19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* //device/libs/android_runtime/android_util_FileObserver.cpp
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** Copyright 2006, The Android Open Source Project
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** Licensed under the Apache License, Version 2.0 (the "License");
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** you may not use this file except in compliance with the License.
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** You may obtain a copy of the License at
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**     http://www.apache.org/licenses/LICENSE-2.0
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** Unless required by applicable law or agreed to in writing, software
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** distributed under the License is distributed on an "AS IS" BASIS,
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** See the License for the specific language governing permissions and
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** limitations under the License.
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project*/
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "JNIHelp.h"
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "jni.h"
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "utils/Log.h"
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "utils/misc.h"
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "android_runtime/AndroidRuntime.h"
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdio.h>
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdlib.h>
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <string.h>
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdint.h>
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <fcntl.h>
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/ioctl.h>
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <errno.h>
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#ifdef HAVE_INOTIFY
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/inotify.h>
349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jmethodID method_onEvent;
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jint android_os_fileobserver_init(JNIEnv* env, jobject object)
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#ifdef HAVE_INOTIFY
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return (jint)inotify_init();
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#else // HAVE_INOTIFY
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return -1;
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif // HAVE_INOTIFY
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic void android_os_fileobserver_observe(JNIEnv* env, jobject object, jint fd)
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#ifdef HAVE_INOTIFY
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char event_buf[512];
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    struct inotify_event* event;
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    while (1)
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int event_pos = 0;
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int num_bytes = read(fd, event_buf, sizeof(event_buf));
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (num_bytes < (int)sizeof(*event))
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        {
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (errno == EINTR)
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                continue;
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
703762c311729fe9f3af085c14c5c1fb471d994c03Steve Block            ALOGE("***** ERROR! android_os_fileobserver_observe() got a short event!");
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7459d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood        while (num_bytes >= (int)sizeof(*event))
7559d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood        {
7659d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            int event_size;
7759d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            event = (struct inotify_event *)(event_buf + event_pos);
7859d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            jstring path = NULL;
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (event->len > 0)
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            {
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                path = env->NewStringUTF(event->name);
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8559d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood
8659d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            env->CallVoidMethod(object, method_onEvent, event->wd, event->mask, path);
8759d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            if (env->ExceptionCheck()) {
8859d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood                env->ExceptionDescribe();
8959d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood                env->ExceptionClear();
9059d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            }
91b22d55b9f8048b2a1ed99b90894c19388bebd7aeThe Android Open Source Project            if (path != NULL)
92b22d55b9f8048b2a1ed99b90894c19388bebd7aeThe Android Open Source Project            {
93b22d55b9f8048b2a1ed99b90894c19388bebd7aeThe Android Open Source Project                env->DeleteLocalRef(path);
94b22d55b9f8048b2a1ed99b90894c19388bebd7aeThe Android Open Source Project            }
9559d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood
9659d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            event_size = sizeof(*event) + event->len;
9759d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            num_bytes -= event_size;
9859d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood            event_pos += event_size;
9959d25d00a17cc41d3b4416f9c90b29fe8e349645Mike Lockwood        }
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif // HAVE_INOTIFY
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jint android_os_fileobserver_startWatching(JNIEnv* env, jobject object, jint fd, jstring pathString, jint mask)
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int res = -1;
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#ifdef HAVE_INOTIFY
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (fd >= 0)
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const char* path = env->GetStringUTFChars(pathString, NULL);
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        res = inotify_add_watch(fd, path, mask);
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringUTFChars(pathString, path);
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif // HAVE_INOTIFY
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return res;
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic void android_os_fileobserver_stopWatching(JNIEnv* env, jobject object, jint fd, jint wfd)
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#ifdef HAVE_INOTIFY
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    inotify_rm_watch((int)fd, (uint32_t)wfd);
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif // HAVE_INOTIFY
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic JNINativeMethod sMethods[] = {
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     /* name, signature, funcPtr */
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    { "init", "()I", (void*)android_os_fileobserver_init },
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    { "observe", "(I)V", (void*)android_os_fileobserver_observe },
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    { "startWatching", "(ILjava/lang/String;I)I", (void*)android_os_fileobserver_startWatching },
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    { "stopWatching", "(II)V", (void*)android_os_fileobserver_stopWatching }
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project};
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectint register_android_os_FileObserver(JNIEnv* env)
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    jclass clazz;
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    clazz = env->FindClass("android/os/FileObserver$ObserverThread");
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (clazz == NULL)
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project	{
1513762c311729fe9f3af085c14c5c1fb471d994c03Steve Block        ALOGE("Can't find android/os/FileObserver$ObserverThread");
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return -1;
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    method_onEvent = env->GetMethodID(clazz, "onEvent", "(IILjava/lang/String;)V");
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (method_onEvent == NULL)
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
1583762c311729fe9f3af085c14c5c1fb471d994c03Steve Block        ALOGE("Can't find FileObserver.onEvent(int, int, String)");
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return -1;
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return AndroidRuntime::registerNativeMethods(env, "android/os/FileObserver$ObserverThread", sMethods, NELEM(sMethods));
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} /* namespace android */
166