mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-25 18:44:54 +00:00
478 lines
11 KiB
C
478 lines
11 KiB
C
/*
|
|
* Copyright (C) 2020 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "fwupd-common-private.h"
|
|
#include "fwupd-enums-private.h"
|
|
#include "fwupd-plugin-private.h"
|
|
|
|
/**
|
|
* FwupdPlugin:
|
|
*
|
|
* A plugin which is used by fwupd to enumerate and update devices.
|
|
*
|
|
* See also: [class@FwupdRelease]
|
|
*/
|
|
|
|
static void
|
|
fwupd_plugin_finalize(GObject *object);
|
|
|
|
typedef struct {
|
|
gchar *name;
|
|
guint64 flags;
|
|
} FwupdPluginPrivate;
|
|
|
|
enum { PROP_0, PROP_NAME, PROP_FLAGS, PROP_LAST };
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE(FwupdPlugin, fwupd_plugin, G_TYPE_OBJECT)
|
|
#define GET_PRIVATE(o) (fwupd_plugin_get_instance_private(o))
|
|
|
|
/**
|
|
* fwupd_plugin_get_name:
|
|
* @self: a #FwupdPlugin
|
|
*
|
|
* Gets the plugin name.
|
|
*
|
|
* Returns: the plugin name, or %NULL if unset
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
const gchar *
|
|
fwupd_plugin_get_name(FwupdPlugin *self)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_val_if_fail(FWUPD_IS_PLUGIN(self), NULL);
|
|
return priv->name;
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_set_name:
|
|
* @self: a #FwupdPlugin
|
|
* @name: the plugin name, e.g. `bios`
|
|
*
|
|
* Sets the plugin name.
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
void
|
|
fwupd_plugin_set_name(FwupdPlugin *self, const gchar *name)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_if_fail(FWUPD_IS_PLUGIN(self));
|
|
g_return_if_fail(name != NULL);
|
|
|
|
/* not changed */
|
|
if (g_strcmp0(priv->name, name) == 0)
|
|
return;
|
|
|
|
g_free(priv->name);
|
|
priv->name = g_strdup(name);
|
|
g_object_notify(G_OBJECT(self), "name");
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_get_flags:
|
|
* @self: a #FwupdPlugin
|
|
*
|
|
* Gets the plugin flags.
|
|
*
|
|
* Returns: plugin flags, or 0 if unset
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
guint64
|
|
fwupd_plugin_get_flags(FwupdPlugin *self)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_val_if_fail(FWUPD_IS_PLUGIN(self), 0);
|
|
return priv->flags;
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_set_flags:
|
|
* @self: a #FwupdPlugin
|
|
* @flags: plugin flags, e.g. %FWUPD_PLUGIN_FLAG_CAPSULES_UNSUPPORTED
|
|
*
|
|
* Sets the plugin flags.
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
void
|
|
fwupd_plugin_set_flags(FwupdPlugin *self, guint64 flags)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_if_fail(FWUPD_IS_PLUGIN(self));
|
|
if (priv->flags == flags)
|
|
return;
|
|
priv->flags = flags;
|
|
g_object_notify(G_OBJECT(self), "flags");
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_add_flag:
|
|
* @self: a #FwupdPlugin
|
|
* @flag: the #FwupdPluginFlags
|
|
*
|
|
* Adds a specific plugin flag to the plugin.
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
void
|
|
fwupd_plugin_add_flag(FwupdPlugin *self, FwupdPluginFlags flag)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_if_fail(FWUPD_IS_PLUGIN(self));
|
|
if (flag == 0)
|
|
return;
|
|
if ((priv->flags & flag) > 0)
|
|
return;
|
|
priv->flags |= flag;
|
|
g_object_notify(G_OBJECT(self), "flags");
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_remove_flag:
|
|
* @self: a #FwupdPlugin
|
|
* @flag: a plugin flag
|
|
*
|
|
* Removes a specific plugin flag from the plugin.
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
void
|
|
fwupd_plugin_remove_flag(FwupdPlugin *self, FwupdPluginFlags flag)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_if_fail(FWUPD_IS_PLUGIN(self));
|
|
if (flag == 0)
|
|
return;
|
|
if ((priv->flags & flag) == 0)
|
|
return;
|
|
priv->flags &= ~flag;
|
|
g_object_notify(G_OBJECT(self), "flags");
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_has_flag:
|
|
* @self: a #FwupdPlugin
|
|
* @flag: a plugin flag
|
|
*
|
|
* Finds if the plugin has a specific plugin flag.
|
|
*
|
|
* Returns: %TRUE if the flag is set
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
gboolean
|
|
fwupd_plugin_has_flag(FwupdPlugin *self, FwupdPluginFlags flag)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_return_val_if_fail(FWUPD_IS_PLUGIN(self), FALSE);
|
|
return (priv->flags & flag) > 0;
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_to_variant:
|
|
* @self: a #FwupdPlugin
|
|
*
|
|
* Serialize the plugin data omitting sensitive fields
|
|
*
|
|
* Returns: the serialized data, or %NULL for error
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
GVariant *
|
|
fwupd_plugin_to_variant(FwupdPlugin *self)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
GVariantBuilder builder;
|
|
|
|
g_return_val_if_fail(FWUPD_IS_PLUGIN(self), NULL);
|
|
|
|
/* create an array with all the metadata in */
|
|
g_variant_builder_init(&builder, G_VARIANT_TYPE_VARDICT);
|
|
if (priv->name != NULL) {
|
|
g_variant_builder_add(&builder,
|
|
"{sv}",
|
|
FWUPD_RESULT_KEY_NAME,
|
|
g_variant_new_string(priv->name));
|
|
}
|
|
if (priv->flags > 0) {
|
|
g_variant_builder_add(&builder,
|
|
"{sv}",
|
|
FWUPD_RESULT_KEY_FLAGS,
|
|
g_variant_new_uint64(priv->flags));
|
|
}
|
|
return g_variant_new("a{sv}", &builder);
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_from_key_value(FwupdPlugin *self, const gchar *key, GVariant *value)
|
|
{
|
|
if (g_strcmp0(key, FWUPD_RESULT_KEY_NAME) == 0) {
|
|
fwupd_plugin_set_name(self, g_variant_get_string(value, NULL));
|
|
return;
|
|
}
|
|
if (g_strcmp0(key, FWUPD_RESULT_KEY_FLAGS) == 0) {
|
|
fwupd_plugin_set_flags(self, g_variant_get_uint64(value));
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
fwupd_pad_kv_str(GString *str, const gchar *key, const gchar *value)
|
|
{
|
|
/* ignore */
|
|
if (key == NULL || value == NULL)
|
|
return;
|
|
g_string_append_printf(str, " %s: ", key);
|
|
for (gsize i = strlen(key); i < 20; i++)
|
|
g_string_append(str, " ");
|
|
g_string_append_printf(str, "%s\n", value);
|
|
}
|
|
|
|
static void
|
|
fwupd_pad_kv_dfl(GString *str, const gchar *key, guint64 plugin_flags)
|
|
{
|
|
g_autoptr(GString) tmp = g_string_new("");
|
|
for (guint i = 0; i < 64; i++) {
|
|
if ((plugin_flags & ((guint64)1 << i)) == 0)
|
|
continue;
|
|
g_string_append_printf(tmp, "%s|", fwupd_plugin_flag_to_string((guint64)1 << i));
|
|
}
|
|
if (tmp->len == 0) {
|
|
g_string_append(tmp, fwupd_plugin_flag_to_string(0));
|
|
} else {
|
|
g_string_truncate(tmp, tmp->len - 1);
|
|
}
|
|
fwupd_pad_kv_str(str, key, tmp->str);
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_to_json:
|
|
* @self: a #FwupdPlugin
|
|
* @builder: a JSON builder
|
|
*
|
|
* Adds a fwupd plugin to a JSON builder
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
void
|
|
fwupd_plugin_to_json(FwupdPlugin *self, JsonBuilder *builder)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
|
|
g_return_if_fail(FWUPD_IS_PLUGIN(self));
|
|
g_return_if_fail(builder != NULL);
|
|
|
|
fwupd_common_json_add_string(builder, FWUPD_RESULT_KEY_NAME, priv->name);
|
|
if (priv->flags != FWUPD_PLUGIN_FLAG_NONE) {
|
|
json_builder_set_member_name(builder, FWUPD_RESULT_KEY_FLAGS);
|
|
json_builder_begin_array(builder);
|
|
for (guint i = 0; i < 64; i++) {
|
|
const gchar *tmp;
|
|
if ((priv->flags & ((guint64)1 << i)) == 0)
|
|
continue;
|
|
tmp = fwupd_plugin_flag_to_string((guint64)1 << i);
|
|
json_builder_add_string_value(builder, tmp);
|
|
}
|
|
json_builder_end_array(builder);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_to_string:
|
|
* @self: a #FwupdPlugin
|
|
*
|
|
* Builds a text representation of the object.
|
|
*
|
|
* Returns: text, or %NULL for invalid
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
gchar *
|
|
fwupd_plugin_to_string(FwupdPlugin *self)
|
|
{
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
GString *str;
|
|
|
|
g_return_val_if_fail(FWUPD_IS_PLUGIN(self), NULL);
|
|
|
|
str = g_string_new(NULL);
|
|
fwupd_pad_kv_str(str, FWUPD_RESULT_KEY_NAME, priv->name);
|
|
fwupd_pad_kv_dfl(str, FWUPD_RESULT_KEY_FLAGS, priv->flags);
|
|
return g_string_free(str, FALSE);
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
|
|
{
|
|
FwupdPlugin *self = FWUPD_PLUGIN(object);
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
switch (prop_id) {
|
|
case PROP_NAME:
|
|
g_value_set_string(value, priv->name);
|
|
break;
|
|
case PROP_FLAGS:
|
|
g_value_set_uint64(value, priv->flags);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
|
|
{
|
|
FwupdPlugin *self = FWUPD_PLUGIN(object);
|
|
switch (prop_id) {
|
|
case PROP_NAME:
|
|
fwupd_plugin_set_name(self, g_value_get_string(value));
|
|
break;
|
|
case PROP_FLAGS:
|
|
fwupd_plugin_set_flags(self, g_value_get_uint64(value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_class_init(FwupdPluginClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS(klass);
|
|
GParamSpec *pspec;
|
|
|
|
object_class->finalize = fwupd_plugin_finalize;
|
|
object_class->get_property = fwupd_plugin_get_property;
|
|
object_class->set_property = fwupd_plugin_set_property;
|
|
|
|
pspec =
|
|
g_param_spec_string("name", NULL, NULL, NULL, G_PARAM_READWRITE | G_PARAM_STATIC_NAME);
|
|
g_object_class_install_property(object_class, PROP_NAME, pspec);
|
|
|
|
pspec = g_param_spec_uint64("flags",
|
|
NULL,
|
|
NULL,
|
|
FWUPD_PLUGIN_FLAG_NONE,
|
|
FWUPD_PLUGIN_FLAG_UNKNOWN,
|
|
FWUPD_PLUGIN_FLAG_NONE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_NAME);
|
|
g_object_class_install_property(object_class, PROP_FLAGS, pspec);
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_init(FwupdPlugin *self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_finalize(GObject *object)
|
|
{
|
|
FwupdPlugin *self = FWUPD_PLUGIN(object);
|
|
FwupdPluginPrivate *priv = GET_PRIVATE(self);
|
|
g_free(priv->name);
|
|
G_OBJECT_CLASS(fwupd_plugin_parent_class)->finalize(object);
|
|
}
|
|
|
|
static void
|
|
fwupd_plugin_set_from_variant_iter(FwupdPlugin *self, GVariantIter *iter)
|
|
{
|
|
GVariant *value;
|
|
const gchar *key;
|
|
while (g_variant_iter_next(iter, "{&sv}", &key, &value)) {
|
|
fwupd_plugin_from_key_value(self, key, value);
|
|
g_variant_unref(value);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_from_variant:
|
|
* @value: the serialized data
|
|
*
|
|
* Creates a new plugin using serialized data.
|
|
*
|
|
* Returns: (transfer full): a new #FwupdPlugin, or %NULL if @value was invalid
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
FwupdPlugin *
|
|
fwupd_plugin_from_variant(GVariant *value)
|
|
{
|
|
FwupdPlugin *self = NULL;
|
|
const gchar *type_string;
|
|
g_autoptr(GVariantIter) iter = NULL;
|
|
|
|
/* format from GetDetails */
|
|
type_string = g_variant_get_type_string(value);
|
|
if (g_strcmp0(type_string, "(a{sv})") == 0) {
|
|
self = fwupd_plugin_new();
|
|
g_variant_get(value, "(a{sv})", &iter);
|
|
fwupd_plugin_set_from_variant_iter(self, iter);
|
|
} else if (g_strcmp0(type_string, "a{sv}") == 0) {
|
|
self = fwupd_plugin_new();
|
|
g_variant_get(value, "a{sv}", &iter);
|
|
fwupd_plugin_set_from_variant_iter(self, iter);
|
|
} else {
|
|
g_warning("type %s not known", type_string);
|
|
}
|
|
return self;
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_array_from_variant:
|
|
* @value: the serialized data
|
|
*
|
|
* Creates an array of new plugins using serialized data.
|
|
*
|
|
* Returns: (transfer container) (element-type FwupdPlugin): plugins, or %NULL if @value was invalid
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
GPtrArray *
|
|
fwupd_plugin_array_from_variant(GVariant *value)
|
|
{
|
|
GPtrArray *array = NULL;
|
|
gsize sz;
|
|
g_autoptr(GVariant) untuple = NULL;
|
|
|
|
array = g_ptr_array_new_with_free_func((GDestroyNotify)g_object_unref);
|
|
untuple = g_variant_get_child_value(value, 0);
|
|
sz = g_variant_n_children(untuple);
|
|
for (guint i = 0; i < sz; i++) {
|
|
FwupdPlugin *self;
|
|
g_autoptr(GVariant) data = NULL;
|
|
data = g_variant_get_child_value(untuple, i);
|
|
self = fwupd_plugin_from_variant(data);
|
|
if (self == NULL)
|
|
continue;
|
|
g_ptr_array_add(array, self);
|
|
}
|
|
return array;
|
|
}
|
|
|
|
/**
|
|
* fwupd_plugin_new:
|
|
*
|
|
* Creates a new plugin.
|
|
*
|
|
* Returns: a new #FwupdPlugin
|
|
*
|
|
* Since: 1.5.0
|
|
**/
|
|
FwupdPlugin *
|
|
fwupd_plugin_new(void)
|
|
{
|
|
FwupdPlugin *self;
|
|
self = g_object_new(FWUPD_TYPE_PLUGIN, NULL);
|
|
return FWUPD_PLUGIN(self);
|
|
}
|