2010-11-05 14:22:13 +00:00
|
|
|
/*
|
|
|
|
* sysinfo.c: get SMBIOS/sysinfo information from the host
|
|
|
|
*
|
2011-02-07 23:16:04 +00:00
|
|
|
* Copyright (C) 2010-2011 Red Hat, Inc.
|
2010-11-05 14:22:13 +00:00
|
|
|
* 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
* Author: Daniel Veillard <veillard@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "virterror_internal.h"
|
|
|
|
#include "sysinfo.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "memory.h"
|
2010-12-03 21:32:12 +00:00
|
|
|
#include "command.h"
|
2010-11-05 14:22:13 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_SYSINFO
|
|
|
|
|
2010-12-07 19:20:46 +00:00
|
|
|
#define virSmbiosReportError(code, ...) \
|
2011-04-16 08:30:22 +00:00
|
|
|
virReportErrorHelper(VIR_FROM_SYSINFO, code, __FILE__, \
|
2010-11-05 14:22:13 +00:00
|
|
|
__FUNCTION__, __LINE__, __VA_ARGS__)
|
|
|
|
|
|
|
|
#define SYSINFO_SMBIOS_DECODER "dmidecode"
|
|
|
|
|
2011-05-27 09:47:30 +00:00
|
|
|
VIR_ENUM_IMPL(virSysinfo, VIR_SYSINFO_LAST,
|
|
|
|
"smbios");
|
|
|
|
|
2010-11-05 14:22:13 +00:00
|
|
|
/**
|
|
|
|
* virSysinfoDefFree:
|
|
|
|
* @def: a sysinfo structure
|
|
|
|
*
|
|
|
|
* Free up the sysinfo structure
|
|
|
|
*/
|
|
|
|
|
|
|
|
void virSysinfoDefFree(virSysinfoDefPtr def)
|
|
|
|
{
|
|
|
|
if (def == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(def->bios_vendor);
|
|
|
|
VIR_FREE(def->bios_version);
|
|
|
|
VIR_FREE(def->bios_date);
|
|
|
|
VIR_FREE(def->bios_release);
|
|
|
|
|
|
|
|
VIR_FREE(def->system_manufacturer);
|
|
|
|
VIR_FREE(def->system_product);
|
|
|
|
VIR_FREE(def->system_version);
|
|
|
|
VIR_FREE(def->system_serial);
|
|
|
|
VIR_FREE(def->system_uuid);
|
|
|
|
VIR_FREE(def->system_sku);
|
2010-12-02 00:07:41 +00:00
|
|
|
VIR_FREE(def->system_family);
|
2010-11-05 14:22:13 +00:00
|
|
|
VIR_FREE(def);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virSysinfoRead:
|
|
|
|
*
|
|
|
|
* Tries to read the SMBIOS information from the current host
|
|
|
|
*
|
|
|
|
* Returns: a filled up sysinfo structure or NULL in case of error
|
|
|
|
*/
|
|
|
|
#ifdef WIN32
|
|
|
|
virSysinfoDefPtr
|
|
|
|
virSysinfoRead(void) {
|
|
|
|
/*
|
|
|
|
* this can probably be extracted from Windows using API or registry
|
|
|
|
* http://www.microsoft.com/whdc/system/platform/firmware/SMBIOS.mspx
|
|
|
|
*/
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Host sysinfo extraction not supported on this platform"));
|
2010-12-07 19:20:46 +00:00
|
|
|
return NULL;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
2011-02-07 23:16:04 +00:00
|
|
|
|
|
|
|
#else /* !WIN32 */
|
|
|
|
|
2010-11-05 14:22:13 +00:00
|
|
|
virSysinfoDefPtr
|
|
|
|
virSysinfoRead(void) {
|
|
|
|
char *path, *cur, *eol, *base;
|
|
|
|
virSysinfoDefPtr ret = NULL;
|
|
|
|
char *outbuf = NULL;
|
2010-12-03 21:32:12 +00:00
|
|
|
virCommandPtr cmd;
|
2010-11-05 14:22:13 +00:00
|
|
|
|
|
|
|
path = virFindFileInPath(SYSINFO_SMBIOS_DECODER);
|
|
|
|
if (path == NULL) {
|
|
|
|
virSmbiosReportError(VIR_ERR_INTERNAL_ERROR,
|
2010-12-07 19:20:46 +00:00
|
|
|
_("Failed to find path for %s binary"),
|
2010-11-05 14:22:13 +00:00
|
|
|
SYSINFO_SMBIOS_DECODER);
|
2010-12-07 19:20:46 +00:00
|
|
|
return NULL;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
|
2010-12-03 21:32:12 +00:00
|
|
|
cmd = virCommandNewArgList(path, "-q", "-t", "0,1", NULL);
|
|
|
|
VIR_FREE(path);
|
|
|
|
virCommandSetOutputBuffer(cmd, &outbuf);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
2010-11-05 14:22:13 +00:00
|
|
|
virSmbiosReportError(VIR_ERR_INTERNAL_ERROR,
|
2010-12-07 19:20:46 +00:00
|
|
|
_("Failed to execute command %s"),
|
2010-11-05 14:22:13 +00:00
|
|
|
path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
goto no_memory;
|
|
|
|
|
2011-05-27 09:47:30 +00:00
|
|
|
ret->type = VIR_SYSINFO_SMBIOS;
|
2010-11-05 14:22:13 +00:00
|
|
|
|
|
|
|
base = outbuf;
|
|
|
|
|
|
|
|
if ((cur = strstr(base, "Vendor: ")) != NULL) {
|
|
|
|
cur += 8;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->bios_vendor = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "Version: ")) != NULL) {
|
|
|
|
cur += 9;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->bios_version = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "Release Date: ")) != NULL) {
|
|
|
|
cur += 14;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->bios_date = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "BIOS Revision: ")) != NULL) {
|
|
|
|
cur += 15;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->bios_release = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((base = strstr(outbuf, "System Information")) == NULL)
|
|
|
|
goto cleanup;
|
|
|
|
if ((cur = strstr(base, "Manufacturer: ")) != NULL) {
|
|
|
|
cur += 14;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) &&
|
|
|
|
((ret->system_manufacturer = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "Product Name: ")) != NULL) {
|
|
|
|
cur += 14;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->system_product = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "Version: ")) != NULL) {
|
|
|
|
cur += 9;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->system_version = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "Serial Number: ")) != NULL) {
|
|
|
|
cur += 15;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->system_serial = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "UUID: ")) != NULL) {
|
|
|
|
cur += 6;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->system_uuid = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
|
|
|
if ((cur = strstr(base, "SKU Number: ")) != NULL) {
|
|
|
|
cur += 12;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->system_sku = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-11-05 14:22:13 +00:00
|
|
|
}
|
2010-12-02 00:07:41 +00:00
|
|
|
if ((cur = strstr(base, "Family: ")) != NULL) {
|
|
|
|
cur += 8;
|
|
|
|
eol = strchr(cur, '\n');
|
|
|
|
if ((eol) && ((ret->system_family = strndup(cur, eol - cur)) == NULL))
|
2010-12-07 19:20:46 +00:00
|
|
|
goto no_memory;
|
2010-12-02 00:07:41 +00:00
|
|
|
}
|
2010-11-05 14:22:13 +00:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(outbuf);
|
2010-12-03 21:32:12 +00:00
|
|
|
virCommandFree(cmd);
|
2010-11-05 14:22:13 +00:00
|
|
|
|
2010-12-07 19:20:46 +00:00
|
|
|
return ret;
|
2010-11-05 14:22:13 +00:00
|
|
|
|
|
|
|
no_memory:
|
|
|
|
virReportOOMError();
|
|
|
|
|
|
|
|
virSysinfoDefFree(ret);
|
|
|
|
ret = NULL;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2011-05-31 13:12:33 +00:00
|
|
|
#endif /* !WIN32 */
|
2011-02-07 23:16:04 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* virSysinfoFormat:
|
|
|
|
* @def: structure to convert to xml string
|
|
|
|
* @prefix: string to prefix before each line of xml
|
|
|
|
*
|
|
|
|
* This returns the XML description of the sysinfo, or NULL after
|
|
|
|
* generating an error message.
|
|
|
|
*/
|
|
|
|
char *
|
|
|
|
virSysinfoFormat(virSysinfoDefPtr def, const char *prefix)
|
|
|
|
{
|
2011-05-27 09:47:30 +00:00
|
|
|
const char *type = virSysinfoTypeToString(def->type);
|
2011-02-07 23:16:04 +00:00
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
size_t len = strlen(prefix);
|
|
|
|
|
|
|
|
if (!type) {
|
|
|
|
virSmbiosReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unexpected sysinfo type model %d"),
|
|
|
|
def->type);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-04-30 16:34:49 +00:00
|
|
|
virBufferAsprintf(&buf, "%s<sysinfo type='%s'>\n", prefix, type);
|
2011-02-07 23:16:04 +00:00
|
|
|
if ((def->bios_vendor != NULL) || (def->bios_version != NULL) ||
|
|
|
|
(def->bios_date != NULL) || (def->bios_release != NULL)) {
|
2011-04-30 16:34:49 +00:00
|
|
|
virBufferAsprintf(&buf, "%s <bios>\n", prefix);
|
2011-02-07 23:16:04 +00:00
|
|
|
if (def->bios_vendor != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='vendor'>%s</entry>\n",
|
|
|
|
def->bios_vendor);
|
|
|
|
}
|
|
|
|
if (def->bios_version != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='version'>%s</entry>\n",
|
|
|
|
def->bios_version);
|
|
|
|
}
|
|
|
|
if (def->bios_date != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='date'>%s</entry>\n",
|
|
|
|
def->bios_date);
|
|
|
|
}
|
|
|
|
if (def->bios_release != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='release'>%s</entry>\n",
|
|
|
|
def->bios_release);
|
|
|
|
}
|
2011-04-30 16:34:49 +00:00
|
|
|
virBufferAsprintf(&buf, "%s </bios>\n", prefix);
|
2011-02-07 23:16:04 +00:00
|
|
|
}
|
|
|
|
if ((def->system_manufacturer != NULL) || (def->system_product != NULL) ||
|
|
|
|
(def->system_version != NULL) || (def->system_serial != NULL) ||
|
|
|
|
(def->system_uuid != NULL) || (def->system_sku != NULL) ||
|
|
|
|
(def->system_family != NULL)) {
|
2011-04-30 16:34:49 +00:00
|
|
|
virBufferAsprintf(&buf, "%s <system>\n", prefix);
|
2011-02-07 23:16:04 +00:00
|
|
|
if (def->system_manufacturer != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='manufacturer'>%s</entry>\n",
|
|
|
|
def->system_manufacturer);
|
|
|
|
}
|
|
|
|
if (def->system_product != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='product'>%s</entry>\n",
|
|
|
|
def->system_product);
|
|
|
|
}
|
|
|
|
if (def->system_version != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='version'>%s</entry>\n",
|
|
|
|
def->system_version);
|
|
|
|
}
|
|
|
|
if (def->system_serial != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='serial'>%s</entry>\n",
|
|
|
|
def->system_serial);
|
|
|
|
}
|
|
|
|
if (def->system_uuid != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='uuid'>%s</entry>\n",
|
|
|
|
def->system_uuid);
|
|
|
|
}
|
|
|
|
if (def->system_sku != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='sku'>%s</entry>\n",
|
|
|
|
def->system_sku);
|
|
|
|
}
|
|
|
|
if (def->system_family != NULL) {
|
|
|
|
virBufferAdd(&buf, prefix, len);
|
|
|
|
virBufferEscapeString(&buf,
|
|
|
|
" <entry name='family'>%s</entry>\n",
|
|
|
|
def->system_family);
|
|
|
|
}
|
2011-04-30 16:34:49 +00:00
|
|
|
virBufferAsprintf(&buf, "%s </system>\n", prefix);
|
2011-02-07 23:16:04 +00:00
|
|
|
}
|
|
|
|
|
2011-04-30 16:34:49 +00:00
|
|
|
virBufferAsprintf(&buf, "%s</sysinfo>\n", prefix);
|
2011-02-07 23:16:04 +00:00
|
|
|
|
|
|
|
return virBufferContentAndReset(&buf);
|
|
|
|
}
|
|
|
|
|
2011-05-27 09:47:30 +00:00
|
|
|
bool virSysinfoIsEqual(virSysinfoDefPtr src,
|
|
|
|
virSysinfoDefPtr dst)
|
|
|
|
{
|
|
|
|
bool identical = false;
|
|
|
|
|
|
|
|
if (!src && !dst)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if ((src && !dst) || (!src && dst)) {
|
|
|
|
virSmbiosReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Target sysinfo does not match source"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (src->type != dst->type) {
|
|
|
|
virSmbiosReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Target sysinfo %s does not match source %s"),
|
|
|
|
virSysinfoTypeToString(dst->type),
|
|
|
|
virSysinfoTypeToString(src->type));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-05-31 13:12:33 +00:00
|
|
|
#define CHECK_FIELD(name, desc) \
|
2011-05-27 09:47:30 +00:00
|
|
|
do { \
|
|
|
|
if (STRNEQ_NULLABLE(src->name, dst->name)) { \
|
|
|
|
virSmbiosReportError(VIR_ERR_CONFIG_UNSUPPORTED, \
|
|
|
|
_("Target sysinfo %s %s does not match source %s"), \
|
|
|
|
desc, NULLSTR(src->name), NULLSTR(dst->name)); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
CHECK_FIELD(bios_vendor, "BIOS vendor");
|
|
|
|
CHECK_FIELD(bios_version, "BIOS version");
|
|
|
|
CHECK_FIELD(bios_date, "BIOS date");
|
|
|
|
CHECK_FIELD(bios_release, "BIOS release");
|
|
|
|
|
|
|
|
CHECK_FIELD(system_manufacturer, "system vendor");
|
|
|
|
CHECK_FIELD(system_product, "system product");
|
|
|
|
CHECK_FIELD(system_version, "system version");
|
|
|
|
CHECK_FIELD(system_serial, "system serial");
|
|
|
|
CHECK_FIELD(system_uuid, "system uuid");
|
|
|
|
CHECK_FIELD(system_sku, "system sku");
|
|
|
|
CHECK_FIELD(system_family, "system family");
|
|
|
|
|
2011-05-31 13:12:33 +00:00
|
|
|
#undef CHECK_FIELD
|
2011-05-27 09:47:30 +00:00
|
|
|
|
|
|
|
identical = true;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
return identical;
|
|
|
|
}
|