/* * 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 "../Macros.h" #include "VibratorInputMapper.h" namespace android { VibratorInputMapper::VibratorInputMapper(InputDeviceContext& deviceContext) : InputMapper(deviceContext), mVibrating(false), mSequence(0) {} VibratorInputMapper::~VibratorInputMapper() {} uint32_t VibratorInputMapper::getSources() { return 0; } void VibratorInputMapper::populateDeviceInfo(InputDeviceInfo* info) { InputMapper::populateDeviceInfo(info); info->setVibrator(true); } void VibratorInputMapper::process(const RawEvent* rawEvent) { // TODO: Handle FF_STATUS, although it does not seem to be widely supported. } void VibratorInputMapper::vibrate(const VibrationSequence& sequence, ssize_t repeat, int32_t token) { #if DEBUG_VIBRATOR ALOGD("vibrate: deviceId=%d, pattern=[%s], repeat=%zd, token=%d", getDeviceId(), sequence.toString().c_str(), repeat, token); #endif mVibrating = true; mSequence = sequence; mRepeat = repeat; mToken = token; mIndex = -1; // Request InputReader to notify InputManagerService for vibration started. NotifyVibratorStateArgs args(getContext()->getNextId(), systemTime(), getDeviceId(), true); getListener()->notifyVibratorState(&args); nextStep(); } void VibratorInputMapper::cancelVibrate(int32_t token) { #if DEBUG_VIBRATOR ALOGD("cancelVibrate: deviceId=%d, token=%d", getDeviceId(), token); #endif if (mVibrating && mToken == token) { stopVibrating(); } } bool VibratorInputMapper::isVibrating() { return mVibrating; } std::vector VibratorInputMapper::getVibratorIds() { return getDeviceContext().getVibratorIds(); } void VibratorInputMapper::timeoutExpired(nsecs_t when) { if (mVibrating) { if (when >= mNextStepTime) { nextStep(); } else { getContext()->requestTimeoutAtTime(mNextStepTime); } } } void VibratorInputMapper::nextStep() { #if DEBUG_VIBRATOR ALOGD("nextStep: index=%d, vibrate deviceId=%d", (int)mIndex, getDeviceId()); #endif mIndex += 1; if (size_t(mIndex) >= mSequence.pattern.size()) { if (mRepeat < 0) { // We are done. stopVibrating(); return; } mIndex = mRepeat; } const VibrationElement& element = mSequence.pattern[mIndex]; if (element.isOn()) { #if DEBUG_VIBRATOR std::string description = element.toString(); ALOGD("nextStep: sending vibrate deviceId=%d, element=%s", getDeviceId(), description.c_str()); #endif getDeviceContext().vibrate(element); } else { #if DEBUG_VIBRATOR ALOGD("nextStep: sending cancel vibrate deviceId=%d", getDeviceId()); #endif getDeviceContext().cancelVibrate(); } nsecs_t now = systemTime(SYSTEM_TIME_MONOTONIC); std::chrono::nanoseconds duration = std::chrono::duration_cast(element.duration); mNextStepTime = now + duration.count(); getContext()->requestTimeoutAtTime(mNextStepTime); #if DEBUG_VIBRATOR ALOGD("nextStep: scheduled timeout in %lldms", element.duration.count()); #endif } void VibratorInputMapper::stopVibrating() { mVibrating = false; #if DEBUG_VIBRATOR ALOGD("stopVibrating: sending cancel vibrate deviceId=%d", getDeviceId()); #endif getDeviceContext().cancelVibrate(); // Request InputReader to notify InputManagerService for vibration complete. NotifyVibratorStateArgs args(getContext()->getNextId(), systemTime(), getDeviceId(), false); getListener()->notifyVibratorState(&args); } void VibratorInputMapper::dump(std::string& dump) { dump += INDENT2 "Vibrator Input Mapper:\n"; dump += StringPrintf(INDENT3 "Vibrating: %s\n", toString(mVibrating)); if (mVibrating) { dump += INDENT3 "Pattern: "; dump += mSequence.toString(); dump += "\n"; dump += StringPrintf(INDENT3 "Repeat Index: %zd\n", mRepeat); } } } // namespace android