Commit 3295814d authored by Mike Waychison's avatar Mike Waychison Committed by Greg Kroah-Hartman

efivars: Parameterize operations.

Instead of letting efivars access struct efi directly when dealing with
variables, use an operations structure.  This allows a later change to
reuse the efivars logic without having to pretend to support everything
in struct efi.
Signed-off-by: default avatarMike Waychison <mikew@google.com>
Cc: Matt Domsch <Matt_Domsch@dell.com>,
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 76b53f7c
...@@ -89,19 +89,26 @@ MODULE_DESCRIPTION("sysfs interface to EFI Variables"); ...@@ -89,19 +89,26 @@ MODULE_DESCRIPTION("sysfs interface to EFI Variables");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_VERSION(EFIVARS_VERSION); MODULE_VERSION(EFIVARS_VERSION);
struct efivar_operations {
efi_get_variable_t *get_variable;
efi_get_next_variable_t *get_next_variable;
efi_set_variable_t *set_variable;
};
struct efivars { struct efivars {
/* /*
* ->lock protects two things: * ->lock protects two things:
* 1) ->list - adds, removals, reads, writes * 1) ->list - adds, removals, reads, writes
* 2) efi.[gs]et_variable() calls. * 2) ops.[gs]et_variable() calls.
* It must not be held when creating sysfs entries or calling kmalloc. * It must not be held when creating sysfs entries or calling kmalloc.
* efi.get_next_variable() is only called from register_efivars(), * ops.get_next_variable() is only called from register_efivars(),
* which is protected by the BKL, so that path is safe. * which is protected by the BKL, so that path is safe.
*/ */
spinlock_t lock; spinlock_t lock;
struct list_head list; struct list_head list;
struct kset *kset; struct kset *kset;
struct bin_attribute *new_var, *del_var; struct bin_attribute *new_var, *del_var;
const struct efivar_operations *ops;
}; };
/* /*
...@@ -182,11 +189,11 @@ get_var_data(struct efivars *efivars, struct efi_variable *var) ...@@ -182,11 +189,11 @@ get_var_data(struct efivars *efivars, struct efi_variable *var)
spin_lock(&efivars->lock); spin_lock(&efivars->lock);
var->DataSize = 1024; var->DataSize = 1024;
status = efi.get_variable(var->VariableName, status = efivars->ops->get_variable(var->VariableName,
&var->VendorGuid, &var->VendorGuid,
&var->Attributes, &var->Attributes,
&var->DataSize, &var->DataSize,
var->Data); var->Data);
spin_unlock(&efivars->lock); spin_unlock(&efivars->lock);
if (status != EFI_SUCCESS) { if (status != EFI_SUCCESS) {
printk(KERN_WARNING "efivars: get_variable() failed 0x%lx!\n", printk(KERN_WARNING "efivars: get_variable() failed 0x%lx!\n",
...@@ -299,11 +306,11 @@ efivar_store_raw(struct efivar_entry *entry, const char *buf, size_t count) ...@@ -299,11 +306,11 @@ efivar_store_raw(struct efivar_entry *entry, const char *buf, size_t count)
} }
spin_lock(&efivars->lock); spin_lock(&efivars->lock);
status = efi.set_variable(new_var->VariableName, status = efivars->ops->set_variable(new_var->VariableName,
&new_var->VendorGuid, &new_var->VendorGuid,
new_var->Attributes, new_var->Attributes,
new_var->DataSize, new_var->DataSize,
new_var->Data); new_var->Data);
spin_unlock(&efivars->lock); spin_unlock(&efivars->lock);
...@@ -446,11 +453,11 @@ static ssize_t efivar_create(struct file *filp, struct kobject *kobj, ...@@ -446,11 +453,11 @@ static ssize_t efivar_create(struct file *filp, struct kobject *kobj,
} }
/* now *really* create the variable via EFI */ /* now *really* create the variable via EFI */
status = efi.set_variable(new_var->VariableName, status = efivars->ops->set_variable(new_var->VariableName,
&new_var->VendorGuid, &new_var->VendorGuid,
new_var->Attributes, new_var->Attributes,
new_var->DataSize, new_var->DataSize,
new_var->Data); new_var->Data);
if (status != EFI_SUCCESS) { if (status != EFI_SUCCESS) {
printk(KERN_WARNING "efivars: set_variable() failed: status=%lx\n", printk(KERN_WARNING "efivars: set_variable() failed: status=%lx\n",
...@@ -511,11 +518,11 @@ static ssize_t efivar_delete(struct file *filp, struct kobject *kobj, ...@@ -511,11 +518,11 @@ static ssize_t efivar_delete(struct file *filp, struct kobject *kobj,
del_var->Attributes = 0; del_var->Attributes = 0;
del_var->DataSize = 0; del_var->DataSize = 0;
status = efi.set_variable(del_var->VariableName, status = efivars->ops->set_variable(del_var->VariableName,
&del_var->VendorGuid, &del_var->VendorGuid,
del_var->Attributes, del_var->Attributes,
del_var->DataSize, del_var->DataSize,
del_var->Data); del_var->Data);
if (status != EFI_SUCCESS) { if (status != EFI_SUCCESS) {
printk(KERN_WARNING "efivars: set_variable() failed: status=%lx\n", printk(KERN_WARNING "efivars: set_variable() failed: status=%lx\n",
...@@ -719,6 +726,7 @@ static void unregister_efivars(struct efivars *efivars) ...@@ -719,6 +726,7 @@ static void unregister_efivars(struct efivars *efivars)
} }
static int register_efivars(struct efivars *efivars, static int register_efivars(struct efivars *efivars,
const struct efivar_operations *ops,
struct kobject *parent_kobj) struct kobject *parent_kobj)
{ {
efi_status_t status = EFI_NOT_FOUND; efi_status_t status = EFI_NOT_FOUND;
...@@ -735,6 +743,7 @@ static int register_efivars(struct efivars *efivars, ...@@ -735,6 +743,7 @@ static int register_efivars(struct efivars *efivars,
spin_lock_init(&efivars->lock); spin_lock_init(&efivars->lock);
INIT_LIST_HEAD(&efivars->list); INIT_LIST_HEAD(&efivars->list);
efivars->ops = ops;
efivars->kset = kset_create_and_add("vars", NULL, parent_kobj); efivars->kset = kset_create_and_add("vars", NULL, parent_kobj);
if (!efivars->kset) { if (!efivars->kset) {
...@@ -751,7 +760,7 @@ static int register_efivars(struct efivars *efivars, ...@@ -751,7 +760,7 @@ static int register_efivars(struct efivars *efivars,
do { do {
variable_name_size = 1024; variable_name_size = 1024;
status = efi.get_next_variable(&variable_name_size, status = ops->get_next_variable(&variable_name_size,
variable_name, variable_name,
&vendor_guid); &vendor_guid);
switch (status) { switch (status) {
...@@ -782,6 +791,7 @@ static int register_efivars(struct efivars *efivars, ...@@ -782,6 +791,7 @@ static int register_efivars(struct efivars *efivars,
} }
static struct efivars __efivars; static struct efivars __efivars;
static struct efivar_operations ops;
/* /*
* For now we register the efi subsystem with the firmware subsystem * For now we register the efi subsystem with the firmware subsystem
...@@ -809,7 +819,10 @@ efivars_init(void) ...@@ -809,7 +819,10 @@ efivars_init(void)
return -ENOMEM; return -ENOMEM;
} }
error = register_efivars(&__efivars, efi_kobj); ops.get_variable = efi.get_variable;
ops.set_variable = efi.set_variable;
ops.get_next_variable = efi.get_next_variable;
error = register_efivars(&__efivars, &ops, efi_kobj);
/* Don't forget the systab entry */ /* Don't forget the systab entry */
error = sysfs_create_group(efi_kobj, &efi_subsys_attr_group); error = sysfs_create_group(efi_kobj, &efi_subsys_attr_group);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment