blob: 48759824f5efbf1d349737bd5bfbb48d478a18a7 [file] [log] [blame]
/*
* Copyright (C) 2019 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.sensors;
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.MemoryFile;
import android.util.Log;
import com.android.internal.util.Preconditions;
import com.android.systemui.dagger.SysUISingleton;
import com.android.systemui.plugins.PluginListener;
import com.android.systemui.plugins.SensorManagerPlugin;
import com.android.systemui.shared.plugins.PluginManager;
import com.android.systemui.util.concurrency.ThreadFactory;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Executor;
import javax.inject.Inject;
/**
* 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.
*/
@SysUISingleton
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 Executor mExecutor;
private final List<SensorManagerPlugin> mPlugins;
@Inject
public AsyncSensorManager(SensorManager sensorManager, ThreadFactory threadFactory,
PluginManager pluginManager) {
mInner = sensorManager;
mExecutor = threadFactory.buildExecutorOnNewThread("async_sensor");
mSensorCache = mInner.getSensorList(Sensor.TYPE_ALL);
mPlugins = new ArrayList<>();
if (pluginManager != null) {
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) {
mExecutor.execute(() -> {
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) {
mExecutor.execute(() -> mInner.registerDynamicSensorCallback(callback, handler));
}
@Override
protected void unregisterDynamicSensorCallbackImpl(DynamicSensorCallback callback) {
mExecutor.execute(() -> mInner.unregisterDynamicSensorCallback(callback));
}
@Override
protected boolean requestTriggerSensorImpl(TriggerEventListener listener, Sensor sensor) {
if (listener == null) {
throw new IllegalArgumentException("listener cannot be null");
}
if (sensor == null) {
throw new IllegalArgumentException("sensor cannot be null");
}
mExecutor.execute(() -> {
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);
mExecutor.execute(() -> {
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
* @return true if there were plugins to register the listener to
*/
public boolean registerPluginListener(SensorManagerPlugin.Sensor sensor,
SensorManagerPlugin.SensorEventListener listener) {
if (mPlugins.isEmpty()) {
Log.w(TAG, "No plugins registered");
return false;
}
mExecutor.execute(() -> {
for (int i = 0; i < mPlugins.size(); i++) {
mPlugins.get(i).registerListener(sensor, listener);
}
});
return true;
}
/**
* Unregisters all sensors that match the give type for all plugins.
* @param sensor
* @param listener
*/
public void unregisterPluginListener(SensorManagerPlugin.Sensor sensor,
SensorManagerPlugin.SensorEventListener listener) {
mExecutor.execute(() -> {
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) {
mExecutor.execute(() -> mInner.setOperationParameter(parameter));
return true;
}
@Override
protected void unregisterListenerImpl(SensorEventListener listener,
Sensor sensor) {
mExecutor.execute(() -> {
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);
}
}