mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-17 05:45:20 +00:00
ALSA: pcm: Add snd_pcm_stop_xrun() helper
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>
This commit is contained in:
parent
31584ed18c
commit
1fb8510cdb
@ -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,
|
dprintk("irq: overrun [full=%d/%d] - Blocks in %d\n",dev->dmasound.read_count,
|
||||||
dev->dmasound.bufsize, dev->dmasound.blocks);
|
dev->dmasound.bufsize, dev->dmasound.blocks);
|
||||||
spin_unlock(&dev->slock);
|
spin_unlock(&dev->slock);
|
||||||
snd_pcm_stream_lock(dev->dmasound.substream);
|
snd_pcm_stop_xrun(dev->dmasound.substream);
|
||||||
snd_pcm_stop(dev->dmasound.substream,SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock(dev->dmasound.substream);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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_start(struct snd_pcm_substream *substream);
|
||||||
int snd_pcm_stop(struct snd_pcm_substream *substream, snd_pcm_state_t status);
|
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_drain_done(struct snd_pcm_substream *substream);
|
||||||
|
int snd_pcm_stop_xrun(struct snd_pcm_substream *substream);
|
||||||
#ifdef CONFIG_PM
|
#ifdef CONFIG_PM
|
||||||
int snd_pcm_suspend(struct snd_pcm_substream *substream);
|
int snd_pcm_suspend(struct snd_pcm_substream *substream);
|
||||||
int snd_pcm_suspend_all(struct snd_pcm *pcm);
|
int snd_pcm_suspend_all(struct snd_pcm *pcm);
|
||||||
|
@ -200,9 +200,7 @@ void pxa2xx_pcm_dma_irq(int dma_ch, void *dev_id)
|
|||||||
} else {
|
} else {
|
||||||
printk(KERN_ERR "DMA error on channel %d (DCSR=%#x)\n",
|
printk(KERN_ERR "DMA error on channel %d (DCSR=%#x)\n",
|
||||||
dma_ch, dcsr);
|
dma_ch, dcsr);
|
||||||
snd_pcm_stream_lock(substream);
|
snd_pcm_stop_xrun(substream);
|
||||||
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock(substream);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(pxa2xx_pcm_dma_irq);
|
EXPORT_SYMBOL(pxa2xx_pcm_dma_irq);
|
||||||
|
@ -1098,6 +1098,29 @@ int snd_pcm_drain_done(struct snd_pcm_substream *substream)
|
|||||||
SNDRV_PCM_STATE_SETUP);
|
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
|
* pause callbacks
|
||||||
*/
|
*/
|
||||||
|
@ -1006,11 +1006,7 @@ void amdtp_stream_pcm_abort(struct amdtp_stream *s)
|
|||||||
struct snd_pcm_substream *pcm;
|
struct snd_pcm_substream *pcm;
|
||||||
|
|
||||||
pcm = ACCESS_ONCE(s->pcm);
|
pcm = ACCESS_ONCE(s->pcm);
|
||||||
if (pcm) {
|
if (pcm)
|
||||||
snd_pcm_stream_lock_irq(pcm);
|
snd_pcm_stop_xrun(pcm);
|
||||||
if (snd_pcm_running(pcm))
|
|
||||||
snd_pcm_stop(pcm, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock_irq(pcm);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(amdtp_stream_pcm_abort);
|
EXPORT_SYMBOL(amdtp_stream_pcm_abort);
|
||||||
|
@ -131,14 +131,8 @@ static void isight_samples(struct isight *isight,
|
|||||||
|
|
||||||
static void isight_pcm_abort(struct isight *isight)
|
static void isight_pcm_abort(struct isight *isight)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
if (ACCESS_ONCE(isight->pcm_active))
|
||||||
|
snd_pcm_stop_xrun(isight->pcm);
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void isight_dropped_samples(struct isight *isight, unsigned int total)
|
static void isight_dropped_samples(struct isight *isight, unsigned int total)
|
||||||
|
@ -769,10 +769,7 @@ static void snd_card_asihpi_timer_function(unsigned long data)
|
|||||||
s->number);
|
s->number);
|
||||||
ds->drained_count++;
|
ds->drained_count++;
|
||||||
if (ds->drained_count > 20) {
|
if (ds->drained_count > 20) {
|
||||||
unsigned long flags;
|
snd_pcm_stop_xrun(s);
|
||||||
snd_pcm_stream_lock_irqsave(s, flags);
|
|
||||||
snd_pcm_stop(s, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock_irqrestore(s, flags);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -688,9 +688,7 @@ static void snd_atiixp_xrun_dma(struct atiixp *chip, struct atiixp_dma *dma)
|
|||||||
if (! dma->substream || ! dma->running)
|
if (! dma->substream || ! dma->running)
|
||||||
return;
|
return;
|
||||||
dev_dbg(chip->card->dev, "XRUN detected (DMA %d)\n", dma->ops->type);
|
dev_dbg(chip->card->dev, "XRUN detected (DMA %d)\n", dma->ops->type);
|
||||||
snd_pcm_stream_lock(dma->substream);
|
snd_pcm_stop_xrun(dma->substream);
|
||||||
snd_pcm_stop(dma->substream, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock(dma->substream);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -638,9 +638,7 @@ static void snd_atiixp_xrun_dma(struct atiixp_modem *chip,
|
|||||||
if (! dma->substream || ! dma->running)
|
if (! dma->substream || ! dma->running)
|
||||||
return;
|
return;
|
||||||
dev_dbg(chip->card->dev, "XRUN detected (DMA %d)\n", dma->ops->type);
|
dev_dbg(chip->card->dev, "XRUN detected (DMA %d)\n", dma->ops->type);
|
||||||
snd_pcm_stream_lock(dma->substream);
|
snd_pcm_stop_xrun(dma->substream);
|
||||||
snd_pcm_stop(dma->substream, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock(dma->substream);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -80,9 +80,7 @@ static void atmel_pcm_dma_irq(u32 ssc_sr,
|
|||||||
|
|
||||||
/* stop RX and capture: will be enabled again at restart */
|
/* stop RX and capture: will be enabled again at restart */
|
||||||
ssc_writex(prtd->ssc->regs, SSC_CR, prtd->mask->ssc_disable);
|
ssc_writex(prtd->ssc->regs, SSC_CR, prtd->mask->ssc_disable);
|
||||||
snd_pcm_stream_lock(substream);
|
snd_pcm_stop_xrun(substream);
|
||||||
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock(substream);
|
|
||||||
|
|
||||||
/* now drain RHR and read status to remove xrun condition */
|
/* now drain RHR and read status to remove xrun condition */
|
||||||
ssc_readx(prtd->ssc->regs, SSC_RHR);
|
ssc_readx(prtd->ssc->regs, SSC_RHR);
|
||||||
|
@ -151,14 +151,7 @@ static const struct snd_pcm_hardware fsl_dma_hardware = {
|
|||||||
*/
|
*/
|
||||||
static void fsl_dma_abort_stream(struct snd_pcm_substream *substream)
|
static void fsl_dma_abort_stream(struct snd_pcm_substream *substream)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
snd_pcm_stop_xrun(substream);
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -679,25 +679,16 @@ int usb6fire_pcm_init(struct sfire_chip *chip)
|
|||||||
void usb6fire_pcm_abort(struct sfire_chip *chip)
|
void usb6fire_pcm_abort(struct sfire_chip *chip)
|
||||||
{
|
{
|
||||||
struct pcm_runtime *rt = chip->pcm;
|
struct pcm_runtime *rt = chip->pcm;
|
||||||
unsigned long flags;
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (rt) {
|
if (rt) {
|
||||||
rt->panic = true;
|
rt->panic = true;
|
||||||
|
|
||||||
if (rt->playback.instance) {
|
if (rt->playback.instance)
|
||||||
snd_pcm_stream_lock_irqsave(rt->playback.instance, flags);
|
snd_pcm_stop_xrun(rt->playback.instance);
|
||||||
snd_pcm_stop(rt->playback.instance,
|
|
||||||
SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock_irqrestore(rt->playback.instance, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rt->capture.instance) {
|
if (rt->capture.instance)
|
||||||
snd_pcm_stream_lock_irqsave(rt->capture.instance, flags);
|
snd_pcm_stop_xrun(rt->capture.instance);
|
||||||
snd_pcm_stop(rt->capture.instance,
|
|
||||||
SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock_irqrestore(rt->capture.instance, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < PCM_N_URBS; i++) {
|
for (i = 0; i < PCM_N_URBS; i++) {
|
||||||
usb_poison_urb(&rt->in_urbs[i].instance);
|
usb_poison_urb(&rt->in_urbs[i].instance);
|
||||||
|
@ -391,9 +391,7 @@ static void snd_complete_urb(struct urb *urb)
|
|||||||
usb_audio_err(ep->chip, "cannot submit urb (err = %d)\n", err);
|
usb_audio_err(ep->chip, "cannot submit urb (err = %d)\n", err);
|
||||||
if (ep->data_subs && ep->data_subs->pcm_substream) {
|
if (ep->data_subs && ep->data_subs->pcm_substream) {
|
||||||
substream = ep->data_subs->pcm_substream;
|
substream = ep->data_subs->pcm_substream;
|
||||||
snd_pcm_stream_lock_irqsave(substream, flags);
|
snd_pcm_stop_xrun(substream);
|
||||||
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
|
|
||||||
snd_pcm_stream_unlock_irqrestore(substream, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
exit_clear:
|
exit_clear:
|
||||||
|
@ -613,24 +613,14 @@ static int start_usb_playback(struct ua101 *ua)
|
|||||||
|
|
||||||
static void abort_alsa_capture(struct ua101 *ua)
|
static void abort_alsa_capture(struct ua101 *ua)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
if (test_bit(ALSA_CAPTURE_RUNNING, &ua->states))
|
||||||
|
snd_pcm_stop_xrun(ua->capture.substream);
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void abort_alsa_playback(struct ua101 *ua)
|
static void abort_alsa_playback(struct ua101 *ua)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
if (test_bit(ALSA_PLAYBACK_RUNNING, &ua->states))
|
||||||
|
snd_pcm_stop_xrun(ua->playback.substream);
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int set_stream_hw(struct ua101 *ua, struct snd_pcm_substream *substream,
|
static int set_stream_hw(struct ua101 *ua, struct snd_pcm_substream *substream,
|
||||||
|
@ -272,13 +272,8 @@ static void usX2Y_clients_stop(struct usX2Ydev *usX2Y)
|
|||||||
for (s = 0; s < 4; s++) {
|
for (s = 0; s < 4; s++) {
|
||||||
struct snd_usX2Y_substream *subs = usX2Y->subs[s];
|
struct snd_usX2Y_substream *subs = usX2Y->subs[s];
|
||||||
if (subs) {
|
if (subs) {
|
||||||
if (atomic_read(&subs->state) >= state_PRERUNNING) {
|
if (atomic_read(&subs->state) >= state_PRERUNNING)
|
||||||
unsigned long flags;
|
snd_pcm_stop_xrun(subs->pcm_substream);
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
for (u = 0; u < NRURBS; u++) {
|
for (u = 0; u < NRURBS; u++) {
|
||||||
struct urb *urb = subs->urb[u];
|
struct urb *urb = subs->urb[u];
|
||||||
if (NULL != urb)
|
if (NULL != urb)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user