Утечка FFmpeg при чтении файлов изображений

При чтении файлов изображений с помощью последней версии FFmpeg я столкнулся с утечкой памяти, которую не могу отследить.

Кажется, что после заполнения AVFrame avcodec_send_packet и avcodec_receive_frame мой вызов av_frame_free на самом деле не освобождает объекты AVBuffer внутри фрейма. Единственное, что я не освобождаю, это AVCodecContext. Если я попытаюсь это сделать, я получу крах.

Я создал этот образец программы, он настолько прост, насколько я могу его понять. Это будет продолжать открывать, читать, а затем закрывать один и тот же файл изображения в цикле. В моей системе происходит утечка памяти с угрожающей скоростью.

#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>

int main(int argc, char **argv) {
    av_register_all();

    while(1) {
        AVFormatContext *fmtCtx = NULL;

        if (avformat_open_input(&fmtCtx, "/path/to/test.jpg", NULL, NULL) == 0) {
            if (avformat_find_stream_info(fmtCtx, NULL) >= 0) {
                for (unsigned int i = 0u; i < fmtCtx -> nb_streams; ++i) {
                    AVStream *stream = fmtCtx -> streams[i];
                    AVCodecContext *codecCtx = stream -> codec;
                    AVCodec *codec = avcodec_find_decoder(codecCtx -> codec_id);

                    if (avcodec_open2(codecCtx, codec, NULL) == 0) {
                        AVPacket packet;

                        if (av_read_frame(fmtCtx, &packet) >= 0) {
                            if (avcodec_send_packet(codecCtx, &packet) == 0) {
                                AVFrame *frame = av_frame_alloc();

                                avcodec_receive_frame(codecCtx, frame);
                                av_frame_free(&frame);
                            }
                        }

                        av_packet_unref(&packet);
                    }
                }
            }

            avformat_close_input(&fmtCtx);
        }
    }

    return 0;
}

person Tim    schedule 16.09.2016    source источник


Ответы (1)


Решение состоит в том, чтобы создать копию AVCodecContext, которая была автоматически создана при открытии файла, и использовать эту копию в avcodec_open2. Это позволяет удалить эту копию с помощью avcodec_free_context.

В последних версиях FFmpeg avcodec_copy_context устарело и заменено на AVCodecParameters. Использование следующего фрагмента в образце программы из вопроса устраняет утечку:

AVCodecParameters *param = avcodec_parameters_alloc();
AVCodecContext *codecCtx = avcodec_alloc_context3(NULL);
AVCodec *codec = avcodec_find_decoder(stream -> codec -> codec_id);

avcodec_parameters_from_context(param, stream -> codec);
avcodec_parameters_to_context(codecCtx, param);
avcodec_parameters_free(&param);
[...]
avcodec_free_context(&codecCtx);
person Tim    schedule 18.09.2016