mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-24 00:39:33 +00:00
192 lines
4.9 KiB
C
192 lines
4.9 KiB
C
/*
|
|
* Copyright (C) 2018 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "fu-plugin-vfuncs.h"
|
|
|
|
#include "fu-superio-it85-device.h"
|
|
#include "fu-superio-it89-device.h"
|
|
|
|
#define FU_QUIRKS_SUPERIO_CHIPSETS "SuperioChipsets"
|
|
|
|
static gboolean
|
|
fu_plugin_superio_coldplug_chipset (FuPlugin *plugin, const gchar *chipset, GError **error)
|
|
{
|
|
g_autoptr(FuSuperioDevice) dev = NULL;
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
g_autofree gchar *key = g_strdup_printf ("SuperIO=%s", chipset);
|
|
guint64 id;
|
|
guint64 port;
|
|
|
|
/* get ID we need for the chipset */
|
|
id = fu_plugin_lookup_quirk_by_id_as_uint64 (plugin, key, "Id");
|
|
if (id == 0x0000 || id > 0xffff) {
|
|
g_set_error (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_SUPPORTED,
|
|
"SuperIO chip %s has invalid Id", chipset);
|
|
return FALSE;
|
|
}
|
|
|
|
/* set address */
|
|
port = fu_plugin_lookup_quirk_by_id_as_uint64 (plugin, key, "Port");
|
|
if (port == 0x0 || port > 0xffff) {
|
|
g_set_error (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_SUPPORTED,
|
|
"SuperIO chip %s has invalid Port", chipset);
|
|
return FALSE;
|
|
}
|
|
|
|
/* create IT89xx or IT89xx */
|
|
if (id >> 8 == 0x85) {
|
|
dev = g_object_new (FU_TYPE_SUPERIO_IT85_DEVICE,
|
|
"chipset", chipset,
|
|
"id", id,
|
|
"port", port,
|
|
NULL);
|
|
} else if (id >> 8 == 0x89) {
|
|
dev = g_object_new (FU_TYPE_SUPERIO_IT89_DEVICE,
|
|
"chipset", chipset,
|
|
"id", id,
|
|
"port", port,
|
|
NULL);
|
|
} else {
|
|
g_set_error (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_SUPPORTED,
|
|
"SuperIO chip %s has unsupported Id", chipset);
|
|
return FALSE;
|
|
}
|
|
|
|
/* unlock */
|
|
locker = fu_device_locker_new (dev, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
fu_plugin_device_add (plugin, FU_DEVICE (dev));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_plugin_superio_coldplug_chipsets (FuPlugin *plugin, const gchar *str, GError **error)
|
|
{
|
|
g_auto(GStrv) chipsets = g_strsplit (str, ",", -1);
|
|
for (guint i = 0; chipsets[i] != NULL; i++) {
|
|
if (!fu_plugin_superio_coldplug_chipset (plugin, chipsets[i], error))
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
fu_plugin_init (FuPlugin *plugin)
|
|
{
|
|
fu_plugin_set_build_hash (plugin, FU_BUILD_HASH);
|
|
fu_plugin_add_rule (plugin, FU_PLUGIN_RULE_SUPPORTS_PROTOCOL, "tw.com.ite.superio");
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_coldplug (FuPlugin *plugin, GError **error)
|
|
{
|
|
GPtrArray *hwids = fu_plugin_get_hwids (plugin);
|
|
for (guint i = 0; i < hwids->len; i++) {
|
|
const gchar *tmp;
|
|
const gchar *guid = g_ptr_array_index (hwids, i);
|
|
g_autofree gchar *key = g_strdup_printf ("HwId=%s", guid);
|
|
tmp = fu_plugin_lookup_quirk_by_id (plugin, key, FU_QUIRKS_SUPERIO_CHIPSETS);
|
|
if (tmp == NULL)
|
|
continue;
|
|
if (!fu_plugin_superio_coldplug_chipsets (plugin, tmp, error))
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_verify_attach (FuPlugin *plugin, FuDevice *device, GError **error)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
locker = fu_device_locker_new (device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
return fu_device_attach (device, error);
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_verify_detach (FuPlugin *plugin, FuDevice *device, GError **error)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
if (fu_device_has_flag (device, FWUPD_DEVICE_FLAG_IS_BOOTLOADER))
|
|
return TRUE;
|
|
locker = fu_device_locker_new (device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
return fu_device_detach (device, error);
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_verify (FuPlugin *plugin, FuDevice *device,
|
|
FuPluginVerifyFlags flags, GError **error)
|
|
{
|
|
g_autoptr(GBytes) fw = NULL;
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
GChecksumType checksum_types[] = {
|
|
G_CHECKSUM_SHA1,
|
|
G_CHECKSUM_SHA256,
|
|
0 };
|
|
|
|
/* get data */
|
|
locker = fu_device_locker_new (device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
fw = fu_device_read_firmware (device, error);
|
|
if (fw == NULL)
|
|
return FALSE;
|
|
for (guint i = 0; checksum_types[i] != 0; i++) {
|
|
g_autofree gchar *hash = NULL;
|
|
hash = g_compute_checksum_for_bytes (checksum_types[i], fw);
|
|
fu_device_add_checksum (device, hash);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_update_detach (FuPlugin *plugin, FuDevice *device, GError **error)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
if (fu_device_has_flag (device, FWUPD_DEVICE_FLAG_IS_BOOTLOADER))
|
|
return TRUE;
|
|
locker = fu_device_locker_new (device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
return fu_device_detach (device, error);
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_update_attach (FuPlugin *plugin, FuDevice *device, GError **error)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = fu_device_locker_new (device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
return fu_device_attach (device, error);
|
|
}
|
|
|
|
gboolean
|
|
fu_plugin_update (FuPlugin *plugin,
|
|
FuDevice *device,
|
|
GBytes *blob_fw,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
locker = fu_device_locker_new (device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
return fu_device_write_firmware (device, blob_fw, error);
|
|
}
|