/* * Copyright (C) 2021 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. */ #include "BaseSensorObject.h" #include "DynamicSensorsSubHal.h" #include #include #include #include #include #include using ::android::hardware::sensors::V1_0::Result; using ::android::hardware::sensors::V2_0::implementation::ScopedWakelock; using ::android::hardware::sensors::V2_1::implementation::convertFromSensorEvent; using ::android::hardware::sensors::V2_1::SensorInfo; using ::android::hardware::sensors::V2_1::SensorType; template using Return = ::android::hardware::Return; using ::android::hardware::Void; namespace dynamic_sensors_flags = com::android::libhardware::dynamic::sensors::flags; namespace android { namespace SensorHalExt { static Result ResultFromStatus(status_t err) { switch (err) { case ::android::OK: return Result::OK; case ::android::PERMISSION_DENIED: return Result::PERMISSION_DENIED; case ::android::NO_MEMORY: return Result::NO_MEMORY; case ::android::BAD_VALUE: return Result::BAD_VALUE; default: return Result::INVALID_OPERATION; } } DynamicSensorsSubHal::DynamicSensorsSubHal() { // initialize dynamic sensor manager mDynamicSensorManager.reset( DynamicSensorManager::createInstance(kDynamicHandleBase, kMaxDynamicHandleCount, this /* callback */)); } // ISensors. Return DynamicSensorsSubHal::setOperationMode(OperationMode mode) { return (mode == static_cast(SENSOR_HAL_NORMAL_MODE) ? Result::OK : Result::BAD_VALUE); } Return DynamicSensorsSubHal::activate(int32_t sensor_handle, bool enabled) { int rc = mDynamicSensorManager->activate(sensor_handle, enabled); return ResultFromStatus(rc); } Return DynamicSensorsSubHal::batch( int32_t sensor_handle, int64_t sampling_period_ns, int64_t max_report_latency_ns) { int rc = mDynamicSensorManager->batch(sensor_handle, sampling_period_ns, max_report_latency_ns); return ResultFromStatus(rc); } Return DynamicSensorsSubHal::flush(int32_t sensor_handle) { int rc = mDynamicSensorManager->flush(sensor_handle); return ResultFromStatus(rc); } Return DynamicSensorsSubHal::registerDirectChannel( const SharedMemInfo& mem __unused, registerDirectChannel_cb callback __unused) { ALOGE("DynamicSensorsSubHal::registerDirectChannel not supported."); return Void(); } Return DynamicSensorsSubHal::unregisterDirectChannel( int32_t channel_handle __unused) { ALOGE("DynamicSensorsSubHal::unregisterDirectChannel not supported."); return Result::INVALID_OPERATION; } Return DynamicSensorsSubHal::configDirectReport( int32_t sensor_handle __unused, int32_t channel_handle __unused, RateLevel rate __unused, configDirectReport_cb callback __unused) { ALOGE("DynamicSensorsSubHal::configDirectReport not supported."); return Void(); } Return DynamicSensorsSubHal::getSensorsList_2_1( getSensorsList_2_1_cb callback) { const sensor_t& sensor_info = mDynamicSensorManager->getDynamicMetaSensor(); std::vector sensors; ALOGD("DynamicSensorsSubHal::getSensorsList_2_1 invoked."); // get the dynamic sensor info sensors.resize(1); sensors[0].sensorHandle = sensor_info.handle; sensors[0].name = sensor_info.name; sensors[0].vendor = sensor_info.vendor; sensors[0].version = sensor_info.version; sensors[0].type = static_cast(sensor_info.type); sensors[0].typeAsString = sensor_info.stringType; sensors[0].maxRange = sensor_info.maxRange; sensors[0].resolution = sensor_info.resolution; sensors[0].power = sensor_info.power; sensors[0].minDelay = sensor_info.minDelay; sensors[0].fifoReservedEventCount = sensor_info.fifoReservedEventCount; sensors[0].fifoMaxEventCount = sensor_info.fifoMaxEventCount; sensors[0].requiredPermission = sensor_info.requiredPermission; sensors[0].maxDelay = sensor_info.maxDelay; sensors[0].flags = sensor_info.flags; callback(sensors); return Void(); } Return DynamicSensorsSubHal::injectSensorData_2_1( const Event& event __unused) { ALOGE("DynamicSensorsSubHal::injectSensorData_2_1 not supported."); return Result::INVALID_OPERATION; } Return DynamicSensorsSubHal::debug( const hidl_handle& handle __unused, const hidl_vec& args __unused) { return Void(); } // ISensorsSubHal. Return DynamicSensorsSubHal::initialize( const sp& hal_proxy_callback) { ALOGD("DynamicSensorsSubHal::initialize invoked."); mHalProxyCallback = hal_proxy_callback; return Result::OK; } // SensorEventCallback. int DynamicSensorsSubHal::submitEvent(SP(BaseSensorObject) sensor, const sensors_event_t& e) { std::vector events; Event hal_event; bool wakeup; bool disconnectDynamicSensorFlag = dynamic_sensors_flags::dynamic_sensors_hal_disconnect_dynamic_sensor(); if (e.type == SENSOR_TYPE_DYNAMIC_SENSOR_META) { const dynamic_sensor_meta_event_t* sensor_meta; sensor_meta = static_cast( &(e.dynamic_sensor_meta)); if (sensor_meta->connected != 0) { onSensorConnected(sensor_meta->handle, sensor_meta->sensor); } else if (disconnectDynamicSensorFlag) { onSensorDisconnected(sensor_meta->handle); } // The sensor framework must be notified of the connected sensor // through the callback before handling the sensor added event. If // it isn't, it will assert when looking up the sensor handle when // processing the sensor added event. // // TODO (b/201529167): Fix dynamic sensors addition / removal when // converting to AIDL. // The sensor framework runs in a separate process from the sensor // HAL, and it processes events in a dedicated thread, so it's // possible the event handling can be done before the callback is // run. Thus, a delay is added after sending notification of the // connected sensor. if (disconnectDynamicSensorFlag || sensor_meta->connected != 0) { std::this_thread::sleep_for(std::chrono::milliseconds(1000)); } } convertFromSensorEvent(e, &hal_event); events.push_back(hal_event); if (sensor && sensor->getSensor()) { wakeup = sensor->getSensor()->flags & SENSOR_FLAG_WAKE_UP; } else { wakeup = false; } ScopedWakelock wakelock = mHalProxyCallback->createScopedWakelock(wakeup); mHalProxyCallback->postEvents(events, std::move(wakelock)); return 0; } void DynamicSensorsSubHal::onSensorConnected( int handle, const sensor_t* sensor_info) { hidl_vec sensor_list; sensor_list.resize(1); sensor_list[0].sensorHandle = handle; sensor_list[0].name = sensor_info->name; sensor_list[0].vendor = sensor_info->vendor; sensor_list[0].version = sensor_info->version; sensor_list[0].type = static_cast(sensor_info->type); sensor_list[0].typeAsString = sensor_info->stringType; sensor_list[0].maxRange = sensor_info->maxRange; sensor_list[0].resolution = sensor_info->resolution; sensor_list[0].power = sensor_info->power; sensor_list[0].minDelay = sensor_info->minDelay; sensor_list[0].fifoReservedEventCount = sensor_info->fifoReservedEventCount; sensor_list[0].fifoMaxEventCount = sensor_info->fifoMaxEventCount; sensor_list[0].requiredPermission = sensor_info->requiredPermission; sensor_list[0].maxDelay = sensor_info->maxDelay; sensor_list[0].flags = sensor_info->flags; mHalProxyCallback->onDynamicSensorsConnected_2_1(sensor_list); } void DynamicSensorsSubHal::onSensorDisconnected(int handle) { hidl_vec handleList; handleList.resize(1); handleList[0] = handle; mHalProxyCallback->onDynamicSensorsDisconnected(handleList); } } // namespace SensorHalExt } // namespace android using ::android::hardware::sensors::V2_1::implementation::ISensorsSubHal; ISensorsSubHal* sensorsHalGetSubHal_2_1(uint32_t* version) { static android::SensorHalExt::DynamicSensorsSubHal subHal; *version = SUB_HAL_2_1_VERSION; return &subHal; }