Commit 373304fe authored by Ming Lei's avatar Ming Lei Committed by Greg Kroah-Hartman

firmware loader: cancel uncache work before caching firmware

Under 'Opportunistic sleep' situation, system sleep might be
triggered very frequently, so the uncahce work may not be completed
before caching firmware during next suspend.

This patch cancels the uncache work before caching firmware to
fix the problem above.

Also this patch optimizes the cacheing firmware mechanism a bit by
only storing one firmware cache entry for one firmware image.

So if the firmware is still cached during suspend, it doesn't need
to be loaded from user space any more.
Signed-off-by: default avatarMing Lei <ming.lei@canonical.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 6f0c0580
...@@ -1142,17 +1142,27 @@ static struct fw_cache_entry *alloc_fw_cache_entry(const char *name) ...@@ -1142,17 +1142,27 @@ static struct fw_cache_entry *alloc_fw_cache_entry(const char *name)
return fce; return fce;
} }
static int fw_cache_piggyback_on_request(const char *name) static int __fw_entry_found(const char *name)
{ {
struct firmware_cache *fwc = &fw_cache; struct firmware_cache *fwc = &fw_cache;
struct fw_cache_entry *fce; struct fw_cache_entry *fce;
int ret = 0;
spin_lock(&fwc->name_lock);
list_for_each_entry(fce, &fwc->fw_names, list) { list_for_each_entry(fce, &fwc->fw_names, list) {
if (!strcmp(fce->name, name)) if (!strcmp(fce->name, name))
goto found; return 1;
} }
return 0;
}
static int fw_cache_piggyback_on_request(const char *name)
{
struct firmware_cache *fwc = &fw_cache;
struct fw_cache_entry *fce;
int ret = 0;
spin_lock(&fwc->name_lock);
if (__fw_entry_found(name))
goto found;
fce = alloc_fw_cache_entry(name); fce = alloc_fw_cache_entry(name);
if (fce) { if (fce) {
...@@ -1229,11 +1239,19 @@ static void dev_cache_fw_image(struct device *dev, void *data) ...@@ -1229,11 +1239,19 @@ static void dev_cache_fw_image(struct device *dev, void *data)
list_del(&fce->list); list_del(&fce->list);
spin_lock(&fwc->name_lock); spin_lock(&fwc->name_lock);
fwc->cnt++; /* only one cache entry for one firmware */
list_add(&fce->list, &fwc->fw_names); if (!__fw_entry_found(fce->name)) {
fwc->cnt++;
list_add(&fce->list, &fwc->fw_names);
} else {
free_fw_cache_entry(fce);
fce = NULL;
}
spin_unlock(&fwc->name_lock); spin_unlock(&fwc->name_lock);
async_schedule(__async_dev_cache_fw_image, (void *)fce); if (fce)
async_schedule(__async_dev_cache_fw_image,
(void *)fce);
} }
} }
...@@ -1275,6 +1293,9 @@ static void device_cache_fw_images(void) ...@@ -1275,6 +1293,9 @@ static void device_cache_fw_images(void)
pr_debug("%s\n", __func__); pr_debug("%s\n", __func__);
/* cancel uncache work */
cancel_delayed_work_sync(&fwc->work);
/* /*
* use small loading timeout for caching devices' firmware * use small loading timeout for caching devices' firmware
* because all these firmware images have been loaded * because all these firmware images have been loaded
......
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