mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-22 09:12:32 +00:00
351 lines
11 KiB
C
351 lines
11 KiB
C
/*
|
|
* Copyright (C) 2017-2019 VIA Corporation
|
|
* Copyright (C) 2019-2020 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "fu-firmware-common.h"
|
|
#include "fu-ihex-firmware.h"
|
|
|
|
#include "fu-vli-usbhub-common.h"
|
|
#include "fu-vli-usbhub-device.h"
|
|
#include "fu-vli-usbhub-i2c-common.h"
|
|
#include "fu-vli-usbhub-msp430-device.h"
|
|
|
|
struct _FuVliUsbhubMsp430Device
|
|
{
|
|
FuDevice parent_instance;
|
|
};
|
|
|
|
G_DEFINE_TYPE (FuVliUsbhubMsp430Device, fu_vli_usbhub_msp430_device, FU_TYPE_DEVICE)
|
|
|
|
/* Texas Instruments BSL */
|
|
#define I2C_ADDR_WRITE 0x18
|
|
#define I2C_ADDR_READ 0x19
|
|
|
|
#define I2C_CMD_WRITE 0x32
|
|
#define I2C_CMD_READ_STATUS 0x33
|
|
#define I2C_CMD_UPGRADE 0x34
|
|
#define I2C_CMD_READ_VERSIONS 0x40
|
|
|
|
#define I2C_R_VDR 0xa0 /* read vendor command */
|
|
#define I2C_W_VDR 0xb0 /* write vendor command */
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_device_i2c_read (FuVliUsbhubDevice *self,
|
|
guint8 cmd, guint8 *buf, gsize bufsz,
|
|
GError **error)
|
|
{
|
|
GUsbDevice *usb_device = fu_usb_device_get_dev (FU_USB_DEVICE (self));
|
|
guint16 value = ((guint16) I2C_ADDR_WRITE << 8) | cmd;
|
|
guint16 index = (guint16) I2C_ADDR_READ << 8;
|
|
if (!g_usb_device_control_transfer (usb_device,
|
|
G_USB_DEVICE_DIRECTION_DEVICE_TO_HOST,
|
|
G_USB_DEVICE_REQUEST_TYPE_VENDOR,
|
|
G_USB_DEVICE_RECIPIENT_DEVICE,
|
|
I2C_R_VDR, value, index,
|
|
buf, bufsz, NULL,
|
|
FU_VLI_DEVICE_TIMEOUT,
|
|
NULL, error)) {
|
|
g_prefix_error (error, "failed to read I2C: ");
|
|
return FALSE;
|
|
}
|
|
if (g_getenv ("FWUPD_VLI_USBHUB_VERBOSE") != NULL)
|
|
fu_common_dump_raw (G_LOG_DOMAIN, "I2cReadData", buf, bufsz);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_device_i2c_read_status (FuVliUsbhubDevice *self,
|
|
FuVliUsbhubI2cStatus *status,
|
|
GError **error)
|
|
{
|
|
guint8 buf[1] = { 0xff };
|
|
if (!fu_vli_usbhub_device_i2c_read (self,
|
|
I2C_CMD_READ_STATUS,
|
|
buf, sizeof(buf),
|
|
error))
|
|
return FALSE;
|
|
if (status != NULL)
|
|
*status = buf[0];
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_device_i2c_write_data (FuVliUsbhubDevice *self,
|
|
guint8 disable_start_bit,
|
|
guint8 disable_end_bit,
|
|
const guint8 *buf,
|
|
gsize bufsz,
|
|
GError **error)
|
|
{
|
|
GUsbDevice *usb_device = fu_usb_device_get_dev (FU_USB_DEVICE (self));
|
|
guint16 value = (((guint16) disable_start_bit) << 8) | disable_end_bit;
|
|
if (g_getenv ("FWUPD_VLI_USBHUB_VERBOSE") != NULL)
|
|
fu_common_dump_raw (G_LOG_DOMAIN, "I2cWriteData", buf, bufsz);
|
|
if (!g_usb_device_control_transfer (usb_device,
|
|
G_USB_DEVICE_DIRECTION_HOST_TO_DEVICE,
|
|
G_USB_DEVICE_REQUEST_TYPE_VENDOR,
|
|
G_USB_DEVICE_RECIPIENT_DEVICE,
|
|
I2C_W_VDR, value, 0x0,
|
|
(guint8 *) buf, bufsz, NULL,
|
|
FU_VLI_DEVICE_TIMEOUT,
|
|
NULL, error)) {
|
|
g_prefix_error (error, "failed to write I2C @0x%x: ", value);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_msp430_device_setup (FuDevice *device, GError **error)
|
|
{
|
|
FuVliUsbhubDevice *parent = FU_VLI_USBHUB_DEVICE (fu_device_get_parent (device));
|
|
guint8 buf[11] = { 0x0 };
|
|
g_autofree gchar *version = NULL;
|
|
|
|
/* get versions */
|
|
if (!fu_vli_usbhub_device_i2c_read (parent,
|
|
I2C_CMD_READ_VERSIONS,
|
|
buf, sizeof(buf), error)) {
|
|
g_prefix_error (error, "failed to read versions: ");
|
|
return FALSE;
|
|
}
|
|
if ((buf[0] == 0x00 && buf[1] == 0x00 && buf[2] == 0x00) ||
|
|
(buf[0] == 0xff && buf[1] == 0xff && buf[2] == 0xff)) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_FOUND,
|
|
"no MSP430 device detected");
|
|
return FALSE;
|
|
}
|
|
|
|
/* set version */
|
|
version = g_strdup_printf ("%x.%x", buf[0], buf[1]);
|
|
fu_device_set_version (device, version);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_msp430_device_detach (FuDevice *device, GError **error)
|
|
{
|
|
FuVliUsbhubDevice *parent = FU_VLI_USBHUB_DEVICE (fu_device_get_parent (device));
|
|
FuVliUsbhubI2cStatus status = 0xff;
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
const guint8 buf[] = {
|
|
I2C_ADDR_WRITE,
|
|
I2C_CMD_UPGRADE,
|
|
};
|
|
|
|
/* open device */
|
|
locker = fu_device_locker_new (parent, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
if (!fu_vli_usbhub_device_i2c_write_data (parent, 0, 0, buf, sizeof(buf), error))
|
|
return FALSE;
|
|
|
|
/* avoid power instability by waiting T1 */
|
|
fu_device_set_status (device, FWUPD_STATUS_DEVICE_RESTART);
|
|
fu_device_sleep_with_progress (device, 1); /* seconds */
|
|
|
|
/* check the device came back */
|
|
if (!fu_vli_usbhub_device_i2c_read_status (parent, &status, error)) {
|
|
g_prefix_error (error, "device did not come back after detach");
|
|
return FALSE;
|
|
}
|
|
return fu_vli_usbhub_i2c_check_status (status, error);
|
|
}
|
|
|
|
static FuFirmware *
|
|
fu_vli_usbhub_msp430_device_prepare_firmware (FuDevice *device,
|
|
GBytes *fw,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuFirmware) firmware = fu_ihex_firmware_new ();
|
|
if (!fu_firmware_tokenize (firmware, fw, flags, error))
|
|
return NULL;
|
|
return g_steal_pointer (&firmware);
|
|
}
|
|
|
|
typedef struct {
|
|
guint8 command;
|
|
guint8 buf[0x40];
|
|
gsize bufsz;
|
|
guint8 len;
|
|
} FuVliUsbhubDeviceRequest;
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_msp430_device_write_firmware_cb (FuDevice *device, gpointer user_data, GError **error)
|
|
{
|
|
FuVliUsbhubDeviceRequest *req = (FuVliUsbhubDeviceRequest *) user_data;
|
|
FuVliUsbhubDevice *parent = FU_VLI_USBHUB_DEVICE (fu_device_get_parent (device));
|
|
FuVliUsbhubI2cStatus status = 0xff;
|
|
|
|
g_usleep (5 * 1000);
|
|
if (fu_usb_device_get_spec (FU_USB_DEVICE (parent)) >= 0x0300 ||
|
|
req->bufsz <= 32) {
|
|
if (!fu_vli_usbhub_device_i2c_write_data (parent,
|
|
0, 0,
|
|
req->buf,
|
|
req->bufsz,
|
|
error))
|
|
return FALSE;
|
|
} else {
|
|
/* for U2, hub data buffer <= 32 bytes */
|
|
if (!fu_vli_usbhub_device_i2c_write_data (parent,
|
|
0, 1,
|
|
req->buf,
|
|
32,
|
|
error))
|
|
return FALSE;
|
|
if (!fu_vli_usbhub_device_i2c_write_data (parent,
|
|
1, 0,
|
|
req->buf + 32,
|
|
req->bufsz - 32,
|
|
error))
|
|
return FALSE;
|
|
}
|
|
|
|
/* end of file, no need to check status */
|
|
if (req->len == 0 && req->buf[6] == 0x01 && req->buf[7] == 0xFF)
|
|
return TRUE;
|
|
|
|
/* read data to check status */
|
|
g_usleep (5 * 1000);
|
|
if (!fu_vli_usbhub_device_i2c_read_status (parent, &status, error))
|
|
return FALSE;
|
|
return fu_vli_usbhub_i2c_check_status (status, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_msp430_device_write_firmware (FuDevice *device,
|
|
FuFirmware *firmware,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
FuVliUsbhubDevice *parent = FU_VLI_USBHUB_DEVICE (fu_device_get_parent (device));
|
|
GPtrArray *records = fu_ihex_firmware_get_records (FU_IHEX_FIRMWARE (firmware));
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
g_autoptr(FuDevice) root = NULL;
|
|
|
|
/* open device */
|
|
locker = fu_device_locker_new (parent, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
|
|
/* transfer by I²C write, and check status by I²C read */
|
|
fu_device_set_status (device, FWUPD_STATUS_DEVICE_WRITE);
|
|
for (guint j = 0; j < records->len; j++) {
|
|
FuIhexFirmwareRecord *rcd = g_ptr_array_index (records, j);
|
|
FuVliUsbhubDeviceRequest req = { 0x0 };
|
|
const gchar *line = rcd->buf->str;
|
|
|
|
/* length, 16-bit address, type */
|
|
req.len = rcd->byte_cnt;
|
|
if (req.len >= sizeof(req.buf) - 7) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"line too long; buffer size is 0x%x bytes",
|
|
(guint) sizeof(req.buf));
|
|
return FALSE;
|
|
}
|
|
if (9 + (guint) req.len * 2 > (guint) rcd->buf->len) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
"line %u malformed", rcd->ln);
|
|
return FALSE;
|
|
}
|
|
|
|
/* write each record directly to the hardware */
|
|
req.buf[0] = I2C_ADDR_WRITE;
|
|
req.buf[1] = I2C_CMD_WRITE;
|
|
req.buf[2] = 0x3a; /* ':' */
|
|
req.buf[3] = req.len;
|
|
req.buf[4] = fu_firmware_strparse_uint8 (line + 3);
|
|
req.buf[5] = fu_firmware_strparse_uint8 (line + 5);
|
|
req.buf[6] = fu_firmware_strparse_uint8 (line + 7);
|
|
for (guint8 i = 0; i < req.len; i++)
|
|
req.buf[7 + i] = fu_firmware_strparse_uint8 (line + 9 + (i * 2));
|
|
req.buf[7 + req.len] = fu_firmware_strparse_uint8 (line + 9+ (req.len * 2));
|
|
req.bufsz = req.len + 8;
|
|
|
|
/* retry this if it fails */
|
|
if (!fu_device_retry (device,
|
|
fu_vli_usbhub_msp430_device_write_firmware_cb,
|
|
5, &req, error))
|
|
return FALSE;
|
|
fu_device_set_progress_full (device, (gsize) j, (gsize) records->len);
|
|
}
|
|
|
|
/* the device automatically reboots */
|
|
fu_device_set_status (device, FWUPD_STATUS_DEVICE_RESTART);
|
|
fu_device_set_progress (device, 0);
|
|
|
|
/* as soon as the parent comes back we can query the child */
|
|
root = fu_device_get_root (device);
|
|
fu_device_add_flag (root, FWUPD_DEVICE_FLAG_WAIT_FOR_REPLUG);
|
|
|
|
/* success */
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_vli_usbhub_msp430_device_probe (FuDevice *device, GError **error)
|
|
{
|
|
FuVliDeviceKind device_kind = FU_VLI_DEVICE_KIND_MSP430;
|
|
FuVliUsbhubDevice *parent = FU_VLI_USBHUB_DEVICE (fu_device_get_parent (device));
|
|
g_autofree gchar *instance_id = NULL;
|
|
|
|
fu_device_set_name (device, fu_vli_common_device_kind_to_string (device_kind));
|
|
|
|
/* add instance ID */
|
|
instance_id = g_strdup_printf ("USB\\VID_%04X&PID_%04X&I2C_%s",
|
|
fu_usb_device_get_vid (FU_USB_DEVICE (parent)),
|
|
fu_usb_device_get_pid (FU_USB_DEVICE (parent)),
|
|
fu_vli_common_device_kind_to_string (device_kind));
|
|
fu_device_add_instance_id (device, instance_id);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_vli_usbhub_msp430_device_init (FuVliUsbhubMsp430Device *self)
|
|
{
|
|
fu_device_add_icon (FU_DEVICE (self), "audio-card");
|
|
fu_device_set_protocol (FU_DEVICE (self), "com.vli.i2c");
|
|
fu_device_add_flag (FU_DEVICE (self), FWUPD_DEVICE_FLAG_UPDATABLE);
|
|
fu_device_add_flag (FU_DEVICE (self), FWUPD_DEVICE_FLAG_NO_GUID_MATCHING);
|
|
fu_device_set_version_format (FU_DEVICE (self), FWUPD_VERSION_FORMAT_PAIR);
|
|
fu_device_set_logical_id (FU_DEVICE (self), "I2C");
|
|
fu_device_set_summary (FU_DEVICE (self), "I²C Dock Management Device");
|
|
|
|
/* the MSP device reboot takes down the entire hub for ~60 seconds */
|
|
fu_device_set_remove_delay (FU_DEVICE (self), 120 * 1000);
|
|
}
|
|
|
|
static void
|
|
fu_vli_usbhub_msp430_device_class_init (FuVliUsbhubMsp430DeviceClass *klass)
|
|
{
|
|
FuDeviceClass *klass_device = FU_DEVICE_CLASS (klass);
|
|
klass_device->probe = fu_vli_usbhub_msp430_device_probe;
|
|
klass_device->setup = fu_vli_usbhub_msp430_device_setup;
|
|
klass_device->detach = fu_vli_usbhub_msp430_device_detach;
|
|
klass_device->write_firmware = fu_vli_usbhub_msp430_device_write_firmware;
|
|
klass_device->prepare_firmware = fu_vli_usbhub_msp430_device_prepare_firmware;
|
|
}
|
|
|
|
FuDevice *
|
|
fu_vli_usbhub_msp430_device_new (FuVliUsbhubDevice *parent)
|
|
{
|
|
FuVliUsbhubMsp430Device *self = g_object_new (FU_TYPE_VLI_USBHUB_MSP430_DEVICE,
|
|
"parent", parent,
|
|
NULL);
|
|
return FU_DEVICE (self);
|
|
}
|