/* * Copyright (C) 2021 The LineageOS Project * * SPDX-License-Identifier: Apache-2.0 */ #include "Vibrator.h" #include #include #include #include #include #include namespace aidl { namespace android { namespace hardware { namespace vibrator { static std::map 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 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& 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& 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* _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; } 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* /*_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& /*composite*/, const std::shared_ptr& /*callback*/) { return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION); } ndk::ScopedAStatus Vibrator::getSupportedAlwaysOnEffects(std::vector* /*_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* /*_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* /*_aidl_return*/) { return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION); } ndk::ScopedAStatus Vibrator::composePwle(const std::vector& /*composite*/, const std::shared_ptr& /*callback*/) { return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION); } ndk::ScopedAStatus Vibrator::activate(uint32_t timeoutMs) { std::lock_guard 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