/* * HMP commands related to UI * * Copyright IBM, Corp. 2011 * * Authors: * Anthony Liguori * * 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 #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); }