mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-25 22:40:50 +00:00

It's actually quite hard to build a front-end for fwupd at the moment as you're never sure when the progress bar is going to zip back to 0% and start all over again. Some plugins go 0..100% for write, others go 0..100% for erase, then again for write, then *again* for verify. By creating a helper object we can easily split up the progress of the specific task, e.g. write_firmware(). We can encode at the plugin level "the erase takes 50% of the time, the write takes 40% and the read takes 10%". This means we can have a progressbar which goes up just once at a consistent speed.
319 lines
9.5 KiB
C
319 lines
9.5 KiB
C
/*
|
|
* Copyright (C) 2016 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "fu-logitech-hidpp-bootloader-nordic.h"
|
|
#include "fu-logitech-hidpp-common.h"
|
|
|
|
struct _FuLogitechHidPpBootloaderNordic {
|
|
FuLogitechHidPpBootloader parent_instance;
|
|
};
|
|
|
|
G_DEFINE_TYPE(FuLogitechHidPpBootloaderNordic,
|
|
fu_logitech_hidpp_bootloader_nordic,
|
|
FU_TYPE_UNIFYING_BOOTLOADER)
|
|
|
|
static gchar *
|
|
fu_logitech_hidpp_bootloader_nordic_get_hw_platform_id(FuLogitechHidPpBootloader *self,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuLogitechHidPpBootloaderRequest) req =
|
|
fu_logitech_hidpp_bootloader_request_new();
|
|
req->cmd = FU_UNIFYING_BOOTLOADER_CMD_GET_HW_PLATFORM_ID;
|
|
if (!fu_logitech_hidpp_bootloader_request(self, req, error)) {
|
|
g_prefix_error(error, "failed to get HW ID: ");
|
|
return NULL;
|
|
}
|
|
return g_strndup((const gchar *)req->data, req->len);
|
|
}
|
|
|
|
static gchar *
|
|
fu_logitech_hidpp_bootloader_nordic_get_fw_version(FuLogitechHidPpBootloader *self, GError **error)
|
|
{
|
|
guint16 micro;
|
|
|
|
g_autoptr(FuLogitechHidPpBootloaderRequest) req =
|
|
fu_logitech_hidpp_bootloader_request_new();
|
|
req->cmd = FU_UNIFYING_BOOTLOADER_CMD_GET_FW_VERSION;
|
|
if (!fu_logitech_hidpp_bootloader_request(self, req, error)) {
|
|
g_prefix_error(error, "failed to get firmware version: ");
|
|
return NULL;
|
|
}
|
|
|
|
/* RRRxx.yy_Bzzzz
|
|
* 012345678901234*/
|
|
micro = (guint16)fu_logitech_hidpp_buffer_read_uint8((const gchar *)req->data + 10) << 8;
|
|
micro += fu_logitech_hidpp_buffer_read_uint8((const gchar *)req->data + 12);
|
|
return fu_logitech_hidpp_format_version(
|
|
"RQR",
|
|
fu_logitech_hidpp_buffer_read_uint8((const gchar *)req->data + 3),
|
|
fu_logitech_hidpp_buffer_read_uint8((const gchar *)req->data + 6),
|
|
micro);
|
|
}
|
|
|
|
static gboolean
|
|
fu_logitech_hidpp_bootloader_nordic_setup(FuDevice *device, GError **error)
|
|
{
|
|
FuLogitechHidPpBootloader *self = FU_UNIFYING_BOOTLOADER(device);
|
|
g_autofree gchar *hw_platform_id = NULL;
|
|
g_autofree gchar *version_fw = NULL;
|
|
g_autoptr(GError) error_local = NULL;
|
|
|
|
/* FuLogitechHidPpBootloader->setup */
|
|
if (!FU_DEVICE_CLASS(fu_logitech_hidpp_bootloader_nordic_parent_class)
|
|
->setup(device, error))
|
|
return FALSE;
|
|
|
|
/* get MCU */
|
|
hw_platform_id = fu_logitech_hidpp_bootloader_nordic_get_hw_platform_id(self, error);
|
|
if (hw_platform_id == NULL)
|
|
return FALSE;
|
|
g_debug("hw-platform-id=%s", hw_platform_id);
|
|
|
|
/* get firmware version, which is not fatal */
|
|
version_fw = fu_logitech_hidpp_bootloader_nordic_get_fw_version(self, &error_local);
|
|
if (version_fw == NULL) {
|
|
g_warning("failed to get firmware version: %s", error_local->message);
|
|
fu_device_set_version(device, "RQR12.00_B0000");
|
|
} else {
|
|
fu_device_set_version(device, version_fw);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_logitech_hidpp_bootloader_nordic_write_signature(FuLogitechHidPpBootloader *self,
|
|
guint16 addr,
|
|
guint8 len,
|
|
const guint8 *data,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuLogitechHidPpBootloaderRequest) req =
|
|
fu_logitech_hidpp_bootloader_request_new();
|
|
req->cmd = 0xC0;
|
|
req->addr = addr;
|
|
req->len = len;
|
|
memcpy(req->data, data, req->len);
|
|
if (!fu_logitech_hidpp_bootloader_request(self, req, error)) {
|
|
g_prefix_error(error, "failed to write sig @0x%02x: ", addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_WRITE_RAM_BUFFER_INVALID_ADDR) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to write @%04x: signature is too big",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_logitech_hidpp_bootloader_nordic_write(FuLogitechHidPpBootloader *self,
|
|
guint16 addr,
|
|
guint8 len,
|
|
const guint8 *data,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuLogitechHidPpBootloaderRequest) req =
|
|
fu_logitech_hidpp_bootloader_request_new();
|
|
req->cmd = FU_UNIFYING_BOOTLOADER_CMD_WRITE;
|
|
req->addr = addr;
|
|
req->len = len;
|
|
if (req->len > 28) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to write @%04x: data length too large %02x",
|
|
addr,
|
|
req->len);
|
|
return FALSE;
|
|
}
|
|
memcpy(req->data, data, req->len);
|
|
if (!fu_logitech_hidpp_bootloader_request(self, req, error)) {
|
|
g_prefix_error(error, "failed to transfer fw @0x%02x: ", addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_WRITE_INVALID_ADDR) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to write @%04x: invalid address",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_WRITE_VERIFY_FAIL) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to write @%04x: failed to verify flash content",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_WRITE_NONZERO_START) {
|
|
g_debug("wrote %d bytes at address %04x, value %02x",
|
|
req->len,
|
|
req->addr,
|
|
req->data[0]);
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to write @%04x: only 1 byte write of 0xff supported",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_WRITE_INVALID_CRC) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to write @%04x: invalid CRC",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_logitech_hidpp_bootloader_nordic_erase(FuLogitechHidPpBootloader *self,
|
|
guint16 addr,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuLogitechHidPpBootloaderRequest) req =
|
|
fu_logitech_hidpp_bootloader_request_new();
|
|
req->cmd = FU_UNIFYING_BOOTLOADER_CMD_ERASE_PAGE;
|
|
req->addr = addr;
|
|
req->len = 0x01;
|
|
if (!fu_logitech_hidpp_bootloader_request(self, req, error)) {
|
|
g_prefix_error(error, "failed to erase fw @0x%02x: ", addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_ERASE_PAGE_INVALID_ADDR) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to erase @%04x: invalid page",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
if (req->cmd == FU_UNIFYING_BOOTLOADER_CMD_ERASE_PAGE_NONZERO_START) {
|
|
g_set_error(error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"failed to erase @%04x: byte 0x00 is not 0xff",
|
|
addr);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_logitech_hidpp_bootloader_nordic_write_firmware(FuDevice *device,
|
|
FuFirmware *firmware,
|
|
FuProgress *progress,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
FuLogitechHidPpBootloader *self = FU_UNIFYING_BOOTLOADER(device);
|
|
const FuLogitechHidPpBootloaderRequest *payload;
|
|
guint16 addr;
|
|
g_autoptr(GBytes) fw = NULL;
|
|
g_autoptr(GPtrArray) reqs = NULL;
|
|
|
|
/* progress */
|
|
fu_progress_set_id(progress, G_STRLOC);
|
|
if (fu_device_has_private_flag(device, FU_LOGITECH_HIDPP_BOOTLOADER_FLAG_IS_SIGNED)) {
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_ERASE, 4);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 13);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 1); /* block 0 */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 82); /* reset vector */
|
|
} else {
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_ERASE, 22);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 72);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 1); /* block 0 */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 6); /* reset vector */
|
|
}
|
|
|
|
/* get default image */
|
|
fw = fu_firmware_get_bytes(firmware, error);
|
|
if (fw == NULL)
|
|
return FALSE;
|
|
|
|
/* erase firmware pages up to the bootloader */
|
|
fu_progress_set_status(progress, FWUPD_STATUS_DEVICE_ERASE);
|
|
for (addr = fu_logitech_hidpp_bootloader_get_addr_lo(self);
|
|
addr < fu_logitech_hidpp_bootloader_get_addr_hi(self);
|
|
addr += fu_logitech_hidpp_bootloader_get_blocksize(self)) {
|
|
if (!fu_logitech_hidpp_bootloader_nordic_erase(self, addr, error))
|
|
return FALSE;
|
|
}
|
|
fu_progress_step_done(progress);
|
|
|
|
/* transfer payload */
|
|
reqs = fu_logitech_hidpp_bootloader_parse_requests(self, fw, error);
|
|
if (reqs == NULL)
|
|
return FALSE;
|
|
fu_progress_set_status(progress, FWUPD_STATUS_DEVICE_WRITE);
|
|
for (guint i = 1; i < reqs->len; i++) {
|
|
gboolean res;
|
|
payload = g_ptr_array_index(reqs, i);
|
|
|
|
if (payload->cmd == FU_UNIFYING_BOOTLOADER_CMD_WRITE_SIGNATURE) {
|
|
res = fu_logitech_hidpp_bootloader_nordic_write_signature(self,
|
|
payload->addr,
|
|
payload->len,
|
|
payload->data,
|
|
error);
|
|
} else {
|
|
res = fu_logitech_hidpp_bootloader_nordic_write(self,
|
|
payload->addr,
|
|
payload->len,
|
|
payload->data,
|
|
error);
|
|
}
|
|
|
|
if (!res)
|
|
return FALSE;
|
|
fu_progress_set_percentage_full(fu_progress_get_child(progress), i + 1, reqs->len);
|
|
}
|
|
fu_progress_step_done(progress);
|
|
|
|
/* send the first managed packet last, excluding the reset vector */
|
|
payload = g_ptr_array_index(reqs, 0);
|
|
if (!fu_logitech_hidpp_bootloader_nordic_write(self,
|
|
payload->addr + 1,
|
|
payload->len - 1,
|
|
payload->data + 1,
|
|
error))
|
|
return FALSE;
|
|
fu_progress_step_done(progress);
|
|
|
|
/* reset vector */
|
|
if (!fu_logitech_hidpp_bootloader_nordic_write(self, 0x0000, 0x01, payload->data, error))
|
|
return FALSE;
|
|
fu_progress_step_done(progress);
|
|
|
|
/* success! */
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_logitech_hidpp_bootloader_nordic_class_init(FuLogitechHidPpBootloaderNordicClass *klass)
|
|
{
|
|
FuDeviceClass *klass_device = FU_DEVICE_CLASS(klass);
|
|
klass_device->write_firmware = fu_logitech_hidpp_bootloader_nordic_write_firmware;
|
|
klass_device->setup = fu_logitech_hidpp_bootloader_nordic_setup;
|
|
}
|
|
|
|
static void
|
|
fu_logitech_hidpp_bootloader_nordic_init(FuLogitechHidPpBootloaderNordic *self)
|
|
{
|
|
}
|