1006a5edeSGerd Hoffmann /* 2006a5edeSGerd Hoffmann * This work is licensed under the terms of the GNU GPL, version 2 or 3006a5edeSGerd Hoffmann * (at your option) any later version. See the COPYING file in the 4006a5edeSGerd Hoffmann * top-level directory. 5006a5edeSGerd Hoffmann */ 6006a5edeSGerd Hoffmann 79b8bfe21SPeter Maydell #include "qemu/osdep.h" 8da34e65cSMarkus Armbruster #include "qapi/error.h" 90b8fa32fSMarkus Armbruster #include "qemu/module.h" 10006a5edeSGerd Hoffmann #include "qemu/sockets.h" 11006a5edeSGerd Hoffmann 12006a5edeSGerd Hoffmann #include "hw/virtio/virtio.h" 13a27bd6c7SMarkus Armbruster #include "hw/qdev-properties.h" 14006a5edeSGerd Hoffmann #include "hw/virtio/virtio-input.h" 15006a5edeSGerd Hoffmann 16e2f6bac3SGerd Hoffmann #include <sys/ioctl.h> 17006a5edeSGerd Hoffmann #include "standard-headers/linux/input.h" 18006a5edeSGerd Hoffmann 19006a5edeSGerd Hoffmann /* ----------------------------------------------------------------- */ 20006a5edeSGerd Hoffmann 21006a5edeSGerd Hoffmann static struct virtio_input_config virtio_input_host_config[] = { 22006a5edeSGerd Hoffmann { /* empty list */ }, 23006a5edeSGerd Hoffmann }; 24006a5edeSGerd Hoffmann 25006a5edeSGerd Hoffmann static void virtio_input_host_event(void *opaque) 26006a5edeSGerd Hoffmann { 27006a5edeSGerd Hoffmann VirtIOInputHost *vih = opaque; 28006a5edeSGerd Hoffmann VirtIOInput *vinput = VIRTIO_INPUT(vih); 29006a5edeSGerd Hoffmann struct virtio_input_event virtio; 30006a5edeSGerd Hoffmann struct input_event evdev; 31006a5edeSGerd Hoffmann int rc; 32006a5edeSGerd Hoffmann 33006a5edeSGerd Hoffmann for (;;) { 34006a5edeSGerd Hoffmann rc = read(vih->fd, &evdev, sizeof(evdev)); 35006a5edeSGerd Hoffmann if (rc != sizeof(evdev)) { 36006a5edeSGerd Hoffmann break; 37006a5edeSGerd Hoffmann } 38006a5edeSGerd Hoffmann 39006a5edeSGerd Hoffmann virtio.type = cpu_to_le16(evdev.type); 40006a5edeSGerd Hoffmann virtio.code = cpu_to_le16(evdev.code); 41006a5edeSGerd Hoffmann virtio.value = cpu_to_le32(evdev.value); 42006a5edeSGerd Hoffmann virtio_input_send(vinput, &virtio); 43006a5edeSGerd Hoffmann } 44006a5edeSGerd Hoffmann } 45006a5edeSGerd Hoffmann 46006a5edeSGerd Hoffmann static void virtio_input_bits_config(VirtIOInputHost *vih, 47006a5edeSGerd Hoffmann int type, int count) 48006a5edeSGerd Hoffmann { 49006a5edeSGerd Hoffmann virtio_input_config bits; 50006a5edeSGerd Hoffmann int rc, i, size = 0; 51006a5edeSGerd Hoffmann 52006a5edeSGerd Hoffmann memset(&bits, 0, sizeof(bits)); 53006a5edeSGerd Hoffmann rc = ioctl(vih->fd, EVIOCGBIT(type, count/8), bits.u.bitmap); 54006a5edeSGerd Hoffmann if (rc < 0) { 55006a5edeSGerd Hoffmann return; 56006a5edeSGerd Hoffmann } 57006a5edeSGerd Hoffmann 58006a5edeSGerd Hoffmann for (i = 0; i < count/8; i++) { 59006a5edeSGerd Hoffmann if (bits.u.bitmap[i]) { 60006a5edeSGerd Hoffmann size = i+1; 61006a5edeSGerd Hoffmann } 62006a5edeSGerd Hoffmann } 63006a5edeSGerd Hoffmann if (size == 0) { 64006a5edeSGerd Hoffmann return; 65006a5edeSGerd Hoffmann } 66006a5edeSGerd Hoffmann 67006a5edeSGerd Hoffmann bits.select = VIRTIO_INPUT_CFG_EV_BITS; 68006a5edeSGerd Hoffmann bits.subsel = type; 69006a5edeSGerd Hoffmann bits.size = size; 70006a5edeSGerd Hoffmann virtio_input_add_config(VIRTIO_INPUT(vih), &bits); 71006a5edeSGerd Hoffmann } 72006a5edeSGerd Hoffmann 73b065e275SLadi Prosek static void virtio_input_abs_config(VirtIOInputHost *vih, int axis) 74b065e275SLadi Prosek { 75b065e275SLadi Prosek virtio_input_config config; 76b065e275SLadi Prosek struct input_absinfo absinfo; 77b065e275SLadi Prosek int rc; 78b065e275SLadi Prosek 79b065e275SLadi Prosek rc = ioctl(vih->fd, EVIOCGABS(axis), &absinfo); 80b065e275SLadi Prosek if (rc < 0) { 81b065e275SLadi Prosek return; 82b065e275SLadi Prosek } 83b065e275SLadi Prosek 84b065e275SLadi Prosek memset(&config, 0, sizeof(config)); 85b065e275SLadi Prosek config.select = VIRTIO_INPUT_CFG_ABS_INFO; 86b065e275SLadi Prosek config.subsel = axis; 87b065e275SLadi Prosek config.size = sizeof(virtio_input_absinfo); 88b065e275SLadi Prosek 89b065e275SLadi Prosek config.u.abs.min = cpu_to_le32(absinfo.minimum); 90b065e275SLadi Prosek config.u.abs.max = cpu_to_le32(absinfo.maximum); 91b065e275SLadi Prosek config.u.abs.fuzz = cpu_to_le32(absinfo.fuzz); 92b065e275SLadi Prosek config.u.abs.flat = cpu_to_le32(absinfo.flat); 93b065e275SLadi Prosek config.u.abs.res = cpu_to_le32(absinfo.resolution); 94b065e275SLadi Prosek 95b065e275SLadi Prosek virtio_input_add_config(VIRTIO_INPUT(vih), &config); 96b065e275SLadi Prosek } 97b065e275SLadi Prosek 98006a5edeSGerd Hoffmann static void virtio_input_host_realize(DeviceState *dev, Error **errp) 99006a5edeSGerd Hoffmann { 100006a5edeSGerd Hoffmann VirtIOInputHost *vih = VIRTIO_INPUT_HOST(dev); 101006a5edeSGerd Hoffmann VirtIOInput *vinput = VIRTIO_INPUT(dev); 102b065e275SLadi Prosek virtio_input_config id, *abs; 103006a5edeSGerd Hoffmann struct input_id ids; 104b065e275SLadi Prosek int rc, ver, i, axis; 105b065e275SLadi Prosek uint8_t byte; 106006a5edeSGerd Hoffmann 107006a5edeSGerd Hoffmann if (!vih->evdev) { 108006a5edeSGerd Hoffmann error_setg(errp, "evdev property is required"); 109006a5edeSGerd Hoffmann return; 110006a5edeSGerd Hoffmann } 111006a5edeSGerd Hoffmann 112006a5edeSGerd Hoffmann vih->fd = open(vih->evdev, O_RDWR); 113006a5edeSGerd Hoffmann if (vih->fd < 0) { 114006a5edeSGerd Hoffmann error_setg_file_open(errp, errno, vih->evdev); 115006a5edeSGerd Hoffmann return; 116006a5edeSGerd Hoffmann } 117*701544cfSMarc-André Lureau if (!g_unix_set_fd_nonblocking(vih->fd, true, NULL)) { 118*701544cfSMarc-André Lureau error_setg_errno(errp, errno, "Failed to set FD nonblocking"); 119*701544cfSMarc-André Lureau goto err_close; 120*701544cfSMarc-André Lureau } 121006a5edeSGerd Hoffmann 122006a5edeSGerd Hoffmann rc = ioctl(vih->fd, EVIOCGVERSION, &ver); 123006a5edeSGerd Hoffmann if (rc < 0) { 124006a5edeSGerd Hoffmann error_setg(errp, "%s: is not an evdev device", vih->evdev); 125006a5edeSGerd Hoffmann goto err_close; 126006a5edeSGerd Hoffmann } 127006a5edeSGerd Hoffmann 128006a5edeSGerd Hoffmann rc = ioctl(vih->fd, EVIOCGRAB, 1); 129006a5edeSGerd Hoffmann if (rc < 0) { 130006a5edeSGerd Hoffmann error_setg_errno(errp, errno, "%s: failed to get exclusive access", 131006a5edeSGerd Hoffmann vih->evdev); 132006a5edeSGerd Hoffmann goto err_close; 133006a5edeSGerd Hoffmann } 134006a5edeSGerd Hoffmann 135006a5edeSGerd Hoffmann memset(&id, 0, sizeof(id)); 136006a5edeSGerd Hoffmann ioctl(vih->fd, EVIOCGNAME(sizeof(id.u.string)-1), id.u.string); 137006a5edeSGerd Hoffmann id.select = VIRTIO_INPUT_CFG_ID_NAME; 138006a5edeSGerd Hoffmann id.size = strlen(id.u.string); 139006a5edeSGerd Hoffmann virtio_input_add_config(vinput, &id); 140006a5edeSGerd Hoffmann 141006a5edeSGerd Hoffmann if (ioctl(vih->fd, EVIOCGID, &ids) == 0) { 142006a5edeSGerd Hoffmann memset(&id, 0, sizeof(id)); 143006a5edeSGerd Hoffmann id.select = VIRTIO_INPUT_CFG_ID_DEVIDS; 144006a5edeSGerd Hoffmann id.size = sizeof(struct virtio_input_devids); 145006a5edeSGerd Hoffmann id.u.ids.bustype = cpu_to_le16(ids.bustype); 146006a5edeSGerd Hoffmann id.u.ids.vendor = cpu_to_le16(ids.vendor); 147006a5edeSGerd Hoffmann id.u.ids.product = cpu_to_le16(ids.product); 148006a5edeSGerd Hoffmann id.u.ids.version = cpu_to_le16(ids.version); 149006a5edeSGerd Hoffmann virtio_input_add_config(vinput, &id); 150006a5edeSGerd Hoffmann } 151006a5edeSGerd Hoffmann 152006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_KEY, KEY_CNT); 153006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_REL, REL_CNT); 154006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_ABS, ABS_CNT); 155006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_MSC, MSC_CNT); 156006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_SW, SW_CNT); 157848c4d44SLadi Prosek virtio_input_bits_config(vih, EV_LED, LED_CNT); 158006a5edeSGerd Hoffmann 159b065e275SLadi Prosek abs = virtio_input_find_config(VIRTIO_INPUT(vih), 160b065e275SLadi Prosek VIRTIO_INPUT_CFG_EV_BITS, EV_ABS); 161b065e275SLadi Prosek if (abs) { 162b065e275SLadi Prosek for (i = 0; i < abs->size; i++) { 163b065e275SLadi Prosek byte = abs->u.bitmap[i]; 164b065e275SLadi Prosek axis = 8 * i; 165b065e275SLadi Prosek while (byte) { 166b065e275SLadi Prosek if (byte & 1) { 167b065e275SLadi Prosek virtio_input_abs_config(vih, axis); 168b065e275SLadi Prosek } 169b065e275SLadi Prosek axis++; 170b065e275SLadi Prosek byte >>= 1; 171b065e275SLadi Prosek } 172b065e275SLadi Prosek } 173b065e275SLadi Prosek } 174b065e275SLadi Prosek 175006a5edeSGerd Hoffmann qemu_set_fd_handler(vih->fd, virtio_input_host_event, NULL, vih); 176006a5edeSGerd Hoffmann return; 177006a5edeSGerd Hoffmann 178006a5edeSGerd Hoffmann err_close: 179006a5edeSGerd Hoffmann close(vih->fd); 180006a5edeSGerd Hoffmann vih->fd = -1; 181006a5edeSGerd Hoffmann return; 182006a5edeSGerd Hoffmann } 183006a5edeSGerd Hoffmann 184b69c3c21SMarkus Armbruster static void virtio_input_host_unrealize(DeviceState *dev) 185006a5edeSGerd Hoffmann { 186006a5edeSGerd Hoffmann VirtIOInputHost *vih = VIRTIO_INPUT_HOST(dev); 187006a5edeSGerd Hoffmann 188006a5edeSGerd Hoffmann if (vih->fd > 0) { 189006a5edeSGerd Hoffmann qemu_set_fd_handler(vih->fd, NULL, NULL, NULL); 190006a5edeSGerd Hoffmann close(vih->fd); 191006a5edeSGerd Hoffmann } 192006a5edeSGerd Hoffmann } 193006a5edeSGerd Hoffmann 1941a782629SLadi Prosek static void virtio_input_host_handle_status(VirtIOInput *vinput, 1951a782629SLadi Prosek virtio_input_event *event) 1961a782629SLadi Prosek { 1971a782629SLadi Prosek VirtIOInputHost *vih = VIRTIO_INPUT_HOST(vinput); 1981a782629SLadi Prosek struct input_event evdev; 199f7a6df5fSFabrice Fontaine struct timeval tval; 2001a782629SLadi Prosek int rc; 2011a782629SLadi Prosek 202f7a6df5fSFabrice Fontaine if (gettimeofday(&tval, NULL)) { 2031a782629SLadi Prosek perror("virtio_input_host_handle_status: gettimeofday"); 2041a782629SLadi Prosek return; 2051a782629SLadi Prosek } 2061a782629SLadi Prosek 207f7a6df5fSFabrice Fontaine evdev.input_event_sec = tval.tv_sec; 208f7a6df5fSFabrice Fontaine evdev.input_event_usec = tval.tv_usec; 2091a782629SLadi Prosek evdev.type = le16_to_cpu(event->type); 2101a782629SLadi Prosek evdev.code = le16_to_cpu(event->code); 2111a782629SLadi Prosek evdev.value = le32_to_cpu(event->value); 2121a782629SLadi Prosek 2131a782629SLadi Prosek rc = write(vih->fd, &evdev, sizeof(evdev)); 2141a782629SLadi Prosek if (rc == -1) { 2151a782629SLadi Prosek perror("virtio_input_host_handle_status: write"); 2161a782629SLadi Prosek } 2171a782629SLadi Prosek } 2181a782629SLadi Prosek 219006a5edeSGerd Hoffmann static const VMStateDescription vmstate_virtio_input_host = { 220006a5edeSGerd Hoffmann .name = "virtio-input-host", 221006a5edeSGerd Hoffmann .unmigratable = 1, 222006a5edeSGerd Hoffmann }; 223006a5edeSGerd Hoffmann 224006a5edeSGerd Hoffmann static Property virtio_input_host_properties[] = { 225006a5edeSGerd Hoffmann DEFINE_PROP_STRING("evdev", VirtIOInputHost, evdev), 226006a5edeSGerd Hoffmann DEFINE_PROP_END_OF_LIST(), 227006a5edeSGerd Hoffmann }; 228006a5edeSGerd Hoffmann 229006a5edeSGerd Hoffmann static void virtio_input_host_class_init(ObjectClass *klass, void *data) 230006a5edeSGerd Hoffmann { 231006a5edeSGerd Hoffmann VirtIOInputClass *vic = VIRTIO_INPUT_CLASS(klass); 232006a5edeSGerd Hoffmann DeviceClass *dc = DEVICE_CLASS(klass); 233006a5edeSGerd Hoffmann 234006a5edeSGerd Hoffmann dc->vmsd = &vmstate_virtio_input_host; 2354f67d30bSMarc-André Lureau device_class_set_props(dc, virtio_input_host_properties); 236006a5edeSGerd Hoffmann vic->realize = virtio_input_host_realize; 237006a5edeSGerd Hoffmann vic->unrealize = virtio_input_host_unrealize; 2381a782629SLadi Prosek vic->handle_status = virtio_input_host_handle_status; 239006a5edeSGerd Hoffmann } 240006a5edeSGerd Hoffmann 241006a5edeSGerd Hoffmann static void virtio_input_host_init(Object *obj) 242006a5edeSGerd Hoffmann { 243006a5edeSGerd Hoffmann VirtIOInput *vinput = VIRTIO_INPUT(obj); 244006a5edeSGerd Hoffmann 245006a5edeSGerd Hoffmann virtio_input_init_config(vinput, virtio_input_host_config); 246006a5edeSGerd Hoffmann } 247006a5edeSGerd Hoffmann 248006a5edeSGerd Hoffmann static const TypeInfo virtio_input_host_info = { 249006a5edeSGerd Hoffmann .name = TYPE_VIRTIO_INPUT_HOST, 250006a5edeSGerd Hoffmann .parent = TYPE_VIRTIO_INPUT, 251006a5edeSGerd Hoffmann .instance_size = sizeof(VirtIOInputHost), 252006a5edeSGerd Hoffmann .instance_init = virtio_input_host_init, 253006a5edeSGerd Hoffmann .class_init = virtio_input_host_class_init, 254006a5edeSGerd Hoffmann }; 255006a5edeSGerd Hoffmann 256006a5edeSGerd Hoffmann /* ----------------------------------------------------------------- */ 257006a5edeSGerd Hoffmann 258006a5edeSGerd Hoffmann static void virtio_register_types(void) 259006a5edeSGerd Hoffmann { 260006a5edeSGerd Hoffmann type_register_static(&virtio_input_host_info); 261006a5edeSGerd Hoffmann } 262006a5edeSGerd Hoffmann 263006a5edeSGerd Hoffmann type_init(virtio_register_types) 264