提交 e0c64d08 编写于 作者: G Gerd Hoffmann

spice: don't create updates in spice server context.

This patch moves the creation of spice screen updates from the spice
server context to qemu iothread context (display refresh timer to be
exact).  This way we avoid accessing qemu internals (display surface)
from spice thread context which in turn allows us to simplify locking.
Signed-off-by: NGerd Hoffmann <kraxel@redhat.com>
上级 14da8345
...@@ -343,18 +343,22 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext) ...@@ -343,18 +343,22 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext)
SimpleSpiceUpdate *update; SimpleSpiceUpdate *update;
QXLCommandRing *ring; QXLCommandRing *ring;
QXLCommand *cmd; QXLCommand *cmd;
int notify; int notify, ret;
switch (qxl->mode) { switch (qxl->mode) {
case QXL_MODE_VGA: case QXL_MODE_VGA:
dprint(qxl, 2, "%s: vga\n", __FUNCTION__); dprint(qxl, 2, "%s: vga\n", __FUNCTION__);
update = qemu_spice_create_update(&qxl->ssd); ret = false;
if (update == NULL) { qemu_mutex_lock(&qxl->ssd.lock);
return false; if (qxl->ssd.update != NULL) {
update = qxl->ssd.update;
qxl->ssd.update = NULL;
*ext = update->ext;
ret = true;
} }
*ext = update->ext; qemu_mutex_unlock(&qxl->ssd.lock);
qxl_log_command(qxl, "vga", ext); qxl_log_command(qxl, "vga", ext);
return true; return ret;
case QXL_MODE_COMPAT: case QXL_MODE_COMPAT:
case QXL_MODE_NATIVE: case QXL_MODE_NATIVE:
case QXL_MODE_UNDEFINED: case QXL_MODE_UNDEFINED:
...@@ -1304,6 +1308,7 @@ static int qxl_init_primary(PCIDevice *dev) ...@@ -1304,6 +1308,7 @@ static int qxl_init_primary(PCIDevice *dev)
vga->ds = graphic_console_init(qxl_hw_update, qxl_hw_invalidate, vga->ds = graphic_console_init(qxl_hw_update, qxl_hw_invalidate,
qxl_hw_screen_dump, qxl_hw_text_update, qxl); qxl_hw_screen_dump, qxl_hw_text_update, qxl);
qxl->ssd.ds = vga->ds; qxl->ssd.ds = vga->ds;
qemu_mutex_init(&qxl->ssd.lock);
qxl->ssd.bufsize = (16 * 1024 * 1024); qxl->ssd.bufsize = (16 * 1024 * 1024);
qxl->ssd.buf = qemu_malloc(qxl->ssd.bufsize); qxl->ssd.buf = qemu_malloc(qxl->ssd.bufsize);
......
...@@ -62,14 +62,7 @@ void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r) ...@@ -62,14 +62,7 @@ void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r)
dest->right = MAX(dest->right, r->right); dest->right = MAX(dest->right, r->right);
} }
/* static SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *ssd)
* Called from spice server thread context (via interface_get_command).
*
* We must aquire the global qemu mutex here to make sure the
* DisplayState (+DisplaySurface) we are accessing doesn't change
* underneath us.
*/
SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *ssd)
{ {
SimpleSpiceUpdate *update; SimpleSpiceUpdate *update;
QXLDrawable *drawable; QXLDrawable *drawable;
...@@ -78,9 +71,7 @@ SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *ssd) ...@@ -78,9 +71,7 @@ SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *ssd)
uint8_t *src, *dst; uint8_t *src, *dst;
int by, bw, bh; int by, bw, bh;
qemu_mutex_lock_iothread();
if (qemu_spice_rect_is_empty(&ssd->dirty)) { if (qemu_spice_rect_is_empty(&ssd->dirty)) {
qemu_mutex_unlock_iothread();
return NULL; return NULL;
}; };
...@@ -141,7 +132,6 @@ SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *ssd) ...@@ -141,7 +132,6 @@ SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *ssd)
cmd->data = (intptr_t)drawable; cmd->data = (intptr_t)drawable;
memset(&ssd->dirty, 0, sizeof(ssd->dirty)); memset(&ssd->dirty, 0, sizeof(ssd->dirty));
qemu_mutex_unlock_iothread();
return update; return update;
} }
...@@ -241,6 +231,12 @@ void qemu_spice_display_resize(SimpleSpiceDisplay *ssd) ...@@ -241,6 +231,12 @@ void qemu_spice_display_resize(SimpleSpiceDisplay *ssd)
qemu_pf_conv_put(ssd->conv); qemu_pf_conv_put(ssd->conv);
ssd->conv = NULL; ssd->conv = NULL;
qemu_mutex_lock(&ssd->lock);
if (ssd->update != NULL) {
qemu_spice_destroy_update(ssd, ssd->update);
ssd->update = NULL;
}
qemu_mutex_unlock(&ssd->lock);
qemu_spice_destroy_host_primary(ssd); qemu_spice_destroy_host_primary(ssd);
qemu_spice_create_host_primary(ssd); qemu_spice_create_host_primary(ssd);
...@@ -252,6 +248,14 @@ void qemu_spice_display_refresh(SimpleSpiceDisplay *ssd) ...@@ -252,6 +248,14 @@ void qemu_spice_display_refresh(SimpleSpiceDisplay *ssd)
{ {
dprint(3, "%s:\n", __FUNCTION__); dprint(3, "%s:\n", __FUNCTION__);
vga_hw_update(); vga_hw_update();
qemu_mutex_lock(&ssd->lock);
if (ssd->update == NULL) {
ssd->update = qemu_spice_create_update(ssd);
ssd->notify++;
}
qemu_mutex_unlock(&ssd->lock);
if (ssd->notify) { if (ssd->notify) {
ssd->notify = 0; ssd->notify = 0;
ssd->worker->wakeup(ssd->worker); ssd->worker->wakeup(ssd->worker);
...@@ -298,14 +302,20 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext) ...@@ -298,14 +302,20 @@ static int interface_get_command(QXLInstance *sin, struct QXLCommandExt *ext)
{ {
SimpleSpiceDisplay *ssd = container_of(sin, SimpleSpiceDisplay, qxl); SimpleSpiceDisplay *ssd = container_of(sin, SimpleSpiceDisplay, qxl);
SimpleSpiceUpdate *update; SimpleSpiceUpdate *update;
int ret = false;
dprint(3, "%s:\n", __FUNCTION__); dprint(3, "%s:\n", __FUNCTION__);
update = qemu_spice_create_update(ssd);
if (update == NULL) { qemu_mutex_lock(&ssd->lock);
return false; if (ssd->update != NULL) {
update = ssd->update;
ssd->update = NULL;
*ext = update->ext;
ret = true;
} }
*ext = update->ext; qemu_mutex_unlock(&ssd->lock);
return true;
return ret;
} }
static int interface_req_cmd_notification(QXLInstance *sin) static int interface_req_cmd_notification(QXLInstance *sin)
...@@ -398,6 +408,7 @@ void qemu_spice_display_init(DisplayState *ds) ...@@ -398,6 +408,7 @@ void qemu_spice_display_init(DisplayState *ds)
{ {
assert(sdpy.ds == NULL); assert(sdpy.ds == NULL);
sdpy.ds = ds; sdpy.ds = ds;
qemu_mutex_init(&sdpy.lock);
sdpy.bufsize = (16 * 1024 * 1024); sdpy.bufsize = (16 * 1024 * 1024);
sdpy.buf = qemu_malloc(sdpy.bufsize); sdpy.buf = qemu_malloc(sdpy.bufsize);
register_displaychangelistener(ds, &display_listener); register_displaychangelistener(ds, &display_listener);
......
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
#include <spice/enums.h> #include <spice/enums.h>
#include <spice/qxl_dev.h> #include <spice/qxl_dev.h>
#include "qemu-thread.h"
#include "pflib.h" #include "pflib.h"
#define NUM_MEMSLOTS 8 #define NUM_MEMSLOTS 8
...@@ -31,7 +32,10 @@ ...@@ -31,7 +32,10 @@
#define NUM_SURFACES 1024 #define NUM_SURFACES 1024
typedef struct SimpleSpiceDisplay { typedef struct SimpleSpiceDisplay SimpleSpiceDisplay;
typedef struct SimpleSpiceUpdate SimpleSpiceUpdate;
struct SimpleSpiceDisplay {
DisplayState *ds; DisplayState *ds;
void *buf; void *buf;
int bufsize; int bufsize;
...@@ -43,19 +47,26 @@ typedef struct SimpleSpiceDisplay { ...@@ -43,19 +47,26 @@ typedef struct SimpleSpiceDisplay {
QXLRect dirty; QXLRect dirty;
int notify; int notify;
int running; int running;
} SimpleSpiceDisplay;
typedef struct SimpleSpiceUpdate { /*
* All struct members below this comment can be accessed from
* both spice server and qemu (iothread) context and any access
* to them must be protected by the lock.
*/
QemuMutex lock;
SimpleSpiceUpdate *update;
};
struct SimpleSpiceUpdate {
QXLDrawable drawable; QXLDrawable drawable;
QXLImage image; QXLImage image;
QXLCommandExt ext; QXLCommandExt ext;
uint8_t *bitmap; uint8_t *bitmap;
} SimpleSpiceUpdate; };
int qemu_spice_rect_is_empty(const QXLRect* r); int qemu_spice_rect_is_empty(const QXLRect* r);
void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r); void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r);
SimpleSpiceUpdate *qemu_spice_create_update(SimpleSpiceDisplay *sdpy);
void qemu_spice_destroy_update(SimpleSpiceDisplay *sdpy, SimpleSpiceUpdate *update); void qemu_spice_destroy_update(SimpleSpiceDisplay *sdpy, SimpleSpiceUpdate *update);
void qemu_spice_create_host_memslot(SimpleSpiceDisplay *ssd); void qemu_spice_create_host_memslot(SimpleSpiceDisplay *ssd);
void qemu_spice_create_host_primary(SimpleSpiceDisplay *ssd); void qemu_spice_create_host_primary(SimpleSpiceDisplay *ssd);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册