152924deaSMarkus Armbruster /* 252924deaSMarkus Armbruster * QMP commands related to machines and CPUs 352924deaSMarkus Armbruster * 452924deaSMarkus Armbruster * Copyright (C) 2014 Red Hat Inc 552924deaSMarkus Armbruster * 652924deaSMarkus Armbruster * This work is licensed under the terms of the GNU GPL, version 2 or later. 752924deaSMarkus Armbruster * See the COPYING file in the top-level directory. 852924deaSMarkus Armbruster */ 952924deaSMarkus Armbruster 1052924deaSMarkus Armbruster #include "qemu/osdep.h" 1152924deaSMarkus Armbruster #include "hw/boards.h" 1252924deaSMarkus Armbruster #include "qapi/error.h" 1344a17fe0SMarkus Armbruster #include "qapi/qapi-builtin-visit.h" 1452924deaSMarkus Armbruster #include "qapi/qapi-commands-machine.h" 1552924deaSMarkus Armbruster #include "qapi/qmp/qerror.h" 1644a17fe0SMarkus Armbruster #include "qapi/qmp/qobject.h" 1744a17fe0SMarkus Armbruster #include "qapi/qobject-input-visitor.h" 18*1b8ae799SDaniel P. Berrangé #include "qapi/type-helpers.h" 19db725815SMarkus Armbruster #include "qemu/main-loop.h" 2044a17fe0SMarkus Armbruster #include "qom/qom-qobject.h" 2152924deaSMarkus Armbruster #include "sysemu/hostmem.h" 2252924deaSMarkus Armbruster #include "sysemu/hw_accel.h" 2352924deaSMarkus Armbruster #include "sysemu/numa.h" 2454d31236SMarkus Armbruster #include "sysemu/runstate.h" 2552924deaSMarkus Armbruster 2652924deaSMarkus Armbruster static void cpustate_to_cpuinfo_s390(CpuInfoS390 *info, const CPUState *cpu) 2752924deaSMarkus Armbruster { 2852924deaSMarkus Armbruster #ifdef TARGET_S390X 2952924deaSMarkus Armbruster S390CPU *s390_cpu = S390_CPU(cpu); 3052924deaSMarkus Armbruster CPUS390XState *env = &s390_cpu->env; 3152924deaSMarkus Armbruster 3252924deaSMarkus Armbruster info->cpu_state = env->cpu_state; 3352924deaSMarkus Armbruster #else 3452924deaSMarkus Armbruster abort(); 3552924deaSMarkus Armbruster #endif 3652924deaSMarkus Armbruster } 3752924deaSMarkus Armbruster 3852924deaSMarkus Armbruster /* 3952924deaSMarkus Armbruster * fast means: we NEVER interrupt vCPU threads to retrieve 4052924deaSMarkus Armbruster * information from KVM. 4152924deaSMarkus Armbruster */ 4252924deaSMarkus Armbruster CpuInfoFastList *qmp_query_cpus_fast(Error **errp) 4352924deaSMarkus Armbruster { 4452924deaSMarkus Armbruster MachineState *ms = MACHINE(qdev_get_machine()); 4552924deaSMarkus Armbruster MachineClass *mc = MACHINE_GET_CLASS(ms); 4695b3a8c8SEric Blake CpuInfoFastList *head = NULL, **tail = &head; 4752924deaSMarkus Armbruster SysEmuTarget target = qapi_enum_parse(&SysEmuTarget_lookup, TARGET_NAME, 4852924deaSMarkus Armbruster -1, &error_abort); 4952924deaSMarkus Armbruster CPUState *cpu; 5052924deaSMarkus Armbruster 5152924deaSMarkus Armbruster CPU_FOREACH(cpu) { 5295b3a8c8SEric Blake CpuInfoFast *value = g_malloc0(sizeof(*value)); 5352924deaSMarkus Armbruster 5495b3a8c8SEric Blake value->cpu_index = cpu->cpu_index; 5595b3a8c8SEric Blake value->qom_path = object_get_canonical_path(OBJECT(cpu)); 5695b3a8c8SEric Blake value->thread_id = cpu->thread_id; 5752924deaSMarkus Armbruster 5895b3a8c8SEric Blake value->has_props = !!mc->cpu_index_to_instance_props; 5995b3a8c8SEric Blake if (value->has_props) { 6052924deaSMarkus Armbruster CpuInstanceProperties *props; 6152924deaSMarkus Armbruster props = g_malloc0(sizeof(*props)); 6252924deaSMarkus Armbruster *props = mc->cpu_index_to_instance_props(ms, cpu->cpu_index); 6395b3a8c8SEric Blake value->props = props; 6452924deaSMarkus Armbruster } 6552924deaSMarkus Armbruster 6695b3a8c8SEric Blake value->target = target; 6752924deaSMarkus Armbruster if (target == SYS_EMU_TARGET_S390X) { 6895b3a8c8SEric Blake cpustate_to_cpuinfo_s390(&value->u.s390x, cpu); 6952924deaSMarkus Armbruster } 7052924deaSMarkus Armbruster 7195b3a8c8SEric Blake QAPI_LIST_APPEND(tail, value); 7252924deaSMarkus Armbruster } 7352924deaSMarkus Armbruster 7452924deaSMarkus Armbruster return head; 7552924deaSMarkus Armbruster } 7652924deaSMarkus Armbruster 7752924deaSMarkus Armbruster MachineInfoList *qmp_query_machines(Error **errp) 7852924deaSMarkus Armbruster { 7952924deaSMarkus Armbruster GSList *el, *machines = object_class_get_list(TYPE_MACHINE, false); 8052924deaSMarkus Armbruster MachineInfoList *mach_list = NULL; 8152924deaSMarkus Armbruster 8252924deaSMarkus Armbruster for (el = machines; el; el = el->next) { 8352924deaSMarkus Armbruster MachineClass *mc = el->data; 8452924deaSMarkus Armbruster MachineInfo *info; 8552924deaSMarkus Armbruster 8652924deaSMarkus Armbruster info = g_malloc0(sizeof(*info)); 8752924deaSMarkus Armbruster if (mc->is_default) { 8852924deaSMarkus Armbruster info->has_is_default = true; 8952924deaSMarkus Armbruster info->is_default = true; 9052924deaSMarkus Armbruster } 9152924deaSMarkus Armbruster 9252924deaSMarkus Armbruster if (mc->alias) { 9352924deaSMarkus Armbruster info->has_alias = true; 9452924deaSMarkus Armbruster info->alias = g_strdup(mc->alias); 9552924deaSMarkus Armbruster } 9652924deaSMarkus Armbruster 9752924deaSMarkus Armbruster info->name = g_strdup(mc->name); 9852924deaSMarkus Armbruster info->cpu_max = !mc->max_cpus ? 1 : mc->max_cpus; 9952924deaSMarkus Armbruster info->hotpluggable_cpus = mc->has_hotpluggable_cpus; 100cd5ff833SIgor Mammedov info->numa_mem_supported = mc->numa_mem_supported; 10179974027SEduardo Habkost info->deprecated = !!mc->deprecation_reason; 10204109957SDaniel P. Berrangé if (mc->default_cpu_type) { 10304109957SDaniel P. Berrangé info->default_cpu_type = g_strdup(mc->default_cpu_type); 10404109957SDaniel P. Berrangé info->has_default_cpu_type = true; 10504109957SDaniel P. Berrangé } 106c5566005SMichal Privoznik if (mc->default_ram_id) { 107c5566005SMichal Privoznik info->default_ram_id = g_strdup(mc->default_ram_id); 108c5566005SMichal Privoznik info->has_default_ram_id = true; 109c5566005SMichal Privoznik } 11052924deaSMarkus Armbruster 11154aa3de7SEric Blake QAPI_LIST_PREPEND(mach_list, info); 11252924deaSMarkus Armbruster } 11352924deaSMarkus Armbruster 11452924deaSMarkus Armbruster g_slist_free(machines); 11552924deaSMarkus Armbruster return mach_list; 11652924deaSMarkus Armbruster } 11752924deaSMarkus Armbruster 11852924deaSMarkus Armbruster CurrentMachineParams *qmp_query_current_machine(Error **errp) 11952924deaSMarkus Armbruster { 12052924deaSMarkus Armbruster CurrentMachineParams *params = g_malloc0(sizeof(*params)); 12152924deaSMarkus Armbruster params->wakeup_suspend_support = qemu_wakeup_suspend_enabled(); 12252924deaSMarkus Armbruster 12352924deaSMarkus Armbruster return params; 12452924deaSMarkus Armbruster } 12552924deaSMarkus Armbruster 126ffaee83bSMarkus Armbruster TargetInfo *qmp_query_target(Error **errp) 127ffaee83bSMarkus Armbruster { 128ffaee83bSMarkus Armbruster TargetInfo *info = g_malloc0(sizeof(*info)); 129ffaee83bSMarkus Armbruster 130ffaee83bSMarkus Armbruster info->arch = qapi_enum_parse(&SysEmuTarget_lookup, TARGET_NAME, -1, 131ffaee83bSMarkus Armbruster &error_abort); 132ffaee83bSMarkus Armbruster 133ffaee83bSMarkus Armbruster return info; 134ffaee83bSMarkus Armbruster } 135ffaee83bSMarkus Armbruster 13652924deaSMarkus Armbruster HotpluggableCPUList *qmp_query_hotpluggable_cpus(Error **errp) 13752924deaSMarkus Armbruster { 13852924deaSMarkus Armbruster MachineState *ms = MACHINE(qdev_get_machine()); 13952924deaSMarkus Armbruster MachineClass *mc = MACHINE_GET_CLASS(ms); 14052924deaSMarkus Armbruster 14152924deaSMarkus Armbruster if (!mc->has_hotpluggable_cpus) { 14252924deaSMarkus Armbruster error_setg(errp, QERR_FEATURE_DISABLED, "query-hotpluggable-cpus"); 14352924deaSMarkus Armbruster return NULL; 14452924deaSMarkus Armbruster } 14552924deaSMarkus Armbruster 14652924deaSMarkus Armbruster return machine_query_hotpluggable_cpus(ms); 14752924deaSMarkus Armbruster } 14852924deaSMarkus Armbruster 14952924deaSMarkus Armbruster void qmp_set_numa_node(NumaOptions *cmd, Error **errp) 15052924deaSMarkus Armbruster { 1512f181fbdSPaolo Bonzini if (phase_check(PHASE_MACHINE_INITIALIZED)) { 152164dafd1SPaolo Bonzini error_setg(errp, "The command is permitted only before the machine has been created"); 15352924deaSMarkus Armbruster return; 15452924deaSMarkus Armbruster } 15552924deaSMarkus Armbruster 15652924deaSMarkus Armbruster set_numa_options(MACHINE(qdev_get_machine()), cmd, errp); 15752924deaSMarkus Armbruster } 15852924deaSMarkus Armbruster 15952924deaSMarkus Armbruster static int query_memdev(Object *obj, void *opaque) 16052924deaSMarkus Armbruster { 16169647f9dSDavid Hildenbrand Error *err = NULL; 16252924deaSMarkus Armbruster MemdevList **list = opaque; 163240ee8bdSEric Blake Memdev *m; 16444a17fe0SMarkus Armbruster QObject *host_nodes; 16544a17fe0SMarkus Armbruster Visitor *v; 16652924deaSMarkus Armbruster 16752924deaSMarkus Armbruster if (object_dynamic_cast(obj, TYPE_MEMORY_BACKEND)) { 16852924deaSMarkus Armbruster m = g_malloc0(sizeof(*m)); 16952924deaSMarkus Armbruster 170240ee8bdSEric Blake m->id = g_strdup(object_get_canonical_path_component(obj)); 171240ee8bdSEric Blake m->has_id = !!m->id; 17252924deaSMarkus Armbruster 173240ee8bdSEric Blake m->size = object_property_get_uint(obj, "size", &error_abort); 174240ee8bdSEric Blake m->merge = object_property_get_bool(obj, "merge", &error_abort); 175240ee8bdSEric Blake m->dump = object_property_get_bool(obj, "dump", &error_abort); 176240ee8bdSEric Blake m->prealloc = object_property_get_bool(obj, "prealloc", &error_abort); 177d300fc54SDavid Hildenbrand m->share = object_property_get_bool(obj, "share", &error_abort); 17869647f9dSDavid Hildenbrand m->reserve = object_property_get_bool(obj, "reserve", &err); 17969647f9dSDavid Hildenbrand if (err) { 18069647f9dSDavid Hildenbrand error_free_or_abort(&err); 18169647f9dSDavid Hildenbrand } else { 18269647f9dSDavid Hildenbrand m->has_reserve = true; 18369647f9dSDavid Hildenbrand } 184240ee8bdSEric Blake m->policy = object_property_get_enum(obj, "policy", "HostMemPolicy", 18552924deaSMarkus Armbruster &error_abort); 18644a17fe0SMarkus Armbruster host_nodes = object_property_get_qobject(obj, 18744a17fe0SMarkus Armbruster "host-nodes", 18852924deaSMarkus Armbruster &error_abort); 18944a17fe0SMarkus Armbruster v = qobject_input_visitor_new(host_nodes); 190240ee8bdSEric Blake visit_type_uint16List(v, NULL, &m->host_nodes, &error_abort); 19144a17fe0SMarkus Armbruster visit_free(v); 19244a17fe0SMarkus Armbruster qobject_unref(host_nodes); 19352924deaSMarkus Armbruster 194240ee8bdSEric Blake QAPI_LIST_PREPEND(*list, m); 19552924deaSMarkus Armbruster } 19652924deaSMarkus Armbruster 19752924deaSMarkus Armbruster return 0; 19852924deaSMarkus Armbruster } 19952924deaSMarkus Armbruster 20052924deaSMarkus Armbruster MemdevList *qmp_query_memdev(Error **errp) 20152924deaSMarkus Armbruster { 20252924deaSMarkus Armbruster Object *obj = object_get_objects_root(); 20352924deaSMarkus Armbruster MemdevList *list = NULL; 20452924deaSMarkus Armbruster 20552924deaSMarkus Armbruster object_child_foreach(obj, query_memdev, &list); 20652924deaSMarkus Armbruster return list; 20752924deaSMarkus Armbruster } 208*1b8ae799SDaniel P. Berrangé 209*1b8ae799SDaniel P. Berrangé HumanReadableText *qmp_x_query_numa(Error **errp) 210*1b8ae799SDaniel P. Berrangé { 211*1b8ae799SDaniel P. Berrangé g_autoptr(GString) buf = g_string_new(""); 212*1b8ae799SDaniel P. Berrangé int i, nb_numa_nodes; 213*1b8ae799SDaniel P. Berrangé NumaNodeMem *node_mem; 214*1b8ae799SDaniel P. Berrangé CpuInfoFastList *cpu_list, *cpu; 215*1b8ae799SDaniel P. Berrangé MachineState *ms = MACHINE(qdev_get_machine()); 216*1b8ae799SDaniel P. Berrangé 217*1b8ae799SDaniel P. Berrangé nb_numa_nodes = ms->numa_state ? ms->numa_state->num_nodes : 0; 218*1b8ae799SDaniel P. Berrangé g_string_append_printf(buf, "%d nodes\n", nb_numa_nodes); 219*1b8ae799SDaniel P. Berrangé if (!nb_numa_nodes) { 220*1b8ae799SDaniel P. Berrangé goto done; 221*1b8ae799SDaniel P. Berrangé } 222*1b8ae799SDaniel P. Berrangé 223*1b8ae799SDaniel P. Berrangé cpu_list = qmp_query_cpus_fast(&error_abort); 224*1b8ae799SDaniel P. Berrangé node_mem = g_new0(NumaNodeMem, nb_numa_nodes); 225*1b8ae799SDaniel P. Berrangé 226*1b8ae799SDaniel P. Berrangé query_numa_node_mem(node_mem, ms); 227*1b8ae799SDaniel P. Berrangé for (i = 0; i < nb_numa_nodes; i++) { 228*1b8ae799SDaniel P. Berrangé g_string_append_printf(buf, "node %d cpus:", i); 229*1b8ae799SDaniel P. Berrangé for (cpu = cpu_list; cpu; cpu = cpu->next) { 230*1b8ae799SDaniel P. Berrangé if (cpu->value->has_props && cpu->value->props->has_node_id && 231*1b8ae799SDaniel P. Berrangé cpu->value->props->node_id == i) { 232*1b8ae799SDaniel P. Berrangé g_string_append_printf(buf, " %" PRIi64, cpu->value->cpu_index); 233*1b8ae799SDaniel P. Berrangé } 234*1b8ae799SDaniel P. Berrangé } 235*1b8ae799SDaniel P. Berrangé g_string_append_printf(buf, "\n"); 236*1b8ae799SDaniel P. Berrangé g_string_append_printf(buf, "node %d size: %" PRId64 " MB\n", i, 237*1b8ae799SDaniel P. Berrangé node_mem[i].node_mem >> 20); 238*1b8ae799SDaniel P. Berrangé g_string_append_printf(buf, "node %d plugged: %" PRId64 " MB\n", i, 239*1b8ae799SDaniel P. Berrangé node_mem[i].node_plugged_mem >> 20); 240*1b8ae799SDaniel P. Berrangé } 241*1b8ae799SDaniel P. Berrangé qapi_free_CpuInfoFastList(cpu_list); 242*1b8ae799SDaniel P. Berrangé g_free(node_mem); 243*1b8ae799SDaniel P. Berrangé 244*1b8ae799SDaniel P. Berrangé done: 245*1b8ae799SDaniel P. Berrangé return human_readable_text_from_str(buf); 246*1b8ae799SDaniel P. Berrangé } 247