/* * Copyright (C) 2010 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 "InputManager" //#define LOG_NDEBUG 0 #include "InputManager.h" #include "InputDispatcherFactory.h" #include "InputReaderFactory.h" #include #include #include #include namespace android { InputManager::InputManager( const sp& readerPolicy, const sp& dispatcherPolicy) { mDispatcher = createInputDispatcher(dispatcherPolicy); mClassifier = new InputClassifier(mDispatcher); mReader = createInputReader(readerPolicy, mClassifier); } InputManager::~InputManager() { stop(); } status_t InputManager::start() { status_t result = mDispatcher->start(); if (result) { ALOGE("Could not start InputDispatcher thread due to error %d.", result); return result; } result = mReader->start(); if (result) { ALOGE("Could not start InputReader due to error %d.", result); mDispatcher->stop(); return result; } return OK; } status_t InputManager::stop() { status_t status = OK; status_t result = mReader->stop(); if (result) { ALOGW("Could not stop InputReader due to error %d.", result); status = result; } result = mDispatcher->stop(); if (result) { ALOGW("Could not stop InputDispatcher thread due to error %d.", result); status = result; } return status; } sp InputManager::getReader() { return mReader; } sp InputManager::getClassifier() { return mClassifier; } sp InputManager::getDispatcher() { return mDispatcher; } class BinderWindowHandle : public InputWindowHandle { public: BinderWindowHandle(const InputWindowInfo& info) { mInfo = info; } bool updateInfo() override { return true; } }; void InputManager::setInputWindows(const std::vector& infos, const sp& setInputWindowsListener) { std::unordered_map>> handlesPerDisplay; std::vector> handles; for (const auto& info : infos) { handlesPerDisplay.emplace(info.displayId, std::vector>()); handlesPerDisplay[info.displayId].push_back(new BinderWindowHandle(info)); } mDispatcher->setInputWindows(handlesPerDisplay); if (setInputWindowsListener) { setInputWindowsListener->onSetInputWindowsFinished(); } } // Used by tests only. void InputManager::registerInputChannel(const sp& channel) { IPCThreadState* ipc = IPCThreadState::self(); const int uid = ipc->getCallingUid(); if (uid != AID_SHELL && uid != AID_ROOT) { ALOGE("Invalid attempt to register input channel over IPC" "from non shell/root entity (PID: %d)", ipc->getCallingPid()); return; } mDispatcher->registerInputChannel(channel); } void InputManager::unregisterInputChannel(const sp& channel) { mDispatcher->unregisterInputChannel(channel); } void InputManager::setMotionClassifierEnabled(bool enabled) { mClassifier->setMotionClassifierEnabled(enabled); } } // namespace android