mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-26 20:31:42 +00:00 
			
		
		
		
	 1b8ae799d8
			
		
	
	
		1b8ae799d8
		
	
	
	
	
		
			
			This is a counterpart to the HMP "info numa" command. It is being added with an "x-" prefix because this QMP command is intended as an adhoc debugging tool and will thus not be modelled in QAPI as fully structured data, nor will it have long term guaranteed stability. The existing HMP command is rewritten to call the QMP command. Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com> Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
		
			
				
	
	
		
			247 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * QMP commands related to machines and CPUs
 | |
|  *
 | |
|  * Copyright (C) 2014 Red Hat Inc
 | |
|  *
 | |
|  * 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/boards.h"
 | |
| #include "qapi/error.h"
 | |
| #include "qapi/qapi-builtin-visit.h"
 | |
| #include "qapi/qapi-commands-machine.h"
 | |
| #include "qapi/qmp/qerror.h"
 | |
| #include "qapi/qmp/qobject.h"
 | |
| #include "qapi/qobject-input-visitor.h"
 | |
| #include "qapi/type-helpers.h"
 | |
| #include "qemu/main-loop.h"
 | |
| #include "qom/qom-qobject.h"
 | |
| #include "sysemu/hostmem.h"
 | |
| #include "sysemu/hw_accel.h"
 | |
| #include "sysemu/numa.h"
 | |
| #include "sysemu/runstate.h"
 | |
| 
 | |
| static void cpustate_to_cpuinfo_s390(CpuInfoS390 *info, const CPUState *cpu)
 | |
| {
 | |
| #ifdef TARGET_S390X
 | |
|     S390CPU *s390_cpu = S390_CPU(cpu);
 | |
|     CPUS390XState *env = &s390_cpu->env;
 | |
| 
 | |
|     info->cpu_state = env->cpu_state;
 | |
| #else
 | |
|     abort();
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * fast means: we NEVER interrupt vCPU threads to retrieve
 | |
|  * information from KVM.
 | |
|  */
 | |
| CpuInfoFastList *qmp_query_cpus_fast(Error **errp)
 | |
| {
 | |
|     MachineState *ms = MACHINE(qdev_get_machine());
 | |
|     MachineClass *mc = MACHINE_GET_CLASS(ms);
 | |
|     CpuInfoFastList *head = NULL, **tail = &head;
 | |
|     SysEmuTarget target = qapi_enum_parse(&SysEmuTarget_lookup, TARGET_NAME,
 | |
|                                           -1, &error_abort);
 | |
|     CPUState *cpu;
 | |
| 
 | |
|     CPU_FOREACH(cpu) {
 | |
|         CpuInfoFast *value = g_malloc0(sizeof(*value));
 | |
| 
 | |
|         value->cpu_index = cpu->cpu_index;
 | |
|         value->qom_path = object_get_canonical_path(OBJECT(cpu));
 | |
|         value->thread_id = cpu->thread_id;
 | |
| 
 | |
|         value->has_props = !!mc->cpu_index_to_instance_props;
 | |
|         if (value->has_props) {
 | |
|             CpuInstanceProperties *props;
 | |
|             props = g_malloc0(sizeof(*props));
 | |
|             *props = mc->cpu_index_to_instance_props(ms, cpu->cpu_index);
 | |
|             value->props = props;
 | |
|         }
 | |
| 
 | |
|         value->target = target;
 | |
|         if (target == SYS_EMU_TARGET_S390X) {
 | |
|             cpustate_to_cpuinfo_s390(&value->u.s390x, cpu);
 | |
|         }
 | |
| 
 | |
|         QAPI_LIST_APPEND(tail, value);
 | |
|     }
 | |
| 
 | |
|     return head;
 | |
| }
 | |
| 
 | |
| MachineInfoList *qmp_query_machines(Error **errp)
 | |
| {
 | |
|     GSList *el, *machines = object_class_get_list(TYPE_MACHINE, false);
 | |
|     MachineInfoList *mach_list = NULL;
 | |
| 
 | |
|     for (el = machines; el; el = el->next) {
 | |
|         MachineClass *mc = el->data;
 | |
|         MachineInfo *info;
 | |
| 
 | |
|         info = g_malloc0(sizeof(*info));
 | |
|         if (mc->is_default) {
 | |
|             info->has_is_default = true;
 | |
|             info->is_default = true;
 | |
|         }
 | |
| 
 | |
|         if (mc->alias) {
 | |
|             info->has_alias = true;
 | |
|             info->alias = g_strdup(mc->alias);
 | |
|         }
 | |
| 
 | |
|         info->name = g_strdup(mc->name);
 | |
|         info->cpu_max = !mc->max_cpus ? 1 : mc->max_cpus;
 | |
|         info->hotpluggable_cpus = mc->has_hotpluggable_cpus;
 | |
|         info->numa_mem_supported = mc->numa_mem_supported;
 | |
|         info->deprecated = !!mc->deprecation_reason;
 | |
|         if (mc->default_cpu_type) {
 | |
|             info->default_cpu_type = g_strdup(mc->default_cpu_type);
 | |
|             info->has_default_cpu_type = true;
 | |
|         }
 | |
|         if (mc->default_ram_id) {
 | |
|             info->default_ram_id = g_strdup(mc->default_ram_id);
 | |
|             info->has_default_ram_id = true;
 | |
|         }
 | |
| 
 | |
|         QAPI_LIST_PREPEND(mach_list, info);
 | |
|     }
 | |
| 
 | |
|     g_slist_free(machines);
 | |
|     return mach_list;
 | |
| }
 | |
| 
 | |
| CurrentMachineParams *qmp_query_current_machine(Error **errp)
 | |
| {
 | |
|     CurrentMachineParams *params = g_malloc0(sizeof(*params));
 | |
|     params->wakeup_suspend_support = qemu_wakeup_suspend_enabled();
 | |
| 
 | |
|     return params;
 | |
| }
 | |
| 
 | |
| TargetInfo *qmp_query_target(Error **errp)
 | |
| {
 | |
|     TargetInfo *info = g_malloc0(sizeof(*info));
 | |
| 
 | |
|     info->arch = qapi_enum_parse(&SysEmuTarget_lookup, TARGET_NAME, -1,
 | |
|                                  &error_abort);
 | |
| 
 | |
|     return info;
 | |
| }
 | |
| 
 | |
| HotpluggableCPUList *qmp_query_hotpluggable_cpus(Error **errp)
 | |
| {
 | |
|     MachineState *ms = MACHINE(qdev_get_machine());
 | |
|     MachineClass *mc = MACHINE_GET_CLASS(ms);
 | |
| 
 | |
|     if (!mc->has_hotpluggable_cpus) {
 | |
|         error_setg(errp, QERR_FEATURE_DISABLED, "query-hotpluggable-cpus");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return machine_query_hotpluggable_cpus(ms);
 | |
| }
 | |
| 
 | |
| void qmp_set_numa_node(NumaOptions *cmd, Error **errp)
 | |
| {
 | |
|     if (phase_check(PHASE_MACHINE_INITIALIZED)) {
 | |
|         error_setg(errp, "The command is permitted only before the machine has been created");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     set_numa_options(MACHINE(qdev_get_machine()), cmd, errp);
 | |
| }
 | |
| 
 | |
| static int query_memdev(Object *obj, void *opaque)
 | |
| {
 | |
|     Error *err = NULL;
 | |
|     MemdevList **list = opaque;
 | |
|     Memdev *m;
 | |
|     QObject *host_nodes;
 | |
|     Visitor *v;
 | |
| 
 | |
|     if (object_dynamic_cast(obj, TYPE_MEMORY_BACKEND)) {
 | |
|         m = g_malloc0(sizeof(*m));
 | |
| 
 | |
|         m->id = g_strdup(object_get_canonical_path_component(obj));
 | |
|         m->has_id = !!m->id;
 | |
| 
 | |
|         m->size = object_property_get_uint(obj, "size", &error_abort);
 | |
|         m->merge = object_property_get_bool(obj, "merge", &error_abort);
 | |
|         m->dump = object_property_get_bool(obj, "dump", &error_abort);
 | |
|         m->prealloc = object_property_get_bool(obj, "prealloc", &error_abort);
 | |
|         m->share = object_property_get_bool(obj, "share", &error_abort);
 | |
|         m->reserve = object_property_get_bool(obj, "reserve", &err);
 | |
|         if (err) {
 | |
|             error_free_or_abort(&err);
 | |
|         } else {
 | |
|             m->has_reserve = true;
 | |
|         }
 | |
|         m->policy = object_property_get_enum(obj, "policy", "HostMemPolicy",
 | |
|                                              &error_abort);
 | |
|         host_nodes = object_property_get_qobject(obj,
 | |
|                                                  "host-nodes",
 | |
|                                                  &error_abort);
 | |
|         v = qobject_input_visitor_new(host_nodes);
 | |
|         visit_type_uint16List(v, NULL, &m->host_nodes, &error_abort);
 | |
|         visit_free(v);
 | |
|         qobject_unref(host_nodes);
 | |
| 
 | |
|         QAPI_LIST_PREPEND(*list, m);
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| MemdevList *qmp_query_memdev(Error **errp)
 | |
| {
 | |
|     Object *obj = object_get_objects_root();
 | |
|     MemdevList *list = NULL;
 | |
| 
 | |
|     object_child_foreach(obj, query_memdev, &list);
 | |
|     return list;
 | |
| }
 | |
| 
 | |
| HumanReadableText *qmp_x_query_numa(Error **errp)
 | |
| {
 | |
|     g_autoptr(GString) buf = g_string_new("");
 | |
|     int i, nb_numa_nodes;
 | |
|     NumaNodeMem *node_mem;
 | |
|     CpuInfoFastList *cpu_list, *cpu;
 | |
|     MachineState *ms = MACHINE(qdev_get_machine());
 | |
| 
 | |
|     nb_numa_nodes = ms->numa_state ? ms->numa_state->num_nodes : 0;
 | |
|     g_string_append_printf(buf, "%d nodes\n", nb_numa_nodes);
 | |
|     if (!nb_numa_nodes) {
 | |
|         goto done;
 | |
|     }
 | |
| 
 | |
|     cpu_list = qmp_query_cpus_fast(&error_abort);
 | |
|     node_mem = g_new0(NumaNodeMem, nb_numa_nodes);
 | |
| 
 | |
|     query_numa_node_mem(node_mem, ms);
 | |
|     for (i = 0; i < nb_numa_nodes; i++) {
 | |
|         g_string_append_printf(buf, "node %d cpus:", i);
 | |
|         for (cpu = cpu_list; cpu; cpu = cpu->next) {
 | |
|             if (cpu->value->has_props && cpu->value->props->has_node_id &&
 | |
|                 cpu->value->props->node_id == i) {
 | |
|                 g_string_append_printf(buf, " %" PRIi64, cpu->value->cpu_index);
 | |
|             }
 | |
|         }
 | |
|         g_string_append_printf(buf, "\n");
 | |
|         g_string_append_printf(buf, "node %d size: %" PRId64 " MB\n", i,
 | |
|                                node_mem[i].node_mem >> 20);
 | |
|         g_string_append_printf(buf, "node %d plugged: %" PRId64 " MB\n", i,
 | |
|                                node_mem[i].node_plugged_mem >> 20);
 | |
|     }
 | |
|     qapi_free_CpuInfoFastList(cpu_list);
 | |
|     g_free(node_mem);
 | |
| 
 | |
|  done:
 | |
|     return human_readable_text_from_str(buf);
 | |
| }
 |