mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-22 08:30:31 +00:00
177 lines
4.4 KiB
C
177 lines
4.4 KiB
C
/*
|
|
* Copyright (C) 2020 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "fu-common.h"
|
|
|
|
#include "fu-bcm57xx-common.h"
|
|
#include "fu-bcm57xx-dict-image.h"
|
|
|
|
struct _FuBcm57xxDictImage {
|
|
FuFirmwareImage parent_instance;
|
|
guint8 target;
|
|
guint8 kind;
|
|
};
|
|
|
|
G_DEFINE_TYPE (FuBcm57xxDictImage, fu_bcm57xx_dict_image, FU_TYPE_FIRMWARE_IMAGE)
|
|
|
|
static void
|
|
fu_bcm57xx_dict_image_to_string (FuFirmwareImage *image, guint idt, GString *str)
|
|
{
|
|
FuBcm57xxDictImage *self = FU_BCM57XX_DICT_IMAGE (image);
|
|
if (self->target != 0xff)
|
|
fu_common_string_append_kx (str, idt, "Target", self->target);
|
|
if (self->kind != 0xff)
|
|
fu_common_string_append_kx (str, idt, "Kind", self->kind);
|
|
}
|
|
|
|
static gboolean
|
|
fu_bcm57xx_dict_image_parse (FuFirmwareImage *image,
|
|
GBytes *fw,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
g_autoptr(GBytes) fw_nocrc = NULL;
|
|
if ((flags & FWUPD_INSTALL_FLAG_IGNORE_CHECKSUM) == 0) {
|
|
if (!fu_bcm57xx_verify_crc (fw, error))
|
|
return FALSE;
|
|
}
|
|
fw_nocrc = g_bytes_new_from_bytes (fw, 0x0, g_bytes_get_size (fw) - sizeof(guint32));
|
|
fu_firmware_image_set_bytes (image, fw_nocrc);
|
|
return TRUE;
|
|
}
|
|
|
|
static GBytes *
|
|
fu_bcm57xx_dict_image_write (FuFirmwareImage *image, GError **error)
|
|
{
|
|
const guint8 *buf;
|
|
gsize bufsz = 0;
|
|
guint32 crc;
|
|
g_autoptr(GByteArray) blob = NULL;
|
|
g_autoptr(GBytes) fw_nocrc = NULL;
|
|
|
|
/* get the CRC-less data */
|
|
fw_nocrc = fu_firmware_image_get_bytes (image);
|
|
if (fw_nocrc == NULL) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"not supported");
|
|
return NULL;
|
|
}
|
|
|
|
/* add to a mutable buffer */
|
|
buf = g_bytes_get_data (fw_nocrc, &bufsz);
|
|
blob = g_byte_array_sized_new (bufsz + sizeof(guint32));
|
|
g_byte_array_append (blob, buf, bufsz);
|
|
|
|
/* add CRC */
|
|
crc = fu_bcm57xx_nvram_crc (buf, bufsz);
|
|
fu_byte_array_append_uint32 (blob, crc, G_BIG_ENDIAN);
|
|
return g_byte_array_free_to_bytes (g_steal_pointer (&blob));
|
|
}
|
|
|
|
static gboolean
|
|
fu_bcm57xx_dict_image_build (FuFirmwareImage *image, XbNode *n, GError **error)
|
|
{
|
|
FuBcm57xxDictImage *self = FU_BCM57XX_DICT_IMAGE (image);
|
|
guint64 tmp;
|
|
|
|
/* two simple properties */
|
|
tmp = xb_node_query_text_as_uint (n, "kind", NULL);
|
|
if (tmp != G_MAXUINT64 && tmp <= G_MAXUINT8)
|
|
fu_bcm57xx_dict_image_set_kind (self, tmp);
|
|
tmp = xb_node_query_text_as_uint (n, "target", NULL);
|
|
if (tmp != G_MAXUINT64 && tmp <= G_MAXUINT8)
|
|
fu_bcm57xx_dict_image_set_target (self, tmp);
|
|
|
|
/* success */
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_bcm57xx_dict_image_ensure_id (FuBcm57xxDictImage *self)
|
|
{
|
|
g_autofree gchar *id = NULL;
|
|
struct {
|
|
guint8 target;
|
|
guint8 kind;
|
|
const gchar *id;
|
|
} ids[] = {
|
|
{ 0x00, 0x00, "pxe" },
|
|
{ 0x0D, 0x00, "ape" },
|
|
{ 0x09, 0x00, "iscsi1" },
|
|
{ 0x05, 0x00, "iscsi2" },
|
|
{ 0x0b, 0x00, "iscsi3" },
|
|
{ 0x00, 0x01, "cfg1000" },
|
|
{ 0x04, 0x01, "vpd2" },
|
|
{ 0xff, 0xff, NULL }
|
|
};
|
|
if (self->target == 0xff || self->kind == 0xff)
|
|
return;
|
|
for (guint i = 0; ids[i].id != NULL; i++) {
|
|
if (self->target == ids[i].target && self->kind == ids[i].kind) {
|
|
g_debug ("using %s for %02x:%02x",
|
|
ids[i].id, self->target, self->kind);
|
|
fu_firmware_image_set_id (FU_FIRMWARE_IMAGE (self), ids[i].id);
|
|
return;
|
|
}
|
|
}
|
|
id = g_strdup_printf ("dict-%02x-%02x", self->target, self->kind);
|
|
g_warning ("falling back to %s, please report", id);
|
|
fu_firmware_image_set_id (FU_FIRMWARE_IMAGE (self), id);
|
|
}
|
|
|
|
void
|
|
fu_bcm57xx_dict_image_set_target (FuBcm57xxDictImage *self, guint8 target)
|
|
{
|
|
self->target = target;
|
|
fu_bcm57xx_dict_image_ensure_id (self);
|
|
}
|
|
|
|
guint8
|
|
fu_bcm57xx_dict_image_get_target (FuBcm57xxDictImage *self)
|
|
{
|
|
return self->target;
|
|
}
|
|
|
|
void
|
|
fu_bcm57xx_dict_image_set_kind (FuBcm57xxDictImage *self, guint8 kind)
|
|
{
|
|
self->kind = kind;
|
|
fu_bcm57xx_dict_image_ensure_id (self);
|
|
}
|
|
|
|
guint8
|
|
fu_bcm57xx_dict_image_get_kind (FuBcm57xxDictImage *self)
|
|
{
|
|
return self->kind;
|
|
}
|
|
|
|
static void
|
|
fu_bcm57xx_dict_image_init (FuBcm57xxDictImage *self)
|
|
{
|
|
self->target = 0xff;
|
|
self->kind = 0xff;
|
|
}
|
|
|
|
static void
|
|
fu_bcm57xx_dict_image_class_init (FuBcm57xxDictImageClass *klass)
|
|
{
|
|
FuFirmwareImageClass *klass_image = FU_FIRMWARE_IMAGE_CLASS (klass);
|
|
klass_image->parse = fu_bcm57xx_dict_image_parse;
|
|
klass_image->write = fu_bcm57xx_dict_image_write;
|
|
klass_image->build = fu_bcm57xx_dict_image_build;
|
|
klass_image->to_string = fu_bcm57xx_dict_image_to_string;
|
|
}
|
|
|
|
FuFirmwareImage *
|
|
fu_bcm57xx_dict_image_new (void)
|
|
{
|
|
return FU_FIRMWARE_IMAGE (g_object_new (FU_TYPE_BCM57XX_DICT_IMAGE, NULL));
|
|
}
|