You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

177 lines
6.0 KiB

/*
* 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.
*/
#include <utils/SystemClock.h>
#include "OccupantAwareness.h"
namespace android {
namespace hardware {
namespace automotive {
namespace occupant_awareness {
namespace V1_0 {
namespace implementation {
using ndk::ScopedAStatus;
static const int32_t kAllCapabilities = OccupantAwareness::CAP_PRESENCE_DETECTION |
OccupantAwareness::CAP_GAZE_DETECTION |
OccupantAwareness::CAP_DRIVER_MONITORING_DETECTION;
constexpr int64_t kNanoSecondsPerMilliSecond = 1000 * 1000;
ScopedAStatus OccupantAwareness::startDetection(OccupantAwarenessStatus* status) {
std::lock_guard<std::mutex> lock(mMutex);
if (mStatus != OccupantAwarenessStatus::NOT_INITIALIZED) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
mStatus = OccupantAwarenessStatus::READY;
mWorkerThread = std::thread(startWorkerThread, this);
if (mCallback) {
mCallback->onSystemStatusChanged(kAllCapabilities, mStatus);
}
*status = mStatus;
return ScopedAStatus::ok();
}
ScopedAStatus OccupantAwareness::stopDetection(OccupantAwarenessStatus* status) {
std::lock_guard<std::mutex> lock(mMutex);
if (mStatus != OccupantAwarenessStatus::READY) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
mStatus = OccupantAwarenessStatus::NOT_INITIALIZED;
mWorkerThread.join();
if (mCallback) {
mCallback->onSystemStatusChanged(kAllCapabilities, mStatus);
}
*status = mStatus;
return ScopedAStatus::ok();
}
ScopedAStatus OccupantAwareness::getCapabilityForRole(Role occupantRole, int32_t* capabilities) {
if (!isValidRole(occupantRole)) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
int intVal = static_cast<int>(occupantRole);
if ((intVal & DetectionGenerator::getSupportedRoles()) == intVal) {
int capabilities_ = DetectionGenerator::getSupportedCapabilities();
if (occupantRole != Role::DRIVER) {
capabilities_ &= ~CAP_DRIVER_MONITORING_DETECTION;
}
*capabilities = capabilities_;
} else {
*capabilities = 0;
}
return ScopedAStatus::ok();
}
ScopedAStatus OccupantAwareness::getState(Role occupantRole, int detectionCapability,
OccupantAwarenessStatus* status) {
if (!isValidRole(occupantRole)) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
if (!isValidDetectionCapabilities(detectionCapability) ||
!isSingularCapability(detectionCapability)) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
int roleVal = static_cast<int>(occupantRole);
if (((roleVal & DetectionGenerator::getSupportedRoles()) != roleVal) ||
((detectionCapability & DetectionGenerator::getSupportedCapabilities()) !=
detectionCapability)) {
*status = OccupantAwarenessStatus::NOT_SUPPORTED;
return ScopedAStatus::ok();
}
std::lock_guard<std::mutex> lock(mMutex);
*status = mStatus;
return ScopedAStatus::ok();
}
ScopedAStatus OccupantAwareness::setCallback(
const std::shared_ptr<IOccupantAwarenessClientCallback>& callback) {
if (callback == nullptr) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
std::lock_guard<std::mutex> lock(mMutex);
mCallback = callback;
return ScopedAStatus::ok();
}
ScopedAStatus OccupantAwareness::getLatestDetection(OccupantDetections* detections) {
std::lock_guard<std::mutex> lock(mMutex);
if (mStatus != OccupantAwarenessStatus::READY) {
return ScopedAStatus::fromExceptionCode(EX_TRANSACTION_FAILED);
}
*detections = mLatestDetections;
return ScopedAStatus::ok();
}
bool OccupantAwareness::isValidRole(Role occupantRole) {
int intVal = static_cast<int>(occupantRole);
int allOccupants = static_cast<int>(Role::ALL_OCCUPANTS);
return (occupantRole != Role::INVALID) && ((intVal & (~allOccupants)) == 0);
}
bool OccupantAwareness::isValidDetectionCapabilities(int detectionCapabilities) {
return (detectionCapabilities != OccupantAwareness::CAP_NONE) &&
((detectionCapabilities & (~kAllCapabilities)) == 0);
}
bool OccupantAwareness::isSingularCapability(int detectionCapability) {
// Check whether the value is 0, or the value has only one bit set.
return (detectionCapability & (detectionCapability - 1)) == 0;
}
void OccupantAwareness::startWorkerThread(OccupantAwareness* occupantAwareness) {
occupantAwareness->workerThreadFunction();
}
void OccupantAwareness::workerThreadFunction() {
bool isFirstDetection = true;
int64_t prevDetectionTimeMs;
while (mStatus == OccupantAwarenessStatus::READY) {
int64_t currentTimeMs = android::elapsedRealtimeNano() / kNanoSecondsPerMilliSecond;
if ((isFirstDetection) || (currentTimeMs - prevDetectionTimeMs > mDetectionDurationMs)) {
std::lock_guard<std::mutex> lock(mMutex);
mLatestDetections = mGenerator.GetNextDetections();
if (mCallback != nullptr) {
mCallback->onDetectionEvent(mLatestDetections);
}
isFirstDetection = false;
prevDetectionTimeMs = currentTimeMs;
}
}
}
} // namespace implementation
} // namespace V1_0
} // namespace occupant_awareness
} // namespace automotive
} // namespace hardware
} // namespace android