初始化用于使用 FFmpeg 混合 mkv 的输出文件

Initializing an output file for muxing mkv with FFmpeg

我是 ffmpeg 的第一次用户,我在为 mkv 混合初始化输出文件时遇到了问题。 我按照描述 on the middle of this page 但是它不起作用。

当我调用 avformat_write_header() 函数时 returns 错误代码 -1094995529,这意味着:处理输入时发现无效数据。

当我将输出上下文分配为 mp4 格式时(我在两种情况下都使用 avformat_alloc_output_context2),我没有收到此错误,它成功地写入了 header。 我试图分析 the code of this example 但我无法弄清楚问题是什么(我也尝试了不同的功能,但没有结果)。

这是我的代码:

const char *filename = "c:\Users\MPM\Desktop\test.mkv";
AVFormatContext *mkvVideo = NULL;

int ret = 0;
ret = avformat_alloc_output_context2(&mkvVideo, av_guess_format("matroska", "c:\Users\MPM\Desktop\test.mkv", NULL), "mkv", filename);

std::cout << "\n avformat_alloc_output_context2: " << ret<<"\n";

AVStream* video_stream = avformat_new_stream(mkvVideo, NULL);

video_stream->time_base.den = 25;
video_stream->time_base.num = 1;
video_stream->id = mkvVideo->nb_streams - 1;

video_stream->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
video_stream->codecpar->codec_id = AV_CODEC_ID_H264;
video_stream->codecpar->bit_rate = 40000;
video_stream->codecpar->width = 1920;
video_stream->codecpar->height = 1080;
video_stream->codecpar->format = 0;

// AVCodec *video_codec = avcodec_find_encoder(AV_CODEC_ID_H264);
// AVCodecContext *codecCTx = NULL;

// codecCTx = avcodec_alloc_context3(video_codec);
// avcodec_get_context_defaults3(codecCTx, video_codec);

// ret = avcodec_parameters_to_context(codecCTx, video_stream->codecpar);
// std::cout << "\n avcodec_parameters_to_context: " << ret;

// codecCTx->framerate = { 25, 1 };
// codecCTx->gop_size = 12; 
// codecCTx->pix_fmt = AV_PIX_FMT_YUV420;
// codecCTx->max_b_frames = 1;
// codecCTx->codec_tag = 1;
// codecCTx->codec_type = AVMEDIA_TYPE_VIDEO;

std::cout << std::endl;
av_dump_format(mkvVideo, 0, filename, 1);

ret = avio_open(&mkvVideo->pb, "c:\Users\MPM\Desktop\test.mp4", AVIO_FLAG_READ_WRITE);
std::cout << "\n avio open: " << ret;

ret = avformat_write_header(mkvVideo, NULL);
std::cout << "\n ret: " << ret;

if (ret < 0)
    {
        char buffer[50];
        size_t bufsize = 50;
        av_strerror(ret, buffer, bufsize);
        std::cout << "\n error message: ";
        for (int i = 0; i < bufsize; i++) std::cout << buffer[i];
        exit(1);
    }

有关输出格式(按av_dump_format)的打印输出数据如下:

Output #0, matroska, to 'c:\Users\MPM\Desktop\test.mkv':
Stream #0:0: Video: h264, yuv420p, 1920x1080, q=2-31, 40 kb/s, 25 tbn

我认为它不起作用的原因可能有两个:

  1. AVStream

  2. 缺少一些参数
  3. 一个mkv输出格式需要不同的初始化

所以谁能告诉我我做错了什么?

原来是H264编解码器初始化的问题。 videostream->codecpar->extradata 字段必须分配并填充 NAL units which are streaming parameters. Doing this is an .

因为我的项目不需要流式处理,所以我简单地分配了一些 space 和 avmalloc() 到 extradata 字段,然后我能够初始化输出文件。