Commit 576d28a7 authored by Larry Finger's avatar Larry Finger Committed by John W. Linville

b43legacy: Fix firmware loading when driver is built into the kernel

Recent versions of udev cause synchronous firmware loading from the
probe routine to fail because the request to user space times out.
The original fix for b43legacy (commit a3ea2c76) moved the firmware
load from the probe routine to a work queue, but it still used synchronous
firmware loading. This method is OK when b43legacy is built as a module;
however, it fails when the driver is compiled into the kernel.

This version changes the code to load the initial firmware file
using request_firmware_nowait(). A completion event is used to
hold the work queue until that file is available. The remaining
firmware files are read synchronously.
Signed-off-by: default avatarLarry Finger <Larry.Finger@lwfinger.net>
Cc: Stable <stable@vger.kernel.org>  (V3.4+)
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 6f80f014
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
#include <linux/ssb/ssb.h> #include <linux/ssb/ssb.h>
#include <linux/ssb/ssb_driver_chipcommon.h> #include <linux/ssb/ssb_driver_chipcommon.h>
#include <linux/completion.h>
#include <net/mac80211.h> #include <net/mac80211.h>
...@@ -733,6 +734,10 @@ struct b43legacy_wldev { ...@@ -733,6 +734,10 @@ struct b43legacy_wldev {
/* Firmware data */ /* Firmware data */
struct b43legacy_firmware fw; struct b43legacy_firmware fw;
const struct firmware *fwp; /* needed to pass fw pointer */
/* completion struct for firmware loading */
struct completion fw_load_complete;
/* Devicelist in struct b43legacy_wl (all 802.11 cores) */ /* Devicelist in struct b43legacy_wl (all 802.11 cores) */
struct list_head list; struct list_head list;
......
...@@ -1513,9 +1513,17 @@ static void b43legacy_print_fw_helptext(struct b43legacy_wl *wl) ...@@ -1513,9 +1513,17 @@ static void b43legacy_print_fw_helptext(struct b43legacy_wl *wl)
"and download the correct firmware (version 3).\n"); "and download the correct firmware (version 3).\n");
} }
static void b43legacy_fw_cb(const struct firmware *firmware, void *context)
{
struct b43legacy_wldev *dev = context;
dev->fwp = firmware;
complete(&dev->fw_load_complete);
}
static int do_request_fw(struct b43legacy_wldev *dev, static int do_request_fw(struct b43legacy_wldev *dev,
const char *name, const char *name,
const struct firmware **fw) const struct firmware **fw, bool async)
{ {
char path[sizeof(modparam_fwpostfix) + 32]; char path[sizeof(modparam_fwpostfix) + 32];
struct b43legacy_fw_header *hdr; struct b43legacy_fw_header *hdr;
...@@ -1528,7 +1536,24 @@ static int do_request_fw(struct b43legacy_wldev *dev, ...@@ -1528,7 +1536,24 @@ static int do_request_fw(struct b43legacy_wldev *dev,
snprintf(path, ARRAY_SIZE(path), snprintf(path, ARRAY_SIZE(path),
"b43legacy%s/%s.fw", "b43legacy%s/%s.fw",
modparam_fwpostfix, name); modparam_fwpostfix, name);
err = request_firmware(fw, path, dev->dev->dev); b43legacyinfo(dev->wl, "Loading firmware %s\n", path);
if (async) {
init_completion(&dev->fw_load_complete);
err = request_firmware_nowait(THIS_MODULE, 1, path,
dev->dev->dev, GFP_KERNEL,
dev, b43legacy_fw_cb);
if (err) {
b43legacyerr(dev->wl, "Unable to load firmware\n");
return err;
}
/* stall here until fw ready */
wait_for_completion(&dev->fw_load_complete);
if (!dev->fwp)
err = -EINVAL;
*fw = dev->fwp;
} else {
err = request_firmware(fw, path, dev->dev->dev);
}
if (err) { if (err) {
b43legacyerr(dev->wl, "Firmware file \"%s\" not found " b43legacyerr(dev->wl, "Firmware file \"%s\" not found "
"or load failed.\n", path); "or load failed.\n", path);
...@@ -1580,7 +1605,7 @@ static void b43legacy_request_firmware(struct work_struct *work) ...@@ -1580,7 +1605,7 @@ static void b43legacy_request_firmware(struct work_struct *work)
filename = "ucode4"; filename = "ucode4";
else else
filename = "ucode5"; filename = "ucode5";
err = do_request_fw(dev, filename, &fw->ucode); err = do_request_fw(dev, filename, &fw->ucode, true);
if (err) if (err)
goto err_load; goto err_load;
} }
...@@ -1589,7 +1614,7 @@ static void b43legacy_request_firmware(struct work_struct *work) ...@@ -1589,7 +1614,7 @@ static void b43legacy_request_firmware(struct work_struct *work)
filename = "pcm4"; filename = "pcm4";
else else
filename = "pcm5"; filename = "pcm5";
err = do_request_fw(dev, filename, &fw->pcm); err = do_request_fw(dev, filename, &fw->pcm, false);
if (err) if (err)
goto err_load; goto err_load;
} }
...@@ -1607,7 +1632,7 @@ static void b43legacy_request_firmware(struct work_struct *work) ...@@ -1607,7 +1632,7 @@ static void b43legacy_request_firmware(struct work_struct *work)
default: default:
goto err_no_initvals; goto err_no_initvals;
} }
err = do_request_fw(dev, filename, &fw->initvals); err = do_request_fw(dev, filename, &fw->initvals, false);
if (err) if (err)
goto err_load; goto err_load;
} }
...@@ -1627,7 +1652,7 @@ static void b43legacy_request_firmware(struct work_struct *work) ...@@ -1627,7 +1652,7 @@ static void b43legacy_request_firmware(struct work_struct *work)
default: default:
goto err_no_initvals; goto err_no_initvals;
} }
err = do_request_fw(dev, filename, &fw->initvals_band); err = do_request_fw(dev, filename, &fw->initvals_band, false);
if (err) if (err)
goto err_load; goto err_load;
} }
......
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