• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.os;
18 
19 import android.util.Config;
20 import android.util.Log;
21 
22 import java.io.FileDescriptor;
23 import java.io.FileOutputStream;
24 import java.io.IOException;
25 import java.io.PrintWriter;
26 import java.lang.ref.WeakReference;
27 import java.lang.reflect.Modifier;
28 
29 /**
30  * Base class for a remotable object, the core part of a lightweight
31  * remote procedure call mechanism defined by {@link IBinder}.
32  * This class is an implementation of IBinder that provides
33  * the standard support creating a local implementation of such an object.
34  *
35  * <p>Most developers will not implement this class directly, instead using the
36  * <a href="{@docRoot}guide/developing/tools/aidl.html">aidl</a> tool to describe the desired
37  * interface, having it generate the appropriate Binder subclass.  You can,
38  * however, derive directly from Binder to implement your own custom RPC
39  * protocol or simply instantiate a raw Binder object directly to use as a
40  * token that can be shared across processes.
41  *
42  * @see IBinder
43  */
44 public class Binder implements IBinder {
45     /*
46      * Set this flag to true to detect anonymous, local or member classes
47      * that extend this Binder class and that are not static. These kind
48      * of classes can potentially create leaks.
49      */
50     private static final boolean FIND_POTENTIAL_LEAKS = false;
51     private static final String TAG = "Binder";
52 
53     private int mObject;
54     private IInterface mOwner;
55     private String mDescriptor;
56 
57     /**
58      * Return the ID of the process that sent you the current transaction
59      * that is being processed.  This pid can be used with higher-level
60      * system services to determine its identity and check permissions.
61      * If the current thread is not currently executing an incoming transaction,
62      * then its own pid is returned.
63      */
getCallingPid()64     public static final native int getCallingPid();
65 
66     /**
67      * Return the ID of the user assigned to the process that sent you the
68      * current transaction that is being processed.  This uid can be used with
69      * higher-level system services to determine its identity and check
70      * permissions.  If the current thread is not currently executing an
71      * incoming transaction, then its own uid is returned.
72      */
getCallingUid()73     public static final native int getCallingUid();
74 
75     /**
76      * Reset the identity of the incoming IPC on the current thread.  This can
77      * be useful if, while handling an incoming call, you will be calling
78      * on interfaces of other objects that may be local to your process and
79      * need to do permission checks on the calls coming into them (so they
80      * will check the permission of your own local process, and not whatever
81      * process originally called you).
82      *
83      * @return Returns an opaque token that can be used to restore the
84      * original calling identity by passing it to
85      * {@link #restoreCallingIdentity(long)}.
86      *
87      * @see #getCallingPid()
88      * @see #getCallingUid()
89      * @see #restoreCallingIdentity(long)
90      */
clearCallingIdentity()91     public static final native long clearCallingIdentity();
92 
93     /**
94      * Restore the identity of the incoming IPC on the current thread
95      * back to a previously identity that was returned by {@link
96      * #clearCallingIdentity}.
97      *
98      * @param token The opaque token that was previously returned by
99      * {@link #clearCallingIdentity}.
100      *
101      * @see #clearCallingIdentity
102      */
restoreCallingIdentity(long token)103     public static final native void restoreCallingIdentity(long token);
104 
105     /**
106      * Sets the native thread-local StrictMode policy mask.
107      *
108      * <p>The StrictMode settings are kept in two places: a Java-level
109      * threadlocal for libcore/Dalvik, and a native threadlocal (set
110      * here) for propagation via Binder calls.  This is a little
111      * unfortunate, but necessary to break otherwise more unfortunate
112      * dependencies either of Dalvik on Android, or Android
113      * native-only code on Dalvik.
114      *
115      * @see StrictMode
116      * @hide
117      */
setThreadStrictModePolicy(int policyMask)118     public static final native void setThreadStrictModePolicy(int policyMask);
119 
120     /**
121      * Gets the current native thread-local StrictMode policy mask.
122      *
123      * @see #setThreadStrictModePolicy
124      * @hide
125      */
getThreadStrictModePolicy()126     public static final native int getThreadStrictModePolicy();
127 
128     /**
129      * Flush any Binder commands pending in the current thread to the kernel
130      * driver.  This can be
131      * useful to call before performing an operation that may block for a long
132      * time, to ensure that any pending object references have been released
133      * in order to prevent the process from holding on to objects longer than
134      * it needs to.
135      */
flushPendingCommands()136     public static final native void flushPendingCommands();
137 
138     /**
139      * Add the calling thread to the IPC thread pool.  This function does
140      * not return until the current process is exiting.
141      */
joinThreadPool()142     public static final native void joinThreadPool();
143 
144     /**
145      * Default constructor initializes the object.
146      */
Binder()147     public Binder() {
148         init();
149 
150         if (FIND_POTENTIAL_LEAKS) {
151             final Class<? extends Binder> klass = getClass();
152             if ((klass.isAnonymousClass() || klass.isMemberClass() || klass.isLocalClass()) &&
153                     (klass.getModifiers() & Modifier.STATIC) == 0) {
154                 Log.w(TAG, "The following Binder class should be static or leaks might occur: " +
155                     klass.getCanonicalName());
156             }
157         }
158     }
159 
160     /**
161      * Convenience method for associating a specific interface with the Binder.
162      * After calling, queryLocalInterface() will be implemented for you
163      * to return the given owner IInterface when the corresponding
164      * descriptor is requested.
165      */
attachInterface(IInterface owner, String descriptor)166     public void attachInterface(IInterface owner, String descriptor) {
167         mOwner = owner;
168         mDescriptor = descriptor;
169     }
170 
171     /**
172      * Default implementation returns an empty interface name.
173      */
getInterfaceDescriptor()174     public String getInterfaceDescriptor() {
175         return mDescriptor;
176     }
177 
178     /**
179      * Default implementation always returns true -- if you got here,
180      * the object is alive.
181      */
pingBinder()182     public boolean pingBinder() {
183         return true;
184     }
185 
186     /**
187      * {@inheritDoc}
188      *
189      * Note that if you're calling on a local binder, this always returns true
190      * because your process is alive if you're calling it.
191      */
isBinderAlive()192     public boolean isBinderAlive() {
193         return true;
194     }
195 
196     /**
197      * Use information supplied to attachInterface() to return the
198      * associated IInterface if it matches the requested
199      * descriptor.
200      */
queryLocalInterface(String descriptor)201     public IInterface queryLocalInterface(String descriptor) {
202         if (mDescriptor.equals(descriptor)) {
203             return mOwner;
204         }
205         return null;
206     }
207 
208     /**
209      * Default implementation is a stub that returns false.  You will want
210      * to override this to do the appropriate unmarshalling of transactions.
211      *
212      * <p>If you want to call this, call transact().
213      */
onTransact(int code, Parcel data, Parcel reply, int flags)214     protected boolean onTransact(int code, Parcel data, Parcel reply,
215             int flags) throws RemoteException {
216         if (code == INTERFACE_TRANSACTION) {
217             reply.writeString(getInterfaceDescriptor());
218             return true;
219         } else if (code == DUMP_TRANSACTION) {
220             ParcelFileDescriptor fd = data.readFileDescriptor();
221             String[] args = data.readStringArray();
222             if (fd != null) {
223                 try {
224                     dump(fd.getFileDescriptor(), args);
225                 } finally {
226                     try {
227                         fd.close();
228                     } catch (IOException e) {
229                         // swallowed, not propagated back to the caller
230                     }
231                 }
232             }
233             // Write the StrictMode header.
234             if (reply != null) {
235                 reply.writeNoException();
236             } else {
237                 StrictMode.clearGatheredViolations();
238             }
239             return true;
240         }
241         return false;
242     }
243 
244     /**
245      * Implemented to call the more convenient version
246      * {@link #dump(FileDescriptor, PrintWriter, String[])}.
247      */
dump(FileDescriptor fd, String[] args)248     public void dump(FileDescriptor fd, String[] args) {
249         FileOutputStream fout = new FileOutputStream(fd);
250         PrintWriter pw = new PrintWriter(fout);
251         try {
252             dump(fd, pw, args);
253         } finally {
254             pw.flush();
255         }
256     }
257 
258     /**
259      * Print the object's state into the given stream.
260      *
261      * @param fd The raw file descriptor that the dump is being sent to.
262      * @param fout The file to which you should dump your state.  This will be
263      * closed for you after you return.
264      * @param args additional arguments to the dump request.
265      */
dump(FileDescriptor fd, PrintWriter fout, String[] args)266     protected void dump(FileDescriptor fd, PrintWriter fout, String[] args) {
267     }
268 
269     /**
270      * Default implementation rewinds the parcels and calls onTransact.  On
271      * the remote side, transact calls into the binder to do the IPC.
272      */
transact(int code, Parcel data, Parcel reply, int flags)273     public final boolean transact(int code, Parcel data, Parcel reply,
274             int flags) throws RemoteException {
275         if (Config.LOGV) Log.v("Binder", "Transact: " + code + " to " + this);
276         if (data != null) {
277             data.setDataPosition(0);
278         }
279         boolean r = onTransact(code, data, reply, flags);
280         if (reply != null) {
281             reply.setDataPosition(0);
282         }
283         return r;
284     }
285 
286     /**
287      * Local implementation is a no-op.
288      */
linkToDeath(DeathRecipient recipient, int flags)289     public void linkToDeath(DeathRecipient recipient, int flags) {
290     }
291 
292     /**
293      * Local implementation is a no-op.
294      */
unlinkToDeath(DeathRecipient recipient, int flags)295     public boolean unlinkToDeath(DeathRecipient recipient, int flags) {
296         return true;
297     }
298 
finalize()299     protected void finalize() throws Throwable {
300         try {
301             destroy();
302         } finally {
303             super.finalize();
304         }
305     }
306 
init()307     private native final void init();
destroy()308     private native final void destroy();
309 
310     // Entry point from android_util_Binder.cpp's onTransact
execTransact(int code, int dataObj, int replyObj, int flags)311     private boolean execTransact(int code, int dataObj, int replyObj,
312             int flags) {
313         Parcel data = Parcel.obtain(dataObj);
314         Parcel reply = Parcel.obtain(replyObj);
315         // theoretically, we should call transact, which will call onTransact,
316         // but all that does is rewind it, and we just got these from an IPC,
317         // so we'll just call it directly.
318         boolean res;
319         try {
320             res = onTransact(code, data, reply, flags);
321         } catch (RemoteException e) {
322             reply.writeException(e);
323             res = true;
324         } catch (RuntimeException e) {
325             reply.writeException(e);
326             res = true;
327         }
328         reply.recycle();
329         data.recycle();
330         return res;
331     }
332 }
333 
334 final class BinderProxy implements IBinder {
pingBinder()335     public native boolean pingBinder();
isBinderAlive()336     public native boolean isBinderAlive();
337 
queryLocalInterface(String descriptor)338     public IInterface queryLocalInterface(String descriptor) {
339         return null;
340     }
341 
getInterfaceDescriptor()342     public native String getInterfaceDescriptor() throws RemoteException;
transact(int code, Parcel data, Parcel reply, int flags)343     public native boolean transact(int code, Parcel data, Parcel reply,
344             int flags) throws RemoteException;
linkToDeath(DeathRecipient recipient, int flags)345     public native void linkToDeath(DeathRecipient recipient, int flags)
346             throws RemoteException;
unlinkToDeath(DeathRecipient recipient, int flags)347     public native boolean unlinkToDeath(DeathRecipient recipient, int flags);
348 
dump(FileDescriptor fd, String[] args)349     public void dump(FileDescriptor fd, String[] args) throws RemoteException {
350         Parcel data = Parcel.obtain();
351         Parcel reply = Parcel.obtain();
352         data.writeFileDescriptor(fd);
353         data.writeStringArray(args);
354         try {
355             transact(DUMP_TRANSACTION, data, reply, 0);
356             reply.readException();
357         } finally {
358             data.recycle();
359             reply.recycle();
360         }
361     }
362 
BinderProxy()363     BinderProxy() {
364         mSelf = new WeakReference(this);
365     }
366 
367     @Override
finalize()368     protected void finalize() throws Throwable {
369         try {
370             destroy();
371         } finally {
372             super.finalize();
373         }
374     }
375 
destroy()376     private native final void destroy();
377 
sendDeathNotice(DeathRecipient recipient)378     private static final void sendDeathNotice(DeathRecipient recipient) {
379         if (Config.LOGV) Log.v("JavaBinder", "sendDeathNotice to " + recipient);
380         try {
381             recipient.binderDied();
382         }
383         catch (RuntimeException exc) {
384             Log.w("BinderNative", "Uncaught exception from death notification",
385                     exc);
386         }
387     }
388 
389     final private WeakReference mSelf;
390     private int mObject;
391 }
392