| /* | 
 |  * 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)); | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | #ifdef CONFIG_PIXMAN | 
 | 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); | 
 | } | 
 | #endif | 
 |  | 
 | 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); | 
 | } |