/* * 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 { static int32_t exceptionCodeFromStatusT(status_t status) { switch (status) { case OK: return binder::Status::EX_NONE; case INVALID_OPERATION: return binder::Status::EX_UNSUPPORTED_OPERATION; case BAD_VALUE: case BAD_TYPE: case NAME_NOT_FOUND: return binder::Status::EX_ILLEGAL_ARGUMENT; case NO_INIT: return binder::Status::EX_ILLEGAL_STATE; case PERMISSION_DENIED: return binder::Status::EX_SECURITY; default: return binder::Status::EX_TRANSACTION_FAILED; } } 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::createInputChannel(const std::string& name, InputChannel* outChannel) { 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(); } base::Result> channel = mDispatcher->createInputChannel(name); if (!channel.ok()) { return binder::Status::fromExceptionCode(exceptionCodeFromStatusT(channel.error().code()), channel.error().message().c_str()); } (*channel)->copyTo(*outChannel); return binder::Status::ok(); } binder::Status InputManager::removeInputChannel(const sp& connectionToken) { mDispatcher->removeInputChannel(connectionToken); 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