blob: 5ceb218b8bff3f3cc25039764c65fc33bda5d0d9 [file] [log] [blame]
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -08001/*
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
Mathias Agopianc5b2c0b2009-05-19 19:08:10 -070020#include <binder/BpBinder.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080021
Mathias Agopianc5b2c0b2009-05-19 19:08:10 -070022#include <binder/IPCThreadState.h>
Dianne Hackborn23eb1e22015-10-07 17:35:27 -070023#include <binder/IResultReceiver.h>
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -070024#include <cutils/compiler.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080025#include <utils/Log.h>
26
27#include <stdio.h>
28
Steve Block6807e592011-10-20 11:56:00 +010029//#undef ALOGV
30//#define ALOGV(...) fprintf(stderr, __VA_ARGS__)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080031
32namespace android {
33
34// ---------------------------------------------------------------------------
35
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -070036Mutex BpBinder::sTrackingLock;
37std::unordered_map<int32_t,uint32_t> BpBinder::sTrackingMap;
38int BpBinder::sNumTrackedUids = 0;
39std::atomic_bool BpBinder::sCountByUidEnabled(false);
40binder_proxy_limit_callback BpBinder::sLimitCallback;
41bool BpBinder::sBinderProxyThrottleCreate = false;
42
43// Arbitrarily high value that probably distinguishes a bad behaving app
44uint32_t BpBinder::sBinderProxyCountHighWatermark = 2500;
45// Another arbitrary value a binder count needs to drop below before another callback will be called
46uint32_t BpBinder::sBinderProxyCountLowWatermark = 2000;
47
48enum {
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -070049 LIMIT_REACHED_MASK = 0x80000000, // A flag denoting that the limit has been reached
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -070050 COUNTING_VALUE_MASK = 0x7FFFFFFF, // A mask of the remaining bits for the count value
51};
52
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080053BpBinder::ObjectManager::ObjectManager()
54{
55}
56
57BpBinder::ObjectManager::~ObjectManager()
58{
59 kill();
60}
61
62void 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) {
Steve Blocke6f43dd2012-01-06 19:20:56 +000072 ALOGE("Trying to attach object ID %p to binder ObjectManager %p with object %p, but object ID already in use",
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080073 objectID, this, object);
74 return;
75 }
76
77 mObjects.add(objectID, e);
78}
79
80void* BpBinder::ObjectManager::find(const void* objectID) const
81{
82 const ssize_t i = mObjects.indexOfKey(objectID);
Yi Kongfdd8da92018-06-07 17:52:27 -070083 if (i < 0) return nullptr;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080084 return mObjects.valueAt(i).object;
85}
86
87void BpBinder::ObjectManager::detach(const void* objectID)
88{
89 mObjects.removeItem(objectID);
90}
91
92void BpBinder::ObjectManager::kill()
93{
94 const size_t N = mObjects.size();
Mark Salyzynd4ecccf2014-05-30 16:35:57 -070095 ALOGV("Killing %zu objects in manager %p", N, this);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080096 for (size_t i=0; i<N; i++) {
97 const entry_t& e = mObjects.valueAt(i);
Yi Kongfdd8da92018-06-07 17:52:27 -070098 if (e.func != nullptr) {
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080099 e.func(mObjects.keyAt(i), e.object, e.cleanupCookie);
100 }
101 }
102
103 mObjects.clear();
104}
105
106// ---------------------------------------------------------------------------
107
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700108
109BpBinder* BpBinder::create(int32_t handle) {
110 int32_t trackedUid = -1;
111 if (sCountByUidEnabled) {
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700112 trackedUid = IPCThreadState::self()->getCallingUid();
113 AutoMutex _l(sTrackingLock);
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700114 uint32_t trackedValue = sTrackingMap[trackedUid];
115 if (CC_UNLIKELY(trackedValue & LIMIT_REACHED_MASK)) {
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700116 if (sBinderProxyThrottleCreate) {
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700117 return nullptr;
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700118 }
119 } else {
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700120 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 }
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700132 }
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700133 sTrackingMap[trackedUid]++;
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700134 }
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700135 return new BpBinder(handle, trackedUid);
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700136}
137
138BpBinder::BpBinder(int32_t handle, int32_t trackedUid)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800139 : mHandle(handle)
140 , mAlive(1)
141 , mObitsSent(0)
Yi Kongfdd8da92018-06-07 17:52:27 -0700142 , mObituaries(nullptr)
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700143 , mTrackedUid(trackedUid)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800144{
Steve Block6807e592011-10-20 11:56:00 +0100145 ALOGV("Creating BpBinder %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800146
147 extendObjectLifetime(OBJECT_LIFETIME_WEAK);
Martijn Coenen7c170bb2018-05-04 17:28:55 -0700148 IPCThreadState::self()->incWeakHandle(handle, this);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800149}
150
Steven Moreland85180c02019-07-16 14:24:20 -0700151int32_t BpBinder::handle() const {
152 return mHandle;
153}
154
Mathias Agopian83c04462009-05-22 19:00:22 -0700155bool BpBinder::isDescriptorCached() const {
156 Mutex::Autolock _l(mLock);
157 return mDescriptorCache.size() ? true : false;
158}
159
160const String16& BpBinder::getInterfaceDescriptor() const
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800161{
Mathias Agopian83c04462009-05-22 19:00:22 -0700162 if (isDescriptorCached() == false) {
163 Parcel send, reply;
164 // do the IPC without a lock held.
165 status_t err = const_cast<BpBinder*>(this)->transact(
166 INTERFACE_TRANSACTION, send, &reply);
167 if (err == NO_ERROR) {
168 String16 res(reply.readString16());
169 Mutex::Autolock _l(mLock);
170 // mDescriptorCache could have been assigned while the lock was
171 // released.
172 if (mDescriptorCache.size() == 0)
173 mDescriptorCache = res;
174 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800175 }
Mark Salyzynd4ecccf2014-05-30 16:35:57 -0700176
Mathias Agopian83c04462009-05-22 19:00:22 -0700177 // we're returning a reference to a non-static object here. Usually this
Mark Salyzynd4ecccf2014-05-30 16:35:57 -0700178 // is not something smart to do, however, with binder objects it is
Mathias Agopian83c04462009-05-22 19:00:22 -0700179 // (usually) safe because they are reference-counted.
Mark Salyzynd4ecccf2014-05-30 16:35:57 -0700180
Mathias Agopian83c04462009-05-22 19:00:22 -0700181 return mDescriptorCache;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800182}
183
184bool BpBinder::isBinderAlive() const
185{
186 return mAlive != 0;
187}
188
189status_t BpBinder::pingBinder()
190{
191 Parcel send;
192 Parcel reply;
Steven Moreland6e69d652019-07-10 14:17:55 -0700193 return transact(PING_TRANSACTION, send, &reply);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800194}
195
196status_t BpBinder::dump(int fd, const Vector<String16>& args)
197{
198 Parcel send;
199 Parcel reply;
200 send.writeFileDescriptor(fd);
201 const size_t numArgs = args.size();
202 send.writeInt32(numArgs);
203 for (size_t i = 0; i < numArgs; i++) {
204 send.writeString16(args[i]);
205 }
206 status_t err = transact(DUMP_TRANSACTION, send, &reply);
207 return err;
208}
209
Jiyong Parkb86c8662018-10-29 23:01:57 +0900210// NOLINTNEXTLINE(google-default-arguments)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800211status_t BpBinder::transact(
212 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
213{
214 // Once a binder has died, it will never come back to life.
215 if (mAlive) {
216 status_t status = IPCThreadState::self()->transact(
217 mHandle, code, data, reply, flags);
218 if (status == DEAD_OBJECT) mAlive = 0;
219 return status;
220 }
221
222 return DEAD_OBJECT;
223}
224
Jiyong Parkb86c8662018-10-29 23:01:57 +0900225// NOLINTNEXTLINE(google-default-arguments)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800226status_t BpBinder::linkToDeath(
227 const sp<DeathRecipient>& recipient, void* cookie, uint32_t flags)
228{
229 Obituary ob;
230 ob.recipient = recipient;
231 ob.cookie = cookie;
232 ob.flags = flags;
233
Yi Kongfdd8da92018-06-07 17:52:27 -0700234 LOG_ALWAYS_FATAL_IF(recipient == nullptr,
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800235 "linkToDeath(): recipient must be non-NULL");
236
237 {
238 AutoMutex _l(mLock);
239
240 if (!mObitsSent) {
241 if (!mObituaries) {
242 mObituaries = new Vector<Obituary>;
243 if (!mObituaries) {
244 return NO_MEMORY;
245 }
Steve Block6807e592011-10-20 11:56:00 +0100246 ALOGV("Requesting death notification: %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800247 getWeakRefs()->incWeak(this);
248 IPCThreadState* self = IPCThreadState::self();
249 self->requestDeathNotification(mHandle, this);
250 self->flushCommands();
251 }
252 ssize_t res = mObituaries->add(ob);
253 return res >= (ssize_t)NO_ERROR ? (status_t)NO_ERROR : res;
254 }
255 }
256
257 return DEAD_OBJECT;
258}
259
Jiyong Parkb86c8662018-10-29 23:01:57 +0900260// NOLINTNEXTLINE(google-default-arguments)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800261status_t BpBinder::unlinkToDeath(
262 const wp<DeathRecipient>& recipient, void* cookie, uint32_t flags,
263 wp<DeathRecipient>* outRecipient)
264{
265 AutoMutex _l(mLock);
266
267 if (mObitsSent) {
268 return DEAD_OBJECT;
269 }
270
271 const size_t N = mObituaries ? mObituaries->size() : 0;
272 for (size_t i=0; i<N; i++) {
273 const Obituary& obit = mObituaries->itemAt(i);
274 if ((obit.recipient == recipient
Yi Kongfdd8da92018-06-07 17:52:27 -0700275 || (recipient == nullptr && obit.cookie == cookie))
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800276 && obit.flags == flags) {
Yi Kongfdd8da92018-06-07 17:52:27 -0700277 if (outRecipient != nullptr) {
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800278 *outRecipient = mObituaries->itemAt(i).recipient;
279 }
280 mObituaries->removeAt(i);
281 if (mObituaries->size() == 0) {
Steve Block6807e592011-10-20 11:56:00 +0100282 ALOGV("Clearing death notification: %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800283 IPCThreadState* self = IPCThreadState::self();
284 self->clearDeathNotification(mHandle, this);
285 self->flushCommands();
286 delete mObituaries;
Yi Kongfdd8da92018-06-07 17:52:27 -0700287 mObituaries = nullptr;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800288 }
289 return NO_ERROR;
290 }
291 }
292
293 return NAME_NOT_FOUND;
294}
295
296void BpBinder::sendObituary()
297{
Steve Block6807e592011-10-20 11:56:00 +0100298 ALOGV("Sending obituary for proxy %p handle %d, mObitsSent=%s\n",
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800299 this, mHandle, mObitsSent ? "true" : "false");
300
301 mAlive = 0;
302 if (mObitsSent) return;
303
304 mLock.lock();
305 Vector<Obituary>* obits = mObituaries;
Yi Kongfdd8da92018-06-07 17:52:27 -0700306 if(obits != nullptr) {
Steve Block6807e592011-10-20 11:56:00 +0100307 ALOGV("Clearing sent death notification: %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800308 IPCThreadState* self = IPCThreadState::self();
309 self->clearDeathNotification(mHandle, this);
310 self->flushCommands();
Yi Kongfdd8da92018-06-07 17:52:27 -0700311 mObituaries = nullptr;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800312 }
313 mObitsSent = 1;
314 mLock.unlock();
315
Mark Salyzynd4ecccf2014-05-30 16:35:57 -0700316 ALOGV("Reporting death of proxy %p for %zu recipients\n",
317 this, obits ? obits->size() : 0U);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800318
Yi Kongfdd8da92018-06-07 17:52:27 -0700319 if (obits != nullptr) {
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800320 const size_t N = obits->size();
321 for (size_t i=0; i<N; i++) {
322 reportOneDeath(obits->itemAt(i));
323 }
324
325 delete obits;
326 }
327}
328
329void BpBinder::reportOneDeath(const Obituary& obit)
330{
331 sp<DeathRecipient> recipient = obit.recipient.promote();
Steve Block6807e592011-10-20 11:56:00 +0100332 ALOGV("Reporting death to recipient: %p\n", recipient.get());
Yi Kongfdd8da92018-06-07 17:52:27 -0700333 if (recipient == nullptr) return;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800334
335 recipient->binderDied(this);
336}
337
338
339void BpBinder::attachObject(
340 const void* objectID, void* object, void* cleanupCookie,
341 object_cleanup_func func)
342{
343 AutoMutex _l(mLock);
Steve Block6807e592011-10-20 11:56:00 +0100344 ALOGV("Attaching object %p to binder %p (manager=%p)", object, this, &mObjects);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800345 mObjects.attach(objectID, object, cleanupCookie, func);
346}
347
348void* BpBinder::findObject(const void* objectID) const
349{
350 AutoMutex _l(mLock);
351 return mObjects.find(objectID);
352}
353
354void BpBinder::detachObject(const void* objectID)
355{
356 AutoMutex _l(mLock);
357 mObjects.detach(objectID);
358}
359
360BpBinder* BpBinder::remoteBinder()
361{
362 return this;
363}
364
365BpBinder::~BpBinder()
366{
Steve Block6807e592011-10-20 11:56:00 +0100367 ALOGV("Destroying BpBinder %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800368
369 IPCThreadState* ipc = IPCThreadState::self();
370
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700371 if (mTrackedUid >= 0) {
372 AutoMutex _l(sTrackingLock);
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700373 uint32_t trackedValue = sTrackingMap[mTrackedUid];
374 if (CC_UNLIKELY((trackedValue & COUNTING_VALUE_MASK) == 0)) {
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700375 ALOGE("Unexpected Binder Proxy tracking decrement in %p handle %d\n", this, mHandle);
376 } else {
377 if (CC_UNLIKELY(
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700378 (trackedValue & LIMIT_REACHED_MASK) &&
379 ((trackedValue & COUNTING_VALUE_MASK) <= sBinderProxyCountLowWatermark)
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700380 )) {
Michael Wachenschwanz74d967a2018-05-15 15:03:57 -0700381 ALOGI("Limit reached bit reset for uid %d (fewer than %d proxies from uid %d held)",
382 getuid(), mTrackedUid, sBinderProxyCountLowWatermark);
383 sTrackingMap[mTrackedUid] &= ~LIMIT_REACHED_MASK;
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700384 }
385 if (--sTrackingMap[mTrackedUid] == 0) {
386 sTrackingMap.erase(mTrackedUid);
387 }
388 }
389 }
390
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800391 if (ipc) {
392 ipc->expungeHandle(mHandle, this);
393 ipc->decWeakHandle(mHandle);
394 }
395}
396
397void BpBinder::onFirstRef()
398{
Steve Block6807e592011-10-20 11:56:00 +0100399 ALOGV("onFirstRef BpBinder %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800400 IPCThreadState* ipc = IPCThreadState::self();
Martijn Coenen7c170bb2018-05-04 17:28:55 -0700401 if (ipc) ipc->incStrongHandle(mHandle, this);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800402}
403
Colin Cross6f4f3ab2014-02-05 17:42:44 -0800404void BpBinder::onLastStrongRef(const void* /*id*/)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800405{
Steve Block6807e592011-10-20 11:56:00 +0100406 ALOGV("onLastStrongRef BpBinder %p handle %d\n", this, mHandle);
407 IF_ALOGV() {
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800408 printRefs();
409 }
410 IPCThreadState* ipc = IPCThreadState::self();
411 if (ipc) ipc->decStrongHandle(mHandle);
Steven Moreland80d23932019-06-07 12:43:27 -0700412
413 mLock.lock();
414 Vector<Obituary>* obits = mObituaries;
415 if(obits != nullptr) {
416 if (!obits->isEmpty()) {
417 ALOGI("onLastStrongRef automatically unlinking death recipients");
418 }
419
420 if (ipc) ipc->clearDeathNotification(mHandle, this);
421 mObituaries = nullptr;
422 }
423 mLock.unlock();
424
425 if (obits != nullptr) {
426 // XXX Should we tell any remaining DeathRecipient
427 // objects that the last strong ref has gone away, so they
428 // are no longer linked?
429 delete obits;
430 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800431}
432
Colin Cross6f4f3ab2014-02-05 17:42:44 -0800433bool BpBinder::onIncStrongAttempted(uint32_t /*flags*/, const void* /*id*/)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800434{
Steve Block6807e592011-10-20 11:56:00 +0100435 ALOGV("onIncStrongAttempted BpBinder %p handle %d\n", this, mHandle);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800436 IPCThreadState* ipc = IPCThreadState::self();
437 return ipc ? ipc->attemptIncStrongHandle(mHandle) == NO_ERROR : false;
438}
439
Michael Wachenschwanzd296d0c2017-08-15 00:57:14 -0700440uint32_t BpBinder::getBinderProxyCount(uint32_t uid)
441{
442 AutoMutex _l(sTrackingLock);
443 auto it = sTrackingMap.find(uid);
444 if (it != sTrackingMap.end()) {
445 return it->second & COUNTING_VALUE_MASK;
446 }
447 return 0;
448}
449
450void BpBinder::getCountByUid(Vector<uint32_t>& uids, Vector<uint32_t>& counts)
451{
452 AutoMutex _l(sTrackingLock);
453 uids.setCapacity(sTrackingMap.size());
454 counts.setCapacity(sTrackingMap.size());
455 for (const auto& it : sTrackingMap) {
456 uids.push_back(it.first);
457 counts.push_back(it.second & COUNTING_VALUE_MASK);
458 }
459}
460
461void BpBinder::enableCountByUid() { sCountByUidEnabled.store(true); }
462void BpBinder::disableCountByUid() { sCountByUidEnabled.store(false); }
463void BpBinder::setCountByUidEnabled(bool enable) { sCountByUidEnabled.store(enable); }
464
465void BpBinder::setLimitCallback(binder_proxy_limit_callback cb) {
466 AutoMutex _l(sTrackingLock);
467 sLimitCallback = cb;
468}
469
470void BpBinder::setBinderProxyCountWatermarks(int high, int low) {
471 AutoMutex _l(sTrackingLock);
472 sBinderProxyCountHighWatermark = high;
473 sBinderProxyCountLowWatermark = low;
474}
475
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800476// ---------------------------------------------------------------------------
477
478}; // namespace android