output/alsa: support 32 bit DSD-over-USB
This commit is contained in:
parent
ddd4f675a2
commit
2803ec2e96
@ -588,7 +588,8 @@ error:
|
|||||||
|
|
||||||
static bool
|
static bool
|
||||||
alsa_setup_dsd(struct alsa_data *ad, struct audio_format *audio_format,
|
alsa_setup_dsd(struct alsa_data *ad, struct audio_format *audio_format,
|
||||||
bool *packed_r, bool *reverse_endian_r, GError **error_r)
|
bool *shift8_r, bool *packed_r, bool *reverse_endian_r,
|
||||||
|
GError **error_r)
|
||||||
{
|
{
|
||||||
assert(ad->dsd_usb);
|
assert(ad->dsd_usb);
|
||||||
assert(audio_format->format == SAMPLE_FORMAT_DSD);
|
assert(audio_format->format == SAMPLE_FORMAT_DSD);
|
||||||
@ -604,6 +605,15 @@ alsa_setup_dsd(struct alsa_data *ad, struct audio_format *audio_format,
|
|||||||
if (!alsa_setup(ad, &usb_format, packed_r, reverse_endian_r, error_r))
|
if (!alsa_setup(ad, &usb_format, packed_r, reverse_endian_r, error_r))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
/* if the device allows only 32 bit, shift all DSD-over-USB
|
||||||
|
samples left by 8 bit and leave the lower 8 bit cleared;
|
||||||
|
the DSD-over-USB documentation does not specify whether
|
||||||
|
this is legal, but there is anecdotical evidence that this
|
||||||
|
is possible (and the only option for some devices) */
|
||||||
|
*shift8_r = usb_format.format == SAMPLE_FORMAT_S32;
|
||||||
|
if (usb_format.format == SAMPLE_FORMAT_S32)
|
||||||
|
usb_format.format = SAMPLE_FORMAT_S24_P32;
|
||||||
|
|
||||||
if (!audio_format_equals(&usb_format, &check)) {
|
if (!audio_format_equals(&usb_format, &check)) {
|
||||||
/* no bit-perfect playback, which is required
|
/* no bit-perfect playback, which is required
|
||||||
for DSD over USB */
|
for DSD over USB */
|
||||||
@ -620,12 +630,13 @@ static bool
|
|||||||
alsa_setup_or_dsd(struct alsa_data *ad, struct audio_format *audio_format,
|
alsa_setup_or_dsd(struct alsa_data *ad, struct audio_format *audio_format,
|
||||||
GError **error_r)
|
GError **error_r)
|
||||||
{
|
{
|
||||||
bool packed, reverse_endian;
|
bool shift8 = false, packed, reverse_endian;
|
||||||
|
|
||||||
const bool dsd_usb = ad->dsd_usb &&
|
const bool dsd_usb = ad->dsd_usb &&
|
||||||
audio_format->format == SAMPLE_FORMAT_DSD;
|
audio_format->format == SAMPLE_FORMAT_DSD;
|
||||||
const bool success = dsd_usb
|
const bool success = dsd_usb
|
||||||
? alsa_setup_dsd(ad, audio_format, &packed, &reverse_endian,
|
? alsa_setup_dsd(ad, audio_format,
|
||||||
|
&shift8, &packed, &reverse_endian,
|
||||||
error_r)
|
error_r)
|
||||||
: alsa_setup(ad, audio_format, &packed, &reverse_endian,
|
: alsa_setup(ad, audio_format, &packed, &reverse_endian,
|
||||||
error_r);
|
error_r);
|
||||||
@ -634,7 +645,7 @@ alsa_setup_or_dsd(struct alsa_data *ad, struct audio_format *audio_format,
|
|||||||
|
|
||||||
pcm_export_open(&ad->export,
|
pcm_export_open(&ad->export,
|
||||||
audio_format->format, audio_format->channels,
|
audio_format->format, audio_format->channels,
|
||||||
dsd_usb, false, packed, reverse_endian);
|
dsd_usb, shift8, packed, reverse_endian);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user