blob: b96ee39e65a6623a48d546e6375e7a5b4fe560c5 [file] [log] [blame]
/*
* Copyright (C) 2017 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.
*/
#include "virtual_device_base.h"
#include <android-base/logging.h>
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <string.h>
#include <unistd.h>
#include "log/log.h"
using cuttlefish_input_service::VirtualDeviceBase;
namespace {
bool DoIoctl(int fd, int request, const uint32_t value) {
int rc = ioctl(fd, request, value);
if (rc < 0) {
SLOGE("ioctl failed (%s)", strerror(errno));
return false;
}
return true;
}
} // namespace
VirtualDeviceBase::VirtualDeviceBase(const char* device_name,
uint16_t product_id)
: device_name_(device_name),
bus_type_(BUS_USB),
vendor_id_(0x6006),
product_id_(product_id),
version_(1) {}
VirtualDeviceBase::~VirtualDeviceBase() {
if (fd_ >= 0) {
close(fd_);
fd_ = -1;
}
}
bool VirtualDeviceBase::SetUp() {
fd_ = open("/dev/uinput", O_WRONLY | O_NONBLOCK);
if (fd_ < 0) {
SLOGE("Failed to open /dev/uinput (%s)", strerror(errno));
return false;
}
strncpy(dev_.name, device_name_, sizeof(dev_.name));
dev_.id.bustype = bus_type_;
dev_.id.vendor = vendor_id_;
dev_.id.product = product_id_;
dev_.id.version = version_;
for (uint32_t evt_type : GetEventTypes()) {
if (!DoIoctl(fd_, UI_SET_EVBIT, evt_type)) {
SLOGE("Error setting event type: %" PRIu32, evt_type);
return false;
}
}
for (uint32_t key : GetKeys()) {
if (!DoIoctl(fd_, UI_SET_KEYBIT, key)) {
SLOGE("Error setting key: %" PRIu32, key);
return false;
}
}
for (uint32_t property : GetProperties()) {
if (!DoIoctl(fd_, UI_SET_PROPBIT, property)) {
SLOGE("Error setting property: %" PRIu32, property);
return false;
}
}
for (uint32_t abs : GetAbs()) {
if (!DoIoctl(fd_, UI_SET_ABSBIT, abs)) {
SLOGE("Error setting abs: %" PRIu32, abs);
return false;
}
}
if (write(fd_, &dev_, sizeof(dev_)) < 0) {
SLOGE("Unable to set input device info (%s)", strerror(errno));
return false;
}
if (ioctl(fd_, UI_DEV_CREATE) < 0) {
SLOGE("Unable to create input device (%s)", strerror(errno));
return false;
}
LOG(INFO) << "set up virtual device";
return true;
}
bool VirtualDeviceBase::EmitEvent(uint16_t type,
uint16_t code,
uint32_t value) {
struct input_event event {};
event.type = type;
event.code = code;
event.value = value;
if (write(fd_, &event, sizeof(event)) < static_cast<ssize_t>(sizeof(event))) {
SLOGE("Event write failed (%s): aborting", strerror(errno));
return false;
}
return true;
}
// By default devices have no event types, keys, properties or absolutes,
// subclasses can override this behavior if necessary.
const std::vector<const uint32_t>& VirtualDeviceBase::GetEventTypes() const {
static const std::vector<const uint32_t> evt_types{};
return evt_types;
}
const std::vector<const uint32_t>& VirtualDeviceBase::GetKeys() const {
static const std::vector<const uint32_t> keys{};
return keys;
}
const std::vector<const uint32_t>& VirtualDeviceBase::GetProperties() const {
static const std::vector<const uint32_t> properties{};
return properties;
}
const std::vector<const uint32_t>& VirtualDeviceBase::GetAbs() const {
static const std::vector<const uint32_t> abs{};
return abs;
}