4 #include "ui/qemu-pixman.h"
5 #include "qom/object.h"
6 #include "qemu/notify.h"
7 #include "qemu/error-report.h"
8 #include "qapi/qapi-types-ui.h"
11 # include <epoxy/gl.h>
12 # include "ui/shader.h"
15 /* keyboard/mouse support */
17 #define MOUSE_EVENT_LBUTTON 0x01
18 #define MOUSE_EVENT_RBUTTON 0x02
19 #define MOUSE_EVENT_MBUTTON 0x04
20 #define MOUSE_EVENT_WHEELUP 0x08
21 #define MOUSE_EVENT_WHEELDN 0x10
23 /* identical to the ps/2 keyboard bits */
24 #define QEMU_SCROLL_LOCK_LED (1 << 0)
25 #define QEMU_NUM_LOCK_LED (1 << 1)
26 #define QEMU_CAPS_LOCK_LED (1 << 2)
29 #define GUI_REFRESH_INTERVAL_DEFAULT 30
30 #define GUI_REFRESH_INTERVAL_IDLE 3000
32 /* Color number is match to standard vga palette */
33 enum qemu_color_names
{
39 QEMU_COLOR_MAGENTA
= 5,
40 QEMU_COLOR_YELLOW
= 6,
43 /* Convert to curses char attributes */
44 #define ATTR2CHTYPE(c, fg, bg, bold) \
45 ((bold) << 21 | (bg) << 11 | (fg) << 8 | (c))
47 typedef void QEMUPutKBDEvent(void *opaque
, int keycode
);
48 typedef void QEMUPutLEDEvent(void *opaque
, int ledstate
);
49 typedef void QEMUPutMouseEvent(void *opaque
, int dx
, int dy
, int dz
, int buttons_state
);
51 typedef struct QEMUPutMouseEntry QEMUPutMouseEntry
;
52 typedef struct QEMUPutKbdEntry QEMUPutKbdEntry
;
53 typedef struct QEMUPutLEDEntry QEMUPutLEDEntry
;
55 QEMUPutKbdEntry
*qemu_add_kbd_event_handler(QEMUPutKBDEvent
*func
,
57 QEMUPutMouseEntry
*qemu_add_mouse_event_handler(QEMUPutMouseEvent
*func
,
58 void *opaque
, int absolute
,
60 void qemu_remove_mouse_event_handler(QEMUPutMouseEntry
*entry
);
61 void qemu_activate_mouse_event_handler(QEMUPutMouseEntry
*entry
);
63 QEMUPutLEDEntry
*qemu_add_led_event_handler(QEMUPutLEDEvent
*func
, void *opaque
);
64 void qemu_remove_led_event_handler(QEMUPutLEDEntry
*entry
);
66 void kbd_put_ledstate(int ledstate
);
68 struct MouseTransformInfo
{
69 /* Touchscreen resolution */
72 /* Calibration values as used/generated by tslib */
76 void hmp_mouse_set(Monitor
*mon
, const QDict
*qdict
);
78 /* keysym is a unicode code except for special keys (see QEMU_KEY_xxx
80 #define QEMU_KEY_ESC1(c) ((c) | 0xe100)
81 #define QEMU_KEY_BACKSPACE 0x007f
82 #define QEMU_KEY_UP QEMU_KEY_ESC1('A')
83 #define QEMU_KEY_DOWN QEMU_KEY_ESC1('B')
84 #define QEMU_KEY_RIGHT QEMU_KEY_ESC1('C')
85 #define QEMU_KEY_LEFT QEMU_KEY_ESC1('D')
86 #define QEMU_KEY_HOME QEMU_KEY_ESC1(1)
87 #define QEMU_KEY_END QEMU_KEY_ESC1(4)
88 #define QEMU_KEY_PAGEUP QEMU_KEY_ESC1(5)
89 #define QEMU_KEY_PAGEDOWN QEMU_KEY_ESC1(6)
90 #define QEMU_KEY_DELETE QEMU_KEY_ESC1(3)
92 #define QEMU_KEY_CTRL_UP 0xe400
93 #define QEMU_KEY_CTRL_DOWN 0xe401
94 #define QEMU_KEY_CTRL_LEFT 0xe402
95 #define QEMU_KEY_CTRL_RIGHT 0xe403
96 #define QEMU_KEY_CTRL_HOME 0xe404
97 #define QEMU_KEY_CTRL_END 0xe405
98 #define QEMU_KEY_CTRL_PAGEUP 0xe406
99 #define QEMU_KEY_CTRL_PAGEDOWN 0xe407
101 void kbd_put_keysym_console(QemuConsole
*s
, int keysym
);
102 bool kbd_put_qcode_console(QemuConsole
*s
, int qcode
, bool ctrl
);
103 void kbd_put_string_console(QemuConsole
*s
, const char *str
, int len
);
104 void kbd_put_keysym(int keysym
);
108 #define TYPE_QEMU_CONSOLE "qemu-console"
109 #define QEMU_CONSOLE(obj) \
110 OBJECT_CHECK(QemuConsole, (obj), TYPE_QEMU_CONSOLE)
111 #define QEMU_CONSOLE_GET_CLASS(obj) \
112 OBJECT_GET_CLASS(QemuConsoleClass, (obj), TYPE_QEMU_CONSOLE)
113 #define QEMU_CONSOLE_CLASS(klass) \
114 OBJECT_CLASS_CHECK(QemuConsoleClass, (klass), TYPE_QEMU_CONSOLE)
116 typedef struct QemuConsoleClass QemuConsoleClass
;
118 struct QemuConsoleClass
{
119 ObjectClass parent_class
;
122 #define QEMU_ALLOCATED_FLAG 0x01
125 uint8_t bits_per_pixel
;
126 uint8_t bytes_per_pixel
;
127 uint8_t depth
; /* color depth in bits */
128 uint32_t rmask
, gmask
, bmask
, amask
;
129 uint8_t rshift
, gshift
, bshift
, ashift
;
130 uint8_t rmax
, gmax
, bmax
, amax
;
131 uint8_t rbits
, gbits
, bbits
, abits
;
134 struct DisplaySurface
{
135 pixman_format_code_t format
;
136 pixman_image_t
*image
;
145 typedef struct QemuUIInfo
{
153 /* cursor data format is 32bit RGBA */
154 typedef struct QEMUCursor
{
161 QEMUCursor
*cursor_alloc(int width
, int height
);
162 void cursor_get(QEMUCursor
*c
);
163 void cursor_put(QEMUCursor
*c
);
164 QEMUCursor
*cursor_builtin_hidden(void);
165 QEMUCursor
*cursor_builtin_left_ptr(void);
166 void cursor_print_ascii_art(QEMUCursor
*c
, const char *prefix
);
167 int cursor_get_mono_bpl(QEMUCursor
*c
);
168 void cursor_set_mono(QEMUCursor
*c
,
169 uint32_t foreground
, uint32_t background
, uint8_t *image
,
170 int transparent
, uint8_t *mask
);
171 void cursor_get_mono_image(QEMUCursor
*c
, int foreground
, uint8_t *mask
);
172 void cursor_get_mono_mask(QEMUCursor
*c
, int transparent
, uint8_t *mask
);
174 typedef void *QEMUGLContext
;
175 typedef struct QEMUGLParams QEMUGLParams
;
177 struct QEMUGLParams
{
191 typedef struct DisplayChangeListenerOps
{
192 const char *dpy_name
;
194 void (*dpy_refresh
)(DisplayChangeListener
*dcl
);
196 void (*dpy_gfx_update
)(DisplayChangeListener
*dcl
,
197 int x
, int y
, int w
, int h
);
198 void (*dpy_gfx_switch
)(DisplayChangeListener
*dcl
,
199 struct DisplaySurface
*new_surface
);
200 bool (*dpy_gfx_check_format
)(DisplayChangeListener
*dcl
,
201 pixman_format_code_t format
);
203 void (*dpy_text_cursor
)(DisplayChangeListener
*dcl
,
205 void (*dpy_text_resize
)(DisplayChangeListener
*dcl
,
207 void (*dpy_text_update
)(DisplayChangeListener
*dcl
,
208 int x
, int y
, int w
, int h
);
210 void (*dpy_mouse_set
)(DisplayChangeListener
*dcl
,
211 int x
, int y
, int on
);
212 void (*dpy_cursor_define
)(DisplayChangeListener
*dcl
,
215 QEMUGLContext (*dpy_gl_ctx_create
)(DisplayChangeListener
*dcl
,
216 QEMUGLParams
*params
);
217 void (*dpy_gl_ctx_destroy
)(DisplayChangeListener
*dcl
,
219 int (*dpy_gl_ctx_make_current
)(DisplayChangeListener
*dcl
,
221 QEMUGLContext (*dpy_gl_ctx_get_current
)(DisplayChangeListener
*dcl
);
223 void (*dpy_gl_scanout_disable
)(DisplayChangeListener
*dcl
);
224 void (*dpy_gl_scanout_texture
)(DisplayChangeListener
*dcl
,
226 bool backing_y_0_top
,
227 uint32_t backing_width
,
228 uint32_t backing_height
,
229 uint32_t x
, uint32_t y
,
230 uint32_t w
, uint32_t h
);
231 void (*dpy_gl_scanout_dmabuf
)(DisplayChangeListener
*dcl
,
233 void (*dpy_gl_cursor_dmabuf
)(DisplayChangeListener
*dcl
,
234 QemuDmaBuf
*dmabuf
, bool have_hot
,
235 uint32_t hot_x
, uint32_t hot_y
);
236 void (*dpy_gl_cursor_position
)(DisplayChangeListener
*dcl
,
237 uint32_t pos_x
, uint32_t pos_y
);
238 void (*dpy_gl_release_dmabuf
)(DisplayChangeListener
*dcl
,
240 void (*dpy_gl_update
)(DisplayChangeListener
*dcl
,
241 uint32_t x
, uint32_t y
, uint32_t w
, uint32_t h
);
243 } DisplayChangeListenerOps
;
245 struct DisplayChangeListener
{
246 uint64_t update_interval
;
247 const DisplayChangeListenerOps
*ops
;
251 QLIST_ENTRY(DisplayChangeListener
) next
;
254 DisplayState
*init_displaystate(void);
255 DisplaySurface
*qemu_create_displaysurface_from(int width
, int height
,
256 pixman_format_code_t format
,
257 int linesize
, uint8_t *data
);
258 DisplaySurface
*qemu_create_displaysurface_pixman(pixman_image_t
*image
);
259 DisplaySurface
*qemu_create_displaysurface_guestmem(int width
, int height
,
260 pixman_format_code_t format
,
263 DisplaySurface
*qemu_create_message_surface(int w
, int h
,
265 PixelFormat
qemu_default_pixelformat(int bpp
);
267 DisplaySurface
*qemu_create_displaysurface(int width
, int height
);
268 void qemu_free_displaysurface(DisplaySurface
*surface
);
270 static inline int is_surface_bgr(DisplaySurface
*surface
)
272 if (PIXMAN_FORMAT_BPP(surface
->format
) == 32 &&
273 PIXMAN_FORMAT_TYPE(surface
->format
) == PIXMAN_TYPE_ABGR
) {
280 static inline int is_buffer_shared(DisplaySurface
*surface
)
282 return !(surface
->flags
& QEMU_ALLOCATED_FLAG
);
285 void register_displaychangelistener(DisplayChangeListener
*dcl
);
286 void update_displaychangelistener(DisplayChangeListener
*dcl
,
288 void unregister_displaychangelistener(DisplayChangeListener
*dcl
);
290 bool dpy_ui_info_supported(QemuConsole
*con
);
291 int dpy_set_ui_info(QemuConsole
*con
, QemuUIInfo
*info
);
293 void dpy_gfx_update(QemuConsole
*con
, int x
, int y
, int w
, int h
);
294 void dpy_gfx_update_full(QemuConsole
*con
);
295 void dpy_gfx_replace_surface(QemuConsole
*con
,
296 DisplaySurface
*surface
);
297 void dpy_text_cursor(QemuConsole
*con
, int x
, int y
);
298 void dpy_text_update(QemuConsole
*con
, int x
, int y
, int w
, int h
);
299 void dpy_text_resize(QemuConsole
*con
, int w
, int h
);
300 void dpy_mouse_set(QemuConsole
*con
, int x
, int y
, int on
);
301 void dpy_cursor_define(QemuConsole
*con
, QEMUCursor
*cursor
);
302 bool dpy_cursor_define_supported(QemuConsole
*con
);
303 bool dpy_gfx_check_format(QemuConsole
*con
,
304 pixman_format_code_t format
);
306 void dpy_gl_scanout_disable(QemuConsole
*con
);
307 void dpy_gl_scanout_texture(QemuConsole
*con
,
308 uint32_t backing_id
, bool backing_y_0_top
,
309 uint32_t backing_width
, uint32_t backing_height
,
310 uint32_t x
, uint32_t y
, uint32_t w
, uint32_t h
);
311 void dpy_gl_scanout_dmabuf(QemuConsole
*con
,
313 void dpy_gl_cursor_dmabuf(QemuConsole
*con
, QemuDmaBuf
*dmabuf
,
314 bool have_hot
, uint32_t hot_x
, uint32_t hot_y
);
315 void dpy_gl_cursor_position(QemuConsole
*con
,
316 uint32_t pos_x
, uint32_t pos_y
);
317 void dpy_gl_release_dmabuf(QemuConsole
*con
,
319 void dpy_gl_update(QemuConsole
*con
,
320 uint32_t x
, uint32_t y
, uint32_t w
, uint32_t h
);
322 QEMUGLContext
dpy_gl_ctx_create(QemuConsole
*con
,
323 QEMUGLParams
*params
);
324 void dpy_gl_ctx_destroy(QemuConsole
*con
, QEMUGLContext ctx
);
325 int dpy_gl_ctx_make_current(QemuConsole
*con
, QEMUGLContext ctx
);
326 QEMUGLContext
dpy_gl_ctx_get_current(QemuConsole
*con
);
328 bool console_has_gl(QemuConsole
*con
);
329 bool console_has_gl_dmabuf(QemuConsole
*con
);
331 static inline int surface_stride(DisplaySurface
*s
)
333 return pixman_image_get_stride(s
->image
);
336 static inline void *surface_data(DisplaySurface
*s
)
338 return pixman_image_get_data(s
->image
);
341 static inline int surface_width(DisplaySurface
*s
)
343 return pixman_image_get_width(s
->image
);
346 static inline int surface_height(DisplaySurface
*s
)
348 return pixman_image_get_height(s
->image
);
351 static inline int surface_bits_per_pixel(DisplaySurface
*s
)
353 int bits
= PIXMAN_FORMAT_BPP(s
->format
);
357 static inline int surface_bytes_per_pixel(DisplaySurface
*s
)
359 int bits
= PIXMAN_FORMAT_BPP(s
->format
);
360 return DIV_ROUND_UP(bits
, 8);
363 static inline pixman_format_code_t
surface_format(DisplaySurface
*s
)
368 typedef uint32_t console_ch_t
;
370 static inline void console_write_ch(console_ch_t
*dest
, uint32_t ch
)
375 typedef struct GraphicHwOps
{
376 void (*invalidate
)(void *opaque
);
377 void (*gfx_update
)(void *opaque
);
378 void (*text_update
)(void *opaque
, console_ch_t
*text
);
379 void (*update_interval
)(void *opaque
, uint64_t interval
);
380 int (*ui_info
)(void *opaque
, uint32_t head
, QemuUIInfo
*info
);
381 void (*gl_block
)(void *opaque
, bool block
);
384 QemuConsole
*graphic_console_init(DeviceState
*dev
, uint32_t head
,
385 const GraphicHwOps
*ops
,
387 void graphic_console_set_hwops(QemuConsole
*con
,
388 const GraphicHwOps
*hw_ops
,
390 void graphic_console_close(QemuConsole
*con
);
392 void graphic_hw_update(QemuConsole
*con
);
393 void graphic_hw_invalidate(QemuConsole
*con
);
394 void graphic_hw_text_update(QemuConsole
*con
, console_ch_t
*chardata
);
395 void graphic_hw_gl_block(QemuConsole
*con
, bool block
);
397 void qemu_console_early_init(void);
399 QemuConsole
*qemu_console_lookup_by_index(unsigned int index
);
400 QemuConsole
*qemu_console_lookup_by_device(DeviceState
*dev
, uint32_t head
);
401 QemuConsole
*qemu_console_lookup_by_device_name(const char *device_id
,
402 uint32_t head
, Error
**errp
);
403 QemuConsole
*qemu_console_lookup_unused(void);
404 bool qemu_console_is_visible(QemuConsole
*con
);
405 bool qemu_console_is_graphic(QemuConsole
*con
);
406 bool qemu_console_is_fixedsize(QemuConsole
*con
);
407 bool qemu_console_is_gl_blocked(QemuConsole
*con
);
408 char *qemu_console_get_label(QemuConsole
*con
);
409 int qemu_console_get_index(QemuConsole
*con
);
410 uint32_t qemu_console_get_head(QemuConsole
*con
);
411 QemuUIInfo
*qemu_console_get_ui_info(QemuConsole
*con
);
412 int qemu_console_get_width(QemuConsole
*con
, int fallback
);
413 int qemu_console_get_height(QemuConsole
*con
, int fallback
);
414 /* Return the low-level window id for the console */
415 int qemu_console_get_window_id(QemuConsole
*con
);
416 /* Set the low-level window id for the console */
417 void qemu_console_set_window_id(QemuConsole
*con
, int window_id
);
419 void console_select(unsigned int index
);
420 void qemu_console_resize(QemuConsole
*con
, int width
, int height
);
421 DisplaySurface
*qemu_console_surface(QemuConsole
*con
);
425 bool console_gl_check_format(DisplayChangeListener
*dcl
,
426 pixman_format_code_t format
);
427 void surface_gl_create_texture(QemuGLShader
*gls
,
428 DisplaySurface
*surface
);
429 void surface_gl_update_texture(QemuGLShader
*gls
,
430 DisplaySurface
*surface
,
431 int x
, int y
, int w
, int h
);
432 void surface_gl_render_texture(QemuGLShader
*gls
,
433 DisplaySurface
*surface
);
434 void surface_gl_destroy_texture(QemuGLShader
*gls
,
435 DisplaySurface
*surface
);
436 void surface_gl_setup_viewport(QemuGLShader
*gls
,
437 DisplaySurface
*surface
,
441 typedef struct QemuDisplay QemuDisplay
;
445 void (*early_init
)(DisplayOptions
*opts
);
446 void (*init
)(DisplayState
*ds
, DisplayOptions
*opts
);
449 void qemu_display_register(QemuDisplay
*ui
);
450 bool qemu_display_find_default(DisplayOptions
*opts
);
451 void qemu_display_early_init(DisplayOptions
*opts
);
452 void qemu_display_init(DisplayState
*ds
, DisplayOptions
*opts
);
455 void vnc_display_init(const char *id
);
456 void vnc_display_open(const char *id
, Error
**errp
);
457 void vnc_display_add_client(const char *id
, int csock
, bool skipauth
);
458 int vnc_display_password(const char *id
, const char *password
);
459 int vnc_display_pw_expire(const char *id
, time_t expires
);
460 QemuOpts
*vnc_parse(const char *str
, Error
**errp
);
461 int vnc_init_func(void *opaque
, QemuOpts
*opts
, Error
**errp
);
464 int index_from_key(const char *key
, size_t key_length
);