Mercurial > mplayer.hg
view av_helpers.c @ 36545:f0e92381681d
Add support for files with MLTI chunks.
Fixes 06.rm and 08_lect_01.rm from FFmpeg trac #2152
Fixes also files from ftp://ftp.aduni.org/videos/
The patch has these limitations:
- no multirate files with MLTI chunks
- no mixed files (eg. MLTI for audio but not for video)
- no MLTI for video together with non-ra audio (eg. ralf)
- only DATA v0 (no v1, versions greater than 1 should not exist)
Files with these feature shuld not really exist anyway.
Also video MLTI files with only one stream are supported but untested
author | rtogni |
---|---|
date | Sun, 19 Jan 2014 15:12:46 +0000 |
parents | d9d308031e9e |
children | 636c79b71de9 |
line wrap: on
line source
/* * Generic libav* helpers * * This file is part of MPlayer. * * MPlayer is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * MPlayer is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with MPlayer; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "libavcodec/avcodec.h" #include "libavformat/avformat.h" #include "mp_msg.h" #include "av_helpers.h" #include "libaf/reorder_ch.h" int avcodec_initialized; int avformat_initialized; static void mp_msp_av_log_callback(void *ptr, int level, const char *fmt, va_list vl) { static int print_prefix=1; AVClass *avc= ptr ? *(AVClass **)ptr : NULL; int type= MSGT_FIXME; int mp_level; switch(level){ case AV_LOG_VERBOSE: mp_level = MSGL_V ; break; case AV_LOG_DEBUG: mp_level= MSGL_V ; break; case AV_LOG_INFO : mp_level= MSGL_INFO; break; case AV_LOG_ERROR: mp_level= MSGL_ERR ; break; default : mp_level= level > AV_LOG_DEBUG ? MSGL_DBG2 : MSGL_ERR; break; } if (ptr && !avc) mp_msg(MSGT_DECVIDEO, MSGL_ERR, "libav* called av_log with context containing a broken AVClass!\n"); if (avc) { if(!strcmp(avc->class_name, "AVCodecContext")){ AVCodecContext *s= ptr; if(s->codec){ if(s->codec->type == AVMEDIA_TYPE_AUDIO){ if(s->codec->decode) type= MSGT_DECAUDIO; }else if(s->codec->type == AVMEDIA_TYPE_VIDEO){ if(s->codec->decode) type= MSGT_DECVIDEO; } //FIXME subtitles, encoders (what msgt for them? there is no appropriate ...) } }else if(!strcmp(avc->class_name, "AVFormatContext")){ AVFormatContext *s= ptr; if(s->iformat) type= MSGT_DEMUXER; else if(s->oformat) type= MSGT_MUXER; } } if (!mp_msg_test(type, mp_level)) return; if(print_prefix && avc) { mp_msg(type, mp_level, "[%s @ %p]", avc->item_name(ptr), avc); } print_prefix= strchr(fmt, '\n') != NULL; mp_msg_va(type, mp_level, fmt, vl); } static void show_av_version(int type, const char *name, int header_ver, int ver, const char *conf) { #ifdef CONFIG_FFMPEG_SO #define FFMPEG_TYPE "external" #else #define FFMPEG_TYPE "internal" #endif mp_msg(type, MSGL_INFO, "%s version %d.%d.%d (" FFMPEG_TYPE ")\n", name, ver >> 16, (ver >> 8) & 0xFF, ver & 0xFF); if (header_ver != ver) mp_msg(type, MSGL_INFO, "Mismatching header version %d.%d.%d\n", header_ver >> 16, (header_ver >> 8) & 0xFF, header_ver & 0xFF); mp_msg(type, MSGL_V, "Configuration: %s\n", conf); } void init_avcodec(void) { if (!avcodec_initialized) { show_av_version(MSGT_DECVIDEO, "libavcodec", LIBAVCODEC_VERSION_INT, avcodec_version(), avcodec_configuration()); avcodec_register_all(); avcodec_initialized = 1; av_log_set_callback(mp_msp_av_log_callback); } } void init_avformat(void) { if (!avformat_initialized) { show_av_version(MSGT_DEMUX, "libavformat", LIBAVFORMAT_VERSION_INT, avformat_version(), avformat_configuration()); av_register_all(); avformat_initialized = 1; av_log_set_callback(mp_msp_av_log_callback); } } int lavc_encode_audio(AVCodecContext *ctx, void *src, int src_len, void *dst, int dst_len) { void *orig_src = src; int bps = av_get_bytes_per_sample(ctx->sample_fmt); int planar = ctx->channels > 1 && av_sample_fmt_is_planar(ctx->sample_fmt); int isac3 = ctx->codec->id == AV_CODEC_ID_AC3; int n; int got; AVPacket pkt; AVFrame *frame = avcodec_alloc_frame(); if ((ctx->channels == 6 || ctx->channels == 5) && (isac3 || !strcmp(ctx->codec->name,"libfaac"))) { reorder_channel_nch(src, AF_CHANNEL_LAYOUT_MPLAYER_DEFAULT, isac3 ? AF_CHANNEL_LAYOUT_LAVC_DEFAULT : AF_CHANNEL_LAYOUT_AAC_DEFAULT, ctx->channels, src_len / bps, bps); } pkt.data = dst; pkt.size = dst_len; frame->nb_samples = src_len / ctx->channels / bps; if (planar) { // TODO: this is horribly inefficient. int ch; src = av_mallocz(src_len); for (ch = 0; ch < ctx->channels; ch++) { uint8_t *tmps = (uint8_t *)orig_src + ch*bps; uint8_t *tmpd = (uint8_t *)src + ch*src_len/ctx->channels; int s; for (s = 0; s < frame->nb_samples; s++) { memcpy(tmpd, tmps, bps); tmps += ctx->channels * bps; tmpd += bps; } } } n = avcodec_fill_audio_frame(frame, ctx->channels, ctx->sample_fmt, src, src_len, 1); if (n < 0) return 0; n = avcodec_encode_audio2(ctx, &pkt, frame, &got); avcodec_free_frame(&frame); if (planar) av_free(src); if (n < 0) return n; return got ? pkt.size : 0; }