128ecbaeeSPaolo Bonzini /* 228ecbaeeSPaolo Bonzini * Copyright (C) 2010 Red Hat, Inc. 328ecbaeeSPaolo Bonzini * 428ecbaeeSPaolo Bonzini * This program is free software; you can redistribute it and/or 528ecbaeeSPaolo Bonzini * modify it under the terms of the GNU General Public License as 628ecbaeeSPaolo Bonzini * published by the Free Software Foundation; either version 2 or 728ecbaeeSPaolo Bonzini * (at your option) version 3 of the License. 828ecbaeeSPaolo Bonzini * 928ecbaeeSPaolo Bonzini * This program is distributed in the hope that it will be useful, 1028ecbaeeSPaolo Bonzini * but WITHOUT ANY WARRANTY; without even the implied warranty of 1128ecbaeeSPaolo Bonzini * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 1228ecbaeeSPaolo Bonzini * GNU General Public License for more details. 1328ecbaeeSPaolo Bonzini * 1428ecbaeeSPaolo Bonzini * You should have received a copy of the GNU General Public License 1528ecbaeeSPaolo Bonzini * along with this program; if not, see <http://www.gnu.org/licenses/>. 1628ecbaeeSPaolo Bonzini */ 1728ecbaeeSPaolo Bonzini 18f91005e1SMarkus Armbruster #ifndef UI_SPICE_DISPLAY_H 19f91005e1SMarkus Armbruster #define UI_SPICE_DISPLAY_H 20f91005e1SMarkus Armbruster 21ec150c7eSMarkus Armbruster #include <spice.h> 2228ecbaeeSPaolo Bonzini #include <spice/ipc_ring.h> 2328ecbaeeSPaolo Bonzini #include <spice/enums.h> 2428ecbaeeSPaolo Bonzini #include <spice/qxl_dev.h> 2528ecbaeeSPaolo Bonzini 261de7afc9SPaolo Bonzini #include "qemu/thread.h" 2728ecbaeeSPaolo Bonzini #include "ui/qemu-pixman.h" 280b087861SPeter Maydell #include "ui/console.h" 2928ecbaeeSPaolo Bonzini 30bc6a3565SAkihiko Odaki #if defined(CONFIG_OPENGL) && defined(CONFIG_GBM) 319f5c6d06SMichal Privoznik # if SPICE_SERVER_VERSION >= 0x000d01 /* release 0.13.1 */ 32474114b7SGerd Hoffmann # define HAVE_SPICE_GL 1 33474114b7SGerd Hoffmann # include "ui/egl-helpers.h" 34474114b7SGerd Hoffmann # include "ui/egl-context.h" 35474114b7SGerd Hoffmann # endif 36474114b7SGerd Hoffmann #endif 37474114b7SGerd Hoffmann 3828ecbaeeSPaolo Bonzini #define NUM_MEMSLOTS 8 3928ecbaeeSPaolo Bonzini #define MEMSLOT_GENERATION_BITS 8 4028ecbaeeSPaolo Bonzini #define MEMSLOT_SLOT_BITS 8 4128ecbaeeSPaolo Bonzini 4228ecbaeeSPaolo Bonzini #define MEMSLOT_GROUP_HOST 0 4328ecbaeeSPaolo Bonzini #define MEMSLOT_GROUP_GUEST 1 4428ecbaeeSPaolo Bonzini #define NUM_MEMSLOTS_GROUPS 2 4528ecbaeeSPaolo Bonzini 4628ecbaeeSPaolo Bonzini /* 4728ecbaeeSPaolo Bonzini * Internal enum to differenciate between options for 4828ecbaeeSPaolo Bonzini * io calls that have a sync (old) version and an _async (new) 4928ecbaeeSPaolo Bonzini * version: 5028ecbaeeSPaolo Bonzini * QXL_SYNC: use the old version 5128ecbaeeSPaolo Bonzini * QXL_ASYNC: use the new version and make sure there are no two 5228ecbaeeSPaolo Bonzini * happening at the same time. This is used for guest initiated 5328ecbaeeSPaolo Bonzini * calls 5428ecbaeeSPaolo Bonzini */ 5528ecbaeeSPaolo Bonzini typedef enum qxl_async_io { 5628ecbaeeSPaolo Bonzini QXL_SYNC, 5728ecbaeeSPaolo Bonzini QXL_ASYNC, 5828ecbaeeSPaolo Bonzini } qxl_async_io; 5928ecbaeeSPaolo Bonzini 6028ecbaeeSPaolo Bonzini enum { 6128ecbaeeSPaolo Bonzini QXL_COOKIE_TYPE_IO, 6228ecbaeeSPaolo Bonzini QXL_COOKIE_TYPE_RENDER_UPDATE_AREA, 6328ecbaeeSPaolo Bonzini QXL_COOKIE_TYPE_POST_LOAD_MONITORS_CONFIG, 64474114b7SGerd Hoffmann QXL_COOKIE_TYPE_GL_DRAW_DONE, 6528ecbaeeSPaolo Bonzini }; 6628ecbaeeSPaolo Bonzini 6728ecbaeeSPaolo Bonzini typedef struct QXLCookie { 6828ecbaeeSPaolo Bonzini int type; 6928ecbaeeSPaolo Bonzini uint64_t io; 7028ecbaeeSPaolo Bonzini union { 7128ecbaeeSPaolo Bonzini uint32_t surface_id; 7228ecbaeeSPaolo Bonzini QXLRect area; 7328ecbaeeSPaolo Bonzini struct { 7428ecbaeeSPaolo Bonzini QXLRect area; 7528ecbaeeSPaolo Bonzini int redraw; 7628ecbaeeSPaolo Bonzini } render; 7739414ef4SGerd Hoffmann void *data; 7828ecbaeeSPaolo Bonzini } u; 7928ecbaeeSPaolo Bonzini } QXLCookie; 8028ecbaeeSPaolo Bonzini 8128ecbaeeSPaolo Bonzini QXLCookie *qxl_cookie_new(int type, uint64_t io); 8228ecbaeeSPaolo Bonzini 8328ecbaeeSPaolo Bonzini typedef struct SimpleSpiceDisplay SimpleSpiceDisplay; 8428ecbaeeSPaolo Bonzini typedef struct SimpleSpiceUpdate SimpleSpiceUpdate; 855643fc01SGerd Hoffmann typedef struct SimpleSpiceCursor SimpleSpiceCursor; 8628ecbaeeSPaolo Bonzini 8728ecbaeeSPaolo Bonzini struct SimpleSpiceDisplay { 8871874c17SGerd Hoffmann DisplaySurface *ds; 89*5e79d516SMarc-André Lureau DisplayGLCtx dgc; 907c20b4a3SGerd Hoffmann DisplayChangeListener dcl; 9128ecbaeeSPaolo Bonzini void *buf; 9228ecbaeeSPaolo Bonzini int bufsize; 9328ecbaeeSPaolo Bonzini QXLInstance qxl; 9428ecbaeeSPaolo Bonzini uint32_t unique; 9528ecbaeeSPaolo Bonzini pixman_image_t *surface; 9628ecbaeeSPaolo Bonzini pixman_image_t *mirror; 9728ecbaeeSPaolo Bonzini int32_t num_surfaces; 9828ecbaeeSPaolo Bonzini 9928ecbaeeSPaolo Bonzini QXLRect dirty; 10028ecbaeeSPaolo Bonzini int notify; 10128ecbaeeSPaolo Bonzini 10228ecbaeeSPaolo Bonzini /* 10328ecbaeeSPaolo Bonzini * All struct members below this comment can be accessed from 10428ecbaeeSPaolo Bonzini * both spice server and qemu (iothread) context and any access 10528ecbaeeSPaolo Bonzini * to them must be protected by the lock. 10628ecbaeeSPaolo Bonzini */ 10728ecbaeeSPaolo Bonzini QemuMutex lock; 10828ecbaeeSPaolo Bonzini QTAILQ_HEAD(, SimpleSpiceUpdate) updates; 1095643fc01SGerd Hoffmann 1105643fc01SGerd Hoffmann /* cursor (without qxl): displaychangelistener -> spice server */ 1115643fc01SGerd Hoffmann SimpleSpiceCursor *ptr_define; 1125643fc01SGerd Hoffmann SimpleSpiceCursor *ptr_move; 113dc8dceeeSMarc-André Lureau int16_t ptr_x, ptr_y; 114dc8dceeeSMarc-André Lureau int16_t hot_x, hot_y; 1155643fc01SGerd Hoffmann 1165643fc01SGerd Hoffmann /* cursor (with qxl): qxl local renderer -> displaychangelistener */ 11728ecbaeeSPaolo Bonzini QEMUCursor *cursor; 11828ecbaeeSPaolo Bonzini int mouse_x, mouse_y; 1190b2824e5SGerd Hoffmann QEMUBH *cursor_bh; 120474114b7SGerd Hoffmann 121474114b7SGerd Hoffmann #ifdef HAVE_SPICE_GL 122474114b7SGerd Hoffmann /* opengl rendering */ 123474114b7SGerd Hoffmann QEMUBH *gl_unblock_bh; 1248e388e90SGerd Hoffmann QEMUTimer *gl_unblock_timer; 12546e19e14SGerd Hoffmann QemuGLShader *gls; 12644231843SGerd Hoffmann int gl_updates; 12744231843SGerd Hoffmann bool have_scanout; 12844231843SGerd Hoffmann bool have_surface; 129b153f901SGerd Hoffmann 130b153f901SGerd Hoffmann QemuDmaBuf *guest_dmabuf; 131b153f901SGerd Hoffmann bool guest_dmabuf_refresh; 132b153f901SGerd Hoffmann bool render_cursor; 133b153f901SGerd Hoffmann 134b153f901SGerd Hoffmann egl_fb guest_fb; 135b153f901SGerd Hoffmann egl_fb blit_fb; 136b153f901SGerd Hoffmann egl_fb cursor_fb; 137b153f901SGerd Hoffmann bool have_hot; 138474114b7SGerd Hoffmann #endif 13928ecbaeeSPaolo Bonzini }; 14028ecbaeeSPaolo Bonzini 14128ecbaeeSPaolo Bonzini struct SimpleSpiceUpdate { 14228ecbaeeSPaolo Bonzini QXLDrawable drawable; 14328ecbaeeSPaolo Bonzini QXLImage image; 14428ecbaeeSPaolo Bonzini QXLCommandExt ext; 14528ecbaeeSPaolo Bonzini uint8_t *bitmap; 14628ecbaeeSPaolo Bonzini QTAILQ_ENTRY(SimpleSpiceUpdate) next; 14728ecbaeeSPaolo Bonzini }; 14828ecbaeeSPaolo Bonzini 1495643fc01SGerd Hoffmann struct SimpleSpiceCursor { 1505643fc01SGerd Hoffmann QXLCursorCmd cmd; 1515643fc01SGerd Hoffmann QXLCommandExt ext; 1525643fc01SGerd Hoffmann QXLCursor cursor; 1535643fc01SGerd Hoffmann }; 1545643fc01SGerd Hoffmann 155fe5c44f9SGerd Hoffmann extern bool spice_opengl; 156fe5c44f9SGerd Hoffmann 15728ecbaeeSPaolo Bonzini int qemu_spice_rect_is_empty(const QXLRect* r); 15828ecbaeeSPaolo Bonzini void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r); 15928ecbaeeSPaolo Bonzini 16028ecbaeeSPaolo Bonzini void qemu_spice_destroy_update(SimpleSpiceDisplay *sdpy, SimpleSpiceUpdate *update); 16128ecbaeeSPaolo Bonzini void qemu_spice_create_host_memslot(SimpleSpiceDisplay *ssd); 16228ecbaeeSPaolo Bonzini void qemu_spice_create_host_primary(SimpleSpiceDisplay *ssd); 16328ecbaeeSPaolo Bonzini void qemu_spice_destroy_host_primary(SimpleSpiceDisplay *ssd); 164c78f7137SGerd Hoffmann void qemu_spice_display_init_common(SimpleSpiceDisplay *ssd); 16528ecbaeeSPaolo Bonzini 16628ecbaeeSPaolo Bonzini void qemu_spice_display_update(SimpleSpiceDisplay *ssd, 16728ecbaeeSPaolo Bonzini int x, int y, int w, int h); 168c12aeb86SGerd Hoffmann void qemu_spice_display_switch(SimpleSpiceDisplay *ssd, 169c12aeb86SGerd Hoffmann DisplaySurface *surface); 17028ecbaeeSPaolo Bonzini void qemu_spice_display_refresh(SimpleSpiceDisplay *ssd); 1710b2824e5SGerd Hoffmann void qemu_spice_cursor_refresh_bh(void *opaque); 17228ecbaeeSPaolo Bonzini 17328ecbaeeSPaolo Bonzini void qemu_spice_add_memslot(SimpleSpiceDisplay *ssd, QXLDevMemSlot *memslot, 17428ecbaeeSPaolo Bonzini qxl_async_io async); 17528ecbaeeSPaolo Bonzini void qemu_spice_del_memslot(SimpleSpiceDisplay *ssd, uint32_t gid, 17628ecbaeeSPaolo Bonzini uint32_t sid); 17728ecbaeeSPaolo Bonzini void qemu_spice_create_primary_surface(SimpleSpiceDisplay *ssd, uint32_t id, 17828ecbaeeSPaolo Bonzini QXLDevSurfaceCreate *surface, 17928ecbaeeSPaolo Bonzini qxl_async_io async); 18028ecbaeeSPaolo Bonzini void qemu_spice_destroy_primary_surface(SimpleSpiceDisplay *ssd, 18128ecbaeeSPaolo Bonzini uint32_t id, qxl_async_io async); 18228ecbaeeSPaolo Bonzini void qemu_spice_wakeup(SimpleSpiceDisplay *ssd); 18328ecbaeeSPaolo Bonzini void qemu_spice_display_start(void); 18428ecbaeeSPaolo Bonzini void qemu_spice_display_stop(void); 18528ecbaeeSPaolo Bonzini int qemu_spice_display_is_running(SimpleSpiceDisplay *ssd); 186be812c0aSLukáš Hrázký 187be812c0aSLukáš Hrázký bool qemu_spice_fill_device_address(QemuConsole *con, 188be812c0aSLukáš Hrázký char *device_address, 189be812c0aSLukáš Hrázký size_t size); 190f91005e1SMarkus Armbruster 191f91005e1SMarkus Armbruster #endif 192