/* * Copyright (C) 2020 Richard Hughes * * SPDX-License-Identifier: LGPL-2.1+ */ #include "config.h" #include "fu-common.h" #include "fu-elantp-common.h" #include "fu-elantp-firmware.h" struct _FuElantpFirmware { FuFirmwareClass parent_instance; guint16 module_id; guint16 iap_addr; }; G_DEFINE_TYPE (FuElantpFirmware, fu_elantp_firmware, FU_TYPE_FIRMWARE) /* firmware block update */ #define ETP_IAP_START_ADDR_WRDS 0x0083 guint16 fu_elantp_firmware_get_module_id (FuElantpFirmware *self) { g_return_val_if_fail (FU_IS_ELANTP_FIRMWARE (self), 0); return self->module_id; } guint16 fu_elantp_firmware_get_iap_addr (FuElantpFirmware *self) { g_return_val_if_fail (FU_IS_ELANTP_FIRMWARE (self), 0); return self->iap_addr; } static void fu_elantp_firmware_to_string (FuFirmware *firmware, guint idt, GString *str) { FuElantpFirmware *self = FU_ELANTP_FIRMWARE (firmware); fu_common_string_append_kx (str, idt, "IapAddr", self->iap_addr); fu_common_string_append_kx (str, idt, "ModuleId", self->module_id); } static gboolean fu_elantp_firmware_parse (FuFirmware *firmware, GBytes *fw, guint64 addr_start, guint64 addr_end, FwupdInstallFlags flags, GError **error) { FuElantpFirmware *self = FU_ELANTP_FIRMWARE (firmware); gsize bufsz = 0; guint16 iap_addr_wrds; guint16 module_id_wrds; const guint8 *buf = g_bytes_get_data (fw, &bufsz); g_autoptr(FuFirmwareImage) img = fu_firmware_image_new (fw); /* presumably in words */ if (!fu_common_read_uint16_safe (buf, bufsz, ETP_IAP_START_ADDR_WRDS * 2, &iap_addr_wrds, G_LITTLE_ENDIAN, error)) return FALSE; if (iap_addr_wrds < ETP_IAP_START_ADDR_WRDS || iap_addr_wrds > 0x7FFF) { g_set_error (error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE, "IAP address invalid: 0x%x", iap_addr_wrds); return FALSE; } self->iap_addr = iap_addr_wrds * 2; /* read module ID */ if (!fu_common_read_uint16_safe (buf, bufsz, self->iap_addr, &module_id_wrds, G_LITTLE_ENDIAN, error)) return FALSE; if (module_id_wrds > 0x7FFF) { g_set_error (error, FWUPD_ERROR, FWUPD_ERROR_INVALID_FILE, "module ID address invalid: 0x%x", module_id_wrds); return FALSE; } if (!fu_common_read_uint16_safe (buf, bufsz, module_id_wrds * 2, &self->module_id, G_LITTLE_ENDIAN, error)) return FALSE; /* whole image */ fu_firmware_add_image (firmware, img); return TRUE; } static gboolean fu_elantp_firmware_build (FuFirmware *firmware, XbNode *n, GError **error) { FuElantpFirmware *self = FU_ELANTP_FIRMWARE (firmware); guint64 tmp; /* two simple properties */ tmp = xb_node_query_text_as_uint (n, "module_id", NULL); if (tmp != G_MAXUINT64 && tmp <= G_MAXUINT16) self->module_id = tmp; tmp = xb_node_query_text_as_uint (n, "iap_addr", NULL); if (tmp != G_MAXUINT64 && tmp <= G_MAXUINT16) self->iap_addr = tmp; /* success */ return TRUE; } static GBytes * fu_elantp_firmware_write (FuFirmware *firmware, GError **error) { FuElantpFirmware *self = FU_ELANTP_FIRMWARE (firmware); g_autoptr(GByteArray) buf = g_byte_array_new (); g_autoptr(GBytes) blob = NULL; /* only one image supported */ blob = fu_firmware_get_image_default_bytes (firmware, error); if (blob == NULL) return NULL; /* lets build a simple firmware like this: * ------ 0x0 * HEADER (containing IAP offset and module ID) * ------ ~0x10a * DATA * ------ */ fu_byte_array_set_size (buf, self->iap_addr + 0x2 + 0x2); fu_common_write_uint16 (buf->data + ETP_IAP_START_ADDR_WRDS * 2, self->iap_addr / 2, G_LITTLE_ENDIAN); fu_common_write_uint16 (buf->data + self->iap_addr, (self->iap_addr + 2) / 2, G_LITTLE_ENDIAN); fu_common_write_uint16 (buf->data + self->iap_addr + 0x2, self->module_id, G_LITTLE_ENDIAN); g_byte_array_append (buf, g_bytes_get_data (blob, NULL), g_bytes_get_size (blob)); return g_byte_array_free_to_bytes (g_steal_pointer (&buf)); } static void fu_elantp_firmware_init (FuElantpFirmware *self) { } static void fu_elantp_firmware_class_init (FuElantpFirmwareClass *klass) { FuFirmwareClass *klass_firmware = FU_FIRMWARE_CLASS (klass); klass_firmware->parse = fu_elantp_firmware_parse; klass_firmware->build = fu_elantp_firmware_build; klass_firmware->write = fu_elantp_firmware_write; klass_firmware->to_string = fu_elantp_firmware_to_string; } FuFirmware * fu_elantp_firmware_new (void) { return FU_FIRMWARE (g_object_new (FU_TYPE_ELANTP_FIRMWARE, NULL)); }