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 7006a5edeSGerd Hoffmann #include "qemu-common.h" 8006a5edeSGerd Hoffmann #include "qemu/sockets.h" 9006a5edeSGerd Hoffmann 10006a5edeSGerd Hoffmann #include "hw/qdev.h" 11006a5edeSGerd Hoffmann #include "hw/virtio/virtio.h" 12006a5edeSGerd Hoffmann #include "hw/virtio/virtio-input.h" 13006a5edeSGerd Hoffmann 14*e2f6bac3SGerd Hoffmann #include <sys/ioctl.h> 15006a5edeSGerd Hoffmann #include "standard-headers/linux/input.h" 16006a5edeSGerd Hoffmann 17006a5edeSGerd Hoffmann /* ----------------------------------------------------------------- */ 18006a5edeSGerd Hoffmann 19006a5edeSGerd Hoffmann static struct virtio_input_config virtio_input_host_config[] = { 20006a5edeSGerd Hoffmann { /* empty list */ }, 21006a5edeSGerd Hoffmann }; 22006a5edeSGerd Hoffmann 23006a5edeSGerd Hoffmann static void virtio_input_host_event(void *opaque) 24006a5edeSGerd Hoffmann { 25006a5edeSGerd Hoffmann VirtIOInputHost *vih = opaque; 26006a5edeSGerd Hoffmann VirtIOInput *vinput = VIRTIO_INPUT(vih); 27006a5edeSGerd Hoffmann struct virtio_input_event virtio; 28006a5edeSGerd Hoffmann struct input_event evdev; 29006a5edeSGerd Hoffmann int rc; 30006a5edeSGerd Hoffmann 31006a5edeSGerd Hoffmann for (;;) { 32006a5edeSGerd Hoffmann rc = read(vih->fd, &evdev, sizeof(evdev)); 33006a5edeSGerd Hoffmann if (rc != sizeof(evdev)) { 34006a5edeSGerd Hoffmann break; 35006a5edeSGerd Hoffmann } 36006a5edeSGerd Hoffmann 37006a5edeSGerd Hoffmann virtio.type = cpu_to_le16(evdev.type); 38006a5edeSGerd Hoffmann virtio.code = cpu_to_le16(evdev.code); 39006a5edeSGerd Hoffmann virtio.value = cpu_to_le32(evdev.value); 40006a5edeSGerd Hoffmann virtio_input_send(vinput, &virtio); 41006a5edeSGerd Hoffmann } 42006a5edeSGerd Hoffmann } 43006a5edeSGerd Hoffmann 44006a5edeSGerd Hoffmann static void virtio_input_bits_config(VirtIOInputHost *vih, 45006a5edeSGerd Hoffmann int type, int count) 46006a5edeSGerd Hoffmann { 47006a5edeSGerd Hoffmann virtio_input_config bits; 48006a5edeSGerd Hoffmann int rc, i, size = 0; 49006a5edeSGerd Hoffmann 50006a5edeSGerd Hoffmann memset(&bits, 0, sizeof(bits)); 51006a5edeSGerd Hoffmann rc = ioctl(vih->fd, EVIOCGBIT(type, count/8), bits.u.bitmap); 52006a5edeSGerd Hoffmann if (rc < 0) { 53006a5edeSGerd Hoffmann return; 54006a5edeSGerd Hoffmann } 55006a5edeSGerd Hoffmann 56006a5edeSGerd Hoffmann for (i = 0; i < count/8; i++) { 57006a5edeSGerd Hoffmann if (bits.u.bitmap[i]) { 58006a5edeSGerd Hoffmann size = i+1; 59006a5edeSGerd Hoffmann } 60006a5edeSGerd Hoffmann } 61006a5edeSGerd Hoffmann if (size == 0) { 62006a5edeSGerd Hoffmann return; 63006a5edeSGerd Hoffmann } 64006a5edeSGerd Hoffmann 65006a5edeSGerd Hoffmann bits.select = VIRTIO_INPUT_CFG_EV_BITS; 66006a5edeSGerd Hoffmann bits.subsel = type; 67006a5edeSGerd Hoffmann bits.size = size; 68006a5edeSGerd Hoffmann virtio_input_add_config(VIRTIO_INPUT(vih), &bits); 69006a5edeSGerd Hoffmann } 70006a5edeSGerd Hoffmann 71006a5edeSGerd Hoffmann static void virtio_input_host_realize(DeviceState *dev, Error **errp) 72006a5edeSGerd Hoffmann { 73006a5edeSGerd Hoffmann VirtIOInputHost *vih = VIRTIO_INPUT_HOST(dev); 74006a5edeSGerd Hoffmann VirtIOInput *vinput = VIRTIO_INPUT(dev); 75006a5edeSGerd Hoffmann virtio_input_config id; 76006a5edeSGerd Hoffmann struct input_id ids; 77006a5edeSGerd Hoffmann int rc, ver; 78006a5edeSGerd Hoffmann 79006a5edeSGerd Hoffmann if (!vih->evdev) { 80006a5edeSGerd Hoffmann error_setg(errp, "evdev property is required"); 81006a5edeSGerd Hoffmann return; 82006a5edeSGerd Hoffmann } 83006a5edeSGerd Hoffmann 84006a5edeSGerd Hoffmann vih->fd = open(vih->evdev, O_RDWR); 85006a5edeSGerd Hoffmann if (vih->fd < 0) { 86006a5edeSGerd Hoffmann error_setg_file_open(errp, errno, vih->evdev); 87006a5edeSGerd Hoffmann return; 88006a5edeSGerd Hoffmann } 89006a5edeSGerd Hoffmann qemu_set_nonblock(vih->fd); 90006a5edeSGerd Hoffmann 91006a5edeSGerd Hoffmann rc = ioctl(vih->fd, EVIOCGVERSION, &ver); 92006a5edeSGerd Hoffmann if (rc < 0) { 93006a5edeSGerd Hoffmann error_setg(errp, "%s: is not an evdev device", vih->evdev); 94006a5edeSGerd Hoffmann goto err_close; 95006a5edeSGerd Hoffmann } 96006a5edeSGerd Hoffmann 97006a5edeSGerd Hoffmann rc = ioctl(vih->fd, EVIOCGRAB, 1); 98006a5edeSGerd Hoffmann if (rc < 0) { 99006a5edeSGerd Hoffmann error_setg_errno(errp, errno, "%s: failed to get exclusive access", 100006a5edeSGerd Hoffmann vih->evdev); 101006a5edeSGerd Hoffmann goto err_close; 102006a5edeSGerd Hoffmann } 103006a5edeSGerd Hoffmann 104006a5edeSGerd Hoffmann memset(&id, 0, sizeof(id)); 105006a5edeSGerd Hoffmann ioctl(vih->fd, EVIOCGNAME(sizeof(id.u.string)-1), id.u.string); 106006a5edeSGerd Hoffmann id.select = VIRTIO_INPUT_CFG_ID_NAME; 107006a5edeSGerd Hoffmann id.size = strlen(id.u.string); 108006a5edeSGerd Hoffmann virtio_input_add_config(vinput, &id); 109006a5edeSGerd Hoffmann 110006a5edeSGerd Hoffmann if (ioctl(vih->fd, EVIOCGID, &ids) == 0) { 111006a5edeSGerd Hoffmann memset(&id, 0, sizeof(id)); 112006a5edeSGerd Hoffmann id.select = VIRTIO_INPUT_CFG_ID_DEVIDS; 113006a5edeSGerd Hoffmann id.size = sizeof(struct virtio_input_devids); 114006a5edeSGerd Hoffmann id.u.ids.bustype = cpu_to_le16(ids.bustype); 115006a5edeSGerd Hoffmann id.u.ids.vendor = cpu_to_le16(ids.vendor); 116006a5edeSGerd Hoffmann id.u.ids.product = cpu_to_le16(ids.product); 117006a5edeSGerd Hoffmann id.u.ids.version = cpu_to_le16(ids.version); 118006a5edeSGerd Hoffmann virtio_input_add_config(vinput, &id); 119006a5edeSGerd Hoffmann } 120006a5edeSGerd Hoffmann 121006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_KEY, KEY_CNT); 122006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_REL, REL_CNT); 123006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_ABS, ABS_CNT); 124006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_MSC, MSC_CNT); 125006a5edeSGerd Hoffmann virtio_input_bits_config(vih, EV_SW, SW_CNT); 126006a5edeSGerd Hoffmann 127006a5edeSGerd Hoffmann qemu_set_fd_handler(vih->fd, virtio_input_host_event, NULL, vih); 128006a5edeSGerd Hoffmann return; 129006a5edeSGerd Hoffmann 130006a5edeSGerd Hoffmann err_close: 131006a5edeSGerd Hoffmann close(vih->fd); 132006a5edeSGerd Hoffmann vih->fd = -1; 133006a5edeSGerd Hoffmann return; 134006a5edeSGerd Hoffmann } 135006a5edeSGerd Hoffmann 136006a5edeSGerd Hoffmann static void virtio_input_host_unrealize(DeviceState *dev, Error **errp) 137006a5edeSGerd Hoffmann { 138006a5edeSGerd Hoffmann VirtIOInputHost *vih = VIRTIO_INPUT_HOST(dev); 139006a5edeSGerd Hoffmann 140006a5edeSGerd Hoffmann if (vih->fd > 0) { 141006a5edeSGerd Hoffmann qemu_set_fd_handler(vih->fd, NULL, NULL, NULL); 142006a5edeSGerd Hoffmann close(vih->fd); 143006a5edeSGerd Hoffmann } 144006a5edeSGerd Hoffmann } 145006a5edeSGerd Hoffmann 146006a5edeSGerd Hoffmann static const VMStateDescription vmstate_virtio_input_host = { 147006a5edeSGerd Hoffmann .name = "virtio-input-host", 148006a5edeSGerd Hoffmann .unmigratable = 1, 149006a5edeSGerd Hoffmann }; 150006a5edeSGerd Hoffmann 151006a5edeSGerd Hoffmann static Property virtio_input_host_properties[] = { 152006a5edeSGerd Hoffmann DEFINE_PROP_STRING("evdev", VirtIOInputHost, evdev), 153006a5edeSGerd Hoffmann DEFINE_PROP_END_OF_LIST(), 154006a5edeSGerd Hoffmann }; 155006a5edeSGerd Hoffmann 156006a5edeSGerd Hoffmann static void virtio_input_host_class_init(ObjectClass *klass, void *data) 157006a5edeSGerd Hoffmann { 158006a5edeSGerd Hoffmann VirtIOInputClass *vic = VIRTIO_INPUT_CLASS(klass); 159006a5edeSGerd Hoffmann DeviceClass *dc = DEVICE_CLASS(klass); 160006a5edeSGerd Hoffmann 161006a5edeSGerd Hoffmann dc->vmsd = &vmstate_virtio_input_host; 162006a5edeSGerd Hoffmann dc->props = virtio_input_host_properties; 163006a5edeSGerd Hoffmann vic->realize = virtio_input_host_realize; 164006a5edeSGerd Hoffmann vic->unrealize = virtio_input_host_unrealize; 165006a5edeSGerd Hoffmann } 166006a5edeSGerd Hoffmann 167006a5edeSGerd Hoffmann static void virtio_input_host_init(Object *obj) 168006a5edeSGerd Hoffmann { 169006a5edeSGerd Hoffmann VirtIOInput *vinput = VIRTIO_INPUT(obj); 170006a5edeSGerd Hoffmann 171006a5edeSGerd Hoffmann virtio_input_init_config(vinput, virtio_input_host_config); 172006a5edeSGerd Hoffmann } 173006a5edeSGerd Hoffmann 174006a5edeSGerd Hoffmann static const TypeInfo virtio_input_host_info = { 175006a5edeSGerd Hoffmann .name = TYPE_VIRTIO_INPUT_HOST, 176006a5edeSGerd Hoffmann .parent = TYPE_VIRTIO_INPUT, 177006a5edeSGerd Hoffmann .instance_size = sizeof(VirtIOInputHost), 178006a5edeSGerd Hoffmann .instance_init = virtio_input_host_init, 179006a5edeSGerd Hoffmann .class_init = virtio_input_host_class_init, 180006a5edeSGerd Hoffmann }; 181006a5edeSGerd Hoffmann 182006a5edeSGerd Hoffmann /* ----------------------------------------------------------------- */ 183006a5edeSGerd Hoffmann 184006a5edeSGerd Hoffmann static void virtio_register_types(void) 185006a5edeSGerd Hoffmann { 186006a5edeSGerd Hoffmann type_register_static(&virtio_input_host_info); 187006a5edeSGerd Hoffmann } 188006a5edeSGerd Hoffmann 189006a5edeSGerd Hoffmann type_init(virtio_register_types) 190