/*
* 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;
}