mirror of
https://git.proxmox.com/git/fwupd
synced 2025-06-06 11:14:14 +00:00

Provide a device instance builder that allows plugins to easily create multiple instance IDs based on parent attributes. Also fix a lot of the instance ID orders, so that we add more generic IDs first, and more specific IDs after.
199 lines
5.3 KiB
C
199 lines
5.3 KiB
C
/*#
|
|
* Copyright (C) 2021 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <fwupdplugin.h>
|
|
|
|
#include "fu-cfu-module.h"
|
|
|
|
struct _FuCfuModule {
|
|
FuDevice parent_instance;
|
|
guint8 component_id;
|
|
guint8 bank;
|
|
};
|
|
|
|
G_DEFINE_TYPE(FuCfuModule, fu_cfu_module, FU_TYPE_DEVICE)
|
|
|
|
static void
|
|
fu_cfu_module_to_string(FuDevice *device, guint idt, GString *str)
|
|
{
|
|
FuCfuModule *self = FU_CFU_MODULE(device);
|
|
fu_common_string_append_kx(str, idt, "ComponentId", self->component_id);
|
|
fu_common_string_append_kx(str, idt, "Bank", self->bank);
|
|
}
|
|
|
|
guint8
|
|
fu_cfu_module_get_component_id(FuCfuModule *self)
|
|
{
|
|
return self->component_id;
|
|
}
|
|
|
|
guint8
|
|
fu_cfu_module_get_bank(FuCfuModule *self)
|
|
{
|
|
return self->bank;
|
|
}
|
|
|
|
gboolean
|
|
fu_cfu_module_setup(FuCfuModule *self, const guint8 *buf, gsize bufsz, gsize offset, GError **error)
|
|
{
|
|
FuDevice *device = FU_DEVICE(self);
|
|
FuDevice *parent = fu_device_get_proxy(device);
|
|
guint32 version_raw = 0;
|
|
guint8 tmp = 0;
|
|
g_autofree gchar *logical_id = NULL;
|
|
g_autofree gchar *version = NULL;
|
|
|
|
/* component ID */
|
|
if (!fu_common_read_uint8_safe(buf, bufsz, offset + 0x5, &self->component_id, error))
|
|
return FALSE;
|
|
|
|
/* these GUIDs may cause the name or version-format to be overwritten */
|
|
fu_device_add_instance_u8(device, "CID", self->component_id);
|
|
if (!fu_device_build_instance_id(device, error, "HIDRAW", "VEN", "DEV", NULL))
|
|
return FALSE;
|
|
if (!fu_device_build_instance_id(device, error, "HIDRAW", "VEN", "DEV", "CID", NULL))
|
|
return FALSE;
|
|
|
|
/* bank */
|
|
if (!fu_common_read_uint8_safe(buf, bufsz, offset + 0x4, &tmp, error))
|
|
return FALSE;
|
|
self->bank = tmp & 0b11;
|
|
fu_device_add_instance_u4(device, "BANK", self->bank);
|
|
if (!fu_device_build_instance_id(device,
|
|
error,
|
|
"HIDRAW",
|
|
"VEN",
|
|
"DEV",
|
|
"CID",
|
|
"BANK",
|
|
NULL))
|
|
return FALSE;
|
|
|
|
/* set name, if not already set using a quirk */
|
|
if (fu_device_get_name(device) == NULL) {
|
|
g_autofree gchar *name = NULL;
|
|
name = g_strdup_printf("%s (0x%02X:0x%02x)",
|
|
fu_device_get_name(parent),
|
|
self->component_id,
|
|
self->bank);
|
|
fu_device_set_name(device, name);
|
|
}
|
|
|
|
/* version */
|
|
if (!fu_common_read_uint32_safe(buf, bufsz, offset, &version_raw, G_LITTLE_ENDIAN, error))
|
|
return FALSE;
|
|
fu_device_set_version_raw(device, version_raw);
|
|
version = fu_common_version_from_uint32(version_raw, fu_device_get_version_format(device));
|
|
fu_device_set_version(device, version);
|
|
|
|
/* logical ID */
|
|
logical_id = g_strdup_printf("CID:0x%02x,BANK:0x%02x", self->component_id, self->bank);
|
|
fu_device_set_logical_id(device, logical_id);
|
|
|
|
/* success */
|
|
return TRUE;
|
|
}
|
|
|
|
static FuFirmware *
|
|
fu_cfu_module_prepare_firmware(FuDevice *device,
|
|
GBytes *fw,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuFirmware) firmware = fu_firmware_new();
|
|
g_autoptr(FuFirmware) offer = fu_cfu_offer_new();
|
|
g_autoptr(FuFirmware) payload = fu_cfu_payload_new();
|
|
g_autoptr(GBytes) fw_offset = NULL;
|
|
|
|
/* offer */
|
|
if (!fu_firmware_parse(offer, fw, flags, error))
|
|
return NULL;
|
|
fu_firmware_set_id(offer, FU_FIRMWARE_ID_HEADER);
|
|
fu_firmware_add_image(firmware, offer);
|
|
|
|
/* payload */
|
|
fw_offset = fu_common_bytes_new_offset(fw, 0x10, g_bytes_get_size(fw) - 0x10, error);
|
|
if (fw_offset == NULL)
|
|
return NULL;
|
|
if (!fu_firmware_parse(payload, fw_offset, flags, error))
|
|
return NULL;
|
|
fu_firmware_set_id(payload, FU_FIRMWARE_ID_PAYLOAD);
|
|
fu_firmware_add_image(firmware, payload);
|
|
|
|
/* success */
|
|
return g_steal_pointer(&firmware);
|
|
}
|
|
|
|
static gboolean
|
|
fu_cfu_module_write_firmware(FuDevice *device,
|
|
FuFirmware *firmware,
|
|
FuProgress *progress,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
FuDevice *proxy;
|
|
g_autoptr(GBytes) fw = NULL;
|
|
|
|
/* get the whole image */
|
|
fw = fu_firmware_get_bytes(firmware, error);
|
|
if (fw == NULL)
|
|
return FALSE;
|
|
|
|
/* process by the parent */
|
|
proxy = fu_device_get_proxy(device);
|
|
if (proxy == NULL) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_SUPPORTED,
|
|
"no proxy device assigned");
|
|
return FALSE;
|
|
}
|
|
return fu_device_write_firmware(proxy, fw, progress, flags, error);
|
|
}
|
|
|
|
static void
|
|
fu_cfu_module_set_progress(FuDevice *self, FuProgress *progress)
|
|
{
|
|
fu_progress_set_id(progress, G_STRLOC);
|
|
fu_progress_add_flag(progress, FU_PROGRESS_FLAG_GUESSED);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_RESTART, 2); /* detach */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 96); /* write */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_RESTART, 2); /* attach */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_BUSY, 2); /* reload */
|
|
}
|
|
|
|
static void
|
|
fu_cfu_module_init(FuCfuModule *self)
|
|
{
|
|
fu_device_add_protocol(FU_DEVICE(self), "com.microsoft.cfu");
|
|
fu_device_set_version_format(FU_DEVICE(self), FWUPD_VERSION_FORMAT_QUAD);
|
|
}
|
|
|
|
static void
|
|
fu_cfu_module_class_init(FuCfuModuleClass *klass)
|
|
{
|
|
FuDeviceClass *klass_device = FU_DEVICE_CLASS(klass);
|
|
klass_device->to_string = fu_cfu_module_to_string;
|
|
klass_device->prepare_firmware = fu_cfu_module_prepare_firmware;
|
|
klass_device->write_firmware = fu_cfu_module_write_firmware;
|
|
klass_device->set_progress = fu_cfu_module_set_progress;
|
|
}
|
|
|
|
FuCfuModule *
|
|
fu_cfu_module_new(FuDevice *parent)
|
|
{
|
|
FuCfuModule *self;
|
|
self = g_object_new(FU_TYPE_CFU_MODULE,
|
|
"ctx",
|
|
fu_device_get_context(parent),
|
|
"proxy",
|
|
parent,
|
|
NULL);
|
|
return self;
|
|
}
|