/* * 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; } }; binder::Status 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(); } return binder::Status::ok(); } // Used by tests only. binder::Status InputManager::registerInputChannel(const InputChannel& 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 binder::Status::ok(); } mDispatcher->registerInputChannel(channel.dup()); return binder::Status::ok(); } binder::Status InputManager::unregisterInputChannel(const InputChannel& channel) { mDispatcher->unregisterInputChannel(channel); return binder::Status::ok(); } status_t InputManager::dump(int fd, const Vector& args) { std::string dump; dump += " InputFlinger dump\n"; ::write(fd, dump.c_str(), dump.size()); return NO_ERROR; } binder::Status InputManager::setFocusedWindow(const FocusRequest& request) { mDispatcher->setFocusedWindow(request); return binder::Status::ok(); } } // namespace android