1
mirror of https://github.com/mpv-player/mpv synced 2025-01-24 19:37:30 +01:00

ad_lavc: force channel layout pass-through with demux_rawaudio

Using demux_rawaudio and the --rawaudio-channels option is useful for
testing channel map stuff. The libavcodec PCM decoder normalizes the
channel map to ffmpeg order, though. Prevent this by forcing the
original channel map when using the mp-pcm pseudo decoder entry (used by
demux_rawaudio and stream/tv.c only).
This commit is contained in:
wm4 2013-04-07 01:27:33 +02:00
parent ade08d676f
commit 1c601e84ff

View File

@ -51,6 +51,7 @@ struct priv {
int output_left; int output_left;
int unitsize; int unitsize;
int previous_data_left; // input demuxer packet data int previous_data_left; // input demuxer packet data
bool force_channel_map;
}; };
#define OPT_BASE_STRUCT struct MPOpts #define OPT_BASE_STRUCT struct MPOpts
@ -155,6 +156,7 @@ static int preinit(sh_audio_t *sh)
static int setup_format(sh_audio_t *sh_audio, static int setup_format(sh_audio_t *sh_audio,
const AVCodecContext *lavc_context) const AVCodecContext *lavc_context)
{ {
struct priv *priv = sh_audio->context;
int sample_format = int sample_format =
af_from_avformat(av_get_packed_sample_fmt(lavc_context->sample_fmt)); af_from_avformat(av_get_packed_sample_fmt(lavc_context->sample_fmt));
bool broken_srate = false; bool broken_srate = false;
@ -173,6 +175,10 @@ static int setup_format(sh_audio_t *sh_audio,
// No channel layout or layout disagrees with channel count // No channel layout or layout disagrees with channel count
if (lavc_chmap.num != lavc_context->channels) if (lavc_chmap.num != lavc_context->channels)
mp_chmap_from_channels(&lavc_chmap, lavc_context->channels); mp_chmap_from_channels(&lavc_chmap, lavc_context->channels);
if (priv->force_channel_map) {
if (lavc_chmap.num == sh_audio->channels.num)
lavc_chmap = sh_audio->channels;
}
if (!mp_chmap_equals(&lavc_chmap, &sh_audio->channels) || if (!mp_chmap_equals(&lavc_chmap, &sh_audio->channels) ||
samplerate != sh_audio->samplerate || samplerate != sh_audio->samplerate ||
@ -211,22 +217,25 @@ static int init(sh_audio_t *sh_audio, const char *decoder)
AVCodecContext *lavc_context; AVCodecContext *lavc_context;
AVCodec *lavc_codec; AVCodec *lavc_codec;
struct priv *ctx = talloc_zero(NULL, struct priv);
sh_audio->context = ctx;
if (sh_audio->wf && strcmp(decoder, "pcm") == 0) { if (sh_audio->wf && strcmp(decoder, "pcm") == 0) {
decoder = find_pcm_decoder(tag_map, sh_audio->format, decoder = find_pcm_decoder(tag_map, sh_audio->format,
sh_audio->wf->wBitsPerSample); sh_audio->wf->wBitsPerSample);
} else if (sh_audio->wf && strcmp(decoder, "mp-pcm") == 0) { } else if (sh_audio->wf && strcmp(decoder, "mp-pcm") == 0) {
decoder = find_pcm_decoder(af_map, sh_audio->format, 0); decoder = find_pcm_decoder(af_map, sh_audio->format, 0);
ctx->force_channel_map = true;
} }
lavc_codec = avcodec_find_decoder_by_name(decoder); lavc_codec = avcodec_find_decoder_by_name(decoder);
if (!lavc_codec) { if (!lavc_codec) {
mp_tmsg(MSGT_DECAUDIO, MSGL_ERR, mp_tmsg(MSGT_DECAUDIO, MSGL_ERR,
"Cannot find codec '%s' in libavcodec...\n", decoder); "Cannot find codec '%s' in libavcodec...\n", decoder);
uninit(sh_audio);
return 0; return 0;
} }
struct priv *ctx = talloc_zero(NULL, struct priv);
sh_audio->context = ctx;
lavc_context = avcodec_alloc_context3(lavc_codec); lavc_context = avcodec_alloc_context3(lavc_codec);
ctx->avctx = lavc_context; ctx->avctx = lavc_context;
ctx->avframe = avcodec_alloc_frame(); ctx->avframe = avcodec_alloc_frame();