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.IntDef; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.compat.annotation.UnsupportedAppUsage; 23 import android.util.Log; 24 25 import java.io.File; 26 import java.lang.annotation.Retention; 27 import java.lang.annotation.RetentionPolicy; 28 import java.lang.ref.WeakReference; 29 import java.util.Arrays; 30 import java.util.HashMap; 31 import java.util.List; 32 33 /** 34 * Monitors files (using <a href="http://en.wikipedia.org/wiki/Inotify">inotify</a>) 35 * to fire an event after files are accessed or changed by any process on 36 * the device (including this one). FileObserver is an abstract class; 37 * subclasses must implement the event handler {@link #onEvent(int, String)}. 38 * 39 * <p>Each FileObserver instance can monitor multiple files or directories. 40 * If a directory is monitored, events will be triggered for all files and 41 * subdirectories inside the monitored directory.</p> 42 * 43 * <p>An event mask is used to specify which changes or actions to report. 44 * Event type constants are used to describe the possible changes in the 45 * event mask as well as what actually happened in event callbacks.</p> 46 * 47 * <p class="caution"><b>Warning</b>: If a FileObserver is garbage collected, it 48 * will stop sending events. To ensure you keep receiving events, you must 49 * keep a reference to the FileObserver instance from some other live object.</p> 50 */ 51 public abstract class FileObserver { 52 /** @hide */ 53 @IntDef(flag = true, value = { 54 ACCESS, 55 MODIFY, 56 ATTRIB, 57 CLOSE_WRITE, 58 CLOSE_NOWRITE, 59 OPEN, 60 MOVED_FROM, 61 MOVED_TO, 62 CREATE, 63 DELETE, 64 DELETE_SELF, 65 MOVE_SELF 66 }) 67 @Retention(RetentionPolicy.SOURCE) 68 public @interface NotifyEventType {} 69 70 /** Event type: Data was read from a file */ 71 public static final int ACCESS = 0x00000001; 72 /** Event type: Data was written to a file */ 73 public static final int MODIFY = 0x00000002; 74 /** Event type: Metadata (permissions, owner, timestamp) was changed explicitly */ 75 public static final int ATTRIB = 0x00000004; 76 /** Event type: Someone had a file or directory open for writing, and closed it */ 77 public static final int CLOSE_WRITE = 0x00000008; 78 /** Event type: Someone had a file or directory open read-only, and closed it */ 79 public static final int CLOSE_NOWRITE = 0x00000010; 80 /** Event type: A file or directory was opened */ 81 public static final int OPEN = 0x00000020; 82 /** Event type: A file or subdirectory was moved from the monitored directory */ 83 public static final int MOVED_FROM = 0x00000040; 84 /** Event type: A file or subdirectory was moved to the monitored directory */ 85 public static final int MOVED_TO = 0x00000080; 86 /** Event type: A new file or subdirectory was created under the monitored directory */ 87 public static final int CREATE = 0x00000100; 88 /** Event type: A file was deleted from the monitored directory */ 89 public static final int DELETE = 0x00000200; 90 /** Event type: The monitored file or directory was deleted; monitoring effectively stops */ 91 public static final int DELETE_SELF = 0x00000400; 92 /** Event type: The monitored file or directory was moved; monitoring continues */ 93 public static final int MOVE_SELF = 0x00000800; 94 95 /** Event mask: All valid event types, combined */ 96 @NotifyEventType 97 public static final int ALL_EVENTS = ACCESS | MODIFY | ATTRIB | CLOSE_WRITE 98 | CLOSE_NOWRITE | OPEN | MOVED_FROM | MOVED_TO | DELETE | CREATE 99 | DELETE_SELF | MOVE_SELF; 100 101 private static final String LOG_TAG = "FileObserver"; 102 103 private static class ObserverThread extends Thread { 104 private HashMap<Integer, WeakReference> m_observers = new HashMap<Integer, WeakReference>(); 105 private int m_fd; 106 ObserverThread()107 public ObserverThread() { 108 super("FileObserver"); 109 m_fd = init(); 110 } 111 run()112 public void run() { 113 observe(m_fd); 114 } 115 startWatching(List<File> files, @NotifyEventType int mask, FileObserver observer)116 public int[] startWatching(List<File> files, 117 @NotifyEventType int mask, FileObserver observer) { 118 final int count = files.size(); 119 final String[] paths = new String[count]; 120 for (int i = 0; i < count; ++i) { 121 paths[i] = files.get(i).getAbsolutePath(); 122 } 123 final int[] wfds = new int[count]; 124 Arrays.fill(wfds, -1); 125 126 startWatching(m_fd, paths, mask, wfds); 127 128 final WeakReference<FileObserver> fileObserverWeakReference = 129 new WeakReference<>(observer); 130 synchronized (m_observers) { 131 for (int wfd : wfds) { 132 if (wfd >= 0) { 133 m_observers.put(wfd, fileObserverWeakReference); 134 } 135 } 136 } 137 138 return wfds; 139 } 140 stopWatching(int[] descriptors)141 public void stopWatching(int[] descriptors) { 142 stopWatching(m_fd, descriptors); 143 } 144 145 @UnsupportedAppUsage onEvent(int wfd, @NotifyEventType int mask, String path)146 public void onEvent(int wfd, @NotifyEventType int mask, String path) { 147 // look up our observer, fixing up the map if necessary... 148 FileObserver observer = null; 149 150 synchronized (m_observers) { 151 WeakReference weak = m_observers.get(wfd); 152 if (weak != null) { // can happen with lots of events from a dead wfd 153 observer = (FileObserver) weak.get(); 154 if (observer == null) { 155 m_observers.remove(wfd); 156 } 157 } 158 } 159 160 // ...then call out to the observer without the sync lock held 161 if (observer != null) { 162 try { 163 observer.onEvent(mask, path); 164 } catch (Throwable throwable) { 165 Log.wtf(LOG_TAG, "Unhandled exception in FileObserver " + observer, throwable); 166 } 167 } 168 } 169 init()170 private native int init(); observe(int fd)171 private native void observe(int fd); startWatching(int fd, String[] paths, @NotifyEventType int mask, int[] wfds)172 private native void startWatching(int fd, String[] paths, 173 @NotifyEventType int mask, int[] wfds); stopWatching(int fd, int[] wfds)174 private native void stopWatching(int fd, int[] wfds); 175 } 176 177 @UnsupportedAppUsage 178 private static ObserverThread s_observerThread; 179 180 static { 181 s_observerThread = new ObserverThread(); s_observerThread.start()182 s_observerThread.start(); 183 } 184 185 // instance 186 private final List<File> mFiles; 187 private int[] mDescriptors; 188 private final int mMask; 189 190 /** 191 * Equivalent to FileObserver(path, FileObserver.ALL_EVENTS). 192 * 193 * @deprecated use {@link #FileObserver(File)} instead. 194 */ 195 @Deprecated FileObserver(String path)196 public FileObserver(String path) { 197 this(new File(path)); 198 } 199 200 /** 201 * Equivalent to FileObserver(file, FileObserver.ALL_EVENTS). 202 */ FileObserver(@onNull File file)203 public FileObserver(@NonNull File file) { 204 this(Arrays.asList(file)); 205 } 206 207 /** 208 * Equivalent to FileObserver(paths, FileObserver.ALL_EVENTS). 209 * 210 * @param files The files or directories to monitor 211 */ FileObserver(@onNull List<File> files)212 public FileObserver(@NonNull List<File> files) { 213 this(files, ALL_EVENTS); 214 } 215 216 /** 217 * Create a new file observer for a certain file or directory. 218 * Monitoring does not start on creation! You must call 219 * {@link #startWatching()} before you will receive events. 220 * 221 * @param path The file or directory to monitor 222 * @param mask The event or events (added together) to watch for 223 * 224 * @deprecated use {@link #FileObserver(File, int)} instead. 225 */ 226 @Deprecated FileObserver(String path, @NotifyEventType int mask)227 public FileObserver(String path, @NotifyEventType int mask) { 228 this(new File(path), mask); 229 } 230 231 /** 232 * Create a new file observer for a certain file or directory. 233 * Monitoring does not start on creation! You must call 234 * {@link #startWatching()} before you will receive events. 235 * 236 * @param file The file or directory to monitor 237 * @param mask The event or events (added together) to watch for 238 */ FileObserver(@onNull File file, @NotifyEventType int mask)239 public FileObserver(@NonNull File file, @NotifyEventType int mask) { 240 this(Arrays.asList(file), mask); 241 } 242 243 /** 244 * Version of {@link #FileObserver(File, int)} that allows callers to monitor 245 * multiple files or directories. 246 * 247 * @param files The files or directories to monitor 248 * @param mask The event or events (added together) to watch for 249 */ FileObserver(@onNull List<File> files, @NotifyEventType int mask)250 public FileObserver(@NonNull List<File> files, @NotifyEventType int mask) { 251 mFiles = files; 252 mMask = mask; 253 } 254 finalize()255 protected void finalize() { 256 stopWatching(); 257 } 258 259 /** 260 * Start watching for events. The monitored file or directory must exist at 261 * this time, or else no events will be reported (even if it appears later). 262 * If monitoring is already started, this call has no effect. 263 */ startWatching()264 public void startWatching() { 265 if (mDescriptors == null) { 266 mDescriptors = s_observerThread.startWatching(mFiles, mMask, this); 267 } 268 } 269 270 /** 271 * Stop watching for events. Some events may be in process, so events 272 * may continue to be reported even after this method completes. If 273 * monitoring is already stopped, this call has no effect. 274 */ stopWatching()275 public void stopWatching() { 276 if (mDescriptors != null) { 277 s_observerThread.stopWatching(mDescriptors); 278 mDescriptors = null; 279 } 280 } 281 282 /** 283 * The event handler, which must be implemented by subclasses. 284 * 285 * <p class="note">This method is invoked on a special FileObserver thread. 286 * It runs independently of any threads, so take care to use appropriate 287 * synchronization! Consider using {@link Handler#post(Runnable)} to shift 288 * event handling work to the main thread to avoid concurrency problems.</p> 289 * 290 * <p>Event handlers must not throw exceptions.</p> 291 * 292 * @param event The type of event which happened 293 * @param path The path, relative to the main monitored file or directory, 294 * of the file or directory which triggered the event. This value can 295 * be {@code null} for certain events, such as {@link #MOVE_SELF}. 296 */ onEvent(int event, @Nullable String path)297 public abstract void onEvent(int event, @Nullable String path); 298 } 299