1 /* //device/libs/android_runtime/android_util_AssetManager.cpp
2 **
3 ** Copyright 2006, 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 "asset"
19 
20 #include <android_runtime/android_util_AssetManager.h>
21 
22 #include <inttypes.h>
23 #include <linux/capability.h>
24 #include <stdio.h>
25 #include <sys/types.h>
26 #include <sys/wait.h>
27 #include <sys/stat.h>
28 #include <sys/system_properties.h>
29 
30 #include <private/android_filesystem_config.h> // for AID_SYSTEM
31 
32 #include "androidfw/Asset.h"
33 #include "androidfw/AssetManager.h"
34 #include "androidfw/AttributeResolution.h"
35 #include "androidfw/ResourceTypes.h"
36 #include "android_runtime/AndroidRuntime.h"
37 #include "android_util_Binder.h"
38 #include "core_jni_helpers.h"
39 #include "jni.h"
40 #include "JNIHelp.h"
41 #include "ScopedStringChars.h"
42 #include "ScopedUtfChars.h"
43 #include "utils/Log.h"
44 #include "utils/misc.h"
45 #include "utils/String8.h"
46 
47 extern "C" int capget(cap_user_header_t hdrp, cap_user_data_t datap);
48 extern "C" int capset(cap_user_header_t hdrp, const cap_user_data_t datap);
49 
50 
51 namespace android {
52 
53 static const bool kThrowOnBadId = false;
54 
55 // ----------------------------------------------------------------------------
56 
57 static struct typedvalue_offsets_t
58 {
59     jfieldID mType;
60     jfieldID mData;
61     jfieldID mString;
62     jfieldID mAssetCookie;
63     jfieldID mResourceId;
64     jfieldID mChangingConfigurations;
65     jfieldID mDensity;
66 } gTypedValueOffsets;
67 
68 static struct assetfiledescriptor_offsets_t
69 {
70     jfieldID mFd;
71     jfieldID mStartOffset;
72     jfieldID mLength;
73 } gAssetFileDescriptorOffsets;
74 
75 static struct assetmanager_offsets_t
76 {
77     jfieldID mObject;
78 } gAssetManagerOffsets;
79 
80 static struct sparsearray_offsets_t
81 {
82     jclass classObject;
83     jmethodID constructor;
84     jmethodID put;
85 } gSparseArrayOffsets;
86 
87 static struct configuration_offsets_t
88 {
89     jclass classObject;
90     jmethodID constructor;
91     jfieldID mSmallestScreenWidthDpOffset;
92     jfieldID mScreenWidthDpOffset;
93     jfieldID mScreenHeightDpOffset;
94 } gConfigurationOffsets;
95 
96 jclass g_stringClass = NULL;
97 
98 // ----------------------------------------------------------------------------
99 
100 static jint copyValue(JNIEnv* env, jobject outValue, const ResTable* table,
101                       const Res_value& value, uint32_t ref, ssize_t block,
102                       uint32_t typeSpecFlags, ResTable_config* config = NULL);
103 
copyValue(JNIEnv * env,jobject outValue,const ResTable * table,const Res_value & value,uint32_t ref,ssize_t block,uint32_t typeSpecFlags,ResTable_config * config)104 jint copyValue(JNIEnv* env, jobject outValue, const ResTable* table,
105                const Res_value& value, uint32_t ref, ssize_t block,
106                uint32_t typeSpecFlags, ResTable_config* config)
107 {
108     env->SetIntField(outValue, gTypedValueOffsets.mType, value.dataType);
109     env->SetIntField(outValue, gTypedValueOffsets.mAssetCookie,
110                      static_cast<jint>(table->getTableCookie(block)));
111     env->SetIntField(outValue, gTypedValueOffsets.mData, value.data);
112     env->SetObjectField(outValue, gTypedValueOffsets.mString, NULL);
113     env->SetIntField(outValue, gTypedValueOffsets.mResourceId, ref);
114     env->SetIntField(outValue, gTypedValueOffsets.mChangingConfigurations,
115             typeSpecFlags);
116     if (config != NULL) {
117         env->SetIntField(outValue, gTypedValueOffsets.mDensity, config->density);
118     }
119     return block;
120 }
121 
122 // This is called by zygote (running as user root) as part of preloadResources.
verifySystemIdmaps()123 static void verifySystemIdmaps()
124 {
125     pid_t pid;
126     char system_id[10];
127 
128     snprintf(system_id, sizeof(system_id), "%d", AID_SYSTEM);
129 
130     switch (pid = fork()) {
131         case -1:
132             ALOGE("failed to fork for idmap: %s", strerror(errno));
133             break;
134         case 0: // child
135             {
136                 struct __user_cap_header_struct capheader;
137                 struct __user_cap_data_struct capdata;
138 
139                 memset(&capheader, 0, sizeof(capheader));
140                 memset(&capdata, 0, sizeof(capdata));
141 
142                 capheader.version = _LINUX_CAPABILITY_VERSION;
143                 capheader.pid = 0;
144 
145                 if (capget(&capheader, &capdata) != 0) {
146                     ALOGE("capget: %s\n", strerror(errno));
147                     exit(1);
148                 }
149 
150                 capdata.effective = capdata.permitted;
151                 if (capset(&capheader, &capdata) != 0) {
152                     ALOGE("capset: %s\n", strerror(errno));
153                     exit(1);
154                 }
155 
156                 if (setgid(AID_SYSTEM) != 0) {
157                     ALOGE("setgid: %s\n", strerror(errno));
158                     exit(1);
159                 }
160 
161                 if (setuid(AID_SYSTEM) != 0) {
162                     ALOGE("setuid: %s\n", strerror(errno));
163                     exit(1);
164                 }
165 
166                 // Generic idmap parameters
167                 const char* argv[8];
168                 int argc = 0;
169                 struct stat st;
170 
171                 memset(argv, NULL, sizeof(argv));
172                 argv[argc++] = AssetManager::IDMAP_BIN;
173                 argv[argc++] = "--scan";
174                 argv[argc++] = AssetManager::TARGET_PACKAGE_NAME;
175                 argv[argc++] = AssetManager::TARGET_APK_PATH;
176                 argv[argc++] = AssetManager::IDMAP_DIR;
177 
178                 // Directories to scan for overlays: if OVERLAY_THEME_DIR_PROPERTY is defined,
179                 // use OVERLAY_DIR/<value of OVERLAY_THEME_DIR_PROPERTY> in addition to OVERLAY_DIR.
180                 char subdir[PROP_VALUE_MAX];
181                 int len = __system_property_get(AssetManager::OVERLAY_THEME_DIR_PROPERTY, subdir);
182                 if (len > 0) {
183                     String8 overlayPath = String8(AssetManager::OVERLAY_DIR) + "/" + subdir;
184                     if (stat(overlayPath.string(), &st) == 0) {
185                         argv[argc++] = overlayPath.string();
186                     }
187                 }
188                 if (stat(AssetManager::OVERLAY_DIR, &st) == 0) {
189                     argv[argc++] = AssetManager::OVERLAY_DIR;
190                 }
191 
192                 // Finally, invoke idmap (if any overlay directory exists)
193                 if (argc > 5) {
194                     execv(AssetManager::IDMAP_BIN, (char* const*)argv);
195                     ALOGE("failed to execv for idmap: %s", strerror(errno));
196                     exit(1); // should never get here
197                 } else {
198                     exit(0);
199                 }
200             }
201             break;
202         default: // parent
203             waitpid(pid, NULL, 0);
204             break;
205     }
206 }
207 
208 // ----------------------------------------------------------------------------
209 
210 // this guy is exported to other jni routines
assetManagerForJavaObject(JNIEnv * env,jobject obj)211 AssetManager* assetManagerForJavaObject(JNIEnv* env, jobject obj)
212 {
213     jlong amHandle = env->GetLongField(obj, gAssetManagerOffsets.mObject);
214     AssetManager* am = reinterpret_cast<AssetManager*>(amHandle);
215     if (am != NULL) {
216         return am;
217     }
218     jniThrowException(env, "java/lang/IllegalStateException", "AssetManager has been finalized!");
219     return NULL;
220 }
221 
android_content_AssetManager_openAsset(JNIEnv * env,jobject clazz,jstring fileName,jint mode)222 static jlong android_content_AssetManager_openAsset(JNIEnv* env, jobject clazz,
223                                                 jstring fileName, jint mode)
224 {
225     AssetManager* am = assetManagerForJavaObject(env, clazz);
226     if (am == NULL) {
227         return 0;
228     }
229 
230     ALOGV("openAsset in %p (Java object %p)\n", am, clazz);
231 
232     ScopedUtfChars fileName8(env, fileName);
233     if (fileName8.c_str() == NULL) {
234         jniThrowException(env, "java/lang/IllegalArgumentException", "Empty file name");
235         return -1;
236     }
237 
238     if (mode != Asset::ACCESS_UNKNOWN && mode != Asset::ACCESS_RANDOM
239         && mode != Asset::ACCESS_STREAMING && mode != Asset::ACCESS_BUFFER) {
240         jniThrowException(env, "java/lang/IllegalArgumentException", "Bad access mode");
241         return -1;
242     }
243 
244     Asset* a = am->open(fileName8.c_str(), (Asset::AccessMode)mode);
245 
246     if (a == NULL) {
247         jniThrowException(env, "java/io/FileNotFoundException", fileName8.c_str());
248         return -1;
249     }
250 
251     //printf("Created Asset Stream: %p\n", a);
252 
253     return reinterpret_cast<jlong>(a);
254 }
255 
returnParcelFileDescriptor(JNIEnv * env,Asset * a,jlongArray outOffsets)256 static jobject returnParcelFileDescriptor(JNIEnv* env, Asset* a, jlongArray outOffsets)
257 {
258     off64_t startOffset, length;
259     int fd = a->openFileDescriptor(&startOffset, &length);
260     delete a;
261 
262     if (fd < 0) {
263         jniThrowException(env, "java/io/FileNotFoundException",
264                 "This file can not be opened as a file descriptor; it is probably compressed");
265         return NULL;
266     }
267 
268     jlong* offsets = (jlong*)env->GetPrimitiveArrayCritical(outOffsets, 0);
269     if (offsets == NULL) {
270         close(fd);
271         return NULL;
272     }
273 
274     offsets[0] = startOffset;
275     offsets[1] = length;
276 
277     env->ReleasePrimitiveArrayCritical(outOffsets, offsets, 0);
278 
279     jobject fileDesc = jniCreateFileDescriptor(env, fd);
280     if (fileDesc == NULL) {
281         close(fd);
282         return NULL;
283     }
284 
285     return newParcelFileDescriptor(env, fileDesc);
286 }
287 
android_content_AssetManager_openAssetFd(JNIEnv * env,jobject clazz,jstring fileName,jlongArray outOffsets)288 static jobject android_content_AssetManager_openAssetFd(JNIEnv* env, jobject clazz,
289                                                 jstring fileName, jlongArray outOffsets)
290 {
291     AssetManager* am = assetManagerForJavaObject(env, clazz);
292     if (am == NULL) {
293         return NULL;
294     }
295 
296     ALOGV("openAssetFd in %p (Java object %p)\n", am, clazz);
297 
298     ScopedUtfChars fileName8(env, fileName);
299     if (fileName8.c_str() == NULL) {
300         return NULL;
301     }
302 
303     Asset* a = am->open(fileName8.c_str(), Asset::ACCESS_RANDOM);
304 
305     if (a == NULL) {
306         jniThrowException(env, "java/io/FileNotFoundException", fileName8.c_str());
307         return NULL;
308     }
309 
310     //printf("Created Asset Stream: %p\n", a);
311 
312     return returnParcelFileDescriptor(env, a, outOffsets);
313 }
314 
android_content_AssetManager_openNonAssetNative(JNIEnv * env,jobject clazz,jint cookie,jstring fileName,jint mode)315 static jlong android_content_AssetManager_openNonAssetNative(JNIEnv* env, jobject clazz,
316                                                          jint cookie,
317                                                          jstring fileName,
318                                                          jint mode)
319 {
320     AssetManager* am = assetManagerForJavaObject(env, clazz);
321     if (am == NULL) {
322         return 0;
323     }
324 
325     ALOGV("openNonAssetNative in %p (Java object %p)\n", am, clazz);
326 
327     ScopedUtfChars fileName8(env, fileName);
328     if (fileName8.c_str() == NULL) {
329         return -1;
330     }
331 
332     if (mode != Asset::ACCESS_UNKNOWN && mode != Asset::ACCESS_RANDOM
333         && mode != Asset::ACCESS_STREAMING && mode != Asset::ACCESS_BUFFER) {
334         jniThrowException(env, "java/lang/IllegalArgumentException", "Bad access mode");
335         return -1;
336     }
337 
338     Asset* a = cookie
339         ? am->openNonAsset(static_cast<int32_t>(cookie), fileName8.c_str(),
340                 (Asset::AccessMode)mode)
341         : am->openNonAsset(fileName8.c_str(), (Asset::AccessMode)mode);
342 
343     if (a == NULL) {
344         jniThrowException(env, "java/io/FileNotFoundException", fileName8.c_str());
345         return -1;
346     }
347 
348     //printf("Created Asset Stream: %p\n", a);
349 
350     return reinterpret_cast<jlong>(a);
351 }
352 
android_content_AssetManager_openNonAssetFdNative(JNIEnv * env,jobject clazz,jint cookie,jstring fileName,jlongArray outOffsets)353 static jobject android_content_AssetManager_openNonAssetFdNative(JNIEnv* env, jobject clazz,
354                                                          jint cookie,
355                                                          jstring fileName,
356                                                          jlongArray outOffsets)
357 {
358     AssetManager* am = assetManagerForJavaObject(env, clazz);
359     if (am == NULL) {
360         return NULL;
361     }
362 
363     ALOGV("openNonAssetFd in %p (Java object %p)\n", am, clazz);
364 
365     ScopedUtfChars fileName8(env, fileName);
366     if (fileName8.c_str() == NULL) {
367         return NULL;
368     }
369 
370     Asset* a = cookie
371         ? am->openNonAsset(static_cast<int32_t>(cookie), fileName8.c_str(), Asset::ACCESS_RANDOM)
372         : am->openNonAsset(fileName8.c_str(), Asset::ACCESS_RANDOM);
373 
374     if (a == NULL) {
375         jniThrowException(env, "java/io/FileNotFoundException", fileName8.c_str());
376         return NULL;
377     }
378 
379     //printf("Created Asset Stream: %p\n", a);
380 
381     return returnParcelFileDescriptor(env, a, outOffsets);
382 }
383 
android_content_AssetManager_list(JNIEnv * env,jobject clazz,jstring fileName)384 static jobjectArray android_content_AssetManager_list(JNIEnv* env, jobject clazz,
385                                                    jstring fileName)
386 {
387     AssetManager* am = assetManagerForJavaObject(env, clazz);
388     if (am == NULL) {
389         return NULL;
390     }
391 
392     ScopedUtfChars fileName8(env, fileName);
393     if (fileName8.c_str() == NULL) {
394         return NULL;
395     }
396 
397     AssetDir* dir = am->openDir(fileName8.c_str());
398 
399     if (dir == NULL) {
400         jniThrowException(env, "java/io/FileNotFoundException", fileName8.c_str());
401         return NULL;
402     }
403 
404     size_t N = dir->getFileCount();
405 
406     jobjectArray array = env->NewObjectArray(dir->getFileCount(),
407                                                 g_stringClass, NULL);
408     if (array == NULL) {
409         delete dir;
410         return NULL;
411     }
412 
413     for (size_t i=0; i<N; i++) {
414         const String8& name = dir->getFileName(i);
415         jstring str = env->NewStringUTF(name.string());
416         if (str == NULL) {
417             delete dir;
418             return NULL;
419         }
420         env->SetObjectArrayElement(array, i, str);
421         env->DeleteLocalRef(str);
422     }
423 
424     delete dir;
425 
426     return array;
427 }
428 
android_content_AssetManager_destroyAsset(JNIEnv * env,jobject clazz,jlong assetHandle)429 static void android_content_AssetManager_destroyAsset(JNIEnv* env, jobject clazz,
430                                                       jlong assetHandle)
431 {
432     Asset* a = reinterpret_cast<Asset*>(assetHandle);
433 
434     //printf("Destroying Asset Stream: %p\n", a);
435 
436     if (a == NULL) {
437         jniThrowNullPointerException(env, "asset");
438         return;
439     }
440 
441     delete a;
442 }
443 
android_content_AssetManager_readAssetChar(JNIEnv * env,jobject clazz,jlong assetHandle)444 static jint android_content_AssetManager_readAssetChar(JNIEnv* env, jobject clazz,
445                                                        jlong assetHandle)
446 {
447     Asset* a = reinterpret_cast<Asset*>(assetHandle);
448 
449     if (a == NULL) {
450         jniThrowNullPointerException(env, "asset");
451         return -1;
452     }
453 
454     uint8_t b;
455     ssize_t res = a->read(&b, 1);
456     return res == 1 ? b : -1;
457 }
458 
android_content_AssetManager_readAsset(JNIEnv * env,jobject clazz,jlong assetHandle,jbyteArray bArray,jint off,jint len)459 static jint android_content_AssetManager_readAsset(JNIEnv* env, jobject clazz,
460                                                 jlong assetHandle, jbyteArray bArray,
461                                                 jint off, jint len)
462 {
463     Asset* a = reinterpret_cast<Asset*>(assetHandle);
464 
465     if (a == NULL || bArray == NULL) {
466         jniThrowNullPointerException(env, "asset");
467         return -1;
468     }
469 
470     if (len == 0) {
471         return 0;
472     }
473 
474     jsize bLen = env->GetArrayLength(bArray);
475     if (off < 0 || off >= bLen || len < 0 || len > bLen || (off+len) > bLen) {
476         jniThrowException(env, "java/lang/IndexOutOfBoundsException", "");
477         return -1;
478     }
479 
480     jbyte* b = env->GetByteArrayElements(bArray, NULL);
481     ssize_t res = a->read(b+off, len);
482     env->ReleaseByteArrayElements(bArray, b, 0);
483 
484     if (res > 0) return static_cast<jint>(res);
485 
486     if (res < 0) {
487         jniThrowException(env, "java/io/IOException", "");
488     }
489     return -1;
490 }
491 
android_content_AssetManager_seekAsset(JNIEnv * env,jobject clazz,jlong assetHandle,jlong offset,jint whence)492 static jlong android_content_AssetManager_seekAsset(JNIEnv* env, jobject clazz,
493                                                  jlong assetHandle,
494                                                  jlong offset, jint whence)
495 {
496     Asset* a = reinterpret_cast<Asset*>(assetHandle);
497 
498     if (a == NULL) {
499         jniThrowNullPointerException(env, "asset");
500         return -1;
501     }
502 
503     return a->seek(
504         offset, (whence > 0) ? SEEK_END : (whence < 0 ? SEEK_SET : SEEK_CUR));
505 }
506 
android_content_AssetManager_getAssetLength(JNIEnv * env,jobject clazz,jlong assetHandle)507 static jlong android_content_AssetManager_getAssetLength(JNIEnv* env, jobject clazz,
508                                                       jlong assetHandle)
509 {
510     Asset* a = reinterpret_cast<Asset*>(assetHandle);
511 
512     if (a == NULL) {
513         jniThrowNullPointerException(env, "asset");
514         return -1;
515     }
516 
517     return a->getLength();
518 }
519 
android_content_AssetManager_getAssetRemainingLength(JNIEnv * env,jobject clazz,jlong assetHandle)520 static jlong android_content_AssetManager_getAssetRemainingLength(JNIEnv* env, jobject clazz,
521                                                                jlong assetHandle)
522 {
523     Asset* a = reinterpret_cast<Asset*>(assetHandle);
524 
525     if (a == NULL) {
526         jniThrowNullPointerException(env, "asset");
527         return -1;
528     }
529 
530     return a->getRemainingLength();
531 }
532 
android_content_AssetManager_addAssetPath(JNIEnv * env,jobject clazz,jstring path,jboolean appAsLib)533 static jint android_content_AssetManager_addAssetPath(JNIEnv* env, jobject clazz,
534                                                        jstring path, jboolean appAsLib)
535 {
536     ScopedUtfChars path8(env, path);
537     if (path8.c_str() == NULL) {
538         return 0;
539     }
540 
541     AssetManager* am = assetManagerForJavaObject(env, clazz);
542     if (am == NULL) {
543         return 0;
544     }
545 
546     int32_t cookie;
547     bool res = am->addAssetPath(String8(path8.c_str()), &cookie, appAsLib);
548 
549     return (res) ? static_cast<jint>(cookie) : 0;
550 }
551 
android_content_AssetManager_addOverlayPath(JNIEnv * env,jobject clazz,jstring idmapPath)552 static jint android_content_AssetManager_addOverlayPath(JNIEnv* env, jobject clazz,
553                                                      jstring idmapPath)
554 {
555     ScopedUtfChars idmapPath8(env, idmapPath);
556     if (idmapPath8.c_str() == NULL) {
557         return 0;
558     }
559 
560     AssetManager* am = assetManagerForJavaObject(env, clazz);
561     if (am == NULL) {
562         return 0;
563     }
564 
565     int32_t cookie;
566     bool res = am->addOverlayPath(String8(idmapPath8.c_str()), &cookie);
567 
568     return (res) ? (jint)cookie : 0;
569 }
570 
android_content_AssetManager_isUpToDate(JNIEnv * env,jobject clazz)571 static jboolean android_content_AssetManager_isUpToDate(JNIEnv* env, jobject clazz)
572 {
573     AssetManager* am = assetManagerForJavaObject(env, clazz);
574     if (am == NULL) {
575         return JNI_TRUE;
576     }
577     return am->isUpToDate() ? JNI_TRUE : JNI_FALSE;
578 }
579 
getLocales(JNIEnv * env,jobject clazz,bool includeSystemLocales)580 static jobjectArray getLocales(JNIEnv* env, jobject clazz, bool includeSystemLocales)
581 {
582     Vector<String8> locales;
583 
584     AssetManager* am = assetManagerForJavaObject(env, clazz);
585     if (am == NULL) {
586         return NULL;
587     }
588 
589     am->getLocales(&locales, includeSystemLocales);
590 
591     const int N = locales.size();
592 
593     jobjectArray result = env->NewObjectArray(N, g_stringClass, NULL);
594     if (result == NULL) {
595         return NULL;
596     }
597 
598     for (int i=0; i<N; i++) {
599         jstring str = env->NewStringUTF(locales[i].string());
600         if (str == NULL) {
601             return NULL;
602         }
603         env->SetObjectArrayElement(result, i, str);
604         env->DeleteLocalRef(str);
605     }
606 
607     return result;
608 }
609 
android_content_AssetManager_getLocales(JNIEnv * env,jobject clazz)610 static jobjectArray android_content_AssetManager_getLocales(JNIEnv* env, jobject clazz)
611 {
612     return getLocales(env, clazz, true /* include system locales */);
613 }
614 
android_content_AssetManager_getNonSystemLocales(JNIEnv * env,jobject clazz)615 static jobjectArray android_content_AssetManager_getNonSystemLocales(JNIEnv* env, jobject clazz)
616 {
617     return getLocales(env, clazz, false /* don't include system locales */);
618 }
619 
constructConfigurationObject(JNIEnv * env,const ResTable_config & config)620 static jobject constructConfigurationObject(JNIEnv* env, const ResTable_config& config) {
621     jobject result = env->NewObject(gConfigurationOffsets.classObject,
622             gConfigurationOffsets.constructor);
623     if (result == NULL) {
624         return NULL;
625     }
626 
627     env->SetIntField(result, gConfigurationOffsets.mSmallestScreenWidthDpOffset,
628             config.smallestScreenWidthDp);
629     env->SetIntField(result, gConfigurationOffsets.mScreenWidthDpOffset, config.screenWidthDp);
630     env->SetIntField(result, gConfigurationOffsets.mScreenHeightDpOffset, config.screenHeightDp);
631 
632     return result;
633 }
634 
getSizeConfigurationsInternal(JNIEnv * env,const Vector<ResTable_config> & configs)635 static jobjectArray getSizeConfigurationsInternal(JNIEnv* env,
636         const Vector<ResTable_config>& configs) {
637     const int N = configs.size();
638     jobjectArray result = env->NewObjectArray(N, gConfigurationOffsets.classObject, NULL);
639     if (result == NULL) {
640         return NULL;
641     }
642 
643     for (int i=0; i<N; i++) {
644         jobject config = constructConfigurationObject(env, configs[i]);
645         if (config == NULL) {
646             env->DeleteLocalRef(result);
647             return NULL;
648         }
649 
650         env->SetObjectArrayElement(result, i, config);
651         env->DeleteLocalRef(config);
652     }
653 
654     return result;
655 }
656 
android_content_AssetManager_getSizeConfigurations(JNIEnv * env,jobject clazz)657 static jobjectArray android_content_AssetManager_getSizeConfigurations(JNIEnv* env, jobject clazz) {
658     AssetManager* am = assetManagerForJavaObject(env, clazz);
659     if (am == NULL) {
660         return NULL;
661     }
662 
663     const ResTable& res(am->getResources());
664     Vector<ResTable_config> configs;
665     res.getConfigurations(&configs, false /* ignoreMipmap */, true /* ignoreAndroidPackage */);
666 
667     return getSizeConfigurationsInternal(env, configs);
668 }
669 
android_content_AssetManager_setConfiguration(JNIEnv * env,jobject clazz,jint mcc,jint mnc,jstring locale,jint orientation,jint touchscreen,jint density,jint keyboard,jint keyboardHidden,jint navigation,jint screenWidth,jint screenHeight,jint smallestScreenWidthDp,jint screenWidthDp,jint screenHeightDp,jint screenLayout,jint uiMode,jint colorMode,jint sdkVersion)670 static void android_content_AssetManager_setConfiguration(JNIEnv* env, jobject clazz,
671                                                           jint mcc, jint mnc,
672                                                           jstring locale, jint orientation,
673                                                           jint touchscreen, jint density,
674                                                           jint keyboard, jint keyboardHidden,
675                                                           jint navigation,
676                                                           jint screenWidth, jint screenHeight,
677                                                           jint smallestScreenWidthDp,
678                                                           jint screenWidthDp, jint screenHeightDp,
679                                                           jint screenLayout, jint uiMode,
680                                                           jint colorMode, jint sdkVersion)
681 {
682     AssetManager* am = assetManagerForJavaObject(env, clazz);
683     if (am == NULL) {
684         return;
685     }
686 
687     ResTable_config config;
688     memset(&config, 0, sizeof(config));
689 
690     const char* locale8 = locale != NULL ? env->GetStringUTFChars(locale, NULL) : NULL;
691 
692     // Constants duplicated from Java class android.content.res.Configuration.
693     static const jint kScreenLayoutRoundMask = 0x300;
694     static const jint kScreenLayoutRoundShift = 8;
695 
696     config.mcc = (uint16_t)mcc;
697     config.mnc = (uint16_t)mnc;
698     config.orientation = (uint8_t)orientation;
699     config.touchscreen = (uint8_t)touchscreen;
700     config.density = (uint16_t)density;
701     config.keyboard = (uint8_t)keyboard;
702     config.inputFlags = (uint8_t)keyboardHidden;
703     config.navigation = (uint8_t)navigation;
704     config.screenWidth = (uint16_t)screenWidth;
705     config.screenHeight = (uint16_t)screenHeight;
706     config.smallestScreenWidthDp = (uint16_t)smallestScreenWidthDp;
707     config.screenWidthDp = (uint16_t)screenWidthDp;
708     config.screenHeightDp = (uint16_t)screenHeightDp;
709     config.screenLayout = (uint8_t)screenLayout;
710     config.uiMode = (uint8_t)uiMode;
711     config.colorMode = (uint8_t)colorMode;
712     config.sdkVersion = (uint16_t)sdkVersion;
713     config.minorVersion = 0;
714 
715     // In Java, we use a 32bit integer for screenLayout, while we only use an 8bit integer
716     // in C++. We must extract the round qualifier out of the Java screenLayout and put it
717     // into screenLayout2.
718     config.screenLayout2 =
719             (uint8_t)((screenLayout & kScreenLayoutRoundMask) >> kScreenLayoutRoundShift);
720 
721     am->setConfiguration(config, locale8);
722 
723     if (locale != NULL) env->ReleaseStringUTFChars(locale, locale8);
724 }
725 
android_content_AssetManager_getResourceIdentifier(JNIEnv * env,jobject clazz,jstring name,jstring defType,jstring defPackage)726 static jint android_content_AssetManager_getResourceIdentifier(JNIEnv* env, jobject clazz,
727                                                             jstring name,
728                                                             jstring defType,
729                                                             jstring defPackage)
730 {
731     ScopedStringChars name16(env, name);
732     if (name16.get() == NULL) {
733         return 0;
734     }
735 
736     AssetManager* am = assetManagerForJavaObject(env, clazz);
737     if (am == NULL) {
738         return 0;
739     }
740 
741     const char16_t* defType16 = reinterpret_cast<const char16_t*>(defType)
742         ? reinterpret_cast<const char16_t*>(env->GetStringChars(defType, NULL))
743         : NULL;
744     jsize defTypeLen = defType
745         ? env->GetStringLength(defType) : 0;
746     const char16_t* defPackage16 = reinterpret_cast<const char16_t*>(defPackage)
747         ? reinterpret_cast<const char16_t*>(env->GetStringChars(defPackage,
748                                                                 NULL))
749         : NULL;
750     jsize defPackageLen = defPackage
751         ? env->GetStringLength(defPackage) : 0;
752 
753     jint ident = am->getResources().identifierForName(
754         reinterpret_cast<const char16_t*>(name16.get()), name16.size(),
755         defType16, defTypeLen, defPackage16, defPackageLen);
756 
757     if (defPackage16) {
758         env->ReleaseStringChars(defPackage,
759                                 reinterpret_cast<const jchar*>(defPackage16));
760     }
761     if (defType16) {
762         env->ReleaseStringChars(defType,
763                                 reinterpret_cast<const jchar*>(defType16));
764     }
765 
766     return ident;
767 }
768 
android_content_AssetManager_getResourceName(JNIEnv * env,jobject clazz,jint resid)769 static jstring android_content_AssetManager_getResourceName(JNIEnv* env, jobject clazz,
770                                                             jint resid)
771 {
772     AssetManager* am = assetManagerForJavaObject(env, clazz);
773     if (am == NULL) {
774         return NULL;
775     }
776 
777     ResTable::resource_name name;
778     if (!am->getResources().getResourceName(resid, true, &name)) {
779         return NULL;
780     }
781 
782     String16 str;
783     if (name.package != NULL) {
784         str.setTo(name.package, name.packageLen);
785     }
786     if (name.type8 != NULL || name.type != NULL) {
787         if (str.size() > 0) {
788             char16_t div = ':';
789             str.append(&div, 1);
790         }
791         if (name.type8 != NULL) {
792             str.append(String16(name.type8, name.typeLen));
793         } else {
794             str.append(name.type, name.typeLen);
795         }
796     }
797     if (name.name8 != NULL || name.name != NULL) {
798         if (str.size() > 0) {
799             char16_t div = '/';
800             str.append(&div, 1);
801         }
802         if (name.name8 != NULL) {
803             str.append(String16(name.name8, name.nameLen));
804         } else {
805             str.append(name.name, name.nameLen);
806         }
807     }
808 
809     return env->NewString((const jchar*)str.string(), str.size());
810 }
811 
android_content_AssetManager_getResourcePackageName(JNIEnv * env,jobject clazz,jint resid)812 static jstring android_content_AssetManager_getResourcePackageName(JNIEnv* env, jobject clazz,
813                                                                    jint resid)
814 {
815     AssetManager* am = assetManagerForJavaObject(env, clazz);
816     if (am == NULL) {
817         return NULL;
818     }
819 
820     ResTable::resource_name name;
821     if (!am->getResources().getResourceName(resid, true, &name)) {
822         return NULL;
823     }
824 
825     if (name.package != NULL) {
826         return env->NewString((const jchar*)name.package, name.packageLen);
827     }
828 
829     return NULL;
830 }
831 
android_content_AssetManager_getResourceTypeName(JNIEnv * env,jobject clazz,jint resid)832 static jstring android_content_AssetManager_getResourceTypeName(JNIEnv* env, jobject clazz,
833                                                                 jint resid)
834 {
835     AssetManager* am = assetManagerForJavaObject(env, clazz);
836     if (am == NULL) {
837         return NULL;
838     }
839 
840     ResTable::resource_name name;
841     if (!am->getResources().getResourceName(resid, true, &name)) {
842         return NULL;
843     }
844 
845     if (name.type8 != NULL) {
846         return env->NewStringUTF(name.type8);
847     }
848 
849     if (name.type != NULL) {
850         return env->NewString((const jchar*)name.type, name.typeLen);
851     }
852 
853     return NULL;
854 }
855 
android_content_AssetManager_getResourceEntryName(JNIEnv * env,jobject clazz,jint resid)856 static jstring android_content_AssetManager_getResourceEntryName(JNIEnv* env, jobject clazz,
857                                                                  jint resid)
858 {
859     AssetManager* am = assetManagerForJavaObject(env, clazz);
860     if (am == NULL) {
861         return NULL;
862     }
863 
864     ResTable::resource_name name;
865     if (!am->getResources().getResourceName(resid, true, &name)) {
866         return NULL;
867     }
868 
869     if (name.name8 != NULL) {
870         return env->NewStringUTF(name.name8);
871     }
872 
873     if (name.name != NULL) {
874         return env->NewString((const jchar*)name.name, name.nameLen);
875     }
876 
877     return NULL;
878 }
879 
android_content_AssetManager_loadResourceValue(JNIEnv * env,jobject clazz,jint ident,jshort density,jobject outValue,jboolean resolve)880 static jint android_content_AssetManager_loadResourceValue(JNIEnv* env, jobject clazz,
881                                                            jint ident,
882                                                            jshort density,
883                                                            jobject outValue,
884                                                            jboolean resolve)
885 {
886     if (outValue == NULL) {
887          jniThrowNullPointerException(env, "outValue");
888          return 0;
889     }
890     AssetManager* am = assetManagerForJavaObject(env, clazz);
891     if (am == NULL) {
892         return 0;
893     }
894     const ResTable& res(am->getResources());
895 
896     Res_value value;
897     ResTable_config config;
898     uint32_t typeSpecFlags;
899     ssize_t block = res.getResource(ident, &value, false, density, &typeSpecFlags, &config);
900     if (kThrowOnBadId) {
901         if (block == BAD_INDEX) {
902             jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
903             return 0;
904         }
905     }
906     uint32_t ref = ident;
907     if (resolve) {
908         block = res.resolveReference(&value, block, &ref, &typeSpecFlags, &config);
909         if (kThrowOnBadId) {
910             if (block == BAD_INDEX) {
911                 jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
912                 return 0;
913             }
914         }
915     }
916     if (block >= 0) {
917         return copyValue(env, outValue, &res, value, ref, block, typeSpecFlags, &config);
918     }
919 
920     return static_cast<jint>(block);
921 }
922 
android_content_AssetManager_loadResourceBagValue(JNIEnv * env,jobject clazz,jint ident,jint bagEntryId,jobject outValue,jboolean resolve)923 static jint android_content_AssetManager_loadResourceBagValue(JNIEnv* env, jobject clazz,
924                                                            jint ident, jint bagEntryId,
925                                                            jobject outValue, jboolean resolve)
926 {
927     AssetManager* am = assetManagerForJavaObject(env, clazz);
928     if (am == NULL) {
929         return 0;
930     }
931     const ResTable& res(am->getResources());
932 
933     // Now lock down the resource object and start pulling stuff from it.
934     res.lock();
935 
936     ssize_t block = -1;
937     Res_value value;
938 
939     const ResTable::bag_entry* entry = NULL;
940     uint32_t typeSpecFlags;
941     ssize_t entryCount = res.getBagLocked(ident, &entry, &typeSpecFlags);
942 
943     for (ssize_t i=0; i<entryCount; i++) {
944         if (((uint32_t)bagEntryId) == entry->map.name.ident) {
945             block = entry->stringBlock;
946             value = entry->map.value;
947         }
948         entry++;
949     }
950 
951     res.unlock();
952 
953     if (block < 0) {
954         return static_cast<jint>(block);
955     }
956 
957     uint32_t ref = ident;
958     if (resolve) {
959         block = res.resolveReference(&value, block, &ref, &typeSpecFlags);
960         if (kThrowOnBadId) {
961             if (block == BAD_INDEX) {
962                 jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
963                 return 0;
964             }
965         }
966     }
967     if (block >= 0) {
968         return copyValue(env, outValue, &res, value, ref, block, typeSpecFlags);
969     }
970 
971     return static_cast<jint>(block);
972 }
973 
android_content_AssetManager_getStringBlockCount(JNIEnv * env,jobject clazz)974 static jint android_content_AssetManager_getStringBlockCount(JNIEnv* env, jobject clazz)
975 {
976     AssetManager* am = assetManagerForJavaObject(env, clazz);
977     if (am == NULL) {
978         return 0;
979     }
980     return am->getResources().getTableCount();
981 }
982 
android_content_AssetManager_getNativeStringBlock(JNIEnv * env,jobject clazz,jint block)983 static jlong android_content_AssetManager_getNativeStringBlock(JNIEnv* env, jobject clazz,
984                                                            jint block)
985 {
986     AssetManager* am = assetManagerForJavaObject(env, clazz);
987     if (am == NULL) {
988         return 0;
989     }
990     return reinterpret_cast<jlong>(am->getResources().getTableStringBlock(block));
991 }
992 
android_content_AssetManager_getCookieName(JNIEnv * env,jobject clazz,jint cookie)993 static jstring android_content_AssetManager_getCookieName(JNIEnv* env, jobject clazz,
994                                                        jint cookie)
995 {
996     AssetManager* am = assetManagerForJavaObject(env, clazz);
997     if (am == NULL) {
998         return NULL;
999     }
1000     String8 name(am->getAssetPath(static_cast<int32_t>(cookie)));
1001     if (name.length() == 0) {
1002         jniThrowException(env, "java/lang/IndexOutOfBoundsException", "Empty cookie name");
1003         return NULL;
1004     }
1005     jstring str = env->NewStringUTF(name.string());
1006     return str;
1007 }
1008 
android_content_AssetManager_getAssignedPackageIdentifiers(JNIEnv * env,jobject clazz)1009 static jobject android_content_AssetManager_getAssignedPackageIdentifiers(JNIEnv* env, jobject clazz)
1010 {
1011     AssetManager* am = assetManagerForJavaObject(env, clazz);
1012     if (am == NULL) {
1013         return 0;
1014     }
1015 
1016     const ResTable& res = am->getResources();
1017 
1018     jobject sparseArray = env->NewObject(gSparseArrayOffsets.classObject,
1019             gSparseArrayOffsets.constructor);
1020     const size_t N = res.getBasePackageCount();
1021     for (size_t i = 0; i < N; i++) {
1022         const String16 name = res.getBasePackageName(i);
1023         env->CallVoidMethod(
1024             sparseArray, gSparseArrayOffsets.put,
1025             static_cast<jint>(res.getBasePackageId(i)),
1026             env->NewString(reinterpret_cast<const jchar*>(name.string()),
1027                            name.size()));
1028     }
1029     return sparseArray;
1030 }
1031 
android_content_AssetManager_newTheme(JNIEnv * env,jobject clazz)1032 static jlong android_content_AssetManager_newTheme(JNIEnv* env, jobject clazz)
1033 {
1034     AssetManager* am = assetManagerForJavaObject(env, clazz);
1035     if (am == NULL) {
1036         return 0;
1037     }
1038     return reinterpret_cast<jlong>(new ResTable::Theme(am->getResources()));
1039 }
1040 
android_content_AssetManager_deleteTheme(JNIEnv * env,jobject clazz,jlong themeHandle)1041 static void android_content_AssetManager_deleteTheme(JNIEnv* env, jobject clazz,
1042                                                      jlong themeHandle)
1043 {
1044     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeHandle);
1045     delete theme;
1046 }
1047 
android_content_AssetManager_applyThemeStyle(JNIEnv * env,jobject clazz,jlong themeHandle,jint styleRes,jboolean force)1048 static void android_content_AssetManager_applyThemeStyle(JNIEnv* env, jobject clazz,
1049                                                          jlong themeHandle,
1050                                                          jint styleRes,
1051                                                          jboolean force)
1052 {
1053     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeHandle);
1054     theme->applyStyle(styleRes, force ? true : false);
1055 }
1056 
android_content_AssetManager_copyTheme(JNIEnv * env,jobject clazz,jlong destHandle,jlong srcHandle)1057 static void android_content_AssetManager_copyTheme(JNIEnv* env, jobject clazz,
1058                                                    jlong destHandle, jlong srcHandle)
1059 {
1060     ResTable::Theme* dest = reinterpret_cast<ResTable::Theme*>(destHandle);
1061     ResTable::Theme* src = reinterpret_cast<ResTable::Theme*>(srcHandle);
1062     dest->setTo(*src);
1063 }
1064 
android_content_AssetManager_clearTheme(JNIEnv * env,jobject clazz,jlong themeHandle)1065 static void android_content_AssetManager_clearTheme(JNIEnv* env, jobject clazz, jlong themeHandle)
1066 {
1067     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeHandle);
1068     theme->clear();
1069 }
1070 
android_content_AssetManager_loadThemeAttributeValue(JNIEnv * env,jobject clazz,jlong themeHandle,jint ident,jobject outValue,jboolean resolve)1071 static jint android_content_AssetManager_loadThemeAttributeValue(
1072     JNIEnv* env, jobject clazz, jlong themeHandle, jint ident, jobject outValue, jboolean resolve)
1073 {
1074     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeHandle);
1075     const ResTable& res(theme->getResTable());
1076 
1077     Res_value value;
1078     // XXX value could be different in different configs!
1079     uint32_t typeSpecFlags = 0;
1080     ssize_t block = theme->getAttribute(ident, &value, &typeSpecFlags);
1081     uint32_t ref = 0;
1082     if (resolve) {
1083         block = res.resolveReference(&value, block, &ref, &typeSpecFlags);
1084         if (kThrowOnBadId) {
1085             if (block == BAD_INDEX) {
1086                 jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
1087                 return 0;
1088             }
1089         }
1090     }
1091     return block >= 0 ? copyValue(env, outValue, &res, value, ref, block, typeSpecFlags) : block;
1092 }
1093 
android_content_AssetManager_getThemeChangingConfigurations(JNIEnv * env,jobject clazz,jlong themeHandle)1094 static jint android_content_AssetManager_getThemeChangingConfigurations(JNIEnv* env, jobject clazz,
1095                                                                         jlong themeHandle)
1096 {
1097     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeHandle);
1098     return theme->getChangingConfigurations();
1099 }
1100 
android_content_AssetManager_dumpTheme(JNIEnv * env,jobject clazz,jlong themeHandle,jint pri,jstring tag,jstring prefix)1101 static void android_content_AssetManager_dumpTheme(JNIEnv* env, jobject clazz,
1102                                                    jlong themeHandle, jint pri,
1103                                                    jstring tag, jstring prefix)
1104 {
1105     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeHandle);
1106     const ResTable& res(theme->getResTable());
1107     (void)res;
1108 
1109     // XXX Need to use params.
1110     theme->dumpToLog();
1111 }
1112 
android_content_AssetManager_resolveAttrs(JNIEnv * env,jobject clazz,jlong themeToken,jint defStyleAttr,jint defStyleRes,jintArray inValues,jintArray attrs,jintArray outValues,jintArray outIndices)1113 static jboolean android_content_AssetManager_resolveAttrs(JNIEnv* env, jobject clazz,
1114                                                           jlong themeToken,
1115                                                           jint defStyleAttr,
1116                                                           jint defStyleRes,
1117                                                           jintArray inValues,
1118                                                           jintArray attrs,
1119                                                           jintArray outValues,
1120                                                           jintArray outIndices)
1121 {
1122     if (themeToken == 0) {
1123         jniThrowNullPointerException(env, "theme token");
1124         return JNI_FALSE;
1125     }
1126     if (attrs == NULL) {
1127         jniThrowNullPointerException(env, "attrs");
1128         return JNI_FALSE;
1129     }
1130     if (outValues == NULL) {
1131         jniThrowNullPointerException(env, "out values");
1132         return JNI_FALSE;
1133     }
1134 
1135     const jsize NI = env->GetArrayLength(attrs);
1136     const jsize NV = env->GetArrayLength(outValues);
1137     if (NV < (NI*STYLE_NUM_ENTRIES)) {
1138         jniThrowException(env, "java/lang/IndexOutOfBoundsException", "out values too small");
1139         return JNI_FALSE;
1140     }
1141 
1142     jint* src = (jint*)env->GetPrimitiveArrayCritical(attrs, 0);
1143     if (src == NULL) {
1144         return JNI_FALSE;
1145     }
1146 
1147     jint* srcValues = (jint*)env->GetPrimitiveArrayCritical(inValues, 0);
1148     const jsize NSV = srcValues == NULL ? 0 : env->GetArrayLength(inValues);
1149 
1150     jint* baseDest = (jint*)env->GetPrimitiveArrayCritical(outValues, 0);
1151     if (baseDest == NULL) {
1152         env->ReleasePrimitiveArrayCritical(attrs, src, 0);
1153         return JNI_FALSE;
1154     }
1155 
1156     jint* indices = NULL;
1157     if (outIndices != NULL) {
1158         if (env->GetArrayLength(outIndices) > NI) {
1159             indices = (jint*)env->GetPrimitiveArrayCritical(outIndices, 0);
1160         }
1161     }
1162 
1163     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeToken);
1164     bool result = ResolveAttrs(theme, defStyleAttr, defStyleRes,
1165                                (uint32_t*) srcValues, NSV,
1166                                (uint32_t*) src, NI,
1167                                (uint32_t*) baseDest,
1168                                (uint32_t*) indices);
1169 
1170     if (indices != NULL) {
1171         env->ReleasePrimitiveArrayCritical(outIndices, indices, 0);
1172     }
1173     env->ReleasePrimitiveArrayCritical(outValues, baseDest, 0);
1174     env->ReleasePrimitiveArrayCritical(inValues, srcValues, 0);
1175     env->ReleasePrimitiveArrayCritical(attrs, src, 0);
1176     return result ? JNI_TRUE : JNI_FALSE;
1177 }
1178 
android_content_AssetManager_applyStyle(JNIEnv * env,jobject,jlong themeToken,jint defStyleAttr,jint defStyleRes,jlong xmlParserToken,jintArray attrsObj,jint length,jlong outValuesAddress,jlong outIndicesAddress)1179 static void android_content_AssetManager_applyStyle(JNIEnv* env, jobject, jlong themeToken,
1180         jint defStyleAttr, jint defStyleRes, jlong xmlParserToken, jintArray attrsObj, jint length,
1181         jlong outValuesAddress, jlong outIndicesAddress) {
1182     jint* attrs = env->GetIntArrayElements(attrsObj, 0);
1183     ResTable::Theme* theme = reinterpret_cast<ResTable::Theme*>(themeToken);
1184     ResXMLParser* xmlParser = reinterpret_cast<ResXMLParser*>(xmlParserToken);
1185     uint32_t* outValues = reinterpret_cast<uint32_t*>(static_cast<uintptr_t>(outValuesAddress));
1186     uint32_t* outIndices = reinterpret_cast<uint32_t*>(static_cast<uintptr_t>(outIndicesAddress));
1187     ApplyStyle(theme, xmlParser, defStyleAttr, defStyleRes,
1188             reinterpret_cast<const uint32_t*>(attrs), length, outValues, outIndices);
1189     env->ReleaseIntArrayElements(attrsObj, attrs, JNI_ABORT);
1190 }
1191 
android_content_AssetManager_retrieveAttributes(JNIEnv * env,jobject clazz,jlong xmlParserToken,jintArray attrs,jintArray outValues,jintArray outIndices)1192 static jboolean android_content_AssetManager_retrieveAttributes(JNIEnv* env, jobject clazz,
1193                                                         jlong xmlParserToken,
1194                                                         jintArray attrs,
1195                                                         jintArray outValues,
1196                                                         jintArray outIndices)
1197 {
1198     if (xmlParserToken == 0) {
1199         jniThrowNullPointerException(env, "xmlParserToken");
1200         return JNI_FALSE;
1201     }
1202     if (attrs == NULL) {
1203         jniThrowNullPointerException(env, "attrs");
1204         return JNI_FALSE;
1205     }
1206     if (outValues == NULL) {
1207         jniThrowNullPointerException(env, "out values");
1208         return JNI_FALSE;
1209     }
1210 
1211     AssetManager* am = assetManagerForJavaObject(env, clazz);
1212     if (am == NULL) {
1213         return JNI_FALSE;
1214     }
1215     const ResTable& res(am->getResources());
1216     ResXMLParser* xmlParser = (ResXMLParser*)xmlParserToken;
1217 
1218     const jsize NI = env->GetArrayLength(attrs);
1219     const jsize NV = env->GetArrayLength(outValues);
1220     if (NV < (NI*STYLE_NUM_ENTRIES)) {
1221         jniThrowException(env, "java/lang/IndexOutOfBoundsException", "out values too small");
1222         return JNI_FALSE;
1223     }
1224 
1225     jint* src = (jint*)env->GetPrimitiveArrayCritical(attrs, 0);
1226     if (src == NULL) {
1227         return JNI_FALSE;
1228     }
1229 
1230     jint* baseDest = (jint*)env->GetPrimitiveArrayCritical(outValues, 0);
1231     if (baseDest == NULL) {
1232         env->ReleasePrimitiveArrayCritical(attrs, src, 0);
1233         return JNI_FALSE;
1234     }
1235 
1236     jint* indices = NULL;
1237     if (outIndices != NULL) {
1238         if (env->GetArrayLength(outIndices) > NI) {
1239             indices = (jint*)env->GetPrimitiveArrayCritical(outIndices, 0);
1240         }
1241     }
1242 
1243     bool result = RetrieveAttributes(&res, xmlParser,
1244                                      (uint32_t*) src, NI,
1245                                      (uint32_t*) baseDest,
1246                                      (uint32_t*) indices);
1247 
1248     if (indices != NULL) {
1249         env->ReleasePrimitiveArrayCritical(outIndices, indices, 0);
1250     }
1251     env->ReleasePrimitiveArrayCritical(outValues, baseDest, 0);
1252     env->ReleasePrimitiveArrayCritical(attrs, src, 0);
1253     return result ? JNI_TRUE : JNI_FALSE;
1254 }
1255 
android_content_AssetManager_getArraySize(JNIEnv * env,jobject clazz,jint id)1256 static jint android_content_AssetManager_getArraySize(JNIEnv* env, jobject clazz,
1257                                                        jint id)
1258 {
1259     AssetManager* am = assetManagerForJavaObject(env, clazz);
1260     if (am == NULL) {
1261         return 0;
1262     }
1263     const ResTable& res(am->getResources());
1264 
1265     res.lock();
1266     const ResTable::bag_entry* defStyleEnt = NULL;
1267     ssize_t bagOff = res.getBagLocked(id, &defStyleEnt);
1268     res.unlock();
1269 
1270     return static_cast<jint>(bagOff);
1271 }
1272 
android_content_AssetManager_retrieveArray(JNIEnv * env,jobject clazz,jint id,jintArray outValues)1273 static jint android_content_AssetManager_retrieveArray(JNIEnv* env, jobject clazz,
1274                                                         jint id,
1275                                                         jintArray outValues)
1276 {
1277     if (outValues == NULL) {
1278         jniThrowNullPointerException(env, "out values");
1279         return JNI_FALSE;
1280     }
1281 
1282     AssetManager* am = assetManagerForJavaObject(env, clazz);
1283     if (am == NULL) {
1284         return JNI_FALSE;
1285     }
1286     const ResTable& res(am->getResources());
1287     ResTable_config config;
1288     Res_value value;
1289     ssize_t block;
1290 
1291     const jsize NV = env->GetArrayLength(outValues);
1292 
1293     jint* baseDest = (jint*)env->GetPrimitiveArrayCritical(outValues, 0);
1294     jint* dest = baseDest;
1295     if (dest == NULL) {
1296         jniThrowException(env, "java/lang/OutOfMemoryError", "");
1297         return JNI_FALSE;
1298     }
1299 
1300     // Now lock down the resource object and start pulling stuff from it.
1301     res.lock();
1302 
1303     const ResTable::bag_entry* arrayEnt = NULL;
1304     uint32_t arrayTypeSetFlags = 0;
1305     ssize_t bagOff = res.getBagLocked(id, &arrayEnt, &arrayTypeSetFlags);
1306     const ResTable::bag_entry* endArrayEnt = arrayEnt +
1307         (bagOff >= 0 ? bagOff : 0);
1308 
1309     int i = 0;
1310     uint32_t typeSetFlags;
1311     while (i < NV && arrayEnt < endArrayEnt) {
1312         block = arrayEnt->stringBlock;
1313         typeSetFlags = arrayTypeSetFlags;
1314         config.density = 0;
1315         value = arrayEnt->map.value;
1316 
1317         uint32_t resid = 0;
1318         if (value.dataType != Res_value::TYPE_NULL) {
1319             // Take care of resolving the found resource to its final value.
1320             //printf("Resolving attribute reference\n");
1321             ssize_t newBlock = res.resolveReference(&value, block, &resid,
1322                     &typeSetFlags, &config);
1323             if (kThrowOnBadId) {
1324                 if (newBlock == BAD_INDEX) {
1325                     jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
1326                     return JNI_FALSE;
1327                 }
1328             }
1329             if (newBlock >= 0) block = newBlock;
1330         }
1331 
1332         // Deal with the special @null value -- it turns back to TYPE_NULL.
1333         if (value.dataType == Res_value::TYPE_REFERENCE && value.data == 0) {
1334             value.dataType = Res_value::TYPE_NULL;
1335             value.data = Res_value::DATA_NULL_UNDEFINED;
1336         }
1337 
1338         //printf("Attribute 0x%08x: final type=0x%x, data=0x%08x\n", curIdent, value.dataType, value.data);
1339 
1340         // Write the final value back to Java.
1341         dest[STYLE_TYPE] = value.dataType;
1342         dest[STYLE_DATA] = value.data;
1343         dest[STYLE_ASSET_COOKIE] = reinterpret_cast<jint>(res.getTableCookie(block));
1344         dest[STYLE_RESOURCE_ID] = resid;
1345         dest[STYLE_CHANGING_CONFIGURATIONS] = typeSetFlags;
1346         dest[STYLE_DENSITY] = config.density;
1347         dest += STYLE_NUM_ENTRIES;
1348         i+= STYLE_NUM_ENTRIES;
1349         arrayEnt++;
1350     }
1351 
1352     i /= STYLE_NUM_ENTRIES;
1353 
1354     res.unlock();
1355 
1356     env->ReleasePrimitiveArrayCritical(outValues, baseDest, 0);
1357 
1358     return i;
1359 }
1360 
android_content_AssetManager_openXmlAssetNative(JNIEnv * env,jobject clazz,jint cookie,jstring fileName)1361 static jlong android_content_AssetManager_openXmlAssetNative(JNIEnv* env, jobject clazz,
1362                                                          jint cookie,
1363                                                          jstring fileName)
1364 {
1365     AssetManager* am = assetManagerForJavaObject(env, clazz);
1366     if (am == NULL) {
1367         return 0;
1368     }
1369 
1370     ALOGV("openXmlAsset in %p (Java object %p)\n", am, clazz);
1371 
1372     ScopedUtfChars fileName8(env, fileName);
1373     if (fileName8.c_str() == NULL) {
1374         return 0;
1375     }
1376 
1377     int32_t assetCookie = static_cast<int32_t>(cookie);
1378     Asset* a = assetCookie
1379         ? am->openNonAsset(assetCookie, fileName8.c_str(), Asset::ACCESS_BUFFER)
1380         : am->openNonAsset(fileName8.c_str(), Asset::ACCESS_BUFFER, &assetCookie);
1381 
1382     if (a == NULL) {
1383         jniThrowException(env, "java/io/FileNotFoundException", fileName8.c_str());
1384         return 0;
1385     }
1386 
1387     const DynamicRefTable* dynamicRefTable =
1388             am->getResources().getDynamicRefTableForCookie(assetCookie);
1389     ResXMLTree* block = new ResXMLTree(dynamicRefTable);
1390     status_t err = block->setTo(a->getBuffer(true), a->getLength(), true);
1391     a->close();
1392     delete a;
1393 
1394     if (err != NO_ERROR) {
1395         jniThrowException(env, "java/io/FileNotFoundException", "Corrupt XML binary file");
1396         return 0;
1397     }
1398 
1399     return reinterpret_cast<jlong>(block);
1400 }
1401 
android_content_AssetManager_getArrayStringInfo(JNIEnv * env,jobject clazz,jint arrayResId)1402 static jintArray android_content_AssetManager_getArrayStringInfo(JNIEnv* env, jobject clazz,
1403                                                                  jint arrayResId)
1404 {
1405     AssetManager* am = assetManagerForJavaObject(env, clazz);
1406     if (am == NULL) {
1407         return NULL;
1408     }
1409     const ResTable& res(am->getResources());
1410 
1411     const ResTable::bag_entry* startOfBag;
1412     const ssize_t N = res.lockBag(arrayResId, &startOfBag);
1413     if (N < 0) {
1414         return NULL;
1415     }
1416 
1417     jintArray array = env->NewIntArray(N * 2);
1418     if (array == NULL) {
1419         res.unlockBag(startOfBag);
1420         return NULL;
1421     }
1422 
1423     Res_value value;
1424     const ResTable::bag_entry* bag = startOfBag;
1425     for (size_t i = 0, j = 0; ((ssize_t)i)<N; i++, bag++) {
1426         jint stringIndex = -1;
1427         jint stringBlock = 0;
1428         value = bag->map.value;
1429 
1430         // Take care of resolving the found resource to its final value.
1431         stringBlock = res.resolveReference(&value, bag->stringBlock, NULL);
1432         if (value.dataType == Res_value::TYPE_STRING) {
1433             stringIndex = value.data;
1434         }
1435 
1436         if (kThrowOnBadId) {
1437             if (stringBlock == BAD_INDEX) {
1438                 jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
1439                 return array;
1440             }
1441         }
1442 
1443         //todo: It might be faster to allocate a C array to contain
1444         //      the blocknums and indices, put them in there and then
1445         //      do just one SetIntArrayRegion()
1446         env->SetIntArrayRegion(array, j, 1, &stringBlock);
1447         env->SetIntArrayRegion(array, j + 1, 1, &stringIndex);
1448         j = j + 2;
1449     }
1450     res.unlockBag(startOfBag);
1451     return array;
1452 }
1453 
android_content_AssetManager_getArrayStringResource(JNIEnv * env,jobject clazz,jint arrayResId)1454 static jobjectArray android_content_AssetManager_getArrayStringResource(JNIEnv* env, jobject clazz,
1455                                                                         jint arrayResId)
1456 {
1457     AssetManager* am = assetManagerForJavaObject(env, clazz);
1458     if (am == NULL) {
1459         return NULL;
1460     }
1461     const ResTable& res(am->getResources());
1462 
1463     const ResTable::bag_entry* startOfBag;
1464     const ssize_t N = res.lockBag(arrayResId, &startOfBag);
1465     if (N < 0) {
1466         return NULL;
1467     }
1468 
1469     jobjectArray array = env->NewObjectArray(N, g_stringClass, NULL);
1470     if (env->ExceptionCheck()) {
1471         res.unlockBag(startOfBag);
1472         return NULL;
1473     }
1474 
1475     Res_value value;
1476     const ResTable::bag_entry* bag = startOfBag;
1477     size_t strLen = 0;
1478     for (size_t i=0; ((ssize_t)i)<N; i++, bag++) {
1479         value = bag->map.value;
1480         jstring str = NULL;
1481 
1482         // Take care of resolving the found resource to its final value.
1483         ssize_t block = res.resolveReference(&value, bag->stringBlock, NULL);
1484         if (kThrowOnBadId) {
1485             if (block == BAD_INDEX) {
1486                 jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
1487                 return array;
1488             }
1489         }
1490         if (value.dataType == Res_value::TYPE_STRING) {
1491             const ResStringPool* pool = res.getTableStringBlock(block);
1492             const char* str8 = pool->string8At(value.data, &strLen);
1493             if (str8 != NULL) {
1494                 str = env->NewStringUTF(str8);
1495             } else {
1496                 const char16_t* str16 = pool->stringAt(value.data, &strLen);
1497                 str = env->NewString(reinterpret_cast<const jchar*>(str16),
1498                                      strLen);
1499             }
1500 
1501             // If one of our NewString{UTF} calls failed due to memory, an
1502             // exception will be pending.
1503             if (env->ExceptionCheck()) {
1504                 res.unlockBag(startOfBag);
1505                 return NULL;
1506             }
1507 
1508             env->SetObjectArrayElement(array, i, str);
1509 
1510             // str is not NULL at that point, otherwise ExceptionCheck would have been true.
1511             // If we have a large amount of strings in our array, we might
1512             // overflow the local reference table of the VM.
1513             env->DeleteLocalRef(str);
1514         }
1515     }
1516     res.unlockBag(startOfBag);
1517     return array;
1518 }
1519 
android_content_AssetManager_getArrayIntResource(JNIEnv * env,jobject clazz,jint arrayResId)1520 static jintArray android_content_AssetManager_getArrayIntResource(JNIEnv* env, jobject clazz,
1521                                                                         jint arrayResId)
1522 {
1523     AssetManager* am = assetManagerForJavaObject(env, clazz);
1524     if (am == NULL) {
1525         return NULL;
1526     }
1527     const ResTable& res(am->getResources());
1528 
1529     const ResTable::bag_entry* startOfBag;
1530     const ssize_t N = res.lockBag(arrayResId, &startOfBag);
1531     if (N < 0) {
1532         return NULL;
1533     }
1534 
1535     jintArray array = env->NewIntArray(N);
1536     if (array == NULL) {
1537         res.unlockBag(startOfBag);
1538         return NULL;
1539     }
1540 
1541     Res_value value;
1542     const ResTable::bag_entry* bag = startOfBag;
1543     for (size_t i=0; ((ssize_t)i)<N; i++, bag++) {
1544         value = bag->map.value;
1545 
1546         // Take care of resolving the found resource to its final value.
1547         ssize_t block = res.resolveReference(&value, bag->stringBlock, NULL);
1548         if (kThrowOnBadId) {
1549             if (block == BAD_INDEX) {
1550                 jniThrowException(env, "java/lang/IllegalStateException", "Bad resource!");
1551                 return array;
1552             }
1553         }
1554         if (value.dataType >= Res_value::TYPE_FIRST_INT
1555                 && value.dataType <= Res_value::TYPE_LAST_INT) {
1556             int intVal = value.data;
1557             env->SetIntArrayRegion(array, i, 1, &intVal);
1558         }
1559     }
1560     res.unlockBag(startOfBag);
1561     return array;
1562 }
1563 
android_content_AssetManager_getStyleAttributes(JNIEnv * env,jobject clazz,jint styleId)1564 static jintArray android_content_AssetManager_getStyleAttributes(JNIEnv* env, jobject clazz,
1565                                                                  jint styleId)
1566 {
1567     AssetManager* am = assetManagerForJavaObject(env, clazz);
1568     if (am == NULL) {
1569         return NULL;
1570     }
1571     const ResTable& res(am->getResources());
1572 
1573     const ResTable::bag_entry* startOfBag;
1574     const ssize_t N = res.lockBag(styleId, &startOfBag);
1575     if (N < 0) {
1576         return NULL;
1577     }
1578 
1579     jintArray array = env->NewIntArray(N);
1580     if (array == NULL) {
1581         res.unlockBag(startOfBag);
1582         return NULL;
1583     }
1584 
1585     const ResTable::bag_entry* bag = startOfBag;
1586     for (size_t i=0; ((ssize_t)i)<N; i++, bag++) {
1587         int resourceId = bag->map.name.ident;
1588         env->SetIntArrayRegion(array, i, 1, &resourceId);
1589     }
1590     res.unlockBag(startOfBag);
1591     return array;
1592 }
1593 
android_content_AssetManager_init(JNIEnv * env,jobject clazz,jboolean isSystem)1594 static void android_content_AssetManager_init(JNIEnv* env, jobject clazz, jboolean isSystem)
1595 {
1596     if (isSystem) {
1597         verifySystemIdmaps();
1598     }
1599     AssetManager* am = new AssetManager();
1600     if (am == NULL) {
1601         jniThrowException(env, "java/lang/OutOfMemoryError", "");
1602         return;
1603     }
1604 
1605     am->addDefaultAssets();
1606 
1607     ALOGV("Created AssetManager %p for Java object %p\n", am, clazz);
1608     env->SetLongField(clazz, gAssetManagerOffsets.mObject, reinterpret_cast<jlong>(am));
1609 }
1610 
android_content_AssetManager_destroy(JNIEnv * env,jobject clazz)1611 static void android_content_AssetManager_destroy(JNIEnv* env, jobject clazz)
1612 {
1613     AssetManager* am = (AssetManager*)
1614         (env->GetLongField(clazz, gAssetManagerOffsets.mObject));
1615     ALOGV("Destroying AssetManager %p for Java object %p\n", am, clazz);
1616     if (am != NULL) {
1617         delete am;
1618         env->SetLongField(clazz, gAssetManagerOffsets.mObject, 0);
1619     }
1620 }
1621 
android_content_AssetManager_getGlobalAssetCount(JNIEnv * env,jobject clazz)1622 static jint android_content_AssetManager_getGlobalAssetCount(JNIEnv* env, jobject clazz)
1623 {
1624     return Asset::getGlobalCount();
1625 }
1626 
android_content_AssetManager_getAssetAllocations(JNIEnv * env,jobject clazz)1627 static jobject android_content_AssetManager_getAssetAllocations(JNIEnv* env, jobject clazz)
1628 {
1629     String8 alloc = Asset::getAssetAllocations();
1630     if (alloc.length() <= 0) {
1631         return NULL;
1632     }
1633 
1634     jstring str = env->NewStringUTF(alloc.string());
1635     return str;
1636 }
1637 
android_content_AssetManager_getGlobalAssetManagerCount(JNIEnv * env,jobject clazz)1638 static jint android_content_AssetManager_getGlobalAssetManagerCount(JNIEnv* env, jobject clazz)
1639 {
1640     return AssetManager::getGlobalCount();
1641 }
1642 
1643 // ----------------------------------------------------------------------------
1644 
1645 /*
1646  * JNI registration.
1647  */
1648 static const JNINativeMethod gAssetManagerMethods[] = {
1649     /* name, signature, funcPtr */
1650 
1651     // Basic asset stuff.
1652     { "openAsset",      "(Ljava/lang/String;I)J",
1653         (void*) android_content_AssetManager_openAsset },
1654     { "openAssetFd",      "(Ljava/lang/String;[J)Landroid/os/ParcelFileDescriptor;",
1655         (void*) android_content_AssetManager_openAssetFd },
1656     { "openNonAssetNative", "(ILjava/lang/String;I)J",
1657         (void*) android_content_AssetManager_openNonAssetNative },
1658     { "openNonAssetFdNative", "(ILjava/lang/String;[J)Landroid/os/ParcelFileDescriptor;",
1659         (void*) android_content_AssetManager_openNonAssetFdNative },
1660     { "list",           "(Ljava/lang/String;)[Ljava/lang/String;",
1661         (void*) android_content_AssetManager_list },
1662     { "destroyAsset",   "(J)V",
1663         (void*) android_content_AssetManager_destroyAsset },
1664     { "readAssetChar",  "(J)I",
1665         (void*) android_content_AssetManager_readAssetChar },
1666     { "readAsset",      "(J[BII)I",
1667         (void*) android_content_AssetManager_readAsset },
1668     { "seekAsset",      "(JJI)J",
1669         (void*) android_content_AssetManager_seekAsset },
1670     { "getAssetLength", "(J)J",
1671         (void*) android_content_AssetManager_getAssetLength },
1672     { "getAssetRemainingLength", "(J)J",
1673         (void*) android_content_AssetManager_getAssetRemainingLength },
1674     { "addAssetPathNative", "(Ljava/lang/String;Z)I",
1675         (void*) android_content_AssetManager_addAssetPath },
1676     { "addOverlayPathNative",   "(Ljava/lang/String;)I",
1677         (void*) android_content_AssetManager_addOverlayPath },
1678     { "isUpToDate",     "()Z",
1679         (void*) android_content_AssetManager_isUpToDate },
1680 
1681     // Resources.
1682     { "getLocales",      "()[Ljava/lang/String;",
1683         (void*) android_content_AssetManager_getLocales },
1684     { "getNonSystemLocales", "()[Ljava/lang/String;",
1685         (void*) android_content_AssetManager_getNonSystemLocales },
1686     { "getSizeConfigurations", "()[Landroid/content/res/Configuration;",
1687         (void*) android_content_AssetManager_getSizeConfigurations },
1688     { "setConfiguration", "(IILjava/lang/String;IIIIIIIIIIIIIII)V",
1689         (void*) android_content_AssetManager_setConfiguration },
1690     { "getResourceIdentifier","(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)I",
1691         (void*) android_content_AssetManager_getResourceIdentifier },
1692     { "getResourceName","(I)Ljava/lang/String;",
1693         (void*) android_content_AssetManager_getResourceName },
1694     { "getResourcePackageName","(I)Ljava/lang/String;",
1695         (void*) android_content_AssetManager_getResourcePackageName },
1696     { "getResourceTypeName","(I)Ljava/lang/String;",
1697         (void*) android_content_AssetManager_getResourceTypeName },
1698     { "getResourceEntryName","(I)Ljava/lang/String;",
1699         (void*) android_content_AssetManager_getResourceEntryName },
1700     { "loadResourceValue","(ISLandroid/util/TypedValue;Z)I",
1701         (void*) android_content_AssetManager_loadResourceValue },
1702     { "loadResourceBagValue","(IILandroid/util/TypedValue;Z)I",
1703         (void*) android_content_AssetManager_loadResourceBagValue },
1704     { "getStringBlockCount","()I",
1705         (void*) android_content_AssetManager_getStringBlockCount },
1706     { "getNativeStringBlock","(I)J",
1707         (void*) android_content_AssetManager_getNativeStringBlock },
1708     { "getCookieName","(I)Ljava/lang/String;",
1709         (void*) android_content_AssetManager_getCookieName },
1710     { "getAssignedPackageIdentifiers","()Landroid/util/SparseArray;",
1711         (void*) android_content_AssetManager_getAssignedPackageIdentifiers },
1712 
1713     // Themes.
1714     { "newTheme", "()J",
1715         (void*) android_content_AssetManager_newTheme },
1716     { "deleteTheme", "(J)V",
1717         (void*) android_content_AssetManager_deleteTheme },
1718     { "applyThemeStyle", "(JIZ)V",
1719         (void*) android_content_AssetManager_applyThemeStyle },
1720     { "copyTheme", "(JJ)V",
1721         (void*) android_content_AssetManager_copyTheme },
1722     { "clearTheme", "(J)V",
1723         (void*) android_content_AssetManager_clearTheme },
1724     { "loadThemeAttributeValue", "(JILandroid/util/TypedValue;Z)I",
1725         (void*) android_content_AssetManager_loadThemeAttributeValue },
1726     { "getThemeChangingConfigurations", "(J)I",
1727         (void*) android_content_AssetManager_getThemeChangingConfigurations },
1728     { "dumpTheme", "(JILjava/lang/String;Ljava/lang/String;)V",
1729         (void*) android_content_AssetManager_dumpTheme },
1730     { "applyStyle","(JIIJ[IIJJ)V",
1731         (void*) android_content_AssetManager_applyStyle },
1732     { "resolveAttrs","(JII[I[I[I[I)Z",
1733         (void*) android_content_AssetManager_resolveAttrs },
1734     { "retrieveAttributes","(J[I[I[I)Z",
1735         (void*) android_content_AssetManager_retrieveAttributes },
1736     { "getArraySize","(I)I",
1737         (void*) android_content_AssetManager_getArraySize },
1738     { "retrieveArray","(I[I)I",
1739         (void*) android_content_AssetManager_retrieveArray },
1740 
1741     // XML files.
1742     { "openXmlAssetNative", "(ILjava/lang/String;)J",
1743         (void*) android_content_AssetManager_openXmlAssetNative },
1744 
1745     // Arrays.
1746     { "getArrayStringResource","(I)[Ljava/lang/String;",
1747         (void*) android_content_AssetManager_getArrayStringResource },
1748     { "getArrayStringInfo","(I)[I",
1749         (void*) android_content_AssetManager_getArrayStringInfo },
1750     { "getArrayIntResource","(I)[I",
1751         (void*) android_content_AssetManager_getArrayIntResource },
1752     { "getStyleAttributes","(I)[I",
1753         (void*) android_content_AssetManager_getStyleAttributes },
1754 
1755     // Bookkeeping.
1756     { "init",           "(Z)V",
1757         (void*) android_content_AssetManager_init },
1758     { "destroy",        "()V",
1759         (void*) android_content_AssetManager_destroy },
1760     { "getGlobalAssetCount", "()I",
1761         (void*) android_content_AssetManager_getGlobalAssetCount },
1762     { "getAssetAllocations", "()Ljava/lang/String;",
1763         (void*) android_content_AssetManager_getAssetAllocations },
1764     { "getGlobalAssetManagerCount", "()I",
1765         (void*) android_content_AssetManager_getGlobalAssetManagerCount },
1766 };
1767 
register_android_content_AssetManager(JNIEnv * env)1768 int register_android_content_AssetManager(JNIEnv* env)
1769 {
1770     jclass typedValue = FindClassOrDie(env, "android/util/TypedValue");
1771     gTypedValueOffsets.mType = GetFieldIDOrDie(env, typedValue, "type", "I");
1772     gTypedValueOffsets.mData = GetFieldIDOrDie(env, typedValue, "data", "I");
1773     gTypedValueOffsets.mString = GetFieldIDOrDie(env, typedValue, "string",
1774                                                  "Ljava/lang/CharSequence;");
1775     gTypedValueOffsets.mAssetCookie = GetFieldIDOrDie(env, typedValue, "assetCookie", "I");
1776     gTypedValueOffsets.mResourceId = GetFieldIDOrDie(env, typedValue, "resourceId", "I");
1777     gTypedValueOffsets.mChangingConfigurations = GetFieldIDOrDie(env, typedValue,
1778                                                                  "changingConfigurations", "I");
1779     gTypedValueOffsets.mDensity = GetFieldIDOrDie(env, typedValue, "density", "I");
1780 
1781     jclass assetFd = FindClassOrDie(env, "android/content/res/AssetFileDescriptor");
1782     gAssetFileDescriptorOffsets.mFd = GetFieldIDOrDie(env, assetFd, "mFd",
1783                                                       "Landroid/os/ParcelFileDescriptor;");
1784     gAssetFileDescriptorOffsets.mStartOffset = GetFieldIDOrDie(env, assetFd, "mStartOffset", "J");
1785     gAssetFileDescriptorOffsets.mLength = GetFieldIDOrDie(env, assetFd, "mLength", "J");
1786 
1787     jclass assetManager = FindClassOrDie(env, "android/content/res/AssetManager");
1788     gAssetManagerOffsets.mObject = GetFieldIDOrDie(env, assetManager, "mObject", "J");
1789 
1790     jclass stringClass = FindClassOrDie(env, "java/lang/String");
1791     g_stringClass = MakeGlobalRefOrDie(env, stringClass);
1792 
1793     jclass sparseArrayClass = FindClassOrDie(env, "android/util/SparseArray");
1794     gSparseArrayOffsets.classObject = MakeGlobalRefOrDie(env, sparseArrayClass);
1795     gSparseArrayOffsets.constructor = GetMethodIDOrDie(env, gSparseArrayOffsets.classObject,
1796                                                        "<init>", "()V");
1797     gSparseArrayOffsets.put = GetMethodIDOrDie(env, gSparseArrayOffsets.classObject, "put",
1798                                                "(ILjava/lang/Object;)V");
1799 
1800     jclass configurationClass = FindClassOrDie(env, "android/content/res/Configuration");
1801     gConfigurationOffsets.classObject = MakeGlobalRefOrDie(env, configurationClass);
1802     gConfigurationOffsets.constructor = GetMethodIDOrDie(env, configurationClass,
1803             "<init>", "()V");
1804     gConfigurationOffsets.mSmallestScreenWidthDpOffset = GetFieldIDOrDie(env, configurationClass,
1805             "smallestScreenWidthDp", "I");
1806     gConfigurationOffsets.mScreenWidthDpOffset = GetFieldIDOrDie(env, configurationClass,
1807             "screenWidthDp", "I");
1808     gConfigurationOffsets.mScreenHeightDpOffset = GetFieldIDOrDie(env, configurationClass,
1809             "screenHeightDp", "I");
1810 
1811     return RegisterMethodsOrDie(env, "android/content/res/AssetManager", gAssetManagerMethods,
1812                                 NELEM(gAssetManagerMethods));
1813 }
1814 
1815 }; // namespace android
1816