提交 acde8376 编写于 作者: A Anthony Liguori

Merge remote-tracking branch 'spice/spice.v54' into staging

* spice/spice.v54:
  qxl: don't assert on guest create_guest_primary
  qxl: ioport_write: remove guest trigerrable abort
  qxl: qxl_add_memslot: remove guest trigerrable panics
  qxl: interface_notify_update: remove guest trigerrable abort
  qxl: cleanup s/__FUNCTION__/__func__/
  qxl: don't abort on guest trigerrable ring indices mismatch
  qxl: fix > 80 chars line
  qxl: replace panic with guest bug in qxl_track_command
  qxl: check for NULL return from qxl_phys2virt
  hw/qxl.c: qxl_phys2virt: replace panics with guest_bug
  spice_info: add mouse_mode
  spice: require spice-protocol >= 0.8.1
...@@ -2592,6 +2592,7 @@ EOF ...@@ -2592,6 +2592,7 @@ EOF
spice_cflags=$($pkg_config --cflags spice-protocol spice-server 2>/dev/null) spice_cflags=$($pkg_config --cflags spice-protocol spice-server 2>/dev/null)
spice_libs=$($pkg_config --libs spice-protocol spice-server 2>/dev/null) spice_libs=$($pkg_config --libs spice-protocol spice-server 2>/dev/null)
if $pkg_config --atleast-version=0.8.2 spice-server >/dev/null 2>&1 && \ if $pkg_config --atleast-version=0.8.2 spice-server >/dev/null 2>&1 && \
$pkg_config --atleast-version=0.8.1 spice-protocol > /dev/null 2>&1 && \
compile_prog "$spice_cflags" "$spice_libs" ; then compile_prog "$spice_cflags" "$spice_libs" ; then
spice="yes" spice="yes"
libs_softmmu="$libs_softmmu $spice_libs" libs_softmmu="$libs_softmmu $spice_libs"
......
...@@ -350,6 +350,8 @@ void hmp_info_spice(Monitor *mon) ...@@ -350,6 +350,8 @@ void hmp_info_spice(Monitor *mon)
} }
monitor_printf(mon, " auth: %s\n", info->auth); monitor_printf(mon, " auth: %s\n", info->auth);
monitor_printf(mon, " compiled: %s\n", info->compiled_version); monitor_printf(mon, " compiled: %s\n", info->compiled_version);
monitor_printf(mon, " mouse-mode: %s\n",
SpiceQueryMouseMode_lookup[info->mouse_mode]);
if (!info->has_channels || info->channels == NULL) { if (!info->has_channels || info->channels == NULL) {
monitor_printf(mon, "Channels: none\n"); monitor_printf(mon, "Channels: none\n");
......
...@@ -100,12 +100,15 @@ static const char *qxl_v2n(const char *n[], size_t l, int v) ...@@ -100,12 +100,15 @@ static const char *qxl_v2n(const char *n[], size_t l, int v)
} }
#define qxl_name(_list, _value) qxl_v2n(_list, ARRAY_SIZE(_list), _value) #define qxl_name(_list, _value) qxl_v2n(_list, ARRAY_SIZE(_list), _value)
static void qxl_log_image(PCIQXLDevice *qxl, QXLPHYSICAL addr, int group_id) static int qxl_log_image(PCIQXLDevice *qxl, QXLPHYSICAL addr, int group_id)
{ {
QXLImage *image; QXLImage *image;
QXLImageDescriptor *desc; QXLImageDescriptor *desc;
image = qxl_phys2virt(qxl, addr, group_id); image = qxl_phys2virt(qxl, addr, group_id);
if (!image) {
return 1;
}
desc = &image->descriptor; desc = &image->descriptor;
fprintf(stderr, " (id %" PRIx64 " type %d flags %d width %d height %d", fprintf(stderr, " (id %" PRIx64 " type %d flags %d width %d height %d",
desc->id, desc->type, desc->flags, desc->width, desc->height); desc->id, desc->type, desc->flags, desc->width, desc->height);
...@@ -120,6 +123,7 @@ static void qxl_log_image(PCIQXLDevice *qxl, QXLPHYSICAL addr, int group_id) ...@@ -120,6 +123,7 @@ static void qxl_log_image(PCIQXLDevice *qxl, QXLPHYSICAL addr, int group_id)
break; break;
} }
fprintf(stderr, ")"); fprintf(stderr, ")");
return 0;
} }
static void qxl_log_rect(QXLRect *rect) static void qxl_log_rect(QXLRect *rect)
...@@ -130,17 +134,24 @@ static void qxl_log_rect(QXLRect *rect) ...@@ -130,17 +134,24 @@ static void qxl_log_rect(QXLRect *rect)
rect->left, rect->top); rect->left, rect->top);
} }
static void qxl_log_cmd_draw_copy(PCIQXLDevice *qxl, QXLCopy *copy, int group_id) static int qxl_log_cmd_draw_copy(PCIQXLDevice *qxl, QXLCopy *copy,
int group_id)
{ {
int ret;
fprintf(stderr, " src %" PRIx64, fprintf(stderr, " src %" PRIx64,
copy->src_bitmap); copy->src_bitmap);
qxl_log_image(qxl, copy->src_bitmap, group_id); ret = qxl_log_image(qxl, copy->src_bitmap, group_id);
if (ret != 0) {
return ret;
}
fprintf(stderr, " area"); fprintf(stderr, " area");
qxl_log_rect(&copy->src_area); qxl_log_rect(&copy->src_area);
fprintf(stderr, " rop %d", copy->rop_descriptor); fprintf(stderr, " rop %d", copy->rop_descriptor);
return 0;
} }
static void qxl_log_cmd_draw(PCIQXLDevice *qxl, QXLDrawable *draw, int group_id) static int qxl_log_cmd_draw(PCIQXLDevice *qxl, QXLDrawable *draw, int group_id)
{ {
fprintf(stderr, ": surface_id %d type %s effect %s", fprintf(stderr, ": surface_id %d type %s effect %s",
draw->surface_id, draw->surface_id,
...@@ -148,12 +159,13 @@ static void qxl_log_cmd_draw(PCIQXLDevice *qxl, QXLDrawable *draw, int group_id) ...@@ -148,12 +159,13 @@ static void qxl_log_cmd_draw(PCIQXLDevice *qxl, QXLDrawable *draw, int group_id)
qxl_name(qxl_draw_effect, draw->effect)); qxl_name(qxl_draw_effect, draw->effect));
switch (draw->type) { switch (draw->type) {
case QXL_DRAW_COPY: case QXL_DRAW_COPY:
qxl_log_cmd_draw_copy(qxl, &draw->u.copy, group_id); return qxl_log_cmd_draw_copy(qxl, &draw->u.copy, group_id);
break; break;
} }
return 0;
} }
static void qxl_log_cmd_draw_compat(PCIQXLDevice *qxl, QXLCompatDrawable *draw, static int qxl_log_cmd_draw_compat(PCIQXLDevice *qxl, QXLCompatDrawable *draw,
int group_id) int group_id)
{ {
fprintf(stderr, ": type %s effect %s", fprintf(stderr, ": type %s effect %s",
...@@ -166,9 +178,10 @@ static void qxl_log_cmd_draw_compat(PCIQXLDevice *qxl, QXLCompatDrawable *draw, ...@@ -166,9 +178,10 @@ static void qxl_log_cmd_draw_compat(PCIQXLDevice *qxl, QXLCompatDrawable *draw,
} }
switch (draw->type) { switch (draw->type) {
case QXL_DRAW_COPY: case QXL_DRAW_COPY:
qxl_log_cmd_draw_copy(qxl, &draw->u.copy, group_id); return qxl_log_cmd_draw_copy(qxl, &draw->u.copy, group_id);
break; break;
} }
return 0;
} }
static void qxl_log_cmd_surface(PCIQXLDevice *qxl, QXLSurfaceCmd *cmd) static void qxl_log_cmd_surface(PCIQXLDevice *qxl, QXLSurfaceCmd *cmd)
...@@ -189,7 +202,7 @@ static void qxl_log_cmd_surface(PCIQXLDevice *qxl, QXLSurfaceCmd *cmd) ...@@ -189,7 +202,7 @@ static void qxl_log_cmd_surface(PCIQXLDevice *qxl, QXLSurfaceCmd *cmd)
} }
} }
void qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id) int qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id)
{ {
QXLCursor *cursor; QXLCursor *cursor;
...@@ -203,6 +216,9 @@ void qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id) ...@@ -203,6 +216,9 @@ void qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id)
cmd->u.set.visible ? "yes" : "no", cmd->u.set.visible ? "yes" : "no",
cmd->u.set.shape); cmd->u.set.shape);
cursor = qxl_phys2virt(qxl, cmd->u.set.shape, group_id); cursor = qxl_phys2virt(qxl, cmd->u.set.shape, group_id);
if (!cursor) {
return 1;
}
fprintf(stderr, " type %s size %dx%d hot-spot +%d+%d" fprintf(stderr, " type %s size %dx%d hot-spot +%d+%d"
" unique 0x%" PRIx64 " data-size %d", " unique 0x%" PRIx64 " data-size %d",
qxl_name(spice_cursor_type, cursor->header.type), qxl_name(spice_cursor_type, cursor->header.type),
...@@ -214,15 +230,17 @@ void qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id) ...@@ -214,15 +230,17 @@ void qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id)
fprintf(stderr, " +%d+%d", cmd->u.position.x, cmd->u.position.y); fprintf(stderr, " +%d+%d", cmd->u.position.x, cmd->u.position.y);
break; break;
} }
return 0;
} }
void qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext) int qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext)
{ {
bool compat = ext->flags & QXL_COMMAND_FLAG_COMPAT; bool compat = ext->flags & QXL_COMMAND_FLAG_COMPAT;
void *data; void *data;
int ret;
if (!qxl->cmdlog) { if (!qxl->cmdlog) {
return; return 0;
} }
fprintf(stderr, "%" PRId64 " qxl-%d/%s:", qemu_get_clock_ns(vm_clock), fprintf(stderr, "%" PRId64 " qxl-%d/%s:", qemu_get_clock_ns(vm_clock),
qxl->id, ring); qxl->id, ring);
...@@ -231,12 +249,18 @@ void qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext) ...@@ -231,12 +249,18 @@ void qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext)
compat ? "(compat)" : ""); compat ? "(compat)" : "");
data = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id); data = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id);
if (!data) {
return 1;
}
switch (ext->cmd.type) { switch (ext->cmd.type) {
case QXL_CMD_DRAW: case QXL_CMD_DRAW:
if (!compat) { if (!compat) {
qxl_log_cmd_draw(qxl, data, ext->group_id); ret = qxl_log_cmd_draw(qxl, data, ext->group_id);
} else { } else {
qxl_log_cmd_draw_compat(qxl, data, ext->group_id); ret = qxl_log_cmd_draw_compat(qxl, data, ext->group_id);
}
if (ret) {
return ret;
} }
break; break;
case QXL_CMD_SURFACE: case QXL_CMD_SURFACE:
...@@ -247,4 +271,5 @@ void qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext) ...@@ -247,4 +271,5 @@ void qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext)
break; break;
} }
fprintf(stderr, "\n"); fprintf(stderr, "\n");
return 0;
} }
...@@ -228,14 +228,18 @@ fail: ...@@ -228,14 +228,18 @@ fail:
/* called from spice server thread context only */ /* called from spice server thread context only */
void qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext) int qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext)
{ {
QXLCursorCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id); QXLCursorCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id);
QXLCursor *cursor; QXLCursor *cursor;
QEMUCursor *c; QEMUCursor *c;
if (!cmd) {
return 1;
}
if (!qxl->ssd.ds->mouse_set || !qxl->ssd.ds->cursor_define) { if (!qxl->ssd.ds->mouse_set || !qxl->ssd.ds->cursor_define) {
return; return 0;
} }
if (qxl->debug > 1 && cmd->type != QXL_CURSOR_MOVE) { if (qxl->debug > 1 && cmd->type != QXL_CURSOR_MOVE) {
...@@ -246,9 +250,12 @@ void qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext) ...@@ -246,9 +250,12 @@ void qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext)
switch (cmd->type) { switch (cmd->type) {
case QXL_CURSOR_SET: case QXL_CURSOR_SET:
cursor = qxl_phys2virt(qxl, cmd->u.set.shape, ext->group_id); cursor = qxl_phys2virt(qxl, cmd->u.set.shape, ext->group_id);
if (!cursor) {
return 1;
}
if (cursor->chunk.data_size != cursor->data_size) { if (cursor->chunk.data_size != cursor->data_size) {
fprintf(stderr, "%s: multiple chunks\n", __FUNCTION__); fprintf(stderr, "%s: multiple chunks\n", __FUNCTION__);
return; return 1;
} }
c = qxl_cursor(qxl, cursor); c = qxl_cursor(qxl, cursor);
if (c == NULL) { if (c == NULL) {
...@@ -270,4 +277,5 @@ void qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext) ...@@ -270,4 +277,5 @@ void qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext)
qemu_mutex_unlock(&qxl->ssd.lock); qemu_mutex_unlock(&qxl->ssd.lock);
break; break;
} }
return 0;
} }
...@@ -27,28 +27,42 @@ ...@@ -27,28 +27,42 @@
#include "qxl.h" #include "qxl.h"
/*
* NOTE: SPICE_RING_PROD_ITEM accesses memory on the pci bar and as
* such can be changed by the guest, so to avoid a guest trigerrable
* abort we just set qxl_guest_bug and set the return to NULL. Still
* it may happen as a result of emulator bug as well.
*/
#undef SPICE_RING_PROD_ITEM #undef SPICE_RING_PROD_ITEM
#define SPICE_RING_PROD_ITEM(r, ret) { \ #define SPICE_RING_PROD_ITEM(qxl, r, ret) { \
typeof(r) start = r; \ typeof(r) start = r; \
typeof(r) end = r + 1; \ typeof(r) end = r + 1; \
uint32_t prod = (r)->prod & SPICE_RING_INDEX_MASK(r); \ uint32_t prod = (r)->prod & SPICE_RING_INDEX_MASK(r); \
typeof(&(r)->items[prod]) m_item = &(r)->items[prod]; \ typeof(&(r)->items[prod]) m_item = &(r)->items[prod]; \
if (!((uint8_t*)m_item >= (uint8_t*)(start) && (uint8_t*)(m_item + 1) <= (uint8_t*)(end))) { \ if (!((uint8_t*)m_item >= (uint8_t*)(start) && (uint8_t*)(m_item + 1) <= (uint8_t*)(end))) { \
abort(); \ qxl_guest_bug(qxl, "SPICE_RING_PROD_ITEM indices mismatch " \
} \ "! %p <= %p < %p", (uint8_t *)start, \
(uint8_t *)m_item, (uint8_t *)end); \
ret = NULL; \
} else { \
ret = &m_item->el; \ ret = &m_item->el; \
} \
} }
#undef SPICE_RING_CONS_ITEM #undef SPICE_RING_CONS_ITEM
#define SPICE_RING_CONS_ITEM(r, ret) { \ #define SPICE_RING_CONS_ITEM(qxl, r, ret) { \
typeof(r) start = r; \ typeof(r) start = r; \
typeof(r) end = r + 1; \ typeof(r) end = r + 1; \
uint32_t cons = (r)->cons & SPICE_RING_INDEX_MASK(r); \ uint32_t cons = (r)->cons & SPICE_RING_INDEX_MASK(r); \
typeof(&(r)->items[cons]) m_item = &(r)->items[cons]; \ typeof(&(r)->items[cons]) m_item = &(r)->items[cons]; \
if (!((uint8_t*)m_item >= (uint8_t*)(start) && (uint8_t*)(m_item + 1) <= (uint8_t*)(end))) { \ if (!((uint8_t*)m_item >= (uint8_t*)(start) && (uint8_t*)(m_item + 1) <= (uint8_t*)(end))) { \
abort(); \ qxl_guest_bug(qxl, "SPICE_RING_CONS_ITEM indices mismatch " \
} \ "! %p <= %p < %p", (uint8_t *)start, \
(uint8_t *)m_item, (uint8_t *)end); \
ret = NULL; \
} else { \
ret = &m_item->el; \ ret = &m_item->el; \
} \
} }
#undef ALIGN #undef ALIGN
...@@ -343,7 +357,8 @@ static void init_qxl_ram(PCIQXLDevice *d) ...@@ -343,7 +357,8 @@ static void init_qxl_ram(PCIQXLDevice *d)
SPICE_RING_INIT(&d->ram->cmd_ring); SPICE_RING_INIT(&d->ram->cmd_ring);
SPICE_RING_INIT(&d->ram->cursor_ring); SPICE_RING_INIT(&d->ram->cursor_ring);
SPICE_RING_INIT(&d->ram->release_ring); SPICE_RING_INIT(&d->ram->release_ring);
SPICE_RING_PROD_ITEM(&d->ram->release_ring, item); SPICE_RING_PROD_ITEM(d, &d->ram->release_ring, item);
assert(item);
*item = 0; *item = 0;
qxl_ring_set_dirty(d); qxl_ring_set_dirty(d);
} }
...@@ -383,14 +398,22 @@ static void qxl_ring_set_dirty(PCIQXLDevice *qxl) ...@@ -383,14 +398,22 @@ static void qxl_ring_set_dirty(PCIQXLDevice *qxl)
* keep track of some command state, for savevm/loadvm. * keep track of some command state, for savevm/loadvm.
* called from spice server thread context only * called from spice server thread context only
*/ */
static void qxl_track_command(PCIQXLDevice *qxl, struct QXLCommandExt *ext) static int qxl_track_command(PCIQXLDevice *qxl, struct QXLCommandExt *ext)
{ {
switch (le32_to_cpu(ext->cmd.type)) { switch (le32_to_cpu(ext->cmd.type)) {
case QXL_CMD_SURFACE: case QXL_CMD_SURFACE:
{ {
QXLSurfaceCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id); QXLSurfaceCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id);
if (!cmd) {
return 1;
}
uint32_t id = le32_to_cpu(cmd->surface_id); uint32_t id = le32_to_cpu(cmd->surface_id);
PANIC_ON(id >= NUM_SURFACES);
if (id >= NUM_SURFACES) {
qxl_guest_bug(qxl, "QXL_CMD_SURFACE id %d >= %d", id, NUM_SURFACES);
return 1;
}
qemu_mutex_lock(&qxl->track_lock); qemu_mutex_lock(&qxl->track_lock);
if (cmd->type == QXL_SURFACE_CMD_CREATE) { if (cmd->type == QXL_SURFACE_CMD_CREATE) {
qxl->guest_surfaces.cmds[id] = ext->cmd.data; qxl->guest_surfaces.cmds[id] = ext->cmd.data;
...@@ -408,6 +431,10 @@ static void qxl_track_command(PCIQXLDevice *qxl, struct QXLCommandExt *ext) ...@@ -408,6 +431,10 @@ static void qxl_track_command(PCIQXLDevice *qxl, struct QXLCommandExt *ext)
case QXL_CMD_CURSOR: case QXL_CMD_CURSOR:
{ {
QXLCursorCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id); QXLCursorCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id);
if (!cmd) {
return 1;
}
if (cmd->type == QXL_CURSOR_SET) { if (cmd->type == QXL_CURSOR_SET) {
qemu_mutex_lock(&qxl->track_lock); qemu_mutex_lock(&qxl->track_lock);
qxl->guest_cursor = ext->cmd.data; qxl->guest_cursor = ext->cmd.data;
...@@ -416,6 +443,7 @@ static void qxl_track_command(PCIQXLDevice *qxl, struct QXLCommandExt *ext) ...@@ -416,6 +443,7 @@ static void qxl_track_command(PCIQXLDevice *qxl, struct QXLCommandExt *ext)
break; break;
} }
} }
return 0;
} }
/* spice display interface callbacks */ /* spice display interface callbacks */
...@@ -546,8 +574,10 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext) ...@@ -546,8 +574,10 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext)
if (SPICE_RING_IS_EMPTY(ring)) { if (SPICE_RING_IS_EMPTY(ring)) {
return false; return false;
} }
trace_qxl_ring_command_get(qxl->id, qxl_mode_to_string(qxl->mode)); SPICE_RING_CONS_ITEM(qxl, ring, cmd);
SPICE_RING_CONS_ITEM(ring, cmd); if (!cmd) {
return false;
}
ext->cmd = *cmd; ext->cmd = *cmd;
ext->group_id = MEMSLOT_GROUP_GUEST; ext->group_id = MEMSLOT_GROUP_GUEST;
ext->flags = qxl->cmdflags; ext->flags = qxl->cmdflags;
...@@ -559,6 +589,7 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext) ...@@ -559,6 +589,7 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext)
qxl->guest_primary.commands++; qxl->guest_primary.commands++;
qxl_track_command(qxl, ext); qxl_track_command(qxl, ext);
qxl_log_command(qxl, "cmd", ext); qxl_log_command(qxl, "cmd", ext);
trace_qxl_ring_command_get(qxl->id, qxl_mode_to_string(qxl->mode));
return true; return true;
default: default:
return false; return false;
...@@ -617,7 +648,10 @@ static inline void qxl_push_free_res(PCIQXLDevice *d, int flush) ...@@ -617,7 +648,10 @@ static inline void qxl_push_free_res(PCIQXLDevice *d, int flush)
if (notify) { if (notify) {
qxl_send_events(d, QXL_INTERRUPT_DISPLAY); qxl_send_events(d, QXL_INTERRUPT_DISPLAY);
} }
SPICE_RING_PROD_ITEM(ring, item); SPICE_RING_PROD_ITEM(d, ring, item);
if (!item) {
return;
}
*item = 0; *item = 0;
d->num_free_res = 0; d->num_free_res = 0;
d->last_release = NULL; d->last_release = NULL;
...@@ -643,7 +677,10 @@ static void interface_release_resource(QXLInstance *sin, ...@@ -643,7 +677,10 @@ static void interface_release_resource(QXLInstance *sin,
* pci bar 0, $command.release_info * pci bar 0, $command.release_info
*/ */
ring = &qxl->ram->release_ring; ring = &qxl->ram->release_ring;
SPICE_RING_PROD_ITEM(ring, item); SPICE_RING_PROD_ITEM(qxl, ring, item);
if (!item) {
return;
}
if (*item == 0) { if (*item == 0) {
/* stick head into the ring */ /* stick head into the ring */
id = ext.info->id; id = ext.info->id;
...@@ -682,7 +719,10 @@ static int interface_get_cursor_command(QXLInstance *sin, struct QXLCommandExt * ...@@ -682,7 +719,10 @@ static int interface_get_cursor_command(QXLInstance *sin, struct QXLCommandExt *
if (SPICE_RING_IS_EMPTY(ring)) { if (SPICE_RING_IS_EMPTY(ring)) {
return false; return false;
} }
SPICE_RING_CONS_ITEM(ring, cmd); SPICE_RING_CONS_ITEM(qxl, ring, cmd);
if (!cmd) {
return false;
}
ext->cmd = *cmd; ext->cmd = *cmd;
ext->group_id = MEMSLOT_GROUP_GUEST; ext->group_id = MEMSLOT_GROUP_GUEST;
ext->flags = qxl->cmdflags; ext->flags = qxl->cmdflags;
...@@ -728,8 +768,13 @@ static int interface_req_cursor_notification(QXLInstance *sin) ...@@ -728,8 +768,13 @@ static int interface_req_cursor_notification(QXLInstance *sin)
/* called from spice server thread context */ /* called from spice server thread context */
static void interface_notify_update(QXLInstance *sin, uint32_t update_id) static void interface_notify_update(QXLInstance *sin, uint32_t update_id)
{ {
fprintf(stderr, "%s: abort()\n", __FUNCTION__); /*
abort(); * Called by spice-server as a result of a QXL_CMD_UPDATE which is not in
* use by xf86-video-qxl and is defined out in the qxl windows driver.
* Probably was at some earlier version that is prior to git start (2009),
* and is still guest trigerrable.
*/
fprintf(stderr, "%s: deprecated\n", __func__);
} }
/* called from spice server thread context only */ /* called from spice server thread context only */
...@@ -764,8 +809,8 @@ static void interface_async_complete_io(PCIQXLDevice *qxl, QXLCookie *cookie) ...@@ -764,8 +809,8 @@ static void interface_async_complete_io(PCIQXLDevice *qxl, QXLCookie *cookie)
} }
if (cookie && current_async != cookie->io) { if (cookie && current_async != cookie->io) {
fprintf(stderr, fprintf(stderr,
"qxl: %s: error: current_async = %d != %" PRId64 " = cookie->io\n", "qxl: %s: error: current_async = %d != %"
__func__, current_async, cookie->io); PRId64 " = cookie->io\n", __func__, current_async, cookie->io);
} }
switch (current_async) { switch (current_async) {
case QXL_IO_MEMSLOT_ADD_ASYNC: case QXL_IO_MEMSLOT_ADD_ASYNC:
...@@ -993,7 +1038,7 @@ static const MemoryRegionPortio qxl_vga_portio_list[] = { ...@@ -993,7 +1038,7 @@ static const MemoryRegionPortio qxl_vga_portio_list[] = {
PORTIO_END_OF_LIST(), PORTIO_END_OF_LIST(),
}; };
static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta, static int qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta,
qxl_async_io async) qxl_async_io async)
{ {
static const int regions[] = { static const int regions[] = {
...@@ -1015,8 +1060,16 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta, ...@@ -1015,8 +1060,16 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta,
trace_qxl_memslot_add_guest(d->id, slot_id, guest_start, guest_end); trace_qxl_memslot_add_guest(d->id, slot_id, guest_start, guest_end);
PANIC_ON(slot_id >= NUM_MEMSLOTS); if (slot_id >= NUM_MEMSLOTS) {
PANIC_ON(guest_start > guest_end); qxl_guest_bug(d, "%s: slot_id >= NUM_MEMSLOTS %d >= %d", __func__,
slot_id, NUM_MEMSLOTS);
return 1;
}
if (guest_start > guest_end) {
qxl_guest_bug(d, "%s: guest_start > guest_end 0x%" PRIx64
" > 0x%" PRIx64, __func__, guest_start, guest_end);
return 1;
}
for (i = 0; i < ARRAY_SIZE(regions); i++) { for (i = 0; i < ARRAY_SIZE(regions); i++) {
pci_region = regions[i]; pci_region = regions[i];
...@@ -1037,7 +1090,10 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta, ...@@ -1037,7 +1090,10 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta,
/* passed */ /* passed */
break; break;
} }
PANIC_ON(i == ARRAY_SIZE(regions)); /* finished loop without match */ if (i == ARRAY_SIZE(regions)) {
qxl_guest_bug(d, "%s: finished loop without match", __func__);
return 1;
}
switch (pci_region) { switch (pci_region) {
case QXL_RAM_RANGE_INDEX: case QXL_RAM_RANGE_INDEX:
...@@ -1049,7 +1105,8 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta, ...@@ -1049,7 +1105,8 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta,
break; break;
default: default:
/* should not happen */ /* should not happen */
abort(); qxl_guest_bug(d, "%s: pci_region = %d", __func__, pci_region);
return 1;
} }
memslot.slot_id = slot_id; memslot.slot_id = slot_id;
...@@ -1065,6 +1122,7 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta, ...@@ -1065,6 +1122,7 @@ static void qxl_add_memslot(PCIQXLDevice *d, uint32_t slot_id, uint64_t delta,
d->guest_slots[slot_id].size = memslot.virt_end - memslot.virt_start; d->guest_slots[slot_id].size = memslot.virt_end - memslot.virt_start;
d->guest_slots[slot_id].delta = delta; d->guest_slots[slot_id].delta = delta;
d->guest_slots[slot_id].active = 1; d->guest_slots[slot_id].active = 1;
return 0;
} }
static void qxl_del_memslot(PCIQXLDevice *d, uint32_t slot_id) static void qxl_del_memslot(PCIQXLDevice *d, uint32_t slot_id)
...@@ -1097,15 +1155,28 @@ void *qxl_phys2virt(PCIQXLDevice *qxl, QXLPHYSICAL pqxl, int group_id) ...@@ -1097,15 +1155,28 @@ void *qxl_phys2virt(PCIQXLDevice *qxl, QXLPHYSICAL pqxl, int group_id)
case MEMSLOT_GROUP_HOST: case MEMSLOT_GROUP_HOST:
return (void *)(intptr_t)offset; return (void *)(intptr_t)offset;
case MEMSLOT_GROUP_GUEST: case MEMSLOT_GROUP_GUEST:
PANIC_ON(slot >= NUM_MEMSLOTS); if (slot >= NUM_MEMSLOTS) {
PANIC_ON(!qxl->guest_slots[slot].active); qxl_guest_bug(qxl, "slot too large %d >= %d", slot, NUM_MEMSLOTS);
PANIC_ON(offset < qxl->guest_slots[slot].delta); return NULL;
}
if (!qxl->guest_slots[slot].active) {
qxl_guest_bug(qxl, "inactive slot %d\n", slot);
return NULL;
}
if (offset < qxl->guest_slots[slot].delta) {
qxl_guest_bug(qxl, "slot %d offset %"PRIu64" < delta %"PRIu64"\n",
slot, offset, qxl->guest_slots[slot].delta);
return NULL;
}
offset -= qxl->guest_slots[slot].delta; offset -= qxl->guest_slots[slot].delta;
PANIC_ON(offset > qxl->guest_slots[slot].size) if (offset > qxl->guest_slots[slot].size) {
qxl_guest_bug(qxl, "slot %d offset %"PRIu64" > size %"PRIu64"\n",
slot, offset, qxl->guest_slots[slot].size);
return NULL;
}
return qxl->guest_slots[slot].ptr + offset; return qxl->guest_slots[slot].ptr + offset;
default:
PANIC_ON(1);
} }
return NULL;
} }
static void qxl_create_guest_primary_complete(PCIQXLDevice *qxl) static void qxl_create_guest_primary_complete(PCIQXLDevice *qxl)
...@@ -1120,7 +1191,10 @@ static void qxl_create_guest_primary(PCIQXLDevice *qxl, int loadvm, ...@@ -1120,7 +1191,10 @@ static void qxl_create_guest_primary(PCIQXLDevice *qxl, int loadvm,
QXLDevSurfaceCreate surface; QXLDevSurfaceCreate surface;
QXLSurfaceCreate *sc = &qxl->guest_primary.surface; QXLSurfaceCreate *sc = &qxl->guest_primary.surface;
assert(qxl->mode != QXL_MODE_NATIVE); if (qxl->mode == QXL_MODE_NATIVE) {
qxl_guest_bug(qxl, "%s: nop since already in QXL_MODE_NATIVE",
__func__);
}
qxl_exit_vga_mode(qxl); qxl_exit_vga_mode(qxl);
surface.format = le32_to_cpu(sc->format); surface.format = le32_to_cpu(sc->format);
...@@ -1192,7 +1266,7 @@ static void qxl_set_mode(PCIQXLDevice *d, int modenr, int loadvm) ...@@ -1192,7 +1266,7 @@ static void qxl_set_mode(PCIQXLDevice *d, int modenr, int loadvm)
} }
d->guest_slots[0].slot = slot; d->guest_slots[0].slot = slot;
qxl_add_memslot(d, 0, devmem, QXL_SYNC); assert(qxl_add_memslot(d, 0, devmem, QXL_SYNC) == 0);
d->guest_primary.surface = surface; d->guest_primary.surface = surface;
qxl_create_guest_primary(d, 0, QXL_SYNC); qxl_create_guest_primary(d, 0, QXL_SYNC);
...@@ -1393,8 +1467,7 @@ async_common: ...@@ -1393,8 +1467,7 @@ async_common:
qxl_spice_destroy_surfaces(d, async); qxl_spice_destroy_surfaces(d, async);
break; break;
default: default:
fprintf(stderr, "%s: ioport=0x%x, abort()\n", __FUNCTION__, io_port); qxl_guest_bug(d, "%s: unexpected ioport=0x%x\n", __func__, io_port);
abort();
} }
return; return;
cancel_async: cancel_async:
...@@ -1450,7 +1523,7 @@ static void qxl_send_events(PCIQXLDevice *d, uint32_t events) ...@@ -1450,7 +1523,7 @@ static void qxl_send_events(PCIQXLDevice *d, uint32_t events)
qxl_update_irq(d); qxl_update_irq(d);
} else { } else {
if (write(d->pipe[1], d, 1) != 1) { if (write(d->pipe[1], d, 1) != 1) {
dprint(d, 1, "%s: write to pipe failed\n", __FUNCTION__); dprint(d, 1, "%s: write to pipe failed\n", __func__);
} }
} }
} }
...@@ -1555,10 +1628,12 @@ static void qxl_dirty_surfaces(PCIQXLDevice *qxl) ...@@ -1555,10 +1628,12 @@ static void qxl_dirty_surfaces(PCIQXLDevice *qxl)
cmd = qxl_phys2virt(qxl, qxl->guest_surfaces.cmds[i], cmd = qxl_phys2virt(qxl, qxl->guest_surfaces.cmds[i],
MEMSLOT_GROUP_GUEST); MEMSLOT_GROUP_GUEST);
assert(cmd);
assert(cmd->type == QXL_SURFACE_CMD_CREATE); assert(cmd->type == QXL_SURFACE_CMD_CREATE);
surface_offset = (intptr_t)qxl_phys2virt(qxl, surface_offset = (intptr_t)qxl_phys2virt(qxl,
cmd->u.surface_create.data, cmd->u.surface_create.data,
MEMSLOT_GROUP_GUEST); MEMSLOT_GROUP_GUEST);
assert(surface_offset);
surface_offset -= vram_start; surface_offset -= vram_start;
surface_size = cmd->u.surface_create.height * surface_size = cmd->u.surface_create.height *
abs(cmd->u.surface_create.stride); abs(cmd->u.surface_create.stride);
......
...@@ -142,12 +142,12 @@ void qxl_spice_reset_image_cache(PCIQXLDevice *qxl); ...@@ -142,12 +142,12 @@ void qxl_spice_reset_image_cache(PCIQXLDevice *qxl);
void qxl_spice_reset_cursor(PCIQXLDevice *qxl); void qxl_spice_reset_cursor(PCIQXLDevice *qxl);
/* qxl-logger.c */ /* qxl-logger.c */
void qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id); int qxl_log_cmd_cursor(PCIQXLDevice *qxl, QXLCursorCmd *cmd, int group_id);
void qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext); int qxl_log_command(PCIQXLDevice *qxl, const char *ring, QXLCommandExt *ext);
/* qxl-render.c */ /* qxl-render.c */
void qxl_render_resize(PCIQXLDevice *qxl); void qxl_render_resize(PCIQXLDevice *qxl);
void qxl_render_update(PCIQXLDevice *qxl); void qxl_render_update(PCIQXLDevice *qxl);
void qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext); int qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext);
void qxl_render_update_area_done(PCIQXLDevice *qxl, QXLCookie *cookie); void qxl_render_update_area_done(PCIQXLDevice *qxl, QXLCookie *cookie);
void qxl_render_update_area_bh(void *opaque); void qxl_render_update_area_bh(void *opaque);
...@@ -633,6 +633,25 @@ ...@@ -633,6 +633,25 @@
'connection-id': 'int', 'channel-type': 'int', 'channel-id': 'int', 'connection-id': 'int', 'channel-type': 'int', 'channel-id': 'int',
'tls': 'bool'} } 'tls': 'bool'} }
##
# @SpiceQueryMouseMode
#
# An enumation of Spice mouse states.
#
# @client: Mouse cursor position is determined by the client.
#
# @server: Mouse cursor position is determined by the server.
#
# @unknown: No information is available about mouse mode used by
# the spice server.
#
# Note: spice/enums.h has a SpiceMouseMode already, hence the name.
#
# Since: 1.1
##
{ 'enum': 'SpiceQueryMouseMode',
'data': [ 'client', 'server', 'unknown' ] }
## ##
# @SpiceInfo # @SpiceInfo
# #
...@@ -654,6 +673,12 @@ ...@@ -654,6 +673,12 @@
# 'spice' uses SASL or direct TLS authentication, depending on command # 'spice' uses SASL or direct TLS authentication, depending on command
# line options # line options
# #
# @mouse-mode: The mode in which the mouse cursor is displayed currently. Can
# be determined by the client or the server, or unknown if spice
# server doesn't provide this information.
#
# Since: 1.1
#
# @channels: a list of @SpiceChannel for each active spice channel # @channels: a list of @SpiceChannel for each active spice channel
# #
# Since: 0.14.0 # Since: 0.14.0
...@@ -661,7 +686,7 @@ ...@@ -661,7 +686,7 @@
{ 'type': 'SpiceInfo', { 'type': 'SpiceInfo',
'data': {'enabled': 'bool', '*host': 'str', '*port': 'int', 'data': {'enabled': 'bool', '*host': 'str', '*port': 'int',
'*tls-port': 'int', '*auth': 'str', '*compiled-version': 'str', '*tls-port': 'int', '*auth': 'str', '*compiled-version': 'str',
'*channels': ['SpiceChannel']} } 'mouse-mode': 'SpiceQueryMouseMode', '*channels': ['SpiceChannel']} }
## ##
# @query-spice # @query-spice
......
...@@ -462,6 +462,13 @@ SpiceInfo *qmp_query_spice(Error **errp) ...@@ -462,6 +462,13 @@ SpiceInfo *qmp_query_spice(Error **errp)
info->tls_port = tls_port; info->tls_port = tls_port;
} }
#if SPICE_SERVER_VERSION >= 0x000a03 /* 0.10.3 */
info->mouse_mode = spice_server_is_server_mouse(spice_server) ?
SPICE_QUERY_MOUSE_MODE_SERVER :
SPICE_QUERY_MOUSE_MODE_CLIENT;
#else
info->mouse_mode = SPICE_QUERY_MOUSE_MODE_UNKNOWN;
#endif
/* for compatibility with the original command */ /* for compatibility with the original command */
info->has_channels = true; info->has_channels = true;
info->channels = qmp_query_spice_channels(); info->channels = qmp_query_spice_channels();
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册