From ec33b82c42f3dd996a72b80359c87524b51ffb2a Mon Sep 17 00:00:00 2001 From: Mathias Agopian Date: Tue, 14 Sep 2010 16:44:59 -0700 Subject: fix [2931909] SensorManger breaks when using different SensorEventListener w/ the same sensors - Sensor handle can now be arbitrary instead of being limited to 0-31 - make sure to disable sensors only when no listener uses them anymore Change-Id: If656c8c24d9827a7db1de3b3aa89961abb7def0c --- core/java/android/hardware/SensorManager.java | 87 ++++++++++++++++++--------- 1 file changed, 58 insertions(+), 29 deletions(-) (limited to 'core/java/android/hardware') diff --git a/core/java/android/hardware/SensorManager.java b/core/java/android/hardware/SensorManager.java index 0068724..76ce2f8 100644 --- a/core/java/android/hardware/SensorManager.java +++ b/core/java/android/hardware/SensorManager.java @@ -24,6 +24,7 @@ import android.os.Message; import android.os.ServiceManager; import android.util.Log; import android.util.SparseArray; +import android.util.SparseBooleanArray; import android.view.IRotationWatcher; import android.view.IWindowManager; import android.view.Surface; @@ -487,7 +488,7 @@ public class SensorManager private final ArrayList mSensorList = new ArrayList(); private final Handler mHandler; private SensorEvent mValuesPool; - public int mSensors; + public SparseBooleanArray mSensors = new SparseBooleanArray(); ListenerDelegate(SensorEventListener listener, Sensor sensor, Handler handler) { mSensorEventListener = listener; @@ -541,18 +542,17 @@ public class SensorManager return mSensorEventListener; } - int addSensor(Sensor sensor) { - mSensors |= 1< getSensors() { return mSensorList; @@ -971,6 +971,31 @@ public class SensorManager return registerListener(listener, sensor, rate, null); } + private boolean enableSensorLocked(Sensor sensor, int delay) { + boolean result = false; + for (ListenerDelegate i : sListeners) { + if (i.hasSensor(sensor)) { + String name = sensor.getName(); + int handle = sensor.getHandle(); + result = sensors_enable_sensor(sQueue, name, handle, delay); + break; + } + } + return result; + } + + private boolean disableSensorLocked(Sensor sensor) { + for (ListenerDelegate i : sListeners) { + if (i.hasSensor(sensor)) { + // not an error, it's just that this sensor is still in use + return true; + } + } + String name = sensor.getName(); + int handle = sensor.getHandle(); + return sensors_enable_sensor(sQueue, name, handle, SENSOR_DISABLE); + } + /** * Registers a {@link android.hardware.SensorEventListener * SensorEventListener} for the given sensor. @@ -1008,7 +1033,7 @@ public class SensorManager if (listener == null || sensor == null) { return false; } - boolean result; + boolean result = true; int delay = -1; switch (rate) { case SENSOR_DELAY_FASTEST: @@ -1029,6 +1054,7 @@ public class SensorManager } synchronized (sListeners) { + // look for this listener in our list ListenerDelegate l = null; for (ListenerDelegate i : sListeners) { if (i.getListener() == listener) { @@ -1037,29 +1063,37 @@ public class SensorManager } } - String name = sensor.getName(); - int handle = sensor.getHandle(); + // if we don't find it, add it to the list if (l == null) { - result = false; l = new ListenerDelegate(listener, sensor, handler); sListeners.add(l); + // if the list is not empty, start our main thread if (!sListeners.isEmpty()) { - result = sSensorThread.startLocked(); - if (result) { - result = sensors_enable_sensor(sQueue, name, handle, delay); - if (!result) { - // there was an error, remove the listeners + if (sSensorThread.startLocked()) { + if (!enableSensorLocked(sensor, delay)) { + // oops. there was an error sListeners.remove(l); + result = false; } + } else { + // there was an error, remove the listener + sListeners.remove(l); + result = false; } + } else { + // weird, we couldn't add the listener + result = false; } } else { - result = sensors_enable_sensor(sQueue, name, handle, delay); - if (result) { - l.addSensor(sensor); + l.addSensor(sensor); + if (!enableSensorLocked(sensor, delay)) { + // oops. there was an error + l.removeSensor(sensor); + result = false; } } } + return result; } @@ -1072,18 +1106,15 @@ public class SensorManager for (int i=0 ; i