Commit c91a988d authored by Jaroslav Kysela's avatar Jaroslav Kysela

ALSA: pcm_core: Fix wake_up() optimization

This change fixes the "ALSA: pcm_lib - optimize wake_up() calls for PCM I/O"
commit. New sleeping queue is introduced to separate user space and kernel
space wake_ups. runtime->nowake is renamed to twake (transfer wake).
Signed-off-by: default avatarJaroslav Kysela <perex@perex.cz>
parent d1db38c0
...@@ -311,8 +311,9 @@ struct snd_pcm_runtime { ...@@ -311,8 +311,9 @@ struct snd_pcm_runtime {
struct snd_pcm_mmap_control *control; struct snd_pcm_mmap_control *control;
/* -- locking / scheduling -- */ /* -- locking / scheduling -- */
unsigned int nowake: 1; /* no wakeup (data-copy in progress) */ unsigned int twake: 1; /* do transfer (!poll) wakeup */
wait_queue_head_t sleep; wait_queue_head_t sleep; /* poll sleep */
wait_queue_head_t tsleep; /* transfer sleep */
struct fasync_struct *fasync; struct fasync_struct *fasync;
/* -- private section -- */ /* -- private section -- */
......
...@@ -894,6 +894,7 @@ int snd_pcm_attach_substream(struct snd_pcm *pcm, int stream, ...@@ -894,6 +894,7 @@ int snd_pcm_attach_substream(struct snd_pcm *pcm, int stream,
memset((void*)runtime->control, 0, size); memset((void*)runtime->control, 0, size);
init_waitqueue_head(&runtime->sleep); init_waitqueue_head(&runtime->sleep);
init_waitqueue_head(&runtime->tsleep);
runtime->status->state = SNDRV_PCM_STATE_OPEN; runtime->status->state = SNDRV_PCM_STATE_OPEN;
......
...@@ -285,8 +285,8 @@ int snd_pcm_update_state(struct snd_pcm_substream *substream, ...@@ -285,8 +285,8 @@ int snd_pcm_update_state(struct snd_pcm_substream *substream,
return -EPIPE; return -EPIPE;
} }
} }
if (!runtime->nowake && avail >= runtime->control->avail_min) if (avail >= runtime->control->avail_min)
wake_up(&runtime->sleep); wake_up(runtime->twake ? &runtime->tsleep : &runtime->sleep);
return 0; return 0;
} }
...@@ -1692,7 +1692,7 @@ static int wait_for_avail_min(struct snd_pcm_substream *substream, ...@@ -1692,7 +1692,7 @@ static int wait_for_avail_min(struct snd_pcm_substream *substream,
long tout; long tout;
init_waitqueue_entry(&wait, current); init_waitqueue_entry(&wait, current);
add_wait_queue(&runtime->sleep, &wait); add_wait_queue(&runtime->tsleep, &wait);
for (;;) { for (;;) {
if (signal_pending(current)) { if (signal_pending(current)) {
err = -ERESTARTSYS; err = -ERESTARTSYS;
...@@ -1735,7 +1735,7 @@ static int wait_for_avail_min(struct snd_pcm_substream *substream, ...@@ -1735,7 +1735,7 @@ static int wait_for_avail_min(struct snd_pcm_substream *substream,
break; break;
} }
_endloop: _endloop:
remove_wait_queue(&runtime->sleep, &wait); remove_wait_queue(&runtime->tsleep, &wait);
*availp = avail; *availp = avail;
return err; return err;
} }
...@@ -1794,7 +1794,7 @@ static snd_pcm_sframes_t snd_pcm_lib_write1(struct snd_pcm_substream *substream, ...@@ -1794,7 +1794,7 @@ static snd_pcm_sframes_t snd_pcm_lib_write1(struct snd_pcm_substream *substream,
goto _end_unlock; goto _end_unlock;
} }
runtime->nowake = 1; runtime->twake = 1;
while (size > 0) { while (size > 0) {
snd_pcm_uframes_t frames, appl_ptr, appl_ofs; snd_pcm_uframes_t frames, appl_ptr, appl_ofs;
snd_pcm_uframes_t avail; snd_pcm_uframes_t avail;
...@@ -1816,7 +1816,7 @@ static snd_pcm_sframes_t snd_pcm_lib_write1(struct snd_pcm_substream *substream, ...@@ -1816,7 +1816,7 @@ static snd_pcm_sframes_t snd_pcm_lib_write1(struct snd_pcm_substream *substream,
if (frames > cont) if (frames > cont)
frames = cont; frames = cont;
if (snd_BUG_ON(!frames)) { if (snd_BUG_ON(!frames)) {
runtime->nowake = 0; runtime->twake = 0;
snd_pcm_stream_unlock_irq(substream); snd_pcm_stream_unlock_irq(substream);
return -EINVAL; return -EINVAL;
} }
...@@ -1855,7 +1855,7 @@ static snd_pcm_sframes_t snd_pcm_lib_write1(struct snd_pcm_substream *substream, ...@@ -1855,7 +1855,7 @@ static snd_pcm_sframes_t snd_pcm_lib_write1(struct snd_pcm_substream *substream,
} }
} }
_end_unlock: _end_unlock:
runtime->nowake = 0; runtime->twake = 0;
if (xfer > 0 && err >= 0) if (xfer > 0 && err >= 0)
snd_pcm_update_state(substream, runtime); snd_pcm_update_state(substream, runtime);
snd_pcm_stream_unlock_irq(substream); snd_pcm_stream_unlock_irq(substream);
...@@ -2016,7 +2016,7 @@ static snd_pcm_sframes_t snd_pcm_lib_read1(struct snd_pcm_substream *substream, ...@@ -2016,7 +2016,7 @@ static snd_pcm_sframes_t snd_pcm_lib_read1(struct snd_pcm_substream *substream,
goto _end_unlock; goto _end_unlock;
} }
runtime->nowake = 1; runtime->twake = 1;
while (size > 0) { while (size > 0) {
snd_pcm_uframes_t frames, appl_ptr, appl_ofs; snd_pcm_uframes_t frames, appl_ptr, appl_ofs;
snd_pcm_uframes_t avail; snd_pcm_uframes_t avail;
...@@ -2045,7 +2045,7 @@ static snd_pcm_sframes_t snd_pcm_lib_read1(struct snd_pcm_substream *substream, ...@@ -2045,7 +2045,7 @@ static snd_pcm_sframes_t snd_pcm_lib_read1(struct snd_pcm_substream *substream,
if (frames > cont) if (frames > cont)
frames = cont; frames = cont;
if (snd_BUG_ON(!frames)) { if (snd_BUG_ON(!frames)) {
runtime->nowake = 0; runtime->twake = 0;
snd_pcm_stream_unlock_irq(substream); snd_pcm_stream_unlock_irq(substream);
return -EINVAL; return -EINVAL;
} }
...@@ -2078,7 +2078,7 @@ static snd_pcm_sframes_t snd_pcm_lib_read1(struct snd_pcm_substream *substream, ...@@ -2078,7 +2078,7 @@ static snd_pcm_sframes_t snd_pcm_lib_read1(struct snd_pcm_substream *substream,
xfer += frames; xfer += frames;
} }
_end_unlock: _end_unlock:
runtime->nowake = 0; runtime->twake = 0;
if (xfer > 0 && err >= 0) if (xfer > 0 && err >= 0)
snd_pcm_update_state(substream, runtime); snd_pcm_update_state(substream, runtime);
snd_pcm_stream_unlock_irq(substream); snd_pcm_stream_unlock_irq(substream);
......
...@@ -919,6 +919,7 @@ static void snd_pcm_post_stop(struct snd_pcm_substream *substream, int state) ...@@ -919,6 +919,7 @@ static void snd_pcm_post_stop(struct snd_pcm_substream *substream, int state)
runtime->status->state = state; runtime->status->state = state;
} }
wake_up(&runtime->sleep); wake_up(&runtime->sleep);
wake_up(&runtime->tsleep);
} }
static struct action_ops snd_pcm_action_stop = { static struct action_ops snd_pcm_action_stop = {
...@@ -1004,6 +1005,7 @@ static void snd_pcm_post_pause(struct snd_pcm_substream *substream, int push) ...@@ -1004,6 +1005,7 @@ static void snd_pcm_post_pause(struct snd_pcm_substream *substream, int push)
SNDRV_TIMER_EVENT_MPAUSE, SNDRV_TIMER_EVENT_MPAUSE,
&runtime->trigger_tstamp); &runtime->trigger_tstamp);
wake_up(&runtime->sleep); wake_up(&runtime->sleep);
wake_up(&runtime->tsleep);
} else { } else {
runtime->status->state = SNDRV_PCM_STATE_RUNNING; runtime->status->state = SNDRV_PCM_STATE_RUNNING;
if (substream->timer) if (substream->timer)
...@@ -1061,6 +1063,7 @@ static void snd_pcm_post_suspend(struct snd_pcm_substream *substream, int state) ...@@ -1061,6 +1063,7 @@ static void snd_pcm_post_suspend(struct snd_pcm_substream *substream, int state)
runtime->status->suspended_state = runtime->status->state; runtime->status->suspended_state = runtime->status->state;
runtime->status->state = SNDRV_PCM_STATE_SUSPENDED; runtime->status->state = SNDRV_PCM_STATE_SUSPENDED;
wake_up(&runtime->sleep); wake_up(&runtime->sleep);
wake_up(&runtime->tsleep);
} }
static struct action_ops snd_pcm_action_suspend = { static struct action_ops snd_pcm_action_suspend = {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment