177 lines
6 KiB
C++
177 lines
6 KiB
C++
|
/*
|
||
|
* 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
|