mirror of
https://github.com/qemu/qemu.git
synced 2025-01-24 14:34:00 +08:00
b153f9019b
Add support for cursor dmabufs. qemu has to render the cursor for that, so in case a cursor is present qemu allocates a new dmabuf, blits the scanout, blends in the pointer and passes on the new dmabuf to spice-server. Without cursor qemu continues to simply pass on the scanout dmabuf as-is. Signed-off-by: Gerd Hoffmann <kraxel@redhat.com> Reviewed-by: Marc-André Lureau <marcandre.lureau@redhat.com> Message-id: 20180308090618.30147-4-kraxel@redhat.com
182 lines
5.2 KiB
C
182 lines
5.2 KiB
C
/*
|
|
* Copyright (C) 2010 Red Hat, Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 or
|
|
* (at your option) version 3 of the License.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <spice/ipc_ring.h>
|
|
#include <spice/enums.h>
|
|
#include <spice/qxl_dev.h>
|
|
|
|
#include "qemu/thread.h"
|
|
#include "ui/qemu-pixman.h"
|
|
#include "ui/console.h"
|
|
#include "sysemu/sysemu.h"
|
|
|
|
#if defined(CONFIG_OPENGL_DMABUF)
|
|
# if SPICE_SERVER_VERSION >= 0x000d01 /* release 0.13.1 */
|
|
# define HAVE_SPICE_GL 1
|
|
# include "ui/egl-helpers.h"
|
|
# include "ui/egl-context.h"
|
|
# endif
|
|
#endif
|
|
|
|
#define NUM_MEMSLOTS 8
|
|
#define MEMSLOT_GENERATION_BITS 8
|
|
#define MEMSLOT_SLOT_BITS 8
|
|
|
|
#define MEMSLOT_GROUP_HOST 0
|
|
#define MEMSLOT_GROUP_GUEST 1
|
|
#define NUM_MEMSLOTS_GROUPS 2
|
|
|
|
/*
|
|
* Internal enum to differenciate between options for
|
|
* io calls that have a sync (old) version and an _async (new)
|
|
* version:
|
|
* QXL_SYNC: use the old version
|
|
* QXL_ASYNC: use the new version and make sure there are no two
|
|
* happening at the same time. This is used for guest initiated
|
|
* calls
|
|
*/
|
|
typedef enum qxl_async_io {
|
|
QXL_SYNC,
|
|
QXL_ASYNC,
|
|
} qxl_async_io;
|
|
|
|
enum {
|
|
QXL_COOKIE_TYPE_IO,
|
|
QXL_COOKIE_TYPE_RENDER_UPDATE_AREA,
|
|
QXL_COOKIE_TYPE_POST_LOAD_MONITORS_CONFIG,
|
|
QXL_COOKIE_TYPE_GL_DRAW_DONE,
|
|
};
|
|
|
|
typedef struct QXLCookie {
|
|
int type;
|
|
uint64_t io;
|
|
union {
|
|
uint32_t surface_id;
|
|
QXLRect area;
|
|
struct {
|
|
QXLRect area;
|
|
int redraw;
|
|
} render;
|
|
void *data;
|
|
} u;
|
|
} QXLCookie;
|
|
|
|
QXLCookie *qxl_cookie_new(int type, uint64_t io);
|
|
|
|
typedef struct SimpleSpiceDisplay SimpleSpiceDisplay;
|
|
typedef struct SimpleSpiceUpdate SimpleSpiceUpdate;
|
|
typedef struct SimpleSpiceCursor SimpleSpiceCursor;
|
|
|
|
struct SimpleSpiceDisplay {
|
|
DisplaySurface *ds;
|
|
DisplayChangeListener dcl;
|
|
void *buf;
|
|
int bufsize;
|
|
QXLInstance qxl;
|
|
uint32_t unique;
|
|
pixman_image_t *surface;
|
|
pixman_image_t *mirror;
|
|
int32_t num_surfaces;
|
|
|
|
QXLRect dirty;
|
|
int notify;
|
|
|
|
/*
|
|
* All struct members below this comment can be accessed from
|
|
* both spice server and qemu (iothread) context and any access
|
|
* to them must be protected by the lock.
|
|
*/
|
|
QemuMutex lock;
|
|
QTAILQ_HEAD(, SimpleSpiceUpdate) updates;
|
|
|
|
/* cursor (without qxl): displaychangelistener -> spice server */
|
|
SimpleSpiceCursor *ptr_define;
|
|
SimpleSpiceCursor *ptr_move;
|
|
int16_t ptr_x, ptr_y;
|
|
int16_t hot_x, hot_y;
|
|
|
|
/* cursor (with qxl): qxl local renderer -> displaychangelistener */
|
|
QEMUCursor *cursor;
|
|
int mouse_x, mouse_y;
|
|
QEMUBH *cursor_bh;
|
|
|
|
#ifdef HAVE_SPICE_GL
|
|
/* opengl rendering */
|
|
QEMUBH *gl_unblock_bh;
|
|
QEMUTimer *gl_unblock_timer;
|
|
QemuGLShader *gls;
|
|
int gl_updates;
|
|
bool have_scanout;
|
|
bool have_surface;
|
|
|
|
QemuDmaBuf *guest_dmabuf;
|
|
bool guest_dmabuf_refresh;
|
|
bool render_cursor;
|
|
|
|
egl_fb guest_fb;
|
|
egl_fb blit_fb;
|
|
egl_fb cursor_fb;
|
|
bool have_hot;
|
|
#endif
|
|
};
|
|
|
|
struct SimpleSpiceUpdate {
|
|
QXLDrawable drawable;
|
|
QXLImage image;
|
|
QXLCommandExt ext;
|
|
uint8_t *bitmap;
|
|
QTAILQ_ENTRY(SimpleSpiceUpdate) next;
|
|
};
|
|
|
|
struct SimpleSpiceCursor {
|
|
QXLCursorCmd cmd;
|
|
QXLCommandExt ext;
|
|
QXLCursor cursor;
|
|
};
|
|
|
|
extern bool spice_opengl;
|
|
|
|
int qemu_spice_rect_is_empty(const QXLRect* r);
|
|
void qemu_spice_rect_union(QXLRect *dest, const QXLRect *r);
|
|
|
|
void qemu_spice_destroy_update(SimpleSpiceDisplay *sdpy, SimpleSpiceUpdate *update);
|
|
void qemu_spice_create_host_memslot(SimpleSpiceDisplay *ssd);
|
|
void qemu_spice_create_host_primary(SimpleSpiceDisplay *ssd);
|
|
void qemu_spice_destroy_host_primary(SimpleSpiceDisplay *ssd);
|
|
void qemu_spice_display_init_common(SimpleSpiceDisplay *ssd);
|
|
|
|
void qemu_spice_display_update(SimpleSpiceDisplay *ssd,
|
|
int x, int y, int w, int h);
|
|
void qemu_spice_display_switch(SimpleSpiceDisplay *ssd,
|
|
DisplaySurface *surface);
|
|
void qemu_spice_display_refresh(SimpleSpiceDisplay *ssd);
|
|
void qemu_spice_cursor_refresh_bh(void *opaque);
|
|
|
|
void qemu_spice_add_memslot(SimpleSpiceDisplay *ssd, QXLDevMemSlot *memslot,
|
|
qxl_async_io async);
|
|
void qemu_spice_del_memslot(SimpleSpiceDisplay *ssd, uint32_t gid,
|
|
uint32_t sid);
|
|
void qemu_spice_create_primary_surface(SimpleSpiceDisplay *ssd, uint32_t id,
|
|
QXLDevSurfaceCreate *surface,
|
|
qxl_async_io async);
|
|
void qemu_spice_destroy_primary_surface(SimpleSpiceDisplay *ssd,
|
|
uint32_t id, qxl_async_io async);
|
|
void qemu_spice_wakeup(SimpleSpiceDisplay *ssd);
|
|
void qemu_spice_display_start(void);
|
|
void qemu_spice_display_stop(void);
|
|
int qemu_spice_display_is_running(SimpleSpiceDisplay *ssd);
|