2 * QEMU DBus display console
4 * Copyright (c) 2021 Marc-André Lureau <marcandre.lureau@redhat.com>
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 #include "qemu/osdep.h"
25 #include "qapi/error.h"
27 #include "ui/kbd-state.h"
30 #include <gio/gunixfdlist.h>
34 struct _DBusDisplayConsole
{
35 GDBusObjectSkeleton parent_instance
;
36 DisplayChangeListener dcl
;
39 GHashTable
*listeners
;
40 QemuDBusDisplay1Console
*iface
;
42 QemuDBusDisplay1Keyboard
*iface_kbd
;
45 QemuDBusDisplay1Mouse
*iface_mouse
;
49 Notifier mouse_mode_notifier
;
52 G_DEFINE_TYPE(DBusDisplayConsole
,
54 G_TYPE_DBUS_OBJECT_SKELETON
)
57 dbus_display_console_set_size(DBusDisplayConsole
*ddc
,
58 uint32_t width
, uint32_t height
)
60 g_object_set(ddc
->iface
,
67 dbus_gfx_switch(DisplayChangeListener
*dcl
,
68 struct DisplaySurface
*new_surface
)
70 DBusDisplayConsole
*ddc
= container_of(dcl
, DBusDisplayConsole
, dcl
);
72 dbus_display_console_set_size(ddc
,
73 surface_width(new_surface
),
74 surface_height(new_surface
));
78 dbus_gfx_update(DisplayChangeListener
*dcl
,
79 int x
, int y
, int w
, int h
)
84 dbus_gl_scanout_disable(DisplayChangeListener
*dcl
)
89 dbus_gl_scanout_texture(DisplayChangeListener
*dcl
,
92 uint32_t backing_width
,
93 uint32_t backing_height
,
94 uint32_t x
, uint32_t y
,
95 uint32_t w
, uint32_t h
)
97 DBusDisplayConsole
*ddc
= container_of(dcl
, DBusDisplayConsole
, dcl
);
99 dbus_display_console_set_size(ddc
, w
, h
);
103 dbus_gl_scanout_dmabuf(DisplayChangeListener
*dcl
,
106 DBusDisplayConsole
*ddc
= container_of(dcl
, DBusDisplayConsole
, dcl
);
108 dbus_display_console_set_size(ddc
,
114 dbus_gl_scanout_update(DisplayChangeListener
*dcl
,
115 uint32_t x
, uint32_t y
,
116 uint32_t w
, uint32_t h
)
120 const DisplayChangeListenerOps dbus_console_dcl_ops
= {
121 .dpy_name
= "dbus-console",
122 .dpy_gfx_switch
= dbus_gfx_switch
,
123 .dpy_gfx_update
= dbus_gfx_update
,
124 .dpy_gl_scanout_disable
= dbus_gl_scanout_disable
,
125 .dpy_gl_scanout_texture
= dbus_gl_scanout_texture
,
126 .dpy_gl_scanout_dmabuf
= dbus_gl_scanout_dmabuf
,
127 .dpy_gl_update
= dbus_gl_scanout_update
,
131 dbus_display_console_init(DBusDisplayConsole
*object
)
133 DBusDisplayConsole
*ddc
= DBUS_DISPLAY_CONSOLE(object
);
135 ddc
->listeners
= g_hash_table_new_full(g_str_hash
, g_str_equal
,
136 NULL
, g_object_unref
);
137 ddc
->dcl
.ops
= &dbus_console_dcl_ops
;
141 dbus_display_console_dispose(GObject
*object
)
143 DBusDisplayConsole
*ddc
= DBUS_DISPLAY_CONSOLE(object
);
145 unregister_displaychangelistener(&ddc
->dcl
);
146 g_clear_object(&ddc
->iface_kbd
);
147 g_clear_object(&ddc
->iface
);
148 g_clear_pointer(&ddc
->listeners
, g_hash_table_unref
);
149 g_clear_pointer(&ddc
->kbd
, qkbd_state_free
);
151 G_OBJECT_CLASS(dbus_display_console_parent_class
)->dispose(object
);
155 dbus_display_console_class_init(DBusDisplayConsoleClass
*klass
)
157 GObjectClass
*gobject_class
= G_OBJECT_CLASS(klass
);
159 gobject_class
->dispose
= dbus_display_console_dispose
;
163 listener_vanished_cb(DBusDisplayListener
*listener
)
165 DBusDisplayConsole
*ddc
= dbus_display_listener_get_console(listener
);
166 const char *name
= dbus_display_listener_get_bus_name(listener
);
168 trace_dbus_listener_vanished(name
);
170 g_hash_table_remove(ddc
->listeners
, name
);
171 qkbd_state_lift_all_keys(ddc
->kbd
);
175 dbus_console_set_ui_info(DBusDisplayConsole
*ddc
,
176 GDBusMethodInvocation
*invocation
,
177 guint16 arg_width_mm
,
178 guint16 arg_height_mm
,
185 .width_mm
= arg_width_mm
,
186 .height_mm
= arg_height_mm
,
190 .height
= arg_height
,
193 if (!dpy_ui_info_supported(ddc
->dcl
.con
)) {
194 g_dbus_method_invocation_return_error(invocation
,
196 DBUS_DISPLAY_ERROR_UNSUPPORTED
,
197 "SetUIInfo is not supported");
198 return DBUS_METHOD_INVOCATION_HANDLED
;
201 dpy_set_ui_info(ddc
->dcl
.con
, &info
, false);
202 qemu_dbus_display1_console_complete_set_uiinfo(ddc
->iface
, invocation
);
203 return DBUS_METHOD_INVOCATION_HANDLED
;
207 dbus_console_register_listener(DBusDisplayConsole
*ddc
,
208 GDBusMethodInvocation
*invocation
,
209 GUnixFDList
*fd_list
,
210 GVariant
*arg_listener
)
212 const char *sender
= g_dbus_method_invocation_get_sender(invocation
);
213 GDBusConnection
*listener_conn
;
214 g_autoptr(GError
) err
= NULL
;
215 g_autoptr(GSocket
) socket
= NULL
;
216 g_autoptr(GSocketConnection
) socket_conn
= NULL
;
217 g_autofree
char *guid
= g_dbus_generate_guid();
218 DBusDisplayListener
*listener
;
221 if (sender
&& g_hash_table_contains(ddc
->listeners
, sender
)) {
222 g_dbus_method_invocation_return_error(
225 DBUS_DISPLAY_ERROR_INVALID
,
226 "`%s` is already registered!",
228 return DBUS_METHOD_INVOCATION_HANDLED
;
231 fd
= g_unix_fd_list_get(fd_list
, g_variant_get_handle(arg_listener
), &err
);
233 g_dbus_method_invocation_return_error(
236 DBUS_DISPLAY_ERROR_FAILED
,
237 "Couldn't get peer fd: %s", err
->message
);
238 return DBUS_METHOD_INVOCATION_HANDLED
;
241 socket
= g_socket_new_from_fd(fd
, &err
);
243 g_dbus_method_invocation_return_error(
246 DBUS_DISPLAY_ERROR_FAILED
,
247 "Couldn't make a socket: %s", err
->message
);
249 return DBUS_METHOD_INVOCATION_HANDLED
;
251 socket_conn
= g_socket_connection_factory_create_connection(socket
);
253 qemu_dbus_display1_console_complete_register_listener(
254 ddc
->iface
, invocation
, NULL
);
256 listener_conn
= g_dbus_connection_new_sync(
257 G_IO_STREAM(socket_conn
),
259 G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_SERVER
,
262 error_report("Failed to setup peer connection: %s", err
->message
);
263 return DBUS_METHOD_INVOCATION_HANDLED
;
266 listener
= dbus_display_listener_new(sender
, listener_conn
, ddc
);
268 return DBUS_METHOD_INVOCATION_HANDLED
;
271 g_hash_table_insert(ddc
->listeners
,
272 (gpointer
)dbus_display_listener_get_bus_name(listener
),
274 g_object_connect(listener_conn
,
275 "swapped-signal::closed", listener_vanished_cb
, listener
,
278 trace_dbus_registered_listener(sender
);
279 return DBUS_METHOD_INVOCATION_HANDLED
;
283 dbus_kbd_press(DBusDisplayConsole
*ddc
,
284 GDBusMethodInvocation
*invocation
,
287 QKeyCode qcode
= qemu_input_key_number_to_qcode(arg_keycode
);
289 trace_dbus_kbd_press(arg_keycode
);
291 qkbd_state_key_event(ddc
->kbd
, qcode
, true);
293 qemu_dbus_display1_keyboard_complete_press(ddc
->iface_kbd
, invocation
);
295 return DBUS_METHOD_INVOCATION_HANDLED
;
299 dbus_kbd_release(DBusDisplayConsole
*ddc
,
300 GDBusMethodInvocation
*invocation
,
303 QKeyCode qcode
= qemu_input_key_number_to_qcode(arg_keycode
);
305 trace_dbus_kbd_release(arg_keycode
);
307 qkbd_state_key_event(ddc
->kbd
, qcode
, false);
309 qemu_dbus_display1_keyboard_complete_release(ddc
->iface_kbd
, invocation
);
311 return DBUS_METHOD_INVOCATION_HANDLED
;
315 dbus_kbd_qemu_leds_updated(void *data
, int ledstate
)
317 DBusDisplayConsole
*ddc
= DBUS_DISPLAY_CONSOLE(data
);
319 qemu_dbus_display1_keyboard_set_modifiers(ddc
->iface_kbd
, ledstate
);
323 dbus_mouse_rel_motion(DBusDisplayConsole
*ddc
,
324 GDBusMethodInvocation
*invocation
,
327 trace_dbus_mouse_rel_motion(dx
, dy
);
329 if (qemu_input_is_absolute()) {
330 g_dbus_method_invocation_return_error(
331 invocation
, DBUS_DISPLAY_ERROR
,
332 DBUS_DISPLAY_ERROR_INVALID
,
333 "Mouse is not relative");
334 return DBUS_METHOD_INVOCATION_HANDLED
;
337 qemu_input_queue_rel(ddc
->dcl
.con
, INPUT_AXIS_X
, dx
);
338 qemu_input_queue_rel(ddc
->dcl
.con
, INPUT_AXIS_Y
, dy
);
339 qemu_input_event_sync();
341 qemu_dbus_display1_mouse_complete_rel_motion(ddc
->iface_mouse
,
344 return DBUS_METHOD_INVOCATION_HANDLED
;
348 dbus_mouse_set_pos(DBusDisplayConsole
*ddc
,
349 GDBusMethodInvocation
*invocation
,
354 trace_dbus_mouse_set_pos(x
, y
);
356 if (!qemu_input_is_absolute()) {
357 g_dbus_method_invocation_return_error(
358 invocation
, DBUS_DISPLAY_ERROR
,
359 DBUS_DISPLAY_ERROR_INVALID
,
360 "Mouse is not absolute");
361 return DBUS_METHOD_INVOCATION_HANDLED
;
364 width
= qemu_console_get_width(ddc
->dcl
.con
, 0);
365 height
= qemu_console_get_height(ddc
->dcl
.con
, 0);
366 if (x
>= width
|| y
>= height
) {
367 g_dbus_method_invocation_return_error(
368 invocation
, DBUS_DISPLAY_ERROR
,
369 DBUS_DISPLAY_ERROR_INVALID
,
370 "Invalid mouse position");
371 return DBUS_METHOD_INVOCATION_HANDLED
;
373 qemu_input_queue_abs(ddc
->dcl
.con
, INPUT_AXIS_X
, x
, 0, width
);
374 qemu_input_queue_abs(ddc
->dcl
.con
, INPUT_AXIS_Y
, y
, 0, height
);
375 qemu_input_event_sync();
377 qemu_dbus_display1_mouse_complete_set_abs_position(ddc
->iface_mouse
,
380 return DBUS_METHOD_INVOCATION_HANDLED
;
384 dbus_mouse_press(DBusDisplayConsole
*ddc
,
385 GDBusMethodInvocation
*invocation
,
388 trace_dbus_mouse_press(button
);
390 qemu_input_queue_btn(ddc
->dcl
.con
, button
, true);
391 qemu_input_event_sync();
393 qemu_dbus_display1_mouse_complete_press(ddc
->iface_mouse
, invocation
);
395 return DBUS_METHOD_INVOCATION_HANDLED
;
399 dbus_mouse_release(DBusDisplayConsole
*ddc
,
400 GDBusMethodInvocation
*invocation
,
403 trace_dbus_mouse_release(button
);
405 qemu_input_queue_btn(ddc
->dcl
.con
, button
, false);
406 qemu_input_event_sync();
408 qemu_dbus_display1_mouse_complete_release(ddc
->iface_mouse
, invocation
);
410 return DBUS_METHOD_INVOCATION_HANDLED
;
414 dbus_mouse_mode_change(Notifier
*notify
, void *data
)
416 DBusDisplayConsole
*ddc
=
417 container_of(notify
, DBusDisplayConsole
, mouse_mode_notifier
);
419 g_object_set(ddc
->iface_mouse
,
420 "is-absolute", qemu_input_is_absolute(),
424 int dbus_display_console_get_index(DBusDisplayConsole
*ddc
)
426 return qemu_console_get_index(ddc
->dcl
.con
);
430 dbus_display_console_new(DBusDisplay
*display
, QemuConsole
*con
)
432 g_autofree
char *path
= NULL
;
433 g_autofree
char *label
= NULL
;
434 char device_addr
[256] = "";
435 DBusDisplayConsole
*ddc
;
441 label
= qemu_console_get_label(con
);
442 idx
= qemu_console_get_index(con
);
443 path
= g_strdup_printf(DBUS_DISPLAY1_ROOT
"/Console_%d", idx
);
444 ddc
= g_object_new(DBUS_DISPLAY_TYPE_CONSOLE
,
445 "g-object-path", path
,
447 ddc
->display
= display
;
449 /* handle errors, and skip non graphics? */
450 qemu_console_fill_device_address(
451 con
, device_addr
, sizeof(device_addr
), NULL
);
453 ddc
->iface
= qemu_dbus_display1_console_skeleton_new();
454 g_object_set(ddc
->iface
,
456 "type", qemu_console_is_graphic(con
) ? "Graphic" : "Text",
457 "head", qemu_console_get_head(con
),
458 "width", qemu_console_get_width(con
, 0),
459 "height", qemu_console_get_height(con
, 0),
460 "device-address", device_addr
,
462 g_object_connect(ddc
->iface
,
463 "swapped-signal::handle-register-listener",
464 dbus_console_register_listener
, ddc
,
465 "swapped-signal::handle-set-uiinfo",
466 dbus_console_set_ui_info
, ddc
,
468 g_dbus_object_skeleton_add_interface(G_DBUS_OBJECT_SKELETON(ddc
),
469 G_DBUS_INTERFACE_SKELETON(ddc
->iface
));
471 ddc
->kbd
= qkbd_state_init(con
);
472 ddc
->iface_kbd
= qemu_dbus_display1_keyboard_skeleton_new();
473 qemu_add_led_event_handler(dbus_kbd_qemu_leds_updated
, ddc
);
474 g_object_connect(ddc
->iface_kbd
,
475 "swapped-signal::handle-press", dbus_kbd_press
, ddc
,
476 "swapped-signal::handle-release", dbus_kbd_release
, ddc
,
478 g_dbus_object_skeleton_add_interface(G_DBUS_OBJECT_SKELETON(ddc
),
479 G_DBUS_INTERFACE_SKELETON(ddc
->iface_kbd
));
481 ddc
->iface_mouse
= qemu_dbus_display1_mouse_skeleton_new();
482 g_object_connect(ddc
->iface_mouse
,
483 "swapped-signal::handle-set-abs-position", dbus_mouse_set_pos
, ddc
,
484 "swapped-signal::handle-rel-motion", dbus_mouse_rel_motion
, ddc
,
485 "swapped-signal::handle-press", dbus_mouse_press
, ddc
,
486 "swapped-signal::handle-release", dbus_mouse_release
, ddc
,
488 g_dbus_object_skeleton_add_interface(G_DBUS_OBJECT_SKELETON(ddc
),
489 G_DBUS_INTERFACE_SKELETON(ddc
->iface_mouse
));
491 register_displaychangelistener(&ddc
->dcl
);
492 ddc
->mouse_mode_notifier
.notify
= dbus_mouse_mode_change
;
493 qemu_add_mouse_mode_change_notifier(&ddc
->mouse_mode_notifier
);