Fix for force closing HDP app: Included review comments
Included a callback to linkToDeath and notify the HealthService when BluetoothHealth is dead
Change-Id: Ie34a391f78c4fba0a74bcefd3dedccaa63f7bba3
diff --git a/src/com/android/bluetooth/hdp/HealthService.java b/src/com/android/bluetooth/hdp/HealthService.java
index 9ab65ea..4f2d20a 100755
--- a/src/com/android/bluetooth/hdp/HealthService.java
+++ b/src/com/android/bluetooth/hdp/HealthService.java
@@ -16,8 +16,11 @@
import android.os.HandlerThread;
import android.os.Looper;
import android.os.Message;
+import android.os.IBinder;
+import android.os.IBinder.DeathRecipient;
import android.os.ParcelFileDescriptor;
import android.os.RemoteException;
+import java.util.NoSuchElementException;
import android.os.ServiceManager;
import android.util.Log;
import java.io.FileDescriptor;
@@ -87,9 +90,21 @@
if (looper != null) {
looper.quit();
}
+ cleanupApps();
return true;
}
+ private void cleanupApps(){
+ for (Entry<BluetoothHealthAppConfiguration, AppInfo> entry : mApps.entrySet()) {
+ try{
+ AppInfo appInfo = entry.getValue();
+ appInfo.cleanup();
+ mApps.remove(entry.getKey());
+ }catch(Exception e){
+ Log.e(TAG,"Failed to cleanup appInfo for appid ="+entry.getKey());
+ }
+ }
+ }
protected boolean cleanup() {
mHandler = null;
//Cleanup native
@@ -124,6 +139,8 @@
{
BluetoothHealthAppConfiguration appConfig =
(BluetoothHealthAppConfiguration) msg.obj;
+ AppInfo appInfo = mApps.get(appConfig);
+ if (appInfo == null) break;
int halRole = convertRoleToHal(appConfig.getRole());
int halChannelType = convertChannelTypeToHal(appConfig.getChannelType());
if (DBG) log("register datatype: " + appConfig.getDataType() + " role: " +
@@ -131,13 +148,21 @@
halChannelType);
int appId = registerHealthAppNative(appConfig.getDataType(), halRole,
appConfig.getName(), halChannelType);
-
if (appId == -1) {
callStatusCallback(appConfig,
BluetoothHealth.APP_CONFIG_REGISTRATION_FAILURE);
+ appInfo.cleanup();
mApps.remove(appConfig);
} else {
- (mApps.get(appConfig)).mAppId = appId;
+ //link to death with a recipient object to implement binderDead()
+ appInfo.mRcpObj = new BluetoothHealthDeathRecipient(HealthService.this,appConfig);
+ IBinder binder = appInfo.mCallback.asBinder();
+ try {
+ binder.linkToDeath(appInfo.mRcpObj,0);
+ } catch (RemoteException e) {
+ Log.e(TAG,"LinktoDeath Exception:"+e);
+ }
+ appInfo.mAppId = appId;
callStatusCallback(appConfig,
BluetoothHealth.APP_CONFIG_REGISTRATION_SUCCESS);
}
@@ -197,6 +222,9 @@
callStatusCallback(appConfig, regStatus);
if (regStatus == BluetoothHealth.APP_CONFIG_REGISTRATION_FAILURE ||
regStatus == BluetoothHealth.APP_CONFIG_UNREGISTRATION_SUCCESS) {
+ //unlink to death once app is unregistered
+ AppInfo appInfo = mApps.get(appConfig);
+ appInfo.cleanup();
mApps.remove(appConfig);
}
}
@@ -236,6 +264,27 @@
}
}
+//Handler for DeathReceipient
+ private static class BluetoothHealthDeathRecipient implements IBinder.DeathRecipient{
+ private BluetoothHealthAppConfiguration mConfig;
+ private HealthService mService;
+
+ public BluetoothHealthDeathRecipient(HealthService service, BluetoothHealthAppConfiguration config) {
+ mService = service;
+ mConfig = config;
+ }
+
+ public void binderDied() {
+ Log.d(TAG,"Binder is dead.");
+ mService.unregisterAppConfiguration(mConfig);
+ }
+
+ public void cleanup(){
+ mService = null;
+ mConfig = null;
+ }
+ }
+
/**
* Handlers for incoming service calls
*/
@@ -338,7 +387,6 @@
if (DBG) Log.d(TAG,"unregisterAppConfiguration: no app found");
return false;
}
-
Message msg = mHandler.obtainMessage(MESSAGE_UNREGISTER_APPLICATION,config);
mHandler.sendMessage(msg);
return true;
@@ -709,12 +757,34 @@
private static class AppInfo {
private IBluetoothHealthCallback mCallback;
+ private BluetoothHealthDeathRecipient mRcpObj;
private int mAppId;
private AppInfo(IBluetoothHealthCallback callback) {
mCallback = callback;
+ mRcpObj = null;
mAppId = -1;
}
+
+ private void cleanup(){
+ if(mCallback != null){
+ if(mRcpObj != null){
+ IBinder binder = mCallback.asBinder();
+ try{
+ binder.unlinkToDeath(mRcpObj,0);
+ }catch(NoSuchElementException e){
+ Log.e(TAG,"No death recipient registered"+e);
+ }
+ mRcpObj.cleanup();
+ mRcpObj = null;
+ }
+ mCallback = null;
+ }
+ else if(mRcpObj != null){
+ mRcpObj.cleanup();
+ mRcpObj = null;
+ }
+ }
}
private class HealthChannel {