You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
179 lines
5.1 KiB
179 lines
5.1 KiB
/*
|
|
* 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 <binder/IPCThreadState.h>
|
|
|
|
#include <log/log.h>
|
|
#include <unordered_map>
|
|
|
|
#include <private/android_filesystem_config.h>
|
|
|
|
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<InputReaderPolicyInterface>& readerPolicy,
|
|
const sp<InputDispatcherPolicyInterface>& 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<InputReaderInterface> InputManager::getReader() {
|
|
return mReader;
|
|
}
|
|
|
|
sp<InputClassifierInterface> InputManager::getClassifier() {
|
|
return mClassifier;
|
|
}
|
|
|
|
sp<InputDispatcherInterface> 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<InputWindowInfo>& infos,
|
|
const sp<ISetInputWindowsListener>& setInputWindowsListener) {
|
|
std::unordered_map<int32_t, std::vector<sp<InputWindowHandle>>> handlesPerDisplay;
|
|
|
|
std::vector<sp<InputWindowHandle>> handles;
|
|
for (const auto& info : infos) {
|
|
handlesPerDisplay.emplace(info.displayId, std::vector<sp<InputWindowHandle>>());
|
|
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<std::unique_ptr<InputChannel>> 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<IBinder>& connectionToken) {
|
|
mDispatcher->removeInputChannel(connectionToken);
|
|
return binder::Status::ok();
|
|
}
|
|
|
|
status_t InputManager::dump(int fd, const Vector<String16>& 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
|