/* * virsh-host.c: Commands in "Host and Hypervisor" group. * * Copyright (C) 2005, 2007-2014 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . */ #include #include "virsh-host.h" #include "virsh.h" #include #include #include "internal.h" #include "virbitmap.h" #include "viralloc.h" #include "virxml.h" #include "virstring.h" #include "virfile.h" #include "virenum.h" #include "virsh-util.h" /* * "capabilities" command */ static const vshCmdInfo info_capabilities = { .help = N_("capabilities"), .desc = N_("Returns capabilities of hypervisor/driver."), }; static const vshCmdOptDef opts_capabilities[] = { {.name = "xpath", .type = VSH_OT_STRING, .flags = VSH_OFLAG_REQ_OPT, .completer = virshCompleteEmpty, .help = N_("xpath expression to filter the XML document") }, {.name = "wrap", .type = VSH_OT_BOOL, .help = N_("wrap xpath results in an common root element"), }, {.name = NULL} }; static bool cmdCapabilities(vshControl *ctl, const vshCmd *cmd) { g_autofree char *caps = NULL; virshControl *priv = ctl->privData; bool wrap = vshCommandOptBool(cmd, "wrap"); const char *xpath = NULL; if (vshCommandOptStringQuiet(ctl, cmd, "xpath", &xpath) < 0) return false; if ((caps = virConnectGetCapabilities(priv->conn)) == NULL) { vshError(ctl, "%s", _("failed to get capabilities")); return false; } return virshDumpXML(ctl, caps, "capabilities", xpath, wrap); } /* * "domcapabilities" command */ static const vshCmdInfo info_domcapabilities = { .help = N_("domain capabilities"), .desc = N_("Returns capabilities of emulator with respect to host and libvirt."), }; static const vshCmdOptDef opts_domcapabilities[] = { {.name = "virttype", .type = VSH_OT_STRING, .completer = virshDomainVirtTypeCompleter, .help = N_("virtualization type (/domain/@type)"), }, {.name = "emulatorbin", .type = VSH_OT_STRING, .help = N_("path to emulator binary (/domain/devices/emulator)"), }, {.name = "arch", .type = VSH_OT_STRING, .completer = virshArchCompleter, .help = N_("domain architecture (/domain/os/type/@arch)"), }, {.name = "machine", .type = VSH_OT_STRING, .help = N_("machine type (/domain/os/type/@machine)"), }, {.name = "xpath", .type = VSH_OT_STRING, .flags = VSH_OFLAG_REQ_OPT, .completer = virshCompleteEmpty, .help = N_("xpath expression to filter the XML document") }, {.name = "wrap", .type = VSH_OT_BOOL, .help = N_("wrap xpath results in an common root element"), }, {.name = NULL} }; static bool cmdDomCapabilities(vshControl *ctl, const vshCmd *cmd) { g_autofree char *caps = NULL; const char *virttype = NULL; const char *emulatorbin = NULL; const char *arch = NULL; const char *machine = NULL; const char *xpath = NULL; const unsigned int flags = 0; /* No flags so far */ bool wrap = vshCommandOptBool(cmd, "wrap"); virshControl *priv = ctl->privData; if (vshCommandOptStringReq(ctl, cmd, "virttype", &virttype) < 0 || vshCommandOptStringReq(ctl, cmd, "emulatorbin", &emulatorbin) < 0 || vshCommandOptStringReq(ctl, cmd, "arch", &arch) < 0 || vshCommandOptStringReq(ctl, cmd, "machine", &machine) < 0 || vshCommandOptStringQuiet(ctl, cmd, "xpath", &xpath) < 0) return false; caps = virConnectGetDomainCapabilities(priv->conn, emulatorbin, arch, machine, virttype, flags); if (!caps) { vshError(ctl, "%s", _("failed to get emulator capabilities")); return false; } return virshDumpXML(ctl, caps, "domcapabilities", xpath, wrap); } /* * "freecell" command */ static const vshCmdInfo info_freecell = { .help = N_("NUMA free memory"), .desc = N_("display available free memory for the NUMA cell."), }; static const vshCmdOptDef opts_freecell[] = { {.name = "cellno", .type = VSH_OT_INT, .completer = virshCellnoCompleter, .help = N_("NUMA cell number") }, {.name = "all", .type = VSH_OT_BOOL, .help = N_("show free memory for all NUMA cells") }, {.name = NULL} }; static bool cmdFreecell(vshControl *ctl, const vshCmd *cmd) { int cell = -1; unsigned long long memory = 0; g_autofree xmlNodePtr *nodes = NULL; unsigned long nodes_cnt; g_autofree unsigned long *nodes_id = NULL; g_autofree unsigned long long *nodes_free = NULL; bool all = vshCommandOptBool(cmd, "all"); bool cellno = vshCommandOptBool(cmd, "cellno"); size_t i; g_autofree char *cap_xml = NULL; g_autoptr(xmlXPathContext) ctxt = NULL; virshControl *priv = ctl->privData; VSH_EXCLUSIVE_OPTIONS_VAR(all, cellno); if (cellno && vshCommandOptInt(ctl, cmd, "cellno", &cell) < 0) return false; if (!all) { if (cellno) { if (virNodeGetCellsFreeMemory(priv->conn, &memory, cell, 1) != 1) return false; vshPrint(ctl, "%d: %llu KiB\n", cell, (memory/1024)); return true; } if ((memory = virNodeGetFreeMemory(priv->conn)) == 0) return false; vshPrint(ctl, "%s: %llu KiB\n", _("Total"), (memory/1024)); return true; } if (!(cap_xml = virConnectGetCapabilities(priv->conn))) { vshError(ctl, "%s", _("unable to get node capabilities")); return false; } if (!virXMLParseStringCtxt(cap_xml, _("(capabilities)"), &ctxt)) { vshError(ctl, "%s", _("unable to get node capabilities")); return false; } nodes_cnt = virXPathNodeSet("/capabilities/host/topology/cells/cell", ctxt, &nodes); if (nodes_cnt == -1) { vshError(ctl, "%s", _("could not get information about NUMA topology")); return false; } nodes_free = g_new0(unsigned long long, nodes_cnt); nodes_id = g_new0(unsigned long, nodes_cnt); for (i = 0; i < nodes_cnt; i++) { unsigned long id; g_autofree char *val = virXMLPropString(nodes[i], "id"); if (virStrToLong_ulp(val, NULL, 10, &id)) { vshError(ctl, "%s", _("conversion from string failed")); return false; } nodes_id[i] = id; if (virNodeGetCellsFreeMemory(priv->conn, &(nodes_free[i]), id, 1) != 1) { vshError(ctl, _("failed to get free memory for NUMA node number: %1$lu"), id); return false; } } for (cell = 0; cell < nodes_cnt; cell++) { vshPrint(ctl, "%5lu: %10llu KiB\n", nodes_id[cell], (nodes_free[cell]/1024)); memory += nodes_free[cell]; } vshPrintExtra(ctl, "--------------------\n"); vshPrintExtra(ctl, "%5s: %10llu KiB\n", _("Total"), memory/1024); return true; } /* * "freepages" command */ static const vshCmdInfo info_freepages = { .help = N_("NUMA free pages"), .desc = N_("display available free pages for the NUMA cell."), }; static const vshCmdOptDef opts_freepages[] = { {.name = "cellno", .type = VSH_OT_INT, .completer = virshCellnoCompleter, .help = N_("NUMA cell number") }, {.name = "pagesize", .type = VSH_OT_INT, .completer = virshAllocpagesPagesizeCompleter, .help = N_("page size (in kibibytes)") }, {.name = "all", .type = VSH_OT_BOOL, .help = N_("show free pages for all NUMA cells") }, {.name = NULL} }; static int vshPageSizeSorter(const void *a, const void *b, void *opaque G_GNUC_UNUSED) { unsigned int pa = *(unsigned int *)a; unsigned int pb = *(unsigned int *)b; return pa - pb; } static bool cmdFreepages(vshControl *ctl, const vshCmd *cmd) { bool ret = false; unsigned int npages; g_autofree unsigned int *pagesize = NULL; unsigned long long bytes = 0; unsigned int kibibytes = 0; int cell; g_autofree unsigned long long *counts = NULL; size_t i, j; xmlNodePtr *nodes = NULL; int nodes_cnt; g_autofree char *cap_xml = NULL; g_autoptr(xmlDoc) doc = NULL; g_autoptr(xmlXPathContext) ctxt = NULL; bool all = vshCommandOptBool(cmd, "all"); bool cellno = vshCommandOptBool(cmd, "cellno"); bool pagesz = vshCommandOptBool(cmd, "pagesize"); virshControl *priv = ctl->privData; bool pagesize_missing = false; int rv = -1; VSH_EXCLUSIVE_OPTIONS_VAR(all, cellno); if (vshCommandOptScaledInt(ctl, cmd, "pagesize", &bytes, 1024, UINT_MAX * 1024ULL) < 0) { goto cleanup; } kibibytes = VIR_DIV_UP(bytes, 1024); if (all) { if (!(cap_xml = virConnectGetCapabilities(priv->conn))) { vshError(ctl, "%s", _("unable to get node capabilities")); goto cleanup; } if (!(doc = virXMLParseStringCtxt(cap_xml, _("capabilities"), &ctxt))) { vshError(ctl, "%s", _("unable to parse node capabilities")); goto cleanup; } if (!pagesz) { nodes_cnt = virXPathNodeSet("/capabilities/host/cpu/pages", ctxt, &nodes); if (nodes_cnt <= 0) { /* Some drivers don't export page sizes under the * XPath above. Do another trick to get them. */ nodes_cnt = virXPathNodeSet("/capabilities/host/topology/cells/cell/pages", ctxt, &nodes); if (nodes_cnt <= 0) { vshError(ctl, "%s", _("could not get information about supported page sizes")); goto cleanup; } } pagesize = g_new0(unsigned int, nodes_cnt); for (i = 0; i < nodes_cnt; i++) { g_autofree char *val = virXMLPropString(nodes[i], "size"); if (virStrToLong_uip(val, NULL, 10, &pagesize[i]) < 0) { vshError(ctl, _("unable to parse page size: %1$s"), val); goto cleanup; } } /* Here, if we've done the trick few lines above, * @pagesize array will contain duplicates. We should * remove them otherwise not very nice output will be * produced. */ g_qsort_with_data(pagesize, nodes_cnt, sizeof(*pagesize), vshPageSizeSorter, NULL); for (i = 0; i < nodes_cnt - 1;) { if (pagesize[i] == pagesize[i + 1]) { memmove(pagesize + i, pagesize + i + 1, (nodes_cnt - i + 1) * sizeof(*pagesize)); nodes_cnt--; } else { i++; } } npages = nodes_cnt; VIR_FREE(nodes); } else { pagesize = g_new0(unsigned int, 1); pagesize[0] = kibibytes; npages = 1; } counts = g_new0(unsigned long long, npages); nodes_cnt = virXPathNodeSet("/capabilities/host/topology/cells/cell", ctxt, &nodes); for (i = 0; i < nodes_cnt; i++) { g_autofree char *val = virXMLPropString(nodes[i], "id"); if (virStrToLong_i(val, NULL, 10, &cell) < 0) { vshError(ctl, _("unable to parse numa node id: %1$s"), val); goto cleanup; } rv = virNodeGetFreePages(priv->conn, npages, pagesize, cell, 1, counts, 0); if (rv < 0) goto cleanup; if (rv < npages) { pagesize_missing = true; vshError(ctl, _("Did not get all free page data for node %1$d"), cell); continue; } vshPrint(ctl, _("Node %1$d:\n"), cell); for (j = 0; j < npages; j++) vshPrint(ctl, "%uKiB: %lld\n", pagesize[j], counts[j]); vshPrint(ctl, "%c", '\n'); } } else { if (!cellno) { vshError(ctl, "%s", _("missing cellno argument")); goto cleanup; } if (vshCommandOptInt(ctl, cmd, "cellno", &cell) < 0) goto cleanup; if (cell < -1) { vshError(ctl, "%s", _("cell number must be non-negative integer or -1")); goto cleanup; } if (!pagesz) { vshError(ctl, "%s", _("missing pagesize argument")); goto cleanup; } /* page size is expected in kibibytes */ pagesize = g_new0(unsigned int, 1); pagesize[0] = kibibytes; counts = g_new0(unsigned long long, 1); rv = virNodeGetFreePages(priv->conn, 1, pagesize, cell, 1, counts, 0); if (rv < 0) goto cleanup; if (rv == 0) { vshError(ctl, "Could not get count of free %uKiB pages, no data returned", *pagesize); goto cleanup; } vshPrint(ctl, "%uKiB: %lld\n", *pagesize, counts[0]); } ret = !pagesize_missing; cleanup: VIR_FREE(nodes); return ret; } /* * "allocpages" command */ static const vshCmdInfo info_allocpages = { .help = N_("Manipulate pages pool size"), .desc = N_("Allocate or free some pages in the pool for NUMA cell."), }; static const vshCmdOptDef opts_allocpages[] = { {.name = "pagesize", .type = VSH_OT_INT, .positional = true, .required = true, .flags = VSH_OFLAG_REQ, .completer = virshAllocpagesPagesizeCompleter, .help = N_("page size (in kibibytes)") }, {.name = "pagecount", .type = VSH_OT_INT, .positional = true, .required = true, .flags = VSH_OFLAG_REQ, .help = N_("page count") }, {.name = "cellno", .type = VSH_OT_INT, .completer = virshCellnoCompleter, .help = N_("NUMA cell number") }, {.name = "add", .type = VSH_OT_BOOL, .help = N_("instead of setting new pool size add pages to it") }, {.name = "all", .type = VSH_OT_BOOL, .help = N_("set on all NUMA cells") }, {.name = NULL} }; static bool cmdAllocpages(vshControl *ctl, const vshCmd *cmd) { bool add = vshCommandOptBool(cmd, "add"); bool all = vshCommandOptBool(cmd, "all"); bool cellno = vshCommandOptBool(cmd, "cellno"); int startCell = -1; int cellCount = 1; unsigned int pageSizes[1]; unsigned long long pageCounts[1], tmp; unsigned int flags = 0; g_autofree char *cap_xml = NULL; g_autoptr(xmlDoc) xml = NULL; g_autoptr(xmlXPathContext) ctxt = NULL; g_autofree xmlNodePtr *nodes = NULL; virshControl *priv = ctl->privData; VSH_EXCLUSIVE_OPTIONS_VAR(all, cellno); if (cellno && vshCommandOptInt(ctl, cmd, "cellno", &startCell) < 0) return false; if (vshCommandOptScaledInt(ctl, cmd, "pagesize", &tmp, 1024, UINT_MAX * 1024ULL) < 0) return false; pageSizes[0] = VIR_DIV_UP(tmp, 1024); if (vshCommandOptULongLong(ctl, cmd, "pagecount", &pageCounts[0]) < 0) return false; flags |= add ? VIR_NODE_ALLOC_PAGES_ADD : VIR_NODE_ALLOC_PAGES_SET; if (all) { unsigned long nodes_cnt; size_t i; if (!(cap_xml = virConnectGetCapabilities(priv->conn))) { vshError(ctl, "%s", _("unable to get node capabilities")); return false; } xml = virXMLParseStringCtxt(cap_xml, _("(capabilities)"), &ctxt); if (!xml) { vshError(ctl, "%s", _("unable to get node capabilities")); return false; } nodes_cnt = virXPathNodeSet("/capabilities/host/topology/cells/cell", ctxt, &nodes); if (nodes_cnt == -1) { vshError(ctl, "%s", _("could not get information about NUMA topology")); return false; } for (i = 0; i < nodes_cnt; i++) { unsigned long id; g_autofree char *val = virXMLPropString(nodes[i], "id"); if (virStrToLong_ulp(val, NULL, 10, &id)) { vshError(ctl, "%s", _("conversion from string failed")); return false; } if (virNodeAllocPages(priv->conn, 1, pageSizes, pageCounts, id, 1, flags) < 0) return false; } } else { if (virNodeAllocPages(priv->conn, 1, pageSizes, pageCounts, startCell, cellCount, flags) < 0) return false; } return true; } /* * "maxvcpus" command */ static const vshCmdInfo info_maxvcpus = { .help = N_("connection vcpu maximum"), .desc = N_("Show maximum number of virtual CPUs for guests on this connection."), }; static const vshCmdOptDef opts_maxvcpus[] = { {.name = "type", .type = VSH_OT_STRING, .completer = virshDomainVirtTypeCompleter, .help = N_("domain type") }, {.name = NULL} }; static bool cmdMaxvcpus(vshControl *ctl, const vshCmd *cmd) { const char *type = NULL; int vcpus = -1; g_autofree char *caps = NULL; g_autoptr(xmlDoc) xml = NULL; g_autoptr(xmlXPathContext) ctxt = NULL; virshControl *priv = ctl->privData; if (vshCommandOptStringReq(ctl, cmd, "type", &type) < 0) return false; if ((caps = virConnectGetDomainCapabilities(priv->conn, NULL, NULL, NULL, type, 0))) { if (!(xml = virXMLParseStringCtxt(caps, _("(domainCapabilities)"), &ctxt))) return false; ignore_value(virXPathInt("string(./vcpu[1]/@max)", ctxt, &vcpus)); } else { vshResetLibvirtError(); } if (vcpus < 0 && (vcpus = virConnectGetMaxVcpus(priv->conn, type)) < 0) return false; vshPrint(ctl, "%d\n", vcpus); return true; } /* * "nodeinfo" command */ static const vshCmdInfo info_nodeinfo = { .help = N_("node information"), .desc = N_("Returns basic information about the node."), }; static bool cmdNodeinfo(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { virNodeInfo info; virshControl *priv = ctl->privData; if (virNodeGetInfo(priv->conn, &info) < 0) { vshError(ctl, "%s", _("failed to get node information")); return false; } vshPrint(ctl, "%-20s %s\n", _("CPU model:"), info.model); vshPrint(ctl, "%-20s %d\n", _("CPU(s):"), info.cpus); if (info.mhz) vshPrint(ctl, "%-20s %d MHz\n", _("CPU frequency:"), info.mhz); vshPrint(ctl, "%-20s %d\n", _("CPU socket(s):"), info.sockets); vshPrint(ctl, "%-20s %d\n", _("Core(s) per socket:"), info.cores); vshPrint(ctl, "%-20s %d\n", _("Thread(s) per core:"), info.threads); vshPrint(ctl, "%-20s %d\n", _("NUMA cell(s):"), info.nodes); vshPrint(ctl, "%-20s %lu KiB\n", _("Memory size:"), info.memory); return true; } /* * "nodecpumap" command */ static const vshCmdInfo info_node_cpumap = { .help = N_("node cpu map"), .desc = N_("Displays the node's total number of CPUs, the number of" " online CPUs and the list of online CPUs."), }; static const vshCmdOptDef opts_node_cpumap[] = { {.name = "pretty", .type = VSH_OT_BOOL, .help = N_("return human readable output") }, {.name = NULL} }; static bool cmdNodeCpuMap(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { int cpu, cpunum; g_autofree unsigned char *cpumap = NULL; unsigned int online; bool pretty = vshCommandOptBool(cmd, "pretty"); virshControl *priv = ctl->privData; cpunum = virNodeGetCPUMap(priv->conn, &cpumap, &online, 0); if (cpunum < 0) { vshError(ctl, "%s", _("Unable to get cpu map")); return false; } vshPrint(ctl, "%-15s %d\n", _("CPUs present:"), cpunum); vshPrint(ctl, "%-15s %d\n", _("CPUs online:"), online); vshPrint(ctl, "%-15s ", _("CPU map:")); if (pretty) { g_autofree char *str = virBitmapDataFormat(cpumap, VIR_CPU_MAPLEN(cpunum)); if (!str) return false; vshPrint(ctl, "%s", str); } else { for (cpu = 0; cpu < cpunum; cpu++) vshPrint(ctl, "%c", VIR_CPU_USED(cpumap, cpu) ? 'y' : '-'); } vshPrint(ctl, "\n"); return true; } /* * "nodecpustats" command */ static const vshCmdInfo info_nodecpustats = { .help = N_("Prints cpu stats of the node."), .desc = N_("Returns cpu stats of the node, in nanoseconds."), }; static const vshCmdOptDef opts_node_cpustats[] = { {.name = "cpu", .type = VSH_OT_INT, .completer = virshNodeCpuCompleter, .help = N_("prints specified cpu statistics only.") }, {.name = "percent", .type = VSH_OT_BOOL, .help = N_("prints by percentage during 1 second.") }, {.name = NULL} }; typedef enum { VIRSH_CPU_USER, VIRSH_CPU_SYSTEM, VIRSH_CPU_IDLE, VIRSH_CPU_IOWAIT, VIRSH_CPU_INTR, VIRSH_CPU_USAGE, VIRSH_CPU_LAST } virshCPUStats; VIR_ENUM_DECL(virshCPUStats); VIR_ENUM_IMPL(virshCPUStats, VIRSH_CPU_LAST, VIR_NODE_CPU_STATS_USER, VIR_NODE_CPU_STATS_KERNEL, VIR_NODE_CPU_STATS_IDLE, VIR_NODE_CPU_STATS_IOWAIT, VIR_NODE_CPU_STATS_INTR, VIR_NODE_CPU_STATS_UTILIZATION); const char *virshCPUOutput[] = { N_("user:"), N_("system:"), N_("idle:"), N_("iowait:"), N_("intr:"), N_("usage:") }; static bool cmdNodeCpuStats(vshControl *ctl, const vshCmd *cmd) { size_t i, j; bool flag_percent = vshCommandOptBool(cmd, "percent"); int cpuNum = VIR_NODE_CPU_STATS_ALL_CPUS; g_autofree virNodeCPUStatsPtr params = NULL; int nparams = 0; unsigned long long cpu_stats[VIRSH_CPU_LAST] = { 0 }; bool present[VIRSH_CPU_LAST] = { false }; virshControl *priv = ctl->privData; if (vshCommandOptInt(ctl, cmd, "cpu", &cpuNum) < 0) return false; if (virNodeGetCPUStats(priv->conn, cpuNum, NULL, &nparams, 0) != 0) { vshError(ctl, "%s", _("Unable to get number of cpu stats")); return false; } if (nparams == 0) { /* nothing to output */ return true; } memset(cpu_stats, 0, sizeof(cpu_stats)); params = g_new0(virNodeCPUStats, nparams); for (i = 0; i < 2; i++) { if (virNodeGetCPUStats(priv->conn, cpuNum, params, &nparams, 0) != 0) { vshError(ctl, "%s", _("Unable to get node cpu stats")); return false; } for (j = 0; j < nparams; j++) { int field = virshCPUStatsTypeFromString(params[j].field); if (field < 0) continue; if (i == 0) { cpu_stats[field] = params[j].value; present[field] = true; } else if (present[field]) { cpu_stats[field] = params[j].value - cpu_stats[field]; } } if (present[VIRSH_CPU_USAGE] || !flag_percent) break; sleep(1); } if (!flag_percent) { for (i = 0; i < VIRSH_CPU_USAGE; i++) { if (present[i]) { vshPrint(ctl, "%-15s %20llu\n", _(virshCPUOutput[i]), cpu_stats[i]); } } return true; } if (present[VIRSH_CPU_USAGE]) { vshPrint(ctl, "%-15s %5.1llu%%\n", _("usage:"), cpu_stats[VIRSH_CPU_USAGE]); vshPrint(ctl, "%-15s %5.1llu%%\n", _("idle:"), 100 - cpu_stats[VIRSH_CPU_USAGE]); } else { double usage, total_time = 0; for (i = 0; i < VIRSH_CPU_USAGE; i++) total_time += cpu_stats[i]; usage = (cpu_stats[VIRSH_CPU_USER] + cpu_stats[VIRSH_CPU_SYSTEM]) / total_time * 100; vshPrint(ctl, "%-15s %5.1lf%%\n", _("usage:"), usage); for (i = 0; i < VIRSH_CPU_USAGE; i++) { if (present[i]) { vshPrint(ctl, "%-15s %5.1lf%%\n", _(virshCPUOutput[i]), cpu_stats[i] / total_time * 100); } } } return true; } /* * "nodememstats" command */ static const vshCmdInfo info_nodememstats = { .help = N_("Prints memory stats of the node."), .desc = N_("Returns memory stats of the node, in kilobytes."), }; static const vshCmdOptDef opts_node_memstats[] = { {.name = "cell", .type = VSH_OT_INT, .help = N_("prints specified cell statistics only.") }, {.name = NULL} }; static bool cmdNodeMemStats(vshControl *ctl, const vshCmd *cmd) { int nparams = 0; size_t i; int cellNum = VIR_NODE_MEMORY_STATS_ALL_CELLS; g_autofree virNodeMemoryStatsPtr params = NULL; virshControl *priv = ctl->privData; if (vshCommandOptInt(ctl, cmd, "cell", &cellNum) < 0) return false; /* get the number of memory parameters */ if (virNodeGetMemoryStats(priv->conn, cellNum, NULL, &nparams, 0) != 0) { vshError(ctl, "%s", _("Unable to get number of memory stats")); return false; } if (nparams == 0) { /* nothing to output */ return true; } /* now go get all the memory parameters */ params = g_new0(virNodeMemoryStats, nparams); if (virNodeGetMemoryStats(priv->conn, cellNum, params, &nparams, 0) != 0) { vshError(ctl, "%s", _("Unable to get memory stats")); return false; } for (i = 0; i < nparams; i++) vshPrint(ctl, "%-7s: %20llu KiB\n", params[i].field, params[i].value); return true; } /* * "nodesevinfo" command */ static const vshCmdInfo info_nodesevinfo = { .help = N_("node SEV information"), .desc = N_("Returns basic SEV information about the node."), }; static bool cmdNodeSEVInfo(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { virshControl *priv = ctl->privData; size_t i; int nparams = 0; virTypedParameterPtr params = NULL; bool ret = false; if (virNodeGetSEVInfo(priv->conn, ¶ms, &nparams, 0) != 0) { vshError(ctl, "%s", _("Unable to get host SEV information")); goto cleanup; } for (i = 0; i < nparams; i++) { g_autofree char *str = vshGetTypedParamValue(ctl, ¶ms[i]); vshPrint(ctl, "%-18s: %s\n", params[i].field, str); } ret = true; cleanup: virTypedParamsFree(params, nparams); return ret; } /* * "nodesuspend" command */ VIR_ENUM_IMPL(virshNodeSuspendTarget, VIR_NODE_SUSPEND_TARGET_LAST, "mem", "disk", "hybrid"); static const vshCmdInfo info_nodesuspend = { .help = N_("suspend the host node for a given time duration"), .desc = N_("Suspend the host node for a given time duration " "and attempt to resume thereafter."), }; static const vshCmdOptDef opts_node_suspend[] = { {.name = "target", .type = VSH_OT_DATA, .positional = true, .required = true, .flags = VSH_OFLAG_REQ, .completer = virshNodeSuspendTargetCompleter, .help = N_("mem(Suspend-to-RAM), disk(Suspend-to-Disk), " "hybrid(Hybrid-Suspend)") }, {.name = "duration", .type = VSH_OT_INT, .positional = true, .required = true, .flags = VSH_OFLAG_REQ, .help = N_("Suspend duration in seconds, at least 60") }, {.name = NULL} }; static bool cmdNodeSuspend(vshControl *ctl, const vshCmd *cmd) { const char *target = NULL; int suspendTarget; long long duration; virshControl *priv = ctl->privData; if (vshCommandOptStringReq(ctl, cmd, "target", &target) < 0) return false; if (vshCommandOptLongLong(ctl, cmd, "duration", &duration) < 0) return false; if ((suspendTarget = virshNodeSuspendTargetTypeFromString(target)) < 0) { vshError(ctl, "%s", _("Invalid target")); return false; } if (duration < 0) { vshError(ctl, "%s", _("Invalid duration")); return false; } if (virNodeSuspendForDuration(priv->conn, suspendTarget, duration, 0) < 0) { vshError(ctl, "%s", _("The host was not suspended")); return false; } return true; } /* * "sysinfo" command */ static const vshCmdInfo info_sysinfo = { .help = N_("print the hypervisor sysinfo"), .desc = N_("output an XML string for the hypervisor sysinfo, if available"), }; static bool cmdSysinfo(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { g_autofree char *sysinfo = NULL; virshControl *priv = ctl->privData; sysinfo = virConnectGetSysinfo(priv->conn, 0); if (sysinfo == NULL) { vshError(ctl, "%s", _("failed to get sysinfo")); return false; } vshPrint(ctl, "%s", sysinfo); return true; } /* * "hostname" command */ static const vshCmdInfo info_hostname = { .help = N_("print the hypervisor hostname"), .desc = "", }; static bool cmdHostname(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { g_autofree char *hostname = NULL; virshControl *priv = ctl->privData; hostname = virConnectGetHostname(priv->conn); if (hostname == NULL) { vshError(ctl, "%s", _("failed to get hostname")); return false; } vshPrint(ctl, "%s\n", hostname); return true; } /* * "uri" command */ static const vshCmdInfo info_uri = { .help = N_("print the hypervisor canonical URI"), .desc = "", }; static bool cmdURI(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { g_autofree char *uri = NULL; virshControl *priv = ctl->privData; uri = virConnectGetURI(priv->conn); if (uri == NULL) { vshError(ctl, "%s", _("failed to get URI")); return false; } vshPrint(ctl, "%s\n", uri); return true; } /* Extracts the CPU definition XML strings from a file which may contain either * - just the CPU definitions, * - domain XMLs, * - capabilities XMLs, or * - domain capabilities XMLs. * * Returns NULL terminated string list. */ static char ** vshExtractCPUDefXMLs(vshControl *ctl, const char *xmlFile) { g_auto(GStrv) cpus = NULL; g_autofree char *buffer = NULL; g_autofree char *xmlStr = NULL; g_autoptr(xmlDoc) xml = NULL; g_autoptr(xmlXPathContext) ctxt = NULL; g_autofree xmlNodePtr *nodes = NULL; char *doc; size_t i; int n; if (virFileReadAll(xmlFile, VSH_MAX_XML_FILE, &buffer) < 0) return NULL; /* Strip possible XML declaration */ if (STRPREFIX(buffer, ""))) doc += 2; else doc = buffer; xmlStr = g_strdup_printf("%s", doc); if (!(xml = virXMLParseStringCtxt(xmlStr, xmlFile, &ctxt))) return NULL; n = virXPathNodeSet("/container/cpu|" "/container/domain/cpu|" "/container/capabilities/host/cpu|" "/container/domainCapabilities/cpu/" "mode[@name='host-model' and @supported='yes']", ctxt, &nodes); if (n < 0) return NULL; if (n == 0) { vshError(ctl, _("File '%1$s' does not contain any element or valid domain XML, host capabilities XML, or domain capabilities XML"), xmlFile); return NULL; } cpus = g_new0(char *, n + 1); for (i = 0; i < n; i++) { /* If the user provided domain capabilities XML, we need to replace * element with . */ if (xmlStrEqual(nodes[i]->name, BAD_CAST "mode")) { xmlNodeSetName(nodes[i], (const xmlChar *)"cpu"); while (nodes[i]->properties) { if (xmlRemoveProp(nodes[i]->properties) < 0) { vshError(ctl, _("Cannot extract CPU definition from domain capabilities XML")); return NULL; } } } if (!(cpus[i] = virXMLNodeToString(xml, nodes[i]))) { vshSaveLibvirtError(); return NULL; } } return g_steal_pointer(&cpus); } /* * "cpu-compare" command */ static const vshCmdInfo info_cpu_compare = { .help = N_("compare host CPU with a CPU described by an XML file"), .desc = N_("compare CPU with host CPU"), }; static const vshCmdOptDef opts_cpu_compare[] = { VIRSH_COMMON_OPT_FILE(N_("file containing an XML CPU description")), {.name = "error", .type = VSH_OT_BOOL, .help = N_("report error if CPUs are incompatible") }, {.name = "validate", .type = VSH_OT_BOOL, .help = N_("validate the XML document against schema") }, {.name = NULL} }; static bool cmdCPUCompare(vshControl *ctl, const vshCmd *cmd) { const char *from = NULL; int result; g_auto(GStrv) cpus = NULL; unsigned int flags = 0; virshControl *priv = ctl->privData; if (vshCommandOptBool(cmd, "error")) flags |= VIR_CONNECT_COMPARE_CPU_FAIL_INCOMPATIBLE; if (vshCommandOptBool(cmd, "validate")) flags |= VIR_CONNECT_COMPARE_CPU_VALIDATE_XML; if (vshCommandOptStringReq(ctl, cmd, "file", &from) < 0) return false; if (!(cpus = vshExtractCPUDefXMLs(ctl, from))) return false; result = virConnectCompareCPU(priv->conn, cpus[0], flags); switch (result) { case VIR_CPU_COMPARE_INCOMPATIBLE: vshPrint(ctl, _("CPU described in %1$s is incompatible with host CPU\n"), from); return false; break; case VIR_CPU_COMPARE_IDENTICAL: vshPrint(ctl, _("CPU described in %1$s is identical to host CPU\n"), from); break; case VIR_CPU_COMPARE_SUPERSET: vshPrint(ctl, _("Host CPU is a superset of CPU described in %1$s\n"), from); break; case VIR_CPU_COMPARE_ERROR: default: vshError(ctl, _("Failed to compare host CPU with %1$s"), from); return false; } return true; } /* * "cpu-baseline" command */ static const vshCmdInfo info_cpu_baseline = { .help = N_("compute baseline CPU"), .desc = N_("Compute baseline CPU for a set of given CPUs."), }; static const vshCmdOptDef opts_cpu_baseline[] = { VIRSH_COMMON_OPT_FILE(N_("file containing XML CPU descriptions")), {.name = "features", .type = VSH_OT_BOOL, .help = N_("Show features that are part of the CPU model type") }, {.name = "migratable", .type = VSH_OT_BOOL, .help = N_("Do not include features that block migration") }, {.name = NULL} }; static bool cmdCPUBaseline(vshControl *ctl, const vshCmd *cmd) { const char *from = NULL; g_autofree char *result = NULL; g_auto(GStrv) list = NULL; unsigned int flags = 0; virshControl *priv = ctl->privData; if (vshCommandOptBool(cmd, "features")) flags |= VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES; if (vshCommandOptBool(cmd, "migratable")) flags |= VIR_CONNECT_BASELINE_CPU_MIGRATABLE; if (vshCommandOptStringReq(ctl, cmd, "file", &from) < 0) return false; if (!(list = vshExtractCPUDefXMLs(ctl, from))) return false; if (!(result = virConnectBaselineCPU(priv->conn, (const char **)list, g_strv_length(list), flags))) return false; vshPrint(ctl, "%s", result); return true; } /* * "cpu-models" command */ static const vshCmdInfo info_cpu_models = { .help = N_("CPU models"), .desc = N_("Get the CPU models for an arch."), }; static const vshCmdOptDef opts_cpu_models[] = { {.name = "arch", .type = VSH_OT_DATA, .positional = true, .required = true, .completer = virshArchCompleter, .flags = VSH_OFLAG_REQ, .help = N_("architecture") }, {.name = NULL} }; static bool cmdCPUModelNames(vshControl *ctl, const vshCmd *cmd) { char **models; size_t i; int nmodels; const char *arch = NULL; virshControl *priv = ctl->privData; if (vshCommandOptStringReq(ctl, cmd, "arch", &arch) < 0) return false; nmodels = virConnectGetCPUModelNames(priv->conn, arch, &models, 0); if (nmodels < 0) { vshError(ctl, "%s", _("failed to get CPU model names")); return false; } if (nmodels == 0) { vshPrintExtra(ctl, "%s\n", _("all CPU models are accepted")); } else { for (i = 0; i < nmodels; i++) { vshPrint(ctl, "%s\n", models[i]); VIR_FREE(models[i]); } } VIR_FREE(models); return true; } /* * "version" command */ static const vshCmdInfo info_version = { .help = N_("show version"), .desc = N_("Display the system version information."), }; static const vshCmdOptDef opts_version[] = { {.name = "daemon", .type = VSH_OT_BOOL, .help = N_("report daemon version too") }, {.name = NULL} }; static bool cmdVersion(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED) { unsigned long hvVersion; const char *hvType; unsigned long libVersion; unsigned long includeVersion; unsigned long apiVersion; unsigned long daemonVersion; unsigned int major; unsigned int minor; unsigned int rel; virshControl *priv = ctl->privData; hvType = virConnectGetType(priv->conn); if (hvType == NULL) { vshError(ctl, "%s", _("failed to get hypervisor type")); return false; } includeVersion = LIBVIR_VERSION_NUMBER; major = includeVersion / 1000000; includeVersion %= 1000000; minor = includeVersion / 1000; rel = includeVersion % 1000; vshPrint(ctl, _("Compiled against library: libvirt %1$d.%2$d.%3$d\n"), major, minor, rel); if (virGetVersion(&libVersion, hvType, &apiVersion) < 0) { vshError(ctl, "%s", _("failed to get the library version")); return false; } major = libVersion / 1000000; libVersion %= 1000000; minor = libVersion / 1000; rel = libVersion % 1000; vshPrint(ctl, _("Using library: libvirt %1$d.%2$d.%3$d\n"), major, minor, rel); major = apiVersion / 1000000; apiVersion %= 1000000; minor = apiVersion / 1000; rel = apiVersion % 1000; vshPrint(ctl, _("Using API: %1$s %2$d.%3$d.%4$d\n"), hvType, major, minor, rel); if (virConnectGetVersion(priv->conn, &hvVersion) < 0) { if (last_error->code == VIR_ERR_NO_SUPPORT) { vshResetLibvirtError(); } else { vshError(ctl, "%s", _("failed to get the hypervisor version")); return false; } } else { if (hvVersion == 0) { vshPrint(ctl, _("Cannot extract running %1$s hypervisor version\n"), hvType); } else { major = hvVersion / 1000000; hvVersion %= 1000000; minor = hvVersion / 1000; rel = hvVersion % 1000; vshPrint(ctl, _("Running hypervisor: %1$s %2$d.%3$d.%4$d\n"), hvType, major, minor, rel); } } if (vshCommandOptBool(cmd, "daemon")) { if (virConnectGetLibVersion(priv->conn, &daemonVersion) < 0) { vshError(ctl, "%s", _("failed to get the daemon version")); } else { major = daemonVersion / 1000000; daemonVersion %= 1000000; minor = daemonVersion / 1000; rel = daemonVersion % 1000; vshPrint(ctl, _("Running against daemon: %1$d.%2$d.%3$d\n"), major, minor, rel); } } return true; } static const vshCmdInfo info_node_memory_tune = { .help = N_("Get or set node memory parameters"), .desc = N_("Get or set node memory parameters"), }; static const vshCmdOptDef opts_node_memory_tune[] = { {.name = "shm-pages-to-scan", .type = VSH_OT_INT, .help = N_("number of pages to scan before the shared memory service " "goes to sleep") }, {.name = "shm-sleep-millisecs", .type = VSH_OT_INT, .help = N_("number of millisecs the shared memory service should " "sleep before next scan") }, {.name = "shm-merge-across-nodes", .type = VSH_OT_INT, .help = N_("Specifies if pages from different numa nodes can be merged") }, {.name = NULL} }; static bool cmdNodeMemoryTune(vshControl *ctl, const vshCmd *cmd) { virTypedParameterPtr params = NULL; int nparams = 0; int maxparams = 0; unsigned int flags = 0; unsigned int value; bool ret = false; int rc = -1; size_t i; virshControl *priv = ctl->privData; if ((rc = vshCommandOptUInt(ctl, cmd, "shm-pages-to-scan", &value)) < 0) { goto cleanup; } else if (rc > 0) { if (virTypedParamsAddUInt(¶ms, &nparams, &maxparams, VIR_NODE_MEMORY_SHARED_PAGES_TO_SCAN, value) < 0) goto save_error; } if ((rc = vshCommandOptUInt(ctl, cmd, "shm-sleep-millisecs", &value)) < 0) { goto cleanup; } else if (rc > 0) { if (virTypedParamsAddUInt(¶ms, &nparams, &maxparams, VIR_NODE_MEMORY_SHARED_SLEEP_MILLISECS, value) < 0) goto save_error; } if ((rc = vshCommandOptUInt(ctl, cmd, "shm-merge-across-nodes", &value)) < 0) { goto cleanup; } else if (rc > 0) { if (virTypedParamsAddUInt(¶ms, &nparams, &maxparams, VIR_NODE_MEMORY_SHARED_MERGE_ACROSS_NODES, value) < 0) goto save_error; } if (nparams == 0) { /* Get the number of memory parameters */ if (virNodeGetMemoryParameters(priv->conn, NULL, &nparams, flags) != 0) { vshError(ctl, "%s", _("Unable to get number of memory parameters")); goto cleanup; } if (nparams == 0) { ret = true; goto cleanup; } /* Now go get all the memory parameters */ params = g_new0(virTypedParameter, nparams); if (virNodeGetMemoryParameters(priv->conn, params, &nparams, flags) != 0) { vshError(ctl, "%s", _("Unable to get memory parameters")); goto cleanup; } /* XXX: Need to sort the returned params once new parameter * fields not of shared memory are added. */ vshPrint(ctl, _("Shared memory:\n")); for (i = 0; i < nparams; i++) { g_autofree char *str = vshGetTypedParamValue(ctl, ¶ms[i]); vshPrint(ctl, "\t%-15s %s\n", params[i].field, str); } } else { if (virNodeSetMemoryParameters(priv->conn, params, nparams, flags) != 0) goto error; } ret = true; cleanup: virTypedParamsFree(params, nparams); return ret; save_error: vshSaveLibvirtError(); error: vshError(ctl, "%s", _("Unable to change memory parameters")); goto cleanup; } /* * "hypervisor-cpu-compare" command */ static const vshCmdInfo info_hypervisor_cpu_compare = { .help = N_("compare a CPU with the CPU created by a hypervisor on the host"), .desc = N_("compare CPU with hypervisor CPU"), }; static const vshCmdOptDef opts_hypervisor_cpu_compare[] = { VIRSH_COMMON_OPT_FILE(N_("file containing an XML CPU description")), {.name = "virttype", .type = VSH_OT_STRING, .completer = virshDomainVirtTypeCompleter, .help = N_("virtualization type (/domain/@type)"), }, {.name = "emulator", .type = VSH_OT_STRING, .help = N_("path to emulator binary (/domain/devices/emulator)"), }, {.name = "arch", .type = VSH_OT_STRING, .completer = virshArchCompleter, .help = N_("CPU architecture (/domain/os/type/@arch)"), }, {.name = "machine", .type = VSH_OT_STRING, .help = N_("machine type (/domain/os/type/@machine)"), }, {.name = "error", .type = VSH_OT_BOOL, .help = N_("report error if CPUs are incompatible") }, {.name = "validate", .type = VSH_OT_BOOL, .help = N_("validate the XML document against schema") }, {.name = NULL} }; static bool cmdHypervisorCPUCompare(vshControl *ctl, const vshCmd *cmd) { const char *from = NULL; const char *virttype = NULL; const char *emulator = NULL; const char *arch = NULL; const char *machine = NULL; int result; g_auto(GStrv) cpus = NULL; unsigned int flags = 0; virshControl *priv = ctl->privData; if (vshCommandOptBool(cmd, "error")) flags |= VIR_CONNECT_COMPARE_CPU_FAIL_INCOMPATIBLE; if (vshCommandOptBool(cmd, "validate")) flags |= VIR_CONNECT_COMPARE_CPU_VALIDATE_XML; if (vshCommandOptStringReq(ctl, cmd, "file", &from) < 0 || vshCommandOptStringReq(ctl, cmd, "virttype", &virttype) < 0 || vshCommandOptStringReq(ctl, cmd, "emulator", &emulator) < 0 || vshCommandOptStringReq(ctl, cmd, "arch", &arch) < 0 || vshCommandOptStringReq(ctl, cmd, "machine", &machine) < 0) return false; if (!(cpus = vshExtractCPUDefXMLs(ctl, from))) return false; result = virConnectCompareHypervisorCPU(priv->conn, emulator, arch, machine, virttype, cpus[0], flags); switch (result) { case VIR_CPU_COMPARE_INCOMPATIBLE: vshPrint(ctl, _("CPU described in %1$s is incompatible with the CPU provided by hypervisor on the host\n"), from); return false; break; case VIR_CPU_COMPARE_IDENTICAL: vshPrint(ctl, _("CPU described in %1$s is identical to the CPU provided by hypervisor on the host\n"), from); break; case VIR_CPU_COMPARE_SUPERSET: vshPrint(ctl, _("The CPU provided by hypervisor on the host is a superset of CPU described in %1$s\n"), from); break; case VIR_CPU_COMPARE_ERROR: default: vshError(ctl, _("Failed to compare hypervisor CPU with %1$s"), from); return false; } return true; } /* * "hypervisor-cpu-baseline" command */ static const vshCmdInfo info_hypervisor_cpu_baseline = { .help = N_("compute baseline CPU usable by a specific hypervisor"), .desc = N_("Compute baseline CPU for a set of given CPUs. The result " "will be tailored to the specified hypervisor."), }; static const vshCmdOptDef opts_hypervisor_cpu_baseline[] = { VIRSH_COMMON_OPT_FILE_FULL(N_("file containing XML CPU descriptions"), false), {.name = "virttype", .type = VSH_OT_STRING, .completer = virshDomainVirtTypeCompleter, .help = N_("virtualization type (/domain/@type)"), }, {.name = "emulator", .type = VSH_OT_STRING, .help = N_("path to emulator binary (/domain/devices/emulator)"), }, {.name = "arch", .type = VSH_OT_STRING, .completer = virshArchCompleter, .help = N_("CPU architecture (/domain/os/type/@arch)"), }, {.name = "machine", .type = VSH_OT_STRING, .help = N_("machine type (/domain/os/type/@machine)"), }, {.name = "features", .type = VSH_OT_BOOL, .help = N_("Show features that are part of the CPU model type") }, {.name = "migratable", .type = VSH_OT_BOOL, .help = N_("Do not include features that block migration") }, {.name = "model", .type = VSH_OT_STRING, .completer = virshCPUModelCompleter, .help = N_("Shortcut for calling the command with a single CPU model " "and no additional features") }, {.name = NULL} }; static bool cmdHypervisorCPUBaseline(vshControl *ctl, const vshCmd *cmd) { const char *from = NULL; const char *virttype = NULL; const char *emulator = NULL; const char *arch = NULL; const char *machine = NULL; const char *model = NULL; bool ret = false; g_autofree char *result = NULL; g_auto(GStrv) list = NULL; unsigned int flags = 0; virshControl *priv = ctl->privData; if (vshCommandOptBool(cmd, "features")) flags |= VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES; if (vshCommandOptBool(cmd, "migratable")) flags |= VIR_CONNECT_BASELINE_CPU_MIGRATABLE; if (vshCommandOptStringReq(ctl, cmd, "file", &from) < 0 || vshCommandOptStringReq(ctl, cmd, "virttype", &virttype) < 0 || vshCommandOptStringReq(ctl, cmd, "emulator", &emulator) < 0 || vshCommandOptStringReq(ctl, cmd, "arch", &arch) < 0 || vshCommandOptStringReq(ctl, cmd, "machine", &machine) < 0 || vshCommandOptStringReq(ctl, cmd, "model", &model) < 0) return false; VSH_ALTERNATIVE_OPTIONS_EXPR("file", from, "model", model); if (from) { if (!(list = vshExtractCPUDefXMLs(ctl, from))) return false; } else { list = g_new0(char *, 2); list[0] = g_strdup_printf("%s", model); } result = virConnectBaselineHypervisorCPU(priv->conn, emulator, arch, machine, virttype, (const char **)list, g_strv_length(list), flags); if (result) { vshPrint(ctl, "%s", result); ret = true; } return ret; } const vshCmdDef hostAndHypervisorCmds[] = { {.name = "allocpages", .handler = cmdAllocpages, .opts = opts_allocpages, .info = &info_allocpages, .flags = 0 }, {.name = "capabilities", .handler = cmdCapabilities, .opts = opts_capabilities, .info = &info_capabilities, .flags = 0 }, {.name = "cpu-baseline", .handler = cmdCPUBaseline, .opts = opts_cpu_baseline, .info = &info_cpu_baseline, .flags = 0 }, {.name = "cpu-compare", .handler = cmdCPUCompare, .opts = opts_cpu_compare, .info = &info_cpu_compare, .flags = 0 }, {.name = "cpu-models", .handler = cmdCPUModelNames, .opts = opts_cpu_models, .info = &info_cpu_models, .flags = 0 }, {.name = "domcapabilities", .handler = cmdDomCapabilities, .opts = opts_domcapabilities, .info = &info_domcapabilities, .flags = 0 }, {.name = "freecell", .handler = cmdFreecell, .opts = opts_freecell, .info = &info_freecell, .flags = 0 }, {.name = "freepages", .handler = cmdFreepages, .opts = opts_freepages, .info = &info_freepages, .flags = 0 }, {.name = "hostname", .handler = cmdHostname, .opts = NULL, .info = &info_hostname, .flags = 0 }, {.name = "hypervisor-cpu-baseline", .handler = cmdHypervisorCPUBaseline, .opts = opts_hypervisor_cpu_baseline, .info = &info_hypervisor_cpu_baseline, .flags = 0 }, {.name = "hypervisor-cpu-compare", .handler = cmdHypervisorCPUCompare, .opts = opts_hypervisor_cpu_compare, .info = &info_hypervisor_cpu_compare, .flags = 0 }, {.name = "maxvcpus", .handler = cmdMaxvcpus, .opts = opts_maxvcpus, .info = &info_maxvcpus, .flags = 0 }, {.name = "node-memory-tune", .handler = cmdNodeMemoryTune, .opts = opts_node_memory_tune, .info = &info_node_memory_tune, .flags = 0 }, {.name = "nodecpumap", .handler = cmdNodeCpuMap, .opts = opts_node_cpumap, .info = &info_node_cpumap, .flags = 0 }, {.name = "nodecpustats", .handler = cmdNodeCpuStats, .opts = opts_node_cpustats, .info = &info_nodecpustats, .flags = 0 }, {.name = "nodeinfo", .handler = cmdNodeinfo, .opts = NULL, .info = &info_nodeinfo, .flags = 0 }, {.name = "nodememstats", .handler = cmdNodeMemStats, .opts = opts_node_memstats, .info = &info_nodememstats, .flags = 0 }, {.name = "nodesevinfo", .handler = cmdNodeSEVInfo, .opts = NULL, .info = &info_nodesevinfo, .flags = 0 }, {.name = "nodesuspend", .handler = cmdNodeSuspend, .opts = opts_node_suspend, .info = &info_nodesuspend, .flags = 0 }, {.name = "sysinfo", .handler = cmdSysinfo, .opts = NULL, .info = &info_sysinfo, .flags = 0 }, {.name = "uri", .handler = cmdURI, .opts = NULL, .info = &info_uri, .flags = 0 }, {.name = "version", .handler = cmdVersion, .opts = opts_version, .info = &info_version, .flags = 0 }, {.name = NULL} };