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