]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
ALSA: usb-audio: Fix recursive locking at XRUN during syncing
authorTakashi Iwai <tiwai@suse.de>
Mon, 20 Mar 2023 14:28:38 +0000 (15:28 +0100)
committerTakashi Iwai <tiwai@suse.de>
Tue, 21 Mar 2023 06:10:43 +0000 (07:10 +0100)
The recent support of low latency playback in USB-audio driver made
the snd_usb_queue_pending_output_urbs() function to be called via PCM
ack ops.  In the new code path, the function is performed already in
the PCM stream lock.  The problem is that, when an XRUN is detected,
the function calls snd_pcm_xrun() to notify, but snd_pcm_xrun() is
supposed to be called only outside the stream lock.  As a result, it
leads to a deadlock of PCM stream locking.

For avoiding such a recursive locking, this patch adds an additional
check to the code paths in PCM core that call the ack callback; now it
checks the error code from the callback, and if it's -EPIPE, the XRUN
is handled in the PCM core side gracefully.  Along with it, the
USB-audio driver code is changed to follow that, i.e. -EPIPE is
returned instead of the explicit snd_pcm_xrun() call when the function
is performed already in the stream lock.

Fixes: d5f871f89e21 ("ALSA: usb-audio: Improved lowlatency playback support")
Reported-and-tested-by: John Keeping <john@metanate.com>
Link: https://lore.kernel.org/r/20230317195128.3911155-1-john@metanate.com
Reviewed-by: Jaroslav Kysela <perex@perex.cz>
Reviewed-by; Takashi Sakamoto <o-takashi@sakamocchi.jp>
Link: https://lore.kernel.org/r/20230320142838.494-1-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
sound/core/pcm_lib.c
sound/usb/endpoint.c
sound/usb/endpoint.h
sound/usb/pcm.c

index 8b6aeb8a78f7d3ac2819cda7b2781e1ec5035067..02fd65993e7e52e269b6a7fec59d48d3e5e5fda4 100644 (file)
@@ -2155,6 +2155,8 @@ int pcm_lib_apply_appl_ptr(struct snd_pcm_substream *substream,
                ret = substream->ops->ack(substream);
                if (ret < 0) {
                        runtime->control->appl_ptr = old_appl_ptr;
+                       if (ret == -EPIPE)
+                               __snd_pcm_xrun(substream);
                        return ret;
                }
        }
index 419302e2057e8fbc67c6ee8d940ae9a4876369b2..647fa054d8b1da37a3ec8db6e4608ce8393d8a12 100644 (file)
@@ -455,8 +455,8 @@ static void push_back_to_ready_list(struct snd_usb_endpoint *ep,
  * This function is used both for implicit feedback endpoints and in low-
  * latency playback mode.
  */
-void snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
-                                      bool in_stream_lock)
+int snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
+                                     bool in_stream_lock)
 {
        bool implicit_fb = snd_usb_endpoint_implicit_feedback_sink(ep);
 
@@ -480,7 +480,7 @@ void snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
                spin_unlock_irqrestore(&ep->lock, flags);
 
                if (ctx == NULL)
-                       return;
+                       break;
 
                /* copy over the length information */
                if (implicit_fb) {
@@ -495,11 +495,14 @@ void snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
                        break;
                if (err < 0) {
                        /* push back to ready list again for -EAGAIN */
-                       if (err == -EAGAIN)
+                       if (err == -EAGAIN) {
                                push_back_to_ready_list(ep, ctx);
-                       else
+                               break;
+                       }
+
+                       if (!in_stream_lock)
                                notify_xrun(ep);
-                       return;
+                       return -EPIPE;
                }
 
                err = usb_submit_urb(ctx->urb, GFP_ATOMIC);
@@ -507,13 +510,16 @@ void snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
                        usb_audio_err(ep->chip,
                                      "Unable to submit urb #%d: %d at %s\n",
                                      ctx->index, err, __func__);
-                       notify_xrun(ep);
-                       return;
+                       if (!in_stream_lock)
+                               notify_xrun(ep);
+                       return -EPIPE;
                }
 
                set_bit(ctx->index, &ep->active_mask);
                atomic_inc(&ep->submitted_urbs);
        }
+
+       return 0;
 }
 
 /*
index 924f4351588ce9818fc93826483d91d0dc73f8bf..c09f68ce08b187258c25bc9d8853c5427a82a3b9 100644 (file)
@@ -52,7 +52,7 @@ int snd_usb_endpoint_implicit_feedback_sink(struct snd_usb_endpoint *ep);
 int snd_usb_endpoint_next_packet_size(struct snd_usb_endpoint *ep,
                                      struct snd_urb_ctx *ctx, int idx,
                                      unsigned int avail);
-void snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
-                                      bool in_stream_lock);
+int snd_usb_queue_pending_output_urbs(struct snd_usb_endpoint *ep,
+                                     bool in_stream_lock);
 
 #endif /* __USBAUDIO_ENDPOINT_H */
index d959da7a1afbab5bca4c92c3613025e48a84432d..eec5232f9fb290c7292d824893563e1bd8326032 100644 (file)
@@ -1639,7 +1639,7 @@ static int snd_usb_pcm_playback_ack(struct snd_pcm_substream *substream)
         * outputs here
         */
        if (!ep->active_mask)
-               snd_usb_queue_pending_output_urbs(ep, true);
+               return snd_usb_queue_pending_output_urbs(ep, true);
        return 0;
 }