/* * virsysinfo.c: get SMBIOS/sysinfo information from the host * * Copyright (C) 2010-2014 Red Hat, Inc. * Copyright (C) 2010 Daniel Veillard * * 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 #include #include #include "virerror.h" #include "virsysinfo.h" #include "viralloc.h" #include "vircommand.h" #include "virlog.h" #include "virfile.h" #include "virstring.h" #include "virxml.h" #define LIBVIRT_VIRSYSINFOPRIV_H_ALLOW #include "virsysinfopriv.h" #define VIR_FROM_THIS VIR_FROM_SYSINFO VIR_LOG_INIT("util.sysinfo"); VIR_ENUM_IMPL(virSysinfo, VIR_SYSINFO_LAST, "smbios", "fwcfg" ); static const char *sysinfoSysinfo = "/proc/sysinfo"; static const char *sysinfoCpuinfo = "/proc/cpuinfo"; #define SYSINFO sysinfoSysinfo #define CPUINFO sysinfoCpuinfo #define CPUINFO_FILE_LEN (1024*1024) /* 1MB limit for /proc/cpuinfo file */ void virSysinfoSetup(const char *sysinfo, const char *cpuinfo) { sysinfoSysinfo = sysinfo; sysinfoCpuinfo = cpuinfo; } void virSysinfoBIOSDefFree(virSysinfoBIOSDef *def) { if (def == NULL) return; g_free(def->vendor); g_free(def->version); g_free(def->date); g_free(def->release); g_free(def); } void virSysinfoSystemDefFree(virSysinfoSystemDef *def) { if (def == NULL) return; g_free(def->manufacturer); g_free(def->product); g_free(def->version); g_free(def->serial); g_free(def->uuid); g_free(def->sku); g_free(def->family); g_free(def); } void virSysinfoBaseBoardDefClear(virSysinfoBaseBoardDef *def) { if (def == NULL) return; VIR_FREE(def->manufacturer); VIR_FREE(def->product); VIR_FREE(def->version); VIR_FREE(def->serial); VIR_FREE(def->asset); VIR_FREE(def->location); } void virSysinfoChassisDefFree(virSysinfoChassisDef *def) { if (def == NULL) return; g_free(def->manufacturer); g_free(def->version); g_free(def->serial); g_free(def->asset); g_free(def->sku); g_free(def); } void virSysinfoOEMStringsDefFree(virSysinfoOEMStringsDef *def) { size_t i; if (def == NULL) return; for (i = 0; i < def->nvalues; i++) g_free(def->values[i]); g_free(def->values); g_free(def); } static void virSysinfoFWCfgDefClear(virSysinfoFWCfgDef *def) { if (!def) return; VIR_FREE(def->name); VIR_FREE(def->value); VIR_FREE(def->file); } /** * virSysinfoDefFree: * @def: a sysinfo structure * * Free up the sysinfo structure */ void virSysinfoDefFree(virSysinfoDef *def) { size_t i; if (def == NULL) return; virSysinfoBIOSDefFree(def->bios); virSysinfoSystemDefFree(def->system); for (i = 0; i < def->nbaseBoard; i++) virSysinfoBaseBoardDefClear(def->baseBoard + i); g_free(def->baseBoard); virSysinfoChassisDefFree(def->chassis); for (i = 0; i < def->nprocessor; i++) { g_free(def->processor[i].processor_socket_destination); g_free(def->processor[i].processor_type); g_free(def->processor[i].processor_family); g_free(def->processor[i].processor_manufacturer); g_free(def->processor[i].processor_signature); g_free(def->processor[i].processor_version); g_free(def->processor[i].processor_external_clock); g_free(def->processor[i].processor_max_speed); g_free(def->processor[i].processor_status); g_free(def->processor[i].processor_serial_number); g_free(def->processor[i].processor_part_number); } g_free(def->processor); for (i = 0; i < def->nmemory; i++) { g_free(def->memory[i].memory_size); g_free(def->memory[i].memory_form_factor); g_free(def->memory[i].memory_locator); g_free(def->memory[i].memory_bank_locator); g_free(def->memory[i].memory_type); g_free(def->memory[i].memory_type_detail); g_free(def->memory[i].memory_speed); g_free(def->memory[i].memory_manufacturer); g_free(def->memory[i].memory_serial_number); g_free(def->memory[i].memory_part_number); } g_free(def->memory); virSysinfoOEMStringsDefFree(def->oemStrings); for (i = 0; i < def->nfw_cfgs; i++) virSysinfoFWCfgDefClear(&def->fw_cfgs[i]); g_free(def->fw_cfgs); g_free(def); } static bool virSysinfoDefIsEmpty(const virSysinfoDef *def) { return !(def->bios || def->system || def->nbaseBoard > 0 || def->chassis || def->nprocessor > 0 || def->nmemory > 0 || def->oemStrings); } static int virSysinfoParsePPCSystem(const char *base, virSysinfoSystemDef **sysdef) { int ret = -1; char *eol = NULL; const char *cur; virSysinfoSystemDef *def; if ((cur = strstr(base, "platform")) == NULL) return 0; def = g_new0(virSysinfoSystemDef, 1); base = cur; /* Account for format 'platform : XXXX'*/ cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) def->family = g_strndup(cur, eol - cur); if ((cur = strstr(base, "model")) != NULL) { cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) def->serial = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "machine")) != NULL) { cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) def->version = g_strndup(cur, eol - cur); } if (!def->manufacturer && !def->product && !def->version && !def->serial && !def->uuid && !def->sku && !def->family) { g_clear_pointer(&def, virSysinfoSystemDefFree); } *sysdef = g_steal_pointer(&def); ret = 0; virSysinfoSystemDefFree(def); return ret; } static int virSysinfoParsePPCProcessor(const char *base, virSysinfoDef *ret) { const char *cur; char *eol, *tmp_base; virSysinfoProcessorDef *processor; while ((tmp_base = strstr(base, "processor")) != NULL) { base = tmp_base; eol = strchr(base, '\n'); cur = strchr(base, ':') + 1; VIR_EXPAND_N(ret->processor, ret->nprocessor, 1); processor = &ret->processor[ret->nprocessor - 1]; virSkipSpaces(&cur); if (eol) processor->processor_socket_destination = g_strndup(cur, eol - cur); base = cur; if ((cur = strstr(base, "cpu")) != NULL) { cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) processor->processor_type = g_strndup(cur, eol - cur); base = cur; } if ((cur = strstr(base, "revision")) != NULL) { cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) processor->processor_version = g_strndup(cur, eol - cur); base = cur; } } return 0; } /* virSysinfoRead for PowerPC * Gathers sysinfo data from /proc/cpuinfo */ virSysinfoDef * virSysinfoReadPPC(void) { g_autoptr(virSysinfoDef) ret = NULL; g_autofree char *outbuf = NULL; ret = g_new0(virSysinfoDef, 1); if (virFileReadAll(CPUINFO, CPUINFO_FILE_LEN, &outbuf) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Failed to open %1$s"), CPUINFO); return NULL; } ret->nprocessor = 0; ret->processor = NULL; if (virSysinfoParsePPCProcessor(outbuf, ret) < 0) return NULL; if (virSysinfoParsePPCSystem(outbuf, &ret->system) < 0) return NULL; return g_steal_pointer(&ret); } static int virSysinfoParseARMSystem(const char *base, virSysinfoSystemDef **sysdef) { int ret = -1; char *eol = NULL; const char *cur; virSysinfoSystemDef *def; if ((cur = strstr(base, "platform")) == NULL) return 0; def = g_new0(virSysinfoSystemDef, 1); base = cur; /* Account for format 'platform : XXXX'*/ cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) def->family = g_strndup(cur, eol - cur); if ((cur = strstr(base, "model")) != NULL) { cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) def->serial = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "machine")) != NULL) { cur = strchr(cur, ':') + 1; eol = strchr(cur, '\n'); virSkipSpaces(&cur); if (eol) def->version = g_strndup(cur, eol - cur); } if (!def->manufacturer && !def->product && !def->version && !def->serial && !def->uuid && !def->sku && !def->family) { g_clear_pointer(&def, virSysinfoSystemDefFree); } *sysdef = g_steal_pointer(&def); ret = 0; virSysinfoSystemDefFree(def); return ret; } static int virSysinfoParseARMProcessor(const char *base, virSysinfoDef *ret) { const char *cur; char *eol, *tmp_base; virSysinfoProcessorDef *processor; char *processor_type = NULL; if (!(tmp_base = strstr(base, "model name")) && !(tmp_base = strstr(base, "Processor"))) return 0; eol = strchr(tmp_base, '\n'); cur = strchr(tmp_base, ':') + 1; virSkipSpaces(&cur); if (eol) processor_type = g_strndup(cur, eol - cur); while ((tmp_base = strstr(base, "processor")) != NULL) { base = tmp_base; eol = strchr(base, '\n'); cur = strchr(base, ':') + 1; VIR_EXPAND_N(ret->processor, ret->nprocessor, 1); processor = &ret->processor[ret->nprocessor - 1]; virSkipSpaces(&cur); if (eol) processor->processor_socket_destination = g_strndup(cur, eol - cur); processor->processor_type = g_strdup(processor_type); base = cur; } VIR_FREE(processor_type); return 0; } /* virSysinfoRead for ARMv7 * Gathers sysinfo data from /proc/cpuinfo */ virSysinfoDef * virSysinfoReadARM(void) { g_autoptr(virSysinfoDef) ret = NULL; g_autofree char *outbuf = NULL; /* Some ARM systems have DMI tables available. */ if ((ret = virSysinfoReadDMI())) { if (!virSysinfoDefIsEmpty(ret)) return g_steal_pointer(&ret); virSysinfoDefFree(ret); } /* Well, we've tried. Fall back to parsing cpuinfo */ virResetLastError(); ret = g_new0(virSysinfoDef, 1); if (virFileReadAll(CPUINFO, CPUINFO_FILE_LEN, &outbuf) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Failed to open %1$s"), CPUINFO); return NULL; } ret->nprocessor = 0; ret->processor = NULL; if (virSysinfoParseARMProcessor(outbuf, ret) < 0) return NULL; if (virSysinfoParseARMSystem(outbuf, &ret->system) < 0) return NULL; return g_steal_pointer(&ret); } static const char * virSysinfoParseS390Delimited(const char *base, const char *name, char **value, char delim1, char delim2) { const char *start; const char *end; if (delim1 != delim2 && (start = strstr(base, name)) && (start = strchr(start, delim1))) { start += 1; end = strchr(start, delim2); if (!end) end = start + strlen(start); virSkipSpaces(&start); *value = g_strndup(start, end - start); virTrimSpaces(*value, NULL); return end; } return NULL; } static const char * virSysinfoParseS390Line(const char *base, const char *name, char **value) { return virSysinfoParseS390Delimited(base, name, value, ':', '\n'); } static int virSysinfoParseS390System(const char *base, virSysinfoSystemDef **sysdef) { int ret = -1; virSysinfoSystemDef *def; def = g_new0(virSysinfoSystemDef, 1); if (!virSysinfoParseS390Line(base, "Manufacturer", &def->manufacturer)) goto cleanup; if (!virSysinfoParseS390Line(base, "Type", &def->family)) goto cleanup; if (!virSysinfoParseS390Line(base, "Sequence Code", &def->serial)) goto cleanup; if (!def->manufacturer && !def->product && !def->version && !def->serial && !def->uuid && !def->sku && !def->family) { g_clear_pointer(&def, virSysinfoSystemDefFree); } *sysdef = g_steal_pointer(&def); ret = 0; cleanup: virSysinfoSystemDefFree(def); return ret; } static int virSysinfoParseS390Processor(const char *base, virSysinfoDef *ret) { const char *tmp_base; char *manufacturer = NULL; char *procline = NULL; char *ncpu = NULL; int result = -1; virSysinfoProcessorDef *processor; if (!(tmp_base = virSysinfoParseS390Line(base, "vendor_id", &manufacturer))) goto error; /* Find processor N: line and gather the processor manufacturer, version, serial number, and family */ while ((tmp_base = strstr(tmp_base, "processor ")) && (tmp_base = virSysinfoParseS390Line(tmp_base, "processor ", &procline))) { VIR_EXPAND_N(ret->processor, ret->nprocessor, 1); processor = &ret->processor[ret->nprocessor - 1]; processor->processor_manufacturer = g_strdup(manufacturer); if (!virSysinfoParseS390Delimited(procline, "version", &processor->processor_version, '=', ',') || !virSysinfoParseS390Delimited(procline, "identification", &processor->processor_serial_number, '=', ',') || !virSysinfoParseS390Delimited(procline, "machine", &processor->processor_family, '=', '\n')) goto error; VIR_FREE(procline); } /* now, for each processor found, extract the frequency information */ tmp_base = base; while ((tmp_base = strstr(tmp_base, "cpu number")) && (tmp_base = virSysinfoParseS390Line(tmp_base, "cpu number", &ncpu))) { unsigned int n; char *mhz = NULL; if (virStrToLong_uip(ncpu, NULL, 10, &n) < 0) goto error; if (n >= ret->nprocessor) { VIR_DEBUG("CPU number '%u' out of range", n); goto cleanup; } if (!(tmp_base = strstr(tmp_base, "cpu MHz static")) || !virSysinfoParseS390Line(tmp_base, "cpu MHz static", &mhz)) goto cleanup; ret->processor[n].processor_max_speed = mhz; VIR_FREE(ncpu); } cleanup: result = 0; error: VIR_FREE(manufacturer); VIR_FREE(procline); VIR_FREE(ncpu); return result; } /* virSysinfoRead for s390x * Gathers sysinfo data from /proc/sysinfo and /proc/cpuinfo */ virSysinfoDef * virSysinfoReadS390(void) { g_autoptr(virSysinfoDef) ret = NULL; g_autofree char *outbuf = NULL; ret = g_new0(virSysinfoDef, 1); /* Gather info from /proc/cpuinfo */ if (virFileReadAll(CPUINFO, CPUINFO_FILE_LEN, &outbuf) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Failed to open %1$s"), CPUINFO); return NULL; } if (virSysinfoParseS390Processor(outbuf, ret) < 0) return NULL; /* Free buffer before reading next file */ VIR_FREE(outbuf); /* Gather info from /proc/sysinfo */ if (virFileReadAll(SYSINFO, 8192, &outbuf) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Failed to open %1$s"), SYSINFO); return NULL; } if (virSysinfoParseS390System(outbuf, &ret->system) < 0) return NULL; return g_steal_pointer(&ret); } static int virSysinfoParseBIOS(const char *base, virSysinfoBIOSDef **bios) { int ret = -1; const char *cur; char *eol = NULL; virSysinfoBIOSDef *def; if ((cur = strstr(base, "BIOS Information")) == NULL) return 0; def = g_new0(virSysinfoBIOSDef, 1); base = cur; if ((cur = strstr(base, "Vendor: ")) != NULL) { cur += 8; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->vendor = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Version: ")) != NULL) { cur += 9; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->version = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Release Date: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->date = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "BIOS Revision: ")) != NULL) { cur += 15; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->release = g_strndup(cur, eol - cur); } if (!def->vendor && !def->version && !def->date && !def->release) { g_clear_pointer(&def, virSysinfoBIOSDefFree); } *bios = g_steal_pointer(&def); ret = 0; virSysinfoBIOSDefFree(def); return ret; } static int virSysinfoParseX86System(const char *base, virSysinfoSystemDef **sysdef) { int ret = -1; const char *cur; char *eol = NULL; virSysinfoSystemDef *def; if ((cur = strstr(base, "System Information")) == NULL) return 0; def = g_new0(virSysinfoSystemDef, 1); base = cur; if ((cur = strstr(base, "Manufacturer: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->manufacturer = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Product Name: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->product = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Version: ")) != NULL) { cur += 9; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->version = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Serial Number: ")) != NULL) { cur += 15; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->serial = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "UUID: ")) != NULL) { cur += 6; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->uuid = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "SKU Number: ")) != NULL) { cur += 12; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->sku = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Family: ")) != NULL) { cur += 8; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->family = g_strndup(cur, eol - cur); } if (!def->manufacturer && !def->product && !def->version && !def->serial && !def->uuid && !def->sku && !def->family) { g_clear_pointer(&def, virSysinfoSystemDefFree); } *sysdef = g_steal_pointer(&def); ret = 0; virSysinfoSystemDefFree(def); return ret; } static int virSysinfoParseX86BaseBoard(const char *base, virSysinfoBaseBoardDef **baseBoard, size_t *nbaseBoard) { const char *cur; char *eol = NULL; virSysinfoBaseBoardDef *boards = NULL; size_t nboards = 0; while (base && (cur = strstr(base, "Base Board Information"))) { virSysinfoBaseBoardDef *def; VIR_EXPAND_N(boards, nboards, 1); def = &boards[nboards - 1]; base = cur + 22; if ((cur = strstr(base, "Manufacturer: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->manufacturer = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Product Name: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->product = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Version: ")) != NULL) { cur += 9; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->version = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Serial Number: ")) != NULL) { cur += 15; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->serial = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Asset Tag: ")) != NULL) { cur += 11; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->asset = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Location In Chassis: ")) != NULL) { cur += 21; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->location = g_strndup(cur, eol - cur); } if (!def->manufacturer && !def->product && !def->version && !def->serial && !def->asset && !def->location) nboards--; } if (nboards == 0) { VIR_FREE(boards); } else { VIR_REALLOC_N(boards, nboards); } *nbaseBoard = nboards; *baseBoard = g_steal_pointer(&boards); return 0; } static int virSysinfoParseX86Chassis(const char *base, virSysinfoChassisDef **chassisdef) { int ret = -1; const char *cur; char *eol = NULL; virSysinfoChassisDef *def; if ((cur = strstr(base, "Chassis Information")) == NULL) return 0; def = g_new0(virSysinfoChassisDef, 1); base = cur; if ((cur = strstr(base, "Manufacturer: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->manufacturer = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Version: ")) != NULL) { cur += 9; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->version = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Serial Number: ")) != NULL) { cur += 15; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->serial = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Asset Tag: ")) != NULL) { cur += 11; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->asset = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "SKU Number: ")) != NULL) { cur += 12; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) def->sku = g_strndup(cur, eol - cur); } if (!def->manufacturer && !def->version && !def->serial && !def->asset && !def->sku) { g_clear_pointer(&def, virSysinfoChassisDefFree); } *chassisdef = g_steal_pointer(&def); ret = 0; virSysinfoChassisDefFree(def); return ret; } static int virSysinfoDMIDecodeOEMString(unsigned int idx, char **str) { g_autofree char *err = NULL; g_autoptr(virCommand) cmd = virCommandNewArgList(DMIDECODE, "--dump", "--oem-string", NULL); virCommandAddArgFormat(cmd, "%u", idx); virCommandSetOutputBuffer(cmd, str); virCommandSetErrorBuffer(cmd, &err); if (virCommandRun(cmd, NULL) < 0) return -1; /* Unfortunately, dmidecode returns 0 even if OEM String index is out * of bounds, but it prints an error message in that case. Check stderr * and return success/failure accordingly. * To make matters worse, if there are two or more 'OEM String' * sections then: * * a) we have no way of distinguishing them as dmidecode prints * strings from all sections, * b) if one section contains a valid string, but the other doesn't, * then stdout contains the valid string and stderr contains the * error "No OEM string number X*. * * Let's just hope there is not many systems like that. */ if ((!*str || **str == '\0') && err && *err != '\0') return -1; virStringTrimOptionalNewline(*str); return 0; } static int virSysinfoParseOEMStrings(const char *base, virSysinfoOEMStringsDef **stringsRet) { virSysinfoOEMStringsDef *strings = NULL; int ret = -1; const char *cur; if (!(cur = strstr(base, "OEM Strings"))) return 0; strings = g_new0(virSysinfoOEMStringsDef, 1); while ((cur = strstr(cur, "String "))) { char *collon = NULL; unsigned int idx = 0; char *eol; cur += 7; if (!(eol = strchr(cur, '\n'))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed output of dmidecode")); goto cleanup; } if (virStrToLong_ui(cur, &collon, 10, &idx) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed output of dmidecode")); goto cleanup; } cur = collon; if (*cur != ':') { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed output of dmidecode")); goto cleanup; } cur += 2; virSkipSpacesBackwards(cur, &eol); if (!eol) continue; VIR_EXPAND_N(strings->values, strings->nvalues, 1); /* If OEM String contains newline, dmidecode escapes it as a dot. * If this is the case then run dmidecode again to get raw string. * Unfortunately, we can't dinstinguish between dot an new line at * this level. */ if (memchr(cur, '.', eol - cur)) { char *str; if (virSysinfoDMIDecodeOEMString(idx, &str) < 0) goto cleanup; strings->values[strings->nvalues - 1] = g_steal_pointer(&str); } else { strings->values[strings->nvalues - 1] = g_strndup(cur, eol - cur); } cur = eol; } *stringsRet = g_steal_pointer(&strings); ret = 0; cleanup: virSysinfoOEMStringsDefFree(strings); return ret; } static int virSysinfoParseX86Processor(const char *base, virSysinfoDef *ret) { const char *cur, *tmp_base; char *eol; virSysinfoProcessorDef *processor; while ((tmp_base = strstr(base, "Processor Information")) != NULL) { base = tmp_base; eol = NULL; VIR_EXPAND_N(ret->processor, ret->nprocessor, 1); processor = &ret->processor[ret->nprocessor - 1]; if ((cur = strstr(base, "Socket Designation: ")) != NULL) { cur += 20; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_socket_destination = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Type: ")) != NULL) { cur += 6; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_type = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Family: ")) != NULL) { cur += 8; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_family = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Manufacturer: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_manufacturer = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Signature: ")) != NULL) { cur += 11; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_signature = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Version: ")) != NULL) { cur += 9; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_version = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "External Clock: ")) != NULL) { cur += 16; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_external_clock = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Max Speed: ")) != NULL) { cur += 11; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_max_speed = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Status: ")) != NULL) { cur += 8; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_status = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Serial Number: ")) != NULL) { cur += 15; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_serial_number = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Part Number: ")) != NULL) { cur += 13; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) processor->processor_part_number = g_strndup(cur, eol - cur); } base += strlen("Processor Information"); } return 0; } static int virSysinfoParseX86Memory(const char *base, virSysinfoDef *ret) { const char *cur, *tmp_base; char *eol; virSysinfoMemoryDef *memory; while ((tmp_base = strstr(base, "Memory Device")) != NULL) { base = tmp_base; eol = NULL; VIR_EXPAND_N(ret->memory, ret->nmemory, 1); memory = &ret->memory[ret->nmemory - 1]; if ((cur = strstr(base, "Size: ")) != NULL) { cur += 6; eol = strchr(cur, '\n'); if (STREQLEN(cur, "No Module Installed", eol - cur)) goto next; virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_size = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Form Factor: ")) != NULL) { cur += 13; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_form_factor = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Locator: ")) != NULL) { cur += 9; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_locator = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Bank Locator: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_bank_locator = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Type: ")) != NULL) { cur += 6; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_type = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Type Detail: ")) != NULL) { cur += 13; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_type_detail = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Speed: ")) != NULL) { cur += 7; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_speed = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Manufacturer: ")) != NULL) { cur += 14; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_manufacturer = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Serial Number: ")) != NULL) { cur += 15; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_serial_number = g_strndup(cur, eol - cur); } if ((cur = strstr(base, "Part Number: ")) != NULL) { cur += 13; eol = strchr(cur, '\n'); virSkipSpacesBackwards(cur, &eol); if (eol) memory->memory_part_number = g_strndup(cur, eol - cur); } next: base += strlen("Memory Device"); } return 0; } virSysinfoDef * virSysinfoReadDMI(void) { g_autoptr(virSysinfoDef) ret = NULL; g_autofree char *outbuf = NULL; g_autoptr(virCommand) cmd = NULL; cmd = virCommandNewArgList(DMIDECODE, "-q", "-t", "0,1,2,3,4,11,17", NULL); virCommandSetOutputBuffer(cmd, &outbuf); if (virCommandRun(cmd, NULL) < 0) return NULL; ret = g_new0(virSysinfoDef, 1); ret->type = VIR_SYSINFO_SMBIOS; if (virSysinfoParseBIOS(outbuf, &ret->bios) < 0) return NULL; if (virSysinfoParseX86System(outbuf, &ret->system) < 0) return NULL; if (virSysinfoParseX86BaseBoard(outbuf, &ret->baseBoard, &ret->nbaseBoard) < 0) return NULL; if (virSysinfoParseX86Chassis(outbuf, &ret->chassis) < 0) return NULL; if (virSysinfoParseOEMStrings(outbuf, &ret->oemStrings) < 0) return NULL; ret->nprocessor = 0; ret->processor = NULL; if (virSysinfoParseX86Processor(outbuf, ret) < 0) return NULL; ret->nmemory = 0; ret->memory = NULL; if (virSysinfoParseX86Memory(outbuf, ret) < 0) return NULL; return g_steal_pointer(&ret); } /** * virSysinfoRead: * * Tries to read the SMBIOS information from the current host * * Returns: a filled up sysinfo structure or NULL in case of error */ virSysinfoDef * virSysinfoRead(void) { #if defined(__powerpc__) return virSysinfoReadPPC(); #elif defined(__arm__) || defined(__aarch64__) return virSysinfoReadARM(); #elif defined(__s390__) || defined(__s390x__) return virSysinfoReadS390(); #elif !defined(WIN32) && \ (defined(__x86_64__) || \ defined(__i386__) || \ defined(__amd64__) || \ defined(__riscv__) || \ defined(__mips__) || \ defined(__loongarch__)) return virSysinfoReadDMI(); #else /* WIN32 || not supported arch */ /* * this can probably be extracted from Windows using API or registry * https://www.microsoft.com/whdc/system/platform/firmware/SMBIOS.mspx */ virReportSystemError(ENOSYS, "%s", _("Host sysinfo extraction not supported on this platform")); return NULL; #endif /* WIN32 || not supported arch */ } static void virSysinfoBIOSFormat(virBuffer *buf, virSysinfoBIOSDef *def) { if (!def) return; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferEscapeString(buf, "%s\n", def->vendor); virBufferEscapeString(buf, "%s\n", def->version); virBufferEscapeString(buf, "%s\n", def->date); virBufferEscapeString(buf, "%s\n", def->release); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } static void virSysinfoSystemFormat(virBuffer *buf, virSysinfoSystemDef *def) { if (!def) return; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferEscapeString(buf, "%s\n", def->manufacturer); virBufferEscapeString(buf, "%s\n", def->product); virBufferEscapeString(buf, "%s\n", def->version); virBufferEscapeString(buf, "%s\n", def->serial); virBufferEscapeString(buf, "%s\n", def->uuid); virBufferEscapeString(buf, "%s\n", def->sku); virBufferEscapeString(buf, "%s\n", def->family); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } static void virSysinfoBaseBoardFormat(virBuffer *buf, virSysinfoBaseBoardDef *baseBoard, size_t nbaseBoard) { virSysinfoBaseBoardDef *def; size_t i; for (i = 0; i < nbaseBoard; i++) { def = baseBoard + i; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferEscapeString(buf, "%s\n", def->manufacturer); virBufferEscapeString(buf, "%s\n", def->product); virBufferEscapeString(buf, "%s\n", def->version); virBufferEscapeString(buf, "%s\n", def->serial); virBufferEscapeString(buf, "%s\n", def->asset); virBufferEscapeString(buf, "%s\n", def->location); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } } static void virSysinfoChassisFormat(virBuffer *buf, virSysinfoChassisDef *def) { if (!def) return; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferEscapeString(buf, "%s\n", def->manufacturer); virBufferEscapeString(buf, "%s\n", def->version); virBufferEscapeString(buf, "%s\n", def->serial); virBufferEscapeString(buf, "%s\n", def->asset); virBufferEscapeString(buf, "%s\n", def->sku); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } static void virSysinfoProcessorFormat(virBuffer *buf, virSysinfoDef *def) { size_t i; virSysinfoProcessorDef *processor; for (i = 0; i < def->nprocessor; i++) { processor = &def->processor[i]; if (!processor->processor_socket_destination && !processor->processor_type && !processor->processor_family && !processor->processor_manufacturer && !processor->processor_signature && !processor->processor_version && !processor->processor_external_clock && !processor->processor_max_speed && !processor->processor_status && !processor->processor_serial_number && !processor->processor_part_number) continue; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferEscapeString(buf, "%s\n", processor->processor_socket_destination); virBufferEscapeString(buf, "%s\n", processor->processor_type); virBufferEscapeString(buf, "%s\n", processor->processor_family); virBufferEscapeString(buf, "%s\n", processor->processor_manufacturer); virBufferEscapeString(buf, "%s\n", processor->processor_signature); virBufferEscapeString(buf, "%s\n", processor->processor_version); virBufferEscapeString(buf, "%s\n", processor->processor_external_clock); virBufferEscapeString(buf, "%s\n", processor->processor_max_speed); virBufferEscapeString(buf, "%s\n", processor->processor_status); virBufferEscapeString(buf, "%s\n", processor->processor_serial_number); virBufferEscapeString(buf, "%s\n", processor->processor_part_number); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } } static void virSysinfoMemoryFormat(virBuffer *buf, virSysinfoDef *def) { size_t i; virSysinfoMemoryDef *memory; for (i = 0; i < def->nmemory; i++) { memory = &def->memory[i]; if (!memory->memory_size && !memory->memory_form_factor && !memory->memory_locator && !memory->memory_bank_locator && !memory->memory_type && !memory->memory_type_detail && !memory->memory_speed && !memory->memory_manufacturer && !memory->memory_serial_number && !memory->memory_part_number) continue; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferEscapeString(buf, "%s\n", memory->memory_size); virBufferEscapeString(buf, "%s\n", memory->memory_form_factor); virBufferEscapeString(buf, "%s\n", memory->memory_locator); virBufferEscapeString(buf, "%s\n", memory->memory_bank_locator); virBufferEscapeString(buf, "%s\n", memory->memory_type); virBufferEscapeString(buf, "%s\n", memory->memory_type_detail); virBufferEscapeString(buf, "%s\n", memory->memory_speed); virBufferEscapeString(buf, "%s\n", memory->memory_manufacturer); virBufferEscapeString(buf, "%s\n", memory->memory_serial_number); virBufferEscapeString(buf, "%s\n", memory->memory_part_number); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } } static void virSysinfoOEMStringsFormat(virBuffer *buf, virSysinfoOEMStringsDef *def) { size_t i; if (!def) return; virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); for (i = 0; i < def->nvalues; i++) { virBufferEscapeString(buf, "%s\n", def->values[i]); } virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } static void virSysinfoFormatSMBIOS(virBuffer *buf, virSysinfoDef *def) { virSysinfoBIOSFormat(buf, def->bios); virSysinfoSystemFormat(buf, def->system); virSysinfoBaseBoardFormat(buf, def->baseBoard, def->nbaseBoard); virSysinfoChassisFormat(buf, def->chassis); virSysinfoProcessorFormat(buf, def); virSysinfoMemoryFormat(buf, def); virSysinfoOEMStringsFormat(buf, def->oemStrings); } static void virSysinfoFormatFWCfg(virBuffer *buf, virSysinfoDef *def) { size_t i; for (i = 0; i < def->nfw_cfgs; i++) { const virSysinfoFWCfgDef *f = &def->fw_cfgs[i]; virBufferAsprintf(buf, "name); if (f->file) virBufferEscapeString(buf, " file='%s'/>\n", f->file); else virBufferEscapeString(buf, ">%s\n", f->value); } } /** * virSysinfoFormat: * @buf: buffer to append output to (may use auto-indentation) * @def: structure to convert to xml string * * Returns 0 on success, -1 on failure after generating an error message. */ int virSysinfoFormat(virBuffer *buf, virSysinfoDef *def) { g_auto(virBuffer) attrBuf = VIR_BUFFER_INITIALIZER; g_auto(virBuffer) childrenBuf = VIR_BUFFER_INIT_CHILD(buf); const char *type = virSysinfoTypeToString(def->type); if (!type) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unexpected sysinfo type model %1$d"), def->type); return -1; } switch (def->type) { case VIR_SYSINFO_SMBIOS: virSysinfoFormatSMBIOS(&childrenBuf, def); break; case VIR_SYSINFO_FWCFG: virSysinfoFormatFWCfg(&childrenBuf, def); break; case VIR_SYSINFO_LAST: break; } virBufferAsprintf(&attrBuf, " type='%s'", type); virXMLFormatElement(buf, "sysinfo", &attrBuf, &childrenBuf); return 0; } #define CHECK_FIELD(name, desc) \ do { \ if (STRNEQ_NULLABLE(src->name, dst->name)) { \ virReportError(VIR_ERR_CONFIG_UNSUPPORTED, \ _("Target sysinfo %1$s %2$s does not match source %3$s"), \ desc, NULLSTR(dst->name), NULLSTR(src->name)); \ return false; \ } \ } while (0) static bool virSysinfoBIOSIsEqual(virSysinfoBIOSDef *src, virSysinfoBIOSDef *dst) { if (!src && !dst) return true; if ((src && !dst) || (!src && dst)) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s", _("Target sysinfo does not match source")); return false; } CHECK_FIELD(vendor, "BIOS vendor"); CHECK_FIELD(version, "BIOS version"); CHECK_FIELD(date, "BIOS date"); CHECK_FIELD(release, "BIOS release"); return true; } static bool virSysinfoSystemIsEqual(virSysinfoSystemDef *src, virSysinfoSystemDef *dst) { if (!src && !dst) return true; if ((src && !dst) || (!src && dst)) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s", _("Target sysinfo does not match source")); return false; } CHECK_FIELD(manufacturer, "system vendor"); CHECK_FIELD(product, "system product"); CHECK_FIELD(version, "system version"); CHECK_FIELD(serial, "system serial"); CHECK_FIELD(uuid, "system uuid"); CHECK_FIELD(sku, "system sku"); CHECK_FIELD(family, "system family"); return true; } static bool virSysinfoBaseBoardIsEqual(virSysinfoBaseBoardDef *src, virSysinfoBaseBoardDef *dst) { if (!src && !dst) return true; if ((src && !dst) || (!src && dst)) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s", _("Target base board does not match source")); return false; } CHECK_FIELD(manufacturer, "base board vendor"); CHECK_FIELD(product, "base board product"); CHECK_FIELD(version, "base board version"); CHECK_FIELD(serial, "base board serial"); CHECK_FIELD(asset, "base board asset"); CHECK_FIELD(location, "base board location"); return true; } static bool virSysinfoChassisIsEqual(virSysinfoChassisDef *src, virSysinfoChassisDef *dst) { if (!src && !dst) return true; if ((src && !dst) || (!src && dst)) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s", _("Target chassis does not match source")); return false; } CHECK_FIELD(manufacturer, "chassis vendor"); CHECK_FIELD(version, "chassis version"); CHECK_FIELD(serial, "chassis serial"); CHECK_FIELD(asset, "chassis asset"); CHECK_FIELD(sku, "chassis sku"); return true; } #undef CHECK_FIELD bool virSysinfoIsEqual(virSysinfoDef *src, virSysinfoDef *dst) { size_t i; if (!src && !dst) return true; if ((src && !dst) || (!src && dst)) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s", _("Target sysinfo does not match source")); return false; } if (src->type != dst->type) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, _("Target sysinfo %1$s does not match source %2$s"), virSysinfoTypeToString(dst->type), virSysinfoTypeToString(src->type)); return false; } if (!virSysinfoBIOSIsEqual(src->bios, dst->bios)) return false; if (!virSysinfoSystemIsEqual(src->system, dst->system)) return false; if (src->nbaseBoard != dst->nbaseBoard) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, _("Target sysinfo base board count '%1$zu' does not match source '%2$zu'"), dst->nbaseBoard, src->nbaseBoard); return false; } for (i = 0; i < src->nbaseBoard; i++) if (!virSysinfoBaseBoardIsEqual(src->baseBoard + i, dst->baseBoard + i)) return false; if (!virSysinfoChassisIsEqual(src->chassis, dst->chassis)) return false; return true; }