1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.os;
18 
19 import android.annotation.Nullable;
20 import android.util.Log;
21 
22 import java.lang.ref.WeakReference;
23 import java.util.HashMap;
24 
25 /**
26  * Monitors files (using <a href="http://en.wikipedia.org/wiki/Inotify">inotify</a>)
27  * to fire an event after files are accessed or changed by by any process on
28  * the device (including this one).  FileObserver is an abstract class;
29  * subclasses must implement the event handler {@link #onEvent(int, String)}.
30  *
31  * <p>Each FileObserver instance monitors a single file or directory.
32  * If a directory is monitored, events will be triggered for all files and
33  * subdirectories inside the monitored directory.</p>
34  *
35  * <p>An event mask is used to specify which changes or actions to report.
36  * Event type constants are used to describe the possible changes in the
37  * event mask as well as what actually happened in event callbacks.</p>
38  *
39  * <p class="caution"><b>Warning</b>: If a FileObserver is garbage collected, it
40  * will stop sending events.  To ensure you keep receiving events, you must
41  * keep a reference to the FileObserver instance from some other live object.</p>
42  */
43 public abstract class FileObserver {
44     /** Event type: Data was read from a file */
45     public static final int ACCESS = 0x00000001;
46     /** Event type: Data was written to a file */
47     public static final int MODIFY = 0x00000002;
48     /** Event type: Metadata (permissions, owner, timestamp) was changed explicitly */
49     public static final int ATTRIB = 0x00000004;
50     /** Event type: Someone had a file or directory open for writing, and closed it */
51     public static final int CLOSE_WRITE = 0x00000008;
52     /** Event type: Someone had a file or directory open read-only, and closed it */
53     public static final int CLOSE_NOWRITE = 0x00000010;
54     /** Event type: A file or directory was opened */
55     public static final int OPEN = 0x00000020;
56     /** Event type: A file or subdirectory was moved from the monitored directory */
57     public static final int MOVED_FROM = 0x00000040;
58     /** Event type: A file or subdirectory was moved to the monitored directory */
59     public static final int MOVED_TO = 0x00000080;
60     /** Event type: A new file or subdirectory was created under the monitored directory */
61     public static final int CREATE = 0x00000100;
62     /** Event type: A file was deleted from the monitored directory */
63     public static final int DELETE = 0x00000200;
64     /** Event type: The monitored file or directory was deleted; monitoring effectively stops */
65     public static final int DELETE_SELF = 0x00000400;
66     /** Event type: The monitored file or directory was moved; monitoring continues */
67     public static final int MOVE_SELF = 0x00000800;
68 
69     /** Event mask: All valid event types, combined */
70     public static final int ALL_EVENTS = ACCESS | MODIFY | ATTRIB | CLOSE_WRITE
71             | CLOSE_NOWRITE | OPEN | MOVED_FROM | MOVED_TO | DELETE | CREATE
72             | DELETE_SELF | MOVE_SELF;
73 
74     private static final String LOG_TAG = "FileObserver";
75 
76     private static class ObserverThread extends Thread {
77         private HashMap<Integer, WeakReference> m_observers = new HashMap<Integer, WeakReference>();
78         private int m_fd;
79 
ObserverThread()80         public ObserverThread() {
81             super("FileObserver");
82             m_fd = init();
83         }
84 
run()85         public void run() {
86             observe(m_fd);
87         }
88 
startWatching(String path, int mask, FileObserver observer)89         public int startWatching(String path, int mask, FileObserver observer) {
90             int wfd = startWatching(m_fd, path, mask);
91 
92             Integer i = new Integer(wfd);
93             if (wfd >= 0) {
94                 synchronized (m_observers) {
95                     m_observers.put(i, new WeakReference(observer));
96                 }
97             }
98 
99             return i;
100         }
101 
stopWatching(int descriptor)102         public void stopWatching(int descriptor) {
103             stopWatching(m_fd, descriptor);
104         }
105 
onEvent(int wfd, int mask, String path)106         public void onEvent(int wfd, int mask, String path) {
107             // look up our observer, fixing up the map if necessary...
108             FileObserver observer = null;
109 
110             synchronized (m_observers) {
111                 WeakReference weak = m_observers.get(wfd);
112                 if (weak != null) {  // can happen with lots of events from a dead wfd
113                     observer = (FileObserver) weak.get();
114                     if (observer == null) {
115                         m_observers.remove(wfd);
116                     }
117                 }
118             }
119 
120             // ...then call out to the observer without the sync lock held
121             if (observer != null) {
122                 try {
123                     observer.onEvent(mask, path);
124                 } catch (Throwable throwable) {
125                     Log.wtf(LOG_TAG, "Unhandled exception in FileObserver " + observer, throwable);
126                 }
127             }
128         }
129 
init()130         private native int init();
observe(int fd)131         private native void observe(int fd);
startWatching(int fd, String path, int mask)132         private native int startWatching(int fd, String path, int mask);
stopWatching(int fd, int wfd)133         private native void stopWatching(int fd, int wfd);
134     }
135 
136     private static ObserverThread s_observerThread;
137 
138     static {
139         s_observerThread = new ObserverThread();
s_observerThread.start()140         s_observerThread.start();
141     }
142 
143     // instance
144     private String m_path;
145     private Integer m_descriptor;
146     private int m_mask;
147 
148     /**
149      * Equivalent to FileObserver(path, FileObserver.ALL_EVENTS).
150      */
FileObserver(String path)151     public FileObserver(String path) {
152         this(path, ALL_EVENTS);
153     }
154 
155     /**
156      * Create a new file observer for a certain file or directory.
157      * Monitoring does not start on creation!  You must call
158      * {@link #startWatching()} before you will receive events.
159      *
160      * @param path The file or directory to monitor
161      * @param mask The event or events (added together) to watch for
162      */
FileObserver(String path, int mask)163     public FileObserver(String path, int mask) {
164         m_path = path;
165         m_mask = mask;
166         m_descriptor = -1;
167     }
168 
finalize()169     protected void finalize() {
170         stopWatching();
171     }
172 
173     /**
174      * Start watching for events.  The monitored file or directory must exist at
175      * this time, or else no events will be reported (even if it appears later).
176      * If monitoring is already started, this call has no effect.
177      */
startWatching()178     public void startWatching() {
179         if (m_descriptor < 0) {
180             m_descriptor = s_observerThread.startWatching(m_path, m_mask, this);
181         }
182     }
183 
184     /**
185      * Stop watching for events.  Some events may be in process, so events
186      * may continue to be reported even after this method completes.  If
187      * monitoring is already stopped, this call has no effect.
188      */
stopWatching()189     public void stopWatching() {
190         if (m_descriptor >= 0) {
191             s_observerThread.stopWatching(m_descriptor);
192             m_descriptor = -1;
193         }
194     }
195 
196     /**
197      * The event handler, which must be implemented by subclasses.
198      *
199      * <p class="note">This method is invoked on a special FileObserver thread.
200      * It runs independently of any threads, so take care to use appropriate
201      * synchronization!  Consider using {@link Handler#post(Runnable)} to shift
202      * event handling work to the main thread to avoid concurrency problems.</p>
203      *
204      * <p>Event handlers must not throw exceptions.</p>
205      *
206      * @param event The type of event which happened
207      * @param path The path, relative to the main monitored file or directory,
208      *     of the file or directory which triggered the event.  This value can
209      *     be {@code null} for certain events, such as {@link #MOVE_SELF}.
210      */
onEvent(int event, @Nullable String path)211     public abstract void onEvent(int event, @Nullable String path);
212 }
213