mirror of
https://git.proxmox.com/git/fwupd
synced 2025-06-05 06:51:40 +00:00

This allows creating the silo when starting the engine with custom plugin keys such as WacomI2cFlashBaseAddr. If we move the plugin initialization earlier then we don't get the HwID matches, so we really do have to split this into a 4-stage startup, e.g. ->load(), ->init(), ->startup() and ->coldplug().
175 lines
5.1 KiB
C
175 lines
5.1 KiB
C
/*
|
|
* Copyright (C) 2017 Mario Limonciello <mario.limonciello@dell.com>
|
|
* Copyright (C) 2017 Peichen Huang <peichenhuang@tw.synaptics.com>
|
|
* Copyright (C) 2017 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <fwupdplugin.h>
|
|
|
|
#include "fu-synaptics-mst-common.h"
|
|
#include "fu-synaptics-mst-device.h"
|
|
#include "fu-synaptics-mst-firmware.h"
|
|
|
|
#define FU_SYNAPTICS_MST_DRM_REPLUG_DELAY 5 /* s */
|
|
|
|
struct FuPluginData {
|
|
GPtrArray *devices;
|
|
guint drm_changed_id;
|
|
};
|
|
|
|
static void
|
|
fu_plugin_synaptics_mst_device_rescan(FuPlugin *plugin, FuDevice *device)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
g_autoptr(GError) error_local = NULL;
|
|
|
|
/* open fd */
|
|
locker = fu_device_locker_new(device, &error_local);
|
|
if (locker == NULL) {
|
|
g_debug("failed to open device %s: %s",
|
|
fu_device_get_logical_id(device),
|
|
error_local->message);
|
|
return;
|
|
}
|
|
if (!fu_device_rescan(device, &error_local)) {
|
|
g_debug("no device found on %s: %s",
|
|
fu_device_get_logical_id(device),
|
|
error_local->message);
|
|
if (fu_device_has_flag(device, FWUPD_DEVICE_FLAG_REGISTERED))
|
|
fu_plugin_device_remove(plugin, device);
|
|
} else {
|
|
fu_plugin_device_add(plugin, device);
|
|
}
|
|
}
|
|
|
|
/* reprobe all existing devices added by this plugin */
|
|
static void
|
|
fu_plugin_synaptics_mst_rescan(FuPlugin *plugin)
|
|
{
|
|
FuPluginData *priv = fu_plugin_get_data(plugin);
|
|
for (guint i = 0; i < priv->devices->len; i++) {
|
|
FuDevice *device = FU_DEVICE(g_ptr_array_index(priv->devices, i));
|
|
fu_plugin_synaptics_mst_device_rescan(plugin, device);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
fu_plugin_synaptics_mst_rescan_cb(gpointer user_data)
|
|
{
|
|
FuPlugin *plugin = FU_PLUGIN(user_data);
|
|
FuPluginData *priv = fu_plugin_get_data(plugin);
|
|
fu_plugin_synaptics_mst_rescan(plugin);
|
|
priv->drm_changed_id = 0;
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_plugin_synaptics_mst_backend_device_changed(FuPlugin *plugin, FuDevice *device, GError **error)
|
|
{
|
|
FuPluginData *priv = fu_plugin_get_data(plugin);
|
|
|
|
/* interesting device? */
|
|
if (!FU_IS_UDEV_DEVICE(device))
|
|
return TRUE;
|
|
if (g_strcmp0(fu_udev_device_get_subsystem(FU_UDEV_DEVICE(device)), "drm") != 0)
|
|
return TRUE;
|
|
|
|
/* recoldplug all drm_dp_aux_dev devices after a *long* delay */
|
|
if (priv->drm_changed_id != 0)
|
|
g_source_remove(priv->drm_changed_id);
|
|
priv->drm_changed_id = g_timeout_add_seconds(FU_SYNAPTICS_MST_DRM_REPLUG_DELAY,
|
|
fu_plugin_synaptics_mst_rescan_cb,
|
|
plugin);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_plugin_synaptics_mst_backend_device_added(FuPlugin *plugin, FuDevice *device, GError **error)
|
|
{
|
|
FuContext *ctx = fu_plugin_get_context(plugin);
|
|
FuPluginData *priv = fu_plugin_get_data(plugin);
|
|
g_autoptr(FuDeviceLocker) locker = NULL;
|
|
g_autoptr(FuSynapticsMstDevice) dev = NULL;
|
|
|
|
/* interesting device? */
|
|
if (!FU_IS_UDEV_DEVICE(device))
|
|
return TRUE;
|
|
|
|
dev = fu_synaptics_mst_device_new(FU_UDEV_DEVICE(device));
|
|
locker = fu_device_locker_new(dev, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
|
|
/* for SynapticsMstDeviceKind=system devices */
|
|
fu_synaptics_mst_device_set_system_type(
|
|
FU_SYNAPTICS_MST_DEVICE(dev),
|
|
fu_context_get_hwid_value(ctx, FU_HWIDS_KEY_PRODUCT_SKU));
|
|
|
|
/* this might fail if there is nothing connected */
|
|
fu_plugin_synaptics_mst_device_rescan(plugin, FU_DEVICE(dev));
|
|
g_ptr_array_add(priv->devices, g_steal_pointer(&dev));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
fu_plugin_synaptics_mst_write_firmware(FuPlugin *plugin,
|
|
FuDevice *device,
|
|
GBytes *blob_fw,
|
|
FuProgress *progress,
|
|
FwupdInstallFlags flags,
|
|
GError **error)
|
|
{
|
|
g_autoptr(FuDeviceLocker) locker = fu_device_locker_new(device, error);
|
|
if (locker == NULL)
|
|
return FALSE;
|
|
if (!fu_device_write_firmware(device, blob_fw, progress, flags, error))
|
|
return FALSE;
|
|
if (!fu_device_has_flag(device, FWUPD_DEVICE_FLAG_SKIPS_RESTART))
|
|
fu_plugin_device_remove(plugin, device);
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_plugin_synaptics_mst_init(FuPlugin *plugin)
|
|
{
|
|
FuPluginData *priv = fu_plugin_alloc_data(plugin, sizeof(FuPluginData));
|
|
|
|
/* devices added by this plugin */
|
|
priv->devices = g_ptr_array_new_with_free_func((GDestroyNotify)g_object_unref);
|
|
|
|
fu_plugin_add_udev_subsystem(plugin, "drm"); /* used for uevent only */
|
|
fu_plugin_add_udev_subsystem(plugin, "drm_dp_aux_dev");
|
|
fu_plugin_add_firmware_gtype(plugin, NULL, FU_TYPE_SYNAPTICS_MST_FIRMWARE);
|
|
}
|
|
|
|
static void
|
|
fu_plugin_synaptics_mst_load(FuContext *ctx)
|
|
{
|
|
fu_context_add_quirk_key(ctx, "SynapticsMstDeviceKind");
|
|
}
|
|
|
|
static void
|
|
fu_plugin_synaptics_mst_destroy(FuPlugin *plugin)
|
|
{
|
|
FuPluginData *priv = fu_plugin_get_data(plugin);
|
|
if (priv->drm_changed_id != 0)
|
|
g_source_remove(priv->drm_changed_id);
|
|
g_ptr_array_unref(priv->devices);
|
|
}
|
|
|
|
void
|
|
fu_plugin_init_vfuncs(FuPluginVfuncs *vfuncs)
|
|
{
|
|
vfuncs->build_hash = FU_BUILD_HASH;
|
|
vfuncs->load = fu_plugin_synaptics_mst_load;
|
|
vfuncs->init = fu_plugin_synaptics_mst_init;
|
|
vfuncs->destroy = fu_plugin_synaptics_mst_destroy;
|
|
vfuncs->write_firmware = fu_plugin_synaptics_mst_write_firmware;
|
|
vfuncs->backend_device_added = fu_plugin_synaptics_mst_backend_device_added;
|
|
vfuncs->backend_device_changed = fu_plugin_synaptics_mst_backend_device_changed;
|
|
}
|