参照元

説明

引数

返り値

参考

実装

/* CAUTION: call it with irq disabled */
int snd_pcm_update_hw_ptr(struct snd_pcm_substream *substream)
{
	struct snd_pcm_runtime *runtime = substream->runtime;
	snd_pcm_uframes_t pos;
	snd_pcm_uframes_t old_hw_ptr, new_hw_ptr, hw_base;
	snd_pcm_sframes_t delta;
	unsigned long jdelta;
	old_hw_ptr = runtime->status->hw_ptr;
	pos = snd_pcm_update_hw_ptr_pos(substream, runtime);
	if (pos == SNDRV_PCM_POS_XRUN) {
		xrun(substream);
		return -EPIPE;
	}
	if (xrun_debug(substream, 16)) {
		char name[16];
		pcm_debug_name(substream, name, sizeof(name));
		snd_printd("hw_update: %s: pos=0x%x/0x%x/0x%x, "
			   "hwptr=0x%lx, hw_base=0x%lx, hw_intr=0x%lx\n",
			   name, (unsigned int)pos,
			   (unsigned int)runtime->period_size,
			   (unsigned int)runtime->buffer_size,
			   (unsigned long)old_hw_ptr,
			   (unsigned long)runtime->hw_ptr_base,
			   (unsigned long)runtime->hw_ptr_interrupt);
	}
	hw_base = runtime->hw_ptr_base;
	new_hw_ptr = hw_base + pos;

	delta = new_hw_ptr - old_hw_ptr;
	jdelta = jiffies - runtime->hw_ptr_jiffies;
	if (delta < 0) {
		delta += runtime->buffer_size;
		if (delta < 0) {
			hw_ptr_error(substream, 
				     "Unexpected hw_pointer value [2] "
				     "(stream=%i, pos=%ld, old_ptr=%ld, jdelta=%li)\n",
				     substream->stream, (long)pos,
				     (long)old_hw_ptr, jdelta);
			return 0;
		}
		hw_base += runtime->buffer_size;
		if (hw_base >= runtime->boundary)
			hw_base = 0;
		new_hw_ptr = hw_base + pos;
	}
	/* Do jiffies check only in xrun_debug mode */
	if (!xrun_debug(substream, 4))
		goto no_jiffies_check;
	if (delta < runtime->delay)
		goto no_jiffies_check;
	delta -= runtime->delay;
	if (((delta * HZ) / runtime->rate) > jdelta + HZ/100) {
		hw_ptr_error(substream,
			     "hw_ptr skipping! "
			     "(pos=%ld, delta=%ld, period=%ld, jdelta=%lu/%lu)\n",
			     (long)pos, (long)delta,
			     (long)runtime->period_size, jdelta,
			     ((delta * HZ) / runtime->rate));
		return 0;
	}
 no_jiffies_check:
	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK &&
	    runtime->silence_size > 0)
		snd_pcm_playback_silence(substream, new_hw_ptr);
	if (runtime->status->hw_ptr == new_hw_ptr)
		return 0;

	runtime->hw_ptr_base = hw_base;
	runtime->status->hw_ptr = new_hw_ptr;
	runtime->hw_ptr_jiffies = jiffies;
	if (runtime->tstamp_mode == SNDRV_PCM_TSTAMP_ENABLE)
		snd_pcm_gettime(runtime, (struct timespec *)&runtime->status->tstamp);
	return snd_pcm_update_hw_ptr_post(substream, runtime);
}

コメント


トップ   編集 凍結 差分 履歴 添付 複製 名前変更 リロード   新規 一覧 検索 最終更新   ヘルプ   最終更新のRSS
Last-modified: 2014-09-13 (土) 08:26:43