mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-29 06:30:19 +00:00

In many plugins we've wanted to use ->prepare_firmware() to parse the firmware ahead of ->detach() and ->write_firmware() but this has the limitation that it can only return a single blob of data. For many devices, multiple binary blobs are required from one parsed image, for instance providing signatures, config and data blobs that have to be pushed to the device in different way. This also means we parse the firmware *before* we ask the user to detach. Break the internal FuDevice API to support these firmware types as they become more popular. This also allows us to move the Intel HEX and SREC parsing out of the dfu plugin as they are used by a few plugins now, and resolving symbols between plugins isn't exactly awesome.
242 lines
5.4 KiB
C
242 lines
5.4 KiB
C
/*
|
|
* Copyright (C) 2015 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
/**
|
|
* SECTION:dfu-element
|
|
* @short_description: Object representing a binary element
|
|
*
|
|
* This object represents an binary blob of data at a specific address.
|
|
*
|
|
* This allows relocatable data segments to be stored in different
|
|
* locations on the device itself.
|
|
*
|
|
* See also: #DfuImage, #DfuFirmware
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
|
|
#include "dfu-common.h"
|
|
#include "dfu-element.h"
|
|
|
|
#include "fwupd-error.h"
|
|
|
|
static void dfu_element_finalize (GObject *object);
|
|
|
|
typedef struct {
|
|
GBytes *contents;
|
|
guint32 target_size;
|
|
guint32 address;
|
|
guint8 padding_value;
|
|
} DfuElementPrivate;
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (DfuElement, dfu_element, G_TYPE_OBJECT)
|
|
#define GET_PRIVATE(o) (dfu_element_get_instance_private (o))
|
|
|
|
static void
|
|
dfu_element_class_init (DfuElementClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
object_class->finalize = dfu_element_finalize;
|
|
}
|
|
|
|
static void
|
|
dfu_element_init (DfuElement *element)
|
|
{
|
|
}
|
|
|
|
static void
|
|
dfu_element_finalize (GObject *object)
|
|
{
|
|
DfuElement *element = DFU_ELEMENT (object);
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
|
|
if (priv->contents != NULL)
|
|
g_bytes_unref (priv->contents);
|
|
|
|
G_OBJECT_CLASS (dfu_element_parent_class)->finalize (object);
|
|
}
|
|
|
|
/**
|
|
* dfu_element_new:
|
|
*
|
|
* Creates a new DFU element object.
|
|
*
|
|
* Return value: a new #DfuElement
|
|
**/
|
|
DfuElement *
|
|
dfu_element_new (void)
|
|
{
|
|
DfuElement *element;
|
|
element = g_object_new (DFU_TYPE_ELEMENT, NULL);
|
|
return element;
|
|
}
|
|
|
|
/**
|
|
* dfu_element_get_contents:
|
|
* @element: a #DfuElement
|
|
*
|
|
* Gets the element data.
|
|
*
|
|
* Return value: (transfer none): element data
|
|
**/
|
|
GBytes *
|
|
dfu_element_get_contents (DfuElement *element)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
g_return_val_if_fail (DFU_IS_ELEMENT (element), NULL);
|
|
return priv->contents;
|
|
}
|
|
|
|
/**
|
|
* dfu_element_get_address:
|
|
* @element: a #DfuElement
|
|
*
|
|
* Gets the offset address of the element.
|
|
*
|
|
* Return value: memory offset value, or 0x00 for unset
|
|
**/
|
|
guint32
|
|
dfu_element_get_address (DfuElement *element)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
g_return_val_if_fail (DFU_IS_ELEMENT (element), 0x00);
|
|
return priv->address;
|
|
}
|
|
|
|
/**
|
|
* dfu_element_set_contents:
|
|
* @element: a #DfuElement
|
|
* @contents: element data
|
|
*
|
|
* Sets the element data.
|
|
**/
|
|
void
|
|
dfu_element_set_contents (DfuElement *element, GBytes *contents)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
g_return_if_fail (DFU_IS_ELEMENT (element));
|
|
g_return_if_fail (contents != NULL);
|
|
if (priv->contents == contents)
|
|
return;
|
|
if (priv->contents != NULL)
|
|
g_bytes_unref (priv->contents);
|
|
priv->contents = g_bytes_ref (contents);
|
|
}
|
|
|
|
/**
|
|
* dfu_element_set_address:
|
|
* @element: a #DfuElement
|
|
* @address: memory offset value
|
|
*
|
|
* Sets the offset address of the element.
|
|
**/
|
|
void
|
|
dfu_element_set_address (DfuElement *element, guint32 address)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
g_return_if_fail (DFU_IS_ELEMENT (element));
|
|
priv->address = address;
|
|
}
|
|
|
|
/**
|
|
* dfu_element_to_string:
|
|
* @element: a #DfuElement
|
|
*
|
|
* Returns a string representaiton of the object.
|
|
*
|
|
* Return value: NULL terminated string, or %NULL for invalid
|
|
**/
|
|
gchar *
|
|
dfu_element_to_string (DfuElement *element)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
GString *str;
|
|
|
|
g_return_val_if_fail (DFU_IS_ELEMENT (element), NULL);
|
|
|
|
str = g_string_new ("");
|
|
g_string_append_printf (str, "address: 0x%02x\n", priv->address);
|
|
if (priv->target_size > 0) {
|
|
g_string_append_printf (str, "target: 0x%04x\n",
|
|
priv->target_size);
|
|
}
|
|
if (priv->contents != NULL) {
|
|
g_string_append_printf (str, "contents: 0x%04x\n",
|
|
(guint32) g_bytes_get_size (priv->contents));
|
|
}
|
|
|
|
g_string_truncate (str, str->len - 1);
|
|
return g_string_free (str, FALSE);
|
|
}
|
|
|
|
/**
|
|
* dfu_element_set_padding_value:
|
|
* @element: a #DfuElement
|
|
* @padding_value: char value, typically 0x00 or 0xff
|
|
*
|
|
* Sets a the value of the padding byte to be used in the function
|
|
* dfu_element_set_target_size().
|
|
**/
|
|
void
|
|
dfu_element_set_padding_value (DfuElement *element, guint8 padding_value)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
g_return_if_fail (DFU_IS_ELEMENT (element));
|
|
priv->padding_value = padding_value;
|
|
}
|
|
|
|
/**
|
|
* dfu_element_set_target_size:
|
|
* @element: a #DfuElement
|
|
* @target_size: size in bytes
|
|
*
|
|
* Sets a target size for the element. If the prepared element is smaller
|
|
* than this then it will be padded up to the required size.
|
|
*
|
|
* If a padding byte other than 0x00 is required then the function
|
|
* dfu_element_set_padding_value() should be used before this function is
|
|
* called.
|
|
**/
|
|
void
|
|
dfu_element_set_target_size (DfuElement *element, guint32 target_size)
|
|
{
|
|
DfuElementPrivate *priv = GET_PRIVATE (element);
|
|
const guint8 *data;
|
|
gsize length;
|
|
guint8 *buf;
|
|
|
|
g_return_if_fail (DFU_IS_ELEMENT (element));
|
|
|
|
/* save for dump */
|
|
priv->target_size = target_size;
|
|
|
|
/* no need to pad */
|
|
if (priv->contents == NULL)
|
|
return;
|
|
if (g_bytes_get_size (priv->contents) >= target_size)
|
|
return;
|
|
|
|
/* reallocate and pad */
|
|
data = g_bytes_get_data (priv->contents, &length);
|
|
buf = g_malloc0 (target_size);
|
|
g_assert (buf != NULL);
|
|
memcpy (buf, data, length);
|
|
|
|
/* set the padding value */
|
|
if (priv->padding_value != 0x00) {
|
|
memset (buf + length,
|
|
priv->padding_value,
|
|
target_size - length);
|
|
}
|
|
|
|
/* replace */
|
|
g_bytes_unref (priv->contents);
|
|
priv->contents = g_bytes_new_take (buf, target_size);
|
|
}
|