| /* |
| * HMP commands related to QOM |
| * |
| * This work is licensed under the terms of the GNU GPL, version 2 or |
| * later. See the COPYING file in the top-level directory. |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "hw/qdev-core.h" |
| #include "monitor/hmp.h" |
| #include "monitor/monitor.h" |
| #include "qapi/error.h" |
| #include "qapi/qapi-commands-qom.h" |
| #include "qapi/qmp/qdict.h" |
| #include "qapi/qmp/qjson.h" |
| #include "qemu/readline.h" |
| #include "qom/object.h" |
| #include "qom/object_interfaces.h" |
| |
| void hmp_qom_list(Monitor *mon, const QDict *qdict) |
| { |
| const char *path = qdict_get_try_str(qdict, "path"); |
| ObjectPropertyInfoList *list; |
| Error *err = NULL; |
| |
| if (path == NULL) { |
| monitor_printf(mon, "/\n"); |
| return; |
| } |
| |
| list = qmp_qom_list(path, &err); |
| if (err == NULL) { |
| ObjectPropertyInfoList *start = list; |
| while (list != NULL) { |
| ObjectPropertyInfo *value = list->value; |
| |
| monitor_printf(mon, "%s (%s)\n", |
| value->name, value->type); |
| list = list->next; |
| } |
| qapi_free_ObjectPropertyInfoList(start); |
| } |
| hmp_handle_error(mon, err); |
| } |
| |
| void hmp_qom_set(Monitor *mon, const QDict *qdict) |
| { |
| const bool json = qdict_get_try_bool(qdict, "json", false); |
| const char *path = qdict_get_str(qdict, "path"); |
| const char *property = qdict_get_str(qdict, "property"); |
| const char *value = qdict_get_str(qdict, "value"); |
| Error *err = NULL; |
| |
| if (!json) { |
| Object *obj = object_resolve_path(path, NULL); |
| |
| if (!obj) { |
| error_set(&err, ERROR_CLASS_DEVICE_NOT_FOUND, |
| "Device '%s' not found", path); |
| } else { |
| object_property_parse(obj, property, value, &err); |
| } |
| } else { |
| QObject *obj = qobject_from_json(value, &err); |
| |
| if (!err) { |
| qmp_qom_set(path, property, obj, &err); |
| } |
| } |
| |
| hmp_handle_error(mon, err); |
| } |
| |
| void hmp_qom_get(Monitor *mon, const QDict *qdict) |
| { |
| const char *path = qdict_get_str(qdict, "path"); |
| const char *property = qdict_get_str(qdict, "property"); |
| Error *err = NULL; |
| QObject *obj = qmp_qom_get(path, property, &err); |
| |
| if (err == NULL) { |
| GString *str = qobject_to_json_pretty(obj, true); |
| monitor_printf(mon, "%s\n", str->str); |
| g_string_free(str, true); |
| } |
| |
| qobject_unref(obj); |
| hmp_handle_error(mon, err); |
| } |
| |
| typedef struct QOMCompositionState { |
| Monitor *mon; |
| int indent; |
| } QOMCompositionState; |
| |
| static void print_qom_composition(Monitor *mon, Object *obj, int indent); |
| |
| static int qom_composition_compare(const void *a, const void *b) |
| { |
| return g_strcmp0(object_get_canonical_path_component(*(Object **)a), |
| object_get_canonical_path_component(*(Object **)b)); |
| } |
| |
| static int insert_qom_composition_child(Object *obj, void *opaque) |
| { |
| g_array_append_val(opaque, obj); |
| return 0; |
| } |
| |
| static void print_qom_composition(Monitor *mon, Object *obj, int indent) |
| { |
| GArray *children = g_array_new(false, false, sizeof(Object *)); |
| const char *name; |
| int i; |
| |
| if (obj == object_get_root()) { |
| name = ""; |
| } else { |
| name = object_get_canonical_path_component(obj); |
| } |
| monitor_printf(mon, "%*s/%s (%s)\n", indent, "", name, |
| object_get_typename(obj)); |
| |
| object_child_foreach(obj, insert_qom_composition_child, children); |
| g_array_sort(children, qom_composition_compare); |
| |
| for (i = 0; i < children->len; i++) { |
| print_qom_composition(mon, g_array_index(children, Object *, i), |
| indent + 2); |
| } |
| g_array_free(children, TRUE); |
| } |
| |
| void hmp_info_qom_tree(Monitor *mon, const QDict *dict) |
| { |
| const char *path = qdict_get_try_str(dict, "path"); |
| Object *obj; |
| bool ambiguous = false; |
| |
| if (path) { |
| obj = object_resolve_path(path, &ambiguous); |
| if (!obj) { |
| monitor_printf(mon, "Path '%s' could not be resolved.\n", path); |
| return; |
| } |
| if (ambiguous) { |
| monitor_printf(mon, "Warning: Path '%s' is ambiguous.\n", path); |
| return; |
| } |
| } else { |
| obj = qdev_get_machine(); |
| } |
| print_qom_composition(mon, obj, 0); |
| } |
| |
| void hmp_object_add(Monitor *mon, const QDict *qdict) |
| { |
| const char *options = qdict_get_str(qdict, "object"); |
| Error *err = NULL; |
| |
| user_creatable_add_from_str(options, &err); |
| hmp_handle_error(mon, err); |
| } |
| |
| void hmp_object_del(Monitor *mon, const QDict *qdict) |
| { |
| const char *id = qdict_get_str(qdict, "id"); |
| Error *err = NULL; |
| |
| user_creatable_del(id, &err); |
| hmp_handle_error(mon, err); |
| } |
| |
| void object_add_completion(ReadLineState *rs, int nb_args, const char *str) |
| { |
| GSList *list, *elt; |
| size_t len; |
| |
| if (nb_args != 2) { |
| return; |
| } |
| |
| len = strlen(str); |
| readline_set_completion_index(rs, len); |
| list = elt = object_class_get_list(TYPE_USER_CREATABLE, false); |
| while (elt) { |
| const char *name; |
| |
| name = object_class_get_name(OBJECT_CLASS(elt->data)); |
| if (strcmp(name, TYPE_USER_CREATABLE)) { |
| readline_add_completion_of(rs, str, name); |
| } |
| elt = elt->next; |
| } |
| g_slist_free(list); |
| } |
| |
| void object_del_completion(ReadLineState *rs, int nb_args, const char *str) |
| { |
| ObjectPropertyInfoList *list, *start; |
| size_t len; |
| |
| if (nb_args != 2) { |
| return; |
| } |
| len = strlen(str); |
| readline_set_completion_index(rs, len); |
| |
| start = list = qmp_qom_list("/objects", NULL); |
| while (list) { |
| ObjectPropertyInfo *info = list->value; |
| |
| if (!strncmp(info->type, "child<", 5)) { |
| readline_add_completion_of(rs, str, info->name); |
| } |
| list = list->next; |
| } |
| qapi_free_ObjectPropertyInfoList(start); |
| } |