1 /* 2 * Copyright (C) 2005 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 "BpBinder" 18 //#define LOG_NDEBUG 0 19 20 #include <binder/BpBinder.h> 21 22 #include <binder/IPCThreadState.h> 23 #include <binder/IResultReceiver.h> 24 #include <cutils/compiler.h> 25 #include <utils/Log.h> 26 27 #include <stdio.h> 28 29 //#undef ALOGV 30 //#define ALOGV(...) fprintf(stderr, __VA_ARGS__) 31 32 namespace android { 33 34 // --------------------------------------------------------------------------- 35 36 Mutex BpBinder::sTrackingLock; 37 std::unordered_map<int32_t,uint32_t> BpBinder::sTrackingMap; 38 int BpBinder::sNumTrackedUids = 0; 39 std::atomic_bool BpBinder::sCountByUidEnabled(false); 40 binder_proxy_limit_callback BpBinder::sLimitCallback; 41 bool BpBinder::sBinderProxyThrottleCreate = false; 42 43 // Arbitrarily high value that probably distinguishes a bad behaving app 44 uint32_t BpBinder::sBinderProxyCountHighWatermark = 2500; 45 // Another arbitrary value a binder count needs to drop below before another callback will be called 46 uint32_t BpBinder::sBinderProxyCountLowWatermark = 2000; 47 48 enum { 49 LIMIT_REACHED_MASK = 0x80000000, // A flag denoting that the limit has been reached 50 COUNTING_VALUE_MASK = 0x7FFFFFFF, // A mask of the remaining bits for the count value 51 }; 52 53 BpBinder::ObjectManager::ObjectManager() 54 { 55 } 56 57 BpBinder::ObjectManager::~ObjectManager() 58 { 59 kill(); 60 } 61 62 void BpBinder::ObjectManager::attach( 63 const void* objectID, void* object, void* cleanupCookie, 64 IBinder::object_cleanup_func func) 65 { 66 entry_t e; 67 e.object = object; 68 e.cleanupCookie = cleanupCookie; 69 e.func = func; 70 71 if (mObjects.indexOfKey(objectID) >= 0) { 72 ALOGE("Trying to attach object ID %p to binder ObjectManager %p with object %p, but object ID already in use", 73 objectID, this, object); 74 return; 75 } 76 77 mObjects.add(objectID, e); 78 } 79 80 void* BpBinder::ObjectManager::find(const void* objectID) const 81 { 82 const ssize_t i = mObjects.indexOfKey(objectID); 83 if (i < 0) return nullptr; 84 return mObjects.valueAt(i).object; 85 } 86 87 void BpBinder::ObjectManager::detach(const void* objectID) 88 { 89 mObjects.removeItem(objectID); 90 } 91 92 void BpBinder::ObjectManager::kill() 93 { 94 const size_t N = mObjects.size(); 95 ALOGV("Killing %zu objects in manager %p", N, this); 96 for (size_t i=0; i<N; i++) { 97 const entry_t& e = mObjects.valueAt(i); 98 if (e.func != nullptr) { 99 e.func(mObjects.keyAt(i), e.object, e.cleanupCookie); 100 } 101 } 102 103 mObjects.clear(); 104 } 105 106 // --------------------------------------------------------------------------- 107 108 109 BpBinder* BpBinder::create(int32_t handle) { 110 int32_t trackedUid = -1; 111 if (sCountByUidEnabled) { 112 trackedUid = IPCThreadState::self()->getCallingUid(); 113 AutoMutex _l(sTrackingLock); 114 uint32_t trackedValue = sTrackingMap[trackedUid]; 115 if (CC_UNLIKELY(trackedValue & LIMIT_REACHED_MASK)) { 116 if (sBinderProxyThrottleCreate) { 117 return nullptr; 118 } 119 } else { 120 if ((trackedValue & COUNTING_VALUE_MASK) >= sBinderProxyCountHighWatermark) { 121 ALOGE("Too many binder proxy objects sent to uid %d from uid %d (%d proxies held)", 122 getuid(), trackedUid, trackedValue); 123 sTrackingMap[trackedUid] |= LIMIT_REACHED_MASK; 124 if (sLimitCallback) sLimitCallback(trackedUid); 125 if (sBinderProxyThrottleCreate) { 126 ALOGI("Throttling binder proxy creates from uid %d in uid %d until binder proxy" 127 " count drops below %d", 128 trackedUid, getuid(), sBinderProxyCountLowWatermark); 129 return nullptr; 130 } 131 } 132 } 133 sTrackingMap[trackedUid]++; 134 } 135 return new BpBinder(handle, trackedUid); 136 } 137 138 BpBinder::BpBinder(int32_t handle, int32_t trackedUid) 139 : mHandle(handle) 140 , mAlive(1) 141 , mObitsSent(0) 142 , mObituaries(nullptr) 143 , mTrackedUid(trackedUid) 144 { 145 ALOGV("Creating BpBinder %p handle %d\n", this, mHandle); 146 147 extendObjectLifetime(OBJECT_LIFETIME_WEAK); 148 IPCThreadState::self()->incWeakHandle(handle, this); 149 } 150 151 bool BpBinder::isDescriptorCached() const { 152 Mutex::Autolock _l(mLock); 153 return mDescriptorCache.size() ? true : false; 154 } 155 156 const String16& BpBinder::getInterfaceDescriptor() const 157 { 158 if (isDescriptorCached() == false) { 159 Parcel send, reply; 160 // do the IPC without a lock held. 161 status_t err = const_cast<BpBinder*>(this)->transact( 162 INTERFACE_TRANSACTION, send, &reply); 163 if (err == NO_ERROR) { 164 String16 res(reply.readString16()); 165 Mutex::Autolock _l(mLock); 166 // mDescriptorCache could have been assigned while the lock was 167 // released. 168 if (mDescriptorCache.size() == 0) 169 mDescriptorCache = res; 170 } 171 } 172 173 // we're returning a reference to a non-static object here. Usually this 174 // is not something smart to do, however, with binder objects it is 175 // (usually) safe because they are reference-counted. 176 177 return mDescriptorCache; 178 } 179 180 bool BpBinder::isBinderAlive() const 181 { 182 return mAlive != 0; 183 } 184 185 status_t BpBinder::pingBinder() 186 { 187 Parcel send; 188 Parcel reply; 189 status_t err = transact(PING_TRANSACTION, send, &reply); 190 if (err != NO_ERROR) return err; 191 if (reply.dataSize() < sizeof(status_t)) return NOT_ENOUGH_DATA; 192 return (status_t)reply.readInt32(); 193 } 194 195 status_t BpBinder::dump(int fd, const Vector<String16>& args) 196 { 197 Parcel send; 198 Parcel reply; 199 send.writeFileDescriptor(fd); 200 const size_t numArgs = args.size(); 201 send.writeInt32(numArgs); 202 for (size_t i = 0; i < numArgs; i++) { 203 send.writeString16(args[i]); 204 } 205 status_t err = transact(DUMP_TRANSACTION, send, &reply); 206 return err; 207 } 208 209 // NOLINTNEXTLINE(google-default-arguments) 210 status_t BpBinder::transact( 211 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) 212 { 213 // Once a binder has died, it will never come back to life. 214 if (mAlive) { 215 status_t status = IPCThreadState::self()->transact( 216 mHandle, code, data, reply, flags); 217 if (status == DEAD_OBJECT) mAlive = 0; 218 return status; 219 } 220 221 return DEAD_OBJECT; 222 } 223 224 // NOLINTNEXTLINE(google-default-arguments) 225 status_t BpBinder::linkToDeath( 226 const sp<DeathRecipient>& recipient, void* cookie, uint32_t flags) 227 { 228 Obituary ob; 229 ob.recipient = recipient; 230 ob.cookie = cookie; 231 ob.flags = flags; 232 233 LOG_ALWAYS_FATAL_IF(recipient == nullptr, 234 "linkToDeath(): recipient must be non-NULL"); 235 236 { 237 AutoMutex _l(mLock); 238 239 if (!mObitsSent) { 240 if (!mObituaries) { 241 mObituaries = new Vector<Obituary>; 242 if (!mObituaries) { 243 return NO_MEMORY; 244 } 245 ALOGV("Requesting death notification: %p handle %d\n", this, mHandle); 246 getWeakRefs()->incWeak(this); 247 IPCThreadState* self = IPCThreadState::self(); 248 self->requestDeathNotification(mHandle, this); 249 self->flushCommands(); 250 } 251 ssize_t res = mObituaries->add(ob); 252 return res >= (ssize_t)NO_ERROR ? (status_t)NO_ERROR : res; 253 } 254 } 255 256 return DEAD_OBJECT; 257 } 258 259 // NOLINTNEXTLINE(google-default-arguments) 260 status_t BpBinder::unlinkToDeath( 261 const wp<DeathRecipient>& recipient, void* cookie, uint32_t flags, 262 wp<DeathRecipient>* outRecipient) 263 { 264 AutoMutex _l(mLock); 265 266 if (mObitsSent) { 267 return DEAD_OBJECT; 268 } 269 270 const size_t N = mObituaries ? mObituaries->size() : 0; 271 for (size_t i=0; i<N; i++) { 272 const Obituary& obit = mObituaries->itemAt(i); 273 if ((obit.recipient == recipient 274 || (recipient == nullptr && obit.cookie == cookie)) 275 && obit.flags == flags) { 276 if (outRecipient != nullptr) { 277 *outRecipient = mObituaries->itemAt(i).recipient; 278 } 279 mObituaries->removeAt(i); 280 if (mObituaries->size() == 0) { 281 ALOGV("Clearing death notification: %p handle %d\n", this, mHandle); 282 IPCThreadState* self = IPCThreadState::self(); 283 self->clearDeathNotification(mHandle, this); 284 self->flushCommands(); 285 delete mObituaries; 286 mObituaries = nullptr; 287 } 288 return NO_ERROR; 289 } 290 } 291 292 return NAME_NOT_FOUND; 293 } 294 295 void BpBinder::sendObituary() 296 { 297 ALOGV("Sending obituary for proxy %p handle %d, mObitsSent=%s\n", 298 this, mHandle, mObitsSent ? "true" : "false"); 299 300 mAlive = 0; 301 if (mObitsSent) return; 302 303 mLock.lock(); 304 Vector<Obituary>* obits = mObituaries; 305 if(obits != nullptr) { 306 ALOGV("Clearing sent death notification: %p handle %d\n", this, mHandle); 307 IPCThreadState* self = IPCThreadState::self(); 308 self->clearDeathNotification(mHandle, this); 309 self->flushCommands(); 310 mObituaries = nullptr; 311 } 312 mObitsSent = 1; 313 mLock.unlock(); 314 315 ALOGV("Reporting death of proxy %p for %zu recipients\n", 316 this, obits ? obits->size() : 0U); 317 318 if (obits != nullptr) { 319 const size_t N = obits->size(); 320 for (size_t i=0; i<N; i++) { 321 reportOneDeath(obits->itemAt(i)); 322 } 323 324 delete obits; 325 } 326 } 327 328 void BpBinder::reportOneDeath(const Obituary& obit) 329 { 330 sp<DeathRecipient> recipient = obit.recipient.promote(); 331 ALOGV("Reporting death to recipient: %p\n", recipient.get()); 332 if (recipient == nullptr) return; 333 334 recipient->binderDied(this); 335 } 336 337 338 void BpBinder::attachObject( 339 const void* objectID, void* object, void* cleanupCookie, 340 object_cleanup_func func) 341 { 342 AutoMutex _l(mLock); 343 ALOGV("Attaching object %p to binder %p (manager=%p)", object, this, &mObjects); 344 mObjects.attach(objectID, object, cleanupCookie, func); 345 } 346 347 void* BpBinder::findObject(const void* objectID) const 348 { 349 AutoMutex _l(mLock); 350 return mObjects.find(objectID); 351 } 352 353 void BpBinder::detachObject(const void* objectID) 354 { 355 AutoMutex _l(mLock); 356 mObjects.detach(objectID); 357 } 358 359 BpBinder* BpBinder::remoteBinder() 360 { 361 return this; 362 } 363 364 BpBinder::~BpBinder() 365 { 366 ALOGV("Destroying BpBinder %p handle %d\n", this, mHandle); 367 368 IPCThreadState* ipc = IPCThreadState::self(); 369 370 if (mTrackedUid >= 0) { 371 AutoMutex _l(sTrackingLock); 372 uint32_t trackedValue = sTrackingMap[mTrackedUid]; 373 if (CC_UNLIKELY((trackedValue & COUNTING_VALUE_MASK) == 0)) { 374 ALOGE("Unexpected Binder Proxy tracking decrement in %p handle %d\n", this, mHandle); 375 } else { 376 if (CC_UNLIKELY( 377 (trackedValue & LIMIT_REACHED_MASK) && 378 ((trackedValue & COUNTING_VALUE_MASK) <= sBinderProxyCountLowWatermark) 379 )) { 380 ALOGI("Limit reached bit reset for uid %d (fewer than %d proxies from uid %d held)", 381 getuid(), mTrackedUid, sBinderProxyCountLowWatermark); 382 sTrackingMap[mTrackedUid] &= ~LIMIT_REACHED_MASK; 383 } 384 if (--sTrackingMap[mTrackedUid] == 0) { 385 sTrackingMap.erase(mTrackedUid); 386 } 387 } 388 } 389 390 mLock.lock(); 391 Vector<Obituary>* obits = mObituaries; 392 if(obits != nullptr) { 393 if (ipc) ipc->clearDeathNotification(mHandle, this); 394 mObituaries = nullptr; 395 } 396 mLock.unlock(); 397 398 if (obits != nullptr) { 399 // XXX Should we tell any remaining DeathRecipient 400 // objects that the last strong ref has gone away, so they 401 // are no longer linked? 402 delete obits; 403 } 404 405 if (ipc) { 406 ipc->expungeHandle(mHandle, this); 407 ipc->decWeakHandle(mHandle); 408 } 409 } 410 411 void BpBinder::onFirstRef() 412 { 413 ALOGV("onFirstRef BpBinder %p handle %d\n", this, mHandle); 414 IPCThreadState* ipc = IPCThreadState::self(); 415 if (ipc) ipc->incStrongHandle(mHandle, this); 416 } 417 418 void BpBinder::onLastStrongRef(const void* /*id*/) 419 { 420 ALOGV("onLastStrongRef BpBinder %p handle %d\n", this, mHandle); 421 IF_ALOGV() { 422 printRefs(); 423 } 424 IPCThreadState* ipc = IPCThreadState::self(); 425 if (ipc) ipc->decStrongHandle(mHandle); 426 } 427 428 bool BpBinder::onIncStrongAttempted(uint32_t /*flags*/, const void* /*id*/) 429 { 430 ALOGV("onIncStrongAttempted BpBinder %p handle %d\n", this, mHandle); 431 IPCThreadState* ipc = IPCThreadState::self(); 432 return ipc ? ipc->attemptIncStrongHandle(mHandle) == NO_ERROR : false; 433 } 434 435 uint32_t BpBinder::getBinderProxyCount(uint32_t uid) 436 { 437 AutoMutex _l(sTrackingLock); 438 auto it = sTrackingMap.find(uid); 439 if (it != sTrackingMap.end()) { 440 return it->second & COUNTING_VALUE_MASK; 441 } 442 return 0; 443 } 444 445 void BpBinder::getCountByUid(Vector<uint32_t>& uids, Vector<uint32_t>& counts) 446 { 447 AutoMutex _l(sTrackingLock); 448 uids.setCapacity(sTrackingMap.size()); 449 counts.setCapacity(sTrackingMap.size()); 450 for (const auto& it : sTrackingMap) { 451 uids.push_back(it.first); 452 counts.push_back(it.second & COUNTING_VALUE_MASK); 453 } 454 } 455 456 void BpBinder::enableCountByUid() { sCountByUidEnabled.store(true); } 457 void BpBinder::disableCountByUid() { sCountByUidEnabled.store(false); } 458 void BpBinder::setCountByUidEnabled(bool enable) { sCountByUidEnabled.store(enable); } 459 460 void BpBinder::setLimitCallback(binder_proxy_limit_callback cb) { 461 AutoMutex _l(sTrackingLock); 462 sLimitCallback = cb; 463 } 464 465 void BpBinder::setBinderProxyCountWatermarks(int high, int low) { 466 AutoMutex _l(sTrackingLock); 467 sBinderProxyCountHighWatermark = high; 468 sBinderProxyCountLowWatermark = low; 469 } 470 471 // --------------------------------------------------------------------------- 472 473 }; // namespace android 474