ALSA: pcm: Add snd_pcm_stop_xrun() helper
authorTakashi Iwai <tiwai@suse.de>
Fri, 7 Nov 2014 16:08:28 +0000 (17:08 +0100)
committerTakashi Iwai <tiwai@suse.de>
Sun, 9 Nov 2014 17:20:40 +0000 (18:20 +0100)
Add a new helper function snd_pcm_stop_xrun() to the standard sequnce
lock/snd_pcm_stop(XRUN)/unlock by a single call, and replace the
existing open codes with this helper.

The function checks the PCM running state to prevent setting the wrong
state, too, for more safety.

Signed-off-by: Takashi Iwai <tiwai@suse.de>
15 files changed:
drivers/media/pci/saa7134/saa7134-alsa.c
include/sound/pcm.h
sound/arm/pxa2xx-pcm-lib.c
sound/core/pcm_native.c
sound/firewire/amdtp.c
sound/firewire/isight.c
sound/pci/asihpi/asihpi.c
sound/pci/atiixp.c
sound/pci/atiixp_modem.c
sound/soc/atmel/atmel-pcm-dma.c
sound/soc/fsl/fsl_dma.c
sound/usb/6fire/pcm.c
sound/usb/endpoint.c
sound/usb/misc/ua101.c
sound/usb/usx2y/usbusx2yaudio.c

index 40569894c1c9e6da225dd76cdb270fee8a4c3d19..ac3cd74e824e0414aa99e0b27dbf590eba2ab56e 100644 (file)
@@ -173,9 +173,7 @@ static void saa7134_irq_alsa_done(struct saa7134_dev *dev,
                dprintk("irq: overrun [full=%d/%d] - Blocks in %d\n",dev->dmasound.read_count,
                        dev->dmasound.bufsize, dev->dmasound.blocks);
                spin_unlock(&dev->slock);
-               snd_pcm_stream_lock(dev->dmasound.substream);
-               snd_pcm_stop(dev->dmasound.substream,SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock(dev->dmasound.substream);
+               snd_pcm_stop_xrun(dev->dmasound.substream);
                return;
        }
 
index 0b8daeed0a33ca932b0c228b58b34884dd4914ca..40289ec2451c08e8b227b71521885f084e125cca 100644 (file)
@@ -506,6 +506,7 @@ int snd_pcm_status(struct snd_pcm_substream *substream,
 int snd_pcm_start(struct snd_pcm_substream *substream);
 int snd_pcm_stop(struct snd_pcm_substream *substream, snd_pcm_state_t status);
 int snd_pcm_drain_done(struct snd_pcm_substream *substream);
+int snd_pcm_stop_xrun(struct snd_pcm_substream *substream);
 #ifdef CONFIG_PM
 int snd_pcm_suspend(struct snd_pcm_substream *substream);
 int snd_pcm_suspend_all(struct snd_pcm *pcm);
index a61d7a9a995e86daaeea8b8243c93f9fbfe911d2..01f8fdc42b1b8a3e2c5a83c955bae76f23256071 100644 (file)
@@ -200,9 +200,7 @@ void pxa2xx_pcm_dma_irq(int dma_ch, void *dev_id)
        } else {
                printk(KERN_ERR "DMA error on channel %d (DCSR=%#x)\n",
                        dma_ch, dcsr);
-               snd_pcm_stream_lock(substream);
-               snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock(substream);
+               snd_pcm_stop_xrun(substream);
        }
 }
 EXPORT_SYMBOL(pxa2xx_pcm_dma_irq);
index dfb5031969f881e4e540deb2b72f3bc4a5ebadca..a3d1221097045ac32e7b3f2b301d9545d84c949a 100644 (file)
@@ -1098,6 +1098,29 @@ int snd_pcm_drain_done(struct snd_pcm_substream *substream)
                                     SNDRV_PCM_STATE_SETUP);
 }
 
+/**
+ * snd_pcm_stop_xrun - stop the running streams as XRUN
+ * @substream: the PCM substream instance
+ * @state: PCM state after stopping the stream
+ *
+ * This stops the given running substream (and all linked substreams) as XRUN.
+ * Unlike snd_pcm_stop(), this function takes the substream lock by itself.
+ *
+ * Return: Zero if successful, or a negative error code.
+ */
+int snd_pcm_stop_xrun(struct snd_pcm_substream *substream)
+{
+       unsigned long flags;
+       int ret = 0;
+
+       snd_pcm_stream_lock_irqsave(substream, flags);
+       if (snd_pcm_running(substream))
+               ret = snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
+       snd_pcm_stream_unlock_irqrestore(substream, flags);
+       return ret;
+}
+EXPORT_SYMBOL_GPL(snd_pcm_stop_xrun);
+
 /*
  * pause callbacks
  */
index 95fc2eaf11dc0f91bcb235a2929b300c39448052..3badc70124ab19d9b2e6115198e88c46718676e9 100644 (file)
@@ -1006,11 +1006,7 @@ void amdtp_stream_pcm_abort(struct amdtp_stream *s)
        struct snd_pcm_substream *pcm;
 
        pcm = ACCESS_ONCE(s->pcm);
-       if (pcm) {
-               snd_pcm_stream_lock_irq(pcm);
-               if (snd_pcm_running(pcm))
-                       snd_pcm_stop(pcm, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock_irq(pcm);
-       }
+       if (pcm)
+               snd_pcm_stop_xrun(pcm);
 }
 EXPORT_SYMBOL(amdtp_stream_pcm_abort);
index 7ac94439e7584d8e973796dbc07a7032909f64c9..48d6dca471c6bc5713cd7a6ee7bc466c008a1997 100644 (file)
@@ -131,14 +131,8 @@ static void isight_samples(struct isight *isight,
 
 static void isight_pcm_abort(struct isight *isight)
 {
-       unsigned long flags;
-
-       if (ACCESS_ONCE(isight->pcm_active)) {
-               snd_pcm_stream_lock_irqsave(isight->pcm, flags);
-               if (snd_pcm_running(isight->pcm))
-                       snd_pcm_stop(isight->pcm, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock_irqrestore(isight->pcm, flags);
-       }
+       if (ACCESS_ONCE(isight->pcm_active))
+               snd_pcm_stop_xrun(isight->pcm);
 }
 
 static void isight_dropped_samples(struct isight *isight, unsigned int total)
index ac66b3228a344558dcb345f1b6db5d6bd8675cbf..ff9f9f1c0391ff845c11c153ed2083353c895c52 100644 (file)
@@ -769,10 +769,7 @@ static void snd_card_asihpi_timer_function(unsigned long data)
                                                s->number);
                                ds->drained_count++;
                                if (ds->drained_count > 20) {
-                                       unsigned long flags;
-                                       snd_pcm_stream_lock_irqsave(s, flags);
-                                       snd_pcm_stop(s, SNDRV_PCM_STATE_XRUN);
-                                       snd_pcm_stream_unlock_irqrestore(s, flags);
+                                       snd_pcm_stop_xrun(s);
                                        continue;
                                }
                        } else {
index 7895c5d300c704f2d592abec4cd1f3b16e0d03ce..9c1c4452a8ee416c4ea343780a9dfee89a84b94b 100644 (file)
@@ -688,9 +688,7 @@ static void snd_atiixp_xrun_dma(struct atiixp *chip, struct atiixp_dma *dma)
        if (! dma->substream || ! dma->running)
                return;
        dev_dbg(chip->card->dev, "XRUN detected (DMA %d)\n", dma->ops->type);
-       snd_pcm_stream_lock(dma->substream);
-       snd_pcm_stop(dma->substream, SNDRV_PCM_STATE_XRUN);
-       snd_pcm_stream_unlock(dma->substream);
+       snd_pcm_stop_xrun(dma->substream);
 }
 
 /*
index 3c3241309a301ed99dceafeab875fabe192f8b86..b2f63e0727de36345d5fc5915d13f72c65e14f3b 100644 (file)
@@ -638,9 +638,7 @@ static void snd_atiixp_xrun_dma(struct atiixp_modem *chip,
        if (! dma->substream || ! dma->running)
                return;
        dev_dbg(chip->card->dev, "XRUN detected (DMA %d)\n", dma->ops->type);
-       snd_pcm_stream_lock(dma->substream);
-       snd_pcm_stop(dma->substream, SNDRV_PCM_STATE_XRUN);
-       snd_pcm_stream_unlock(dma->substream);
+       snd_pcm_stop_xrun(dma->substream);
 }
 
 /*
index b79a2a864513e4a071879a0cfcc0033678dc1be2..33fb3bb133df281429be1be2914006e575f011f1 100644 (file)
@@ -80,9 +80,7 @@ static void atmel_pcm_dma_irq(u32 ssc_sr,
 
                /* stop RX and capture: will be enabled again at restart */
                ssc_writex(prtd->ssc->regs, SSC_CR, prtd->mask->ssc_disable);
-               snd_pcm_stream_lock(substream);
-               snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock(substream);
+               snd_pcm_stop_xrun(substream);
 
                /* now drain RHR and read status to remove xrun condition */
                ssc_readx(prtd->ssc->regs, SSC_RHR);
index a609aafc994d109d8116d83110780c39769c99be..b2b108805b248a1505e20e052b5b26e6f410fec2 100644 (file)
@@ -151,14 +151,7 @@ static const struct snd_pcm_hardware fsl_dma_hardware = {
  */
 static void fsl_dma_abort_stream(struct snd_pcm_substream *substream)
 {
-       unsigned long flags;
-
-       snd_pcm_stream_lock_irqsave(substream, flags);
-
-       if (snd_pcm_running(substream))
-               snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
-
-       snd_pcm_stream_unlock_irqrestore(substream, flags);
+       snd_pcm_stop_xrun(substream);
 }
 
 /**
index ba40489b2de4fd26ab69ca8d9fc619f36c6e6493..36f4115eb1cddfcc89aaaead0138ef92d92672f2 100644 (file)
@@ -679,25 +679,16 @@ int usb6fire_pcm_init(struct sfire_chip *chip)
 void usb6fire_pcm_abort(struct sfire_chip *chip)
 {
        struct pcm_runtime *rt = chip->pcm;
-       unsigned long flags;
        int i;
 
        if (rt) {
                rt->panic = true;
 
-               if (rt->playback.instance) {
-                       snd_pcm_stream_lock_irqsave(rt->playback.instance, flags);
-                       snd_pcm_stop(rt->playback.instance,
-                                       SNDRV_PCM_STATE_XRUN);
-                       snd_pcm_stream_unlock_irqrestore(rt->playback.instance, flags);
-               }
+               if (rt->playback.instance)
+                       snd_pcm_stop_xrun(rt->playback.instance);
 
-               if (rt->capture.instance) {
-                       snd_pcm_stream_lock_irqsave(rt->capture.instance, flags);
-                       snd_pcm_stop(rt->capture.instance,
-                                       SNDRV_PCM_STATE_XRUN);
-                       snd_pcm_stream_unlock_irqrestore(rt->capture.instance, flags);
-               }
+               if (rt->capture.instance)
+                       snd_pcm_stop_xrun(rt->capture.instance);
 
                for (i = 0; i < PCM_N_URBS; i++) {
                        usb_poison_urb(&rt->in_urbs[i].instance);
index a4679913b0aa5d663d0064e7be82c3d25c8c4d08..03b074419964967ab16644a916112f1d2c514712 100644 (file)
@@ -391,9 +391,7 @@ static void snd_complete_urb(struct urb *urb)
        usb_audio_err(ep->chip, "cannot submit urb (err = %d)\n", err);
        if (ep->data_subs && ep->data_subs->pcm_substream) {
                substream = ep->data_subs->pcm_substream;
-               snd_pcm_stream_lock_irqsave(substream, flags);
-               snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock_irqrestore(substream, flags);
+               snd_pcm_stop_xrun(substream);
        }
 
 exit_clear:
index a1bab149df4d06ab1ba9fb4c725c19974841eb5d..9581089c28c5e5300930476ee16774543095ed74 100644 (file)
@@ -613,24 +613,14 @@ static int start_usb_playback(struct ua101 *ua)
 
 static void abort_alsa_capture(struct ua101 *ua)
 {
-       unsigned long flags;
-
-       if (test_bit(ALSA_CAPTURE_RUNNING, &ua->states)) {
-               snd_pcm_stream_lock_irqsave(ua->capture.substream, flags);
-               snd_pcm_stop(ua->capture.substream, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock_irqrestore(ua->capture.substream, flags);
-       }
+       if (test_bit(ALSA_CAPTURE_RUNNING, &ua->states))
+               snd_pcm_stop_xrun(ua->capture.substream);
 }
 
 static void abort_alsa_playback(struct ua101 *ua)
 {
-       unsigned long flags;
-
-       if (test_bit(ALSA_PLAYBACK_RUNNING, &ua->states)) {
-               snd_pcm_stream_lock_irqsave(ua->playback.substream, flags);
-               snd_pcm_stop(ua->playback.substream, SNDRV_PCM_STATE_XRUN);
-               snd_pcm_stream_unlock_irqrestore(ua->playback.substream, flags);
-       }
+       if (test_bit(ALSA_PLAYBACK_RUNNING, &ua->states))
+               snd_pcm_stop_xrun(ua->playback.substream);
 }
 
 static int set_stream_hw(struct ua101 *ua, struct snd_pcm_substream *substream,
index a63330dd1407e721285285a559c8b90adddd74c5..61d5dc2a342174a2e6f72c5250ff3ddfd20d7b5b 100644 (file)
@@ -272,13 +272,8 @@ static void usX2Y_clients_stop(struct usX2Ydev *usX2Y)
        for (s = 0; s < 4; s++) {
                struct snd_usX2Y_substream *subs = usX2Y->subs[s];
                if (subs) {
-                       if (atomic_read(&subs->state) >= state_PRERUNNING) {
-                               unsigned long flags;
-
-                               snd_pcm_stream_lock_irqsave(subs->pcm_substream, flags);
-                               snd_pcm_stop(subs->pcm_substream, SNDRV_PCM_STATE_XRUN);
-                               snd_pcm_stream_unlock_irqrestore(subs->pcm_substream, flags);
-                       }
+                       if (atomic_read(&subs->state) >= state_PRERUNNING)
+                               snd_pcm_stop_xrun(subs->pcm_substream);
                        for (u = 0; u < NRURBS; u++) {
                                struct urb *urb = subs->urb[u];
                                if (NULL != urb)