/* * 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. */ #pragma once #include #include #include #include #include #include #include #include #include "EventHub.h" #include "InputReaderBase.h" #include "InputReaderContext.h" #include "NotifyArgs.h" namespace android { class PeripheralController; class PeripheralControllerInterface; class InputDeviceContext; class InputMapper; /* Represents the state of a single input device. */ class InputDevice { public: InputDevice(InputReaderContext* context, int32_t id, int32_t generation, const InputDeviceIdentifier& identifier); ~InputDevice(); inline InputReaderContext* getContext() { return mContext; } inline int32_t getId() const { return mId; } inline int32_t getControllerNumber() const { return mControllerNumber; } inline int32_t getGeneration() const { return mGeneration; } inline const std::string getName() const { return mIdentifier.name; } inline const std::string getDescriptor() { return mIdentifier.descriptor; } inline std::optional getBluetoothAddress() const { return mIdentifier.bluetoothAddress; } inline const std::string getLocation() const { return mIdentifier.location; } inline ftl::Flags getClasses() const { return mClasses; } inline uint32_t getSources() const { return mSources; } inline bool hasEventHubDevices() const { return !mDevices.empty(); } inline bool isExternal() { return mIsExternal; } inline std::optional getAssociatedDisplayPort() const { return mAssociatedDisplayPort; } inline std::optional getAssociatedDisplayUniqueIdByPort() const { return mAssociatedDisplayUniqueIdByPort; } inline std::optional getAssociatedDisplayUniqueIdByDescriptor() const { return mAssociatedDisplayUniqueIdByDescriptor; } inline std::optional getDeviceTypeAssociation() const { return mAssociatedDeviceType; } inline std::optional getAssociatedViewport() const { return mAssociatedViewport; } inline bool hasMic() const { return mHasMic; } inline bool isIgnored() { return !getMapperCount() && !mController; } inline KeyboardType getKeyboardType() const { return mKeyboardType; } bool isEnabled(); void dump(std::string& dump, const std::string& eventHubDevStr); void addEmptyEventHubDevice(int32_t eventHubId); [[nodiscard]] std::list addEventHubDevice( nsecs_t when, int32_t eventHubId, const InputReaderConfiguration& readerConfig); void removeEventHubDevice(int32_t eventHubId); [[nodiscard]] std::list configure(nsecs_t when, const InputReaderConfiguration& readerConfig, ConfigurationChanges changes); [[nodiscard]] std::list reset(nsecs_t when); [[nodiscard]] std::list process(const RawEvent* rawEvents, size_t count); [[nodiscard]] std::list timeoutExpired(nsecs_t when); [[nodiscard]] std::list updateExternalStylusState(const StylusState& state); InputDeviceInfo getDeviceInfo(); int32_t getKeyCodeState(uint32_t sourceMask, int32_t keyCode); int32_t getScanCodeState(uint32_t sourceMask, int32_t scanCode); int32_t getSwitchState(uint32_t sourceMask, int32_t switchCode); int32_t getKeyCodeForKeyLocation(int32_t locationKeyCode) const; bool markSupportedKeyCodes(uint32_t sourceMask, const std::vector& keyCodes, uint8_t* outFlags); [[nodiscard]] std::list vibrate(const VibrationSequence& sequence, ssize_t repeat, int32_t token); [[nodiscard]] std::list cancelVibrate(int32_t token); bool isVibrating(); std::vector getVibratorIds(); [[nodiscard]] std::list cancelTouch(nsecs_t when, nsecs_t readTime); bool enableSensor(InputDeviceSensorType sensorType, std::chrono::microseconds samplingPeriod, std::chrono::microseconds maxBatchReportLatency); void disableSensor(InputDeviceSensorType sensorType); void flushSensor(InputDeviceSensorType sensorType); std::optional getBatteryEventHubId() const; bool setLightColor(int32_t lightId, int32_t color); bool setLightPlayerId(int32_t lightId, int32_t playerId); std::optional getLightColor(int32_t lightId); std::optional getLightPlayerId(int32_t lightId); int32_t getMetaState(); void updateMetaState(int32_t keyCode); void addKeyRemapping(int32_t fromKeyCode, int32_t toKeyCode); void setKeyboardType(KeyboardType keyboardType); void bumpGeneration(); [[nodiscard]] NotifyDeviceResetArgs notifyReset(nsecs_t when); inline const PropertyMap& getConfiguration() { return mConfiguration; } inline EventHubInterface* getEventHub() { return mContext->getEventHub(); } std::optional getAssociatedDisplayId(); void updateLedState(bool reset); size_t getMapperCount(); // construct and add a mapper to the input device template T& addMapper(int32_t eventHubId, Args... args) { // ensure a device entry exists for this eventHubId addEmptyEventHubDevice(eventHubId); // create mapper auto& devicePair = mDevices[eventHubId]; auto& deviceContext = devicePair.first; auto& mappers = devicePair.second; T* mapper = new T(*deviceContext, args...); mappers.emplace_back(mapper); return *mapper; } template T& constructAndAddMapper(int32_t eventHubId, Args... args) { // create mapper auto& devicePair = mDevices[eventHubId]; auto& deviceContext = devicePair.first; auto& mappers = devicePair.second; mappers.push_back(createInputMapper(*deviceContext, args...)); return static_cast(*mappers.back()); } // construct and add a controller to the input device template T& addController(int32_t eventHubId) { // ensure a device entry exists for this eventHubId addEmptyEventHubDevice(eventHubId); // create controller auto& devicePair = mDevices[eventHubId]; auto& deviceContext = devicePair.first; mController = std::make_unique(*deviceContext); return *(reinterpret_cast(mController.get())); } private: InputReaderContext* mContext; int32_t mId; int32_t mGeneration; int32_t mControllerNumber; InputDeviceIdentifier mIdentifier; std::string mAlias; ftl::Flags mClasses; // map from eventHubId to device context and mappers using MapperVector = std::vector>; using DevicePair = std::pair, MapperVector>; // Map from EventHub ID to pair of device context and vector of mapper. std::unordered_map mDevices; // Misc devices controller for lights, battery, etc. std::unique_ptr mController; uint32_t mSources; bool mIsWaking; bool mIsExternal; KeyboardType mKeyboardType = KeyboardType::NONE; std::optional mAssociatedDisplayPort; std::optional mAssociatedDisplayUniqueIdByPort; std::optional mAssociatedDisplayUniqueIdByDescriptor; std::optional mAssociatedDeviceType; std::optional mAssociatedViewport; bool mHasMic; bool mDropUntilNextSync; std::optional mShouldSmoothScroll; typedef int32_t (InputMapper::*GetStateFunc)(uint32_t sourceMask, int32_t code); int32_t getState(uint32_t sourceMask, int32_t code, GetStateFunc getStateFunc); std::vector> createMappers( InputDeviceContext& contextPtr, const InputReaderConfiguration& readerConfig); [[nodiscard]] std::list configureInternal( nsecs_t when, const InputReaderConfiguration& readerConfig, ConfigurationChanges changes, bool forceEnable = false); [[nodiscard]] std::list updateEnableState( nsecs_t when, const InputReaderConfiguration& readerConfig, bool forceEnable = false); PropertyMap mConfiguration; // Runs logic post a `process` call. This can be used to update the generated `NotifyArgs` as // per the properties of the InputDevice. void postProcess(std::list& args) const; // helpers to interate over the devices collection // run a function against every mapper on every subdevice inline void for_each_mapper(std::function f) { for (auto& deviceEntry : mDevices) { auto& devicePair = deviceEntry.second; auto& mappers = devicePair.second; for (auto& mapperPtr : mappers) { f(*mapperPtr); } } } // run a function against every mapper on a specific subdevice inline void for_each_mapper_in_subdevice(int32_t eventHubDevice, std::function f) { auto deviceIt = mDevices.find(eventHubDevice); if (deviceIt != mDevices.end()) { auto& devicePair = deviceIt->second; auto& mappers = devicePair.second; for (auto& mapperPtr : mappers) { f(*mapperPtr); } } } // run a function against every subdevice inline void for_each_subdevice(std::function f) { for (auto& deviceEntry : mDevices) { auto& devicePair = deviceEntry.second; auto& contextPtr = devicePair.first; f(*contextPtr); } } // return the first value returned by a function over every mapper. // if all mappers return nullopt, return nullopt. template inline std::optional first_in_mappers( std::function(InputMapper&)> f) const { for (auto& deviceEntry : mDevices) { auto& devicePair = deviceEntry.second; auto& mappers = devicePair.second; for (auto& mapperPtr : mappers) { std::optional ret = f(*mapperPtr); if (ret) { return ret; } } } return std::nullopt; } }; /* Provides access to EventHub methods, but limits access to the current InputDevice. * Essentially an implementation of EventHubInterface, but for a specific device id. * Helps hide implementation details of InputDevice and EventHub. Used by mappers to * check the status of the associated hardware device */ class InputDeviceContext { public: InputDeviceContext(InputDevice& device, int32_t eventHubId); virtual ~InputDeviceContext(); inline InputReaderContext* getContext() { return mContext; } inline int32_t getId() { return mDeviceId; } inline int32_t getEventHubId() { return mId; } inline ftl::Flags getDeviceClasses() const { return mEventHub->getDeviceClasses(mId); } inline InputDeviceIdentifier getDeviceIdentifier() const { return mEventHub->getDeviceIdentifier(mId); } inline int32_t getDeviceControllerNumber() const { return mEventHub->getDeviceControllerNumber(mId); } inline status_t getAbsoluteAxisInfo(int32_t code, RawAbsoluteAxisInfo* axisInfo) const { if (const auto status = mEventHub->getAbsoluteAxisInfo(mId, code, axisInfo); status != OK) { return status; } // Validate axis info for InputDevice. if (axisInfo->valid && axisInfo->minValue == axisInfo->maxValue) { // Historically, we deem axes with the same min and max values as invalid to avoid // dividing by zero when scaling by max - min. // TODO(b/291772515): Perform axis info validation on a per-axis basis when it is used. axisInfo->valid = false; } return OK; } inline bool hasRelativeAxis(int32_t code) const { return mEventHub->hasRelativeAxis(mId, code); } inline bool hasInputProperty(int32_t property) const { return mEventHub->hasInputProperty(mId, property); } inline bool hasMscEvent(int mscEvent) const { return mEventHub->hasMscEvent(mId, mscEvent); } inline void addKeyRemapping(int32_t fromKeyCode, int32_t toKeyCode) const { mEventHub->addKeyRemapping(mId, fromKeyCode, toKeyCode); } inline status_t mapKey(int32_t scanCode, int32_t usageCode, int32_t metaState, int32_t* outKeycode, int32_t* outMetaState, uint32_t* outFlags) const { return mEventHub->mapKey(mId, scanCode, usageCode, metaState, outKeycode, outMetaState, outFlags); } inline status_t mapAxis(int32_t scanCode, AxisInfo* outAxisInfo) const { return mEventHub->mapAxis(mId, scanCode, outAxisInfo); } inline base::Result> mapSensor(int32_t absCode) { return mEventHub->mapSensor(mId, absCode); } inline const std::vector getRawLightIds() { return mEventHub->getRawLightIds(mId); } inline std::optional getRawLightInfo(int32_t lightId) { return mEventHub->getRawLightInfo(mId, lightId); } inline std::optional getLightBrightness(int32_t lightId) { return mEventHub->getLightBrightness(mId, lightId); } inline void setLightBrightness(int32_t lightId, int32_t brightness) { return mEventHub->setLightBrightness(mId, lightId, brightness); } inline std::optional> getLightIntensities( int32_t lightId) { return mEventHub->getLightIntensities(mId, lightId); } inline void setLightIntensities(int32_t lightId, std::unordered_map intensities) { return mEventHub->setLightIntensities(mId, lightId, intensities); } inline std::vector getVideoFrames() { return mEventHub->getVideoFrames(mId); } inline int32_t getScanCodeState(int32_t scanCode) const { return mEventHub->getScanCodeState(mId, scanCode); } inline int32_t getKeyCodeState(int32_t keyCode) const { return mEventHub->getKeyCodeState(mId, keyCode); } inline int32_t getKeyCodeForKeyLocation(int32_t locationKeyCode) const { return mEventHub->getKeyCodeForKeyLocation(mId, locationKeyCode); } inline int32_t getSwitchState(int32_t sw) const { return mEventHub->getSwitchState(mId, sw); } inline status_t getAbsoluteAxisValue(int32_t code, int32_t* outValue) const { return mEventHub->getAbsoluteAxisValue(mId, code, outValue); } inline base::Result> getMtSlotValues(int32_t axis, size_t slotCount) const { return mEventHub->getMtSlotValues(mId, axis, slotCount); } inline bool markSupportedKeyCodes(const std::vector& keyCodes, uint8_t* outFlags) const { return mEventHub->markSupportedKeyCodes(mId, keyCodes, outFlags); } inline bool hasScanCode(int32_t scanCode) const { return mEventHub->hasScanCode(mId, scanCode); } inline bool hasKeyCode(int32_t keyCode) const { return mEventHub->hasKeyCode(mId, keyCode); } inline bool hasLed(int32_t led) const { return mEventHub->hasLed(mId, led); } inline void setLedState(int32_t led, bool on) { return mEventHub->setLedState(mId, led, on); } inline void getVirtualKeyDefinitions(std::vector& outVirtualKeys) const { return mEventHub->getVirtualKeyDefinitions(mId, outVirtualKeys); } inline const std::shared_ptr getKeyCharacterMap() const { return mEventHub->getKeyCharacterMap(mId); } inline bool setKeyboardLayoutOverlay(std::shared_ptr map) { return mEventHub->setKeyboardLayoutOverlay(mId, map); } inline const std::optional getRawLayoutInfo() { return mEventHub->getRawLayoutInfo(mId); } inline void vibrate(const VibrationElement& element) { return mEventHub->vibrate(mId, element); } inline void cancelVibrate() { return mEventHub->cancelVibrate(mId); } inline std::vector getVibratorIds() { return mEventHub->getVibratorIds(mId); } inline const std::vector getRawBatteryIds() { return mEventHub->getRawBatteryIds(mId); } inline std::optional getRawBatteryInfo(int32_t batteryId) { return mEventHub->getRawBatteryInfo(mId, batteryId); } inline std::optional getBatteryCapacity(int32_t batteryId) { return mEventHub->getBatteryCapacity(mId, batteryId); } inline std::optional getBatteryStatus(int32_t batteryId) { return mEventHub->getBatteryStatus(mId, batteryId); } inline bool hasAbsoluteAxis(int32_t code) const { RawAbsoluteAxisInfo info; mEventHub->getAbsoluteAxisInfo(mId, code, &info); return info.valid; } inline bool isKeyPressed(int32_t scanCode) const { return mEventHub->getScanCodeState(mId, scanCode) == AKEY_STATE_DOWN; } inline bool isKeyCodePressed(int32_t keyCode) const { return mEventHub->getKeyCodeState(mId, keyCode) == AKEY_STATE_DOWN; } inline int32_t getAbsoluteAxisValue(int32_t code) const { int32_t value; mEventHub->getAbsoluteAxisValue(mId, code, &value); return value; } inline bool isDeviceEnabled() { return mEventHub->isDeviceEnabled(mId); } inline status_t enableDevice() { return mEventHub->enableDevice(mId); } inline status_t disableDevice() { return mEventHub->disableDevice(mId); } inline const std::string getName() const { return mDevice.getName(); } inline const std::string getDescriptor() { return mDevice.getDescriptor(); } inline const std::string getLocation() { return mDevice.getLocation(); } inline bool isExternal() const { return mDevice.isExternal(); } inline std::optional getAssociatedDisplayPort() const { return mDevice.getAssociatedDisplayPort(); } inline std::optional getAssociatedDisplayUniqueIdByPort() const { return mDevice.getAssociatedDisplayUniqueIdByPort(); } inline std::optional getAssociatedDisplayUniqueIdByDescriptor() const { return mDevice.getAssociatedDisplayUniqueIdByDescriptor(); } inline std::optional getDeviceTypeAssociation() const { return mDevice.getDeviceTypeAssociation(); } virtual std::optional getAssociatedViewport() const { return mDevice.getAssociatedViewport(); } [[nodiscard]] inline std::list cancelTouch(nsecs_t when, nsecs_t readTime) { return mDevice.cancelTouch(when, readTime); } inline void bumpGeneration() { mDevice.bumpGeneration(); } inline const PropertyMap& getConfiguration() const { return mDevice.getConfiguration(); } inline KeyboardType getKeyboardType() const { return mDevice.getKeyboardType(); } inline void setKeyboardType(KeyboardType keyboardType) { return mDevice.setKeyboardType(keyboardType); } private: InputDevice& mDevice; InputReaderContext* mContext; EventHubInterface* mEventHub; int32_t mId; int32_t mDeviceId; }; } // namespace android