Home | History | Annotate | Download | only in jni
      1 /*
      2  * Copyright (C) 2006 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #define LOG_TAG "JavaBinder"
     18 //#define LOG_NDEBUG 0
     19 
     20 #include "android_util_Binder.h"
     21 #include "JNIHelp.h"
     22 
     23 #include <fcntl.h>
     24 #include <stdio.h>
     25 #include <sys/stat.h>
     26 #include <sys/types.h>
     27 #include <unistd.h>
     28 
     29 #include <utils/Atomic.h>
     30 #include <binder/IInterface.h>
     31 #include <binder/IPCThreadState.h>
     32 #include <utils/Log.h>
     33 #include <utils/SystemClock.h>
     34 #include <utils/List.h>
     35 #include <utils/KeyedVector.h>
     36 #include <cutils/logger.h>
     37 #include <binder/Parcel.h>
     38 #include <binder/ProcessState.h>
     39 #include <binder/IServiceManager.h>
     40 #include <utils/threads.h>
     41 
     42 #include <ScopedUtfChars.h>
     43 #include <ScopedLocalRef.h>
     44 
     45 #include <android_runtime/AndroidRuntime.h>
     46 
     47 //#undef LOGV
     48 //#define LOGV(...) fprintf(stderr, __VA_ARGS__)
     49 
     50 #define DEBUG_DEATH 0
     51 #if DEBUG_DEATH
     52 #define LOGDEATH LOGD
     53 #else
     54 #define LOGDEATH LOGV
     55 #endif
     56 
     57 using namespace android;
     58 
     59 // ----------------------------------------------------------------------------
     60 
     61 static struct bindernative_offsets_t
     62 {
     63     // Class state.
     64     jclass mClass;
     65     jmethodID mExecTransact;
     66 
     67     // Object state.
     68     jfieldID mObject;
     69 
     70 } gBinderOffsets;
     71 
     72 // ----------------------------------------------------------------------------
     73 
     74 static struct binderinternal_offsets_t
     75 {
     76     // Class state.
     77     jclass mClass;
     78     jmethodID mForceGc;
     79 
     80 } gBinderInternalOffsets;
     81 
     82 // ----------------------------------------------------------------------------
     83 
     84 static struct debug_offsets_t
     85 {
     86     // Class state.
     87     jclass mClass;
     88 
     89 } gDebugOffsets;
     90 
     91 // ----------------------------------------------------------------------------
     92 
     93 static struct weakreference_offsets_t
     94 {
     95     // Class state.
     96     jclass mClass;
     97     jmethodID mGet;
     98 
     99 } gWeakReferenceOffsets;
    100 
    101 static struct error_offsets_t
    102 {
    103     jclass mClass;
    104 } gErrorOffsets;
    105 
    106 // ----------------------------------------------------------------------------
    107 
    108 static struct binderproxy_offsets_t
    109 {
    110     // Class state.
    111     jclass mClass;
    112     jmethodID mConstructor;
    113     jmethodID mSendDeathNotice;
    114 
    115     // Object state.
    116     jfieldID mObject;
    117     jfieldID mSelf;
    118     jfieldID mOrgue;
    119 
    120 } gBinderProxyOffsets;
    121 
    122 static struct class_offsets_t
    123 {
    124     jmethodID mGetName;
    125 } gClassOffsets;
    126 
    127 // ----------------------------------------------------------------------------
    128 
    129 static struct parcel_offsets_t
    130 {
    131     jfieldID mObject;
    132     jfieldID mOwnObject;
    133 } gParcelOffsets;
    134 
    135 static struct log_offsets_t
    136 {
    137     // Class state.
    138     jclass mClass;
    139     jmethodID mLogE;
    140 } gLogOffsets;
    141 
    142 static struct parcel_file_descriptor_offsets_t
    143 {
    144     jclass mClass;
    145     jmethodID mConstructor;
    146 } gParcelFileDescriptorOffsets;
    147 
    148 static struct strict_mode_callback_offsets_t
    149 {
    150     jclass mClass;
    151     jmethodID mCallback;
    152 } gStrictModeCallbackOffsets;
    153 
    154 // ****************************************************************************
    155 // ****************************************************************************
    156 // ****************************************************************************
    157 
    158 static volatile int32_t gNumRefsCreated = 0;
    159 static volatile int32_t gNumProxyRefs = 0;
    160 static volatile int32_t gNumLocalRefs = 0;
    161 static volatile int32_t gNumDeathRefs = 0;
    162 
    163 static void incRefsCreated(JNIEnv* env)
    164 {
    165     int old = android_atomic_inc(&gNumRefsCreated);
    166     if (old == 200) {
    167         android_atomic_and(0, &gNumRefsCreated);
    168         env->CallStaticVoidMethod(gBinderInternalOffsets.mClass,
    169                 gBinderInternalOffsets.mForceGc);
    170     } else {
    171         LOGV("Now have %d binder ops", old);
    172     }
    173 }
    174 
    175 static JavaVM* jnienv_to_javavm(JNIEnv* env)
    176 {
    177     JavaVM* vm;
    178     return env->GetJavaVM(&vm) >= 0 ? vm : NULL;
    179 }
    180 
    181 static JNIEnv* javavm_to_jnienv(JavaVM* vm)
    182 {
    183     JNIEnv* env;
    184     return vm->GetEnv((void **)&env, JNI_VERSION_1_4) >= 0 ? env : NULL;
    185 }
    186 
    187 static void report_exception(JNIEnv* env, jthrowable excep, const char* msg)
    188 {
    189     env->ExceptionClear();
    190 
    191     jstring tagstr = env->NewStringUTF(LOG_TAG);
    192     jstring msgstr = env->NewStringUTF(msg);
    193 
    194     if ((tagstr == NULL) || (msgstr == NULL)) {
    195         env->ExceptionClear();      /* assume exception (OOM?) was thrown */
    196         LOGE("Unable to call Log.e()\n");
    197         LOGE("%s", msg);
    198         goto bail;
    199     }
    200 
    201     env->CallStaticIntMethod(
    202         gLogOffsets.mClass, gLogOffsets.mLogE, tagstr, msgstr, excep);
    203     if (env->ExceptionCheck()) {
    204         /* attempting to log the failure has failed */
    205         LOGW("Failed trying to log exception, msg='%s'\n", msg);
    206         env->ExceptionClear();
    207     }
    208 
    209     if (env->IsInstanceOf(excep, gErrorOffsets.mClass)) {
    210         /*
    211          * It's an Error: Reraise the exception, detach this thread, and
    212          * wait for the fireworks. Die even more blatantly after a minute
    213          * if the gentler attempt doesn't do the trick.
    214          *
    215          * The GetJavaVM function isn't on the "approved" list of JNI calls
    216          * that can be made while an exception is pending, so we want to
    217          * get the VM ptr, throw the exception, and then detach the thread.
    218          */
    219         JavaVM* vm = jnienv_to_javavm(env);
    220         env->Throw(excep);
    221         vm->DetachCurrentThread();
    222         sleep(60);
    223         LOGE("Forcefully exiting");
    224         exit(1);
    225         *((int *) 1) = 1;
    226     }
    227 
    228 bail:
    229     /* discard local refs created for us by VM */
    230     env->DeleteLocalRef(tagstr);
    231     env->DeleteLocalRef(msgstr);
    232 }
    233 
    234 static void set_dalvik_blockguard_policy(JNIEnv* env, jint strict_policy)
    235 {
    236     // Call back into android.os.StrictMode#onBinderStrictModePolicyChange
    237     // to sync our state back to it.  See the comments in StrictMode.java.
    238     env->CallStaticVoidMethod(gStrictModeCallbackOffsets.mClass,
    239                               gStrictModeCallbackOffsets.mCallback,
    240                               strict_policy);
    241 }
    242 
    243 class JavaBBinderHolder;
    244 
    245 class JavaBBinder : public BBinder
    246 {
    247 public:
    248     JavaBBinder(JNIEnv* env, jobject object)
    249         : mVM(jnienv_to_javavm(env)), mObject(env->NewGlobalRef(object))
    250     {
    251         LOGV("Creating JavaBBinder %p\n", this);
    252         android_atomic_inc(&gNumLocalRefs);
    253         incRefsCreated(env);
    254     }
    255 
    256     bool    checkSubclass(const void* subclassID) const
    257     {
    258         return subclassID == &gBinderOffsets;
    259     }
    260 
    261     jobject object() const
    262     {
    263         return mObject;
    264     }
    265 
    266 protected:
    267     virtual ~JavaBBinder()
    268     {
    269         LOGV("Destroying JavaBBinder %p\n", this);
    270         android_atomic_dec(&gNumLocalRefs);
    271         JNIEnv* env = javavm_to_jnienv(mVM);
    272         env->DeleteGlobalRef(mObject);
    273     }
    274 
    275     virtual status_t onTransact(
    276         uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags = 0)
    277     {
    278         JNIEnv* env = javavm_to_jnienv(mVM);
    279 
    280         LOGV("onTransact() on %p calling object %p in env %p vm %p\n", this, mObject, env, mVM);
    281 
    282         IPCThreadState* thread_state = IPCThreadState::self();
    283         const int strict_policy_before = thread_state->getStrictModePolicy();
    284         thread_state->setLastTransactionBinderFlags(flags);
    285 
    286         //printf("Transact from %p to Java code sending: ", this);
    287         //data.print();
    288         //printf("\n");
    289         jboolean res = env->CallBooleanMethod(mObject, gBinderOffsets.mExecTransact,
    290             code, (int32_t)&data, (int32_t)reply, flags);
    291         jthrowable excep = env->ExceptionOccurred();
    292 
    293         if (excep) {
    294             report_exception(env, excep,
    295                 "*** Uncaught remote exception!  "
    296                 "(Exceptions are not yet supported across processes.)");
    297             res = JNI_FALSE;
    298 
    299             /* clean up JNI local ref -- we don't return to Java code */
    300             env->DeleteLocalRef(excep);
    301         }
    302 
    303         // Restore the Java binder thread's state if it changed while
    304         // processing a call (as it would if the Parcel's header had a
    305         // new policy mask and Parcel.enforceInterface() changed
    306         // it...)
    307         const int strict_policy_after = thread_state->getStrictModePolicy();
    308         if (strict_policy_after != strict_policy_before) {
    309             // Our thread-local...
    310             thread_state->setStrictModePolicy(strict_policy_before);
    311             // And the Java-level thread-local...
    312             set_dalvik_blockguard_policy(env, strict_policy_before);
    313         }
    314 
    315         jthrowable excep2 = env->ExceptionOccurred();
    316         if (excep2) {
    317             report_exception(env, excep2,
    318                 "*** Uncaught exception in onBinderStrictModePolicyChange");
    319             /* clean up JNI local ref -- we don't return to Java code */
    320             env->DeleteLocalRef(excep2);
    321         }
    322 
    323         //aout << "onTransact to Java code; result=" << res << endl
    324         //    << "Transact from " << this << " to Java code returning "
    325         //    << reply << ": " << *reply << endl;
    326         return res != JNI_FALSE ? NO_ERROR : UNKNOWN_TRANSACTION;
    327     }
    328 
    329     virtual status_t dump(int fd, const Vector<String16>& args)
    330     {
    331         return 0;
    332     }
    333 
    334 private:
    335     JavaVM* const   mVM;
    336     jobject const   mObject;
    337 };
    338 
    339 // ----------------------------------------------------------------------------
    340 
    341 class JavaBBinderHolder : public RefBase
    342 {
    343 public:
    344     sp<JavaBBinder> get(JNIEnv* env, jobject obj)
    345     {
    346         AutoMutex _l(mLock);
    347         sp<JavaBBinder> b = mBinder.promote();
    348         if (b == NULL) {
    349             b = new JavaBBinder(env, obj);
    350             mBinder = b;
    351             LOGV("Creating JavaBinder %p (refs %p) for Object %p, weakCount=%d\n",
    352                  b.get(), b->getWeakRefs(), obj, b->getWeakRefs()->getWeakCount());
    353         }
    354 
    355         return b;
    356     }
    357 
    358     sp<JavaBBinder> getExisting()
    359     {
    360         AutoMutex _l(mLock);
    361         return mBinder.promote();
    362     }
    363 
    364 private:
    365     Mutex           mLock;
    366     wp<JavaBBinder> mBinder;
    367 };
    368 
    369 // ----------------------------------------------------------------------------
    370 
    371 // Per-IBinder death recipient bookkeeping.  This is how we reconcile local jobject
    372 // death recipient references passed in through JNI with the permanent corresponding
    373 // JavaDeathRecipient objects.
    374 
    375 class JavaDeathRecipient;
    376 
    377 class DeathRecipientList : public RefBase {
    378     List< sp<JavaDeathRecipient> > mList;
    379     Mutex mLock;
    380 
    381 public:
    382     DeathRecipientList();
    383     ~DeathRecipientList();
    384 
    385     void add(const sp<JavaDeathRecipient>& recipient);
    386     void remove(const sp<JavaDeathRecipient>& recipient);
    387     sp<JavaDeathRecipient> find(jobject recipient);
    388 };
    389 
    390 // ----------------------------------------------------------------------------
    391 
    392 class JavaDeathRecipient : public IBinder::DeathRecipient
    393 {
    394 public:
    395     JavaDeathRecipient(JNIEnv* env, jobject object, const sp<DeathRecipientList>& list)
    396         : mVM(jnienv_to_javavm(env)), mObject(env->NewGlobalRef(object)),
    397           mObjectWeak(NULL), mList(list)
    398     {
    399         // These objects manage their own lifetimes so are responsible for final bookkeeping.
    400         // The list holds a strong reference to this object.
    401         LOGDEATH("Adding JDR %p to DRL %p", this, list.get());
    402         list->add(this);
    403 
    404         android_atomic_inc(&gNumDeathRefs);
    405         incRefsCreated(env);
    406     }
    407 
    408     void binderDied(const wp<IBinder>& who)
    409     {
    410         LOGDEATH("Receiving binderDied() on JavaDeathRecipient %p\n", this);
    411         if (mObject != NULL) {
    412             JNIEnv* env = javavm_to_jnienv(mVM);
    413 
    414             env->CallStaticVoidMethod(gBinderProxyOffsets.mClass,
    415                     gBinderProxyOffsets.mSendDeathNotice, mObject);
    416             jthrowable excep = env->ExceptionOccurred();
    417             if (excep) {
    418                 report_exception(env, excep,
    419                         "*** Uncaught exception returned from death notification!");
    420             }
    421 
    422             // Demote from strong ref to weak after binderDied() has been delivered,
    423             // to allow the DeathRecipient and BinderProxy to be GC'd if no longer needed.
    424             mObjectWeak = env->NewWeakGlobalRef(mObject);
    425             env->DeleteGlobalRef(mObject);
    426             mObject = NULL;
    427         }
    428     }
    429 
    430     void clearReference()
    431     {
    432         sp<DeathRecipientList> list = mList.promote();
    433         if (list != NULL) {
    434             LOGDEATH("Removing JDR %p from DRL %p", this, list.get());
    435             list->remove(this);
    436         } else {
    437             LOGDEATH("clearReference() on JDR %p but DRL wp purged", this);
    438         }
    439     }
    440 
    441     bool matches(jobject obj) {
    442         bool result;
    443         JNIEnv* env = javavm_to_jnienv(mVM);
    444 
    445         if (mObject != NULL) {
    446             result = env->IsSameObject(obj, mObject);
    447         } else {
    448             jobject me = env->NewLocalRef(mObjectWeak);
    449             result = env->IsSameObject(obj, me);
    450             env->DeleteLocalRef(me);
    451         }
    452         return result;
    453     }
    454 
    455     void warnIfStillLive() {
    456         if (mObject != NULL) {
    457             // Okay, something is wrong -- we have a hard reference to a live death
    458             // recipient on the VM side, but the list is being torn down.
    459             JNIEnv* env = javavm_to_jnienv(mVM);
    460             ScopedLocalRef<jclass> objClassRef(env, env->GetObjectClass(mObject));
    461             ScopedLocalRef<jstring> nameRef(env,
    462                     (jstring) env->CallObjectMethod(objClassRef.get(), gClassOffsets.mGetName));
    463             ScopedUtfChars nameUtf(env, nameRef.get());
    464             if (nameUtf.c_str() != NULL) {
    465                 LOGW("BinderProxy is being destroyed but the application did not call "
    466                         "unlinkToDeath to unlink all of its death recipients beforehand.  "
    467                         "Releasing leaked death recipient: %s", nameUtf.c_str());
    468             } else {
    469                 LOGW("BinderProxy being destroyed; unable to get DR object name");
    470                 env->ExceptionClear();
    471             }
    472         }
    473     }
    474 
    475 protected:
    476     virtual ~JavaDeathRecipient()
    477     {
    478         //LOGI("Removing death ref: recipient=%p\n", mObject);
    479         android_atomic_dec(&gNumDeathRefs);
    480         JNIEnv* env = javavm_to_jnienv(mVM);
    481         if (mObject != NULL) {
    482             env->DeleteGlobalRef(mObject);
    483         } else {
    484             env->DeleteWeakGlobalRef(mObjectWeak);
    485         }
    486     }
    487 
    488 private:
    489     JavaVM* const mVM;
    490     jobject mObject;
    491     jweak mObjectWeak; // will be a weak ref to the same VM-side DeathRecipient after binderDied()
    492     wp<DeathRecipientList> mList;
    493 };
    494 
    495 // ----------------------------------------------------------------------------
    496 
    497 DeathRecipientList::DeathRecipientList() {
    498     LOGDEATH("New DRL @ %p", this);
    499 }
    500 
    501 DeathRecipientList::~DeathRecipientList() {
    502     LOGDEATH("Destroy DRL @ %p", this);
    503     AutoMutex _l(mLock);
    504 
    505     // Should never happen -- the JavaDeathRecipient objects that have added themselves
    506     // to the list are holding references on the list object.  Only when they are torn
    507     // down can the list header be destroyed.
    508     if (mList.size() > 0) {
    509         List< sp<JavaDeathRecipient> >::iterator iter;
    510         for (iter = mList.begin(); iter != mList.end(); iter++) {
    511             (*iter)->warnIfStillLive();
    512         }
    513     }
    514 }
    515 
    516 void DeathRecipientList::add(const sp<JavaDeathRecipient>& recipient) {
    517     AutoMutex _l(mLock);
    518 
    519     LOGDEATH("DRL @ %p : add JDR %p", this, recipient.get());
    520     mList.push_back(recipient);
    521 }
    522 
    523 void DeathRecipientList::remove(const sp<JavaDeathRecipient>& recipient) {
    524     AutoMutex _l(mLock);
    525 
    526     List< sp<JavaDeathRecipient> >::iterator iter;
    527     for (iter = mList.begin(); iter != mList.end(); iter++) {
    528         if (*iter == recipient) {
    529             LOGDEATH("DRL @ %p : remove JDR %p", this, recipient.get());
    530             mList.erase(iter);
    531             return;
    532         }
    533     }
    534 }
    535 
    536 sp<JavaDeathRecipient> DeathRecipientList::find(jobject recipient) {
    537     AutoMutex _l(mLock);
    538 
    539     List< sp<JavaDeathRecipient> >::iterator iter;
    540     for (iter = mList.begin(); iter != mList.end(); iter++) {
    541         if ((*iter)->matches(recipient)) {
    542             return *iter;
    543         }
    544     }
    545     return NULL;
    546 }
    547 
    548 // ----------------------------------------------------------------------------
    549 
    550 namespace android {
    551 
    552 static void proxy_cleanup(const void* id, void* obj, void* cleanupCookie)
    553 {
    554     android_atomic_dec(&gNumProxyRefs);
    555     JNIEnv* env = javavm_to_jnienv((JavaVM*)cleanupCookie);
    556     env->DeleteGlobalRef((jobject)obj);
    557 }
    558 
    559 static Mutex mProxyLock;
    560 
    561 jobject javaObjectForIBinder(JNIEnv* env, const sp<IBinder>& val)
    562 {
    563     if (val == NULL) return NULL;
    564 
    565     if (val->checkSubclass(&gBinderOffsets)) {
    566         // One of our own!
    567         jobject object = static_cast<JavaBBinder*>(val.get())->object();
    568         LOGDEATH("objectForBinder %p: it's our own %p!\n", val.get(), object);
    569         return object;
    570     }
    571 
    572     // For the rest of the function we will hold this lock, to serialize
    573     // looking/creation of Java proxies for native Binder proxies.
    574     AutoMutex _l(mProxyLock);
    575 
    576     // Someone else's...  do we know about it?
    577     jobject object = (jobject)val->findObject(&gBinderProxyOffsets);
    578     if (object != NULL) {
    579         jobject res = env->CallObjectMethod(object, gWeakReferenceOffsets.mGet);
    580         if (res != NULL) {
    581             LOGV("objectForBinder %p: found existing %p!\n", val.get(), res);
    582             return res;
    583         }
    584         LOGDEATH("Proxy object %p of IBinder %p no longer in working set!!!", object, val.get());
    585         android_atomic_dec(&gNumProxyRefs);
    586         val->detachObject(&gBinderProxyOffsets);
    587         env->DeleteGlobalRef(object);
    588     }
    589 
    590     object = env->NewObject(gBinderProxyOffsets.mClass, gBinderProxyOffsets.mConstructor);
    591     if (object != NULL) {
    592         LOGDEATH("objectForBinder %p: created new proxy %p !\n", val.get(), object);
    593         // The proxy holds a reference to the native object.
    594         env->SetIntField(object, gBinderProxyOffsets.mObject, (int)val.get());
    595         val->incStrong(object);
    596 
    597         // The native object needs to hold a weak reference back to the
    598         // proxy, so we can retrieve the same proxy if it is still active.
    599         jobject refObject = env->NewGlobalRef(
    600                 env->GetObjectField(object, gBinderProxyOffsets.mSelf));
    601         val->attachObject(&gBinderProxyOffsets, refObject,
    602                 jnienv_to_javavm(env), proxy_cleanup);
    603 
    604         // Also remember the death recipients registered on this proxy
    605         sp<DeathRecipientList> drl = new DeathRecipientList;
    606         drl->incStrong((void*)javaObjectForIBinder);
    607         env->SetIntField(object, gBinderProxyOffsets.mOrgue, reinterpret_cast<jint>(drl.get()));
    608 
    609         // Note that a new object reference has been created.
    610         android_atomic_inc(&gNumProxyRefs);
    611         incRefsCreated(env);
    612     }
    613 
    614     return object;
    615 }
    616 
    617 sp<IBinder> ibinderForJavaObject(JNIEnv* env, jobject obj)
    618 {
    619     if (obj == NULL) return NULL;
    620 
    621     if (env->IsInstanceOf(obj, gBinderOffsets.mClass)) {
    622         JavaBBinderHolder* jbh = (JavaBBinderHolder*)
    623             env->GetIntField(obj, gBinderOffsets.mObject);
    624         return jbh != NULL ? jbh->get(env, obj) : NULL;
    625     }
    626 
    627     if (env->IsInstanceOf(obj, gBinderProxyOffsets.mClass)) {
    628         return (IBinder*)
    629             env->GetIntField(obj, gBinderProxyOffsets.mObject);
    630     }
    631 
    632     LOGW("ibinderForJavaObject: %p is not a Binder object", obj);
    633     return NULL;
    634 }
    635 
    636 Parcel* parcelForJavaObject(JNIEnv* env, jobject obj)
    637 {
    638     if (obj) {
    639         Parcel* p = (Parcel*)env->GetIntField(obj, gParcelOffsets.mObject);
    640         if (p != NULL) {
    641             return p;
    642         }
    643         jniThrowException(env, "java/lang/IllegalStateException", "Parcel has been finalized!");
    644     }
    645     return NULL;
    646 }
    647 
    648 jobject newParcelFileDescriptor(JNIEnv* env, jobject fileDesc)
    649 {
    650     return env->NewObject(
    651             gParcelFileDescriptorOffsets.mClass, gParcelFileDescriptorOffsets.mConstructor, fileDesc);
    652 }
    653 
    654 void signalExceptionForError(JNIEnv* env, jobject obj, status_t err)
    655 {
    656     switch (err) {
    657         case UNKNOWN_ERROR:
    658             jniThrowException(env, "java/lang/RuntimeException", "Unknown error");
    659             break;
    660         case NO_MEMORY:
    661             jniThrowException(env, "java/lang/OutOfMemoryError", NULL);
    662             break;
    663         case INVALID_OPERATION:
    664             jniThrowException(env, "java/lang/UnsupportedOperationException", NULL);
    665             break;
    666         case BAD_VALUE:
    667             jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
    668             break;
    669         case BAD_INDEX:
    670             jniThrowException(env, "java/lang/IndexOutOfBoundsException", NULL);
    671             break;
    672         case BAD_TYPE:
    673             jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
    674             break;
    675         case NAME_NOT_FOUND:
    676             jniThrowException(env, "java/util/NoSuchElementException", NULL);
    677             break;
    678         case PERMISSION_DENIED:
    679             jniThrowException(env, "java/lang/SecurityException", NULL);
    680             break;
    681         case NOT_ENOUGH_DATA:
    682             jniThrowException(env, "android/os/ParcelFormatException", "Not enough data");
    683             break;
    684         case NO_INIT:
    685             jniThrowException(env, "java/lang/RuntimeException", "Not initialized");
    686             break;
    687         case ALREADY_EXISTS:
    688             jniThrowException(env, "java/lang/RuntimeException", "Item already exists");
    689             break;
    690         case DEAD_OBJECT:
    691             jniThrowException(env, "android/os/DeadObjectException", NULL);
    692             break;
    693         case UNKNOWN_TRANSACTION:
    694             jniThrowException(env, "java/lang/RuntimeException", "Unknown transaction code");
    695             break;
    696         case FAILED_TRANSACTION:
    697             LOGE("!!! FAILED BINDER TRANSACTION !!!");
    698             //jniThrowException(env, "java/lang/OutOfMemoryError", "Binder transaction too large");
    699             break;
    700         case FDS_NOT_ALLOWED:
    701             jniThrowException(env, "java/lang/RuntimeException",
    702                     "Not allowed to write file descriptors here");
    703             break;
    704         default:
    705             LOGE("Unknown binder error code. 0x%x", err);
    706     }
    707 }
    708 
    709 }
    710 
    711 // ----------------------------------------------------------------------------
    712 
    713 static jint android_os_Binder_getCallingPid(JNIEnv* env, jobject clazz)
    714 {
    715     return IPCThreadState::self()->getCallingPid();
    716 }
    717 
    718 static jint android_os_Binder_getCallingUid(JNIEnv* env, jobject clazz)
    719 {
    720     return IPCThreadState::self()->getCallingUid();
    721 }
    722 
    723 static jlong android_os_Binder_clearCallingIdentity(JNIEnv* env, jobject clazz)
    724 {
    725     return IPCThreadState::self()->clearCallingIdentity();
    726 }
    727 
    728 static void android_os_Binder_restoreCallingIdentity(JNIEnv* env, jobject clazz, jlong token)
    729 {
    730     // XXX temporary sanity check to debug crashes.
    731     int uid = (int)(token>>32);
    732     if (uid > 0 && uid < 999) {
    733         // In Android currently there are no uids in this range.
    734         char buf[128];
    735         sprintf(buf, "Restoring bad calling ident: 0x%Lx", token);
    736         jniThrowException(env, "java/lang/IllegalStateException", buf);
    737         return;
    738     }
    739     IPCThreadState::self()->restoreCallingIdentity(token);
    740 }
    741 
    742 static void android_os_Binder_setThreadStrictModePolicy(JNIEnv* env, jobject clazz, jint policyMask)
    743 {
    744     IPCThreadState::self()->setStrictModePolicy(policyMask);
    745 }
    746 
    747 static jint android_os_Binder_getThreadStrictModePolicy(JNIEnv* env, jobject clazz)
    748 {
    749     return IPCThreadState::self()->getStrictModePolicy();
    750 }
    751 
    752 static void android_os_Binder_flushPendingCommands(JNIEnv* env, jobject clazz)
    753 {
    754     IPCThreadState::self()->flushCommands();
    755 }
    756 
    757 static void android_os_Binder_init(JNIEnv* env, jobject obj)
    758 {
    759     JavaBBinderHolder* jbh = new JavaBBinderHolder();
    760     if (jbh == NULL) {
    761         jniThrowException(env, "java/lang/OutOfMemoryError", NULL);
    762         return;
    763     }
    764     LOGV("Java Binder %p: acquiring first ref on holder %p", obj, jbh);
    765     jbh->incStrong((void*)android_os_Binder_init);
    766     env->SetIntField(obj, gBinderOffsets.mObject, (int)jbh);
    767 }
    768 
    769 static void android_os_Binder_destroy(JNIEnv* env, jobject obj)
    770 {
    771     JavaBBinderHolder* jbh = (JavaBBinderHolder*)
    772         env->GetIntField(obj, gBinderOffsets.mObject);
    773     if (jbh != NULL) {
    774         env->SetIntField(obj, gBinderOffsets.mObject, 0);
    775         LOGV("Java Binder %p: removing ref on holder %p", obj, jbh);
    776         jbh->decStrong((void*)android_os_Binder_init);
    777     } else {
    778         // Encountering an uninitialized binder is harmless.  All it means is that
    779         // the Binder was only partially initialized when its finalizer ran and called
    780         // destroy().  The Binder could be partially initialized for several reasons.
    781         // For example, a Binder subclass constructor might have thrown an exception before
    782         // it could delegate to its superclass's constructor.  Consequently init() would
    783         // not have been called and the holder pointer would remain NULL.
    784         LOGV("Java Binder %p: ignoring uninitialized binder", obj);
    785     }
    786 }
    787 
    788 // ----------------------------------------------------------------------------
    789 
    790 static const JNINativeMethod gBinderMethods[] = {
    791      /* name, signature, funcPtr */
    792     { "getCallingPid", "()I", (void*)android_os_Binder_getCallingPid },
    793     { "getCallingUid", "()I", (void*)android_os_Binder_getCallingUid },
    794     { "clearCallingIdentity", "()J", (void*)android_os_Binder_clearCallingIdentity },
    795     { "restoreCallingIdentity", "(J)V", (void*)android_os_Binder_restoreCallingIdentity },
    796     { "setThreadStrictModePolicy", "(I)V", (void*)android_os_Binder_setThreadStrictModePolicy },
    797     { "getThreadStrictModePolicy", "()I", (void*)android_os_Binder_getThreadStrictModePolicy },
    798     { "flushPendingCommands", "()V", (void*)android_os_Binder_flushPendingCommands },
    799     { "init", "()V", (void*)android_os_Binder_init },
    800     { "destroy", "()V", (void*)android_os_Binder_destroy }
    801 };
    802 
    803 const char* const kBinderPathName = "android/os/Binder";
    804 
    805 static int int_register_android_os_Binder(JNIEnv* env)
    806 {
    807     jclass clazz;
    808 
    809     clazz = env->FindClass(kBinderPathName);
    810     LOG_FATAL_IF(clazz == NULL, "Unable to find class android.os.Binder");
    811 
    812     gBinderOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
    813     gBinderOffsets.mExecTransact
    814         = env->GetMethodID(clazz, "execTransact", "(IIII)Z");
    815     assert(gBinderOffsets.mExecTransact);
    816 
    817     gBinderOffsets.mObject
    818         = env->GetFieldID(clazz, "mObject", "I");
    819     assert(gBinderOffsets.mObject);
    820 
    821     return AndroidRuntime::registerNativeMethods(
    822         env, kBinderPathName,
    823         gBinderMethods, NELEM(gBinderMethods));
    824 }
    825 
    826 // ****************************************************************************
    827 // ****************************************************************************
    828 // ****************************************************************************
    829 
    830 namespace android {
    831 
    832 jint android_os_Debug_getLocalObjectCount(JNIEnv* env, jobject clazz)
    833 {
    834     return gNumLocalRefs;
    835 }
    836 
    837 jint android_os_Debug_getProxyObjectCount(JNIEnv* env, jobject clazz)
    838 {
    839     return gNumProxyRefs;
    840 }
    841 
    842 jint android_os_Debug_getDeathObjectCount(JNIEnv* env, jobject clazz)
    843 {
    844     return gNumDeathRefs;
    845 }
    846 
    847 }
    848 
    849 // ****************************************************************************
    850 // ****************************************************************************
    851 // ****************************************************************************
    852 
    853 static jobject android_os_BinderInternal_getContextObject(JNIEnv* env, jobject clazz)
    854 {
    855     sp<IBinder> b = ProcessState::self()->getContextObject(NULL);
    856     return javaObjectForIBinder(env, b);
    857 }
    858 
    859 static void android_os_BinderInternal_joinThreadPool(JNIEnv* env, jobject clazz)
    860 {
    861     sp<IBinder> b = ProcessState::self()->getContextObject(NULL);
    862     android::IPCThreadState::self()->joinThreadPool();
    863 }
    864 
    865 static void android_os_BinderInternal_disableBackgroundScheduling(JNIEnv* env,
    866         jobject clazz, jboolean disable)
    867 {
    868     IPCThreadState::disableBackgroundScheduling(disable ? true : false);
    869 }
    870 
    871 static void android_os_BinderInternal_handleGc(JNIEnv* env, jobject clazz)
    872 {
    873     LOGV("Gc has executed, clearing binder ops");
    874     android_atomic_and(0, &gNumRefsCreated);
    875 }
    876 
    877 // ----------------------------------------------------------------------------
    878 
    879 static const JNINativeMethod gBinderInternalMethods[] = {
    880      /* name, signature, funcPtr */
    881     { "getContextObject", "()Landroid/os/IBinder;", (void*)android_os_BinderInternal_getContextObject },
    882     { "joinThreadPool", "()V", (void*)android_os_BinderInternal_joinThreadPool },
    883     { "disableBackgroundScheduling", "(Z)V", (void*)android_os_BinderInternal_disableBackgroundScheduling },
    884     { "handleGc", "()V", (void*)android_os_BinderInternal_handleGc }
    885 };
    886 
    887 const char* const kBinderInternalPathName = "com/android/internal/os/BinderInternal";
    888 
    889 static int int_register_android_os_BinderInternal(JNIEnv* env)
    890 {
    891     jclass clazz;
    892 
    893     clazz = env->FindClass(kBinderInternalPathName);
    894     LOG_FATAL_IF(clazz == NULL, "Unable to find class com.android.internal.os.BinderInternal");
    895 
    896     gBinderInternalOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
    897     gBinderInternalOffsets.mForceGc
    898         = env->GetStaticMethodID(clazz, "forceBinderGc", "()V");
    899     assert(gBinderInternalOffsets.mForceGc);
    900 
    901     return AndroidRuntime::registerNativeMethods(
    902         env, kBinderInternalPathName,
    903         gBinderInternalMethods, NELEM(gBinderInternalMethods));
    904 }
    905 
    906 // ****************************************************************************
    907 // ****************************************************************************
    908 // ****************************************************************************
    909 
    910 static jboolean android_os_BinderProxy_pingBinder(JNIEnv* env, jobject obj)
    911 {
    912     IBinder* target = (IBinder*)
    913         env->GetIntField(obj, gBinderProxyOffsets.mObject);
    914     if (target == NULL) {
    915         return JNI_FALSE;
    916     }
    917     status_t err = target->pingBinder();
    918     return err == NO_ERROR ? JNI_TRUE : JNI_FALSE;
    919 }
    920 
    921 static jstring android_os_BinderProxy_getInterfaceDescriptor(JNIEnv* env, jobject obj)
    922 {
    923     IBinder* target = (IBinder*) env->GetIntField(obj, gBinderProxyOffsets.mObject);
    924     if (target != NULL) {
    925         const String16& desc = target->getInterfaceDescriptor();
    926         return env->NewString(desc.string(), desc.size());
    927     }
    928     jniThrowException(env, "java/lang/RuntimeException",
    929             "No binder found for object");
    930     return NULL;
    931 }
    932 
    933 static jboolean android_os_BinderProxy_isBinderAlive(JNIEnv* env, jobject obj)
    934 {
    935     IBinder* target = (IBinder*)
    936         env->GetIntField(obj, gBinderProxyOffsets.mObject);
    937     if (target == NULL) {
    938         return JNI_FALSE;
    939     }
    940     bool alive = target->isBinderAlive();
    941     return alive ? JNI_TRUE : JNI_FALSE;
    942 }
    943 
    944 static int getprocname(pid_t pid, char *buf, size_t len) {
    945     char filename[20];
    946     FILE *f;
    947 
    948     sprintf(filename, "/proc/%d/cmdline", pid);
    949     f = fopen(filename, "r");
    950     if (!f) { *buf = '\0'; return 1; }
    951     if (!fgets(buf, len, f)) { *buf = '\0'; return 2; }
    952     fclose(f);
    953     return 0;
    954 }
    955 
    956 static bool push_eventlog_string(char** pos, const char* end, const char* str) {
    957     jint len = strlen(str);
    958     int space_needed = 1 + sizeof(len) + len;
    959     if (end - *pos < space_needed) {
    960         LOGW("not enough space for string. remain=%d; needed=%d",
    961              (end - *pos), space_needed);
    962         return false;
    963     }
    964     **pos = EVENT_TYPE_STRING;
    965     (*pos)++;
    966     memcpy(*pos, &len, sizeof(len));
    967     *pos += sizeof(len);
    968     memcpy(*pos, str, len);
    969     *pos += len;
    970     return true;
    971 }
    972 
    973 static bool push_eventlog_int(char** pos, const char* end, jint val) {
    974     int space_needed = 1 + sizeof(val);
    975     if (end - *pos < space_needed) {
    976         LOGW("not enough space for int.  remain=%d; needed=%d",
    977              (end - *pos), space_needed);
    978         return false;
    979     }
    980     **pos = EVENT_TYPE_INT;
    981     (*pos)++;
    982     memcpy(*pos, &val, sizeof(val));
    983     *pos += sizeof(val);
    984     return true;
    985 }
    986 
    987 // From frameworks/base/core/java/android/content/EventLogTags.logtags:
    988 #define LOGTAG_BINDER_OPERATION 52004
    989 
    990 static void conditionally_log_binder_call(int64_t start_millis,
    991                                           IBinder* target, jint code) {
    992     int duration_ms = static_cast<int>(uptimeMillis() - start_millis);
    993 
    994     int sample_percent;
    995     if (duration_ms >= 500) {
    996         sample_percent = 100;
    997     } else {
    998         sample_percent = 100 * duration_ms / 500;
    999         if (sample_percent == 0) {
   1000             return;
   1001         }
   1002         if (sample_percent < (random() % 100 + 1)) {
   1003             return;
   1004         }
   1005     }
   1006 
   1007     char process_name[40];
   1008     getprocname(getpid(), process_name, sizeof(process_name));
   1009     String8 desc(target->getInterfaceDescriptor());
   1010 
   1011     char buf[LOGGER_ENTRY_MAX_PAYLOAD];
   1012     buf[0] = EVENT_TYPE_LIST;
   1013     buf[1] = 5;
   1014     char* pos = &buf[2];
   1015     char* end = &buf[LOGGER_ENTRY_MAX_PAYLOAD - 1];  // leave room for final \n
   1016     if (!push_eventlog_string(&pos, end, desc.string())) return;
   1017     if (!push_eventlog_int(&pos, end, code)) return;
   1018     if (!push_eventlog_int(&pos, end, duration_ms)) return;
   1019     if (!push_eventlog_string(&pos, end, process_name)) return;
   1020     if (!push_eventlog_int(&pos, end, sample_percent)) return;
   1021     *(pos++) = '\n';   // conventional with EVENT_TYPE_LIST apparently.
   1022     android_bWriteLog(LOGTAG_BINDER_OPERATION, buf, pos - buf);
   1023 }
   1024 
   1025 // We only measure binder call durations to potentially log them if
   1026 // we're on the main thread.  Unfortunately sim-eng doesn't seem to
   1027 // have gettid, so we just ignore this and don't log if we can't
   1028 // get the thread id.
   1029 static bool should_time_binder_calls() {
   1030 #ifdef HAVE_GETTID
   1031   return (getpid() == androidGetTid());
   1032 #else
   1033 #warning no gettid(), so not logging Binder calls...
   1034   return false;
   1035 #endif
   1036 }
   1037 
   1038 static jboolean android_os_BinderProxy_transact(JNIEnv* env, jobject obj,
   1039                                                 jint code, jobject dataObj,
   1040                                                 jobject replyObj, jint flags)
   1041 {
   1042     if (dataObj == NULL) {
   1043         jniThrowNullPointerException(env, NULL);
   1044         return JNI_FALSE;
   1045     }
   1046 
   1047     Parcel* data = parcelForJavaObject(env, dataObj);
   1048     if (data == NULL) {
   1049         return JNI_FALSE;
   1050     }
   1051     Parcel* reply = parcelForJavaObject(env, replyObj);
   1052     if (reply == NULL && replyObj != NULL) {
   1053         return JNI_FALSE;
   1054     }
   1055 
   1056     IBinder* target = (IBinder*)
   1057         env->GetIntField(obj, gBinderProxyOffsets.mObject);
   1058     if (target == NULL) {
   1059         jniThrowException(env, "java/lang/IllegalStateException", "Binder has been finalized!");
   1060         return JNI_FALSE;
   1061     }
   1062 
   1063     LOGV("Java code calling transact on %p in Java object %p with code %d\n",
   1064             target, obj, code);
   1065 
   1066     // Only log the binder call duration for things on the Java-level main thread.
   1067     // But if we don't
   1068     const bool time_binder_calls = should_time_binder_calls();
   1069 
   1070     int64_t start_millis;
   1071     if (time_binder_calls) {
   1072         start_millis = uptimeMillis();
   1073     }
   1074     //printf("Transact from Java code to %p sending: ", target); data->print();
   1075     status_t err = target->transact(code, *data, reply, flags);
   1076     //if (reply) printf("Transact from Java code to %p received: ", target); reply->print();
   1077     if (time_binder_calls) {
   1078         conditionally_log_binder_call(start_millis, target, code);
   1079     }
   1080 
   1081     if (err == NO_ERROR) {
   1082         return JNI_TRUE;
   1083     } else if (err == UNKNOWN_TRANSACTION) {
   1084         return JNI_FALSE;
   1085     }
   1086 
   1087     signalExceptionForError(env, obj, err);
   1088     return JNI_FALSE;
   1089 }
   1090 
   1091 static void android_os_BinderProxy_linkToDeath(JNIEnv* env, jobject obj,
   1092                                                jobject recipient, jint flags)
   1093 {
   1094     if (recipient == NULL) {
   1095         jniThrowNullPointerException(env, NULL);
   1096         return;
   1097     }
   1098 
   1099     IBinder* target = (IBinder*)
   1100         env->GetIntField(obj, gBinderProxyOffsets.mObject);
   1101     if (target == NULL) {
   1102         LOGW("Binder has been finalized when calling linkToDeath() with recip=%p)\n", recipient);
   1103         assert(false);
   1104     }
   1105 
   1106     LOGDEATH("linkToDeath: binder=%p recipient=%p\n", target, recipient);
   1107 
   1108     if (!target->localBinder()) {
   1109         DeathRecipientList* list = (DeathRecipientList*)
   1110                 env->GetIntField(obj, gBinderProxyOffsets.mOrgue);
   1111         sp<JavaDeathRecipient> jdr = new JavaDeathRecipient(env, recipient, list);
   1112         status_t err = target->linkToDeath(jdr, NULL, flags);
   1113         if (err != NO_ERROR) {
   1114             // Failure adding the death recipient, so clear its reference
   1115             // now.
   1116             jdr->clearReference();
   1117             signalExceptionForError(env, obj, err);
   1118         }
   1119     }
   1120 }
   1121 
   1122 static jboolean android_os_BinderProxy_unlinkToDeath(JNIEnv* env, jobject obj,
   1123                                                  jobject recipient, jint flags)
   1124 {
   1125     jboolean res = JNI_FALSE;
   1126     if (recipient == NULL) {
   1127         jniThrowNullPointerException(env, NULL);
   1128         return res;
   1129     }
   1130 
   1131     IBinder* target = (IBinder*)
   1132         env->GetIntField(obj, gBinderProxyOffsets.mObject);
   1133     if (target == NULL) {
   1134         LOGW("Binder has been finalized when calling linkToDeath() with recip=%p)\n", recipient);
   1135         return JNI_FALSE;
   1136     }
   1137 
   1138     LOGDEATH("unlinkToDeath: binder=%p recipient=%p\n", target, recipient);
   1139 
   1140     if (!target->localBinder()) {
   1141         status_t err = NAME_NOT_FOUND;
   1142 
   1143         // If we find the matching recipient, proceed to unlink using that
   1144         DeathRecipientList* list = (DeathRecipientList*)
   1145                 env->GetIntField(obj, gBinderProxyOffsets.mOrgue);
   1146         sp<JavaDeathRecipient> origJDR = list->find(recipient);
   1147         LOGDEATH("   unlink found list %p and JDR %p", list, origJDR.get());
   1148         if (origJDR != NULL) {
   1149             wp<IBinder::DeathRecipient> dr;
   1150             err = target->unlinkToDeath(origJDR, NULL, flags, &dr);
   1151             if (err == NO_ERROR && dr != NULL) {
   1152                 sp<IBinder::DeathRecipient> sdr = dr.promote();
   1153                 JavaDeathRecipient* jdr = static_cast<JavaDeathRecipient*>(sdr.get());
   1154                 if (jdr != NULL) {
   1155                     jdr->clearReference();
   1156                 }
   1157             }
   1158         }
   1159 
   1160         if (err == NO_ERROR || err == DEAD_OBJECT) {
   1161             res = JNI_TRUE;
   1162         } else {
   1163             jniThrowException(env, "java/util/NoSuchElementException",
   1164                               "Death link does not exist");
   1165         }
   1166     }
   1167 
   1168     return res;
   1169 }
   1170 
   1171 static void android_os_BinderProxy_destroy(JNIEnv* env, jobject obj)
   1172 {
   1173     IBinder* b = (IBinder*)
   1174             env->GetIntField(obj, gBinderProxyOffsets.mObject);
   1175     DeathRecipientList* drl = (DeathRecipientList*)
   1176             env->GetIntField(obj, gBinderProxyOffsets.mOrgue);
   1177 
   1178     LOGDEATH("Destroying BinderProxy %p: binder=%p drl=%p\n", obj, b, drl);
   1179     env->SetIntField(obj, gBinderProxyOffsets.mObject, 0);
   1180     env->SetIntField(obj, gBinderProxyOffsets.mOrgue, 0);
   1181     drl->decStrong((void*)javaObjectForIBinder);
   1182     b->decStrong(obj);
   1183 
   1184     IPCThreadState::self()->flushCommands();
   1185 }
   1186 
   1187 // ----------------------------------------------------------------------------
   1188 
   1189 static const JNINativeMethod gBinderProxyMethods[] = {
   1190      /* name, signature, funcPtr */
   1191     {"pingBinder",          "()Z", (void*)android_os_BinderProxy_pingBinder},
   1192     {"isBinderAlive",       "()Z", (void*)android_os_BinderProxy_isBinderAlive},
   1193     {"getInterfaceDescriptor", "()Ljava/lang/String;", (void*)android_os_BinderProxy_getInterfaceDescriptor},
   1194     {"transact",            "(ILandroid/os/Parcel;Landroid/os/Parcel;I)Z", (void*)android_os_BinderProxy_transact},
   1195     {"linkToDeath",         "(Landroid/os/IBinder$DeathRecipient;I)V", (void*)android_os_BinderProxy_linkToDeath},
   1196     {"unlinkToDeath",       "(Landroid/os/IBinder$DeathRecipient;I)Z", (void*)android_os_BinderProxy_unlinkToDeath},
   1197     {"destroy",             "()V", (void*)android_os_BinderProxy_destroy},
   1198 };
   1199 
   1200 const char* const kBinderProxyPathName = "android/os/BinderProxy";
   1201 
   1202 static int int_register_android_os_BinderProxy(JNIEnv* env)
   1203 {
   1204     jclass clazz;
   1205 
   1206     clazz = env->FindClass("java/lang/ref/WeakReference");
   1207     LOG_FATAL_IF(clazz == NULL, "Unable to find class java.lang.ref.WeakReference");
   1208     gWeakReferenceOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
   1209     gWeakReferenceOffsets.mGet
   1210         = env->GetMethodID(clazz, "get", "()Ljava/lang/Object;");
   1211     assert(gWeakReferenceOffsets.mGet);
   1212 
   1213     clazz = env->FindClass("java/lang/Error");
   1214     LOG_FATAL_IF(clazz == NULL, "Unable to find class java.lang.Error");
   1215     gErrorOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
   1216 
   1217     clazz = env->FindClass(kBinderProxyPathName);
   1218     LOG_FATAL_IF(clazz == NULL, "Unable to find class android.os.BinderProxy");
   1219 
   1220     gBinderProxyOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
   1221     gBinderProxyOffsets.mConstructor
   1222         = env->GetMethodID(clazz, "<init>", "()V");
   1223     assert(gBinderProxyOffsets.mConstructor);
   1224     gBinderProxyOffsets.mSendDeathNotice
   1225         = env->GetStaticMethodID(clazz, "sendDeathNotice", "(Landroid/os/IBinder$DeathRecipient;)V");
   1226     assert(gBinderProxyOffsets.mSendDeathNotice);
   1227 
   1228     gBinderProxyOffsets.mObject
   1229         = env->GetFieldID(clazz, "mObject", "I");
   1230     assert(gBinderProxyOffsets.mObject);
   1231     gBinderProxyOffsets.mSelf
   1232         = env->GetFieldID(clazz, "mSelf", "Ljava/lang/ref/WeakReference;");
   1233     assert(gBinderProxyOffsets.mSelf);
   1234     gBinderProxyOffsets.mOrgue
   1235         = env->GetFieldID(clazz, "mOrgue", "I");
   1236     assert(gBinderProxyOffsets.mOrgue);
   1237 
   1238     clazz = env->FindClass("java/lang/Class");
   1239     LOG_FATAL_IF(clazz == NULL, "Unable to find java.lang.Class");
   1240     gClassOffsets.mGetName = env->GetMethodID(clazz, "getName", "()Ljava/lang/String;");
   1241     assert(gClassOffsets.mGetName);
   1242 
   1243     return AndroidRuntime::registerNativeMethods(
   1244         env, kBinderProxyPathName,
   1245         gBinderProxyMethods, NELEM(gBinderProxyMethods));
   1246 }
   1247 
   1248 // ****************************************************************************
   1249 // ****************************************************************************
   1250 // ****************************************************************************
   1251 
   1252 static jint android_os_Parcel_dataSize(JNIEnv* env, jobject clazz)
   1253 {
   1254     Parcel* parcel = parcelForJavaObject(env, clazz);
   1255     return parcel ? parcel->dataSize() : 0;
   1256 }
   1257 
   1258 static jint android_os_Parcel_dataAvail(JNIEnv* env, jobject clazz)
   1259 {
   1260     Parcel* parcel = parcelForJavaObject(env, clazz);
   1261     return parcel ? parcel->dataAvail() : 0;
   1262 }
   1263 
   1264 static jint android_os_Parcel_dataPosition(JNIEnv* env, jobject clazz)
   1265 {
   1266     Parcel* parcel = parcelForJavaObject(env, clazz);
   1267     return parcel ? parcel->dataPosition() : 0;
   1268 }
   1269 
   1270 static jint android_os_Parcel_dataCapacity(JNIEnv* env, jobject clazz)
   1271 {
   1272     Parcel* parcel = parcelForJavaObject(env, clazz);
   1273     return parcel ? parcel->dataCapacity() : 0;
   1274 }
   1275 
   1276 static void android_os_Parcel_setDataSize(JNIEnv* env, jobject clazz, jint size)
   1277 {
   1278     Parcel* parcel = parcelForJavaObject(env, clazz);
   1279     if (parcel != NULL) {
   1280         const status_t err = parcel->setDataSize(size);
   1281         if (err != NO_ERROR) {
   1282             signalExceptionForError(env, clazz, err);
   1283         }
   1284     }
   1285 }
   1286 
   1287 static void android_os_Parcel_setDataPosition(JNIEnv* env, jobject clazz, jint pos)
   1288 {
   1289     Parcel* parcel = parcelForJavaObject(env, clazz);
   1290     if (parcel != NULL) {
   1291         parcel->setDataPosition(pos);
   1292     }
   1293 }
   1294 
   1295 static void android_os_Parcel_setDataCapacity(JNIEnv* env, jobject clazz, jint size)
   1296 {
   1297     Parcel* parcel = parcelForJavaObject(env, clazz);
   1298     if (parcel != NULL) {
   1299         const status_t err = parcel->setDataCapacity(size);
   1300         if (err != NO_ERROR) {
   1301             signalExceptionForError(env, clazz, err);
   1302         }
   1303     }
   1304 }
   1305 
   1306 static jboolean android_os_Parcel_pushAllowFds(JNIEnv* env, jobject clazz, jboolean allowFds)
   1307 {
   1308     Parcel* parcel = parcelForJavaObject(env, clazz);
   1309     jboolean ret = JNI_TRUE;
   1310     if (parcel != NULL) {
   1311         ret = (jboolean)parcel->pushAllowFds(allowFds);
   1312     }
   1313     return ret;
   1314 }
   1315 
   1316 static void android_os_Parcel_restoreAllowFds(JNIEnv* env, jobject clazz, jboolean lastValue)
   1317 {
   1318     Parcel* parcel = parcelForJavaObject(env, clazz);
   1319     if (parcel != NULL) {
   1320         parcel->restoreAllowFds((bool)lastValue);
   1321     }
   1322 }
   1323 
   1324 static void android_os_Parcel_writeNative(JNIEnv* env, jobject clazz,
   1325                                           jobject data, jint offset,
   1326                                           jint length)
   1327 {
   1328     Parcel* parcel = parcelForJavaObject(env, clazz);
   1329     if (parcel == NULL) {
   1330         return;
   1331     }
   1332 
   1333     const status_t err = parcel->writeInt32(length);
   1334     if (err != NO_ERROR) {
   1335         signalExceptionForError(env, clazz, err);
   1336         return;
   1337     }
   1338 
   1339     void* dest = parcel->writeInplace(length);
   1340     if (dest == NULL) {
   1341         signalExceptionForError(env, clazz, NO_MEMORY);
   1342         return;
   1343     }
   1344 
   1345     jbyte* ar = (jbyte*)env->GetPrimitiveArrayCritical((jarray)data, 0);
   1346     if (ar) {
   1347         memcpy(dest, ar + offset, length);
   1348         env->ReleasePrimitiveArrayCritical((jarray)data, ar, 0);
   1349     }
   1350 }
   1351 
   1352 
   1353 static void android_os_Parcel_writeInt(JNIEnv* env, jobject clazz, jint val)
   1354 {
   1355     Parcel* parcel = parcelForJavaObject(env, clazz);
   1356     if (parcel != NULL) {
   1357         const status_t err = parcel->writeInt32(val);
   1358         if (err != NO_ERROR) {
   1359             signalExceptionForError(env, clazz, err);
   1360         }
   1361     }
   1362 }
   1363 
   1364 static void android_os_Parcel_writeLong(JNIEnv* env, jobject clazz, jlong val)
   1365 {
   1366     Parcel* parcel = parcelForJavaObject(env, clazz);
   1367     if (parcel != NULL) {
   1368         const status_t err = parcel->writeInt64(val);
   1369         if (err != NO_ERROR) {
   1370             signalExceptionForError(env, clazz, err);
   1371         }
   1372     }
   1373 }
   1374 
   1375 static void android_os_Parcel_writeFloat(JNIEnv* env, jobject clazz, jfloat val)
   1376 {
   1377     Parcel* parcel = parcelForJavaObject(env, clazz);
   1378     if (parcel != NULL) {
   1379         const status_t err = parcel->writeFloat(val);
   1380         if (err != NO_ERROR) {
   1381             signalExceptionForError(env, clazz, err);
   1382         }
   1383     }
   1384 }
   1385 
   1386 static void android_os_Parcel_writeDouble(JNIEnv* env, jobject clazz, jdouble val)
   1387 {
   1388     Parcel* parcel = parcelForJavaObject(env, clazz);
   1389     if (parcel != NULL) {
   1390         const status_t err = parcel->writeDouble(val);
   1391         if (err != NO_ERROR) {
   1392             signalExceptionForError(env, clazz, err);
   1393         }
   1394     }
   1395 }
   1396 
   1397 static void android_os_Parcel_writeString(JNIEnv* env, jobject clazz, jstring val)
   1398 {
   1399     Parcel* parcel = parcelForJavaObject(env, clazz);
   1400     if (parcel != NULL) {
   1401         status_t err = NO_MEMORY;
   1402         if (val) {
   1403             const jchar* str = env->GetStringCritical(val, 0);
   1404             if (str) {
   1405                 err = parcel->writeString16(str, env->GetStringLength(val));
   1406                 env->ReleaseStringCritical(val, str);
   1407             }
   1408         } else {
   1409             err = parcel->writeString16(NULL, 0);
   1410         }
   1411         if (err != NO_ERROR) {
   1412             signalExceptionForError(env, clazz, err);
   1413         }
   1414     }
   1415 }
   1416 
   1417 static void android_os_Parcel_writeStrongBinder(JNIEnv* env, jobject clazz, jobject object)
   1418 {
   1419     Parcel* parcel = parcelForJavaObject(env, clazz);
   1420     if (parcel != NULL) {
   1421         const status_t err = parcel->writeStrongBinder(ibinderForJavaObject(env, object));
   1422         if (err != NO_ERROR) {
   1423             signalExceptionForError(env, clazz, err);
   1424         }
   1425     }
   1426 }
   1427 
   1428 static void android_os_Parcel_writeFileDescriptor(JNIEnv* env, jobject clazz, jobject object)
   1429 {
   1430     Parcel* parcel = parcelForJavaObject(env, clazz);
   1431     if (parcel != NULL) {
   1432         const status_t err =
   1433                 parcel->writeDupFileDescriptor(jniGetFDFromFileDescriptor(env, object));
   1434         if (err != NO_ERROR) {
   1435             signalExceptionForError(env, clazz, err);
   1436         }
   1437     }
   1438 }
   1439 
   1440 static jbyteArray android_os_Parcel_createByteArray(JNIEnv* env, jobject clazz)
   1441 {
   1442     jbyteArray ret = NULL;
   1443 
   1444     Parcel* parcel = parcelForJavaObject(env, clazz);
   1445     if (parcel != NULL) {
   1446         int32_t len = parcel->readInt32();
   1447 
   1448         // sanity check the stored length against the true data size
   1449         if (len >= 0 && len <= (int32_t)parcel->dataAvail()) {
   1450             ret = env->NewByteArray(len);
   1451 
   1452             if (ret != NULL) {
   1453                 jbyte* a2 = (jbyte*)env->GetPrimitiveArrayCritical(ret, 0);
   1454                 if (a2) {
   1455                     const void* data = parcel->readInplace(len);
   1456                     memcpy(a2, data, len);
   1457                     env->ReleasePrimitiveArrayCritical(ret, a2, 0);
   1458                 }
   1459             }
   1460         }
   1461     }
   1462 
   1463     return ret;
   1464 }
   1465 
   1466 static jint android_os_Parcel_readInt(JNIEnv* env, jobject clazz)
   1467 {
   1468     Parcel* parcel = parcelForJavaObject(env, clazz);
   1469     if (parcel != NULL) {
   1470         return parcel->readInt32();
   1471     }
   1472     return 0;
   1473 }
   1474 
   1475 static jlong android_os_Parcel_readLong(JNIEnv* env, jobject clazz)
   1476 {
   1477     Parcel* parcel = parcelForJavaObject(env, clazz);
   1478     if (parcel != NULL) {
   1479         return parcel->readInt64();
   1480     }
   1481     return 0;
   1482 }
   1483 
   1484 static jfloat android_os_Parcel_readFloat(JNIEnv* env, jobject clazz)
   1485 {
   1486     Parcel* parcel = parcelForJavaObject(env, clazz);
   1487     if (parcel != NULL) {
   1488         return parcel->readFloat();
   1489     }
   1490     return 0;
   1491 }
   1492 
   1493 static jdouble android_os_Parcel_readDouble(JNIEnv* env, jobject clazz)
   1494 {
   1495     Parcel* parcel = parcelForJavaObject(env, clazz);
   1496     if (parcel != NULL) {
   1497         return parcel->readDouble();
   1498     }
   1499     return 0;
   1500 }
   1501 
   1502 static jstring android_os_Parcel_readString(JNIEnv* env, jobject clazz)
   1503 {
   1504     Parcel* parcel = parcelForJavaObject(env, clazz);
   1505     if (parcel != NULL) {
   1506         size_t len;
   1507         const char16_t* str = parcel->readString16Inplace(&len);
   1508         if (str) {
   1509             return env->NewString(str, len);
   1510         }
   1511         return NULL;
   1512     }
   1513     return NULL;
   1514 }
   1515 
   1516 static jobject android_os_Parcel_readStrongBinder(JNIEnv* env, jobject clazz)
   1517 {
   1518     Parcel* parcel = parcelForJavaObject(env, clazz);
   1519     if (parcel != NULL) {
   1520         return javaObjectForIBinder(env, parcel->readStrongBinder());
   1521     }
   1522     return NULL;
   1523 }
   1524 
   1525 static jobject android_os_Parcel_readFileDescriptor(JNIEnv* env, jobject clazz)
   1526 {
   1527     Parcel* parcel = parcelForJavaObject(env, clazz);
   1528     if (parcel != NULL) {
   1529         int fd = parcel->readFileDescriptor();
   1530         if (fd < 0) return NULL;
   1531         fd = dup(fd);
   1532         if (fd < 0) return NULL;
   1533         return jniCreateFileDescriptor(env, fd);
   1534     }
   1535     return NULL;
   1536 }
   1537 
   1538 static jobject android_os_Parcel_openFileDescriptor(JNIEnv* env, jobject clazz,
   1539                                                     jstring name, jint mode)
   1540 {
   1541     if (name == NULL) {
   1542         jniThrowNullPointerException(env, NULL);
   1543         return NULL;
   1544     }
   1545     const jchar* str = env->GetStringCritical(name, 0);
   1546     if (str == NULL) {
   1547         // Whatever, whatever.
   1548         jniThrowException(env, "java/lang/IllegalStateException", NULL);
   1549         return NULL;
   1550     }
   1551     String8 name8(str, env->GetStringLength(name));
   1552     env->ReleaseStringCritical(name, str);
   1553     int flags=0;
   1554     switch (mode&0x30000000) {
   1555         case 0:
   1556         case 0x10000000:
   1557             flags = O_RDONLY;
   1558             break;
   1559         case 0x20000000:
   1560             flags = O_WRONLY;
   1561             break;
   1562         case 0x30000000:
   1563             flags = O_RDWR;
   1564             break;
   1565     }
   1566 
   1567     if (mode&0x08000000) flags |= O_CREAT;
   1568     if (mode&0x04000000) flags |= O_TRUNC;
   1569     if (mode&0x02000000) flags |= O_APPEND;
   1570 
   1571     int realMode = S_IRWXU|S_IRWXG;
   1572     if (mode&0x00000001) realMode |= S_IROTH;
   1573     if (mode&0x00000002) realMode |= S_IWOTH;
   1574 
   1575     int fd = open(name8.string(), flags, realMode);
   1576     if (fd < 0) {
   1577         jniThrowException(env, "java/io/FileNotFoundException", strerror(errno));
   1578         return NULL;
   1579     }
   1580     jobject object = jniCreateFileDescriptor(env, fd);
   1581     if (object == NULL) {
   1582         close(fd);
   1583     }
   1584     return object;
   1585 }
   1586 
   1587 static jobject android_os_Parcel_dupFileDescriptor(JNIEnv* env, jobject clazz, jobject orig)
   1588 {
   1589     if (orig == NULL) {
   1590         jniThrowNullPointerException(env, NULL);
   1591         return NULL;
   1592     }
   1593     int origfd = jniGetFDFromFileDescriptor(env, orig);
   1594     if (origfd < 0) {
   1595         jniThrowException(env, "java/lang/IllegalArgumentException", "bad FileDescriptor");
   1596         return NULL;
   1597     }
   1598 
   1599     int fd = dup(origfd);
   1600     if (fd < 0) {
   1601         jniThrowIOException(env, errno);
   1602         return NULL;
   1603     }
   1604     jobject object = jniCreateFileDescriptor(env, fd);
   1605     if (object == NULL) {
   1606         close(fd);
   1607     }
   1608     return object;
   1609 }
   1610 
   1611 static void android_os_Parcel_closeFileDescriptor(JNIEnv* env, jobject clazz, jobject object)
   1612 {
   1613     if (object == NULL) {
   1614         jniThrowNullPointerException(env, NULL);
   1615         return;
   1616     }
   1617     int fd = jniGetFDFromFileDescriptor(env, object);
   1618     if (fd >= 0) {
   1619         jniSetFileDescriptorOfFD(env, object, -1);
   1620         //LOGI("Closing ParcelFileDescriptor %d\n", fd);
   1621         close(fd);
   1622     }
   1623 }
   1624 
   1625 static void android_os_Parcel_clearFileDescriptor(JNIEnv* env, jobject clazz, jobject object)
   1626 {
   1627     if (object == NULL) {
   1628         jniThrowNullPointerException(env, NULL);
   1629         return;
   1630     }
   1631     int fd = jniGetFDFromFileDescriptor(env, object);
   1632     if (fd >= 0) {
   1633         jniSetFileDescriptorOfFD(env, object, -1);
   1634     }
   1635 }
   1636 
   1637 static void android_os_Parcel_freeBuffer(JNIEnv* env, jobject clazz)
   1638 {
   1639     int32_t own = env->GetIntField(clazz, gParcelOffsets.mOwnObject);
   1640     if (own) {
   1641         Parcel* parcel = parcelForJavaObject(env, clazz);
   1642         if (parcel != NULL) {
   1643             //LOGI("Parcel.freeBuffer() called for C++ Parcel %p\n", parcel);
   1644             parcel->freeData();
   1645         }
   1646     }
   1647 }
   1648 
   1649 static void android_os_Parcel_init(JNIEnv* env, jobject clazz, jint parcelInt)
   1650 {
   1651     Parcel* parcel = (Parcel*)parcelInt;
   1652     int own = 0;
   1653     if (!parcel) {
   1654         //LOGI("Initializing obj %p: creating new Parcel\n", clazz);
   1655         own = 1;
   1656         parcel = new Parcel;
   1657     } else {
   1658         //LOGI("Initializing obj %p: given existing Parcel %p\n", clazz, parcel);
   1659     }
   1660     if (parcel == NULL) {
   1661         jniThrowException(env, "java/lang/OutOfMemoryError", NULL);
   1662         return;
   1663     }
   1664     //LOGI("Initializing obj %p from C++ Parcel %p, own=%d\n", clazz, parcel, own);
   1665     env->SetIntField(clazz, gParcelOffsets.mOwnObject, own);
   1666     env->SetIntField(clazz, gParcelOffsets.mObject, (int)parcel);
   1667 }
   1668 
   1669 static void android_os_Parcel_destroy(JNIEnv* env, jobject clazz)
   1670 {
   1671     int32_t own = env->GetIntField(clazz, gParcelOffsets.mOwnObject);
   1672     if (own) {
   1673         Parcel* parcel = parcelForJavaObject(env, clazz);
   1674         env->SetIntField(clazz, gParcelOffsets.mObject, 0);
   1675         //LOGI("Destroying obj %p: deleting C++ Parcel %p\n", clazz, parcel);
   1676         delete parcel;
   1677     } else {
   1678         env->SetIntField(clazz, gParcelOffsets.mObject, 0);
   1679         //LOGI("Destroying obj %p: leaving C++ Parcel %p\n", clazz);
   1680     }
   1681 }
   1682 
   1683 static jbyteArray android_os_Parcel_marshall(JNIEnv* env, jobject clazz)
   1684 {
   1685     Parcel* parcel = parcelForJavaObject(env, clazz);
   1686     if (parcel == NULL) {
   1687        return NULL;
   1688     }
   1689 
   1690     // do not marshall if there are binder objects in the parcel
   1691     if (parcel->objectsCount())
   1692     {
   1693         jniThrowException(env, "java/lang/RuntimeException", "Tried to marshall a Parcel that contained Binder objects.");
   1694         return NULL;
   1695     }
   1696 
   1697     jbyteArray ret = env->NewByteArray(parcel->dataSize());
   1698 
   1699     if (ret != NULL)
   1700     {
   1701         jbyte* array = (jbyte*)env->GetPrimitiveArrayCritical(ret, 0);
   1702         if (array != NULL)
   1703         {
   1704             memcpy(array, parcel->data(), parcel->dataSize());
   1705             env->ReleasePrimitiveArrayCritical(ret, array, 0);
   1706         }
   1707     }
   1708 
   1709     return ret;
   1710 }
   1711 
   1712 static void android_os_Parcel_unmarshall(JNIEnv* env, jobject clazz, jbyteArray data, jint offset, jint length)
   1713 {
   1714     Parcel* parcel = parcelForJavaObject(env, clazz);
   1715     if (parcel == NULL || length < 0) {
   1716        return;
   1717     }
   1718 
   1719     jbyte* array = (jbyte*)env->GetPrimitiveArrayCritical(data, 0);
   1720     if (array)
   1721     {
   1722         parcel->setDataSize(length);
   1723         parcel->setDataPosition(0);
   1724 
   1725         void* raw = parcel->writeInplace(length);
   1726         memcpy(raw, (array + offset), length);
   1727 
   1728         env->ReleasePrimitiveArrayCritical(data, array, 0);
   1729     }
   1730 }
   1731 
   1732 static void android_os_Parcel_appendFrom(JNIEnv* env, jobject clazz, jobject parcel, jint offset, jint length)
   1733 {
   1734     Parcel* thisParcel = parcelForJavaObject(env, clazz);
   1735     if (thisParcel == NULL) {
   1736        return;
   1737     }
   1738     Parcel* otherParcel = parcelForJavaObject(env, parcel);
   1739     if (otherParcel == NULL) {
   1740        return;
   1741     }
   1742 
   1743     status_t err = thisParcel->appendFrom(otherParcel, offset, length);
   1744     if (err != NO_ERROR) {
   1745         signalExceptionForError(env, clazz, err);
   1746     }
   1747 }
   1748 
   1749 static jboolean android_os_Parcel_hasFileDescriptors(JNIEnv* env, jobject clazz)
   1750 {
   1751     jboolean ret = JNI_FALSE;
   1752     Parcel* parcel = parcelForJavaObject(env, clazz);
   1753     if (parcel != NULL) {
   1754         if (parcel->hasFileDescriptors()) {
   1755             ret = JNI_TRUE;
   1756         }
   1757     }
   1758     return ret;
   1759 }
   1760 
   1761 static void android_os_Parcel_writeInterfaceToken(JNIEnv* env, jobject clazz, jstring name)
   1762 {
   1763     Parcel* parcel = parcelForJavaObject(env, clazz);
   1764     if (parcel != NULL) {
   1765         // In the current implementation, the token is just the serialized interface name that
   1766         // the caller expects to be invoking
   1767         const jchar* str = env->GetStringCritical(name, 0);
   1768         if (str != NULL) {
   1769             parcel->writeInterfaceToken(String16(str, env->GetStringLength(name)));
   1770             env->ReleaseStringCritical(name, str);
   1771         }
   1772     }
   1773 }
   1774 
   1775 static void android_os_Parcel_enforceInterface(JNIEnv* env, jobject clazz, jstring name)
   1776 {
   1777     jboolean ret = JNI_FALSE;
   1778 
   1779     Parcel* parcel = parcelForJavaObject(env, clazz);
   1780     if (parcel != NULL) {
   1781         const jchar* str = env->GetStringCritical(name, 0);
   1782         if (str) {
   1783             IPCThreadState* threadState = IPCThreadState::self();
   1784             const int32_t oldPolicy = threadState->getStrictModePolicy();
   1785             const bool isValid = parcel->enforceInterface(
   1786                 String16(str, env->GetStringLength(name)),
   1787                 threadState);
   1788             env->ReleaseStringCritical(name, str);
   1789             if (isValid) {
   1790                 const int32_t newPolicy = threadState->getStrictModePolicy();
   1791                 if (oldPolicy != newPolicy) {
   1792                     // Need to keep the Java-level thread-local strict
   1793                     // mode policy in sync for the libcore
   1794                     // enforcements, which involves an upcall back
   1795                     // into Java.  (We can't modify the
   1796                     // Parcel.enforceInterface signature, as it's
   1797                     // pseudo-public, and used via AIDL
   1798                     // auto-generation...)
   1799                     set_dalvik_blockguard_policy(env, newPolicy);
   1800                 }
   1801                 return;     // everything was correct -> return silently
   1802             }
   1803         }
   1804     }
   1805 
   1806     // all error conditions wind up here
   1807     jniThrowException(env, "java/lang/SecurityException",
   1808             "Binder invocation to an incorrect interface");
   1809 }
   1810 
   1811 // ----------------------------------------------------------------------------
   1812 
   1813 static const JNINativeMethod gParcelMethods[] = {
   1814     {"dataSize",            "()I", (void*)android_os_Parcel_dataSize},
   1815     {"dataAvail",           "()I", (void*)android_os_Parcel_dataAvail},
   1816     {"dataPosition",        "()I", (void*)android_os_Parcel_dataPosition},
   1817     {"dataCapacity",        "()I", (void*)android_os_Parcel_dataCapacity},
   1818     {"setDataSize",         "(I)V", (void*)android_os_Parcel_setDataSize},
   1819     {"setDataPosition",     "(I)V", (void*)android_os_Parcel_setDataPosition},
   1820     {"setDataCapacity",     "(I)V", (void*)android_os_Parcel_setDataCapacity},
   1821     {"pushAllowFds",        "(Z)Z", (void*)android_os_Parcel_pushAllowFds},
   1822     {"restoreAllowFds",     "(Z)V", (void*)android_os_Parcel_restoreAllowFds},
   1823     {"writeNative",         "([BII)V", (void*)android_os_Parcel_writeNative},
   1824     {"writeInt",            "(I)V", (void*)android_os_Parcel_writeInt},
   1825     {"writeLong",           "(J)V", (void*)android_os_Parcel_writeLong},
   1826     {"writeFloat",          "(F)V", (void*)android_os_Parcel_writeFloat},
   1827     {"writeDouble",         "(D)V", (void*)android_os_Parcel_writeDouble},
   1828     {"writeString",         "(Ljava/lang/String;)V", (void*)android_os_Parcel_writeString},
   1829     {"writeStrongBinder",   "(Landroid/os/IBinder;)V", (void*)android_os_Parcel_writeStrongBinder},
   1830     {"writeFileDescriptor", "(Ljava/io/FileDescriptor;)V", (void*)android_os_Parcel_writeFileDescriptor},
   1831     {"createByteArray",     "()[B", (void*)android_os_Parcel_createByteArray},
   1832     {"readInt",             "()I", (void*)android_os_Parcel_readInt},
   1833     {"readLong",            "()J", (void*)android_os_Parcel_readLong},
   1834     {"readFloat",           "()F", (void*)android_os_Parcel_readFloat},
   1835     {"readDouble",          "()D", (void*)android_os_Parcel_readDouble},
   1836     {"readString",          "()Ljava/lang/String;", (void*)android_os_Parcel_readString},
   1837     {"readStrongBinder",    "()Landroid/os/IBinder;", (void*)android_os_Parcel_readStrongBinder},
   1838     {"internalReadFileDescriptor",  "()Ljava/io/FileDescriptor;", (void*)android_os_Parcel_readFileDescriptor},
   1839     {"openFileDescriptor",  "(Ljava/lang/String;I)Ljava/io/FileDescriptor;", (void*)android_os_Parcel_openFileDescriptor},
   1840     {"dupFileDescriptor",   "(Ljava/io/FileDescriptor;)Ljava/io/FileDescriptor;", (void*)android_os_Parcel_dupFileDescriptor},
   1841     {"closeFileDescriptor", "(Ljava/io/FileDescriptor;)V", (void*)android_os_Parcel_closeFileDescriptor},
   1842     {"clearFileDescriptor", "(Ljava/io/FileDescriptor;)V", (void*)android_os_Parcel_clearFileDescriptor},
   1843     {"freeBuffer",          "()V", (void*)android_os_Parcel_freeBuffer},
   1844     {"init",                "(I)V", (void*)android_os_Parcel_init},
   1845     {"destroy",             "()V", (void*)android_os_Parcel_destroy},
   1846     {"marshall",            "()[B", (void*)android_os_Parcel_marshall},
   1847     {"unmarshall",          "([BII)V", (void*)android_os_Parcel_unmarshall},
   1848     {"appendFrom",          "(Landroid/os/Parcel;II)V", (void*)android_os_Parcel_appendFrom},
   1849     {"hasFileDescriptors",  "()Z", (void*)android_os_Parcel_hasFileDescriptors},
   1850     {"writeInterfaceToken", "(Ljava/lang/String;)V", (void*)android_os_Parcel_writeInterfaceToken},
   1851     {"enforceInterface",    "(Ljava/lang/String;)V", (void*)android_os_Parcel_enforceInterface},
   1852 };
   1853 
   1854 const char* const kParcelPathName = "android/os/Parcel";
   1855 
   1856 static int int_register_android_os_Parcel(JNIEnv* env)
   1857 {
   1858     jclass clazz;
   1859 
   1860     clazz = env->FindClass("android/util/Log");
   1861     LOG_FATAL_IF(clazz == NULL, "Unable to find class android.util.Log");
   1862     gLogOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
   1863     gLogOffsets.mLogE = env->GetStaticMethodID(
   1864         clazz, "e", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/Throwable;)I");
   1865     assert(gLogOffsets.mLogE);
   1866 
   1867     clazz = env->FindClass("android/os/ParcelFileDescriptor");
   1868     LOG_FATAL_IF(clazz == NULL, "Unable to find class android.os.ParcelFileDescriptor");
   1869     gParcelFileDescriptorOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
   1870     gParcelFileDescriptorOffsets.mConstructor
   1871         = env->GetMethodID(clazz, "<init>", "(Ljava/io/FileDescriptor;)V");
   1872 
   1873     clazz = env->FindClass(kParcelPathName);
   1874     LOG_FATAL_IF(clazz == NULL, "Unable to find class android.os.Parcel");
   1875 
   1876     gParcelOffsets.mObject
   1877         = env->GetFieldID(clazz, "mObject", "I");
   1878     gParcelOffsets.mOwnObject
   1879         = env->GetFieldID(clazz, "mOwnObject", "I");
   1880 
   1881     clazz = env->FindClass("android/os/StrictMode");
   1882     LOG_FATAL_IF(clazz == NULL, "Unable to find class android.os.StrictMode");
   1883     gStrictModeCallbackOffsets.mClass = (jclass) env->NewGlobalRef(clazz);
   1884     gStrictModeCallbackOffsets.mCallback = env->GetStaticMethodID(
   1885         clazz, "onBinderStrictModePolicyChange", "(I)V");
   1886     LOG_FATAL_IF(gStrictModeCallbackOffsets.mCallback == NULL,
   1887                  "Unable to find strict mode callback.");
   1888 
   1889     return AndroidRuntime::registerNativeMethods(
   1890         env, kParcelPathName,
   1891         gParcelMethods, NELEM(gParcelMethods));
   1892 }
   1893 
   1894 int register_android_os_Binder(JNIEnv* env)
   1895 {
   1896     if (int_register_android_os_Binder(env) < 0)
   1897         return -1;
   1898     if (int_register_android_os_BinderInternal(env) < 0)
   1899         return -1;
   1900     if (int_register_android_os_BinderProxy(env) < 0)
   1901         return -1;
   1902     if (int_register_android_os_Parcel(env) < 0)
   1903         return -1;
   1904     return 0;
   1905 }
   1906