On Mon, 13 Feb 2012 13:37:06 -0600
Larry Finger <[email protected]> wrote:

> Drivers that load firmware from their probe routine have problems with the
> latest versions of udev as they get timeouts while waiting for user
> space to start. The problem is fixed by loading the firmware and starting
> mac80211 from a delayed_work queue. By using this method, most of the
> original code is preserved.
> 
> Signed-off-by: Larry Finger <[email protected]>
> ---
> This one is compile-tested only.
> 
> Larry
> ---
>  drivers/net/wireless/p54/p54spi.c |   38 ++++++++++++++++++++++++++----------
>  drivers/net/wireless/p54/p54spi.h |    1 +
>  2 files changed, 28 insertions(+), 11 deletions(-)
> 
> diff --git a/drivers/net/wireless/p54/p54spi.c 
> b/drivers/net/wireless/p54/p54spi.c
> index 7faed62..87c2634 100644
> --- a/drivers/net/wireless/p54/p54spi.c
> +++ b/drivers/net/wireless/p54/p54spi.c
> @@ -595,6 +595,30 @@ static void p54spi_op_stop(struct ieee80211_hw *dev)
>       cancel_work_sync(&priv->work);
>  }
>  
> +static void p54s_load_firmware(struct work_struct *work)
> +{
> +     struct p54s_priv *priv = container_of(to_delayed_work(work),
> +                              struct p54s_priv, firmware_load);
> +     struct ieee80211_hw *hw = priv->hw;
> +     int ret;
> +
> +     ret = p54spi_request_firmware(hw);
> +     if (ret < 0)
> +             goto err_free_common;
> +
> +     ret = p54spi_request_eeprom(hw);
> +     if (ret)
> +             goto err_free_common;
> +
> +     ret = p54_register_common(hw, &priv->spi->dev);
> +     if (ret)
> +             goto err_free_common;
> +     return;
> +
> +err_free_common:
> +     dev_err(&priv->spi->dev, "Failed to read firmware\n");
> +}
> +
>  static int __devinit p54spi_probe(struct spi_device *spi)
>  {
>       struct p54s_priv *priv = NULL;
> @@ -658,17 +682,9 @@ static int __devinit p54spi_probe(struct spi_device *spi)
>       priv->common.stop = p54spi_op_stop;
>       priv->common.tx = p54spi_op_tx;
>  
> -     ret = p54spi_request_firmware(hw);
> -     if (ret < 0)
> -             goto err_free_common;
> -
> -     ret = p54spi_request_eeprom(hw);
> -     if (ret)
> -             goto err_free_common;
> -
> -     ret = p54_register_common(hw, &priv->spi->dev);
> -     if (ret)
> -             goto err_free_common;
> +     /* setup and start delayed work to load firmware */
> +     INIT_DELAYED_WORK(&priv->firmware_load, p54s_load_firmware);
> +     schedule_delayed_work(&priv->firmware_load, HZ);

Why delay it one second?
That seems arbitrary. Just use a non-delayed workqueue. That should be good 
enough.

>  
>       return 0;
>  
> diff --git a/drivers/net/wireless/p54/p54spi.h 
> b/drivers/net/wireless/p54/p54spi.h
> index dfaa62a..5f7714b 100644
> --- a/drivers/net/wireless/p54/p54spi.h
> +++ b/drivers/net/wireless/p54/p54spi.h
> @@ -120,6 +120,7 @@ struct p54s_priv {
>  
>       enum fw_state fw_state;
>       const struct firmware *firmware;
> +     struct delayed_work firmware_load;
>  };
>  
>  #endif /* P54SPI_H */

I think we have to cancel the work in the remove handler. Just in case the 
device was removed
before fw load wq could run.

-- 
Greetings, Michael.
_______________________________________________
devel mailing list
[email protected]
http://driverdev.linuxdriverproject.org/mailman/listinfo/devel

Reply via email to