mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-31 20:44:16 +00:00 
			
		
		
		
	 b7d75c0b48
			
		
	
	
		b7d75c0b48
		
	
	
	
	
		
			
			This moves these commands from MAINTAINERS sections "Human Monitor (HMP)" and "QMP" to "Character device backends". Signed-off-by: Markus Armbruster <armbru@redhat.com> Message-Id: <20230124121946.1139465-4-armbru@redhat.com>
		
			
				
	
	
		
			221 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			221 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * HMP commands related to character devices
 | |
|  *
 | |
|  * 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"
 | |
| #include "chardev/char.h"
 | |
| #include "monitor/hmp.h"
 | |
| #include "monitor/monitor.h"
 | |
| #include "qapi/error.h"
 | |
| #include "qapi/qapi-commands-char.h"
 | |
| #include "qapi/qmp/qdict.h"
 | |
| #include "qemu/config-file.h"
 | |
| #include "qemu/option.h"
 | |
| 
 | |
| void hmp_info_chardev(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     ChardevInfoList *char_info, *info;
 | |
| 
 | |
|     char_info = qmp_query_chardev(NULL);
 | |
|     for (info = char_info; info; info = info->next) {
 | |
|         monitor_printf(mon, "%s: filename=%s\n", info->value->label,
 | |
|                                                  info->value->filename);
 | |
|     }
 | |
| 
 | |
|     qapi_free_ChardevInfoList(char_info);
 | |
| }
 | |
| 
 | |
| void hmp_ringbuf_write(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *chardev = qdict_get_str(qdict, "device");
 | |
|     const char *data = qdict_get_str(qdict, "data");
 | |
|     Error *err = NULL;
 | |
| 
 | |
|     qmp_ringbuf_write(chardev, data, false, 0, &err);
 | |
| 
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| void hmp_ringbuf_read(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     uint32_t size = qdict_get_int(qdict, "size");
 | |
|     const char *chardev = qdict_get_str(qdict, "device");
 | |
|     char *data;
 | |
|     Error *err = NULL;
 | |
|     int i;
 | |
| 
 | |
|     data = qmp_ringbuf_read(chardev, size, false, 0, &err);
 | |
|     if (hmp_handle_error(mon, err)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     for (i = 0; data[i]; i++) {
 | |
|         unsigned char ch = data[i];
 | |
| 
 | |
|         if (ch == '\\') {
 | |
|             monitor_printf(mon, "\\\\");
 | |
|         } else if ((ch < 0x20 && ch != '\n' && ch != '\t') || ch == 0x7F) {
 | |
|             monitor_printf(mon, "\\u%04X", ch);
 | |
|         } else {
 | |
|             monitor_printf(mon, "%c", ch);
 | |
|         }
 | |
| 
 | |
|     }
 | |
|     monitor_printf(mon, "\n");
 | |
|     g_free(data);
 | |
| }
 | |
| 
 | |
| void hmp_chardev_add(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *args = qdict_get_str(qdict, "args");
 | |
|     Error *err = NULL;
 | |
|     QemuOpts *opts;
 | |
| 
 | |
|     opts = qemu_opts_parse_noisily(qemu_find_opts("chardev"), args, true);
 | |
|     if (opts == NULL) {
 | |
|         error_setg(&err, "Parsing chardev args failed");
 | |
|     } else {
 | |
|         qemu_chr_new_from_opts(opts, NULL, &err);
 | |
|         qemu_opts_del(opts);
 | |
|     }
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| void hmp_chardev_change(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     const char *args = qdict_get_str(qdict, "args");
 | |
|     const char *id;
 | |
|     Error *err = NULL;
 | |
|     ChardevBackend *backend = NULL;
 | |
|     ChardevReturn *ret = NULL;
 | |
|     QemuOpts *opts = qemu_opts_parse_noisily(qemu_find_opts("chardev"), args,
 | |
|                                              true);
 | |
|     if (!opts) {
 | |
|         error_setg(&err, "Parsing chardev args failed");
 | |
|         goto end;
 | |
|     }
 | |
| 
 | |
|     id = qdict_get_str(qdict, "id");
 | |
|     if (qemu_opts_id(opts)) {
 | |
|         error_setg(&err, "Unexpected 'id' parameter");
 | |
|         goto end;
 | |
|     }
 | |
| 
 | |
|     backend = qemu_chr_parse_opts(opts, &err);
 | |
|     if (!backend) {
 | |
|         goto end;
 | |
|     }
 | |
| 
 | |
|     ret = qmp_chardev_change(id, backend, &err);
 | |
| 
 | |
| end:
 | |
|     qapi_free_ChardevReturn(ret);
 | |
|     qapi_free_ChardevBackend(backend);
 | |
|     qemu_opts_del(opts);
 | |
|     hmp_handle_error(mon, err);
 | |
| }
 | |
| 
 | |
| void hmp_chardev_remove(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     Error *local_err = NULL;
 | |
| 
 | |
|     qmp_chardev_remove(qdict_get_str(qdict, "id"), &local_err);
 | |
|     hmp_handle_error(mon, local_err);
 | |
| }
 | |
| 
 | |
| void hmp_chardev_send_break(Monitor *mon, const QDict *qdict)
 | |
| {
 | |
|     Error *local_err = NULL;
 | |
| 
 | |
|     qmp_chardev_send_break(qdict_get_str(qdict, "id"), &local_err);
 | |
|     hmp_handle_error(mon, local_err);
 | |
| }
 | |
| 
 | |
| void chardev_add_completion(ReadLineState *rs, int nb_args, const char *str)
 | |
| {
 | |
|     size_t len;
 | |
|     ChardevBackendInfoList *list, *start;
 | |
| 
 | |
|     if (nb_args != 2) {
 | |
|         return;
 | |
|     }
 | |
|     len = strlen(str);
 | |
|     readline_set_completion_index(rs, len);
 | |
| 
 | |
|     start = list = qmp_query_chardev_backends(NULL);
 | |
|     while (list) {
 | |
|         const char *chr_name = list->value->name;
 | |
| 
 | |
|         if (!strncmp(chr_name, str, len)) {
 | |
|             readline_add_completion(rs, chr_name);
 | |
|         }
 | |
|         list = list->next;
 | |
|     }
 | |
|     qapi_free_ChardevBackendInfoList(start);
 | |
| }
 | |
| 
 | |
| void chardev_remove_completion(ReadLineState *rs, int nb_args, const char *str)
 | |
| {
 | |
|     size_t len;
 | |
|     ChardevInfoList *list, *start;
 | |
| 
 | |
|     if (nb_args != 2) {
 | |
|         return;
 | |
|     }
 | |
|     len = strlen(str);
 | |
|     readline_set_completion_index(rs, len);
 | |
| 
 | |
|     start = list = qmp_query_chardev(NULL);
 | |
|     while (list) {
 | |
|         ChardevInfo *chr = list->value;
 | |
| 
 | |
|         if (!strncmp(chr->label, str, len)) {
 | |
|             readline_add_completion(rs, chr->label);
 | |
|         }
 | |
|         list = list->next;
 | |
|     }
 | |
|     qapi_free_ChardevInfoList(start);
 | |
| }
 | |
| 
 | |
| static void ringbuf_completion(ReadLineState *rs, const char *str)
 | |
| {
 | |
|     size_t len;
 | |
|     ChardevInfoList *list, *start;
 | |
| 
 | |
|     len = strlen(str);
 | |
|     readline_set_completion_index(rs, len);
 | |
| 
 | |
|     start = list = qmp_query_chardev(NULL);
 | |
|     while (list) {
 | |
|         ChardevInfo *chr_info = list->value;
 | |
| 
 | |
|         if (!strncmp(chr_info->label, str, len)) {
 | |
|             Chardev *chr = qemu_chr_find(chr_info->label);
 | |
|             if (chr && CHARDEV_IS_RINGBUF(chr)) {
 | |
|                 readline_add_completion(rs, chr_info->label);
 | |
|             }
 | |
|         }
 | |
|         list = list->next;
 | |
|     }
 | |
|     qapi_free_ChardevInfoList(start);
 | |
| }
 | |
| 
 | |
| void ringbuf_write_completion(ReadLineState *rs, int nb_args, const char *str)
 | |
| {
 | |
|     if (nb_args != 2) {
 | |
|         return;
 | |
|     }
 | |
|     ringbuf_completion(rs, str);
 | |
| }
 |