 f9e1ef7482
			
		
	
	
		f9e1ef7482
		
	
	
	
	
		
			
			It has nothing to do with migration, except for the "migrate" in the name of the command. Move it with the rest of the ui commands. Signed-off-by: Juan Quintela <quintela@redhat.com> Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
		
			
				
	
	
		
			478 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			478 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * HMP commands related to UI
 | |
|  *
 | |
|  * Copyright IBM, Corp. 2011
 | |
|  *
 | |
|  * Authors:
 | |
|  *  Anthony Liguori   <aliguori@us.ibm.com>
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2.  See
 | |
|  * the COPYING file in the top-level directory.
 | |
|  *
 | |
|  * Contributions after 2012-01-13 are licensed under the terms of the
 | |
|  * GNU GPL, version 2 or (at your option) any later version.
 | |
|  */
 | |
| 
 | |
| #include "qemu/osdep.h"
 | |
| #ifdef CONFIG_SPICE
 | |
| #include <spice/enums.h>
 | |
| #endif
 | |
| #include "monitor/hmp.h"
 | |
| #include "monitor/monitor-internal.h"
 | |
| #include "qapi/error.h"
 | |
| #include "qapi/qapi-commands-ui.h"
 | |
| #include "qapi/qmp/qdict.h"
 | |
| #include "qemu/cutils.h"
 | |
| #include "ui/console.h"
 | |
| #include "ui/input.h"
 | |
| 
 | |
| static int mouse_button_state;
 | |
| 
 | |
| void hmp_mouse_move(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     int dx, dy, dz, button;
 | |
|     const char *dx_str = qdict_get_str(qdict, "dx_str");
 | |
|     const char *dy_str = qdict_get_str(qdict, "dy_str");
 | |
|     const char *dz_str = qdict_get_try_str(qdict, "dz_str");
 | |
| 
 | |
|     dx = strtol(dx_str, NULL, 0);
 | |
|     dy = strtol(dy_str, NULL, 0);
 | |
|     qemu_input_queue_rel(NULL, INPUT_AXIS_X, dx);
 | |
|     qemu_input_queue_rel(NULL, INPUT_AXIS_Y, dy);
 | |
| 
 | |
|     if (dz_str) {
 | |
|         dz = strtol(dz_str, NULL, 0);
 | |
|         if (dz != 0) {
 | |
|             button = (dz > 0) ? INPUT_BUTTON_WHEEL_UP : INPUT_BUTTON_WHEEL_DOWN;
 | |
|             qemu_input_queue_btn(NULL, button, true);
 | |
|             qemu_input_event_sync();
 | |
|             qemu_input_queue_btn(NULL, button, false);
 | |
|         }
 | |
|     }
 | |
|     qemu_input_event_sync();
 | |
| }
 | |
| 
 | |
| void hmp_mouse_button(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     static uint32_t bmap[INPUT_BUTTON__MAX] = {
 | |
|         [INPUT_BUTTON_LEFT]       = MOUSE_EVENT_LBUTTON,
 | |
|         [INPUT_BUTTON_MIDDLE]     = MOUSE_EVENT_MBUTTON,
 | |
|         [INPUT_BUTTON_RIGHT]      = MOUSE_EVENT_RBUTTON,
 | |
|     };
 | |
|     int button_state = qdict_get_int(qdict, "button_state");
 | |
| 
 | |
|     if (mouse_button_state == button_state) {
 | |
|         return;
 | |
|     }
 | |
|     qemu_input_update_buttons(NULL, bmap, mouse_button_state, button_state);
 | |
|     qemu_input_event_sync();
 | |
|     mouse_button_state = button_state;
 | |
| }
 | |
| 
 | |
| void hmp_mouse_set(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     Error *err = NULL;
 | |
| 
 | |
|     qemu_mouse_set(qdict_get_int(qdict, "index"), &err);
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| void hmp_info_mice(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     MouseInfoList *mice_list, *mouse;
 | |
| 
 | |
|     mice_list = qmp_query_mice(NULL);
 | |
|     if (!mice_list) {
 | |
|         monitor_printf(mon, "No mouse devices connected\n");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     for (mouse = mice_list; mouse; mouse = mouse->next) {
 | |
|         monitor_printf(mon, "%c Mouse #%" PRId64 ": %s%s\n",
 | |
|                        mouse->value->current ? '*' : ' ',
 | |
|                        mouse->value->index, mouse->value->name,
 | |
|                        mouse->value->absolute ? " (absolute)" : "");
 | |
|     }
 | |
| 
 | |
|     qapi_free_MouseInfoList(mice_list);
 | |
| }
 | |
| 
 | |
| #ifdef CONFIG_VNC
 | |
| /* Helper for hmp_info_vnc_clients, _servers */
 | |
| static void hmp_info_VncBasicInfo(Monitor *mon, VncBasicInfo *info,
 | |
|                                   const char *name)
 | |
| {
 | |
|     monitor_printf(mon, "  %s: %s:%s (%s%s)\n",
 | |
|                    name,
 | |
|                    info->host,
 | |
|                    info->service,
 | |
|                    NetworkAddressFamily_str(info->family),
 | |
|                    info->websocket ? " (Websocket)" : "");
 | |
| }
 | |
| 
 | |
| /* Helper displaying and auth and crypt info */
 | |
| static void hmp_info_vnc_authcrypt(Monitor *mon, const char *indent,
 | |
|                                    VncPrimaryAuth auth,
 | |
|                                    VncVencryptSubAuth *vencrypt)
 | |
| {
 | |
|     monitor_printf(mon, "%sAuth: %s (Sub: %s)\n", indent,
 | |
|                    VncPrimaryAuth_str(auth),
 | |
|                    vencrypt ? VncVencryptSubAuth_str(*vencrypt) : "none");
 | |
| }
 | |
| 
 | |
| static void hmp_info_vnc_clients(Monitor *mon, VncClientInfoList *client)
 | |
| {
 | |
|     while (client) {
 | |
|         VncClientInfo *cinfo = client->value;
 | |
| 
 | |
|         hmp_info_VncBasicInfo(mon, qapi_VncClientInfo_base(cinfo), "Client");
 | |
|         monitor_printf(mon, "    x509_dname: %s\n",
 | |
|                        cinfo->x509_dname ?: "none");
 | |
|         monitor_printf(mon, "    sasl_username: %s\n",
 | |
|                        cinfo->sasl_username ?: "none");
 | |
| 
 | |
|         client = client->next;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void hmp_info_vnc_servers(Monitor *mon, VncServerInfo2List *server)
 | |
| {
 | |
|     while (server) {
 | |
|         VncServerInfo2 *sinfo = server->value;
 | |
|         hmp_info_VncBasicInfo(mon, qapi_VncServerInfo2_base(sinfo), "Server");
 | |
|         hmp_info_vnc_authcrypt(mon, "    ", sinfo->auth,
 | |
|                                sinfo->has_vencrypt ? &sinfo->vencrypt : NULL);
 | |
|         server = server->next;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void hmp_info_vnc(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     VncInfo2List *info2l, *info2l_head;
 | |
|     Error *err = NULL;
 | |
| 
 | |
|     info2l = qmp_query_vnc_servers(&err);
 | |
|     info2l_head = info2l;
 | |
|     if (hmp_handle_error(mon, err)) {
 | |
|         return;
 | |
|     }
 | |
|     if (!info2l) {
 | |
|         monitor_printf(mon, "None\n");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     while (info2l) {
 | |
|         VncInfo2 *info = info2l->value;
 | |
|         monitor_printf(mon, "%s:\n", info->id);
 | |
|         hmp_info_vnc_servers(mon, info->server);
 | |
|         hmp_info_vnc_clients(mon, info->clients);
 | |
|         if (!info->server) {
 | |
|             /*
 | |
|              * The server entry displays its auth, we only need to
 | |
|              * display in the case of 'reverse' connections where
 | |
|              * there's no server.
 | |
|              */
 | |
|             hmp_info_vnc_authcrypt(mon, "  ", info->auth,
 | |
|                                info->has_vencrypt ? &info->vencrypt : NULL);
 | |
|         }
 | |
|         if (info->display) {
 | |
|             monitor_printf(mon, "  Display: %s\n", info->display);
 | |
|         }
 | |
|         info2l = info2l->next;
 | |
|     }
 | |
| 
 | |
|     qapi_free_VncInfo2List(info2l_head);
 | |
| 
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_SPICE
 | |
| void hmp_info_spice(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     SpiceChannelList *chan;
 | |
|     SpiceInfo *info;
 | |
|     const char *channel_name;
 | |
|     static const char *const channel_names[] = {
 | |
|         [SPICE_CHANNEL_MAIN] = "main",
 | |
|         [SPICE_CHANNEL_DISPLAY] = "display",
 | |
|         [SPICE_CHANNEL_INPUTS] = "inputs",
 | |
|         [SPICE_CHANNEL_CURSOR] = "cursor",
 | |
|         [SPICE_CHANNEL_PLAYBACK] = "playback",
 | |
|         [SPICE_CHANNEL_RECORD] = "record",
 | |
|         [SPICE_CHANNEL_TUNNEL] = "tunnel",
 | |
|         [SPICE_CHANNEL_SMARTCARD] = "smartcard",
 | |
|         [SPICE_CHANNEL_USBREDIR] = "usbredir",
 | |
|         [SPICE_CHANNEL_PORT] = "port",
 | |
|         [SPICE_CHANNEL_WEBDAV] = "webdav",
 | |
|     };
 | |
| 
 | |
|     info = qmp_query_spice(NULL);
 | |
| 
 | |
|     if (!info->enabled) {
 | |
|         monitor_printf(mon, "Server: disabled\n");
 | |
|         goto out;
 | |
|     }
 | |
| 
 | |
|     monitor_printf(mon, "Server:\n");
 | |
|     if (info->has_port) {
 | |
|         monitor_printf(mon, "     address: %s:%" PRId64 "\n",
 | |
|                        info->host, info->port);
 | |
|     }
 | |
|     if (info->has_tls_port) {
 | |
|         monitor_printf(mon, "     address: %s:%" PRId64 " [tls]\n",
 | |
|                        info->host, info->tls_port);
 | |
|     }
 | |
|     monitor_printf(mon, "    migrated: %s\n",
 | |
|                    info->migrated ? "true" : "false");
 | |
|     monitor_printf(mon, "        auth: %s\n", info->auth);
 | |
|     monitor_printf(mon, "    compiled: %s\n", info->compiled_version);
 | |
|     monitor_printf(mon, "  mouse-mode: %s\n",
 | |
|                    SpiceQueryMouseMode_str(info->mouse_mode));
 | |
| 
 | |
|     if (!info->has_channels || info->channels == NULL) {
 | |
|         monitor_printf(mon, "Channels: none\n");
 | |
|     } else {
 | |
|         for (chan = info->channels; chan; chan = chan->next) {
 | |
|             monitor_printf(mon, "Channel:\n");
 | |
|             monitor_printf(mon, "     address: %s:%s%s\n",
 | |
|                            chan->value->host, chan->value->port,
 | |
|                            chan->value->tls ? " [tls]" : "");
 | |
|             monitor_printf(mon, "     session: %" PRId64 "\n",
 | |
|                            chan->value->connection_id);
 | |
|             monitor_printf(mon, "     channel: %" PRId64 ":%" PRId64 "\n",
 | |
|                            chan->value->channel_type, chan->value->channel_id);
 | |
| 
 | |
|             channel_name = "unknown";
 | |
|             if (chan->value->channel_type > 0 &&
 | |
|                 chan->value->channel_type < ARRAY_SIZE(channel_names) &&
 | |
|                 channel_names[chan->value->channel_type]) {
 | |
|                 channel_name = channel_names[chan->value->channel_type];
 | |
|             }
 | |
| 
 | |
|             monitor_printf(mon, "     channel name: %s\n", channel_name);
 | |
|         }
 | |
|     }
 | |
| 
 | |
| out:
 | |
|     qapi_free_SpiceInfo(info);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void hmp_set_password(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *protocol  = qdict_get_str(qdict, "protocol");
 | |
|     const char *password  = qdict_get_str(qdict, "password");
 | |
|     const char *display = qdict_get_try_str(qdict, "display");
 | |
|     const char *connected = qdict_get_try_str(qdict, "connected");
 | |
|     Error *err = NULL;
 | |
| 
 | |
|     SetPasswordOptions opts = {
 | |
|         .password = (char *)password,
 | |
|         .has_connected = !!connected,
 | |
|     };
 | |
| 
 | |
|     opts.connected = qapi_enum_parse(&SetPasswordAction_lookup, connected,
 | |
|                                      SET_PASSWORD_ACTION_KEEP, &err);
 | |
|     if (err) {
 | |
|         goto out;
 | |
|     }
 | |
| 
 | |
|     opts.protocol = qapi_enum_parse(&DisplayProtocol_lookup, protocol,
 | |
|                                     DISPLAY_PROTOCOL_VNC, &err);
 | |
|     if (err) {
 | |
|         goto out;
 | |
|     }
 | |
| 
 | |
|     if (opts.protocol == DISPLAY_PROTOCOL_VNC) {
 | |
|         opts.u.vnc.display = (char *)display;
 | |
|     }
 | |
| 
 | |
|     qmp_set_password(&opts, &err);
 | |
| 
 | |
| out:
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| void hmp_expire_password(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *protocol  = qdict_get_str(qdict, "protocol");
 | |
|     const char *whenstr = qdict_get_str(qdict, "time");
 | |
|     const char *display = qdict_get_try_str(qdict, "display");
 | |
|     Error *err = NULL;
 | |
| 
 | |
|     ExpirePasswordOptions opts = {
 | |
|         .time = (char *)whenstr,
 | |
|     };
 | |
| 
 | |
|     opts.protocol = qapi_enum_parse(&DisplayProtocol_lookup, protocol,
 | |
|                                     DISPLAY_PROTOCOL_VNC, &err);
 | |
|     if (err) {
 | |
|         goto out;
 | |
|     }
 | |
| 
 | |
|     if (opts.protocol == DISPLAY_PROTOCOL_VNC) {
 | |
|         opts.u.vnc.display = (char *)display;
 | |
|     }
 | |
| 
 | |
|     qmp_expire_password(&opts, &err);
 | |
| 
 | |
| out:
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| #ifdef CONFIG_VNC
 | |
| static void hmp_change_read_arg(void *opaque, const char *password,
 | |
|                                 void *readline_opaque)
 | |
| {
 | |
|     qmp_change_vnc_password(password, NULL);
 | |
|     monitor_read_command(opaque, 1);
 | |
| }
 | |
| 
 | |
| void hmp_change_vnc(Monitor *mon, const char *device, const char *target,
 | |
|                     const char *arg, const char *read_only, bool force,
 | |
|                     Error **errp)
 | |
| {
 | |
|     if (read_only) {
 | |
|         error_setg(errp, "Parameter 'read-only-mode' is invalid for VNC");
 | |
|         return;
 | |
|     }
 | |
|     if (strcmp(target, "passwd") && strcmp(target, "password")) {
 | |
|         error_setg(errp, "Expected 'password' after 'vnc'");
 | |
|         return;
 | |
|     }
 | |
|     if (!arg) {
 | |
|         MonitorHMP *hmp_mon = container_of(mon, MonitorHMP, common);
 | |
|         monitor_read_password(hmp_mon, hmp_change_read_arg, NULL);
 | |
|     } else {
 | |
|         qmp_change_vnc_password(arg, errp);
 | |
|     }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void hmp_sendkey(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *keys = qdict_get_str(qdict, "keys");
 | |
|     KeyValue *v = NULL;
 | |
|     KeyValueList *head = NULL, **tail = &head;
 | |
|     int has_hold_time = qdict_haskey(qdict, "hold-time");
 | |
|     int hold_time = qdict_get_try_int(qdict, "hold-time", -1);
 | |
|     Error *err = NULL;
 | |
|     const char *separator;
 | |
|     int keyname_len;
 | |
| 
 | |
|     while (1) {
 | |
|         separator = qemu_strchrnul(keys, '-');
 | |
|         keyname_len = separator - keys;
 | |
| 
 | |
|         /* Be compatible with old interface, convert user inputted "<" */
 | |
|         if (keys[0] == '<' && keyname_len == 1) {
 | |
|             keys = "less";
 | |
|             keyname_len = 4;
 | |
|         }
 | |
| 
 | |
|         v = g_malloc0(sizeof(*v));
 | |
| 
 | |
|         if (strstart(keys, "0x", NULL)) {
 | |
|             const char *endp;
 | |
|             int value;
 | |
| 
 | |
|             if (qemu_strtoi(keys, &endp, 0, &value) < 0) {
 | |
|                 goto err_out;
 | |
|             }
 | |
|             assert(endp <= keys + keyname_len);
 | |
|             if (endp != keys + keyname_len) {
 | |
|                 goto err_out;
 | |
|             }
 | |
|             v->type = KEY_VALUE_KIND_NUMBER;
 | |
|             v->u.number.data = value;
 | |
|         } else {
 | |
|             int idx = index_from_key(keys, keyname_len);
 | |
|             if (idx == Q_KEY_CODE__MAX) {
 | |
|                 goto err_out;
 | |
|             }
 | |
|             v->type = KEY_VALUE_KIND_QCODE;
 | |
|             v->u.qcode.data = idx;
 | |
|         }
 | |
|         QAPI_LIST_APPEND(tail, v);
 | |
|         v = NULL;
 | |
| 
 | |
|         if (!*separator) {
 | |
|             break;
 | |
|         }
 | |
|         keys = separator + 1;
 | |
|     }
 | |
| 
 | |
|     qmp_send_key(head, has_hold_time, hold_time, &err);
 | |
|     hmp_handle_error(mon, err);
 | |
| 
 | |
| out:
 | |
|     qapi_free_KeyValue(v);
 | |
|     qapi_free_KeyValueList(head);
 | |
|     return;
 | |
| 
 | |
| err_out:
 | |
|     monitor_printf(mon, "invalid parameter: %.*s\n", keyname_len, keys);
 | |
|     goto out;
 | |
| }
 | |
| 
 | |
| void sendkey_completion(ReadLineState *rs, int nb_args, const char *str)
 | |
| {
 | |
|     int i;
 | |
|     char *sep;
 | |
|     size_t len;
 | |
| 
 | |
|     if (nb_args != 2) {
 | |
|         return;
 | |
|     }
 | |
|     sep = strrchr(str, '-');
 | |
|     if (sep) {
 | |
|         str = sep + 1;
 | |
|     }
 | |
|     len = strlen(str);
 | |
|     readline_set_completion_index(rs, len);
 | |
|     for (i = 0; i < Q_KEY_CODE__MAX; i++) {
 | |
|         if (!strncmp(str, QKeyCode_str(i), len)) {
 | |
|             readline_add_completion(rs, QKeyCode_str(i));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void coroutine_fn
 | |
| hmp_screendump(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *filename = qdict_get_str(qdict, "filename");
 | |
|     const char *id = qdict_get_try_str(qdict, "device");
 | |
|     int64_t head = qdict_get_try_int(qdict, "head", 0);
 | |
|     const char *input_format  = qdict_get_try_str(qdict, "format");
 | |
|     Error *err = NULL;
 | |
|     ImageFormat format;
 | |
| 
 | |
|     format = qapi_enum_parse(&ImageFormat_lookup, input_format,
 | |
|                               IMAGE_FORMAT_PPM, &err);
 | |
|     if (err) {
 | |
|         goto end;
 | |
|     }
 | |
| 
 | |
|     qmp_screendump(filename, id, id != NULL, head,
 | |
|                    input_format != NULL, format, &err);
 | |
| end:
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| void hmp_client_migrate_info(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     Error *err = NULL;
 | |
|     const char *protocol = qdict_get_str(qdict, "protocol");
 | |
|     const char *hostname = qdict_get_str(qdict, "hostname");
 | |
|     bool has_port        = qdict_haskey(qdict, "port");
 | |
|     int port             = qdict_get_try_int(qdict, "port", -1);
 | |
|     bool has_tls_port    = qdict_haskey(qdict, "tls-port");
 | |
|     int tls_port         = qdict_get_try_int(qdict, "tls-port", -1);
 | |
|     const char *cert_subject = qdict_get_try_str(qdict, "cert-subject");
 | |
| 
 | |
|     qmp_client_migrate_info(protocol, hostname,
 | |
|                             has_port, port, has_tls_port, tls_port,
 | |
|                             cert_subject, &err);
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 |