fwupd/libfwupdplugin/fu-hwids.c
2021-08-24 11:18:40 -05:00

544 lines
15 KiB
C

/*
* Copyright (C) 2017 Richard Hughes <richard@hughsie.com>
*
* SPDX-License-Identifier: LGPL-2.1+
*/
#define G_LOG_DOMAIN "FuHwids"
#include "config.h"
#include <gio/gio.h>
#include <glib.h>
#include <string.h>
#include "fwupd-common.h"
#include "fwupd-error.h"
#include "fu-common.h"
#include "fu-hwids.h"
/**
* FuHwids:
*
* A the hardware IDs on the system.
*
* Note, these are called "CHIDs" in Microsoft Windows and the results here
* will match that of `ComputerHardwareIds.exe`.
*
* See also: [class@FuSmbios]
*/
struct _FuHwids {
GObject parent_instance;
GHashTable *hash_dmi_hw; /* BiosVersion->"1.2.3 " */
GHashTable *hash_dmi_display; /* BiosVersion->"1.2.3" */
GHashTable *hash_smbios_override; /* BiosVersion->"1.2.3" */
GHashTable *hash_guid; /* a-c-b-d->1 */
GPtrArray *array_guids; /* a-c-b-d */
};
G_DEFINE_TYPE(FuHwids, fu_hwids, G_TYPE_OBJECT)
/**
* fu_hwids_get_value:
* @self: a #FuHwids
* @key: a DMI ID, e.g. `BiosVersion`
*
* Gets the cached value for one specific key that is valid ASCII and suitable
* for display.
*
* Returns: the string, e.g. `1.2.3`, or %NULL if not found
*
* Since: 0.9.3
**/
const gchar *
fu_hwids_get_value(FuHwids *self, const gchar *key)
{
return g_hash_table_lookup(self->hash_dmi_display, key);
}
/**
* fu_hwids_has_guid:
* @self: a #FuHwids
* @guid: a GUID, e.g. `059eb22d-6dc7-59af-abd3-94bbe017f67c`
*
* Finds out if a hardware GUID exists.
*
* Returns: %TRUE if the GUID exists
*
* Since: 0.9.3
**/
gboolean
fu_hwids_has_guid(FuHwids *self, const gchar *guid)
{
return g_hash_table_lookup(self->hash_guid, guid) != NULL;
}
/**
* fu_hwids_get_guids:
* @self: a #FuHwids
*
* Returns all the defined HWIDs
*
* Returns: (transfer none) (element-type utf8): an array of GUIDs
*
* Since: 0.9.3
**/
GPtrArray *
fu_hwids_get_guids(FuHwids *self)
{
return self->array_guids;
}
/**
* fu_hwids_get_keys:
* @self: a #FuHwids
*
* Returns all the defined HWID keys.
*
* Returns: (transfer container) (element-type utf8): All the known keys,
* e.g. %FU_HWIDS_KEY_FAMILY
*
* Since: 1.5.6
**/
GPtrArray *
fu_hwids_get_keys(FuHwids *self)
{
GPtrArray *array = g_ptr_array_new();
const gchar *keys[] = {FU_HWIDS_KEY_BIOS_VENDOR,
FU_HWIDS_KEY_BIOS_VERSION,
FU_HWIDS_KEY_BIOS_MAJOR_RELEASE,
FU_HWIDS_KEY_BIOS_MINOR_RELEASE,
FU_HWIDS_KEY_FIRMWARE_MAJOR_RELEASE,
FU_HWIDS_KEY_FIRMWARE_MINOR_RELEASE,
FU_HWIDS_KEY_MANUFACTURER,
FU_HWIDS_KEY_FAMILY,
FU_HWIDS_KEY_PRODUCT_NAME,
FU_HWIDS_KEY_PRODUCT_SKU,
FU_HWIDS_KEY_ENCLOSURE_KIND,
FU_HWIDS_KEY_BASEBOARD_MANUFACTURER,
FU_HWIDS_KEY_BASEBOARD_PRODUCT,
NULL};
for (guint i = 0; keys[i] != NULL; i++)
g_ptr_array_add(array, (gpointer)keys[i]);
return array;
}
static gchar *
fu_hwids_get_guid_for_str(const gchar *str, GError **error)
{
glong items_written = 0;
g_autofree gunichar2 *data = NULL;
/* convert to UTF-16 and convert to GUID using custom namespace */
data = g_utf8_to_utf16(str, -1, NULL, &items_written, error);
if (data == NULL)
return NULL;
if (items_written == 0) {
g_set_error_literal(error,
FWUPD_ERROR,
FWUPD_ERROR_INVALID_FILE,
"no GUIDs in data");
return NULL;
}
/* ensure the data is in little endian format */
for (glong i = 0; i < items_written; i++)
data[i] = GUINT16_TO_LE(data[i]);
/* convert to a GUID */
return fwupd_guid_hash_data((guint8 *)data,
items_written * 2,
FWUPD_GUID_FLAG_NAMESPACE_MICROSOFT);
}
/**
* fu_hwids_get_replace_keys:
* @self: a #FuHwids
* @key: a HardwareID key, e.g. `HardwareID-3`
*
* Gets the replacement key for a well known value.
*
* Returns: the replacement value, e.g. `Manufacturer&ProductName`, or %NULL for error.
*
* Since: 0.9.3
**/
const gchar *
fu_hwids_get_replace_keys(FuHwids *self, const gchar *key)
{
struct {
const gchar *search;
const gchar *replace;
} msdefined[] = {
{"HardwareID-0",
FU_HWIDS_KEY_MANUFACTURER
"&" FU_HWIDS_KEY_FAMILY "&" FU_HWIDS_KEY_PRODUCT_NAME "&" FU_HWIDS_KEY_PRODUCT_SKU
"&" FU_HWIDS_KEY_BIOS_VENDOR "&" FU_HWIDS_KEY_BIOS_VERSION
"&" FU_HWIDS_KEY_BIOS_MAJOR_RELEASE "&" FU_HWIDS_KEY_BIOS_MINOR_RELEASE},
{"HardwareID-1",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_FAMILY "&" FU_HWIDS_KEY_PRODUCT_NAME
"&" FU_HWIDS_KEY_BIOS_VENDOR "&" FU_HWIDS_KEY_BIOS_VERSION
"&" FU_HWIDS_KEY_BIOS_MAJOR_RELEASE
"&" FU_HWIDS_KEY_BIOS_MINOR_RELEASE},
{"HardwareID-2",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_PRODUCT_NAME "&" FU_HWIDS_KEY_BIOS_VENDOR
"&" FU_HWIDS_KEY_BIOS_VERSION
"&" FU_HWIDS_KEY_BIOS_MAJOR_RELEASE
"&" FU_HWIDS_KEY_BIOS_MINOR_RELEASE},
{"HardwareID-3",
FU_HWIDS_KEY_MANUFACTURER
"&" FU_HWIDS_KEY_FAMILY "&" FU_HWIDS_KEY_PRODUCT_NAME "&" FU_HWIDS_KEY_PRODUCT_SKU
"&" FU_HWIDS_KEY_BASEBOARD_MANUFACTURER "&" FU_HWIDS_KEY_BASEBOARD_PRODUCT},
{"HardwareID-4",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_FAMILY "&" FU_HWIDS_KEY_PRODUCT_NAME
"&" FU_HWIDS_KEY_PRODUCT_SKU},
{"HardwareID-5",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_FAMILY "&" FU_HWIDS_KEY_PRODUCT_NAME},
{"HardwareID-6",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_PRODUCT_SKU
"&" FU_HWIDS_KEY_BASEBOARD_MANUFACTURER
"&" FU_HWIDS_KEY_BASEBOARD_PRODUCT},
{"HardwareID-7", FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_PRODUCT_SKU},
{"HardwareID-8",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_PRODUCT_NAME
"&" FU_HWIDS_KEY_BASEBOARD_MANUFACTURER
"&" FU_HWIDS_KEY_BASEBOARD_PRODUCT},
{"HardwareID-9", FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_PRODUCT_NAME},
{"HardwareID-10",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_FAMILY
"&" FU_HWIDS_KEY_BASEBOARD_MANUFACTURER
"&" FU_HWIDS_KEY_BASEBOARD_PRODUCT},
{"HardwareID-11", FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_FAMILY},
{"HardwareID-12", FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_ENCLOSURE_KIND},
{"HardwareID-13",
FU_HWIDS_KEY_MANUFACTURER "&" FU_HWIDS_KEY_BASEBOARD_MANUFACTURER
"&" FU_HWIDS_KEY_BASEBOARD_PRODUCT},
{"HardwareID-14", FU_HWIDS_KEY_MANUFACTURER},
{NULL, NULL}};
/* defined for Windows 10 */
for (guint i = 0; msdefined[i].search != NULL; i++) {
if (g_strcmp0(msdefined[i].search, key) == 0) {
key = msdefined[i].replace;
break;
}
}
return key;
}
/**
* fu_hwids_add_smbios_override:
* @self: a #FuHwids
* @key: a key, e.g. %FU_HWIDS_KEY_PRODUCT_SKU
* @value: (nullable): a new value, e.g. `ExampleModel`
*
* Sets SMBIOS override values so you can emulate another system.
*
* This function has no effect if called after fu_hwids_setup()
*
* Since: 1.5.6
**/
void
fu_hwids_add_smbios_override(FuHwids *self, const gchar *key, const gchar *value)
{
g_return_if_fail(FU_IS_HWIDS(self));
g_return_if_fail(key != NULL);
g_hash_table_insert(self->hash_smbios_override, g_strdup(key), g_strdup(value));
}
/**
* fu_hwids_get_replace_values:
* @self: a #FuHwids
* @keys: a key, e.g. `HardwareID-3` or %FU_HWIDS_KEY_PRODUCT_SKU
* @error: (nullable): optional return location for an error
*
* Gets the replacement values for a HardwareID key or plain key.
*
* Returns: a string, e.g. `LENOVO&ThinkPad T440s`, or %NULL for error.
*
* Since: 0.9.3
**/
gchar *
fu_hwids_get_replace_values(FuHwids *self, const gchar *keys, GError **error)
{
g_auto(GStrv) split = NULL;
g_autoptr(GString) str = g_string_new(NULL);
g_return_val_if_fail(FU_IS_HWIDS(self), NULL);
g_return_val_if_fail(keys != NULL, NULL);
g_return_val_if_fail(error == NULL || *error == NULL, NULL);
/* do any replacements */
keys = fu_hwids_get_replace_keys(self, keys);
/* get each part of the HWID */
split = g_strsplit(keys, "&", -1);
for (guint j = 0; split[j] != NULL; j++) {
const gchar *tmp = g_hash_table_lookup(self->hash_dmi_hw, split[j]);
if (tmp == NULL) {
g_set_error(error,
G_IO_ERROR,
G_IO_ERROR_FAILED,
"not available as '%s' unknown",
split[j]);
return NULL;
}
g_string_append_printf(str, "%s&", tmp);
}
if (str->len > 0)
g_string_truncate(str, str->len - 1);
return g_strdup(str->str);
}
/**
* fu_hwids_get_guid:
* @self: a #FuHwids
* @keys: a key, e.g. `HardwareID-3` or %FU_HWIDS_KEY_PRODUCT_SKU
* @error: (nullable): optional return location for an error
*
* Gets the GUID for a specific key.
*
* Returns: a string, or %NULL for error.
*
* Since: 0.9.3
**/
gchar *
fu_hwids_get_guid(FuHwids *self, const gchar *keys, GError **error)
{
g_autofree gchar *tmp = NULL;
g_return_val_if_fail(FU_IS_HWIDS(self), NULL);
g_return_val_if_fail(keys != NULL, NULL);
g_return_val_if_fail(error == NULL || *error == NULL, NULL);
tmp = fu_hwids_get_replace_values(self, keys, error);
if (tmp == NULL)
return NULL;
return fu_hwids_get_guid_for_str(tmp, error);
}
typedef gchar *(*FuHwidsConvertFunc)(FuSmbios *smbios, guint8 type, guint8 offset, GError **error);
static gchar *
fu_hwids_convert_string_table_cb(FuSmbios *smbios, guint8 type, guint8 offset, GError **error)
{
const gchar *tmp;
tmp = fu_smbios_get_string(smbios, type, offset, error);
if (tmp == NULL)
return NULL;
/* ComputerHardwareIds.exe seems to strip spaces */
return fu_common_strstrip(tmp);
}
static gchar *
fu_hwids_convert_padded_integer_cb(FuSmbios *smbios, guint8 type, guint8 offset, GError **error)
{
guint tmp = fu_smbios_get_integer(smbios, type, offset, error);
if (tmp == G_MAXUINT)
return NULL;
return g_strdup_printf("%02x", tmp);
}
static gchar *
fu_hwids_convert_integer_cb(FuSmbios *smbios, guint8 type, guint8 offset, GError **error)
{
guint tmp = fu_smbios_get_integer(smbios, type, offset, error);
if (tmp == G_MAXUINT)
return NULL;
return g_strdup_printf("%x", tmp);
}
/**
* fu_hwids_setup:
* @self: a #FuHwids
* @smbios: (nullable): a #FuSmbios
* @error: (nullable): optional return location for an error
*
* Reads all the SMBIOS values from the hardware.
*
* Returns: %TRUE for success
*
* Since: 0.9.3
**/
gboolean
fu_hwids_setup(FuHwids *self, FuSmbios *smbios, GError **error)
{
struct {
const gchar *key;
guint8 type;
guint8 offset;
FuHwidsConvertFunc func;
} map[] = {{FU_HWIDS_KEY_MANUFACTURER,
FU_SMBIOS_STRUCTURE_TYPE_SYSTEM,
0x04,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_ENCLOSURE_KIND,
FU_SMBIOS_STRUCTURE_TYPE_CHASSIS,
0x05,
fu_hwids_convert_integer_cb},
{FU_HWIDS_KEY_FAMILY,
FU_SMBIOS_STRUCTURE_TYPE_SYSTEM,
0x1a,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_PRODUCT_NAME,
FU_SMBIOS_STRUCTURE_TYPE_SYSTEM,
0x05,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_PRODUCT_SKU,
FU_SMBIOS_STRUCTURE_TYPE_SYSTEM,
0x19,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_BIOS_VENDOR,
FU_SMBIOS_STRUCTURE_TYPE_BIOS,
0x04,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_BIOS_VERSION,
FU_SMBIOS_STRUCTURE_TYPE_BIOS,
0x05,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_BIOS_MAJOR_RELEASE,
FU_SMBIOS_STRUCTURE_TYPE_BIOS,
0x14,
fu_hwids_convert_padded_integer_cb},
{FU_HWIDS_KEY_BIOS_MINOR_RELEASE,
FU_SMBIOS_STRUCTURE_TYPE_BIOS,
0x15,
fu_hwids_convert_padded_integer_cb},
{FU_HWIDS_KEY_FIRMWARE_MAJOR_RELEASE,
FU_SMBIOS_STRUCTURE_TYPE_BIOS,
0x16,
fu_hwids_convert_padded_integer_cb},
{FU_HWIDS_KEY_FIRMWARE_MINOR_RELEASE,
FU_SMBIOS_STRUCTURE_TYPE_BIOS,
0x17,
fu_hwids_convert_padded_integer_cb},
{FU_HWIDS_KEY_BASEBOARD_MANUFACTURER,
FU_SMBIOS_STRUCTURE_TYPE_BASEBOARD,
0x04,
fu_hwids_convert_string_table_cb},
{FU_HWIDS_KEY_BASEBOARD_PRODUCT,
FU_SMBIOS_STRUCTURE_TYPE_BASEBOARD,
0x05,
fu_hwids_convert_string_table_cb},
{NULL, 0x00, 0x00, NULL}};
g_return_val_if_fail(FU_IS_HWIDS(self), FALSE);
g_return_val_if_fail(FU_IS_SMBIOS(smbios) || smbios == NULL, FALSE);
g_return_val_if_fail(error == NULL || *error == NULL, FALSE);
/* get all DMI data */
for (guint i = 0; map[i].key != NULL; i++) {
const gchar *contents_hdr = NULL;
g_autofree gchar *contents = NULL;
g_autofree gchar *contents_safe = NULL;
g_autoptr(GError) error_local = NULL;
/* get the data from a SMBIOS table unless an override exists */
if (g_hash_table_lookup_extended(self->hash_smbios_override,
map[i].key,
NULL,
(gpointer *)&contents_hdr)) {
if (contents_hdr == NULL) {
g_debug("ignoring %s", map[i].key);
continue;
}
} else if (smbios != NULL) {
contents = map[i].func(smbios, map[i].type, map[i].offset, &error_local);
if (contents == NULL) {
g_debug("ignoring %s: %s", map[i].key, error_local->message);
continue;
}
contents_hdr = contents;
} else {
g_debug("ignoring %s", map[i].key);
continue;
}
g_debug("smbios property %s=%s", map[i].key, contents_hdr);
/* weirdly, remove leading zeros */
while (contents_hdr[0] == '0' && map[i].func != fu_hwids_convert_padded_integer_cb)
contents_hdr++;
g_hash_table_insert(self->hash_dmi_hw,
g_strdup(map[i].key),
g_strdup(contents_hdr));
/* make suitable for display */
contents_safe = g_str_to_ascii(contents_hdr, "C");
g_strdelimit(contents_safe, "\n\r", '\0');
g_strchomp(contents_safe);
g_hash_table_insert(self->hash_dmi_display,
g_strdup(map[i].key),
g_steal_pointer(&contents_safe));
}
/* add GUIDs */
for (guint i = 0; i < 15; i++) {
g_autofree gchar *guid = NULL;
g_autofree gchar *key = NULL;
g_autoptr(GError) error_local = NULL;
/* get the GUID and add to hash */
key = g_strdup_printf("HardwareID-%u", i);
guid = fu_hwids_get_guid(self, key, &error_local);
if (guid == NULL) {
g_debug("%s is not available, %s", key, error_local->message);
continue;
}
g_hash_table_insert(self->hash_guid, g_strdup(guid), GUINT_TO_POINTER(1));
g_ptr_array_add(self->array_guids, g_steal_pointer(&guid));
}
return TRUE;
}
static void
fu_hwids_finalize(GObject *object)
{
FuHwids *self;
g_return_if_fail(FU_IS_HWIDS(object));
self = FU_HWIDS(object);
g_hash_table_unref(self->hash_dmi_hw);
g_hash_table_unref(self->hash_dmi_display);
g_hash_table_unref(self->hash_smbios_override);
g_hash_table_unref(self->hash_guid);
g_ptr_array_unref(self->array_guids);
G_OBJECT_CLASS(fu_hwids_parent_class)->finalize(object);
}
static void
fu_hwids_class_init(FuHwidsClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS(klass);
object_class->finalize = fu_hwids_finalize;
}
static void
fu_hwids_init(FuHwids *self)
{
self->hash_dmi_hw = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
self->hash_dmi_display = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
self->hash_smbios_override = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
self->hash_guid = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, NULL);
self->array_guids = g_ptr_array_new_with_free_func(g_free);
}
/**
* fu_hwids_new:
*
* Creates a new #FuHwids
*
* Since: 0.9.3
**/
FuHwids *
fu_hwids_new(void)
{
FuHwids *self;
self = g_object_new(FU_TYPE_HWIDS, NULL);
return FU_HWIDS(self);
}