• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* //device/libs/media_jni/MediaScanner.cpp
2 **
3 ** Copyright 2007, The Android Open Source Project
4 **
5 ** Licensed under the Apache License, Version 2.0 (the "License");
6 ** you may not use this file except in compliance with the License.
7 ** You may obtain a copy of the License at
8 **
9 **     http://www.apache.org/licenses/LICENSE-2.0
10 **
11 ** Unless required by applicable law or agreed to in writing, software
12 ** distributed under the License is distributed on an "AS IS" BASIS,
13 ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 ** See the License for the specific language governing permissions and
15 ** limitations under the License.
16 */
17 
18 #define LOG_TAG "MediaScanner"
19 #include "utils/Log.h"
20 
21 #include <media/mediascanner.h>
22 #include <stdio.h>
23 #include <assert.h>
24 #include <limits.h>
25 #include <unistd.h>
26 #include <fcntl.h>
27 #include <utils/threads.h>
28 
29 #include "jni.h"
30 #include "JNIHelp.h"
31 #include "android_runtime/AndroidRuntime.h"
32 
33 
34 // ----------------------------------------------------------------------------
35 
36 using namespace android;
37 
38 // ----------------------------------------------------------------------------
39 
40 struct fields_t {
41     jfieldID    context;
42 };
43 static fields_t fields;
44 
45 // ----------------------------------------------------------------------------
46 
47 class MyMediaScannerClient : public MediaScannerClient
48 {
49 public:
MyMediaScannerClient(JNIEnv * env,jobject client)50     MyMediaScannerClient(JNIEnv *env, jobject client)
51         :   mEnv(env),
52             mClient(env->NewGlobalRef(client)),
53             mScanFileMethodID(0),
54             mHandleStringTagMethodID(0),
55             mSetMimeTypeMethodID(0)
56     {
57         jclass mediaScannerClientInterface = env->FindClass("android/media/MediaScannerClient");
58         if (mediaScannerClientInterface == NULL) {
59             fprintf(stderr, "android/media/MediaScannerClient not found\n");
60         }
61         else {
62             mScanFileMethodID = env->GetMethodID(mediaScannerClientInterface, "scanFile",
63                                                      "(Ljava/lang/String;JJ)V");
64             mHandleStringTagMethodID = env->GetMethodID(mediaScannerClientInterface, "handleStringTag",
65                                                      "(Ljava/lang/String;Ljava/lang/String;)V");
66             mSetMimeTypeMethodID = env->GetMethodID(mediaScannerClientInterface, "setMimeType",
67                                                      "(Ljava/lang/String;)V");
68             mAddNoMediaFolderMethodID = env->GetMethodID(mediaScannerClientInterface, "addNoMediaFolder",
69                                                      "(Ljava/lang/String;)V");
70         }
71     }
72 
~MyMediaScannerClient()73     virtual ~MyMediaScannerClient()
74     {
75         mEnv->DeleteGlobalRef(mClient);
76     }
77 
78     // returns true if it succeeded, false if an exception occured in the Java code
scanFile(const char * path,long long lastModified,long long fileSize)79     virtual bool scanFile(const char* path, long long lastModified, long long fileSize)
80     {
81         jstring pathStr;
82         if ((pathStr = mEnv->NewStringUTF(path)) == NULL) return false;
83 
84         mEnv->CallVoidMethod(mClient, mScanFileMethodID, pathStr, lastModified, fileSize);
85 
86         mEnv->DeleteLocalRef(pathStr);
87         return (!mEnv->ExceptionCheck());
88     }
89 
90     // returns true if it succeeded, false if an exception occured in the Java code
handleStringTag(const char * name,const char * value)91     virtual bool handleStringTag(const char* name, const char* value)
92     {
93         jstring nameStr, valueStr;
94         if ((nameStr = mEnv->NewStringUTF(name)) == NULL) return false;
95         if ((valueStr = mEnv->NewStringUTF(value)) == NULL) return false;
96 
97         mEnv->CallVoidMethod(mClient, mHandleStringTagMethodID, nameStr, valueStr);
98 
99         mEnv->DeleteLocalRef(nameStr);
100         mEnv->DeleteLocalRef(valueStr);
101         return (!mEnv->ExceptionCheck());
102     }
103 
104     // returns true if it succeeded, false if an exception occured in the Java code
setMimeType(const char * mimeType)105     virtual bool setMimeType(const char* mimeType)
106     {
107         jstring mimeTypeStr;
108         if ((mimeTypeStr = mEnv->NewStringUTF(mimeType)) == NULL) return false;
109 
110         mEnv->CallVoidMethod(mClient, mSetMimeTypeMethodID, mimeTypeStr);
111 
112         mEnv->DeleteLocalRef(mimeTypeStr);
113         return (!mEnv->ExceptionCheck());
114     }
115 
116     // returns true if it succeeded, false if an exception occured in the Java code
addNoMediaFolder(const char * path)117     virtual bool addNoMediaFolder(const char* path)
118     {
119         jstring pathStr;
120         if ((pathStr = mEnv->NewStringUTF(path)) == NULL) return false;
121 
122         mEnv->CallVoidMethod(mClient, mAddNoMediaFolderMethodID, pathStr);
123 
124         mEnv->DeleteLocalRef(pathStr);
125         return (!mEnv->ExceptionCheck());
126     }
127 
128 
129 private:
130     JNIEnv *mEnv;
131     jobject mClient;
132     jmethodID mScanFileMethodID;
133     jmethodID mHandleStringTagMethodID;
134     jmethodID mSetMimeTypeMethodID;
135     jmethodID mAddNoMediaFolderMethodID;
136 };
137 
138 
139 // ----------------------------------------------------------------------------
140 
ExceptionCheck(void * env)141 static bool ExceptionCheck(void* env)
142 {
143     return ((JNIEnv *)env)->ExceptionCheck();
144 }
145 
146 static void
android_media_MediaScanner_processDirectory(JNIEnv * env,jobject thiz,jstring path,jstring extensions,jobject client)147 android_media_MediaScanner_processDirectory(JNIEnv *env, jobject thiz, jstring path, jstring extensions, jobject client)
148 {
149     MediaScanner *mp = (MediaScanner *)env->GetIntField(thiz, fields.context);
150 
151     if (path == NULL) {
152         jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
153         return;
154     }
155     if (extensions == NULL) {
156         jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
157         return;
158     }
159 
160     const char *pathStr = env->GetStringUTFChars(path, NULL);
161     if (pathStr == NULL) {  // Out of memory
162         jniThrowException(env, "java/lang/RuntimeException", "Out of memory");
163         return;
164     }
165     const char *extensionsStr = env->GetStringUTFChars(extensions, NULL);
166     if (extensionsStr == NULL) {  // Out of memory
167         env->ReleaseStringUTFChars(path, pathStr);
168         jniThrowException(env, "java/lang/RuntimeException", "Out of memory");
169         return;
170     }
171 
172     MyMediaScannerClient myClient(env, client);
173     mp->processDirectory(pathStr, extensionsStr, myClient, ExceptionCheck, env);
174     env->ReleaseStringUTFChars(path, pathStr);
175     env->ReleaseStringUTFChars(extensions, extensionsStr);
176 }
177 
178 static void
android_media_MediaScanner_processFile(JNIEnv * env,jobject thiz,jstring path,jstring mimeType,jobject client)179 android_media_MediaScanner_processFile(JNIEnv *env, jobject thiz, jstring path, jstring mimeType, jobject client)
180 {
181     MediaScanner *mp = (MediaScanner *)env->GetIntField(thiz, fields.context);
182 
183     if (path == NULL) {
184         jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
185         return;
186     }
187 
188     const char *pathStr = env->GetStringUTFChars(path, NULL);
189     if (pathStr == NULL) {  // Out of memory
190         jniThrowException(env, "java/lang/RuntimeException", "Out of memory");
191         return;
192     }
193     const char *mimeTypeStr = (mimeType ? env->GetStringUTFChars(mimeType, NULL) : NULL);
194     if (mimeType && mimeTypeStr == NULL) {  // Out of memory
195         env->ReleaseStringUTFChars(path, pathStr);
196         jniThrowException(env, "java/lang/RuntimeException", "Out of memory");
197         return;
198     }
199 
200     MyMediaScannerClient myClient(env, client);
201     mp->processFile(pathStr, mimeTypeStr, myClient);
202     env->ReleaseStringUTFChars(path, pathStr);
203     if (mimeType) {
204         env->ReleaseStringUTFChars(mimeType, mimeTypeStr);
205     }
206 }
207 
208 static void
android_media_MediaScanner_setLocale(JNIEnv * env,jobject thiz,jstring locale)209 android_media_MediaScanner_setLocale(JNIEnv *env, jobject thiz, jstring locale)
210 {
211     MediaScanner *mp = (MediaScanner *)env->GetIntField(thiz, fields.context);
212 
213     if (locale == NULL) {
214         jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
215         return;
216     }
217     const char *localeStr = env->GetStringUTFChars(locale, NULL);
218     if (localeStr == NULL) {  // Out of memory
219         jniThrowException(env, "java/lang/RuntimeException", "Out of memory");
220         return;
221     }
222     mp->setLocale(localeStr);
223 
224     env->ReleaseStringUTFChars(locale, localeStr);
225 }
226 
227 static jbyteArray
android_media_MediaScanner_extractAlbumArt(JNIEnv * env,jobject thiz,jobject fileDescriptor)228 android_media_MediaScanner_extractAlbumArt(JNIEnv *env, jobject thiz, jobject fileDescriptor)
229 {
230     MediaScanner *mp = (MediaScanner *)env->GetIntField(thiz, fields.context);
231 
232     if (fileDescriptor == NULL) {
233         jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
234         return NULL;
235     }
236 
237     int fd = getParcelFileDescriptorFD(env, fileDescriptor);
238     char* data = mp->extractAlbumArt(fd);
239     if (!data) {
240         return NULL;
241     }
242     long len = *((long*)data);
243 
244     jbyteArray array = env->NewByteArray(len);
245     if (array != NULL) {
246         jbyte* bytes = env->GetByteArrayElements(array, NULL);
247         memcpy(bytes, data + 4, len);
248         env->ReleaseByteArrayElements(array, bytes, 0);
249     }
250 
251 done:
252     free(data);
253     // if NewByteArray() returned NULL, an out-of-memory
254     // exception will have been raised. I just want to
255     // return null in that case.
256     env->ExceptionClear();
257     return array;
258 }
259 
260 // This function gets a field ID, which in turn causes class initialization.
261 // It is called from a static block in MediaScanner, which won't run until the
262 // first time an instance of this class is used.
263 static void
android_media_MediaScanner_native_init(JNIEnv * env)264 android_media_MediaScanner_native_init(JNIEnv *env)
265 {
266      jclass clazz;
267 
268     clazz = env->FindClass("android/media/MediaScanner");
269     if (clazz == NULL) {
270         jniThrowException(env, "java/lang/RuntimeException", "Can't find android/media/MediaScanner");
271         return;
272     }
273 
274     fields.context = env->GetFieldID(clazz, "mNativeContext", "I");
275     if (fields.context == NULL) {
276         jniThrowException(env, "java/lang/RuntimeException", "Can't find MediaScanner.mNativeContext");
277         return;
278     }
279 }
280 
281 static void
android_media_MediaScanner_native_setup(JNIEnv * env,jobject thiz)282 android_media_MediaScanner_native_setup(JNIEnv *env, jobject thiz)
283 {
284     MediaScanner *mp = new MediaScanner();
285     if (mp == NULL) {
286         jniThrowException(env, "java/lang/RuntimeException", "Out of memory");
287         return;
288     }
289 
290     env->SetIntField(thiz, fields.context, (int)mp);
291 }
292 
293 static void
android_media_MediaScanner_native_finalize(JNIEnv * env,jobject thiz)294 android_media_MediaScanner_native_finalize(JNIEnv *env, jobject thiz)
295 {
296     MediaScanner *mp = (MediaScanner *)env->GetIntField(thiz, fields.context);
297 
298     //printf("##### android_media_MediaScanner_native_finalize: ctx=0x%p\n", ctx);
299 
300     if (mp == 0)
301         return;
302 
303     delete mp;
304 }
305 
306 // ----------------------------------------------------------------------------
307 
308 static JNINativeMethod gMethods[] = {
309     {"processDirectory",  "(Ljava/lang/String;Ljava/lang/String;Landroid/media/MediaScannerClient;)V",
310                                                         (void *)android_media_MediaScanner_processDirectory},
311     {"processFile",       "(Ljava/lang/String;Ljava/lang/String;Landroid/media/MediaScannerClient;)V",
312                                                         (void *)android_media_MediaScanner_processFile},
313     {"setLocale",         "(Ljava/lang/String;)V",      (void *)android_media_MediaScanner_setLocale},
314     {"extractAlbumArt",   "(Ljava/io/FileDescriptor;)[B",     (void *)android_media_MediaScanner_extractAlbumArt},
315     {"native_init",        "()V",                      (void *)android_media_MediaScanner_native_init},
316     {"native_setup",        "()V",                      (void *)android_media_MediaScanner_native_setup},
317     {"native_finalize",     "()V",                      (void *)android_media_MediaScanner_native_finalize},
318 };
319 
320 static const char* const kClassPathName = "android/media/MediaScanner";
321 
322 // This function only registers the native methods, and is called from
323 // JNI_OnLoad in android_media_MediaPlayer.cpp
register_android_media_MediaScanner(JNIEnv * env)324 int register_android_media_MediaScanner(JNIEnv *env)
325 {
326     return AndroidRuntime::registerNativeMethods(env,
327                 "android/media/MediaScanner", gMethods, NELEM(gMethods));
328 }
329 
330 
331