mirror of
https://git.proxmox.com/git/fwupd
synced 2025-06-06 09:25:40 +00:00
430 lines
12 KiB
C
430 lines
12 KiB
C
/*
|
|
* Copyright (C) 2017 Richard Hughes <richard@hughsie.com>
|
|
* Copyright (C) 2021 Daniel Campello <campello@chromium.org>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <libflashrom.h>
|
|
|
|
#include "fu-flashrom-cmos.h"
|
|
#include "fu-flashrom-device.h"
|
|
|
|
/*
|
|
* Flag to determine if the CMOS checksum should be reset after the flash
|
|
* is reprogrammed. This will force the CMOS defaults to be reloaded on
|
|
* the next boot.
|
|
*/
|
|
#define FU_FLASHROM_DEVICE_FLAG_RESET_CMOS (1 << 0)
|
|
|
|
/*
|
|
* Flag to determine if manual ME unlocking by pressing Fn + M is supported.
|
|
*/
|
|
#define FU_FLASHROM_DEVICE_FLAG_FN_M_ME_UNLOCK (1 << 1)
|
|
|
|
struct _FuFlashromDevice {
|
|
FuUdevDevice parent_instance;
|
|
FuIfdRegion region;
|
|
struct flashrom_flashctx *flashctx;
|
|
struct flashrom_layout *layout;
|
|
};
|
|
|
|
G_DEFINE_TYPE(FuFlashromDevice, fu_flashrom_device, FU_TYPE_UDEV_DEVICE)
|
|
|
|
enum { PROP_0, PROP_FLASHCTX, PROP_REGION, PROP_LAST };
|
|
|
|
static gboolean
|
|
fu_flashrom_device_set_quirk_kv(FuDevice *device,
|
|
const gchar *key,
|
|
const gchar *value,
|
|
GError **error)
|
|
{
|
|
if (g_strcmp0(key, "PciBcrAddr") == 0) {
|
|
guint64 tmp = 0;
|
|
if (!fu_common_strtoull_full(value, &tmp, 0, G_MAXUINT32, error))
|
|
return FALSE;
|
|
fu_device_set_metadata_integer(device, "PciBcrAddr", tmp);
|
|
return TRUE;
|
|
}
|
|
g_set_error_literal(error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, "no supported");
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_flashrom_device_probe(FuDevice *device, GError **error)
|
|
{
|
|
const gchar *dev_name = NULL;
|
|
const gchar *sysfs_path = NULL;
|
|
|
|
/* FuUdevDevice->probe */
|
|
if (!FU_DEVICE_CLASS(fu_flashrom_device_parent_class)->probe(device, error))
|
|
return FALSE;
|
|
|
|
sysfs_path = fu_udev_device_get_sysfs_path(FU_UDEV_DEVICE(device));
|
|
if (sysfs_path != NULL) {
|
|
g_autofree gchar *physical_id = NULL;
|
|
physical_id = g_strdup_printf("DEVNAME=%s", sysfs_path);
|
|
fu_device_set_physical_id(device, physical_id);
|
|
}
|
|
dev_name = fu_udev_device_get_sysfs_attr(FU_UDEV_DEVICE(device), "name", NULL);
|
|
if (dev_name != NULL) {
|
|
fu_device_add_instance_id_full(device,
|
|
dev_name,
|
|
FU_DEVICE_INSTANCE_FLAG_ONLY_QUIRKS);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_flashrom_device_open(FuDevice *device, GError **error)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(device);
|
|
struct flashrom_layout *layout;
|
|
|
|
/* get the flash size from the device if not already been quirked */
|
|
if (fu_device_get_firmware_size_max(device) == 0) {
|
|
gsize flash_size = flashrom_flash_getsize(self->flashctx);
|
|
if (flash_size == 0) {
|
|
g_set_error_literal(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"flash size zero");
|
|
return FALSE;
|
|
}
|
|
fu_device_set_firmware_size_max(device, flash_size);
|
|
}
|
|
|
|
if (flashrom_layout_read_from_ifd(&layout, self->flashctx, NULL, 0)) {
|
|
g_set_error_literal(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_READ,
|
|
"failed to read layout from Intel ICH descriptor");
|
|
return FALSE;
|
|
}
|
|
|
|
/* update only one specific region of the flash and do not touch others */
|
|
if (flashrom_layout_include_region(layout, fu_ifd_region_to_string(self->region))) {
|
|
flashrom_layout_release(layout);
|
|
|
|
g_set_error_literal(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"invalid region name");
|
|
return FALSE;
|
|
}
|
|
|
|
/* flashrom_layout_set() doesn't transfer ownership, so we must manage layout's lifetime */
|
|
self->layout = layout;
|
|
flashrom_layout_set(self->flashctx, self->layout);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_flashrom_device_close(FuDevice *device, GError **error)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(device);
|
|
|
|
if (self->layout != NULL) {
|
|
flashrom_layout_release(self->layout);
|
|
self->layout = NULL;
|
|
|
|
flashrom_layout_set(self->flashctx, NULL);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GBytes *
|
|
fu_flashrom_device_dump_firmware(FuDevice *device, FuProgress *progress, GError **error)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(device);
|
|
gint rc;
|
|
gsize bufsz = fu_device_get_firmware_size_max(device);
|
|
g_autofree guint8 *buf = g_malloc0(bufsz);
|
|
|
|
fu_progress_set_status(progress, FWUPD_STATUS_DEVICE_READ);
|
|
rc = flashrom_image_read(self->flashctx, buf, bufsz);
|
|
if (rc != 0) {
|
|
g_set_error(error, FWUPD_ERROR, FWUPD_ERROR_READ, "failed to read flash [%i]", rc);
|
|
return NULL;
|
|
}
|
|
return g_bytes_new_take(g_steal_pointer(&buf), bufsz);
|
|
}
|
|
|
|
static gboolean
|
|
fu_flashrom_device_prepare(FuDevice *device,
|
|
FuProgress *progress,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
g_autofree gchar *firmware_orig = NULL;
|
|
g_autofree gchar *localstatedir = NULL;
|
|
g_autofree gchar *basename = NULL;
|
|
|
|
/* if the original firmware doesn't exist, grab it now */
|
|
basename = g_strdup_printf("flashrom-%s.bin", fu_device_get_id(device));
|
|
localstatedir = fu_common_get_path(FU_PATH_KIND_LOCALSTATEDIR_PKG);
|
|
firmware_orig = g_build_filename(localstatedir, "builder", basename, NULL);
|
|
if (!fu_common_mkdir_parent(firmware_orig, error))
|
|
return FALSE;
|
|
if (!g_file_test(firmware_orig, G_FILE_TEST_EXISTS)) {
|
|
g_autoptr(GBytes) buf = NULL;
|
|
buf = fu_flashrom_device_dump_firmware(device, progress, error);
|
|
if (buf == NULL) {
|
|
g_prefix_error(error, "failed to back up original firmware: ");
|
|
return FALSE;
|
|
}
|
|
if (!fu_common_set_contents_bytes(firmware_orig, buf, error))
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_flashrom_device_write_firmware(FuDevice *device,
|
|
FuFirmware *firmware,
|
|
FuProgress *progress,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(device);
|
|
gsize sz = 0;
|
|
gint rc;
|
|
const guint8 *buf;
|
|
g_autoptr(GBytes) blob_fw = NULL;
|
|
|
|
/* progress */
|
|
fu_progress_set_id(progress, G_STRLOC);
|
|
fu_progress_add_flag(progress, FU_PROGRESS_FLAG_GUESSED);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 90);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_VERIFY, 10);
|
|
|
|
/* read early */
|
|
blob_fw = fu_firmware_get_bytes(firmware, error);
|
|
if (blob_fw == NULL)
|
|
return FALSE;
|
|
buf = g_bytes_get_data(blob_fw, &sz);
|
|
|
|
/* write region */
|
|
if (sz != fu_device_get_firmware_size_max(device)) {
|
|
g_set_error(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"invalid image size 0x%x, expected 0x%x",
|
|
(guint)sz,
|
|
(guint)fu_device_get_firmware_size_max(device));
|
|
return FALSE;
|
|
}
|
|
rc = flashrom_image_write(self->flashctx, (void *)buf, sz, NULL /* refbuffer */);
|
|
if (rc != 0) {
|
|
g_set_error(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_WRITE,
|
|
"image write failed, err=%i",
|
|
rc);
|
|
return FALSE;
|
|
}
|
|
fu_progress_step_done(progress);
|
|
|
|
if (flashrom_image_verify(self->flashctx, (void *)buf, sz)) {
|
|
g_set_error(error, FWUPD_ERROR, FWUPD_ERROR_WRITE, "image verify failed");
|
|
return FALSE;
|
|
}
|
|
fu_progress_step_done(progress);
|
|
|
|
/* Check if CMOS needs a reset */
|
|
if (fu_device_has_private_flag(device, FU_FLASHROM_DEVICE_FLAG_RESET_CMOS)) {
|
|
g_debug("Attempting CMOS Reset");
|
|
if (!fu_flashrom_cmos_reset(error)) {
|
|
g_prefix_error(error, "failed CMOS reset: ");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* success */
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_set_progress(FuDevice *self, FuProgress *progress)
|
|
{
|
|
fu_progress_set_id(progress, G_STRLOC);
|
|
fu_progress_add_flag(progress, FU_PROGRESS_FLAG_GUESSED);
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_RESTART, 0); /* detach */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_WRITE, 100); /* write */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_RESTART, 0); /* attach */
|
|
fu_progress_add_step(progress, FWUPD_STATUS_DEVICE_BUSY, 0); /* reload */
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_init(FuFlashromDevice *self)
|
|
{
|
|
fu_device_add_flag(FU_DEVICE(self), FWUPD_DEVICE_FLAG_INTERNAL);
|
|
fu_device_add_flag(FU_DEVICE(self), FWUPD_DEVICE_FLAG_UPDATABLE);
|
|
fu_device_add_flag(FU_DEVICE(self), FWUPD_DEVICE_FLAG_NEEDS_SHUTDOWN);
|
|
fu_device_add_flag(FU_DEVICE(self), FWUPD_DEVICE_FLAG_REQUIRE_AC);
|
|
fu_device_add_flag(FU_DEVICE(self), FWUPD_DEVICE_FLAG_CAN_VERIFY_IMAGE);
|
|
fu_device_add_protocol(FU_DEVICE(self), "org.flashrom");
|
|
fu_device_add_internal_flag(FU_DEVICE(self), FU_DEVICE_INTERNAL_FLAG_ENSURE_SEMVER);
|
|
fu_device_add_internal_flag(FU_DEVICE(self), FU_DEVICE_INTERNAL_FLAG_MD_SET_SIGNED);
|
|
fu_device_set_physical_id(FU_DEVICE(self), "flashrom");
|
|
fu_device_set_version_format(FU_DEVICE(self), FWUPD_VERSION_FORMAT_PAIR);
|
|
fu_device_add_icon(FU_DEVICE(self), "computer");
|
|
fu_device_register_private_flag(FU_DEVICE(self),
|
|
FU_FLASHROM_DEVICE_FLAG_RESET_CMOS,
|
|
"reset-cmos");
|
|
fu_device_register_private_flag(FU_DEVICE(self),
|
|
FU_FLASHROM_DEVICE_FLAG_FN_M_ME_UNLOCK,
|
|
"fn-m-me-unlock");
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_constructed(GObject *obj)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(obj);
|
|
fu_device_add_instance_id(FU_DEVICE(self), "main-system-firmware");
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(object);
|
|
switch (prop_id) {
|
|
case PROP_FLASHCTX:
|
|
g_value_set_pointer(value, self->flashctx);
|
|
break;
|
|
case PROP_REGION:
|
|
g_value_set_uint(value, self->region);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_set_property(GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(object);
|
|
switch (prop_id) {
|
|
case PROP_FLASHCTX:
|
|
self->flashctx = g_value_get_pointer(value);
|
|
break;
|
|
case PROP_REGION:
|
|
self->region = g_value_get_uint(value);
|
|
fu_device_set_logical_id(FU_DEVICE(self), fu_ifd_region_to_string(self->region));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_finalize(GObject *object)
|
|
{
|
|
FuFlashromDevice *self = FU_FLASHROM_DEVICE(object);
|
|
if (self->layout != NULL)
|
|
flashrom_layout_release(self->layout);
|
|
|
|
G_OBJECT_CLASS(fu_flashrom_device_parent_class)->finalize(object);
|
|
}
|
|
|
|
static void
|
|
fu_flashrom_device_class_init(FuFlashromDeviceClass *klass)
|
|
{
|
|
GParamSpec *pspec;
|
|
GObjectClass *object_class = G_OBJECT_CLASS(klass);
|
|
FuDeviceClass *klass_device = FU_DEVICE_CLASS(klass);
|
|
|
|
object_class->get_property = fu_flashrom_device_get_property;
|
|
object_class->set_property = fu_flashrom_device_set_property;
|
|
|
|
/**
|
|
* FuFlashromDevice:region:
|
|
*
|
|
* The IFD region that's being managed.
|
|
*/
|
|
pspec = g_param_spec_uint("region",
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
G_MAXUINT,
|
|
0,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_NAME);
|
|
g_object_class_install_property(object_class, PROP_REGION, pspec);
|
|
|
|
/**
|
|
* FuFlashromDevice:flashctx:
|
|
*
|
|
* The JSON root member for the device.
|
|
*/
|
|
pspec =
|
|
g_param_spec_pointer("flashctx",
|
|
NULL,
|
|
NULL,
|
|
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_NAME);
|
|
g_object_class_install_property(object_class, PROP_FLASHCTX, pspec);
|
|
|
|
object_class->constructed = fu_flashrom_device_constructed;
|
|
object_class->finalize = fu_flashrom_device_finalize;
|
|
klass_device->set_quirk_kv = fu_flashrom_device_set_quirk_kv;
|
|
klass_device->probe = fu_flashrom_device_probe;
|
|
klass_device->open = fu_flashrom_device_open;
|
|
klass_device->close = fu_flashrom_device_close;
|
|
klass_device->set_progress = fu_flashrom_device_set_progress;
|
|
klass_device->prepare = fu_flashrom_device_prepare;
|
|
klass_device->dump_firmware = fu_flashrom_device_dump_firmware;
|
|
klass_device->write_firmware = fu_flashrom_device_write_firmware;
|
|
}
|
|
|
|
FuDevice *
|
|
fu_flashrom_device_new(FuContext *ctx, struct flashrom_flashctx *flashctx, FuIfdRegion region)
|
|
{
|
|
return FU_DEVICE(g_object_new(FU_TYPE_FLASHROM_DEVICE,
|
|
"context",
|
|
ctx,
|
|
"flashctx",
|
|
flashctx,
|
|
"region",
|
|
region,
|
|
NULL));
|
|
}
|
|
|
|
FuIfdRegion
|
|
fu_flashrom_device_get_region(FuFlashromDevice *self)
|
|
{
|
|
return self->region;
|
|
}
|
|
|
|
gboolean
|
|
fu_flashrom_device_unlock(FuFlashromDevice *self, GError **error)
|
|
{
|
|
if (fu_flashrom_device_get_region(self) == FU_IFD_REGION_ME &&
|
|
fu_device_has_private_flag(FU_DEVICE(self), FU_FLASHROM_DEVICE_FLAG_FN_M_ME_UNLOCK)) {
|
|
g_set_error_literal(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOTHING_TO_DO,
|
|
"\n"
|
|
"ME region should be unlocked manually the following way:\n"
|
|
" 1. Power off your device\n"
|
|
" 2. Press and keep holding Fn + M during the next step\n"
|
|
" 3. Press power on button");
|
|
return FALSE;
|
|
}
|
|
|
|
g_set_error(error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_NOT_SUPPORTED,
|
|
"Unlocking of device %s is not supported",
|
|
fu_device_get_name(FU_DEVICE(self)));
|
|
return FALSE;
|
|
}
|