| /* |
| * This work is licensed under the terms of the GNU GPL, version 2 or |
| * (at your option) any later version. See the COPYING file in the |
| * top-level directory. |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "qapi/error.h" |
| #include "qemu/module.h" |
| #include "qemu/sockets.h" |
| |
| #include "hw/virtio/virtio.h" |
| #include "hw/qdev-properties.h" |
| #include "hw/virtio/virtio-input.h" |
| |
| #include <sys/ioctl.h> |
| #include "standard-headers/linux/input.h" |
| |
| /* ----------------------------------------------------------------- */ |
| |
| static struct virtio_input_config virtio_input_host_config[] = { |
| { /* empty list */ }, |
| }; |
| |
| static void virtio_input_host_event(void *opaque) |
| { |
| VirtIOInputHost *vih = opaque; |
| VirtIOInput *vinput = VIRTIO_INPUT(vih); |
| struct virtio_input_event virtio; |
| struct input_event evdev; |
| int rc; |
| |
| for (;;) { |
| rc = read(vih->fd, &evdev, sizeof(evdev)); |
| if (rc != sizeof(evdev)) { |
| break; |
| } |
| |
| virtio.type = cpu_to_le16(evdev.type); |
| virtio.code = cpu_to_le16(evdev.code); |
| virtio.value = cpu_to_le32(evdev.value); |
| virtio_input_send(vinput, &virtio); |
| } |
| } |
| |
| static void virtio_input_bits_config(VirtIOInputHost *vih, |
| int type, int count) |
| { |
| virtio_input_config bits; |
| int rc, i, size = 0; |
| |
| memset(&bits, 0, sizeof(bits)); |
| rc = ioctl(vih->fd, EVIOCGBIT(type, count/8), bits.u.bitmap); |
| if (rc < 0) { |
| return; |
| } |
| |
| for (i = 0; i < count/8; i++) { |
| if (bits.u.bitmap[i]) { |
| size = i+1; |
| } |
| } |
| if (size == 0) { |
| return; |
| } |
| |
| bits.select = VIRTIO_INPUT_CFG_EV_BITS; |
| bits.subsel = type; |
| bits.size = size; |
| virtio_input_add_config(VIRTIO_INPUT(vih), &bits); |
| } |
| |
| static void virtio_input_abs_config(VirtIOInputHost *vih, int axis) |
| { |
| virtio_input_config config; |
| struct input_absinfo absinfo; |
| int rc; |
| |
| rc = ioctl(vih->fd, EVIOCGABS(axis), &absinfo); |
| if (rc < 0) { |
| return; |
| } |
| |
| memset(&config, 0, sizeof(config)); |
| config.select = VIRTIO_INPUT_CFG_ABS_INFO; |
| config.subsel = axis; |
| config.size = sizeof(virtio_input_absinfo); |
| |
| config.u.abs.min = cpu_to_le32(absinfo.minimum); |
| config.u.abs.max = cpu_to_le32(absinfo.maximum); |
| config.u.abs.fuzz = cpu_to_le32(absinfo.fuzz); |
| config.u.abs.flat = cpu_to_le32(absinfo.flat); |
| config.u.abs.res = cpu_to_le32(absinfo.resolution); |
| |
| virtio_input_add_config(VIRTIO_INPUT(vih), &config); |
| } |
| |
| static void virtio_input_host_realize(DeviceState *dev, Error **errp) |
| { |
| VirtIOInputHost *vih = VIRTIO_INPUT_HOST(dev); |
| VirtIOInput *vinput = VIRTIO_INPUT(dev); |
| virtio_input_config id, *abs; |
| struct input_id ids; |
| int rc, ver, i, axis; |
| uint8_t byte; |
| |
| if (!vih->evdev) { |
| error_setg(errp, "evdev property is required"); |
| return; |
| } |
| |
| vih->fd = open(vih->evdev, O_RDWR); |
| if (vih->fd < 0) { |
| error_setg_file_open(errp, errno, vih->evdev); |
| return; |
| } |
| if (!g_unix_set_fd_nonblocking(vih->fd, true, NULL)) { |
| error_setg_errno(errp, errno, "Failed to set FD nonblocking"); |
| goto err_close; |
| } |
| |
| rc = ioctl(vih->fd, EVIOCGVERSION, &ver); |
| if (rc < 0) { |
| error_setg(errp, "%s: is not an evdev device", vih->evdev); |
| goto err_close; |
| } |
| |
| rc = ioctl(vih->fd, EVIOCGRAB, 1); |
| if (rc < 0) { |
| error_setg_errno(errp, errno, "%s: failed to get exclusive access", |
| vih->evdev); |
| goto err_close; |
| } |
| |
| memset(&id, 0, sizeof(id)); |
| ioctl(vih->fd, EVIOCGNAME(sizeof(id.u.string)-1), id.u.string); |
| id.select = VIRTIO_INPUT_CFG_ID_NAME; |
| id.size = strlen(id.u.string); |
| virtio_input_add_config(vinput, &id); |
| |
| if (ioctl(vih->fd, EVIOCGID, &ids) == 0) { |
| memset(&id, 0, sizeof(id)); |
| id.select = VIRTIO_INPUT_CFG_ID_DEVIDS; |
| id.size = sizeof(struct virtio_input_devids); |
| id.u.ids.bustype = cpu_to_le16(ids.bustype); |
| id.u.ids.vendor = cpu_to_le16(ids.vendor); |
| id.u.ids.product = cpu_to_le16(ids.product); |
| id.u.ids.version = cpu_to_le16(ids.version); |
| virtio_input_add_config(vinput, &id); |
| } |
| |
| virtio_input_bits_config(vih, EV_KEY, KEY_CNT); |
| virtio_input_bits_config(vih, EV_REL, REL_CNT); |
| virtio_input_bits_config(vih, EV_ABS, ABS_CNT); |
| virtio_input_bits_config(vih, EV_MSC, MSC_CNT); |
| virtio_input_bits_config(vih, EV_SW, SW_CNT); |
| virtio_input_bits_config(vih, EV_LED, LED_CNT); |
| |
| abs = virtio_input_find_config(VIRTIO_INPUT(vih), |
| VIRTIO_INPUT_CFG_EV_BITS, EV_ABS); |
| if (abs) { |
| for (i = 0; i < abs->size; i++) { |
| byte = abs->u.bitmap[i]; |
| axis = 8 * i; |
| while (byte) { |
| if (byte & 1) { |
| virtio_input_abs_config(vih, axis); |
| } |
| axis++; |
| byte >>= 1; |
| } |
| } |
| } |
| |
| qemu_set_fd_handler(vih->fd, virtio_input_host_event, NULL, vih); |
| return; |
| |
| err_close: |
| close(vih->fd); |
| vih->fd = -1; |
| return; |
| } |
| |
| static void virtio_input_host_unrealize(DeviceState *dev) |
| { |
| VirtIOInputHost *vih = VIRTIO_INPUT_HOST(dev); |
| |
| if (vih->fd > 0) { |
| qemu_set_fd_handler(vih->fd, NULL, NULL, NULL); |
| close(vih->fd); |
| } |
| } |
| |
| static void virtio_input_host_handle_status(VirtIOInput *vinput, |
| virtio_input_event *event) |
| { |
| VirtIOInputHost *vih = VIRTIO_INPUT_HOST(vinput); |
| struct input_event evdev; |
| struct timeval tval; |
| int rc; |
| |
| if (gettimeofday(&tval, NULL)) { |
| perror("virtio_input_host_handle_status: gettimeofday"); |
| return; |
| } |
| |
| evdev.input_event_sec = tval.tv_sec; |
| evdev.input_event_usec = tval.tv_usec; |
| evdev.type = le16_to_cpu(event->type); |
| evdev.code = le16_to_cpu(event->code); |
| evdev.value = le32_to_cpu(event->value); |
| |
| rc = write(vih->fd, &evdev, sizeof(evdev)); |
| if (rc == -1) { |
| perror("virtio_input_host_handle_status: write"); |
| } |
| } |
| |
| static const VMStateDescription vmstate_virtio_input_host = { |
| .name = "virtio-input-host", |
| .unmigratable = 1, |
| }; |
| |
| static Property virtio_input_host_properties[] = { |
| DEFINE_PROP_STRING("evdev", VirtIOInputHost, evdev), |
| DEFINE_PROP_END_OF_LIST(), |
| }; |
| |
| static void virtio_input_host_class_init(ObjectClass *klass, void *data) |
| { |
| VirtIOInputClass *vic = VIRTIO_INPUT_CLASS(klass); |
| DeviceClass *dc = DEVICE_CLASS(klass); |
| |
| dc->vmsd = &vmstate_virtio_input_host; |
| device_class_set_props(dc, virtio_input_host_properties); |
| vic->realize = virtio_input_host_realize; |
| vic->unrealize = virtio_input_host_unrealize; |
| vic->handle_status = virtio_input_host_handle_status; |
| } |
| |
| static void virtio_input_host_init(Object *obj) |
| { |
| VirtIOInput *vinput = VIRTIO_INPUT(obj); |
| |
| virtio_input_init_config(vinput, virtio_input_host_config); |
| } |
| |
| static const TypeInfo virtio_input_host_info = { |
| .name = TYPE_VIRTIO_INPUT_HOST, |
| .parent = TYPE_VIRTIO_INPUT, |
| .instance_size = sizeof(VirtIOInputHost), |
| .instance_init = virtio_input_host_init, |
| .class_init = virtio_input_host_class_init, |
| }; |
| |
| /* ----------------------------------------------------------------- */ |
| |
| static void virtio_register_types(void) |
| { |
| type_register_static(&virtio_input_host_info); |
| } |
| |
| type_init(virtio_register_types) |