blob: 31f4991a82b50d3555ec89fcd7bd5e286c2604b3 [file] [log] [blame]
/*
* Copyright (C) 2017 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 com.android.systemui.util;
import android.content.Context;
import android.hardware.HardwareBuffer;
import android.hardware.Sensor;
import android.hardware.SensorAdditionalInfo;
import android.hardware.SensorDirectChannel;
import android.hardware.SensorEventListener;
import android.hardware.SensorManager;
import android.hardware.TriggerEventListener;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.MemoryFile;
import android.util.Log;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.util.Preconditions;
import com.android.systemui.plugins.PluginListener;
import com.android.systemui.plugins.SensorManagerPlugin;
import com.android.systemui.shared.plugins.PluginManager;
import java.util.ArrayList;
import java.util.List;
import javax.inject.Inject;
import javax.inject.Singleton;
/**
* Wrapper around sensor manager that hides potential sources of latency.
*
* Offloads fetching (non-dynamic) sensors and (un)registering listeners onto a background thread
* without blocking. Note that this means registering listeners now always appears successful even
* if it is not.
*/
@Singleton
public class AsyncSensorManager extends SensorManager
implements PluginListener<SensorManagerPlugin> {
private static final String TAG = "AsyncSensorManager";
private final SensorManager mInner;
private final List<Sensor> mSensorCache;
private final HandlerThread mHandlerThread = new HandlerThread("async_sensor");
@VisibleForTesting final Handler mHandler;
private final List<SensorManagerPlugin> mPlugins;
@Inject
public AsyncSensorManager(Context context, PluginManager pluginManager) {
this(context.getSystemService(SensorManager.class), pluginManager);
}
@VisibleForTesting
AsyncSensorManager(SensorManager sensorManager, PluginManager pluginManager) {
mInner = sensorManager;
mHandlerThread.start();
mHandler = new Handler(mHandlerThread.getLooper());
mSensorCache = mInner.getSensorList(Sensor.TYPE_ALL);
mPlugins = new ArrayList<>();
pluginManager.addPluginListener(this, SensorManagerPlugin.class, true /* allowMultiple */);
}
@Override
protected List<Sensor> getFullSensorList() {
return mSensorCache;
}
@Override
protected List<Sensor> getFullDynamicSensorList() {
return mInner.getSensorList(Sensor.TYPE_ALL);
}
@Override
protected boolean registerListenerImpl(SensorEventListener listener,
Sensor sensor, int delayUs, Handler handler, int maxReportLatencyUs,
int reservedFlags) {
mHandler.post(() -> {
if (!mInner.registerListener(listener, sensor, delayUs, maxReportLatencyUs, handler)) {
Log.e(TAG, "Registering " + listener + " for " + sensor + " failed.");
}
});
return true;
}
@Override
protected boolean flushImpl(SensorEventListener listener) {
return mInner.flush(listener);
}
@Override
protected SensorDirectChannel createDirectChannelImpl(MemoryFile memoryFile,
HardwareBuffer hardwareBuffer) {
throw new UnsupportedOperationException("not implemented");
}
@Override
protected void destroyDirectChannelImpl(SensorDirectChannel channel) {
throw new UnsupportedOperationException("not implemented");
}
@Override
protected int configureDirectChannelImpl(SensorDirectChannel channel, Sensor s, int rate) {
throw new UnsupportedOperationException("not implemented");
}
@Override
protected void registerDynamicSensorCallbackImpl(DynamicSensorCallback callback,
Handler handler) {
mHandler.post(() -> mInner.registerDynamicSensorCallback(callback, handler));
}
@Override
protected void unregisterDynamicSensorCallbackImpl(DynamicSensorCallback callback) {
mHandler.post(() -> mInner.unregisterDynamicSensorCallback(callback));
}
@Override
protected boolean requestTriggerSensorImpl(TriggerEventListener listener, Sensor sensor) {
mHandler.post(() -> {
if (!mInner.requestTriggerSensor(listener, sensor)) {
Log.e(TAG, "Requesting " + listener + " for " + sensor + " failed.");
}
});
return true;
}
@Override
protected boolean cancelTriggerSensorImpl(TriggerEventListener listener, Sensor sensor,
boolean disable) {
Preconditions.checkArgument(disable);
mHandler.post(() -> {
if (!mInner.cancelTriggerSensor(listener, sensor)) {
Log.e(TAG, "Canceling " + listener + " for " + sensor + " failed.");
}
});
return true;
}
/**
* Requests for all sensors that match the given type from all plugins.
* @param sensor
* @param listener
*/
public void registerPluginListener(SensorManagerPlugin.Sensor sensor,
SensorManagerPlugin.SensorEventListener listener) {
if (mPlugins.isEmpty()) {
Log.w(TAG, "No plugins registered");
}
mHandler.post(() -> {
for (int i = 0; i < mPlugins.size(); i++) {
mPlugins.get(i).registerListener(sensor, listener);
}
});
}
/**
* Unregisters all sensors that match the give type for all plugins.
* @param sensor
* @param listener
*/
public void unregisterPluginListener(SensorManagerPlugin.Sensor sensor,
SensorManagerPlugin.SensorEventListener listener) {
mHandler.post(() -> {
for (int i = 0; i < mPlugins.size(); i++) {
mPlugins.get(i).unregisterListener(sensor, listener);
}
});
}
@Override
protected boolean initDataInjectionImpl(boolean enable) {
throw new UnsupportedOperationException("not implemented");
}
@Override
protected boolean injectSensorDataImpl(Sensor sensor, float[] values, int accuracy,
long timestamp) {
throw new UnsupportedOperationException("not implemented");
}
@Override
protected boolean setOperationParameterImpl(SensorAdditionalInfo parameter) {
mHandler.post(() -> mInner.setOperationParameter(parameter));
return true;
}
@Override
protected void unregisterListenerImpl(SensorEventListener listener,
Sensor sensor) {
mHandler.post(() -> {
if (sensor == null) {
mInner.unregisterListener(listener);
} else {
mInner.unregisterListener(listener, sensor);
}
});
}
@Override
public void onPluginConnected(SensorManagerPlugin plugin, Context pluginContext) {
mPlugins.add(plugin);
}
@Override
public void onPluginDisconnected(SensorManagerPlugin plugin) {
mPlugins.remove(plugin);
}
}