diff options
| -rw-r--r-- | Documentation/sound/alsa/Procfile.txt | 5 | ||||
| -rw-r--r-- | sound/core/pcm_lib.c | 10 | ||||
| -rw-r--r-- | sound/core/pcm_native.c | 6 |
3 files changed, 19 insertions, 2 deletions
diff --git a/Documentation/sound/alsa/Procfile.txt b/Documentation/sound/alsa/Procfile.txt index bba2dbb79d81..cfac20cf9e33 100644 --- a/Documentation/sound/alsa/Procfile.txt +++ b/Documentation/sound/alsa/Procfile.txt | |||
| @@ -104,6 +104,11 @@ card*/pcm*/xrun_debug | |||
| 104 | When this value is greater than 1, the driver will show the | 104 | When this value is greater than 1, the driver will show the |
| 105 | stack trace additionally. This may help the debugging. | 105 | stack trace additionally. This may help the debugging. |
| 106 | 106 | ||
| 107 | Since 2.6.30, this option also enables the hwptr check using | ||
| 108 | jiffies. This detects spontaneous invalid pointer callback | ||
| 109 | values, but can be lead to too much corrections for a (mostly | ||
| 110 | buggy) hardware that doesn't give smooth pointer updates. | ||
| 111 | |||
| 107 | card*/pcm*/sub*/info | 112 | card*/pcm*/sub*/info |
| 108 | The general information of this PCM sub-stream. | 113 | The general information of this PCM sub-stream. |
| 109 | 114 | ||
diff --git a/sound/core/pcm_lib.c b/sound/core/pcm_lib.c index a2a792c18c40..d659995ac3ac 100644 --- a/sound/core/pcm_lib.c +++ b/sound/core/pcm_lib.c | |||
| @@ -249,6 +249,11 @@ static int snd_pcm_update_hw_ptr_interrupt(struct snd_pcm_substream *substream) | |||
| 249 | new_hw_ptr = hw_base + pos; | 249 | new_hw_ptr = hw_base + pos; |
| 250 | } | 250 | } |
| 251 | } | 251 | } |
| 252 | |||
| 253 | /* Do jiffies check only in xrun_debug mode */ | ||
| 254 | if (!xrun_debug(substream)) | ||
| 255 | goto no_jiffies_check; | ||
| 256 | |||
| 252 | /* Skip the jiffies check for hardwares with BATCH flag. | 257 | /* Skip the jiffies check for hardwares with BATCH flag. |
| 253 | * Such hardware usually just increases the position at each IRQ, | 258 | * Such hardware usually just increases the position at each IRQ, |
| 254 | * thus it can't give any strange position. | 259 | * thus it can't give any strange position. |
| @@ -336,7 +341,9 @@ int snd_pcm_update_hw_ptr(struct snd_pcm_substream *substream) | |||
| 336 | hw_base = 0; | 341 | hw_base = 0; |
| 337 | new_hw_ptr = hw_base + pos; | 342 | new_hw_ptr = hw_base + pos; |
| 338 | } | 343 | } |
| 339 | if (((delta * HZ) / runtime->rate) > jdelta + HZ/100) { | 344 | /* Do jiffies check only in xrun_debug mode */ |
| 345 | if (xrun_debug(substream) && | ||
| 346 | ((delta * HZ) / runtime->rate) > jdelta + HZ/100) { | ||
| 340 | hw_ptr_error(substream, | 347 | hw_ptr_error(substream, |
| 341 | "hw_ptr skipping! " | 348 | "hw_ptr skipping! " |
| 342 | "(pos=%ld, delta=%ld, period=%ld, jdelta=%lu/%lu)\n", | 349 | "(pos=%ld, delta=%ld, period=%ld, jdelta=%lu/%lu)\n", |
| @@ -1478,7 +1485,6 @@ static int snd_pcm_lib_ioctl_reset(struct snd_pcm_substream *substream, | |||
| 1478 | runtime->status->hw_ptr %= runtime->buffer_size; | 1485 | runtime->status->hw_ptr %= runtime->buffer_size; |
| 1479 | else | 1486 | else |
| 1480 | runtime->status->hw_ptr = 0; | 1487 | runtime->status->hw_ptr = 0; |
| 1481 | runtime->hw_ptr_jiffies = jiffies; | ||
| 1482 | snd_pcm_stream_unlock_irqrestore(substream, flags); | 1488 | snd_pcm_stream_unlock_irqrestore(substream, flags); |
| 1483 | return 0; | 1489 | return 0; |
| 1484 | } | 1490 | } |
diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c index fc6f98e257df..b5da656d1ece 100644 --- a/sound/core/pcm_native.c +++ b/sound/core/pcm_native.c | |||
| @@ -848,6 +848,7 @@ static void snd_pcm_post_start(struct snd_pcm_substream *substream, int state) | |||
| 848 | { | 848 | { |
| 849 | struct snd_pcm_runtime *runtime = substream->runtime; | 849 | struct snd_pcm_runtime *runtime = substream->runtime; |
| 850 | snd_pcm_trigger_tstamp(substream); | 850 | snd_pcm_trigger_tstamp(substream); |
| 851 | runtime->hw_ptr_jiffies = jiffies; | ||
| 851 | runtime->status->state = state; | 852 | runtime->status->state = state; |
| 852 | if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK && | 853 | if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK && |
| 853 | runtime->silence_size > 0) | 854 | runtime->silence_size > 0) |
| @@ -961,6 +962,11 @@ static int snd_pcm_do_pause(struct snd_pcm_substream *substream, int push) | |||
| 961 | { | 962 | { |
| 962 | if (substream->runtime->trigger_master != substream) | 963 | if (substream->runtime->trigger_master != substream) |
| 963 | return 0; | 964 | return 0; |
| 965 | /* The jiffies check in snd_pcm_update_hw_ptr*() is done by | ||
| 966 | * a delta betwen the current jiffies, this gives a large enough | ||
| 967 | * delta, effectively to skip the check once. | ||
| 968 | */ | ||
| 969 | substream->runtime->hw_ptr_jiffies = jiffies - HZ * 1000; | ||
| 964 | return substream->ops->trigger(substream, | 970 | return substream->ops->trigger(substream, |
| 965 | push ? SNDRV_PCM_TRIGGER_PAUSE_PUSH : | 971 | push ? SNDRV_PCM_TRIGGER_PAUSE_PUSH : |
| 966 | SNDRV_PCM_TRIGGER_PAUSE_RELEASE); | 972 | SNDRV_PCM_TRIGGER_PAUSE_RELEASE); |
