|  | /* | 
|  | * This module provides an interface to trigger and test firmware loading. | 
|  | * | 
|  | * It is designed to be used for basic evaluation of the firmware loading | 
|  | * subsystem (for example when validating firmware verification). It lacks | 
|  | * any extra dependencies, and will not normally be loaded by the system | 
|  | * unless explicitly requested by name. | 
|  | */ | 
|  |  | 
|  | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | 
|  |  | 
|  | #include <linux/init.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/printk.h> | 
|  | #include <linux/firmware.h> | 
|  | #include <linux/device.h> | 
|  | #include <linux/fs.h> | 
|  | #include <linux/miscdevice.h> | 
|  | #include <linux/slab.h> | 
|  | #include <linux/uaccess.h> | 
|  |  | 
|  | static DEFINE_MUTEX(test_fw_mutex); | 
|  | static const struct firmware *test_firmware; | 
|  |  | 
|  | static ssize_t test_fw_misc_read(struct file *f, char __user *buf, | 
|  | size_t size, loff_t *offset) | 
|  | { | 
|  | ssize_t rc = 0; | 
|  |  | 
|  | mutex_lock(&test_fw_mutex); | 
|  | if (test_firmware) | 
|  | rc = simple_read_from_buffer(buf, size, offset, | 
|  | test_firmware->data, | 
|  | test_firmware->size); | 
|  | mutex_unlock(&test_fw_mutex); | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | static const struct file_operations test_fw_fops = { | 
|  | .owner          = THIS_MODULE, | 
|  | .read           = test_fw_misc_read, | 
|  | }; | 
|  |  | 
|  | static struct miscdevice test_fw_misc_device = { | 
|  | .minor          = MISC_DYNAMIC_MINOR, | 
|  | .name           = "test_firmware", | 
|  | .fops           = &test_fw_fops, | 
|  | }; | 
|  |  | 
|  | static ssize_t trigger_request_store(struct device *dev, | 
|  | struct device_attribute *attr, | 
|  | const char *buf, size_t count) | 
|  | { | 
|  | int rc; | 
|  | char *name; | 
|  |  | 
|  | name = kzalloc(count + 1, GFP_KERNEL); | 
|  | if (!name) | 
|  | return -ENOSPC; | 
|  | memcpy(name, buf, count); | 
|  |  | 
|  | pr_info("loading '%s'\n", name); | 
|  |  | 
|  | mutex_lock(&test_fw_mutex); | 
|  | release_firmware(test_firmware); | 
|  | test_firmware = NULL; | 
|  | rc = request_firmware(&test_firmware, name, dev); | 
|  | if (rc) | 
|  | pr_info("load of '%s' failed: %d\n", name, rc); | 
|  | pr_info("loaded: %zu\n", test_firmware ? test_firmware->size : 0); | 
|  | mutex_unlock(&test_fw_mutex); | 
|  |  | 
|  | kfree(name); | 
|  |  | 
|  | return count; | 
|  | } | 
|  | static DEVICE_ATTR_WO(trigger_request); | 
|  |  | 
|  | static int __init test_firmware_init(void) | 
|  | { | 
|  | int rc; | 
|  |  | 
|  | rc = misc_register(&test_fw_misc_device); | 
|  | if (rc) { | 
|  | pr_err("could not register misc device: %d\n", rc); | 
|  | return rc; | 
|  | } | 
|  | rc = device_create_file(test_fw_misc_device.this_device, | 
|  | &dev_attr_trigger_request); | 
|  | if (rc) { | 
|  | pr_err("could not create sysfs interface: %d\n", rc); | 
|  | goto dereg; | 
|  | } | 
|  |  | 
|  | pr_warn("interface ready\n"); | 
|  |  | 
|  | return 0; | 
|  | dereg: | 
|  | misc_deregister(&test_fw_misc_device); | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | module_init(test_firmware_init); | 
|  |  | 
|  | static void __exit test_firmware_exit(void) | 
|  | { | 
|  | release_firmware(test_firmware); | 
|  | device_remove_file(test_fw_misc_device.this_device, | 
|  | &dev_attr_trigger_request); | 
|  | misc_deregister(&test_fw_misc_device); | 
|  | pr_warn("removed interface\n"); | 
|  | } | 
|  |  | 
|  | module_exit(test_firmware_exit); | 
|  |  | 
|  | MODULE_AUTHOR("Kees Cook <keescook@chromium.org>"); | 
|  | MODULE_LICENSE("GPL"); |