/* Copyright (C) 2009 Red Hat, Inc. This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, see . */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include "spice.h" #include "red_worker.h" #include "quic.h" #include "reds_sw_canvas.h" #ifdef USE_OPENGL #include "reds_gl_canvas.h" #endif // USE_OPENGL #include "reds.h" #include "red_dispatcher.h" #include "red_parse_qxl.h" static int num_active_workers = 0; //volatile struct RedDispatcher { QXLWorker base; QXLInstance *qxl; int channel; pthread_t worker_thread; uint32_t pending; int primary_active; int x_res; int y_res; int use_hardware_cursor; RedDispatcher *next; RedWorkerMessage async_message; pthread_mutex_t async_lock; QXLDevSurfaceCreate surface_create; }; typedef struct RedWorkeState { uint8_t *io_base; unsigned long phys_delta; uint32_t x_res; uint32_t y_res; uint32_t bits; uint32_t stride; } RedWorkeState; extern uint32_t streaming_video; extern spice_image_compression_t image_compression; extern spice_wan_compression_t jpeg_state; extern spice_wan_compression_t zlib_glz_state; static RedDispatcher *dispatchers = NULL; static void red_dispatcher_set_display_peer(RedChannel *channel, RedClient *client, RedsStream *stream, int migration, int num_common_caps, uint32_t *common_caps, int num_caps, uint32_t *caps) { 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)); } static void red_dispatcher_disconnect_display_peer(RedChannelClient *rcc) { RedDispatcher *dispatcher; if (!rcc->channel) { return; } 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 *)); // 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); } static void red_dispatcher_display_migrate(RedChannelClient *rcc) { 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 *)); } static void red_dispatcher_set_cursor_peer(RedChannel *channel, RedClient *client, RedsStream *stream, int migration, int num_common_caps, uint32_t *common_caps, int num_caps, uint32_t *caps) { 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)); } static void red_dispatcher_disconnect_cursor_peer(RedChannelClient *rcc) { RedDispatcher *dispatcher; if (!rcc->channel) { return; } 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 *)); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); } static void red_dispatcher_cursor_migrate(RedChannelClient *rcc) { 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_CURSOR_MIGRATE; write_message(dispatcher->channel, &message); } typedef struct RendererInfo { int id; const char *name; } RendererInfo; static RendererInfo renderers_info[] = { {RED_RENDERER_SW, "sw"}, #ifdef USE_OPENGL {RED_RENDERER_OGL_PBUF, "oglpbuf"}, {RED_RENDERER_OGL_PIXMAP, "oglpixmap"}, #endif {RED_RENDERER_INVALID, NULL}, }; static uint32_t renderers[RED_MAX_RENDERERS]; static uint32_t num_renderers = 0; static RendererInfo *find_renderer(const char *name) { RendererInfo *inf = renderers_info; while (inf->name) { if (strcmp(name, inf->name) == 0) { return inf; } inf++; } return NULL; } int red_dispatcher_add_renderer(const char *name) { RendererInfo *inf; if (num_renderers == RED_MAX_RENDERERS || !(inf = find_renderer(name))) { return FALSE; } renderers[num_renderers++] = inf->id; return TRUE; } int red_dispatcher_qxl_count(void) { return num_active_workers; } static void update_client_mouse_allowed(void) { static int allowed = FALSE; int allow_now = FALSE; int x_res = 0; int y_res = 0; if (num_active_workers > 0) { allow_now = TRUE; RedDispatcher *now = dispatchers; while (now && allow_now) { if (now->primary_active) { allow_now = now->use_hardware_cursor; if (num_active_workers == 1) { if (allow_now) { x_res = now->x_res; y_res = now->y_res; } break; } } now = now->next; } } if (allow_now || allow_now != allowed) { allowed = allow_now; reds_set_client_mouse_allowed(allowed, x_res, y_res); } } static void red_dispatcher_update_area(RedDispatcher *dispatcher, uint32_t surface_id, QXLRect *qxl_area, QXLRect *qxl_dirty_rects, uint32_t num_dirty_rects, uint32_t clear_dirty_region) { RedWorkerMessage message = RED_WORKER_MESSAGE_UPDATE; 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); } static RedWorkerMessage red_dispatcher_async_start(RedDispatcher *dispatcher, RedWorkerMessage message) { pthread_mutex_lock(&dispatcher->async_lock); if (dispatcher->async_message != RED_WORKER_MESSAGE_NOP) { red_printf("error: async clash. second async ignored"); pthread_mutex_unlock(&dispatcher->async_lock); return RED_WORKER_MESSAGE_NOP; } dispatcher->async_message = message; pthread_mutex_unlock(&dispatcher->async_lock); return message; } static void red_dispatcher_update_area_async(RedDispatcher *dispatcher, uint32_t surface_id, QXLRect *qxl_area, uint32_t clear_dirty_region, uint64_t cookie) { RedWorkerMessage message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_UPDATE_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } write_message(dispatcher->channel, &message); send_data(dispatcher->channel, &cookie, sizeof(cookie)); 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)); } static void qxl_worker_update_area(QXLWorker *qxl_worker, uint32_t surface_id, QXLRect *qxl_area, QXLRect *qxl_dirty_rects, uint32_t num_dirty_rects, uint32_t clear_dirty_region) { red_dispatcher_update_area((RedDispatcher*)qxl_worker, surface_id, qxl_area, qxl_dirty_rects, num_dirty_rects, clear_dirty_region); } static void red_dispatcher_add_memslot(RedDispatcher *dispatcher, QXLDevMemSlot *mem_slot) { RedWorkerMessage message = RED_WORKER_MESSAGE_ADD_MEMSLOT; write_message(dispatcher->channel, &message); send_data(dispatcher->channel, mem_slot, sizeof(QXLDevMemSlot)); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); } static void qxl_worker_add_memslot(QXLWorker *qxl_worker, QXLDevMemSlot *mem_slot) { red_dispatcher_add_memslot((RedDispatcher*)qxl_worker, mem_slot); } static void red_dispatcher_add_memslot_async(RedDispatcher *dispatcher, QXLDevMemSlot *mem_slot, uint64_t cookie) { RedWorkerMessage message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_ADD_MEMSLOT_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } write_message(dispatcher->channel, &message); send_data(dispatcher->channel, &cookie, sizeof(cookie)); send_data(dispatcher->channel, mem_slot, sizeof(QXLDevMemSlot)); } static void red_dispatcher_del_memslot(RedDispatcher *dispatcher, uint32_t slot_group_id, uint32_t slot_id) { 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)); } static void qxl_worker_del_memslot(QXLWorker *qxl_worker, uint32_t slot_group_id, uint32_t slot_id) { red_dispatcher_del_memslot((RedDispatcher*)qxl_worker, slot_group_id, slot_id); } static void red_dispatcher_destroy_surfaces(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_DESTROY_SURFACES; write_message(dispatcher->channel, &message); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); } static void qxl_worker_destroy_surfaces(QXLWorker *qxl_worker) { red_dispatcher_destroy_surfaces((RedDispatcher*)qxl_worker); } static void red_dispatcher_destroy_surfaces_async(RedDispatcher *dispatcher, uint64_t cookie) { RedWorkerMessage message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_DESTROY_SURFACES_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } write_message(dispatcher->channel, &message); send_data(dispatcher->channel, &cookie, sizeof(cookie)); } static void red_dispatcher_destroy_primary_surface_complete(RedDispatcher *dispatcher) { dispatcher->x_res = 0; dispatcher->y_res = 0; dispatcher->use_hardware_cursor = FALSE; dispatcher->primary_active = FALSE; update_client_mouse_allowed(); } static void red_dispatcher_destroy_primary_surface(RedDispatcher *dispatcher, uint32_t surface_id, int async, uint64_t cookie) { RedWorkerMessage message; if (async) { message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } } else { message = RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE; } write_message(dispatcher->channel, &message); if (async) { send_data(dispatcher->channel, &cookie, sizeof(cookie)); } 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); } } static void qxl_worker_destroy_primary_surface(QXLWorker *qxl_worker, uint32_t surface_id) { red_dispatcher_destroy_primary_surface((RedDispatcher*)qxl_worker, surface_id, 0, 0); } static void red_dispatcher_create_primary_surface_complete(RedDispatcher *dispatcher) { QXLDevSurfaceCreate *surface = &dispatcher->surface_create; dispatcher->x_res = surface->width; dispatcher->y_res = surface->height; dispatcher->use_hardware_cursor = surface->mouse_mode; dispatcher->primary_active = TRUE; update_client_mouse_allowed(); memset(&dispatcher->surface_create, 0, sizeof(QXLDevSurfaceCreate)); } static void red_dispatcher_create_primary_surface(RedDispatcher *dispatcher, uint32_t surface_id, QXLDevSurfaceCreate *surface, int async, uint64_t cookie) { RedWorkerMessage message; if (async) { message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } } else { message = RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE; } dispatcher->surface_create = *surface; write_message(dispatcher->channel, &message); if (async) { send_data(dispatcher->channel, &cookie, sizeof(cookie)); } 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); } } static void qxl_worker_create_primary_surface(QXLWorker *qxl_worker, uint32_t surface_id, QXLDevSurfaceCreate *surface) { red_dispatcher_create_primary_surface((RedDispatcher*)qxl_worker, surface_id, surface, 0, 0); } static void red_dispatcher_reset_image_cache(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_RESET_IMAGE_CACHE; write_message(dispatcher->channel, &message); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); } static void qxl_worker_reset_image_cache(QXLWorker *qxl_worker) { red_dispatcher_reset_image_cache((RedDispatcher*)qxl_worker); } static void red_dispatcher_reset_cursor(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_RESET_CURSOR; write_message(dispatcher->channel, &message); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); } 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) { RedWorkerMessage message; if (async ) { message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } } else { message = RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT; } write_message(dispatcher->channel, &message); if (async) { send_data(dispatcher->channel, &cookie, sizeof(cookie)); } send_data(dispatcher->channel, &surface_id, sizeof(uint32_t)); if (async) { return; } 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) { red_dispatcher_destroy_surface_wait((RedDispatcher*)qxl_worker, surface_id, 0, 0); } static void red_dispatcher_reset_memslots(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_RESET_MEMSLOTS; write_message(dispatcher->channel, &message); } static void qxl_worker_reset_memslots(QXLWorker *qxl_worker) { red_dispatcher_reset_memslots((RedDispatcher*)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); } } static void qxl_worker_wakeup(QXLWorker *qxl_worker) { red_dispatcher_wakeup((RedDispatcher*)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); } } static void qxl_worker_oom(QXLWorker *qxl_worker) { red_dispatcher_oom((RedDispatcher*)qxl_worker); } static void red_dispatcher_start(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_START; write_message(dispatcher->channel, &message); } static void qxl_worker_start(QXLWorker *qxl_worker) { red_dispatcher_start((RedDispatcher*)qxl_worker); } static void red_dispatcher_flush_surfaces_async(RedDispatcher *dispatcher, uint64_t cookie) { RedWorkerMessage message = red_dispatcher_async_start(dispatcher, RED_WORKER_MESSAGE_FLUSH_SURFACES_ASYNC); if (message == RED_WORKER_MESSAGE_NOP) { return; } write_message(dispatcher->channel, &message); send_data(dispatcher->channel, &cookie, sizeof(cookie)); } static void red_dispatcher_stop(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_STOP; write_message(dispatcher->channel, &message); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); } static void qxl_worker_stop(QXLWorker *qxl_worker) { red_dispatcher_stop((RedDispatcher*)qxl_worker); } static void red_dispatcher_loadvm_commands(RedDispatcher *dispatcher, struct QXLCommandExt *ext, uint32_t count) { RedWorkerMessage message = RED_WORKER_MESSAGE_LOADVM_COMMANDS; 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); } static void qxl_worker_loadvm_commands(QXLWorker *qxl_worker, struct QXLCommandExt *ext, uint32_t count) { red_dispatcher_loadvm_commands((RedDispatcher*)qxl_worker, ext, count); } void red_dispatcher_set_mm_time(uint32_t mm_time) { RedDispatcher *now = dispatchers; while (now) { now->qxl->st->qif->set_mm_time(now->qxl, mm_time); now = now->next; } } static inline int calc_compression_level(void) { ASSERT(streaming_video != STREAM_VIDEO_INVALID); if ((streaming_video != STREAM_VIDEO_OFF) || (image_compression != SPICE_IMAGE_COMPRESS_QUIC)) { return 0; } else { return 1; } } void red_dispatcher_on_ic_change(void) { 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)); now = now->next; } } void red_dispatcher_on_sv_change(void) { 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)); now = now->next; } } void red_dispatcher_set_mouse_mode(uint32_t mode) { 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)); now = now->next; } } int red_dispatcher_count(void) { RedDispatcher *now = dispatchers; int ret = 0; while (now) { ret++; now = now->next; } return ret; } uint32_t red_dispatcher_qxl_ram_size(void) { QXLDevInitInfo qxl_info; if (!dispatchers) { return 0; } dispatchers->qxl->st->qif->get_init_info(dispatchers->qxl, &qxl_info); return qxl_info.qxl_ram_size; } SPICE_GNUC_VISIBLE void spice_qxl_wakeup(QXLInstance *instance) { red_dispatcher_wakeup(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_oom(QXLInstance *instance) { red_dispatcher_oom(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_start(QXLInstance *instance) { red_dispatcher_start(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_stop(QXLInstance *instance) { red_dispatcher_stop(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_update_area(QXLInstance *instance, uint32_t surface_id, struct QXLRect *area, struct QXLRect *dirty_rects, uint32_t num_dirty_rects, uint32_t clear_dirty_region) { red_dispatcher_update_area(instance->st->dispatcher, surface_id, area, dirty_rects, num_dirty_rects, clear_dirty_region); } SPICE_GNUC_VISIBLE void spice_qxl_add_memslot(QXLInstance *instance, QXLDevMemSlot *slot) { red_dispatcher_add_memslot(instance->st->dispatcher, slot); } SPICE_GNUC_VISIBLE void spice_qxl_del_memslot(QXLInstance *instance, uint32_t slot_group_id, uint32_t slot_id) { red_dispatcher_del_memslot(instance->st->dispatcher, slot_group_id, slot_id); } SPICE_GNUC_VISIBLE void spice_qxl_reset_memslots(QXLInstance *instance) { red_dispatcher_reset_memslots(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_destroy_surfaces(QXLInstance *instance) { red_dispatcher_destroy_surfaces(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_destroy_primary_surface(QXLInstance *instance, uint32_t surface_id) { red_dispatcher_destroy_primary_surface(instance->st->dispatcher, surface_id, 0, 0); } SPICE_GNUC_VISIBLE void spice_qxl_create_primary_surface(QXLInstance *instance, uint32_t surface_id, QXLDevSurfaceCreate *surface) { red_dispatcher_create_primary_surface(instance->st->dispatcher, surface_id, surface, 0, 0); } SPICE_GNUC_VISIBLE void spice_qxl_reset_image_cache(QXLInstance *instance) { red_dispatcher_reset_image_cache(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_reset_cursor(QXLInstance *instance) { red_dispatcher_reset_cursor(instance->st->dispatcher); } SPICE_GNUC_VISIBLE void spice_qxl_destroy_surface_wait(QXLInstance *instance, uint32_t surface_id) { red_dispatcher_destroy_surface_wait(instance->st->dispatcher, surface_id, 0, 0); } SPICE_GNUC_VISIBLE void spice_qxl_loadvm_commands(QXLInstance *instance, struct QXLCommandExt *ext, uint32_t count) { red_dispatcher_loadvm_commands(instance->st->dispatcher, ext, count); } SPICE_GNUC_VISIBLE void spice_qxl_update_area_async(QXLInstance *instance, uint32_t surface_id, QXLRect *qxl_area, uint32_t clear_dirty_region, uint64_t cookie) { red_dispatcher_update_area_async(instance->st->dispatcher, surface_id, qxl_area, clear_dirty_region, cookie); } SPICE_GNUC_VISIBLE void spice_qxl_add_memslot_async(QXLInstance *instance, QXLDevMemSlot *slot, uint64_t cookie) { red_dispatcher_add_memslot_async(instance->st->dispatcher, slot, cookie); } SPICE_GNUC_VISIBLE void spice_qxl_destroy_surfaces_async(QXLInstance *instance, uint64_t cookie) { red_dispatcher_destroy_surfaces_async(instance->st->dispatcher, cookie); } SPICE_GNUC_VISIBLE void spice_qxl_destroy_primary_surface_async(QXLInstance *instance, uint32_t surface_id, uint64_t cookie) { red_dispatcher_destroy_primary_surface(instance->st->dispatcher, surface_id, 1, cookie); } SPICE_GNUC_VISIBLE void spice_qxl_create_primary_surface_async(QXLInstance *instance, uint32_t surface_id, QXLDevSurfaceCreate *surface, uint64_t cookie) { red_dispatcher_create_primary_surface(instance->st->dispatcher, surface_id, surface, 1, cookie); } SPICE_GNUC_VISIBLE void spice_qxl_destroy_surface_async(QXLInstance *instance, uint32_t surface_id, uint64_t cookie) { red_dispatcher_destroy_surface_wait(instance->st->dispatcher, surface_id, 1, cookie); } SPICE_GNUC_VISIBLE void spice_qxl_flush_surfaces_async(QXLInstance *instance, uint64_t cookie) { red_dispatcher_flush_surfaces_async(instance->st->dispatcher, cookie); } void red_dispatcher_async_complete(struct RedDispatcher *dispatcher, uint64_t cookie) { pthread_mutex_lock(&dispatcher->async_lock); switch (dispatcher->async_message) { case RED_WORKER_MESSAGE_UPDATE_ASYNC: break; case RED_WORKER_MESSAGE_ADD_MEMSLOT_ASYNC: break; case RED_WORKER_MESSAGE_DESTROY_SURFACES_ASYNC: break; case RED_WORKER_MESSAGE_CREATE_PRIMARY_SURFACE_ASYNC: red_dispatcher_create_primary_surface_complete(dispatcher); break; case RED_WORKER_MESSAGE_DESTROY_PRIMARY_SURFACE_ASYNC: red_dispatcher_destroy_primary_surface_complete(dispatcher); break; case RED_WORKER_MESSAGE_DESTROY_SURFACE_WAIT_ASYNC: break; case RED_WORKER_MESSAGE_FLUSH_SURFACES_ASYNC: break; default: red_printf("unexpected message"); } dispatcher->async_message = RED_WORKER_MESSAGE_NOP; pthread_mutex_unlock(&dispatcher->async_lock); dispatcher->qxl->st->qif->async_complete(dispatcher->qxl, cookie); } static RedChannel *red_dispatcher_display_channel_create(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_DISPLAY_CHANNEL_CREATE; 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); return display_channel; } static RedChannel *red_dispatcher_cursor_channel_create(RedDispatcher *dispatcher) { RedWorkerMessage message = RED_WORKER_MESSAGE_CURSOR_CHANNEL_CREATE; 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); return cursor_channel; } RedDispatcher *red_dispatcher_init(QXLInstance *qxl) { RedDispatcher *dispatcher; int channels[2]; RedWorkerMessage message; WorkerInitData init_data; QXLDevInitInfo init_info; int r; RedChannel *display_channel; RedChannel *cursor_channel; sigset_t thread_sig_mask; sigset_t curr_sig_mask; ClientCbs client_cbs = {0,}; quic_init(); sw_canvas_init(); #ifdef USE_OPENGL 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]; init_data.qxl = dispatcher->qxl = qxl; init_data.id = qxl->id; init_data.channel = channels[1]; init_data.pending = &dispatcher->pending; init_data.num_renderers = num_renderers; memcpy(init_data.renderers, renderers, sizeof(init_data.renderers)); dispatcher->async_message = RED_WORKER_MESSAGE_NOP; pthread_mutex_init(&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; qxl->st->qif->get_init_info(qxl, &init_info); init_data.memslot_id_bits = init_info.memslot_id_bits; init_data.memslot_gen_bits = init_info.memslot_gen_bits; init_data.num_memslots = init_info.num_memslots; 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; sigfillset(&thread_sig_mask); sigdelset(&thread_sig_mask, SIGILL); 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))) { red_error("create thread failed %d", r); } pthread_sigmask(SIG_SETMASK, &curr_sig_mask, NULL); read_message(dispatcher->channel, &message); ASSERT(message == RED_WORKER_MESSAGE_READY); display_channel = red_dispatcher_display_channel_create(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); reds_register_channel(display_channel); } cursor_channel = red_dispatcher_cursor_channel_create(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); reds_register_channel(cursor_channel); } qxl->st->qif->attache_worker(qxl, &dispatcher->base); qxl->st->qif->set_compression_level(qxl, calc_compression_level()); dispatcher->next = dispatchers; dispatchers = dispatcher; return dispatcher; }