[Spice-devel] [PATCH 5/5] server/red_worker: reuse dispatcher

Alon Levy alevy at redhat.com
Sun Nov 6 08:49:15 PST 2011


This patch reuses Dispatcher in RedDispatcher. It adds two helpers
to red_worker to keep RedWorker opaque to the outside. The dispatcher is
abused in three places that use the underlying socket directly:
 once sending a READY after red_init completes
 once for each channel creation, replying with the RedChannel instance
  for cursor and display.

Bugzilla: 42463

rfc->v1:
* move callbacks to red_worker.c including registration (Yonit)
* rename dispatcher to red_dispatcher in red_worker.c and red_dispatcher.c
* add accessor red_dispatcher_get_dispatcher
* s/dispatcher_handle_recv/dispatcher_handle_recv_read/ and change sig to
  just Dispatcher *dispatcher (was the SpiceCoreInterface one)
* remove SpiceCoreInterface parameter from dispatcher_init (Yonit)
* main_dispatcher needed it for channel_event so it has it in
  struct MainDispatcher
* add dispatcher_get_recv_fd for red_worker
---
 server/dispatcher.c      |   20 +-
 server/dispatcher.h      |   19 +-
 server/main_dispatcher.c |   15 +-
 server/red_dispatcher.c  |  484 ++++++++++++++-----------
 server/red_dispatcher.h  |  145 ++++++++
 server/red_worker.c      |  904 +++++++++++++++++++++++++++++-----------------
 server/red_worker.h      |    5 +-
 7 files changed, 1022 insertions(+), 570 deletions(-)

diff --git a/server/dispatcher.c b/server/dispatcher.c
index fb7920b..e3c7f4a 100644
--- a/server/dispatcher.c
+++ b/server/dispatcher.c
@@ -147,13 +147,11 @@ static int dispatcher_handle_single_read(Dispatcher *dispatcher)
 }
 
 /*
- * dispatcher_handle_read
+ * dispatcher_handle_recv_read
  * doesn't handle being in the middle of a message. all reads are blocking.
  */
-void dispatcher_handle_read(int fd, int event, void *opaque)
+void dispatcher_handle_recv_read(Dispatcher *dispatcher)
 {
-    Dispatcher *dispatcher = opaque;
-
     while (dispatcher_handle_single_read(dispatcher)) {
     }
 }
@@ -210,12 +208,11 @@ void dispatcher_register_handler(Dispatcher *dispatcher, uint32_t message_type,
     }
 }
 
-void dispatcher_init(Dispatcher *dispatcher, SpiceCoreInterface *recv_core,
-                     size_t max_message_type, void *opaque)
+void dispatcher_init(Dispatcher *dispatcher, size_t max_message_type,
+                     void *opaque)
 {
     int channels[2];
 
-    dispatcher->recv_core = recv_core;
     dispatcher->opaque = opaque;
     if (socketpair(AF_LOCAL, SOCK_STREAM, 0, channels) == -1) {
         red_error("socketpair failed %s", strerror(errno));
@@ -229,13 +226,14 @@ void dispatcher_init(Dispatcher *dispatcher, SpiceCoreInterface *recv_core,
     dispatcher->messages = spice_malloc0_n(max_message_type,
                                            sizeof(dispatcher->messages[0]));
     dispatcher->max_message_type = max_message_type;
-    if (recv_core) {
-        recv_core->watch_add(dispatcher->recv_fd, SPICE_WATCH_EVENT_READ,
-                             dispatcher_handle_read, dispatcher);
-    }
 }
 
 void dispatcher_set_opaque(Dispatcher *dispatcher, void *opaque)
 {
     dispatcher->opaque = opaque;
 }
+
+int dispatcher_get_recv_fd(Dispatcher *dispatcher)
+{
+    return dispatcher->recv_fd;
+}
diff --git a/server/dispatcher.h b/server/dispatcher.h
index 44bfb47..04e6b46 100644
--- a/server/dispatcher.h
+++ b/server/dispatcher.h
@@ -38,13 +38,12 @@ void dispatcher_send_message(Dispatcher *dispatcher, uint32_t message_type,
 
 /*
  * dispatcher_init
- * @recv_core: core interface instance used by receiver
  * @max_message_type: number of message types. Allows upfront allocation
  *  of a DispatcherMessage list.
  * up front, and registration in any order wanted.
  */
-void dispatcher_init(Dispatcher *dispatcher, SpiceCoreInterface *recv_core,
-                     size_t max_message_type, void *opaque);
+void dispatcher_init(Dispatcher *dispatcher, size_t max_message_type,
+                     void *opaque);
 
 /*
  * dispatcher_register_handler
@@ -61,12 +60,16 @@ void dispatcher_register_handler(Dispatcher *dispatcher, uint32_t message_type,
                                  int ack);
 
 /*
- *  dispatcher_handle_read
- *  @fd: fd triggered. Value is ignored
- *  @event: event (read/write). Value is ignored
- *  @opaque: must be Dispatcher instance
+ *  dispatcher_handle_recv_read
+ *  @dispatcher: Dispatcher instance
  */
-void dispatcher_handle_read(int fd, int event, void *opaque);
+void dispatcher_handle_recv_read(Dispatcher *);
+
+/*
+ *  dispatcher_get_recv_fd
+ *  @return: receive file descriptor of the dispatcher
+ */
+int dispatcher_get_recv_fd(Dispatcher *);
 
 /*
  * dispatcher_set_opaque
diff --git a/server/main_dispatcher.c b/server/main_dispatcher.c
index 5cddab7..a5967fa 100644
--- a/server/main_dispatcher.c
+++ b/server/main_dispatcher.c
@@ -30,6 +30,7 @@
 
 typedef struct {
     Dispatcher base;
+    SpiceCoreInterface *core;
 } MainDispatcher;
 
 MainDispatcher main_dispatcher;
@@ -50,7 +51,7 @@ static void main_dispatcher_self_handle_channel_event(
                                                 int event,
                                                 SpiceChannelEventInfo *info)
 {
-    main_dispatcher.base.recv_core->channel_event(event, info);
+    main_dispatcher.core->channel_event(event, info);
 }
 
 static void main_dispatcher_handle_channel_event(void *opaque,
@@ -76,10 +77,20 @@ void main_dispatcher_channel_event(int event, SpiceChannelEventInfo *info)
                             &msg);
 }
 
+static void dispatcher_handle_read(int fd, int event, void *opaque)
+{
+    Dispatcher *dispatcher = opaque;
+
+    dispatcher_handle_recv_read(dispatcher);
+}
+
 void main_dispatcher_init(SpiceCoreInterface *core)
 {
     memset(&main_dispatcher, 0, sizeof(main_dispatcher));
-    dispatcher_init(&main_dispatcher.base, core, MAIN_DISPATCHER_NUM_MESSAGES, &main_dispatcher.base);
+    main_dispatcher.core = core;
+    dispatcher_init(&main_dispatcher.base, MAIN_DISPATCHER_NUM_MESSAGES, &main_dispatcher.base);
+    core->watch_add(main_dispatcher.base.recv_fd, SPICE_WATCH_EVENT_READ,
+                    dispatcher_handle_read, &main_dispatcher.base);
     dispatcher_register_handler(&main_dispatcher.base, MAIN_DISPATCHER_CHANNEL_EVENT,
                                 main_dispatcher_handle_channel_event,
                                 sizeof(MainDispatcherChannelEventMessage), 0 /* no ack */);
diff --git a/server/red_dispatcher.c b/server/red_dispatcher.c
index 10e6126..72539dc 100644
--- a/server/red_dispatcher.c
+++ b/server/red_dispatcher.c
@@ -37,6 +37,7 @@
 #include "reds_gl_canvas.h"
 #endif // USE_OPENGL
 #include "reds.h"
+#include "dispatcher.h"
 #include "red_dispatcher.h"
 #include "red_parse_qxl.h"
 
@@ -58,7 +59,7 @@ struct AsyncCommand {
 struct RedDispatcher {
     QXLWorker base;
     QXLInstance *qxl;
-    int channel;
+    Dispatcher dispatcher;
     pthread_t worker_thread;
     uint32_t pending;
     int primary_active;
@@ -93,19 +94,22 @@ static void red_dispatcher_set_display_peer(RedChannel *channel, RedClient *clie
                                             int num_common_caps, uint32_t *common_caps, int num_caps,
                                             uint32_t *caps)
 {
+    RedWorkerMessageDisplayConnect payload;
     RedDispatcher *dispatcher;
 
     red_printf("");
     dispatcher = (RedDispatcher *)channel->data;
-    RedWorkerMessage message = RED_WORKER_MESSAGE_DISPLAY_CONNECT;
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &client, sizeof(RedClient *));
-    send_data(dispatcher->channel, &stream, sizeof(RedsStream *));
-    send_data(dispatcher->channel, &migration, sizeof(int));
+    payload.client = client;
+    payload.stream = stream;
+    payload.migration = migration;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DISPLAY_CONNECT,
+                            &payload);
 }
 
 static void red_dispatcher_disconnect_display_peer(RedChannelClient *rcc)
 {
+    RedWorkerMessageDisplayDisconnect payload;
     RedDispatcher *dispatcher;
 
     if (!rcc->channel) {
@@ -115,27 +119,28 @@ static void red_dispatcher_disconnect_display_peer(RedChannelClient *rcc)
     dispatcher = (RedDispatcher *)rcc->channel->data;
 
     red_printf("");
-    RedWorkerMessage message = RED_WORKER_MESSAGE_DISPLAY_DISCONNECT;
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &rcc, sizeof(RedChannelClient *));
+    payload.rcc = rcc;
 
     // TODO: we turned it to be sync, due to client_destroy . Should we support async? - for this we will need ref count
     // for channels
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DISPLAY_DISCONNECT,
+                            &payload);
 }
 
 static void red_dispatcher_display_migrate(RedChannelClient *rcc)
 {
+    RedWorkerMessageDisplayMigrate payload;
     RedDispatcher *dispatcher;
     if (!rcc->channel) {
         return;
     }
     dispatcher = (RedDispatcher *)rcc->channel->data;
     red_printf("channel type %u id %u", rcc->channel->type, rcc->channel->id);
-    RedWorkerMessage message = RED_WORKER_MESSAGE_DISPLAY_MIGRATE;
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &rcc, sizeof(RedChannelClient *));
+    payload.rcc = rcc;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DISPLAY_MIGRATE,
+                            &payload);
 }
 
 static void red_dispatcher_set_cursor_peer(RedChannel *channel, RedClient *client, RedsStream *stream,
@@ -143,17 +148,20 @@ static void red_dispatcher_set_cursor_peer(RedChannel *channel, RedClient *clien
                                            uint32_t *common_caps, int num_caps,
                                            uint32_t *caps)
 {
+    RedWorkerMessageCursorConnect payload;
     RedDispatcher *dispatcher = (RedDispatcher *)channel->data;
     red_printf("");
-    RedWorkerMessage message = RED_WORKER_MESSAGE_CURSOR_CONNECT;
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &client, sizeof(RedClient *));
-    send_data(dispatcher->channel, &stream, sizeof(RedsStream *));
-    send_data(dispatcher->channel, &migration, sizeof(int));
+    payload.client = client;
+    payload.stream = stream;
+    payload.migration = migration;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_CURSOR_CONNECT,
+                            &payload);
 }
 
 static void red_dispatcher_disconnect_cursor_peer(RedChannelClient *rcc)
 {
+    RedWorkerMessageCursorDisconnect payload;
     RedDispatcher *dispatcher;
 
     if (!rcc->channel) {
@@ -162,16 +170,16 @@ static void red_dispatcher_disconnect_cursor_peer(RedChannelClient *rcc)
 
     dispatcher = (RedDispatcher *)rcc->channel->data;
     red_printf("");
-    RedWorkerMessage message = RED_WORKER_MESSAGE_CURSOR_DISCONNECT;
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &rcc, sizeof(RedChannelClient *));
+    payload.rcc = rcc;
 
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_CURSOR_DISCONNECT,
+                            &payload);
 }
 
 static void red_dispatcher_cursor_migrate(RedChannelClient *rcc)
 {
+    RedWorkerMessageCursorMigrate payload;
     RedDispatcher *dispatcher;
 
     if (!rcc->channel) {
@@ -179,8 +187,10 @@ static void red_dispatcher_cursor_migrate(RedChannelClient *rcc)
     }
     dispatcher = (RedDispatcher *)rcc->channel->data;
     red_printf("channel type %u id %u", rcc->channel->type, rcc->channel->id);
-    RedWorkerMessage message = RED_WORKER_MESSAGE_CURSOR_MIGRATE;
-    write_message(dispatcher->channel, &message);
+    payload.rcc = rcc;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_CURSOR_MIGRATE,
+                            &payload);
 }
 
 typedef struct RendererInfo {
@@ -263,16 +273,16 @@ static void red_dispatcher_update_area(RedDispatcher *dispatcher, uint32_t surfa
                                    QXLRect *qxl_area, QXLRect *qxl_dirty_rects,
                                    uint32_t num_dirty_rects, uint32_t clear_dirty_region)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_UPDATE;
+    RedWorkerMessageUpdate payload;
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &surface_id, sizeof(uint32_t));
-    send_data(dispatcher->channel, &qxl_area, sizeof(QXLRect *));
-    send_data(dispatcher->channel, &qxl_dirty_rects, sizeof(QXLRect *));
-    send_data(dispatcher->channel, &num_dirty_rects, sizeof(uint32_t));
-    send_data(dispatcher->channel, &clear_dirty_region, sizeof(uint32_t));
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    payload.surface_id = surface_id;
+    payload.qxl_area = qxl_area;
+    payload.qxl_dirty_rects = qxl_dirty_rects;
+    payload.num_dirty_rects = num_dirty_rects;
+    payload.clear_dirty_region = clear_dirty_region;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_UPDATE,
+                            &payload);
 }
 
 static AsyncCommand *async_command_alloc(RedDispatcher *dispatcher,
@@ -297,13 +307,15 @@ static void red_dispatcher_update_area_async(RedDispatcher *dispatcher,
                                          uint64_t cookie)
 {
     RedWorkerMessage message = RED_WORKER_MESSAGE_UPDATE_ASYNC;
-    AsyncCommand *cmd = async_command_alloc(dispatcher, message, cookie);
+    RedWorkerMessageUpdateAsync payload;
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &cmd, sizeof(cmd));
-    send_data(dispatcher->channel, &surface_id, sizeof(uint32_t));
-    send_data(dispatcher->channel, qxl_area, sizeof(QXLRect));
-    send_data(dispatcher->channel, &clear_dirty_region, sizeof(uint32_t));
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    payload.surface_id = surface_id;
+    payload.qxl_area = *qxl_area;
+    payload.clear_dirty_region = clear_dirty_region;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            message,
+                            &payload);
 }
 
 static void qxl_worker_update_area(QXLWorker *qxl_worker, uint32_t surface_id,
@@ -316,12 +328,12 @@ static void qxl_worker_update_area(QXLWorker *qxl_worker, uint32_t surface_id,
 
 static void red_dispatcher_add_memslot(RedDispatcher *dispatcher, QXLDevMemSlot *mem_slot)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_ADD_MEMSLOT;
+    RedWorkerMessageAddMemslot payload;
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, mem_slot, sizeof(QXLDevMemSlot));
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    payload.mem_slot = *mem_slot;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_ADD_MEMSLOT,
+                            &payload);
 }
 
 static void qxl_worker_add_memslot(QXLWorker *qxl_worker, QXLDevMemSlot *mem_slot)
@@ -331,21 +343,22 @@ static void qxl_worker_add_memslot(QXLWorker *qxl_worker, QXLDevMemSlot *mem_slo
 
 static void red_dispatcher_add_memslot_async(RedDispatcher *dispatcher, QXLDevMemSlot *mem_slot, uint64_t cookie)
 {
+    RedWorkerMessageAddMemslotAsync payload;
     RedWorkerMessage message = RED_WORKER_MESSAGE_ADD_MEMSLOT_ASYNC;
-    AsyncCommand *cmd = async_command_alloc(dispatcher, message, cookie);
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &cmd, sizeof(cmd));
-    send_data(dispatcher->channel, mem_slot, sizeof(QXLDevMemSlot));
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    payload.mem_slot = *mem_slot;
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
 }
 
 static void red_dispatcher_del_memslot(RedDispatcher *dispatcher, uint32_t slot_group_id, uint32_t slot_id)
 {
+    RedWorkerMessageDelMemslot payload;
     RedWorkerMessage message = RED_WORKER_MESSAGE_DEL_MEMSLOT;
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &slot_group_id, sizeof(uint32_t));
-    send_data(dispatcher->channel, &slot_id, sizeof(uint32_t));
+    payload.slot_group_id = slot_group_id;
+    payload.slot_id = slot_id;
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
 }
 
 static void qxl_worker_del_memslot(QXLWorker *qxl_worker, uint32_t slot_group_id, uint32_t slot_id)
@@ -355,11 +368,11 @@ static void qxl_worker_del_memslot(QXLWorker *qxl_worker, uint32_t slot_group_id
 
 static void red_dispatcher_destroy_surfaces(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_DESTROY_SURFACES;
+    RedWorkerMessageDestroySurfaces payload;
 
-    write_message(dispatcher->channel, &message);
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DESTROY_SURFACES,
+                            &payload);
 }
 
 static void qxl_worker_destroy_surfaces(QXLWorker *qxl_worker)
@@ -369,11 +382,11 @@ static void qxl_worker_destroy_surfaces(QXLWorker *qxl_worker)
 
 static void red_dispatcher_destroy_surfaces_async(RedDispatcher *dispatcher, uint64_t cookie)
 {
+    RedWorkerMessageDestroySurfacesAsync payload;
     RedWorkerMessage message = RED_WORKER_MESSAGE_DESTROY_SURFACES_ASYNC;
-    AsyncCommand *cmd = async_command_alloc(dispatcher, message, cookie);
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &cmd, sizeof(cmd));
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
 }
 
 static void red_dispatcher_destroy_primary_surface_complete(RedDispatcher *dispatcher)
@@ -387,28 +400,37 @@ static void red_dispatcher_destroy_primary_surface_complete(RedDispatcher *dispa
 }
 
 static void
+red_dispatcher_destroy_primary_surface_sync(RedDispatcher *dispatcher,
+                                            uint32_t surface_id)
+{
+    RedWorkerMessageDestroyPrimarySurface payload;
+    payload.surface_id = surface_id;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE,
+                            &payload);
+    red_dispatcher_destroy_primary_surface_complete(dispatcher);
+}
+
+static void
+red_dispatcher_destroy_primary_surface_async(RedDispatcher *dispatcher,
+                                             uint32_t surface_id, uint64_t cookie)
+{
+    RedWorkerMessageDestroyPrimarySurfaceAsync payload;
+    RedWorkerMessage message = RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC;
+
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    payload.surface_id = surface_id;
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
+}
+
+static void
 red_dispatcher_destroy_primary_surface(RedDispatcher *dispatcher,
                                        uint32_t surface_id, int async, uint64_t cookie)
 {
-    RedWorkerMessage message;
-    AsyncCommand *cmd = NULL;
-
     if (async) {
-        message = RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC;
-        cmd = async_command_alloc(dispatcher, message, cookie);
+        red_dispatcher_destroy_primary_surface_async(dispatcher, surface_id, cookie);
     } else {
-        message = RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE;
-    }
-
-    write_message(dispatcher->channel, &message);
-    if (async) {
-        send_data(dispatcher->channel, &cmd, sizeof(cmd));
-    }
-    send_data(dispatcher->channel, &surface_id, sizeof(uint32_t));
-    if (!async) {
-        read_message(dispatcher->channel, &message);
-        ASSERT(message == RED_WORKER_MESSAGE_READY);
-        red_dispatcher_destroy_primary_surface_complete(dispatcher);
+        red_dispatcher_destroy_primary_surface_sync(dispatcher, surface_id);
     }
 }
 
@@ -431,30 +453,42 @@ static void red_dispatcher_create_primary_surface_complete(RedDispatcher *dispat
 }
 
 static void
-red_dispatcher_create_primary_surface(RedDispatcher *dispatcher, uint32_t surface_id,
-                                      QXLDevSurfaceCreate *surface, int async, uint64_t cookie)
+red_dispatcher_create_primary_surface_async(RedDispatcher *dispatcher, uint32_t surface_id,
+                                            QXLDevSurfaceCreate *surface, uint64_t cookie)
 {
-    RedWorkerMessage message;
-    AsyncCommand *cmd = NULL;
+    RedWorkerMessageCreatePrimarySurfaceAsync payload;
+    RedWorkerMessage message = RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC;
+
+    dispatcher->surface_create = *surface;
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    payload.surface_id = surface_id;
+    payload.surface = *surface;
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
+}
+
+static void
+red_dispatcher_create_primary_surface_sync(RedDispatcher *dispatcher, uint32_t surface_id,
+                                           QXLDevSurfaceCreate *surface)
+{
+    RedWorkerMessageCreatePrimarySurface payload;
 
-    if (async) {
-        message = RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC;
-        cmd = async_command_alloc(dispatcher, message, cookie);
-    } else {
-        message = RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE;
-    }
     dispatcher->surface_create = *surface;
+    payload.surface_id = surface_id;
+    payload.surface = *surface;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE,
+                            &payload);
+    red_dispatcher_create_primary_surface_complete(dispatcher);
+}
 
-    write_message(dispatcher->channel, &message);
+static void
+red_dispatcher_create_primary_surface(RedDispatcher *dispatcher, uint32_t surface_id,
+                                      QXLDevSurfaceCreate *surface, int async, uint64_t cookie)
+{
     if (async) {
-        send_data(dispatcher->channel, &cmd, sizeof(cmd));
-    }
-    send_data(dispatcher->channel, &surface_id, sizeof(uint32_t));
-    send_data(dispatcher->channel, surface, sizeof(QXLDevSurfaceCreate));
-    if (!async) {
-        read_message(dispatcher->channel, &message);
-        ASSERT(message == RED_WORKER_MESSAGE_READY);
-        red_dispatcher_create_primary_surface_complete(dispatcher);
+        red_dispatcher_create_primary_surface_async(dispatcher, surface_id, surface, cookie);
+    } else {
+        red_dispatcher_create_primary_surface_sync(dispatcher, surface_id, surface);
     }
 }
 
@@ -466,11 +500,11 @@ static void qxl_worker_create_primary_surface(QXLWorker *qxl_worker, uint32_t su
 
 static void red_dispatcher_reset_image_cache(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_RESET_IMAGE_CACHE;
+    RedWorkerMessageResetImageCache payload;
 
-    write_message(dispatcher->channel, &message);
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_RESET_IMAGE_CACHE,
+                            &payload);
 }
 
 static void qxl_worker_reset_image_cache(QXLWorker *qxl_worker)
@@ -480,11 +514,11 @@ static void qxl_worker_reset_image_cache(QXLWorker *qxl_worker)
 
 static void red_dispatcher_reset_cursor(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_RESET_CURSOR;
+    RedWorkerMessageResetCursor payload;
 
-    write_message(dispatcher->channel, &message);
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_RESET_CURSOR,
+                            &payload);
 }
 
 static void qxl_worker_reset_cursor(QXLWorker *qxl_worker)
@@ -492,29 +526,35 @@ static void qxl_worker_reset_cursor(QXLWorker *qxl_worker)
     red_dispatcher_reset_cursor((RedDispatcher*)qxl_worker);
 }
 
-static void red_dispatcher_destroy_surface_wait(RedDispatcher *dispatcher, uint32_t surface_id,
-                                                int async, uint64_t cookie)
+static void red_dispatcher_destroy_surface_wait_sync(RedDispatcher *dispatcher, uint32_t surface_id)
 {
-    RedWorkerMessage message;
-    AsyncCommand *cmd = NULL;
+    RedWorkerMessageDestroySurfaceWait payload;
 
-    if (async ) {
-        message = RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC;
-        cmd = async_command_alloc(dispatcher, message, cookie);
-    } else {
-        message = RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT;
-    }
+    payload.surface_id = surface_id;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT,
+                            &payload);
+}
+                                                    
+static void red_dispatcher_destroy_surface_wait_async(RedDispatcher *dispatcher, uint32_t surface_id,
+                                                      uint64_t cookie)
+{
+    RedWorkerMessageDestroySurfaceWaitAsync payload;
+    RedWorkerMessage message = RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC;
 
-    write_message(dispatcher->channel, &message);
-    if (async) {
-        send_data(dispatcher->channel, &cmd, sizeof(cmd));
-    }
-    send_data(dispatcher->channel, &surface_id, sizeof(uint32_t));
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    payload.surface_id = surface_id;
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
+}
+
+static void red_dispatcher_destroy_surface_wait(RedDispatcher *dispatcher, uint32_t surface_id,
+                                                int async, uint64_t cookie)
+{
     if (async) {
-        return;
+        red_dispatcher_destroy_surface_wait_async(dispatcher, surface_id, cookie);
+    } else {
+        red_dispatcher_destroy_surface_wait_sync(dispatcher, surface_id);
     }
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
 }
 
 static void qxl_worker_destroy_surface_wait(QXLWorker *qxl_worker, uint32_t surface_id)
@@ -524,9 +564,11 @@ static void qxl_worker_destroy_surface_wait(QXLWorker *qxl_worker, uint32_t surf
 
 static void red_dispatcher_reset_memslots(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_RESET_MEMSLOTS;
+    RedWorkerMessageResetMemslots payload;
 
-    write_message(dispatcher->channel, &message);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_RESET_MEMSLOTS,
+                            &payload);
 }
 
 static void qxl_worker_reset_memslots(QXLWorker *qxl_worker)
@@ -536,11 +578,15 @@ static void qxl_worker_reset_memslots(QXLWorker *qxl_worker)
 
 static void red_dispatcher_wakeup(RedDispatcher *dispatcher)
 {
-    if (!test_bit(RED_WORKER_PENDING_WAKEUP, dispatcher->pending)) {
-        RedWorkerMessage message = RED_WORKER_MESSAGE_WAKEUP;
-        set_bit(RED_WORKER_PENDING_WAKEUP, &dispatcher->pending);
-        write_message(dispatcher->channel, &message);
+    RedWorkerMessageWakeup payload;
+
+    if (test_bit(RED_WORKER_PENDING_WAKEUP, dispatcher->pending)) {
+        return;
     }
+    set_bit(RED_WORKER_PENDING_WAKEUP, &dispatcher->pending);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_WAKEUP,
+                            &payload);
 }
 
 static void qxl_worker_wakeup(QXLWorker *qxl_worker)
@@ -550,11 +596,15 @@ static void qxl_worker_wakeup(QXLWorker *qxl_worker)
 
 static void red_dispatcher_oom(RedDispatcher *dispatcher)
 {
-    if (!test_bit(RED_WORKER_PENDING_OOM, dispatcher->pending)) {
-        RedWorkerMessage message = RED_WORKER_MESSAGE_OOM;
-        set_bit(RED_WORKER_PENDING_OOM, &dispatcher->pending);
-        write_message(dispatcher->channel, &message);
+    RedWorkerMessageOom payload;
+
+    if (test_bit(RED_WORKER_PENDING_OOM, dispatcher->pending)) {
+        return;
     }
+    set_bit(RED_WORKER_PENDING_OOM, &dispatcher->pending);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_OOM,
+                            &payload);
 }
 
 static void qxl_worker_oom(QXLWorker *qxl_worker)
@@ -564,9 +614,11 @@ static void qxl_worker_oom(QXLWorker *qxl_worker)
 
 static void red_dispatcher_start(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_START;
+    RedWorkerMessageStart payload;
 
-    write_message(dispatcher->channel, &message);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_START,
+                            &payload);
 }
 
 static void qxl_worker_start(QXLWorker *qxl_worker)
@@ -576,20 +628,20 @@ static void qxl_worker_start(QXLWorker *qxl_worker)
 
 static void red_dispatcher_flush_surfaces_async(RedDispatcher *dispatcher, uint64_t cookie)
 {
+    RedWorkerMessageFlushSurfacesAsync payload;
     RedWorkerMessage message = RED_WORKER_MESSAGE_FLUSH_SURFACES_ASYNC;
-    AsyncCommand *cmd = async_command_alloc(dispatcher, message, cookie);
 
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &cmd, sizeof(cmd));
+    payload.cmd = async_command_alloc(dispatcher, message, cookie);
+    dispatcher_send_message(&dispatcher->dispatcher, message, &payload);
 }
 
 static void red_dispatcher_stop(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_STOP;
+    RedWorkerMessageStop payload;
 
-    write_message(dispatcher->channel, &message);
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_STOP,
+                            &payload);
 }
 
 static void qxl_worker_stop(QXLWorker *qxl_worker)
@@ -601,14 +653,14 @@ static void red_dispatcher_loadvm_commands(RedDispatcher *dispatcher,
                                            struct QXLCommandExt *ext,
                                            uint32_t count)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_LOADVM_COMMANDS;
+    RedWorkerMessageLoadvmCommands payload;
 
     red_printf("");
-    write_message(dispatcher->channel, &message);
-    send_data(dispatcher->channel, &count, sizeof(uint32_t));
-    send_data(dispatcher->channel, ext, sizeof(QXLCommandExt) * count);
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    payload.count = count;
+    payload.ext = ext;
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_LOADVM_COMMANDS,
+                            &payload);
 }
 
 static void qxl_worker_loadvm_commands(QXLWorker *qxl_worker,
@@ -640,37 +692,44 @@ static inline int calc_compression_level(void)
 
 void red_dispatcher_on_ic_change(void)
 {
+    RedWorkerMessageSetCompression payload;
     int compression_level = calc_compression_level();
     RedDispatcher *now = dispatchers;
+
     while (now) {
-        RedWorkerMessage message = RED_WORKER_MESSAGE_SET_COMPRESSION;
         now->qxl->st->qif->set_compression_level(now->qxl, compression_level);
-        write_message(now->channel, &message);
-        send_data(now->channel, &image_compression, sizeof(spice_image_compression_t));
+        payload.image_compression = image_compression;
+        dispatcher_send_message(&now->dispatcher,
+                                RED_WORKER_MESSAGE_SET_COMPRESSION,
+                                &payload);
         now = now->next;
     }
 }
 
 void red_dispatcher_on_sv_change(void)
 {
+    RedWorkerMessageSetStreamingVideo payload;
     int compression_level = calc_compression_level();
     RedDispatcher *now = dispatchers;
     while (now) {
-        RedWorkerMessage message = RED_WORKER_MESSAGE_SET_STREAMING_VIDEO;
         now->qxl->st->qif->set_compression_level(now->qxl, compression_level);
-        write_message(now->channel, &message);
-        send_data(now->channel, &streaming_video, sizeof(uint32_t));
+        payload.streaming_video = streaming_video;
+        dispatcher_send_message(&now->dispatcher,
+                                RED_WORKER_MESSAGE_SET_STREAMING_VIDEO,
+                                &payload);
         now = now->next;
     }
 }
 
 void red_dispatcher_set_mouse_mode(uint32_t mode)
 {
+    RedWorkerMessageSetMouseMode payload;
     RedDispatcher *now = dispatchers;
     while (now) {
-        RedWorkerMessage message = RED_WORKER_MESSAGE_SET_MOUSE_MODE;
-        write_message(now->channel, &message);
-        send_data(now->channel, &mode, sizeof(uint32_t));
+        payload.mode = mode;
+        dispatcher_send_message(&now->dispatcher,
+                                RED_WORKER_MESSAGE_SET_MOUSE_MODE,
+                                &payload);
         now = now->next;
     }
 }
@@ -873,34 +932,31 @@ void red_dispatcher_async_complete(struct RedDispatcher *dispatcher,
 
 static RedChannel *red_dispatcher_display_channel_create(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE;
+    RedWorkerMessageDisplayChannelCreate payload;
     RedChannel *display_channel;
 
-    write_message(dispatcher->channel, &message);
-
-    receive_data(dispatcher->channel, &display_channel, sizeof(RedChannel *));
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE,
+                            &payload);
+    receive_data(dispatcher->dispatcher.send_fd, &display_channel, sizeof(RedChannel *));
     return display_channel;
 }
 
 static RedChannel *red_dispatcher_cursor_channel_create(RedDispatcher *dispatcher)
 {
-    RedWorkerMessage message = RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE;
+    RedWorkerMessageCursorChannelCreate payload;
     RedChannel *cursor_channel;
 
-    write_message(dispatcher->channel, &message);
-
-    receive_data(dispatcher->channel, &cursor_channel, sizeof(RedChannel *));
-    read_message(dispatcher->channel, &message);
-    ASSERT(message == RED_WORKER_MESSAGE_READY);
+    dispatcher_send_message(&dispatcher->dispatcher,
+                            RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE,
+                            &payload);
+    receive_data(dispatcher->dispatcher.send_fd, &cursor_channel, sizeof(RedChannel *));
     return cursor_channel;
 }
 
 RedDispatcher *red_dispatcher_init(QXLInstance *qxl)
 {
-    RedDispatcher *dispatcher;
-    int channels[2];
+    RedDispatcher *red_dispatcher;
     RedWorkerMessage message;
     WorkerInitData init_data;
     QXLDevInitInfo init_info;
@@ -917,45 +973,41 @@ RedDispatcher *red_dispatcher_init(QXLInstance *qxl)
     gl_canvas_init();
 #endif // USE_OPENGL
 
-    if (socketpair(AF_LOCAL, SOCK_STREAM, 0, channels) == -1) {
-        red_error("socketpair failed %s", strerror(errno));
-    }
-
-    dispatcher = spice_new0(RedDispatcher, 1);
-    dispatcher->channel = channels[0];
-    ring_init(&dispatcher->async_commands);
-    DBG_ASYNC("dispatcher->async_commands.next %p", dispatcher->async_commands.next);
-    init_data.qxl = dispatcher->qxl = qxl;
+    red_dispatcher = spice_new0(RedDispatcher, 1);
+    ring_init(&red_dispatcher->async_commands);
+    DBG_ASYNC("red_dispatcher->async_commands.next %p", red_dispatcher->async_commands.next);
+    dispatcher_init(&red_dispatcher->dispatcher, RED_WORKER_MESSAGE_COUNT, NULL);
+    init_data.qxl = red_dispatcher->qxl = qxl;
     init_data.id = qxl->id;
-    init_data.channel = channels[1];
-    init_data.pending = &dispatcher->pending;
+    init_data.red_dispatcher = red_dispatcher;
+    init_data.pending = &red_dispatcher->pending;
     init_data.num_renderers = num_renderers;
     memcpy(init_data.renderers, renderers, sizeof(init_data.renderers));
 
-    pthread_mutex_init(&dispatcher->async_lock, NULL);
+    pthread_mutex_init(&red_dispatcher->async_lock, NULL);
     init_data.image_compression = image_compression;
     init_data.jpeg_state = jpeg_state;
     init_data.zlib_glz_state = zlib_glz_state;
     init_data.streaming_video = streaming_video;
 
-    dispatcher->base.major_version = SPICE_INTERFACE_QXL_MAJOR;
-    dispatcher->base.minor_version = SPICE_INTERFACE_QXL_MINOR;
-    dispatcher->base.wakeup = qxl_worker_wakeup;
-    dispatcher->base.oom = qxl_worker_oom;
-    dispatcher->base.start = qxl_worker_start;
-    dispatcher->base.stop = qxl_worker_stop;
-    dispatcher->base.update_area = qxl_worker_update_area;
-    dispatcher->base.add_memslot = qxl_worker_add_memslot;
-    dispatcher->base.del_memslot = qxl_worker_del_memslot;
-    dispatcher->base.reset_memslots = qxl_worker_reset_memslots;
-    dispatcher->base.destroy_surfaces = qxl_worker_destroy_surfaces;
-    dispatcher->base.create_primary_surface = qxl_worker_create_primary_surface;
-    dispatcher->base.destroy_primary_surface = qxl_worker_destroy_primary_surface;
-
-    dispatcher->base.reset_image_cache = qxl_worker_reset_image_cache;
-    dispatcher->base.reset_cursor = qxl_worker_reset_cursor;
-    dispatcher->base.destroy_surface_wait = qxl_worker_destroy_surface_wait;
-    dispatcher->base.loadvm_commands = qxl_worker_loadvm_commands;
+    red_dispatcher->base.major_version = SPICE_INTERFACE_QXL_MAJOR;
+    red_dispatcher->base.minor_version = SPICE_INTERFACE_QXL_MINOR;
+    red_dispatcher->base.wakeup = qxl_worker_wakeup;
+    red_dispatcher->base.oom = qxl_worker_oom;
+    red_dispatcher->base.start = qxl_worker_start;
+    red_dispatcher->base.stop = qxl_worker_stop;
+    red_dispatcher->base.update_area = qxl_worker_update_area;
+    red_dispatcher->base.add_memslot = qxl_worker_add_memslot;
+    red_dispatcher->base.del_memslot = qxl_worker_del_memslot;
+    red_dispatcher->base.reset_memslots = qxl_worker_reset_memslots;
+    red_dispatcher->base.destroy_surfaces = qxl_worker_destroy_surfaces;
+    red_dispatcher->base.create_primary_surface = qxl_worker_create_primary_surface;
+    red_dispatcher->base.destroy_primary_surface = qxl_worker_destroy_primary_surface;
+
+    red_dispatcher->base.reset_image_cache = qxl_worker_reset_image_cache;
+    red_dispatcher->base.reset_cursor = qxl_worker_reset_cursor;
+    red_dispatcher->base.destroy_surface_wait = qxl_worker_destroy_surface_wait;
+    red_dispatcher->base.loadvm_commands = qxl_worker_loadvm_commands;
 
     qxl->st->qif->get_init_info(qxl, &init_info);
 
@@ -965,7 +1017,6 @@ RedDispatcher *red_dispatcher_init(QXLInstance *qxl)
     init_data.num_memslots_groups = init_info.num_memslots_groups;
     init_data.internal_groupslot_id = init_info.internal_groupslot_id;
     init_data.n_surfaces = init_info.n_surfaces;
-    init_data.dispatcher = dispatcher;
 
     num_active_workers = 1;
 
@@ -974,40 +1025,51 @@ RedDispatcher *red_dispatcher_init(QXLInstance *qxl)
     sigdelset(&thread_sig_mask, SIGFPE);
     sigdelset(&thread_sig_mask, SIGSEGV);
     pthread_sigmask(SIG_SETMASK, &thread_sig_mask, &curr_sig_mask);
-    if ((r = pthread_create(&dispatcher->worker_thread, NULL, red_worker_main, &init_data))) {
+    if ((r = pthread_create(&red_dispatcher->worker_thread, NULL, red_worker_main, &init_data))) {
         red_error("create thread failed %d", r);
     }
     pthread_sigmask(SIG_SETMASK, &curr_sig_mask, NULL);
 
-    read_message(dispatcher->channel, &message);
+    read_message(red_dispatcher->dispatcher.send_fd, &message);
     ASSERT(message == RED_WORKER_MESSAGE_READY);
 
-    display_channel = red_dispatcher_display_channel_create(dispatcher);
+    display_channel = red_dispatcher_display_channel_create(red_dispatcher);
 
     if (display_channel) {
         client_cbs.connect = red_dispatcher_set_display_peer;
         client_cbs.disconnect = red_dispatcher_disconnect_display_peer;
         client_cbs.migrate = red_dispatcher_display_migrate;
         red_channel_register_client_cbs(display_channel, &client_cbs);
-        red_channel_set_data(display_channel, dispatcher);
+        red_channel_set_data(display_channel, red_dispatcher);
         reds_register_channel(display_channel);
     }
 
-    cursor_channel = red_dispatcher_cursor_channel_create(dispatcher);
+    cursor_channel = red_dispatcher_cursor_channel_create(red_dispatcher);
 
     if (cursor_channel) {
         client_cbs.connect = red_dispatcher_set_cursor_peer;
         client_cbs.disconnect = red_dispatcher_disconnect_cursor_peer;
         client_cbs.migrate = red_dispatcher_cursor_migrate;
         red_channel_register_client_cbs(cursor_channel, &client_cbs);
-        red_channel_set_data(cursor_channel, dispatcher);
+        red_channel_set_data(cursor_channel, red_dispatcher);
         reds_register_channel(cursor_channel);
     }
 
-    qxl->st->qif->attache_worker(qxl, &dispatcher->base);
+    qxl->st->qif->attache_worker(qxl, &red_dispatcher->base);
     qxl->st->qif->set_compression_level(qxl, calc_compression_level());
 
-    dispatcher->next = dispatchers;
-    dispatchers = dispatcher;
-    return dispatcher;
+    red_dispatcher->next = dispatchers;
+    dispatchers = red_dispatcher;
+    return red_dispatcher;
+}
+
+struct Dispatcher *red_dispatcher_get_dispatcher(RedDispatcher *red_dispatcher)
+{
+    return &red_dispatcher->dispatcher;
+}
+
+void red_dispatcher_set_dispatcher_opaque(struct RedDispatcher *red_dispatcher,
+                                          void *opaque)
+{
+    dispatcher_set_opaque(&red_dispatcher->dispatcher, opaque);
 }
diff --git a/server/red_dispatcher.h b/server/red_dispatcher.h
index c2582f4..59edb44 100644
--- a/server/red_dispatcher.h
+++ b/server/red_dispatcher.h
@@ -32,5 +32,150 @@ int red_dispatcher_add_renderer(const char *name);
 uint32_t red_dispatcher_qxl_ram_size(void);
 int red_dispatcher_qxl_count(void);
 void red_dispatcher_async_complete(struct RedDispatcher *, AsyncCommand *);
+struct Dispatcher *red_dispatcher_get_dispatcher(struct RedDispatcher *);
+
+typedef struct RedWorkerMessageDisplayConnect {
+    RedClient * client;
+    RedsStream * stream;
+    int migration;
+} RedWorkerMessageDisplayConnect;
+
+typedef struct RedWorkerMessageDisplayDisconnect {
+    RedChannelClient *rcc;
+} RedWorkerMessageDisplayDisconnect;
+
+typedef struct RedWorkerMessageDisplayMigrate {
+    RedChannelClient *rcc;
+} RedWorkerMessageDisplayMigrate;
+
+typedef struct RedWorkerMessageCursorConnect {
+    RedClient *client;
+    RedsStream *stream;
+    int migration;
+} RedWorkerMessageCursorConnect;
+
+typedef struct RedWorkerMessageCursorDisconnect {
+    RedChannelClient *rcc;
+} RedWorkerMessageCursorDisconnect;
+
+typedef struct RedWorkerMessageCursorMigrate {
+    RedChannelClient *rcc;
+} RedWorkerMessageCursorMigrate;
+
+typedef struct RedWorkerMessageUpdate {
+    uint32_t surface_id;
+    QXLRect * qxl_area;
+    QXLRect * qxl_dirty_rects;
+    uint32_t num_dirty_rects;
+    uint32_t clear_dirty_region;
+} RedWorkerMessageUpdate;
+
+typedef struct RedWorkerMessageUpdateAsync {
+    AsyncCommand *cmd;
+    uint32_t surface_id;
+    QXLRect qxl_area;
+    uint32_t clear_dirty_region;
+} RedWorkerMessageUpdateAsync;
+
+typedef struct RedWorkerMessageAddMemslot {
+    QXLDevMemSlot mem_slot;
+} RedWorkerMessageAddMemslot;
+
+typedef struct RedWorkerMessageAddMemslotAsync {
+    AsyncCommand *cmd;
+    QXLDevMemSlot mem_slot;
+} RedWorkerMessageAddMemslotAsync;
+
+typedef struct RedWorkerMessageDelMemslot {
+    uint32_t slot_group_id;
+    uint32_t slot_id;
+} RedWorkerMessageDelMemslot;
+
+typedef struct RedWorkerMessageDestroySurfaces {
+} RedWorkerMessageDestroySurfaces;
+
+typedef struct RedWorkerMessageDestroySurfacesAsync {
+    AsyncCommand *cmd;
+} RedWorkerMessageDestroySurfacesAsync;
+
+
+typedef struct RedWorkerMessageDestroyPrimarySurface {
+    uint32_t surface_id;
+} RedWorkerMessageDestroyPrimarySurface;
+
+typedef struct RedWorkerMessageDestroyPrimarySurfaceAsync {
+    AsyncCommand *cmd;
+    uint32_t surface_id;
+} RedWorkerMessageDestroyPrimarySurfaceAsync;
+
+typedef struct RedWorkerMessageCreatePrimarySurfaceAsync {
+    AsyncCommand *cmd;
+    uint32_t surface_id;
+    QXLDevSurfaceCreate surface;
+} RedWorkerMessageCreatePrimarySurfaceAsync;
+
+typedef struct RedWorkerMessageCreatePrimarySurface {
+    uint32_t surface_id;
+    QXLDevSurfaceCreate surface;
+} RedWorkerMessageCreatePrimarySurface;
+
+typedef struct RedWorkerMessageResetImageCache {
+} RedWorkerMessageResetImageCache;
+
+typedef struct RedWorkerMessageResetCursor {
+} RedWorkerMessageResetCursor;
+
+typedef struct RedWorkerMessageWakeup {
+} RedWorkerMessageWakeup;
+
+typedef struct RedWorkerMessageOom {
+} RedWorkerMessageOom;
+
+typedef struct RedWorkerMessageStart {
+} RedWorkerMessageStart;
+
+typedef struct RedWorkerMessageFlushSurfacesAsync {
+    AsyncCommand *cmd;
+} RedWorkerMessageFlushSurfacesAsync;
+
+typedef struct RedWorkerMessageStop {
+} RedWorkerMessageStop;
+
+/* this command is sync, so it's ok to pass a pointer */
+typedef struct RedWorkerMessageLoadvmCommands {
+    uint32_t count;
+    QXLCommandExt *ext;
+} RedWorkerMessageLoadvmCommands;
+
+typedef struct RedWorkerMessageSetCompression {
+    spice_image_compression_t image_compression;
+} RedWorkerMessageSetCompression;
+
+typedef struct RedWorkerMessageSetStreamingVideo {
+    uint32_t streaming_video;
+} RedWorkerMessageSetStreamingVideo;
+
+typedef struct RedWorkerMessageSetMouseMode {
+    uint32_t mode;
+} RedWorkerMessageSetMouseMode;
+
+typedef struct RedWorkerMessageDisplayChannelCreate {
+} RedWorkerMessageDisplayChannelCreate;
+
+typedef struct RedWorkerMessageCursorChannelCreate {
+} RedWorkerMessageCursorChannelCreate;
+
+typedef struct RedWorkerMessageDestroySurfaceWait {
+    uint32_t surface_id;
+} RedWorkerMessageDestroySurfaceWait;
+
+typedef struct RedWorkerMessageDestroySurfaceWaitAsync {
+    AsyncCommand *cmd;
+    uint32_t surface_id;
+} RedWorkerMessageDestroySurfaceWaitAsync;
+
+typedef struct RedWorkerMessageResetMemslots {
+} RedWorkerMessageResetMemslots;
+
 
 #endif
diff --git a/server/red_worker.c b/server/red_worker.c
index de8a820..0bb5b0c 100644
--- a/server/red_worker.c
+++ b/server/red_worker.c
@@ -72,6 +72,7 @@
 #include "zlib_encoder.h"
 #include "red_channel.h"
 #include "red_dispatcher.h"
+#include "dispatcher.h"
 #include "main_channel.h"
 
 //#define COMPRESS_STAT
@@ -865,9 +866,10 @@ typedef struct RedWorker {
     DisplayChannel *display_channel;
     CursorChannel *cursor_channel;
     QXLInstance *qxl;
-    RedDispatcher *dispatcher;
-    int id;
+    RedDispatcher *red_dispatcher;
+
     int channel;
+    int id;
     int running;
     uint32_t *pending;
     int epoll;
@@ -10103,21 +10105,19 @@ static void surface_dirty_region_to_rects(RedSurface *surface,
     free(dirty_rects);
 }
 
-static inline void handle_dev_update_async(RedWorker *worker)
+void handle_dev_update_async(void *opaque, void *payload)
 {
-    QXLRect qxl_rect;
+    RedWorker *worker = opaque;
+    RedWorkerMessageUpdateAsync *msg = payload;
     SpiceRect rect;
-    uint32_t surface_id;
-    uint32_t clear_dirty_region;
     QXLRect *qxl_dirty_rects;
     uint32_t num_dirty_rects;
     RedSurface *surface;
+    uint32_t surface_id = msg->surface_id;
+    QXLRect qxl_area = msg->qxl_area;
+    uint32_t clear_dirty_region = msg->clear_dirty_region;
 
-    receive_data(worker->channel, &surface_id, sizeof(uint32_t));
-    receive_data(worker->channel, &qxl_rect, sizeof(QXLRect));
-    receive_data(worker->channel, &clear_dirty_region, sizeof(uint32_t));
-
-    red_get_rect_ptr(&rect, &qxl_rect);
+    red_get_rect_ptr(&rect, &qxl_area);
     flush_display_commands(worker);
 
     ASSERT(worker->running);
@@ -10125,12 +10125,12 @@ static inline void handle_dev_update_async(RedWorker *worker)
     validate_surface(worker, surface_id);
     red_update_area(worker, &rect, surface_id);
     if (!worker->qxl->st->qif->update_area_complete) {
-        return;
+        goto complete;
     }
     surface = &worker->surfaces[surface_id];
     num_dirty_rects = pixman_region32_n_rects(&surface->draw_dirty_region);
     if (num_dirty_rects == 0) {
-        return;
+        goto complete;
     }
     qxl_dirty_rects = spice_new0(QXLRect, num_dirty_rects);
     surface_dirty_region_to_rects(surface, qxl_dirty_rects, num_dirty_rects,
@@ -10138,26 +10138,25 @@ static inline void handle_dev_update_async(RedWorker *worker)
     worker->qxl->st->qif->update_area_complete(worker->qxl, surface_id,
                                           qxl_dirty_rects, num_dirty_rects);
     free(qxl_dirty_rects);
+
+complete:
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
 }
 
-static inline void handle_dev_update(RedWorker *worker)
+void handle_dev_update(void *opaque, void *payload)
 {
-    const QXLRect *qxl_rect;
+    RedWorker *worker = opaque;
+    RedWorkerMessageUpdate *msg = payload;
     SpiceRect *rect = spice_new0(SpiceRect, 1);
-    QXLRect *qxl_dirty_rects;
     RedSurface *surface;
-    uint32_t num_dirty_rects;
-    uint32_t surface_id;
-    uint32_t clear_dirty_region;
-
-    receive_data(worker->channel, &surface_id, sizeof(uint32_t));
-    receive_data(worker->channel, &qxl_rect, sizeof(QXLRect *));
-    receive_data(worker->channel, &qxl_dirty_rects, sizeof(QXLRect *));
-    receive_data(worker->channel, &num_dirty_rects, sizeof(uint32_t));
-    receive_data(worker->channel, &clear_dirty_region, sizeof(uint32_t));
+    uint32_t surface_id = msg->surface_id;
+    const QXLRect *qxl_area = msg->qxl_area;
+    uint32_t num_dirty_rects = msg->num_dirty_rects;
+    QXLRect *qxl_dirty_rects = msg->qxl_dirty_rects;
+    uint32_t clear_dirty_region = msg->clear_dirty_region;
 
     surface = &worker->surfaces[surface_id];
-    red_get_rect_ptr(rect, qxl_rect);
+    red_get_rect_ptr(rect, qxl_area);
     flush_display_commands(worker);
 
     ASSERT(worker->running);
@@ -10170,28 +10169,36 @@ static inline void handle_dev_update(RedWorker *worker)
                                   clear_dirty_region);
 }
 
-static inline void handle_dev_add_memslot(RedWorker *worker)
+static void dev_add_memslot(RedWorker *worker, QXLDevMemSlot mem_slot)
 {
-    QXLDevMemSlot dev_slot;
+    red_memslot_info_add_slot(&worker->mem_slots, mem_slot.slot_group_id, mem_slot.slot_id,
+                              mem_slot.addr_delta, mem_slot.virt_start, mem_slot.virt_end,
+                              mem_slot.generation);
+}
 
-    receive_data(worker->channel, &dev_slot, sizeof(QXLDevMemSlot));
+void handle_dev_add_memslot(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+    RedWorkerMessageAddMemslot *msg = payload;
+    QXLDevMemSlot mem_slot = msg->mem_slot;
 
-    red_memslot_info_add_slot(&worker->mem_slots, dev_slot.slot_group_id, dev_slot.slot_id,
-                              dev_slot.addr_delta, dev_slot.virt_start, dev_slot.virt_end,
-                              dev_slot.generation);
+    red_memslot_info_add_slot(&worker->mem_slots, mem_slot.slot_group_id, mem_slot.slot_id,
+                              mem_slot.addr_delta, mem_slot.virt_start, mem_slot.virt_end,
+                              mem_slot.generation);
 }
 
-static inline void handle_dev_del_memslot(RedWorker *worker)
+void handle_dev_del_memslot(void *opaque, void *payload)
 {
-    uint32_t slot_id;
-    uint32_t slot_group_id;
-
-    receive_data(worker->channel, &slot_group_id, sizeof(uint32_t));
-    receive_data(worker->channel, &slot_id, sizeof(uint32_t));
+    RedWorker *worker = opaque;
+    RedWorkerMessageDelMemslot *msg = payload;
+    uint32_t slot_id = msg->slot_id;
+    uint32_t slot_group_id = msg->slot_group_id;
 
     red_memslot_info_del_slot(&worker->mem_slots, slot_group_id, slot_id);
 }
 
+/* TODO: destroy_surface_wait, dev_destroy_surface_wait - confusing. one asserts
+ * surface_id == 0, maybe move the assert upward and merge the two functions? */
 static inline void destroy_surface_wait(RedWorker *worker, int surface_id)
 {
     if (!worker->surfaces[surface_id].context.canvas) {
@@ -10206,12 +10213,8 @@ static inline void destroy_surface_wait(RedWorker *worker, int surface_id)
     red_clear_surface_drawables_from_pipes(worker, surface_id, TRUE, TRUE);
 }
 
-static inline void handle_dev_destroy_surface_wait(RedWorker *worker)
+static void dev_destroy_surface_wait(RedWorker *worker, uint32_t surface_id)
 {
-    uint32_t surface_id;
-
-    receive_data(worker->channel, &surface_id, sizeof(uint32_t));
-
     ASSERT(surface_id == 0);
 
     flush_all_qxl_commands(worker);
@@ -10221,6 +10224,14 @@ static inline void handle_dev_destroy_surface_wait(RedWorker *worker)
     }
 }
 
+void handle_dev_destroy_surface_wait(void *opaque, void *payload)
+{
+    RedWorkerMessageDestroySurfaceWait *msg = payload;
+    RedWorker *worker = opaque;
+
+    dev_destroy_surface_wait(worker, msg->surface_id);
+}
+
 static inline void red_cursor_reset(RedWorker *worker)
 {
     if (worker->cursor) {
@@ -10244,7 +10255,7 @@ static inline void red_cursor_reset(RedWorker *worker)
 /* called upon device reset */
 
 /* TODO: split me*/
-static inline void handle_dev_destroy_surfaces(RedWorker *worker)
+static inline void dev_destroy_surfaces(RedWorker *worker)
 {
     int i;
 
@@ -10273,14 +10284,17 @@ static inline void handle_dev_destroy_surfaces(RedWorker *worker)
     red_cursor_reset(worker);
 }
 
-static inline void handle_dev_create_primary_surface(RedWorker *worker)
+void handle_dev_destroy_surfaces(void *opaque, void *payload)
 {
-    uint32_t surface_id;
-    QXLDevSurfaceCreate surface;
-    uint8_t *line_0;
+    RedWorker *worker = opaque;
 
-    receive_data(worker->channel, &surface_id, sizeof(uint32_t));
-    receive_data(worker->channel, &surface, sizeof(QXLDevSurfaceCreate));
+    dev_destroy_surfaces(worker);
+}
+
+static void dev_create_primary_surface(RedWorker *worker, uint32_t surface_id,
+                                       QXLDevSurfaceCreate surface)
+{
+    uint8_t *line_0;
 
     PANIC_ON(surface_id != 0);
     PANIC_ON(surface.height == 0);
@@ -10308,12 +10322,16 @@ static inline void handle_dev_create_primary_surface(RedWorker *worker)
     }
 }
 
-static inline void handle_dev_destroy_primary_surface(RedWorker *worker)
+void handle_dev_create_primary_surface(void *opaque, void *payload)
 {
-    uint32_t surface_id;
+    RedWorkerMessageCreatePrimarySurface *msg = payload;
+    RedWorker *worker = opaque;
 
-    receive_data(worker->channel, &surface_id, sizeof(uint32_t));
+    dev_create_primary_surface(worker, msg->surface_id, msg->surface);
+}
 
+static void dev_destroy_primary_surface(RedWorker *worker, uint32_t surface_id)
+{
     PANIC_ON(surface_id != 0);
 
     if (!worker->surfaces[surface_id].context.canvas) {
@@ -10322,7 +10340,7 @@ static inline void handle_dev_destroy_primary_surface(RedWorker *worker)
     }
 
     flush_all_qxl_commands(worker);
-    destroy_surface_wait(worker, 0);
+    dev_destroy_surface_wait(worker, 0);
     red_destroy_surface(worker, 0);
     ASSERT(ring_is_empty(&worker->streams));
 
@@ -10331,6 +10349,25 @@ static inline void handle_dev_destroy_primary_surface(RedWorker *worker)
     red_cursor_reset(worker);
 }
 
+void handle_dev_destroy_primary_surface(void *opaque, void *payload)
+{
+    RedWorkerMessageDestroyPrimarySurface *msg = payload;
+    RedWorker *worker = opaque;
+    uint32_t surface_id = msg->surface_id;
+
+    dev_destroy_primary_surface(worker, surface_id);
+}
+
+void handle_dev_destroy_primary_surface_async(void *opaque, void *payload)
+{
+    RedWorkerMessageDestroyPrimarySurfaceAsync *msg = payload;
+    RedWorker *worker = opaque;
+    uint32_t surface_id = msg->surface_id;
+
+    dev_destroy_primary_surface(worker, surface_id);
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
+}
+
 static void flush_all_surfaces(RedWorker *worker)
 {
     int x;
@@ -10342,7 +10379,7 @@ static void flush_all_surfaces(RedWorker *worker)
     }
 }
 
-static void handle_dev_flush_surfaces(RedWorker *worker)
+static void dev_flush_surfaces(RedWorker *worker)
 {
     flush_all_qxl_commands(worker);
     flush_all_surfaces(worker);
@@ -10350,8 +10387,27 @@ static void handle_dev_flush_surfaces(RedWorker *worker)
     red_wait_outgoing_items(&worker->cursor_channel->common.base);
 }
 
-static void handle_dev_stop(RedWorker *worker)
+void handle_dev_flush_surfaces(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+
+    dev_flush_surfaces(worker);
+}
+
+void handle_dev_flush_surfaces_async(void *opaque, void *payload)
 {
+    RedWorkerMessageFlushSurfacesAsync *msg = payload;
+    RedWorker *worker = opaque;
+
+    dev_flush_surfaces(worker);
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
+}
+
+void handle_dev_stop(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+
+    red_printf("stop");
     ASSERT(worker->running);
     worker->running = FALSE;
     red_display_clear_glz_drawables(worker->display_channel);
@@ -10360,8 +10416,9 @@ static void handle_dev_stop(RedWorker *worker)
     red_wait_outgoing_items(&worker->cursor_channel->common.base);
 }
 
-static void handle_dev_start(RedWorker *worker)
+void handle_dev_start(void *opaque, void *payload)
 {
+    RedWorker *worker = opaque;
     RedChannel *cursor_red_channel = &worker->cursor_channel->common.base;
     RedChannel *display_red_channel = &worker->display_channel->common.base;
 
@@ -10375,308 +10432,479 @@ static void handle_dev_start(RedWorker *worker)
     worker->running = TRUE;
 }
 
-static void handle_dev_input(EventListener *listener, uint32_t events)
+void handle_dev_wakeup(void *opaque, void *payload)
 {
-    RedWorker *worker = SPICE_CONTAINEROF(listener, RedWorker, dev_listener);
-    RedWorkerMessage message;
+    RedWorker *worker = opaque;
+
+    clear_bit(RED_WORKER_PENDING_WAKEUP, worker->pending);
+    stat_inc_counter(worker->wakeup_counter, 1);
+}
+
+void handle_dev_oom(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+
     RedChannel *display_red_channel = &worker->display_channel->common.base;
     int ring_is_empty;
-    int call_async_complete = 0;
-    int write_ready = 0;
-    AsyncCommand *cmd;
-
-    read_message(worker->channel, &message);
-
-    /* for async messages we do the common work in the handler, and
-     * send a ready or call async_complete from here, hence the added switch. */
-    switch (message) {
-    case RED_WORKER_MESSAGE_UPDATE_ASYNC:
-    case RED_WORKER_MESSAGE_ADD_MEMSLOT_ASYNC:
-    case RED_WORKER_MESSAGE_DESTROY_SURFACES_ASYNC:
-    case RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC:
-    case RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC:
-    case RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC:
-    case RED_WORKER_MESSAGE_FLUSH_SURFACES_ASYNC:
-        call_async_complete = 1;
-        receive_data(worker->channel, &cmd, sizeof(cmd));
-        break;
-    case RED_WORKER_MESSAGE_UPDATE:
-    case RED_WORKER_MESSAGE_ADD_MEMSLOT:
-    case RED_WORKER_MESSAGE_DESTROY_SURFACES:
-    case RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE:
-    case RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE:
-    case RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT:
-    case RED_WORKER_MESSAGE_RESET_CURSOR:
-    case RED_WORKER_MESSAGE_RESET_IMAGE_CACHE:
-    case RED_WORKER_MESSAGE_STOP:
-    case RED_WORKER_MESSAGE_LOADVM_COMMANDS:
-    case RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE:
-    case RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE:
-    case RED_WORKER_MESSAGE_DISPLAY_DISCONNECT:
-    case RED_WORKER_MESSAGE_CURSOR_DISCONNECT:
-        write_ready = 1;
-    default:
-        break;
-    }
 
-    switch (message) {
-    case RED_WORKER_MESSAGE_UPDATE_ASYNC:
-        handle_dev_update_async(worker);
-        break;
-    case RED_WORKER_MESSAGE_UPDATE:
-        handle_dev_update(worker);
-        break;
-    case RED_WORKER_MESSAGE_WAKEUP:
-        clear_bit(RED_WORKER_PENDING_WAKEUP, worker->pending);
-        stat_inc_counter(worker->wakeup_counter, 1);
-        break;
-    case RED_WORKER_MESSAGE_OOM:
-        ASSERT(worker->running);
-        // streams? but without streams also leak
-        red_printf_debug(1, "WORKER",
-                         "OOM1 #draw=%u, #red_draw=%u, #glz_draw=%u current %u pipes %u",
-                         worker->drawable_count,
-                         worker->red_drawable_count,
-                         worker->glz_drawable_count,
-                         worker->current_size,
-                         worker->display_channel ?
-                         red_channel_sum_pipes_size(display_red_channel) : 0);
-        while (red_process_commands(worker, MAX_PIPE_SIZE, &ring_is_empty)) {
-            red_channel_push(&worker->display_channel->common.base);
-        }
-        if (worker->qxl->st->qif->flush_resources(worker->qxl) == 0) {
-            red_free_some(worker);
-            worker->qxl->st->qif->flush_resources(worker->qxl);
-        }
-        red_printf_debug(1, "WORKER",
-                         "OOM2 #draw=%u, #red_draw=%u, #glz_draw=%u current %u pipes %u",
-                         worker->drawable_count,
-                         worker->red_drawable_count,
-                         worker->glz_drawable_count,
-                         worker->current_size,
-                         worker->display_channel ?
-                         red_channel_sum_pipes_size(display_red_channel) : 0);
-        clear_bit(RED_WORKER_PENDING_OOM, worker->pending);
-        break;
-    case RED_WORKER_MESSAGE_RESET_CURSOR:
-        red_cursor_reset(worker);
-        break;
-    case RED_WORKER_MESSAGE_RESET_IMAGE_CACHE:
-        image_cache_reset(&worker->image_cache);
-        break;
-    case RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC:
-    case RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT:
-        handle_dev_destroy_surface_wait(worker);
-        break;
-    case RED_WORKER_MESSAGE_DESTROY_SURFACES_ASYNC:
-    case RED_WORKER_MESSAGE_DESTROY_SURFACES:
-        handle_dev_destroy_surfaces(worker);
-        break;
-    case RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC:
-    case RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE:
-        handle_dev_create_primary_surface(worker);
-        break;
-    case RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC:
-    case RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE:
-        handle_dev_destroy_primary_surface(worker);
-        break;
-    case RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE: {
-        RedChannel *red_channel;
-        // TODO: handle seemless migration. Temp, setting migrate to FALSE
-        ensure_display_channel_created(worker, FALSE);
-        red_channel = &worker->display_channel->common.base;
-        send_data(worker->channel, &red_channel, sizeof(RedChannel *));
-        break;
+    ASSERT(worker->running);
+    // streams? but without streams also leak
+    red_printf_debug(1, "WORKER",
+                     "OOM1 #draw=%u, #red_draw=%u, #glz_draw=%u current %u pipes %u",
+                     worker->drawable_count,
+                     worker->red_drawable_count,
+                     worker->glz_drawable_count,
+                     worker->current_size,
+                     worker->display_channel ?
+                     red_channel_sum_pipes_size(display_red_channel) : 0);
+    while (red_process_commands(worker, MAX_PIPE_SIZE, &ring_is_empty)) {
+        red_channel_push(&worker->display_channel->common.base);
     }
-    case RED_WORKER_MESSAGE_DISPLAY_CONNECT: {
-        RedsStream *stream;
-        RedClient *client;
-        int migrate;
-        red_printf("connect");
-
-        receive_data(worker->channel, &client, sizeof(RedClient *));
-        receive_data(worker->channel, &stream, sizeof(RedsStream *));
-        receive_data(worker->channel, &migrate, sizeof(int));
-        handle_new_display_channel(worker, client, stream, migrate);
-        break;
+    if (worker->qxl->st->qif->flush_resources(worker->qxl) == 0) {
+        red_free_some(worker);
+        worker->qxl->st->qif->flush_resources(worker->qxl);
     }
-    case RED_WORKER_MESSAGE_DISPLAY_DISCONNECT: {
-        RedChannelClient *rcc;
+    red_printf_debug(1, "WORKER",
+                     "OOM2 #draw=%u, #red_draw=%u, #glz_draw=%u current %u pipes %u",
+                     worker->drawable_count,
+                     worker->red_drawable_count,
+                     worker->glz_drawable_count,
+                     worker->current_size,
+                     worker->display_channel ?
+                     red_channel_sum_pipes_size(display_red_channel) : 0);
+    clear_bit(RED_WORKER_PENDING_OOM, worker->pending);
+}
 
-        red_printf("disconnect display client");
-        receive_data(worker->channel, &rcc, sizeof(RedChannelClient *));
-        ASSERT(rcc);
-        display_channel_client_disconnect(rcc);
-        break;
-    }
-    case RED_WORKER_MESSAGE_STOP: {
-        red_printf("stop");
-        handle_dev_stop(worker);
+void handle_dev_reset_cursor(void *opaque, void *payload)
+{
+    red_cursor_reset((RedWorker *)opaque);
+}
+
+void handle_dev_reset_image_cache(void *opaque, void *payload)
+{
+    image_cache_reset(&((RedWorker *)opaque)->image_cache);
+}
+
+void handle_dev_destroy_surface_wait_async(void *opaque, void *payload)
+{
+    RedWorkerMessageDestroySurfaceWaitAsync *msg = payload;
+    RedWorker *worker = opaque;
+
+    dev_destroy_surface_wait(worker, msg->surface_id);
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
+}
+
+void handle_dev_destroy_surfaces_async(void *opaque, void *payload)
+{
+    RedWorkerMessageDestroySurfacesAsync *msg = payload;
+    RedWorker *worker = opaque;
+
+    dev_destroy_surfaces(worker);
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
+}
+
+void handle_dev_create_primary_surface_async(void *opaque, void *payload)
+{
+    RedWorkerMessageCreatePrimarySurfaceAsync *msg = payload;
+    RedWorker *worker = opaque;
+
+    dev_create_primary_surface(worker, msg->surface_id, msg->surface);
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
+}
+
+/* exception for Dispatcher, data going from red_worker to main thread,
+ * TODO: use a different dispatcher?
+ * TODO: leave direct usage of channel(fd)? It's only used right after the
+ * pthread is created, since the channel duration is the lifetime of the spice
+ * server. */
+
+void handle_dev_display_channel_create(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+
+    RedChannel *red_channel;
+    // TODO: handle seemless migration. Temp, setting migrate to FALSE
+    ensure_display_channel_created(worker, FALSE);
+    red_channel = &worker->display_channel->common.base;
+    send_data(worker->channel, &red_channel, sizeof(RedChannel *));
+}
+
+void handle_dev_display_connect(void *opaque, void *payload)
+{
+    RedWorkerMessageDisplayConnect *msg = payload;
+    RedWorker *worker = opaque;
+    RedsStream *stream = msg->stream;
+    RedClient *client = msg->client;
+    int migration = msg->migration;
+
+    red_printf("connect");
+    handle_new_display_channel(worker, client, stream, migration);
+}
+
+void handle_dev_display_disconnect(void *opaque, void *payload)
+{
+    RedWorkerMessageDisplayDisconnect *msg = payload;
+    RedChannelClient *rcc = msg->rcc;
+
+    red_printf("disconnect display client");
+    ASSERT(rcc);
+    display_channel_client_disconnect(rcc);
+}
+
+void handle_dev_display_migrate(void *opaque, void *payload)
+{
+    RedWorkerMessageDisplayMigrate *msg = payload;
+    RedWorker *worker = opaque;
+
+    RedChannelClient *rcc = msg->rcc;
+    red_printf("migrate display client");
+    ASSERT(rcc);
+    red_migrate_display(worker, rcc);
+}
+
+/* TODO: special, perhaps use another dispatcher? */
+void handle_dev_cursor_channel_create(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+    RedChannel *red_channel;
+
+    // TODO: handle seemless migration. Temp, setting migrate to FALSE
+    ensure_cursor_channel_created(worker, FALSE);
+    red_channel = &worker->cursor_channel->common.base;
+    send_data(worker->channel, &red_channel, sizeof(RedChannel *));
+}
+
+void handle_dev_cursor_connect(void *opaque, void *payload)
+{
+    RedWorkerMessageCursorConnect *msg = payload;
+    RedWorker *worker = opaque;
+    RedsStream *stream = msg->stream;
+    RedClient *client = msg->client;
+    int migration = msg->migration;
+
+    red_printf("cursor connect");
+    red_connect_cursor(worker, client, stream, migration);
+}
+
+void handle_dev_cursor_disconnect(void *opaque, void *payload)
+{
+    RedWorkerMessageCursorDisconnect *msg = payload;
+    RedChannelClient *rcc = msg->rcc;
+
+    red_printf("disconnect cursor client");
+    ASSERT(rcc);
+    cursor_channel_client_disconnect(rcc);
+}
+
+void handle_dev_cursor_migrate(void *opaque, void *payload)
+{
+    RedWorkerMessageCursorMigrate *msg = payload;
+    RedWorker *worker = opaque;
+    RedChannelClient *rcc = msg->rcc;
+
+    red_printf("migrate cursor client");
+    ASSERT(rcc);
+    red_migrate_cursor(worker, rcc);
+}
+
+void handle_dev_set_compression(void *opaque, void *payload)
+{
+    RedWorkerMessageSetCompression *msg = payload;
+    RedWorker *worker = opaque;
+
+    worker->image_compression = msg->image_compression;
+    switch (worker->image_compression) {
+    case SPICE_IMAGE_COMPRESS_AUTO_LZ:
+        red_printf("ic auto_lz");
         break;
-    }
-    case RED_WORKER_MESSAGE_START:
-        red_printf("start");
-        handle_dev_start(worker);
+    case SPICE_IMAGE_COMPRESS_AUTO_GLZ:
+        red_printf("ic auto_glz");
         break;
-    case RED_WORKER_MESSAGE_DISPLAY_MIGRATE: {
-        RedChannelClient *rcc;
-        red_printf("migrate display client");
-        receive_data(worker->channel, &rcc, sizeof(RedChannelClient *));
-        ASSERT(rcc);
-        red_migrate_display(worker, rcc);
+    case SPICE_IMAGE_COMPRESS_QUIC:
+        red_printf("ic quic");
         break;
-    }
-    case RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE: {
-        RedChannel *red_channel;
-        // TODO: handle seemless migration. Temp, setting migrate to FALSE
-        ensure_cursor_channel_created(worker, FALSE);
-        red_channel = &worker->cursor_channel->common.base;
-        send_data(worker->channel, &red_channel, sizeof(RedChannel *));
+    case SPICE_IMAGE_COMPRESS_LZ:
+        red_printf("ic lz");
         break;
-    }
-    case RED_WORKER_MESSAGE_CURSOR_CONNECT: {
-        RedsStream *stream;
-        RedClient *client;
-        int migrate;
-
-        red_printf("cursor connect");
-        receive_data(worker->channel, &client, sizeof(RedClient *));
-        receive_data(worker->channel, &stream, sizeof(RedsStream *));
-        receive_data(worker->channel, &migrate, sizeof(int));
-        red_connect_cursor(worker, client, stream, migrate);
+    case SPICE_IMAGE_COMPRESS_GLZ:
+        red_printf("ic glz");
         break;
-    }
-    case RED_WORKER_MESSAGE_CURSOR_DISCONNECT: {
-        RedChannelClient *rcc;
-
-        red_printf("disconnect cursor client");
-        receive_data(worker->channel, &rcc, sizeof(RedChannelClient *));
-        ASSERT(rcc);
-        cursor_channel_client_disconnect(rcc);
+    case SPICE_IMAGE_COMPRESS_OFF:
+        red_printf("ic off");
         break;
+    default:
+        red_printf("ic invalid");
     }
-    case RED_WORKER_MESSAGE_CURSOR_MIGRATE: {
-        RedChannelClient *rcc;
-        red_printf("migrate cursor client");
-        receive_data(worker->channel, &rcc, sizeof(RedChannelClient *));
-        ASSERT(rcc);
-        red_migrate_cursor(worker, rcc);
-        break;
+#ifdef COMPRESS_STAT
+    print_compress_stats(worker->display_channel);
+    if (worker->display_channel) {
+        stat_reset(&worker->display_channel->quic_stat);
+        stat_reset(&worker->display_channel->lz_stat);
+        stat_reset(&worker->display_channel->glz_stat);
+        stat_reset(&worker->display_channel->jpeg_stat);
+        stat_reset(&worker->display_channel->zlib_glz_stat);
+        stat_reset(&worker->display_channel->jpeg_alpha_stat);
     }
-    case RED_WORKER_MESSAGE_SET_COMPRESSION:
-        receive_data(worker->channel, &worker->image_compression,
-                     sizeof(spice_image_compression_t));
-        switch (worker->image_compression) {
-        case SPICE_IMAGE_COMPRESS_AUTO_LZ:
-            red_printf("ic auto_lz");
-            break;
-        case SPICE_IMAGE_COMPRESS_AUTO_GLZ:
-            red_printf("ic auto_glz");
+#endif
+}
+
+void handle_dev_set_streaming_video(void *opaque, void *payload)
+{
+    RedWorkerMessageSetStreamingVideo *msg = payload;
+    RedWorker *worker = opaque;
+
+    worker->streaming_video = msg->streaming_video;
+    ASSERT(worker->streaming_video != STREAM_VIDEO_INVALID);
+    switch(worker->streaming_video) {
+        case STREAM_VIDEO_ALL:
+            red_printf("sv all");
             break;
-        case SPICE_IMAGE_COMPRESS_QUIC:
-            red_printf("ic quic");
+        case STREAM_VIDEO_FILTER:
+            red_printf("sv filter");
             break;
-        case SPICE_IMAGE_COMPRESS_LZ:
-            red_printf("ic lz");
+        case STREAM_VIDEO_OFF:
+            red_printf("sv off");
             break;
-        case SPICE_IMAGE_COMPRESS_GLZ:
-            red_printf("ic glz");
+        default:
+            red_printf("sv invalid");
+    }
+}
+
+void handle_dev_set_mouse_mode(void *opaque, void *payload)
+{
+    RedWorkerMessageSetMouseMode *msg = payload;
+    RedWorker *worker = opaque;
+
+    worker->mouse_mode = msg->mode;
+    red_printf("mouse mode %u", worker->mouse_mode);
+}
+
+void handle_dev_add_memslot_async(void *opaque, void *payload)
+{
+    RedWorkerMessageAddMemslotAsync *msg = payload;
+    RedWorker *worker = opaque;
+
+    dev_add_memslot(worker, msg->mem_slot);
+    red_dispatcher_async_complete(worker->red_dispatcher, msg->cmd);
+}
+
+void handle_dev_reset_memslots(void *opaque, void *payload)
+{
+    RedWorker *worker = opaque;
+
+    red_memslot_info_reset(&worker->mem_slots);
+}
+
+void handle_dev_loadvm_commands(void *opaque, void *payload)
+{
+    RedWorkerMessageLoadvmCommands *msg = payload;
+    RedWorker *worker = opaque;
+    uint32_t i;
+    RedCursorCmd *cursor_cmd;
+    RedSurfaceCmd *surface_cmd;
+    uint32_t count = msg->count;
+    QXLCommandExt *ext = msg->ext;
+
+    red_printf("loadvm_commands");
+    for (i = 0 ; i < count ; ++i) {
+        switch (ext[i].cmd.type) {
+        case QXL_CMD_CURSOR:
+            cursor_cmd = spice_new0(RedCursorCmd, 1);
+            red_get_cursor_cmd(&worker->mem_slots, ext[i].group_id,
+                               cursor_cmd, ext[i].cmd.data);
+            qxl_process_cursor(worker, cursor_cmd, ext[i].group_id);
             break;
-        case SPICE_IMAGE_COMPRESS_OFF:
-            red_printf("ic off");
+        case QXL_CMD_SURFACE:
+            surface_cmd = spice_new0(RedSurfaceCmd, 1);
+            red_get_surface_cmd(&worker->mem_slots, ext[i].group_id,
+                                surface_cmd, ext[i].cmd.data);
+            red_process_surface(worker, surface_cmd, ext[i].group_id, TRUE);
             break;
         default:
-            red_printf("ic invalid");
-        }
-#ifdef COMPRESS_STAT
-        print_compress_stats(worker->display_channel);
-        if (worker->display_channel) {
-            stat_reset(&worker->display_channel->quic_stat);
-            stat_reset(&worker->display_channel->lz_stat);
-            stat_reset(&worker->display_channel->glz_stat);
-            stat_reset(&worker->display_channel->jpeg_stat);
-            stat_reset(&worker->display_channel->zlib_glz_stat);
-            stat_reset(&worker->display_channel->jpeg_alpha_stat);
-        }
-#endif
-        break;
-    case RED_WORKER_MESSAGE_SET_STREAMING_VIDEO:
-        receive_data(worker->channel, &worker->streaming_video, sizeof(uint32_t));
-        ASSERT(worker->streaming_video != STREAM_VIDEO_INVALID);
-        switch(worker->streaming_video) {
-            case STREAM_VIDEO_ALL:
-                red_printf("sv all");
-                break;
-            case STREAM_VIDEO_FILTER:
-                red_printf("sv filter");
-                break;
-            case STREAM_VIDEO_OFF:
-                red_printf("sv off");
-                break;
-            default:
-                red_printf("sv invalid");
-        }
-        break;
-    case RED_WORKER_MESSAGE_SET_MOUSE_MODE:
-        receive_data(worker->channel, &worker->mouse_mode, sizeof(uint32_t));
-        red_printf("mouse mode %u", worker->mouse_mode);
-        break;
-    case RED_WORKER_MESSAGE_ADD_MEMSLOT_ASYNC:
-    case RED_WORKER_MESSAGE_ADD_MEMSLOT:
-        handle_dev_add_memslot(worker);
-        break;
-    case RED_WORKER_MESSAGE_DEL_MEMSLOT:
-        handle_dev_del_memslot(worker);
-        break;
-    case RED_WORKER_MESSAGE_RESET_MEMSLOTS:
-        red_memslot_info_reset(&worker->mem_slots);
-        break;
-    case RED_WORKER_MESSAGE_LOADVM_COMMANDS: {
-        uint32_t count;
-        QXLCommandExt ext;
-        RedCursorCmd *cursor_cmd;
-        RedSurfaceCmd *surface_cmd;
-
-        red_printf("loadvm_commands");
-        receive_data(worker->channel, &count, sizeof(uint32_t));
-        while (count > 0) {
-            receive_data(worker->channel, &ext, sizeof(QXLCommandExt));
-            switch (ext.cmd.type) {
-            case QXL_CMD_CURSOR:
-                cursor_cmd = spice_new0(RedCursorCmd, 1);
-                red_get_cursor_cmd(&worker->mem_slots, ext.group_id,
-                                   cursor_cmd, ext.cmd.data);
-                qxl_process_cursor(worker, cursor_cmd, ext.group_id);
-                break;
-            case QXL_CMD_SURFACE:
-                surface_cmd = spice_new0(RedSurfaceCmd, 1);
-                red_get_surface_cmd(&worker->mem_slots, ext.group_id,
-                                    surface_cmd, ext.cmd.data);
-                red_process_surface(worker, surface_cmd, ext.group_id, TRUE);
-                break;
-            default:
-                red_printf("unhandled loadvm command type (%d)", ext.cmd.type);
-                break;
-            }
-            count--;
+            red_printf("unhandled loadvm command type (%d)", ext[i].cmd.type);
+            break;
         }
-        break;
-    }
-    case RED_WORKER_MESSAGE_FLUSH_SURFACES_ASYNC:
-        handle_dev_flush_surfaces(worker);
-        break;
-    default:
-        red_error("message error");
-    }
-    if (call_async_complete) {
-        red_dispatcher_async_complete(worker->dispatcher, cmd);
-    }
-    if (write_ready) {
-        message = RED_WORKER_MESSAGE_READY;
-        write_message(worker->channel, &message);
     }
 }
 
+static void register_callbacks(Dispatcher *dispatcher)
+{
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DISPLAY_CONNECT,
+                                handle_dev_display_connect,
+                                sizeof(RedWorkerMessageDisplayConnect),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DISPLAY_DISCONNECT,
+                                handle_dev_display_disconnect,
+                                sizeof(RedWorkerMessageDisplayDisconnect),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DISPLAY_MIGRATE,
+                                handle_dev_display_migrate,
+                                sizeof(RedWorkerMessageDisplayMigrate),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_CURSOR_CONNECT,
+                                handle_dev_cursor_connect,
+                                sizeof(RedWorkerMessageCursorConnect),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_CURSOR_DISCONNECT,
+                                handle_dev_cursor_disconnect,
+                                sizeof(RedWorkerMessageCursorDisconnect),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_CURSOR_MIGRATE,
+                                handle_dev_cursor_migrate,
+                                sizeof(RedWorkerMessageCursorMigrate),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_UPDATE,
+                                handle_dev_update,
+                                sizeof(RedWorkerMessageUpdate),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_UPDATE_ASYNC,
+                                handle_dev_update_async,
+                                sizeof(RedWorkerMessageUpdateAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_ADD_MEMSLOT,
+                                handle_dev_add_memslot,
+                                sizeof(RedWorkerMessageAddMemslot),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_ADD_MEMSLOT_ASYNC,
+                                handle_dev_add_memslot_async,
+                                sizeof(RedWorkerMessageAddMemslotAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DEL_MEMSLOT,
+                                handle_dev_del_memslot,
+                                sizeof(RedWorkerMessageDelMemslot),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DESTROY_SURFACES,
+                                handle_dev_destroy_surfaces,
+                                sizeof(RedWorkerMessageDestroySurfaces),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DESTROY_SURFACES_ASYNC,
+                                handle_dev_destroy_surfaces_async,
+                                sizeof(RedWorkerMessageDestroySurfacesAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE,
+                                handle_dev_destroy_primary_surface,
+                                sizeof(RedWorkerMessageDestroyPrimarySurface),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC,
+                                handle_dev_destroy_primary_surface_async,
+                                sizeof(RedWorkerMessageDestroyPrimarySurfaceAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC,
+                                handle_dev_create_primary_surface_async,
+                                sizeof(RedWorkerMessageCreatePrimarySurfaceAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE,
+                                handle_dev_create_primary_surface,
+                                sizeof(RedWorkerMessageCreatePrimarySurface),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_RESET_IMAGE_CACHE,
+                                handle_dev_reset_image_cache,
+                                sizeof(RedWorkerMessageResetImageCache),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_RESET_CURSOR,
+                                handle_dev_reset_cursor,
+                                sizeof(RedWorkerMessageResetCursor),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_WAKEUP,
+                                handle_dev_wakeup,
+                                sizeof(RedWorkerMessageWakeup),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_OOM,
+                                handle_dev_oom,
+                                sizeof(RedWorkerMessageOom),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_START,
+                                handle_dev_start,
+                                sizeof(RedWorkerMessageStart),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_FLUSH_SURFACES_ASYNC,
+                                handle_dev_flush_surfaces_async,
+                                sizeof(RedWorkerMessageFlushSurfacesAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_STOP,
+                                handle_dev_stop,
+                                sizeof(RedWorkerMessageStop),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_LOADVM_COMMANDS,
+                                handle_dev_loadvm_commands,
+                                sizeof(RedWorkerMessageLoadvmCommands),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_SET_COMPRESSION,
+                                handle_dev_set_compression,
+                                sizeof(RedWorkerMessageSetCompression),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_SET_STREAMING_VIDEO,
+                                handle_dev_set_streaming_video,
+                                sizeof(RedWorkerMessageSetStreamingVideo),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_SET_MOUSE_MODE,
+                                handle_dev_set_mouse_mode,
+                                sizeof(RedWorkerMessageSetMouseMode),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE,
+                                handle_dev_display_channel_create,
+                                sizeof(RedWorkerMessageDisplayChannelCreate),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE,
+                                handle_dev_cursor_channel_create,
+                                sizeof(RedWorkerMessageCursorChannelCreate),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT,
+                                handle_dev_destroy_surface_wait,
+                                sizeof(RedWorkerMessageDestroySurfaceWait),
+                                1);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC,
+                                handle_dev_destroy_surface_wait_async,
+                                sizeof(RedWorkerMessageDestroySurfaceWaitAsync),
+                                0);
+    dispatcher_register_handler(dispatcher,
+                                RED_WORKER_MESSAGE_RESET_MEMSLOTS,
+                                handle_dev_reset_memslots,
+                                sizeof(RedWorkerMessageResetMemslots),
+                                0);
+}
+
+
+
+static void handle_dev_input(EventListener *listener, uint32_t events)
+{
+    RedWorker *worker = SPICE_CONTAINEROF(listener, RedWorker, dev_listener);
+
+    dispatcher_handle_recv_read(red_dispatcher_get_dispatcher(worker->red_dispatcher));
+}
+
 static void handle_dev_free(EventListener *ctx)
 {
     free(ctx);
@@ -10687,14 +10915,18 @@ static void red_init(RedWorker *worker, WorkerInitData *init_data)
     struct epoll_event event;
     RedWorkerMessage message;
     int epoll;
+    Dispatcher *dispatcher;
 
     ASSERT(sizeof(CursorItem) <= QXL_CURSUR_DEVICE_DATA_SIZE);
 
     memset(worker, 0, sizeof(RedWorker));
-    worker->dispatcher = init_data->dispatcher;
+    dispatcher = red_dispatcher_get_dispatcher(init_data->red_dispatcher);
+    dispatcher_set_opaque(dispatcher, worker);
+    worker->red_dispatcher = init_data->red_dispatcher;
     worker->qxl = init_data->qxl;
     worker->id = init_data->id;
-    worker->channel = init_data->channel;
+    worker->channel = dispatcher_get_recv_fd(dispatcher);
+    register_callbacks(dispatcher);
     worker->pending = init_data->pending;
     worker->dev_listener.refs = 1;
     worker->dev_listener.action = handle_dev_input;
diff --git a/server/red_worker.h b/server/red_worker.h
index 26c43ad..08c7b22 100644
--- a/server/red_worker.h
+++ b/server/red_worker.h
@@ -84,6 +84,8 @@ enum {
 
     RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE,
     RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE,
+
+    RED_WORKER_MESSAGE_COUNT // LAST
 };
 
 typedef uint32_t RedWorkerMessage;
@@ -102,7 +104,6 @@ typedef struct RedDispatcher RedDispatcher;
 typedef struct WorkerInitData {
     struct QXLInstance *qxl;
     int id;
-    int channel;
     uint32_t *pending;
     uint32_t num_renderers;
     uint32_t renderers[RED_MAX_RENDERERS];
@@ -116,7 +117,7 @@ typedef struct WorkerInitData {
     uint8_t memslot_id_bits;
     uint8_t internal_groupslot_id;
     uint32_t n_surfaces;
-    RedDispatcher *dispatcher;
+    RedDispatcher *red_dispatcher;
 } WorkerInitData;
 
 void *red_worker_main(void *arg);
-- 
1.7.7.1



More information about the Spice-devel mailing list