Compare commits
8 Commits
pull-vnc-2
...
pull-input
Author | SHA1 | Date | |
---|---|---|---|
|
2e6a64cb8d | ||
|
2330e9e7cc | ||
|
d4df42c431 | ||
|
c80276b420 | ||
|
d7b7f526b1 | ||
|
96d7c0720e | ||
|
57a4e3b92b | ||
|
cde8dcbc92 |
@@ -25,16 +25,51 @@
|
|||||||
#include "qemu-common.h"
|
#include "qemu-common.h"
|
||||||
#include "sysemu/char.h"
|
#include "sysemu/char.h"
|
||||||
#include "ui/console.h"
|
#include "ui/console.h"
|
||||||
|
#include "ui/input.h"
|
||||||
|
|
||||||
#define MSMOUSE_LO6(n) ((n) & 0x3f)
|
#define MSMOUSE_LO6(n) ((n) & 0x3f)
|
||||||
#define MSMOUSE_HI2(n) (((n) & 0xc0) >> 6)
|
#define MSMOUSE_HI2(n) (((n) & 0xc0) >> 6)
|
||||||
|
|
||||||
static void msmouse_event(void *opaque,
|
typedef struct {
|
||||||
int dx, int dy, int dz, int buttons_state)
|
CharDriverState *chr;
|
||||||
{
|
QemuInputHandlerState *hs;
|
||||||
CharDriverState *chr = (CharDriverState *)opaque;
|
int axis[INPUT_AXIS__MAX];
|
||||||
|
bool btns[INPUT_BUTTON__MAX];
|
||||||
|
bool btnc[INPUT_BUTTON__MAX];
|
||||||
|
uint8_t outbuf[32];
|
||||||
|
int outlen;
|
||||||
|
} MouseState;
|
||||||
|
|
||||||
|
static void msmouse_chr_accept_input(CharDriverState *chr)
|
||||||
|
{
|
||||||
|
MouseState *mouse = chr->opaque;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
len = qemu_chr_be_can_write(chr);
|
||||||
|
if (len > mouse->outlen) {
|
||||||
|
len = mouse->outlen;
|
||||||
|
}
|
||||||
|
if (!len) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
qemu_chr_be_write(chr, mouse->outbuf, len);
|
||||||
|
mouse->outlen -= len;
|
||||||
|
if (mouse->outlen) {
|
||||||
|
memmove(mouse->outbuf, mouse->outbuf + len, mouse->outlen);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void msmouse_queue_event(MouseState *mouse)
|
||||||
|
{
|
||||||
unsigned char bytes[4] = { 0x40, 0x00, 0x00, 0x00 };
|
unsigned char bytes[4] = { 0x40, 0x00, 0x00, 0x00 };
|
||||||
|
int dx, dy, count = 3;
|
||||||
|
|
||||||
|
dx = mouse->axis[INPUT_AXIS_X];
|
||||||
|
mouse->axis[INPUT_AXIS_X] = 0;
|
||||||
|
|
||||||
|
dy = mouse->axis[INPUT_AXIS_Y];
|
||||||
|
mouse->axis[INPUT_AXIS_Y] = 0;
|
||||||
|
|
||||||
/* Movement deltas */
|
/* Movement deltas */
|
||||||
bytes[0] |= (MSMOUSE_HI2(dy) << 2) | MSMOUSE_HI2(dx);
|
bytes[0] |= (MSMOUSE_HI2(dy) << 2) | MSMOUSE_HI2(dx);
|
||||||
@@ -42,14 +77,54 @@ static void msmouse_event(void *opaque,
|
|||||||
bytes[2] |= MSMOUSE_LO6(dy);
|
bytes[2] |= MSMOUSE_LO6(dy);
|
||||||
|
|
||||||
/* Buttons */
|
/* Buttons */
|
||||||
bytes[0] |= (buttons_state & 0x01 ? 0x20 : 0x00);
|
bytes[0] |= (mouse->btns[INPUT_BUTTON_LEFT] ? 0x20 : 0x00);
|
||||||
bytes[0] |= (buttons_state & 0x02 ? 0x10 : 0x00);
|
bytes[0] |= (mouse->btns[INPUT_BUTTON_RIGHT] ? 0x10 : 0x00);
|
||||||
bytes[3] |= (buttons_state & 0x04 ? 0x20 : 0x00);
|
if (mouse->btns[INPUT_BUTTON_MIDDLE] ||
|
||||||
|
mouse->btnc[INPUT_BUTTON_MIDDLE]) {
|
||||||
|
bytes[3] |= (mouse->btns[INPUT_BUTTON_MIDDLE] ? 0x20 : 0x00);
|
||||||
|
mouse->btnc[INPUT_BUTTON_MIDDLE] = false;
|
||||||
|
count = 4;
|
||||||
|
}
|
||||||
|
|
||||||
/* We always send the packet of, so that we do not have to keep track
|
if (mouse->outlen <= sizeof(mouse->outbuf) - count) {
|
||||||
of previous state of the middle button. This can potentially confuse
|
memcpy(mouse->outbuf + mouse->outlen, bytes, count);
|
||||||
some very old drivers for two button mice though. */
|
mouse->outlen += count;
|
||||||
qemu_chr_be_write(chr, bytes, 4);
|
} else {
|
||||||
|
/* queue full -> drop event */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void msmouse_input_event(DeviceState *dev, QemuConsole *src,
|
||||||
|
InputEvent *evt)
|
||||||
|
{
|
||||||
|
MouseState *mouse = (MouseState *)dev;
|
||||||
|
InputMoveEvent *move;
|
||||||
|
InputBtnEvent *btn;
|
||||||
|
|
||||||
|
switch (evt->type) {
|
||||||
|
case INPUT_EVENT_KIND_REL:
|
||||||
|
move = evt->u.rel.data;
|
||||||
|
mouse->axis[move->axis] += move->value;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case INPUT_EVENT_KIND_BTN:
|
||||||
|
btn = evt->u.btn.data;
|
||||||
|
mouse->btns[btn->button] = btn->down;
|
||||||
|
mouse->btnc[btn->button] = true;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
/* keep gcc happy */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void msmouse_input_sync(DeviceState *dev)
|
||||||
|
{
|
||||||
|
MouseState *mouse = (MouseState *)dev;
|
||||||
|
|
||||||
|
msmouse_queue_event(mouse);
|
||||||
|
msmouse_chr_accept_input(mouse->chr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int msmouse_chr_write (struct CharDriverState *s, const uint8_t *buf, int len)
|
static int msmouse_chr_write (struct CharDriverState *s, const uint8_t *buf, int len)
|
||||||
@@ -60,26 +135,41 @@ static int msmouse_chr_write (struct CharDriverState *s, const uint8_t *buf, int
|
|||||||
|
|
||||||
static void msmouse_chr_close (struct CharDriverState *chr)
|
static void msmouse_chr_close (struct CharDriverState *chr)
|
||||||
{
|
{
|
||||||
g_free (chr);
|
MouseState *mouse = chr->opaque;
|
||||||
|
|
||||||
|
qemu_input_handler_unregister(mouse->hs);
|
||||||
|
g_free(mouse);
|
||||||
|
g_free(chr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static QemuInputHandler msmouse_handler = {
|
||||||
|
.name = "QEMU Microsoft Mouse",
|
||||||
|
.mask = INPUT_EVENT_MASK_BTN | INPUT_EVENT_MASK_REL,
|
||||||
|
.event = msmouse_input_event,
|
||||||
|
.sync = msmouse_input_sync,
|
||||||
|
};
|
||||||
|
|
||||||
static CharDriverState *qemu_chr_open_msmouse(const char *id,
|
static CharDriverState *qemu_chr_open_msmouse(const char *id,
|
||||||
ChardevBackend *backend,
|
ChardevBackend *backend,
|
||||||
ChardevReturn *ret,
|
ChardevReturn *ret,
|
||||||
Error **errp)
|
Error **errp)
|
||||||
{
|
{
|
||||||
ChardevCommon *common = backend->u.msmouse.data;
|
ChardevCommon *common = backend->u.msmouse.data;
|
||||||
|
MouseState *mouse;
|
||||||
CharDriverState *chr;
|
CharDriverState *chr;
|
||||||
|
|
||||||
chr = qemu_chr_alloc(common, errp);
|
chr = qemu_chr_alloc(common, errp);
|
||||||
if (!chr) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
chr->chr_write = msmouse_chr_write;
|
chr->chr_write = msmouse_chr_write;
|
||||||
chr->chr_close = msmouse_chr_close;
|
chr->chr_close = msmouse_chr_close;
|
||||||
|
chr->chr_accept_input = msmouse_chr_accept_input;
|
||||||
chr->explicit_be_open = true;
|
chr->explicit_be_open = true;
|
||||||
|
|
||||||
qemu_add_mouse_event_handler(msmouse_event, chr, 0, "QEMU Microsoft Mouse");
|
mouse = g_new0(MouseState, 1);
|
||||||
|
mouse->hs = qemu_input_handler_register((DeviceState *)mouse,
|
||||||
|
&msmouse_handler);
|
||||||
|
|
||||||
|
mouse->chr = chr;
|
||||||
|
chr->opaque = mouse;
|
||||||
|
|
||||||
return chr;
|
return chr;
|
||||||
}
|
}
|
||||||
|
@@ -27,6 +27,7 @@
|
|||||||
#include "ui/console.h"
|
#include "ui/console.h"
|
||||||
#include "qemu/timer.h"
|
#include "qemu/timer.h"
|
||||||
#include "hw/input/hid.h"
|
#include "hw/input/hid.h"
|
||||||
|
#include "trace.h"
|
||||||
|
|
||||||
#define HID_USAGE_ERROR_ROLLOVER 0x01
|
#define HID_USAGE_ERROR_ROLLOVER 0x01
|
||||||
#define HID_USAGE_POSTFAIL 0x02
|
#define HID_USAGE_POSTFAIL 0x02
|
||||||
@@ -234,7 +235,7 @@ static void hid_keyboard_event(DeviceState *dev, QemuConsole *src,
|
|||||||
key->down,
|
key->down,
|
||||||
scancodes);
|
scancodes);
|
||||||
if (hs->n + count > QUEUE_LENGTH) {
|
if (hs->n + count > QUEUE_LENGTH) {
|
||||||
fprintf(stderr, "usb-kbd: warning: key event queue full\n");
|
trace_hid_kbd_queue_full();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
for (i = 0; i < count; i++) {
|
for (i = 0; i < count; i++) {
|
||||||
|
@@ -23,3 +23,9 @@ milkymist_softusb_memory_write(uint32_t addr, uint32_t value) "addr %08x value %
|
|||||||
milkymist_softusb_mevt(uint8_t m) "m %d"
|
milkymist_softusb_mevt(uint8_t m) "m %d"
|
||||||
milkymist_softusb_kevt(uint8_t m) "m %d"
|
milkymist_softusb_kevt(uint8_t m) "m %d"
|
||||||
milkymist_softusb_pulse_irq(void) "Pulse IRQ"
|
milkymist_softusb_pulse_irq(void) "Pulse IRQ"
|
||||||
|
|
||||||
|
# hw/input/hid.c
|
||||||
|
hid_kbd_queue_full(void) "queue full"
|
||||||
|
|
||||||
|
# hw/input/virtio
|
||||||
|
virtio_input_queue_full(void) "queue full"
|
||||||
|
@@ -7,6 +7,7 @@
|
|||||||
#include "qemu/osdep.h"
|
#include "qemu/osdep.h"
|
||||||
#include "qapi/error.h"
|
#include "qapi/error.h"
|
||||||
#include "qemu/iov.h"
|
#include "qemu/iov.h"
|
||||||
|
#include "trace.h"
|
||||||
|
|
||||||
#include "hw/qdev.h"
|
#include "hw/qdev.h"
|
||||||
#include "hw/virtio/virtio.h"
|
#include "hw/virtio/virtio.h"
|
||||||
@@ -47,7 +48,7 @@ void virtio_input_send(VirtIOInput *vinput, virtio_input_event *event)
|
|||||||
virtqueue_get_avail_bytes(vinput->evt, &have, NULL, need, 0);
|
virtqueue_get_avail_bytes(vinput->evt, &have, NULL, need, 0);
|
||||||
if (have < need) {
|
if (have < need) {
|
||||||
vinput->qindex = 0;
|
vinput->qindex = 0;
|
||||||
fprintf(stderr, "%s: ENOSPC in vq, dropping events\n", __func__);
|
trace_virtio_input_queue_full();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
269
ui/input-linux.c
269
ui/input-linux.c
@@ -129,6 +129,17 @@ static int qemu_input_linux_to_qcode(unsigned int lnx)
|
|||||||
return linux_to_qcode[lnx];
|
return linux_to_qcode[lnx];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool linux_is_button(unsigned int lnx)
|
||||||
|
{
|
||||||
|
if (lnx < 0x100) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (lnx >= 0x160 && lnx < 0x2c0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
#define TYPE_INPUT_LINUX "input-linux"
|
#define TYPE_INPUT_LINUX "input-linux"
|
||||||
#define INPUT_LINUX(obj) \
|
#define INPUT_LINUX(obj) \
|
||||||
OBJECT_CHECK(InputLinux, (obj), TYPE_INPUT_LINUX)
|
OBJECT_CHECK(InputLinux, (obj), TYPE_INPUT_LINUX)
|
||||||
@@ -153,6 +164,12 @@ struct InputLinux {
|
|||||||
int keycount;
|
int keycount;
|
||||||
int wheel;
|
int wheel;
|
||||||
bool initialized;
|
bool initialized;
|
||||||
|
|
||||||
|
bool has_rel_x;
|
||||||
|
bool has_abs_x;
|
||||||
|
int num_keys;
|
||||||
|
int num_btns;
|
||||||
|
|
||||||
QTAILQ_ENTRY(InputLinux) next;
|
QTAILQ_ENTRY(InputLinux) next;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -188,71 +205,55 @@ static void input_linux_toggle_grab(InputLinux *il)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void input_linux_event_keyboard(void *opaque)
|
static void input_linux_handle_keyboard(InputLinux *il,
|
||||||
|
struct input_event *event)
|
||||||
{
|
{
|
||||||
InputLinux *il = opaque;
|
if (event->type == EV_KEY) {
|
||||||
struct input_event event;
|
if (event->value > 2 || (event->value > 1 && !il->repeat)) {
|
||||||
int rc;
|
/*
|
||||||
|
* ignore autorepeat + unknown key events
|
||||||
for (;;) {
|
* 0 == up, 1 == down, 2 == autorepeat, other == undefined
|
||||||
rc = read(il->fd, &event, sizeof(event));
|
*/
|
||||||
if (rc != sizeof(event)) {
|
return;
|
||||||
if (rc < 0 && errno != EAGAIN) {
|
}
|
||||||
fprintf(stderr, "%s: read: %s\n", __func__, strerror(errno));
|
if (event->code >= KEY_CNT) {
|
||||||
qemu_set_fd_handler(il->fd, NULL, NULL, NULL);
|
/*
|
||||||
close(il->fd);
|
* Should not happen. But better safe than sorry,
|
||||||
}
|
* and we make Coverity happy too.
|
||||||
break;
|
*/
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (event.type) {
|
/* keep track of key state */
|
||||||
case EV_KEY:
|
if (!il->keydown[event->code] && event->value) {
|
||||||
if (event.value > 2 || (event.value > 1 && !il->repeat)) {
|
il->keydown[event->code] = true;
|
||||||
/*
|
il->keycount++;
|
||||||
* ignore autorepeat + unknown key events
|
}
|
||||||
* 0 == up, 1 == down, 2 == autorepeat, other == undefined
|
if (il->keydown[event->code] && !event->value) {
|
||||||
*/
|
il->keydown[event->code] = false;
|
||||||
continue;
|
il->keycount--;
|
||||||
}
|
}
|
||||||
if (event.code >= KEY_CNT) {
|
|
||||||
/*
|
|
||||||
* Should not happen. But better safe than sorry,
|
|
||||||
* and we make Coverity happy too.
|
|
||||||
*/
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
/* keep track of key state */
|
|
||||||
if (!il->keydown[event.code] && event.value) {
|
|
||||||
il->keydown[event.code] = true;
|
|
||||||
il->keycount++;
|
|
||||||
}
|
|
||||||
if (il->keydown[event.code] && !event.value) {
|
|
||||||
il->keydown[event.code] = false;
|
|
||||||
il->keycount--;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* send event to guest when grab is active */
|
/* send event to guest when grab is active */
|
||||||
if (il->grab_active) {
|
if (il->grab_active) {
|
||||||
int qcode = qemu_input_linux_to_qcode(event.code);
|
int qcode = qemu_input_linux_to_qcode(event->code);
|
||||||
qemu_input_event_send_key_qcode(NULL, qcode, event.value);
|
qemu_input_event_send_key_qcode(NULL, qcode, event->value);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* hotkey -> record switch request ... */
|
/* hotkey -> record switch request ... */
|
||||||
if (il->keydown[KEY_LEFTCTRL] &&
|
if (il->keydown[KEY_LEFTCTRL] &&
|
||||||
il->keydown[KEY_RIGHTCTRL]) {
|
il->keydown[KEY_RIGHTCTRL]) {
|
||||||
il->grab_request = true;
|
il->grab_request = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ... and do the switch when all keys are lifted, so we
|
* ... and do the switch when all keys are lifted, so we
|
||||||
* confuse neither guest nor host with keys which seem to
|
* confuse neither guest nor host with keys which seem to
|
||||||
* be stuck due to missing key-up events.
|
* be stuck due to missing key-up events.
|
||||||
*/
|
*/
|
||||||
if (il->grab_request && !il->keycount) {
|
if (il->grab_request && !il->keycount) {
|
||||||
il->grab_request = false;
|
il->grab_request = false;
|
||||||
input_linux_toggle_grab(il);
|
input_linux_toggle_grab(il);
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -265,7 +266,59 @@ static void input_linux_event_mouse_button(int button)
|
|||||||
qemu_input_event_sync();
|
qemu_input_event_sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void input_linux_event_mouse(void *opaque)
|
static void input_linux_handle_mouse(InputLinux *il, struct input_event *event)
|
||||||
|
{
|
||||||
|
if (!il->grab_active) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (event->type) {
|
||||||
|
case EV_KEY:
|
||||||
|
switch (event->code) {
|
||||||
|
case BTN_LEFT:
|
||||||
|
qemu_input_queue_btn(NULL, INPUT_BUTTON_LEFT, event->value);
|
||||||
|
break;
|
||||||
|
case BTN_RIGHT:
|
||||||
|
qemu_input_queue_btn(NULL, INPUT_BUTTON_RIGHT, event->value);
|
||||||
|
break;
|
||||||
|
case BTN_MIDDLE:
|
||||||
|
qemu_input_queue_btn(NULL, INPUT_BUTTON_MIDDLE, event->value);
|
||||||
|
break;
|
||||||
|
case BTN_GEAR_UP:
|
||||||
|
qemu_input_queue_btn(NULL, INPUT_BUTTON_WHEEL_UP, event->value);
|
||||||
|
break;
|
||||||
|
case BTN_GEAR_DOWN:
|
||||||
|
qemu_input_queue_btn(NULL, INPUT_BUTTON_WHEEL_DOWN,
|
||||||
|
event->value);
|
||||||
|
break;
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
case EV_REL:
|
||||||
|
switch (event->code) {
|
||||||
|
case REL_X:
|
||||||
|
qemu_input_queue_rel(NULL, INPUT_AXIS_X, event->value);
|
||||||
|
break;
|
||||||
|
case REL_Y:
|
||||||
|
qemu_input_queue_rel(NULL, INPUT_AXIS_Y, event->value);
|
||||||
|
break;
|
||||||
|
case REL_WHEEL:
|
||||||
|
il->wheel = event->value;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case EV_SYN:
|
||||||
|
qemu_input_event_sync();
|
||||||
|
if (il->wheel != 0) {
|
||||||
|
input_linux_event_mouse_button((il->wheel > 0)
|
||||||
|
? INPUT_BUTTON_WHEEL_UP
|
||||||
|
: INPUT_BUTTON_WHEEL_DOWN);
|
||||||
|
il->wheel = 0;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void input_linux_event(void *opaque)
|
||||||
{
|
{
|
||||||
InputLinux *il = opaque;
|
InputLinux *il = opaque;
|
||||||
struct input_event event;
|
struct input_event event;
|
||||||
@@ -282,54 +335,11 @@ static void input_linux_event_mouse(void *opaque)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* only send event to guest when grab is active */
|
if (il->num_keys) {
|
||||||
if (!il->grab_active) {
|
input_linux_handle_keyboard(il, &event);
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
if (il->has_rel_x && il->num_btns) {
|
||||||
switch (event.type) {
|
input_linux_handle_mouse(il, &event);
|
||||||
case EV_KEY:
|
|
||||||
switch (event.code) {
|
|
||||||
case BTN_LEFT:
|
|
||||||
qemu_input_queue_btn(NULL, INPUT_BUTTON_LEFT, event.value);
|
|
||||||
break;
|
|
||||||
case BTN_RIGHT:
|
|
||||||
qemu_input_queue_btn(NULL, INPUT_BUTTON_RIGHT, event.value);
|
|
||||||
break;
|
|
||||||
case BTN_MIDDLE:
|
|
||||||
qemu_input_queue_btn(NULL, INPUT_BUTTON_MIDDLE, event.value);
|
|
||||||
break;
|
|
||||||
case BTN_GEAR_UP:
|
|
||||||
qemu_input_queue_btn(NULL, INPUT_BUTTON_WHEEL_UP, event.value);
|
|
||||||
break;
|
|
||||||
case BTN_GEAR_DOWN:
|
|
||||||
qemu_input_queue_btn(NULL, INPUT_BUTTON_WHEEL_DOWN,
|
|
||||||
event.value);
|
|
||||||
break;
|
|
||||||
};
|
|
||||||
break;
|
|
||||||
case EV_REL:
|
|
||||||
switch (event.code) {
|
|
||||||
case REL_X:
|
|
||||||
qemu_input_queue_rel(NULL, INPUT_AXIS_X, event.value);
|
|
||||||
break;
|
|
||||||
case REL_Y:
|
|
||||||
qemu_input_queue_rel(NULL, INPUT_AXIS_Y, event.value);
|
|
||||||
break;
|
|
||||||
case REL_WHEEL:
|
|
||||||
il->wheel = event.value;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case EV_SYN:
|
|
||||||
qemu_input_event_sync();
|
|
||||||
if (il->wheel != 0) {
|
|
||||||
input_linux_event_mouse_button((il->wheel > 0)
|
|
||||||
? INPUT_BUTTON_WHEEL_UP
|
|
||||||
: INPUT_BUTTON_WHEEL_DOWN);
|
|
||||||
il->wheel = 0;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -337,7 +347,8 @@ static void input_linux_event_mouse(void *opaque)
|
|||||||
static void input_linux_complete(UserCreatable *uc, Error **errp)
|
static void input_linux_complete(UserCreatable *uc, Error **errp)
|
||||||
{
|
{
|
||||||
InputLinux *il = INPUT_LINUX(uc);
|
InputLinux *il = INPUT_LINUX(uc);
|
||||||
uint32_t evtmap, relmap, absmap;
|
uint8_t evtmap, relmap, absmap, keymap[KEY_CNT / 8];
|
||||||
|
unsigned int i;
|
||||||
int rc, ver;
|
int rc, ver;
|
||||||
|
|
||||||
if (!il->evdev) {
|
if (!il->evdev) {
|
||||||
@@ -365,36 +376,36 @@ static void input_linux_complete(UserCreatable *uc, Error **errp)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (evtmap & (1 << EV_REL)) {
|
if (evtmap & (1 << EV_REL)) {
|
||||||
|
relmap = 0;
|
||||||
rc = ioctl(il->fd, EVIOCGBIT(EV_REL, sizeof(relmap)), &relmap);
|
rc = ioctl(il->fd, EVIOCGBIT(EV_REL, sizeof(relmap)), &relmap);
|
||||||
if (rc < 0) {
|
if (relmap & (1 << REL_X)) {
|
||||||
relmap = 0;
|
il->has_rel_x = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (evtmap & (1 << EV_ABS)) {
|
if (evtmap & (1 << EV_ABS)) {
|
||||||
ioctl(il->fd, EVIOCGBIT(EV_ABS, sizeof(absmap)), &absmap);
|
absmap = 0;
|
||||||
if (rc < 0) {
|
rc = ioctl(il->fd, EVIOCGBIT(EV_ABS, sizeof(absmap)), &absmap);
|
||||||
absmap = 0;
|
if (absmap & (1 << ABS_X)) {
|
||||||
|
il->has_abs_x = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((evtmap & (1 << EV_REL)) &&
|
if (evtmap & (1 << EV_KEY)) {
|
||||||
(relmap & (1 << REL_X))) {
|
memset(keymap, 0, sizeof(keymap));
|
||||||
/* has relative x axis -> assume mouse */
|
rc = ioctl(il->fd, EVIOCGBIT(EV_KEY, sizeof(keymap)), keymap);
|
||||||
qemu_set_fd_handler(il->fd, input_linux_event_mouse, NULL, il);
|
for (i = 0; i < KEY_CNT; i++) {
|
||||||
} else if ((evtmap & (1 << EV_ABS)) &&
|
if (keymap[i / 8] & (1 << (i % 8))) {
|
||||||
(absmap & (1 << ABS_X))) {
|
if (linux_is_button(i)) {
|
||||||
/* has absolute x axis -> not supported */
|
il->num_btns++;
|
||||||
error_setg(errp, "tablet/touchscreen not supported");
|
} else {
|
||||||
goto err_close;
|
il->num_keys++;
|
||||||
} else if (evtmap & (1 << EV_KEY)) {
|
}
|
||||||
/* has keys/buttons (and no x axis) -> assume keyboard */
|
}
|
||||||
qemu_set_fd_handler(il->fd, input_linux_event_keyboard, NULL, il);
|
}
|
||||||
} else {
|
|
||||||
/* Huh? What is this? */
|
|
||||||
error_setg(errp, "unknown kind of input device");
|
|
||||||
goto err_close;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
qemu_set_fd_handler(il->fd, input_linux_event, NULL, il);
|
||||||
input_linux_toggle_grab(il);
|
input_linux_toggle_grab(il);
|
||||||
QTAILQ_INSERT_TAIL(&inputs, il, next);
|
QTAILQ_INSERT_TAIL(&inputs, il, next);
|
||||||
il->initialized = true;
|
il->initialized = true;
|
||||||
|
@@ -349,7 +349,7 @@ tight_detect_smooth_image(VncState *vs, int w, int h)
|
|||||||
tight_fill_palette##bpp(VncState *vs, int x, int y, \
|
tight_fill_palette##bpp(VncState *vs, int x, int y, \
|
||||||
int max, size_t count, \
|
int max, size_t count, \
|
||||||
uint32_t *bg, uint32_t *fg, \
|
uint32_t *bg, uint32_t *fg, \
|
||||||
VncPalette *palette) { \
|
VncPalette **palette) { \
|
||||||
uint##bpp##_t *data; \
|
uint##bpp##_t *data; \
|
||||||
uint##bpp##_t c0, c1, ci; \
|
uint##bpp##_t c0, c1, ci; \
|
||||||
int i, n0, n1; \
|
int i, n0, n1; \
|
||||||
@@ -396,23 +396,23 @@ tight_detect_smooth_image(VncState *vs, int w, int h)
|
|||||||
return 0; \
|
return 0; \
|
||||||
} \
|
} \
|
||||||
\
|
\
|
||||||
palette_init(palette, max, bpp); \
|
*palette = palette_new(max, bpp); \
|
||||||
palette_put(palette, c0); \
|
palette_put(*palette, c0); \
|
||||||
palette_put(palette, c1); \
|
palette_put(*palette, c1); \
|
||||||
palette_put(palette, ci); \
|
palette_put(*palette, ci); \
|
||||||
\
|
\
|
||||||
for (i++; i < count; i++) { \
|
for (i++; i < count; i++) { \
|
||||||
if (data[i] == ci) { \
|
if (data[i] == ci) { \
|
||||||
continue; \
|
continue; \
|
||||||
} else { \
|
} else { \
|
||||||
ci = data[i]; \
|
ci = data[i]; \
|
||||||
if (!palette_put(palette, (uint32_t)ci)) { \
|
if (!palette_put(*palette, (uint32_t)ci)) { \
|
||||||
return 0; \
|
return 0; \
|
||||||
} \
|
} \
|
||||||
} \
|
} \
|
||||||
} \
|
} \
|
||||||
\
|
\
|
||||||
return palette_size(palette); \
|
return palette_size(*palette); \
|
||||||
}
|
}
|
||||||
|
|
||||||
DEFINE_FILL_PALETTE_FUNCTION(8)
|
DEFINE_FILL_PALETTE_FUNCTION(8)
|
||||||
@@ -421,7 +421,7 @@ DEFINE_FILL_PALETTE_FUNCTION(32)
|
|||||||
|
|
||||||
static int tight_fill_palette(VncState *vs, int x, int y,
|
static int tight_fill_palette(VncState *vs, int x, int y,
|
||||||
size_t count, uint32_t *bg, uint32_t *fg,
|
size_t count, uint32_t *bg, uint32_t *fg,
|
||||||
VncPalette *palette)
|
VncPalette **palette)
|
||||||
{
|
{
|
||||||
int max;
|
int max;
|
||||||
|
|
||||||
@@ -1457,11 +1457,9 @@ static int send_sub_rect_jpeg(VncState *vs, int x, int y, int w, int h,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static __thread VncPalette color_count_palette;
|
|
||||||
|
|
||||||
static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
|
static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
|
||||||
{
|
{
|
||||||
VncPalette *palette = &color_count_palette;
|
VncPalette *palette = NULL;
|
||||||
uint32_t bg = 0, fg = 0;
|
uint32_t bg = 0, fg = 0;
|
||||||
int colors;
|
int colors;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
@@ -1490,7 +1488,7 @@ static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
colors = tight_fill_palette(vs, x, y, w * h, &bg, &fg, palette);
|
colors = tight_fill_palette(vs, x, y, w * h, &bg, &fg, &palette);
|
||||||
|
|
||||||
#ifdef CONFIG_VNC_JPEG
|
#ifdef CONFIG_VNC_JPEG
|
||||||
if (allow_jpeg && vs->tight.quality != (uint8_t)-1) {
|
if (allow_jpeg && vs->tight.quality != (uint8_t)-1) {
|
||||||
@@ -1503,6 +1501,7 @@ static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
|
|||||||
ret = send_sub_rect_nojpeg(vs, x, y, w, h, bg, fg, colors, palette);
|
ret = send_sub_rect_nojpeg(vs, x, y, w, h, bg, fg, colors, palette);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
palette_destroy(palette);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
15
ui/vnc.c
15
ui/vnc.c
@@ -1025,7 +1025,7 @@ static int find_and_clear_dirty_height(VncState *vs,
|
|||||||
static int vnc_update_client(VncState *vs, int has_dirty, bool sync)
|
static int vnc_update_client(VncState *vs, int has_dirty, bool sync)
|
||||||
{
|
{
|
||||||
vs->has_dirty += has_dirty;
|
vs->has_dirty += has_dirty;
|
||||||
if (vs->need_update && !vs->disconnecting) {
|
if (vs->need_update && vs->ioc != NULL) {
|
||||||
VncDisplay *vd = vs->vd;
|
VncDisplay *vd = vs->vd;
|
||||||
VncJob *job;
|
VncJob *job;
|
||||||
int y;
|
int y;
|
||||||
@@ -1436,9 +1436,8 @@ static void vnc_jobs_bh(void *opaque)
|
|||||||
* First function called whenever there is more data to be read from
|
* First function called whenever there is more data to be read from
|
||||||
* the client socket. Will delegate actual work according to whether
|
* the client socket. Will delegate actual work according to whether
|
||||||
* SASL SSF layers are enabled (thus requiring decryption calls)
|
* SASL SSF layers are enabled (thus requiring decryption calls)
|
||||||
* Returns 0 on success, -1 if client disconnected
|
|
||||||
*/
|
*/
|
||||||
static int vnc_client_read(VncState *vs)
|
static void vnc_client_read(VncState *vs)
|
||||||
{
|
{
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
|
|
||||||
@@ -1451,9 +1450,8 @@ static int vnc_client_read(VncState *vs)
|
|||||||
if (!ret) {
|
if (!ret) {
|
||||||
if (vs->disconnecting) {
|
if (vs->disconnecting) {
|
||||||
vnc_disconnect_finish(vs);
|
vnc_disconnect_finish(vs);
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
return 0;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (vs->read_handler && vs->input.offset >= vs->read_handler_expect) {
|
while (vs->read_handler && vs->input.offset >= vs->read_handler_expect) {
|
||||||
@@ -1463,7 +1461,7 @@ static int vnc_client_read(VncState *vs)
|
|||||||
ret = vs->read_handler(vs, vs->input.buffer, len);
|
ret = vs->read_handler(vs, vs->input.buffer, len);
|
||||||
if (vs->disconnecting) {
|
if (vs->disconnecting) {
|
||||||
vnc_disconnect_finish(vs);
|
vnc_disconnect_finish(vs);
|
||||||
return -1;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
@@ -1472,7 +1470,6 @@ static int vnc_client_read(VncState *vs)
|
|||||||
vs->read_handler_expect = ret;
|
vs->read_handler_expect = ret;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean vnc_client_io(QIOChannel *ioc G_GNUC_UNUSED,
|
gboolean vnc_client_io(QIOChannel *ioc G_GNUC_UNUSED,
|
||||||
@@ -1480,9 +1477,7 @@ gboolean vnc_client_io(QIOChannel *ioc G_GNUC_UNUSED,
|
|||||||
{
|
{
|
||||||
VncState *vs = opaque;
|
VncState *vs = opaque;
|
||||||
if (condition & G_IO_IN) {
|
if (condition & G_IO_IN) {
|
||||||
if (vnc_client_read(vs) < 0) {
|
vnc_client_read(vs);
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (condition & G_IO_OUT) {
|
if (condition & G_IO_OUT) {
|
||||||
vnc_client_write(vs);
|
vnc_client_write(vs);
|
||||||
|
Reference in New Issue
Block a user