| /* |
| * 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 "qom/object.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); |
| } |