[alsa-devel] [PATCH 4/4] ALSA: hda - Continue probe in work context to avoid request_module deadlock
Takashi Iwai
tiwai at suse.de
Thu May 23 08:47:56 CEST 2013
At Thu, 23 May 2013 01:04:16 +0800,
Wang Xingchao wrote:
>
> There's deadlock when request_module(i915) in azx_probe.
> It looks like:
> device_lock(audio pci device) -> azx_probe -> module_request
> (or symbol_request) -> modprobe (userspace) -> i915 init ->
> drm_pci_init -> pci_register_driver -> bus_add_driver -> driver_attach ->
> which in turn tries all locks on pci bus, and when it tries the one on the
> audio device, it will deadlock.
>
> This patch introduce a work to store remaining probe stuff, and let
> request_module run in safe work context.
The bug is introduced by your patch, so better to fold into it.
Moreover...
>
> Signed-off-by: Wang Xingchao <xingchao.wang at linux.intel.com>
> ---
> sound/pci/hda/hda_intel.c | 105 +++++++++++++++++++++++++++-------------------
> 1 file changed, 62 insertions(+), 43 deletions(-)
>
> diff --git a/sound/pci/hda/hda_intel.c b/sound/pci/hda/hda_intel.c
> index f20a88c..1bc7c3b 100644
> --- a/sound/pci/hda/hda_intel.c
> +++ b/sound/pci/hda/hda_intel.c
> @@ -76,6 +76,7 @@ static int probe_only[SNDRV_CARDS];
> static int jackpoll_ms[SNDRV_CARDS];
> static bool single_cmd;
> static int enable_msi = -1;
> +static int dev;
Don't do this.
> #ifdef CONFIG_SND_HDA_PATCH_LOADER
> static char *patch[SNDRV_CARDS];
> #endif
> @@ -542,6 +543,8 @@ struct azx {
> /* for pending irqs */
> struct work_struct irq_pending_work;
>
> + struct delayed_work probe_work;
> +
> /* reboot notifier (for mysterious hangup problem at power-down) */
> struct notifier_block reboot_notifier;
>
> @@ -3670,58 +3673,22 @@ static void azx_firmware_cb(const struct firmware *fw, void *context)
> }
> #endif
>
> -static int azx_probe(struct pci_dev *pci,
> - const struct pci_device_id *pci_id)
> +static void azx_probe_work(struct work_struct *work)
> {
> - static int dev;
> - struct snd_card *card;
> - struct azx *chip;
> + struct azx *chip =
> + container_of(work, struct azx, probe_work.work);
> + struct snd_card *card = chip->card;
> + struct pci_dev *pci = chip->pci;
> bool probe_now;
> int err;
>
> - if (dev >= SNDRV_CARDS)
> - return -ENODEV;
> - if (!enable[dev]) {
> - dev++;
> - return -ENOENT;
> - }
> -
> - err = snd_card_create(index[dev], id[dev], THIS_MODULE, 0, &card);
> - if (err < 0) {
> - snd_printk(KERN_ERR "hda-intel: Error creating card!\n");
> - return err;
> - }
> -
> - snd_card_set_dev(card, &pci->dev);
> -
> - err = azx_create(card, pci, dev, pci_id->driver_data, &chip);
> - if (err < 0)
> - goto out_free;
> - card->private_data = chip;
> -
> - pci_set_drvdata(pci, card);
> -
> - err = register_vga_switcheroo(chip);
> - if (err < 0) {
> - snd_printk(KERN_ERR SFX
> - "%s: Error registering VGA-switcheroo client\n", pci_name(pci));
> - goto out_free;
> - }
> -
> - if (check_hdmi_disabled(pci)) {
> - snd_printk(KERN_INFO SFX "%s: VGA controller is disabled\n",
> - pci_name(pci));
> - snd_printk(KERN_INFO SFX "%s: Delaying initialization\n", pci_name(pci));
> - chip->disabled = true;
> - }
> -
> /* Request power well for Haswell HDA controller and codec */
> if (chip->driver_caps & AZX_DCAPS_I915_POWERWELL) {
> #ifdef CONFIG_SND_HDA_I915
> err = hda_i915_init();
> if (err < 0) {
> snd_printk(KERN_ERR SFX "Error request power-well from i915\n");
> - return err;
> + goto out_free;
> }
> hda_display_power(true);
> #else
> @@ -3760,7 +3727,7 @@ static int azx_probe(struct pci_dev *pci,
>
> dev++;
> complete_all(&chip->probe_wait);
> - return 0;
> + return;
> out_free_power:
> if (chip->driver_caps & AZX_DCAPS_I915_POWERWELL) {
> hda_display_power(false);
> @@ -3769,6 +3736,58 @@ out_free_power:
> out_free:
> snd_card_free(card);
> pci_set_drvdata(pci, NULL);
> +}
> +
> +static int azx_probe(struct pci_dev *pci,
> + const struct pci_device_id *pci_id)
> +{
> + struct snd_card *card;
> + struct azx *chip;
> + int err;
> +
> + if (dev >= SNDRV_CARDS)
> + return -ENODEV;
> + if (!enable[dev]) {
> + dev++;
> + return -ENOENT;
> + }
> +
> + err = snd_card_create(index[dev], id[dev], THIS_MODULE, 0, &card);
> + if (err < 0) {
> + snd_printk(KERN_ERR "hda-intel: Error creating card!\n");
> + return err;
> + }
> +
> + snd_card_set_dev(card, &pci->dev);
> +
> + err = azx_create(card, pci, dev, pci_id->driver_data, &chip);
> + if (err < 0)
> + goto out_free;
> + card->private_data = chip;
> +
> + pci_set_drvdata(pci, card);
> +
> + err = register_vga_switcheroo(chip);
> + if (err < 0) {
> + snd_printk(KERN_ERR SFX
> + "%s: Error registering VGA-switcheroo client\n", pci_name(pci));
> + goto out_free;
> + }
> +
> + if (check_hdmi_disabled(pci)) {
> + snd_printk(KERN_INFO SFX "%s: VGA controller is disabled\n",
> + pci_name(pci));
> + snd_printk(KERN_INFO SFX "%s: Delaying initialization\n", pci_name(pci));
> + chip->disabled = true;
> + }
> +
> + /* continue probing in work context as may trigger request module */
> + INIT_DELAYED_WORK(&chip->probe_work, azx_probe_work);
The initialization must be done earlier, in azx_create().
> + schedule_delayed_work(&chip->probe_work, 0);
You shouldn't do async probe unless needed.
That is, this is required only for Haswell.
Also, if you delay the call of hda_i915_init(), you need to have a
flag to indicate whether this initialization has been done, and call
the counterpart in azx_free() only if the flag is set. Otherwise, you
might call hda_i915_exit() & co even before calling hda_i915_init().
Another point to fix is to make sure to cancel the leftover work in
destructor.
Last not but least, I guess the call of pm_runtime_put_noidle() in
azx_probe() might be problematic. In theory it allows the runtime
suspend before hda_i915_init() is done.
Maybe we should move pm_runtime_put_noidle() into
azx_probe_continue(). And put the counterpart to azx_free()
conditionally called with chip->running, or so.
But, this doesn't exclude the explicit suspend/resume -- you are
calling hda_display_power() and this might be also before the actual
initialization. Again, this must be conditional, too.
Takashi
> + return 0;
> +out_free:
> + snd_card_free(card);
> + pci_set_drvdata(pci, NULL);
> return err;
> }
More information about the Alsa-devel
mailing list