使用libavcodec将音频编码为AAC

13
我正在使用 libavcodec(截至 2010 年 3 月 3 日的最新 git 版本)将原始 PCM 编码为 AAC(启用了 libfaac 支持)。我通过每次使用 codec_context->frame_size 个样本调用 avcodec_encode_audio 来完成此操作。前四个调用成功返回,但第五个调用永远不会返回。当我使用 gdb 调试时,堆栈被破坏。
如果我使用 audacity 将 pcm 数据导出到 .wav 文件中,然后我就可以使用命令行 ffmpeg 转换为 aac,而不会出现任何问题,所以我确定是我做错了什么。
我编写了一个小测试程序来重复我的问题。它从文件中读取测试数据,该文件可在此处获得:http://birdie.protoven.com/audio.pcm(约 2 秒的有符号 16 位 LE pcm)。
如果我直接使用 FAAC,我可以使所有内容正常工作,但如果我只使用 libavcodec,代码可能会更加简洁,因为我还要对视频进行编码,并将两者都写入 mp4。
ffmpeg 版本信息:
FFmpeg version git-c280040, Copyright (c) 2000-2010 the FFmpeg developers
  built on Mar  3 2010 15:40:46 with gcc 4.4.1
  configuration: --enable-libfaac --enable-gpl --enable-nonfree --enable-version3 --enable-postproc --enable-pthreads --enable-debug=3 --enable-shared
  libavutil     50.10. 0 / 50.10. 0
  libavcodec    52.55. 0 / 52.55. 0
  libavformat   52.54. 0 / 52.54. 0
  libavdevice   52. 2. 0 / 52. 2. 0
  libswscale     0.10. 0 /  0.10. 0
  libpostproc   51. 2. 0 / 51. 2. 0

我是否在编解码器上下文中漏掉或设置不正确的东西?任何帮助都将不胜感激!

以下是我的测试代码:

#include <stdio.h>
#include <libavcodec/avcodec.h>

void EncodeTest(int sampleRate, int channels, int audioBitrate,
    uint8_t *audioData, size_t audioSize)
{
    AVCodecContext  *audioCodec;
    AVCodec *codec;
    uint8_t *buf;
    int bufSize, frameBytes;

    avcodec_register_all();

    //Set up audio encoder
    codec = avcodec_find_encoder(CODEC_ID_AAC);
    if (codec == NULL) return;
    audioCodec = avcodec_alloc_context();
    audioCodec->bit_rate = audioBitrate;
    audioCodec->sample_fmt = SAMPLE_FMT_S16;
    audioCodec->sample_rate = sampleRate;
    audioCodec->channels = channels;
    audioCodec->profile = FF_PROFILE_AAC_MAIN;
    audioCodec->time_base = (AVRational){1, sampleRate};
    audioCodec->codec_type = CODEC_TYPE_AUDIO;
    if (avcodec_open(audioCodec, codec) < 0) return;

    bufSize = FF_MIN_BUFFER_SIZE * 10;
    buf = (uint8_t *)malloc(bufSize);
    if (buf == NULL) return;

    frameBytes = audioCodec->frame_size * audioCodec->channels * 2;
    while (audioSize >= frameBytes)
    {
        int packetSize;

        packetSize = avcodec_encode_audio(audioCodec, buf, bufSize, (short *)audioData);
        printf("encoder returned %d bytes of data\n", packetSize);
        audioData += frameBytes;
        audioSize -= frameBytes;
    }
}

int main()
{
    FILE *stream = fopen("audio.pcm", "rb");
    size_t size;
    uint8_t *buf;

    if (stream == NULL)
    {
        printf("Unable to open file\n");
        return 1;
    }

    fseek(stream, 0, SEEK_END);
    size = ftell(stream);
    fseek(stream, 0, SEEK_SET);
    buf = (uint8_t *)malloc(size);
    fread(buf, sizeof(uint8_t), size, stream);
    fclose(stream);

    EncodeTest(32000, 2, 448000, buf, size);
}

你能否与我分享整个项目?我也在使用libav编码视频/音频。 - suitianshi
2个回答

4

如果比特率小于386000,问题似乎会消失。我不确定为什么会这样,因为我可以直接使用FAAC进行比该比特率更高的编码。但是对我来说,128000已经足够好了,所以我能够继续前进。


0

我正在尝试使用AAC格式进行压缩,并在编码中遇到了一些问题。FFmpeg的最新版本(2.8.0)中有一些功能。首先,您是否检查过样本格式是否受支持?在我的版本中,唯一支持的格式是AV_SAMPLE_FMT_FLTP。格式检查示例如下:

/* 检查编码器是否支持给定的样本格式 */ int check_sample_fmt(AVCodec *codec, enum AVSampleFormat sample_fmt) { const enum AVSampleFormat *p = codec->sample_fmts;

while (*p != AV_SAMPLE_FMT_NONE) {
    if (*p == sample_fmt)
        return 1;
    p++;
}
return 0;

}

如果您观察支持的格式,只有AV_SAMPLE_FMT_FLTP受AAC编解码器支持。建议使用swresample进行平面浮点格式转换,或者可以手动完成。为了打开编解码器,应使用带有strict sperimental选项的avcodec_open2。

顺祝商祺


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接