/*
* Copyright (C) 2007 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.os;
import android.util.Log;
import java.io.PrintWriter;
import java.util.ArrayList;
import java.util.Set;
import java.util.WeakHashMap;
/**
* A TokenWatcher watches a collection of {@link IBinder}s. IBinders are added
* to the collection by calling {@link #acquire}, and removed by calling {@link
* #release}. IBinders are also implicitly removed when they become weakly
* reachable. Each IBinder may be added at most once.
*
* The {@link #acquired} method is invoked by posting to the specified handler
* whenever the size of the watched collection becomes nonzero. The {@link
* #released} method is invoked on the specified handler whenever the size of
* the watched collection becomes zero.
*/
public abstract class TokenWatcher
{
/**
* Construct the TokenWatcher
*
* @param h A handler to call {@link #acquired} and {@link #released}
* on. If you don't care, just call it like this, although your thread
* will have to be a Looper thread.
* new TokenWatcher(new Handler())
* @param tag A debugging tag for this TokenWatcher
*/
public TokenWatcher(Handler h, String tag)
{
mHandler = h;
mTag = tag != null ? tag : "TokenWatcher";
}
/**
* Called when the number of active tokens goes from 0 to 1.
*/
public abstract void acquired();
/**
* Called when the number of active tokens goes from 1 to 0.
*/
public abstract void released();
/**
* Record that this token has been acquired. When acquire is called, and
* the current count is 0, the acquired method is called on the given
* handler.
*
* Note that the same {@code token} can only be acquired once. If this
* {@code token} has already been acquired, no action is taken. The first
* subsequent call to {@link #release} will release this {@code token}
* immediately.
*
* @param token An IBinder object.
* @param tag A string used by the {@link #dump} method for debugging,
* to see who has references.
*/
public void acquire(IBinder token, String tag)
{
synchronized (mTokens) {
if (mTokens.containsKey(token)) {
return;
}
// explicitly checked to avoid bogus sendNotification calls because
// of the WeakHashMap and the GC
int oldSize = mTokens.size();
Death d = new Death(token, tag);
try {
token.linkToDeath(d, 0);
} catch (RemoteException e) {
return;
}
mTokens.put(token, d);
if (oldSize == 0 && !mAcquired) {
sendNotificationLocked(true);
mAcquired = true;
}
}
}
public void cleanup(IBinder token, boolean unlink)
{
synchronized (mTokens) {
Death d = mTokens.remove(token);
if (unlink && d != null) {
d.token.unlinkToDeath(d, 0);
d.token = null;
}
if (mTokens.size() == 0 && mAcquired) {
sendNotificationLocked(false);
mAcquired = false;
}
}
}
public void release(IBinder token)
{
cleanup(token, true);
}
public boolean isAcquired()
{
synchronized (mTokens) {
return mAcquired;
}
}
public void dump()
{
ArrayList a = dumpInternal();
for (String s : a) {
Log.i(mTag, s);
}
}
public void dump(PrintWriter pw) {
ArrayList a = dumpInternal();
for (String s : a) {
pw.println(s);
}
}
private ArrayList dumpInternal() {
ArrayList a = new ArrayList();
synchronized (mTokens) {
Set keys = mTokens.keySet();
a.add("Token count: " + mTokens.size());
int i = 0;
for (IBinder b: keys) {
a.add("[" + i + "] " + mTokens.get(b).tag + " - " + b);
i++;
}
}
return a;
}
private Runnable mNotificationTask = new Runnable() {
public void run()
{
int value;
synchronized (mTokens) {
value = mNotificationQueue;
mNotificationQueue = -1;
}
if (value == 1) {
acquired();
}
else if (value == 0) {
released();
}
}
};
private void sendNotificationLocked(boolean on)
{
int value = on ? 1 : 0;
if (mNotificationQueue == -1) {
// empty
mNotificationQueue = value;
mHandler.post(mNotificationTask);
}
else if (mNotificationQueue != value) {
// it's a pair, so cancel it
mNotificationQueue = -1;
mHandler.removeCallbacks(mNotificationTask);
}
// else, same so do nothing -- maybe we should warn?
}
private class Death implements IBinder.DeathRecipient
{
IBinder token;
String tag;
Death(IBinder token, String tag)
{
this.token = token;
this.tag = tag;
}
public void binderDied()
{
cleanup(token, false);
}
protected void finalize() throws Throwable
{
try {
if (token != null) {
Log.w(mTag, "cleaning up leaked reference: " + tag);
release(token);
}
}
finally {
super.finalize();
}
}
}
private WeakHashMap mTokens = new WeakHashMap();
private Handler mHandler;
private String mTag;
private int mNotificationQueue = -1;
private volatile boolean mAcquired = false;
}