fwupd/libfwupdplugin/fu-hwids.c
Richard Hughes a02c1073f2 trivial: Fix up some of the developer docs
And add some missing content as requried.
2021-06-11 09:39:03 +01:00

546 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 <glib.h>
#include <gio/gio.h>
#include <string.h>
#include "fu-common.h"
#include "fu-hwids.h"
#include "fwupd-common.h"
#include "fwupd-error.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);
}