1a3e22260SGerd Hoffmann /* 2a3e22260SGerd Hoffmann * Copyright (C) 2010 Red Hat, Inc. 3a3e22260SGerd Hoffmann * 4a3e22260SGerd Hoffmann * This program is free software; you can redistribute it and/or 5a3e22260SGerd Hoffmann * modify it under the terms of the GNU General Public License as 6a3e22260SGerd Hoffmann * published by the Free Software Foundation; either version 2 or 7a3e22260SGerd Hoffmann * (at your option) version 3 of the License. 8a3e22260SGerd Hoffmann * 9a3e22260SGerd Hoffmann * This program is distributed in the hope that it will be useful, 10a3e22260SGerd Hoffmann * but WITHOUT ANY WARRANTY; without even the implied warranty of 11a3e22260SGerd Hoffmann * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12a3e22260SGerd Hoffmann * GNU General Public License for more details. 13a3e22260SGerd Hoffmann * 14a3e22260SGerd Hoffmann * You should have received a copy of the GNU General Public License 15a3e22260SGerd Hoffmann * along with this program; if not, see <http://www.gnu.org/licenses/>. 16a3e22260SGerd Hoffmann */ 17a3e22260SGerd Hoffmann 18a3e22260SGerd Hoffmann #include <spice/ipc_ring.h> 19a3e22260SGerd Hoffmann #include <spice/enums.h> 20a3e22260SGerd Hoffmann #include <spice/qxl_dev.h> 21a3e22260SGerd Hoffmann 22e0c64d08SGerd Hoffmann #include "qemu-thread.h" 2307536094SGerd Hoffmann #include "console.h" 24a3e22260SGerd Hoffmann #include "pflib.h" 251dfb4dd9SLuiz Capitulino #include "sysemu.h" 26a3e22260SGerd Hoffmann 27a3e22260SGerd Hoffmann #define NUM_MEMSLOTS 8 28a3e22260SGerd Hoffmann #define MEMSLOT_GENERATION_BITS 8 29a3e22260SGerd Hoffmann #define MEMSLOT_SLOT_BITS 8 30a3e22260SGerd Hoffmann 31a3e22260SGerd Hoffmann #define MEMSLOT_GROUP_HOST 0 32a3e22260SGerd Hoffmann #define MEMSLOT_GROUP_GUEST 1 33a3e22260SGerd Hoffmann #define NUM_MEMSLOTS_GROUPS 2 34a3e22260SGerd Hoffmann 35a3e22260SGerd Hoffmann #define NUM_SURFACES 1024 36a3e22260SGerd Hoffmann 375ff4e36cSAlon Levy /* 385ff4e36cSAlon Levy * Internal enum to differenciate between options for 395ff4e36cSAlon Levy * io calls that have a sync (old) version and an _async (new) 405ff4e36cSAlon Levy * version: 415ff4e36cSAlon Levy * QXL_SYNC: use the old version 425ff4e36cSAlon Levy * QXL_ASYNC: use the new version and make sure there are no two 435ff4e36cSAlon Levy * happening at the same time. This is used for guest initiated 445ff4e36cSAlon Levy * calls 455ff4e36cSAlon Levy */ 465ff4e36cSAlon Levy typedef enum qxl_async_io { 475ff4e36cSAlon Levy QXL_SYNC, 485ff4e36cSAlon Levy QXL_ASYNC, 495ff4e36cSAlon Levy } qxl_async_io; 505ff4e36cSAlon Levy 512e1a98c9SAlon Levy enum { 522e1a98c9SAlon Levy QXL_COOKIE_TYPE_IO, 5381fb6f15SAlon Levy QXL_COOKIE_TYPE_RENDER_UPDATE_AREA, 54*020af1c4SAlon Levy QXL_COOKIE_TYPE_POST_LOAD_MONITORS_CONFIG, 552e1a98c9SAlon Levy }; 562e1a98c9SAlon Levy 572e1a98c9SAlon Levy typedef struct QXLCookie { 582e1a98c9SAlon Levy int type; 592e1a98c9SAlon Levy uint64_t io; 602e1a98c9SAlon Levy union { 612e1a98c9SAlon Levy uint32_t surface_id; 6281fb6f15SAlon Levy QXLRect area; 6381fb6f15SAlon Levy struct { 6481fb6f15SAlon Levy QXLRect area; 6581fb6f15SAlon Levy int redraw; 6681fb6f15SAlon Levy } render; 672e1a98c9SAlon Levy } u; 682e1a98c9SAlon Levy } QXLCookie; 692e1a98c9SAlon Levy 702e1a98c9SAlon Levy QXLCookie *qxl_cookie_new(int type, uint64_t io); 712e1a98c9SAlon Levy 72e0c64d08SGerd Hoffmann typedef struct SimpleSpiceDisplay SimpleSpiceDisplay; 73e0c64d08SGerd Hoffmann typedef struct SimpleSpiceUpdate SimpleSpiceUpdate; 74e0c64d08SGerd Hoffmann 75e0c64d08SGerd Hoffmann struct SimpleSpiceDisplay { 76a3e22260SGerd Hoffmann DisplayState *ds; 77a3e22260SGerd Hoffmann void *buf; 78a3e22260SGerd Hoffmann int bufsize; 79a3e22260SGerd Hoffmann QXLWorker *worker; 80a3e22260SGerd Hoffmann QXLInstance qxl; 81a3e22260SGerd Hoffmann uint32_t unique; 82a3e22260SGerd Hoffmann QemuPfConv *conv; 83a3e22260SGerd Hoffmann 84a3e22260SGerd Hoffmann QXLRect dirty; 85a3e22260SGerd Hoffmann int notify; 8671d388d4SYonit Halperin #if SPICE_SERVER_VERSION < 0x000b02 /* before 0.11.2 */ 87a3e22260SGerd Hoffmann int running; 8871d388d4SYonit Halperin #endif 89a3e22260SGerd Hoffmann 90e0c64d08SGerd Hoffmann /* 91e0c64d08SGerd Hoffmann * All struct members below this comment can be accessed from 92e0c64d08SGerd Hoffmann * both spice server and qemu (iothread) context and any access 93e0c64d08SGerd Hoffmann * to them must be protected by the lock. 94e0c64d08SGerd Hoffmann */ 95e0c64d08SGerd Hoffmann QemuMutex lock; 96e0c64d08SGerd Hoffmann SimpleSpiceUpdate *update; 9707536094SGerd Hoffmann QEMUCursor *cursor; 9807536094SGerd Hoffmann int mouse_x, mouse_y; 99e0c64d08SGerd Hoffmann }; 100e0c64d08SGerd Hoffmann 101e0c64d08SGerd Hoffmann struct SimpleSpiceUpdate { 102a3e22260SGerd Hoffmann QXLDrawable drawable; 103a3e22260SGerd Hoffmann QXLImage image; 104a3e22260SGerd Hoffmann QXLCommandExt ext; 105a3e22260SGerd Hoffmann uint8_t *bitmap; 106e0c64d08SGerd Hoffmann }; 107a3e22260SGerd Hoffmann 108a3e22260SGerd Hoffmann int qemu_spice_rect_is_empty(const QXLRect* r); 109a3e22260SGerd Hoffmann void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r); 110a3e22260SGerd Hoffmann 111a3e22260SGerd Hoffmann void qemu_spice_destroy_update(SimpleSpiceDisplay *sdpy, SimpleSpiceUpdate *update); 112a3e22260SGerd Hoffmann void qemu_spice_create_host_memslot(SimpleSpiceDisplay *ssd); 113a3e22260SGerd Hoffmann void qemu_spice_create_host_primary(SimpleSpiceDisplay *ssd); 114a3e22260SGerd Hoffmann void qemu_spice_destroy_host_primary(SimpleSpiceDisplay *ssd); 1151dfb4dd9SLuiz Capitulino void qemu_spice_vm_change_state_handler(void *opaque, int running, 1161dfb4dd9SLuiz Capitulino RunState state); 117a963f876SGerd Hoffmann void qemu_spice_display_init_common(SimpleSpiceDisplay *ssd, DisplayState *ds); 118a3e22260SGerd Hoffmann 119a3e22260SGerd Hoffmann void qemu_spice_display_update(SimpleSpiceDisplay *ssd, 120a3e22260SGerd Hoffmann int x, int y, int w, int h); 121a3e22260SGerd Hoffmann void qemu_spice_display_resize(SimpleSpiceDisplay *ssd); 122a3e22260SGerd Hoffmann void qemu_spice_display_refresh(SimpleSpiceDisplay *ssd); 123bb5a8cd5SAlon Levy void qemu_spice_cursor_refresh_unlocked(SimpleSpiceDisplay *ssd); 1245c59d118SGerd Hoffmann 1255ff4e36cSAlon Levy void qemu_spice_add_memslot(SimpleSpiceDisplay *ssd, QXLDevMemSlot *memslot, 1265ff4e36cSAlon Levy qxl_async_io async); 1275c59d118SGerd Hoffmann void qemu_spice_del_memslot(SimpleSpiceDisplay *ssd, uint32_t gid, 1285c59d118SGerd Hoffmann uint32_t sid); 1295c59d118SGerd Hoffmann void qemu_spice_create_primary_surface(SimpleSpiceDisplay *ssd, uint32_t id, 1305ff4e36cSAlon Levy QXLDevSurfaceCreate *surface, 1315ff4e36cSAlon Levy qxl_async_io async); 1325ff4e36cSAlon Levy void qemu_spice_destroy_primary_surface(SimpleSpiceDisplay *ssd, 1335ff4e36cSAlon Levy uint32_t id, qxl_async_io async); 1345c59d118SGerd Hoffmann void qemu_spice_wakeup(SimpleSpiceDisplay *ssd); 13571d388d4SYonit Halperin #if SPICE_SERVER_VERSION >= 0x000b02 /* before 0.11.2 */ 13671d388d4SYonit Halperin void qemu_spice_display_start(void); 13771d388d4SYonit Halperin void qemu_spice_display_stop(void); 13871d388d4SYonit Halperin #endif 13971d388d4SYonit Halperin int qemu_spice_display_is_running(SimpleSpiceDisplay *ssd); 140