Fix various bugs in OS X DSD playback
the most notable bugs are 1. osx_output_set_device_format should use the target asbd rather than AudioFormat. This is because asbd's sample rate calculation reflects the real dop target rate of the DAC, white AudioFormat's sample rate is the original DSD format rate. 2. the original code value the highest rate that's the multiple of the target rate. This cause DOP always have the wrong rate chosen. This is also not necessary for PCM playback --- MPD's goal is bit perfect, and it's meaningless to raise to two or four times the PCM sample rate. 3. if sample_rate cannot be synchronized, the test for falling back to PCM is wrong. If the file format is in DSD format such fallback is necessary, whatever the params.dop setting is.
This commit is contained in:
parent
7df0d3b7fc
commit
2127a482da
@ -339,8 +339,9 @@ osx_output_score_sample_rate(Float64 destination_rate, unsigned int source_rate)
|
|||||||
score += (1 - frac_portion) * 1000;
|
score += (1 - frac_portion) * 1000;
|
||||||
// prefer exact matches over other multiples
|
// prefer exact matches over other multiples
|
||||||
score += (int_portion == 1.0) ? 500 : 0;
|
score += (int_portion == 1.0) ? 500 : 0;
|
||||||
// prefer higher multiples if source rate higher than dest rate
|
if (source_rate == destination_rate)
|
||||||
if(source_rate >= destination_rate)
|
score += 1000;
|
||||||
|
else if(source_rate > destination_rate)
|
||||||
score += (int_portion > 1 && int_portion < 100) ? (100 - int_portion) / 100 * 100 : 0;
|
score += (int_portion > 1 && int_portion < 100) ? (100 - int_portion) / 100 * 100 : 0;
|
||||||
else
|
else
|
||||||
score += (int_portion > 1 && int_portion < 100) ? (100 + int_portion) / 100 * 100 : 0;
|
score += (int_portion > 1 && int_portion < 100) ? (100 + int_portion) / 100 * 100 : 0;
|
||||||
@ -349,23 +350,23 @@ osx_output_score_sample_rate(Float64 destination_rate, unsigned int source_rate)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static float
|
static float
|
||||||
osx_output_score_format(const AudioStreamBasicDescription &format_desc, const AudioFormat &format) {
|
osx_output_score_format(const AudioStreamBasicDescription &format_desc, const AudioStreamBasicDescription &target_format) {
|
||||||
float score = 0;
|
float score = 0;
|
||||||
// Score only linear PCM formats (everything else MPD cannot use)
|
// Score only linear PCM formats (everything else MPD cannot use)
|
||||||
if (format_desc.mFormatID == kAudioFormatLinearPCM) {
|
if (format_desc.mFormatID == kAudioFormatLinearPCM) {
|
||||||
score += osx_output_score_sample_rate(format_desc.mSampleRate, format.sample_rate);
|
score += osx_output_score_sample_rate(format_desc.mSampleRate, target_format.mSampleRate);
|
||||||
|
|
||||||
// Just choose the stream / format with the highest number of output channels
|
// Just choose the stream / format with the highest number of output channels
|
||||||
score += format_desc.mChannelsPerFrame * 5;
|
score += format_desc.mChannelsPerFrame * 5;
|
||||||
|
|
||||||
if (format.format == SampleFormat::FLOAT) {
|
if (target_format.mFormatFlags == kLinearPCMFormatFlagIsFloat) {
|
||||||
// for float, prefer the highest bitdepth we have
|
// for float, prefer the highest bitdepth we have
|
||||||
if (format_desc.mBitsPerChannel >= 16)
|
if (format_desc.mBitsPerChannel >= 16)
|
||||||
score += (format_desc.mBitsPerChannel / 8);
|
score += (format_desc.mBitsPerChannel / 8);
|
||||||
} else {
|
} else {
|
||||||
if (format_desc.mBitsPerChannel == ((format.format == SampleFormat::S24_P32) ? 24 : format.GetSampleSize() * 8))
|
if (format_desc.mBitsPerChannel == target_format.mBitsPerChannel)
|
||||||
score += 5;
|
score += 5;
|
||||||
else if (format_desc.mBitsPerChannel > format.GetSampleSize() * 8)
|
else if (format_desc.mBitsPerChannel > target_format.mBitsPerChannel)
|
||||||
score += 1;
|
score += 1;
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -374,7 +375,7 @@ osx_output_score_format(const AudioStreamBasicDescription &format_desc, const Au
|
|||||||
}
|
}
|
||||||
|
|
||||||
static Float64
|
static Float64
|
||||||
osx_output_set_device_format(AudioDeviceID dev_id, const AudioFormat &audio_format)
|
osx_output_set_device_format(AudioDeviceID dev_id, const AudioStreamBasicDescription &target_format)
|
||||||
{
|
{
|
||||||
AudioObjectPropertyAddress aopa = {
|
AudioObjectPropertyAddress aopa = {
|
||||||
kAudioDevicePropertyStreams,
|
kAudioDevicePropertyStreams,
|
||||||
@ -436,8 +437,8 @@ osx_output_set_device_format(AudioDeviceID dev_id, const AudioFormat &audio_form
|
|||||||
// for devices with kAudioStreamAnyRate
|
// for devices with kAudioStreamAnyRate
|
||||||
// we use the requested samplerate here
|
// we use the requested samplerate here
|
||||||
if (format_desc.mSampleRate == kAudioStreamAnyRate)
|
if (format_desc.mSampleRate == kAudioStreamAnyRate)
|
||||||
format_desc.mSampleRate = audio_format.sample_rate;
|
format_desc.mSampleRate = target_format.mSampleRate;
|
||||||
float score = osx_output_score_format(format_desc, audio_format);
|
float score = osx_output_score_format(format_desc, target_format);
|
||||||
|
|
||||||
// print all (linear pcm) formats and their rating
|
// print all (linear pcm) formats and their rating
|
||||||
if(score > 0.0)
|
if(score > 0.0)
|
||||||
@ -785,6 +786,7 @@ OSXOutput::Open(AudioFormat &audio_format)
|
|||||||
params.dop = true;
|
params.dop = true;
|
||||||
asbd.mSampleRate = params.CalcOutputSampleRate(audio_format.sample_rate);
|
asbd.mSampleRate = params.CalcOutputSampleRate(audio_format.sample_rate);
|
||||||
asbd.mBytesPerPacket = 4 * audio_format.channels;
|
asbd.mBytesPerPacket = 4 * audio_format.channels;
|
||||||
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -792,10 +794,10 @@ OSXOutput::Open(AudioFormat &audio_format)
|
|||||||
asbd.mBytesPerFrame = asbd.mBytesPerPacket;
|
asbd.mBytesPerFrame = asbd.mBytesPerPacket;
|
||||||
asbd.mChannelsPerFrame = audio_format.channels;
|
asbd.mChannelsPerFrame = audio_format.channels;
|
||||||
|
|
||||||
Float64 sample_rate = osx_output_set_device_format(dev_id, audio_format);
|
Float64 sample_rate = osx_output_set_device_format(dev_id, asbd);
|
||||||
|
|
||||||
#ifdef ENABLE_DSD
|
#ifdef ENABLE_DSD
|
||||||
if(params.dop && sample_rate != asbd.mSampleRate) { // fall back to PCM in case sample_rate cannot be synchronized
|
if(audio_format.format == SampleFormat::DSD && sample_rate != asbd.mSampleRate) { // fall back to PCM in case sample_rate cannot be synchronized
|
||||||
params.dop = false;
|
params.dop = false;
|
||||||
audio_format.format = SampleFormat::S32;
|
audio_format.format = SampleFormat::S32;
|
||||||
asbd.mBitsPerChannel = 32;
|
asbd.mBitsPerChannel = 32;
|
||||||
|
Loading…
Reference in New Issue
Block a user