* Stream interrupts seem to arrive too early on most HDA chips:

we adjust buffer descriptors to take this into account. It defaults to one sample, but it should depend also on the sample rate or the chip vendor.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@29322 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Jérôme Duval 2009-02-25 19:12:23 +00:00
parent 3539f6ec6a
commit 346ae57c93

View File

@ -109,7 +109,7 @@ stream_handle_interrupt(hda_controller* controller, hda_stream* stream)
stream->real_time = system_time(); stream->real_time = system_time();
stream->frames_count += stream->buffer_length; stream->frames_count += stream->buffer_length;
stream->buffer_cycle = 1 - (position / (bufferSize + 1)); // added 1 to avoid having 2 stream->buffer_cycle = ((position / bufferSize) + 1) % stream->num_buffers;
release_spinlock(&stream->lock); release_spinlock(&stream->lock);
@ -523,13 +523,18 @@ hda_stream_setup_buffers(hda_audio_group* audioGroup, hda_stream* stream,
stream->buffer_descriptors_area = B_ERROR; stream->buffer_descriptors_area = B_ERROR;
} }
// Stream interrupts seem to arrive too early on most HDA
// so we adjust buffer descriptors to take this into account
// TODO compute this value based on sample rate and depending on the hardware
uint32 offset = 1 * (stream->sample_size * stream->num_channels);
/* Calculate size of buffer (aligned to 128 bytes) */ /* Calculate size of buffer (aligned to 128 bytes) */
bufferSize = stream->sample_size * stream->num_channels bufferSize = stream->sample_size * stream->num_channels
* stream->buffer_length; * stream->buffer_length;
bufferSize = ALIGN(bufferSize, 128); bufferSize = ALIGN(bufferSize, 128);
dprintf("HDA: sample size %ld, num channels %ld, buffer length %ld ****************\n", dprintf("HDA: sample size %ld, num channels %ld, buffer length %ld, offset %ld **********\n",
stream->sample_size, stream->num_channels, stream->buffer_length); stream->sample_size, stream->num_channels, stream->buffer_length, offset);
/* Calculate total size of all buffers (aligned to size of B_PAGE_SIZE) */ /* Calculate total size of all buffers (aligned to size of B_PAGE_SIZE) */
alloc = bufferSize * stream->num_buffers; alloc = bufferSize * stream->num_buffers;
@ -561,7 +566,7 @@ hda_stream_setup_buffers(hda_audio_group* audioGroup, hda_stream* stream,
} }
/* Now allocate BDL for buffer range */ /* Now allocate BDL for buffer range */
alloc = stream->num_buffers * sizeof(bdl_entry_t); alloc = (stream->num_buffers + ((offset > 0) ? 1 : 0)) * sizeof(bdl_entry_t);
alloc = PAGE_ALIGN(alloc); alloc = PAGE_ALIGN(alloc);
stream->buffer_descriptors_area = create_area("hda buffer descriptors", stream->buffer_descriptors_area = create_area("hda buffer descriptors",
@ -585,14 +590,30 @@ hda_stream_setup_buffers(hda_audio_group* audioGroup, hda_stream* stream,
dprintf("%s(%s): Allocated %ld bytes for %ld BDLEs\n", __func__, desc, dprintf("%s(%s): Allocated %ld bytes for %ld BDLEs\n", __func__, desc,
alloc, stream->num_buffers); alloc, stream->num_buffers);
uint32 fragments = 0;
if (offset > 0) {
bufferDescriptors->lower = stream->physical_buffers[0];
bufferDescriptors->upper = 0;
bufferDescriptors->length = offset;
bufferDescriptors->ioc = 1;
bufferDescriptors++;
fragments++;
}
/* Setup buffer descriptor list (BDL) entries */ /* Setup buffer descriptor list (BDL) entries */
for (index = 0; index < stream->num_buffers; index++, bufferDescriptors++) { for (index = 0; index < stream->num_buffers; index++, bufferDescriptors++) {
bufferDescriptors->lower = stream->physical_buffers[index]; bufferDescriptors->lower = stream->physical_buffers[index] + offset;
bufferDescriptors->upper = 0; bufferDescriptors->upper = 0;
fragments++;
if (index == (stream->num_buffers - 1) && offset > 0) {
bufferDescriptors->length = bufferSize - offset;
bufferDescriptors->ioc = 0;
} else {
bufferDescriptors->length = bufferSize; bufferDescriptors->length = bufferSize;
bufferDescriptors->ioc = 1; bufferDescriptors->ioc = 1;
// we want an interrupt after every buffer // we want an interrupt after every buffer
} }
}
/* Configure stream registers */ /* Configure stream registers */
format = (stream->num_channels - 1) & 0xf; format = (stream->num_channels - 1) & 0xf;
@ -624,7 +645,7 @@ hda_stream_setup_buffers(hda_audio_group* audioGroup, hda_stream* stream,
stream->Write32(HDAC_STREAM_BUFFERS_BASE_LOWER, stream->Write32(HDAC_STREAM_BUFFERS_BASE_LOWER,
stream->physical_buffer_descriptors); stream->physical_buffer_descriptors);
stream->Write32(HDAC_STREAM_BUFFERS_BASE_UPPER, 0); stream->Write32(HDAC_STREAM_BUFFERS_BASE_UPPER, 0);
stream->Write16(HDAC_STREAM_LAST_VALID, stream->num_buffers - 1); stream->Write16(HDAC_STREAM_LAST_VALID, fragments);
/* total cyclic buffer size in _bytes_ */ /* total cyclic buffer size in _bytes_ */
stream->Write32(HDAC_STREAM_BUFFER_SIZE, bufferSize stream->Write32(HDAC_STREAM_BUFFER_SIZE, bufferSize
* stream->num_buffers); * stream->num_buffers);