/* * Copyright (C) 2015 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. */ #define LOG_TAG "MouseInputMapper" //#define LOG_NDEBUG 0 #include "MouseInputMapper.h" #include #include #include #include #include "InputHost.h" #include "InputHub.h" namespace android { // Map scancodes to input HAL usages. // The order of these definitions MUST remain in sync with the order they are // defined in linux/input.h. static struct { int32_t scancode; InputUsage usage; } codeMap[] = { {BTN_LEFT, INPUT_USAGE_BUTTON_PRIMARY}, {BTN_RIGHT, INPUT_USAGE_BUTTON_SECONDARY}, {BTN_MIDDLE, INPUT_USAGE_BUTTON_TERTIARY}, {BTN_SIDE, INPUT_USAGE_BUTTON_UNKNOWN}, {BTN_EXTRA, INPUT_USAGE_BUTTON_UNKNOWN}, {BTN_FORWARD, INPUT_USAGE_BUTTON_FORWARD}, {BTN_BACK, INPUT_USAGE_BUTTON_BACK}, {BTN_TASK, INPUT_USAGE_BUTTON_UNKNOWN}, }; bool MouseInputMapper::configureInputReport(InputDeviceNode* devNode, InputReportDefinition* report) { setInputReportDefinition(report); getInputReportDefinition()->addCollection(INPUT_COLLECTION_ID_MOUSE, 1); // Configure mouse axes if (!devNode->hasRelativeAxis(REL_X) || !devNode->hasRelativeAxis(REL_Y)) { ALOGE("Device %s is missing a relative x or y axis. Device cannot be configured.", devNode->getPath().c_str()); return false; } getInputReportDefinition()->declareUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_X, INT32_MIN, INT32_MAX, 1.0f); getInputReportDefinition()->declareUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_Y, INT32_MIN, INT32_MAX, 1.0f); if (devNode->hasRelativeAxis(REL_WHEEL)) { getInputReportDefinition()->declareUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_VSCROLL, -1, 1, 0.0f); } if (devNode->hasRelativeAxis(REL_HWHEEL)) { getInputReportDefinition()->declareUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_HSCROLL, -1, 1, 0.0f); } // Configure mouse buttons InputUsage usages[NELEM(codeMap)]; int numUsages = 0; for (int32_t i = 0; i < NELEM(codeMap); ++i) { if (devNode->hasKey(codeMap[i].scancode)) { usages[numUsages++] = codeMap[i].usage; } } if (numUsages == 0) { ALOGW("MouseInputMapper found no buttons for %s", devNode->getPath().c_str()); } getInputReportDefinition()->declareUsages(INPUT_COLLECTION_ID_MOUSE, usages, numUsages); return true; } void MouseInputMapper::process(const InputEvent& event) { ALOGV("processing mouse event. type=%d code=%d value=%d", event.type, event.code, event.value); switch (event.type) { case EV_KEY: processButton(event.code, event.value); break; case EV_REL: processMotion(event.code, event.value); break; case EV_SYN: if (event.code == SYN_REPORT) { sync(event.when); } break; default: ALOGV("unknown mouse event type: %d", event.type); } } void MouseInputMapper::processMotion(int32_t code, int32_t value) { switch (code) { case REL_X: mRelX = value; break; case REL_Y: mRelY = value; break; case REL_WHEEL: mRelWheel = value; break; case REL_HWHEEL: mRelHWheel = value; break; default: // Unknown code. Ignore. break; } } // Map evdev button codes to bit indices. This function assumes code >= // BTN_MOUSE. uint32_t buttonToBit(int32_t code) { return static_cast(code - BTN_MOUSE); } void MouseInputMapper::processButton(int32_t code, int32_t value) { // Mouse buttons start at BTN_MOUSE and end before BTN_JOYSTICK. There isn't // really enough room after the mouse buttons for another button class, so // the risk of a button type being inserted after mouse is low. if (code >= BTN_MOUSE && code < BTN_JOYSTICK) { if (value) { mButtonValues.markBit(buttonToBit(code)); } else { mButtonValues.clearBit(buttonToBit(code)); } mUpdatedButtonMask.markBit(buttonToBit(code)); } } void MouseInputMapper::sync(nsecs_t when) { // Process updated button states. while (!mUpdatedButtonMask.isEmpty()) { auto bit = mUpdatedButtonMask.clearFirstMarkedBit(); getInputReport()->setBoolUsage(INPUT_COLLECTION_ID_MOUSE, codeMap[bit].usage, mButtonValues.hasBit(bit), 0); } // Process motion and scroll changes. if (mRelX != 0) { getInputReport()->setIntUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_X, mRelX, 0); } if (mRelY != 0) { getInputReport()->setIntUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_Y, mRelY, 0); } if (mRelWheel != 0) { getInputReport()->setIntUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_VSCROLL, mRelWheel, 0); } if (mRelHWheel != 0) { getInputReport()->setIntUsage(INPUT_COLLECTION_ID_MOUSE, INPUT_USAGE_AXIS_HSCROLL, mRelHWheel, 0); } // Report and reset. getInputReport()->reportEvent(getDeviceHandle()); mUpdatedButtonMask.clear(); mButtonValues.clear(); mRelX = 0; mRelY = 0; mRelWheel = 0; mRelHWheel = 0; } } // namespace android