mirror of
https://gitlab.com/qemu-project/qemu
synced 2024-09-20 03:01:32 +00:00
virtio-input: send rel-wheel events for wheel buttons
qemu uses wheel-up/down button events for mouse wheel input, however linux applications typically want REL_WHEEL events. This fixes wheel with linux guests. Tested with X11/wayland, and windows virtio-input driver. Based on a patch from Marc. Added property to enable/disable wheel axis. Cc: Marc-André Lureau <marcandre.lureau@redhat.com> Signed-off-by: Gerd Hoffmann <kraxel@redhat.com> Message-id: 20170926113243.26081-1-kraxel@redhat.com
This commit is contained in:
parent
74083f9c01
commit
f4924974c7
|
@ -190,6 +190,7 @@ static void virtio_input_key_config(VirtIOInput *vinput,
|
||||||
static void virtio_input_handle_event(DeviceState *dev, QemuConsole *src,
|
static void virtio_input_handle_event(DeviceState *dev, QemuConsole *src,
|
||||||
InputEvent *evt)
|
InputEvent *evt)
|
||||||
{
|
{
|
||||||
|
VirtIOInputHID *vhid = VIRTIO_INPUT_HID(dev);
|
||||||
VirtIOInput *vinput = VIRTIO_INPUT(dev);
|
VirtIOInput *vinput = VIRTIO_INPUT(dev);
|
||||||
virtio_input_event event;
|
virtio_input_event event;
|
||||||
int qcode;
|
int qcode;
|
||||||
|
@ -215,7 +216,14 @@ static void virtio_input_handle_event(DeviceState *dev, QemuConsole *src,
|
||||||
break;
|
break;
|
||||||
case INPUT_EVENT_KIND_BTN:
|
case INPUT_EVENT_KIND_BTN:
|
||||||
btn = evt->u.btn.data;
|
btn = evt->u.btn.data;
|
||||||
if (keymap_button[btn->button]) {
|
if (vhid->wheel_axis && (btn->button == INPUT_BUTTON_WHEEL_UP ||
|
||||||
|
btn->button == INPUT_BUTTON_WHEEL_DOWN)) {
|
||||||
|
event.type = cpu_to_le16(EV_REL);
|
||||||
|
event.code = cpu_to_le16(REL_WHEEL);
|
||||||
|
event.value = cpu_to_le32(btn->button == INPUT_BUTTON_WHEEL_UP
|
||||||
|
? 1 : -1);
|
||||||
|
virtio_input_send(vinput, &event);
|
||||||
|
} else if (keymap_button[btn->button]) {
|
||||||
event.type = cpu_to_le16(EV_KEY);
|
event.type = cpu_to_le16(EV_KEY);
|
||||||
event.code = cpu_to_le16(keymap_button[btn->button]);
|
event.code = cpu_to_le16(keymap_button[btn->button]);
|
||||||
event.value = cpu_to_le32(btn->down ? 1 : 0);
|
event.value = cpu_to_le32(btn->down ? 1 : 0);
|
||||||
|
@ -407,7 +415,7 @@ static QemuInputHandler virtio_mouse_handler = {
|
||||||
.sync = virtio_input_handle_sync,
|
.sync = virtio_input_handle_sync,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct virtio_input_config virtio_mouse_config[] = {
|
static struct virtio_input_config virtio_mouse_config_v1[] = {
|
||||||
{
|
{
|
||||||
.select = VIRTIO_INPUT_CFG_ID_NAME,
|
.select = VIRTIO_INPUT_CFG_ID_NAME,
|
||||||
.size = sizeof(VIRTIO_ID_NAME_MOUSE),
|
.size = sizeof(VIRTIO_ID_NAME_MOUSE),
|
||||||
|
@ -432,13 +440,53 @@ static struct virtio_input_config virtio_mouse_config[] = {
|
||||||
{ /* end of list */ },
|
{ /* end of list */ },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct virtio_input_config virtio_mouse_config_v2[] = {
|
||||||
|
{
|
||||||
|
.select = VIRTIO_INPUT_CFG_ID_NAME,
|
||||||
|
.size = sizeof(VIRTIO_ID_NAME_MOUSE),
|
||||||
|
.u.string = VIRTIO_ID_NAME_MOUSE,
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_ID_DEVIDS,
|
||||||
|
.size = sizeof(struct virtio_input_devids),
|
||||||
|
.u.ids = {
|
||||||
|
.bustype = const_le16(BUS_VIRTUAL),
|
||||||
|
.vendor = const_le16(0x0627), /* same we use for usb hid devices */
|
||||||
|
.product = const_le16(0x0002),
|
||||||
|
.version = const_le16(0x0002),
|
||||||
|
},
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_EV_BITS,
|
||||||
|
.subsel = EV_REL,
|
||||||
|
.size = 2,
|
||||||
|
.u.bitmap = {
|
||||||
|
(1 << REL_X) | (1 << REL_Y),
|
||||||
|
(1 << (REL_WHEEL - 8))
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ /* end of list */ },
|
||||||
|
};
|
||||||
|
|
||||||
|
static Property virtio_mouse_properties[] = {
|
||||||
|
DEFINE_PROP_BOOL("wheel-axis", VirtIOInputHID, wheel_axis, true),
|
||||||
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
|
};
|
||||||
|
|
||||||
|
static void virtio_mouse_class_init(ObjectClass *klass, void *data)
|
||||||
|
{
|
||||||
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||||
|
|
||||||
|
dc->props = virtio_mouse_properties;
|
||||||
|
}
|
||||||
|
|
||||||
static void virtio_mouse_init(Object *obj)
|
static void virtio_mouse_init(Object *obj)
|
||||||
{
|
{
|
||||||
VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
|
VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
|
||||||
VirtIOInput *vinput = VIRTIO_INPUT(obj);
|
VirtIOInput *vinput = VIRTIO_INPUT(obj);
|
||||||
|
|
||||||
vhid->handler = &virtio_mouse_handler;
|
vhid->handler = &virtio_mouse_handler;
|
||||||
virtio_input_init_config(vinput, virtio_mouse_config);
|
virtio_input_init_config(vinput, vhid->wheel_axis
|
||||||
|
? virtio_mouse_config_v2
|
||||||
|
: virtio_mouse_config_v1);
|
||||||
virtio_input_key_config(vinput, keymap_button,
|
virtio_input_key_config(vinput, keymap_button,
|
||||||
ARRAY_SIZE(keymap_button));
|
ARRAY_SIZE(keymap_button));
|
||||||
}
|
}
|
||||||
|
@ -448,6 +496,7 @@ static const TypeInfo virtio_mouse_info = {
|
||||||
.parent = TYPE_VIRTIO_INPUT_HID,
|
.parent = TYPE_VIRTIO_INPUT_HID,
|
||||||
.instance_size = sizeof(VirtIOInputHID),
|
.instance_size = sizeof(VirtIOInputHID),
|
||||||
.instance_init = virtio_mouse_init,
|
.instance_init = virtio_mouse_init,
|
||||||
|
.class_init = virtio_mouse_class_init,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* ----------------------------------------------------------------- */
|
/* ----------------------------------------------------------------- */
|
||||||
|
@ -459,7 +508,7 @@ static QemuInputHandler virtio_tablet_handler = {
|
||||||
.sync = virtio_input_handle_sync,
|
.sync = virtio_input_handle_sync,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct virtio_input_config virtio_tablet_config[] = {
|
static struct virtio_input_config virtio_tablet_config_v1[] = {
|
||||||
{
|
{
|
||||||
.select = VIRTIO_INPUT_CFG_ID_NAME,
|
.select = VIRTIO_INPUT_CFG_ID_NAME,
|
||||||
.size = sizeof(VIRTIO_ID_NAME_TABLET),
|
.size = sizeof(VIRTIO_ID_NAME_TABLET),
|
||||||
|
@ -496,13 +545,72 @@ static struct virtio_input_config virtio_tablet_config[] = {
|
||||||
{ /* end of list */ },
|
{ /* end of list */ },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct virtio_input_config virtio_tablet_config_v2[] = {
|
||||||
|
{
|
||||||
|
.select = VIRTIO_INPUT_CFG_ID_NAME,
|
||||||
|
.size = sizeof(VIRTIO_ID_NAME_TABLET),
|
||||||
|
.u.string = VIRTIO_ID_NAME_TABLET,
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_ID_DEVIDS,
|
||||||
|
.size = sizeof(struct virtio_input_devids),
|
||||||
|
.u.ids = {
|
||||||
|
.bustype = const_le16(BUS_VIRTUAL),
|
||||||
|
.vendor = const_le16(0x0627), /* same we use for usb hid devices */
|
||||||
|
.product = const_le16(0x0003),
|
||||||
|
.version = const_le16(0x0002),
|
||||||
|
},
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_EV_BITS,
|
||||||
|
.subsel = EV_ABS,
|
||||||
|
.size = 1,
|
||||||
|
.u.bitmap = {
|
||||||
|
(1 << ABS_X) | (1 << ABS_Y),
|
||||||
|
},
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_EV_BITS,
|
||||||
|
.subsel = EV_REL,
|
||||||
|
.size = 2,
|
||||||
|
.u.bitmap = {
|
||||||
|
0,
|
||||||
|
(1 << (REL_WHEEL - 8))
|
||||||
|
},
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_ABS_INFO,
|
||||||
|
.subsel = ABS_X,
|
||||||
|
.size = sizeof(virtio_input_absinfo),
|
||||||
|
.u.abs.min = const_le32(INPUT_EVENT_ABS_MIN),
|
||||||
|
.u.abs.max = const_le32(INPUT_EVENT_ABS_MAX),
|
||||||
|
},{
|
||||||
|
.select = VIRTIO_INPUT_CFG_ABS_INFO,
|
||||||
|
.subsel = ABS_Y,
|
||||||
|
.size = sizeof(virtio_input_absinfo),
|
||||||
|
.u.abs.min = const_le32(INPUT_EVENT_ABS_MIN),
|
||||||
|
.u.abs.max = const_le32(INPUT_EVENT_ABS_MAX),
|
||||||
|
},
|
||||||
|
{ /* end of list */ },
|
||||||
|
};
|
||||||
|
|
||||||
|
static Property virtio_tablet_properties[] = {
|
||||||
|
DEFINE_PROP_BOOL("wheel-axis", VirtIOInputHID, wheel_axis, true),
|
||||||
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
|
};
|
||||||
|
|
||||||
|
static void virtio_tablet_class_init(ObjectClass *klass, void *data)
|
||||||
|
{
|
||||||
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||||
|
|
||||||
|
dc->props = virtio_tablet_properties;
|
||||||
|
}
|
||||||
|
|
||||||
static void virtio_tablet_init(Object *obj)
|
static void virtio_tablet_init(Object *obj)
|
||||||
{
|
{
|
||||||
VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
|
VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
|
||||||
VirtIOInput *vinput = VIRTIO_INPUT(obj);
|
VirtIOInput *vinput = VIRTIO_INPUT(obj);
|
||||||
|
|
||||||
vhid->handler = &virtio_tablet_handler;
|
vhid->handler = &virtio_tablet_handler;
|
||||||
virtio_input_init_config(vinput, virtio_tablet_config);
|
virtio_input_init_config(vinput, vhid->wheel_axis
|
||||||
|
? virtio_tablet_config_v2
|
||||||
|
: virtio_tablet_config_v1);
|
||||||
virtio_input_key_config(vinput, keymap_button,
|
virtio_input_key_config(vinput, keymap_button,
|
||||||
ARRAY_SIZE(keymap_button));
|
ARRAY_SIZE(keymap_button));
|
||||||
}
|
}
|
||||||
|
@ -512,6 +620,7 @@ static const TypeInfo virtio_tablet_info = {
|
||||||
.parent = TYPE_VIRTIO_INPUT_HID,
|
.parent = TYPE_VIRTIO_INPUT_HID,
|
||||||
.instance_size = sizeof(VirtIOInputHID),
|
.instance_size = sizeof(VirtIOInputHID),
|
||||||
.instance_init = virtio_tablet_init,
|
.instance_init = virtio_tablet_init,
|
||||||
|
.class_init = virtio_tablet_class_init,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* ----------------------------------------------------------------- */
|
/* ----------------------------------------------------------------- */
|
||||||
|
|
|
@ -2,7 +2,15 @@
|
||||||
#define HW_COMPAT_H
|
#define HW_COMPAT_H
|
||||||
|
|
||||||
#define HW_COMPAT_2_10 \
|
#define HW_COMPAT_2_10 \
|
||||||
/* empty */
|
{\
|
||||||
|
.driver = "virtio-mouse-device",\
|
||||||
|
.property = "wheel-axis",\
|
||||||
|
.value = "false",\
|
||||||
|
},{\
|
||||||
|
.driver = "virtio-tablet-device",\
|
||||||
|
.property = "wheel-axis",\
|
||||||
|
.value = "false",\
|
||||||
|
},
|
||||||
|
|
||||||
#define HW_COMPAT_2_9 \
|
#define HW_COMPAT_2_9 \
|
||||||
{\
|
{\
|
||||||
|
|
|
@ -89,6 +89,7 @@ struct VirtIOInputHID {
|
||||||
QemuInputHandler *handler;
|
QemuInputHandler *handler;
|
||||||
QemuInputHandlerState *hs;
|
QemuInputHandlerState *hs;
|
||||||
int ledstate;
|
int ledstate;
|
||||||
|
bool wheel_axis;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct VirtIOInputHost {
|
struct VirtIOInputHost {
|
||||||
|
|
Loading…
Reference in a new issue