|
|
|
/*
|
|
|
|
* Copyright (C) 2021 The LineageOS Project
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Vibrator.h"
|
|
|
|
|
|
|
|
#include <android-base/logging.h>
|
|
|
|
|
|
|
|
#include <cmath>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <map>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
namespace aidl {
|
|
|
|
namespace android {
|
|
|
|
namespace hardware {
|
|
|
|
namespace vibrator {
|
|
|
|
|
|
|
|
static std::map<Effect, int> CP_TRIGGER_EFFECTS {
|
|
|
|
{ Effect::CLICK, 10 },
|
|
|
|
{ Effect::DOUBLE_CLICK, 14 },
|
|
|
|
{ Effect::HEAVY_CLICK, 23 },
|
|
|
|
{ Effect::TEXTURE_TICK, 50 },
|
|
|
|
{ Effect::TICK, 50 }
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write value to path and close file.
|
|
|
|
*/
|
|
|
|
template <typename T>
|
|
|
|
static ndk::ScopedAStatus writeNode(const std::string& path, const T& value) {
|
|
|
|
std::ofstream node(path);
|
|
|
|
if (!node) {
|
|
|
|
LOG(ERROR) << "Failed to open: " << path;
|
|
|
|
return ndk::ScopedAStatus::fromStatus(STATUS_UNKNOWN_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG(DEBUG) << "writeNode node: " << path << " value: " << value;
|
|
|
|
|
|
|
|
node << value << std::endl;
|
|
|
|
if (!node) {
|
|
|
|
LOG(ERROR) << "Failed to write: " << value;
|
|
|
|
return ndk::ScopedAStatus::fromStatus(STATUS_UNKNOWN_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool nodeExists(const std::string& path) {
|
|
|
|
std::ofstream f(path.c_str());
|
|
|
|
return f.good();
|
|
|
|
}
|
|
|
|
|
|
|
|
Vibrator::Vibrator() {
|
|
|
|
mIsTimedOutVibrator = nodeExists(VIBRATOR_TIMEOUT_PATH);
|
|
|
|
mHasTimedOutIntensity = nodeExists(VIBRATOR_INTENSITY_PATH);
|
|
|
|
mHasTimedOutEffect = nodeExists(VIBRATOR_CP_TRIGGER_PATH);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getCapabilities(int32_t* _aidl_return) {
|
|
|
|
*_aidl_return = IVibrator::CAP_ON_CALLBACK | IVibrator::CAP_PERFORM_CALLBACK |
|
|
|
|
IVibrator::CAP_EXTERNAL_CONTROL /*| IVibrator::CAP_COMPOSE_EFFECTS |
|
|
|
|
IVibrator::CAP_ALWAYS_ON_CONTROL*/;
|
|
|
|
|
|
|
|
if (mHasTimedOutIntensity) {
|
|
|
|
*_aidl_return = *_aidl_return | IVibrator::CAP_AMPLITUDE_CONTROL |
|
|
|
|
IVibrator::CAP_EXTERNAL_AMPLITUDE_CONTROL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::off() {
|
|
|
|
return activate(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::on(int32_t timeoutMs, const std::shared_ptr<IVibratorCallback>& callback) {
|
|
|
|
ndk::ScopedAStatus status;
|
|
|
|
|
|
|
|
if (mHasTimedOutEffect)
|
|
|
|
writeNode(VIBRATOR_CP_TRIGGER_PATH, 0); // Clear all effects
|
|
|
|
|
|
|
|
status = activate(timeoutMs);
|
|
|
|
|
|
|
|
if (callback != nullptr) {
|
|
|
|
std::thread([=] {
|
|
|
|
LOG(DEBUG) << "Starting on on another thread";
|
|
|
|
usleep(timeoutMs * 1000);
|
|
|
|
LOG(DEBUG) << "Notifying on complete";
|
|
|
|
if (!callback->onComplete().isOk()) {
|
|
|
|
LOG(ERROR) << "Failed to call onComplete";
|
|
|
|
}
|
|
|
|
}).detach();
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::perform(Effect effect, EffectStrength strength, const std::shared_ptr<IVibratorCallback>& callback, int32_t* _aidl_return) {
|
|
|
|
ndk::ScopedAStatus status;
|
|
|
|
uint32_t amplitude = strengthToAmplitude(strength, &status);
|
|
|
|
uint32_t ms = 1000;
|
|
|
|
|
|
|
|
if (!status.isOk())
|
|
|
|
return status;
|
|
|
|
|
|
|
|
activate(0);
|
|
|
|
setAmplitude(amplitude);
|
|
|
|
|
|
|
|
if (mHasTimedOutEffect && CP_TRIGGER_EFFECTS.find(effect) != CP_TRIGGER_EFFECTS.end()) {
|
|
|
|
writeNode(VIBRATOR_CP_TRIGGER_PATH, CP_TRIGGER_EFFECTS[effect]);
|
|
|
|
} else {
|
|
|
|
if (mHasTimedOutEffect)
|
|
|
|
writeNode(VIBRATOR_CP_TRIGGER_PATH, 0); // Clear previous effect
|
|
|
|
|
|
|
|
ms = effectToMs(effect, &status);
|
|
|
|
|
|
|
|
if (!status.isOk())
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = activate(ms);
|
|
|
|
|
|
|
|
if (callback != nullptr) {
|
|
|
|
std::thread([=] {
|
|
|
|
LOG(DEBUG) << "Starting perform on another thread";
|
|
|
|
usleep(ms * 1000);
|
|
|
|
LOG(DEBUG) << "Notifying perform complete";
|
|
|
|
callback->onComplete();
|
|
|
|
}).detach();
|
|
|
|
}
|
|
|
|
|
|
|
|
*_aidl_return = ms;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getSupportedEffects(std::vector<Effect>* _aidl_return) {
|
|
|
|
*_aidl_return = {Effect::CLICK, Effect::DOUBLE_CLICK, Effect::HEAVY_CLICK,
|
|
|
|
Effect::TICK, Effect::TEXTURE_TICK, Effect::THUD, Effect::POP,
|
|
|
|
Effect::RINGTONE_1, Effect::RINGTONE_2, Effect::RINGTONE_3,
|
|
|
|
Effect::RINGTONE_4, Effect::RINGTONE_5, Effect::RINGTONE_6,
|
|
|
|
Effect::RINGTONE_7, Effect::RINGTONE_7, Effect::RINGTONE_8,
|
|
|
|
Effect::RINGTONE_9, Effect::RINGTONE_10, Effect::RINGTONE_11,
|
|
|
|
Effect::RINGTONE_12, Effect::RINGTONE_13, Effect::RINGTONE_14,
|
|
|
|
Effect::RINGTONE_15};
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::setAmplitude(float amplitude) {
|
|
|
|
uint32_t intensity;
|
|
|
|
|
|
|
|
if (amplitude == 0) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG(DEBUG) << "Setting amplitude: " << (uint32_t)amplitude;
|
|
|
|
|
|
|
|
intensity = std::lround((amplitude - 1) * INTENSITY_MAX / 254.0);
|
|
|
|
if (intensity > INTENSITY_MAX) {
|
|
|
|
intensity = INTENSITY_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (intensity == 0) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG(DEBUG) << "Setting intensity: " << intensity;
|
|
|
|
|
|
|
|
if (mHasTimedOutIntensity) {
|
|
|
|
return writeNode(VIBRATOR_INTENSITY_PATH, intensity);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::setExternalControl(bool enabled) {
|
|
|
|
if (mEnabled) {
|
|
|
|
LOG(WARNING) << "Setting external control while the vibrator is enabled is "
|
|
|
|
"unsupported!";
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG(INFO) << "ExternalControl: " << mExternalControl << " -> " << enabled;
|
|
|
|
mExternalControl = enabled;
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getCompositionDelayMax(int32_t* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getCompositionSizeMax(int32_t* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getSupportedPrimitives(std::vector<CompositePrimitive>* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getPrimitiveDuration(CompositePrimitive /*primitive*/, int32_t* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::compose(const std::vector<CompositeEffect>& /*composite*/, const std::shared_ptr<IVibratorCallback>& /*callback*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getSupportedAlwaysOnEffects(std::vector<Effect>* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::alwaysOnEnable(int32_t /*id*/, Effect /*effect*/, EffectStrength /*strength*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::alwaysOnDisable(int32_t /*id*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getResonantFrequency(float* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getQFactor(float* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getFrequencyResolution(float* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getFrequencyMinimum(float* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getBandwidthAmplitudeMap(std::vector<float>* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getPwlePrimitiveDurationMax(int32_t* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getPwleCompositionSizeMax(int32_t* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::getSupportedBraking(std::vector<Braking>* /*_aidl_return*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::composePwle(const std::vector<PrimitivePwle>& /*composite*/, const std::shared_ptr<IVibratorCallback>& /*callback*/) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Vibrator::activate(uint32_t timeoutMs) {
|
|
|
|
std::lock_guard<std::mutex> lock{mMutex};
|
|
|
|
if (!mIsTimedOutVibrator) {
|
|
|
|
return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
}
|
|
|
|
|
|
|
|
return writeNode(VIBRATOR_TIMEOUT_PATH, timeoutMs);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Vibrator::strengthToAmplitude(EffectStrength strength, ndk::ScopedAStatus* status) {
|
|
|
|
*status = ndk::ScopedAStatus::ok();
|
|
|
|
|
|
|
|
switch (strength) {
|
|
|
|
case EffectStrength::LIGHT:
|
|
|
|
return 64;
|
|
|
|
case EffectStrength::MEDIUM:
|
|
|
|
return 128;
|
|
|
|
case EffectStrength::STRONG:
|
|
|
|
return 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
*status = ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t Vibrator::effectToMs(Effect effect, ndk::ScopedAStatus* status) {
|
|
|
|
*status = ndk::ScopedAStatus::ok();
|
|
|
|
switch (effect) {
|
|
|
|
case Effect::CLICK:
|
|
|
|
return 10;
|
|
|
|
case Effect::DOUBLE_CLICK:
|
|
|
|
return 15;
|
|
|
|
case Effect::TICK:
|
|
|
|
case Effect::TEXTURE_TICK:
|
|
|
|
case Effect::THUD:
|
|
|
|
case Effect::POP:
|
|
|
|
return 5;
|
|
|
|
case Effect::HEAVY_CLICK:
|
|
|
|
return 10;
|
|
|
|
case Effect::RINGTONE_1:
|
|
|
|
case Effect::RINGTONE_2:
|
|
|
|
case Effect::RINGTONE_3:
|
|
|
|
case Effect::RINGTONE_4:
|
|
|
|
case Effect::RINGTONE_5:
|
|
|
|
case Effect::RINGTONE_6:
|
|
|
|
case Effect::RINGTONE_7:
|
|
|
|
case Effect::RINGTONE_8:
|
|
|
|
case Effect::RINGTONE_9:
|
|
|
|
case Effect::RINGTONE_10:
|
|
|
|
case Effect::RINGTONE_11:
|
|
|
|
case Effect::RINGTONE_12:
|
|
|
|
case Effect::RINGTONE_13:
|
|
|
|
case Effect::RINGTONE_14:
|
|
|
|
case Effect::RINGTONE_15:
|
|
|
|
return 30000;
|
|
|
|
}
|
|
|
|
*status = ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vibrator
|
|
|
|
} // namespace hardware
|
|
|
|
} // namespace android
|
|
|
|
} // namespace aidl
|