Skip to content
Snippets Groups Projects
utils.c 128 KiB
Newer Older
  • Learn to ignore specific revisions
  • Fabrice Bellard's avatar
    Fabrice Bellard committed
    /*
     * utils for libavcodec
    
     * Copyright (c) 2001 Fabrice Bellard
    
     * Copyright (c) 2002-2004 Michael Niedermayer <michaelni@gmx.at>
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     *
    
     * This file is part of FFmpeg.
     *
     * FFmpeg is free software; you can redistribute it and/or
    
     * modify it under the terms of the GNU Lesser General Public
     * License as published by the Free Software Foundation; either
    
     * version 2.1 of the License, or (at your option) any later version.
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     *
    
     * FFmpeg is distributed in the hope that it will be useful,
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     * but WITHOUT ANY WARRANTY; without even the implied warranty of
    
     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
     * Lesser General Public License for more details.
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     *
    
     * You should have received a copy of the GNU Lesser General Public
    
     * License along with FFmpeg; if not, write to the Free Software
    
     * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     */
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    /**
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
     * utils.
     */
    
    #include "libavutil/attributes.h"
    
    #include "libavutil/avstring.h"
    
    #include "libavutil/bprint.h"
    
    #include "libavutil/channel_layout.h"
    
    #include "libavutil/crc.h"
    
    #include "libavutil/frame.h"
    
    #include "libavutil/hwcontext.h"
    
    #include "libavutil/mathematics.h"
    
    #include "libavutil/mem_internal.h"
    
    #include "libavutil/imgutils.h"
    #include "libavutil/samplefmt.h"
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    #include "avcodec.h"
    
    #include "libavutil/opt.h"
    
    #include "mpegvideo.h"
    
    #include "frame_thread_encoder.h"
    
    #include "raw.h"
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    #include <float.h>
    
    #if CONFIG_ICONV
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    #include "libavutil/ffversion.h"
    const char av_codec_ffversion[] = "FFmpeg version " FFMPEG_VERSION;
    
    
    #if HAVE_PTHREADS || HAVE_W32THREADS || HAVE_OS2THREADS
    static int default_lockmgr_cb(void **arg, enum AVLockOp op)
    {
        void * volatile * mutex = arg;
        int err;
    
        switch (op) {
        case AV_LOCK_CREATE:
            return 0;
        case AV_LOCK_OBTAIN:
            if (!*mutex) {
                pthread_mutex_t *tmp = av_malloc(sizeof(pthread_mutex_t));
                if (!tmp)
                    return AVERROR(ENOMEM);
                if ((err = pthread_mutex_init(tmp, NULL))) {
                    av_free(tmp);
                    return AVERROR(err);
                }
                if (avpriv_atomic_ptr_cas(mutex, NULL, tmp)) {
                    pthread_mutex_destroy(tmp);
                    av_free(tmp);
                }
            }
    
            if ((err = pthread_mutex_lock(*mutex)))
                return AVERROR(err);
    
            return 0;
        case AV_LOCK_RELEASE:
            if ((err = pthread_mutex_unlock(*mutex)))
                return AVERROR(err);
    
            return 0;
        case AV_LOCK_DESTROY:
            if (*mutex)
                pthread_mutex_destroy(*mutex);
            av_free(*mutex);
            avpriv_atomic_ptr_cas(mutex, *mutex, NULL);
            return 0;
        }
        return 1;
    }
    static int (*lockmgr_cb)(void **mutex, enum AVLockOp op) = default_lockmgr_cb;
    #else
    static int (*lockmgr_cb)(void **mutex, enum AVLockOp op) = NULL;
    #endif
    
    
    
    volatile int ff_avcodec_locked;
    
    Luca Barbato's avatar
    Luca Barbato committed
    static int volatile entangled_thread_counter = 0;
    
    static void *codec_mutex;
    
    void av_fast_padded_malloc(void *ptr, unsigned int *size, size_t min_size)
    {
        uint8_t **p = ptr;
    
        if (min_size > SIZE_MAX - AV_INPUT_BUFFER_PADDING_SIZE) {
    
            av_freep(p);
    
        if (!ff_fast_malloc(p, size, min_size + AV_INPUT_BUFFER_PADDING_SIZE, 1))
            memset(*p + min_size, 0, AV_INPUT_BUFFER_PADDING_SIZE);
    
    void av_fast_padded_mallocz(void *ptr, unsigned int *size, size_t min_size)
    {
        uint8_t **p = ptr;
    
        if (min_size > SIZE_MAX - AV_INPUT_BUFFER_PADDING_SIZE) {
    
            av_freep(p);
            *size = 0;
            return;
        }
    
        if (!ff_fast_malloc(p, size, min_size + AV_INPUT_BUFFER_PADDING_SIZE, 1))
            memset(*p, 0, min_size + AV_INPUT_BUFFER_PADDING_SIZE);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    /* encoder management */
    
    static AVCodec *first_avcodec = NULL;
    
    static AVCodec **last_avcodec = &first_avcodec;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    AVCodec *av_codec_next(const AVCodec *c)
    {
    
    Luca Barbato's avatar
    Luca Barbato committed
        if (c)
            return c->next;
        else
            return first_avcodec;
    
    static av_cold void avcodec_init(void)
    
    {
        static int initialized = 0;
    
        if (initialized != 0)
            return;
        initialized = 1;
    
    
        if (CONFIG_ME_CMP)
            ff_me_cmp_init_static();
    
    int av_codec_is_encoder(const AVCodec *codec)
    
        return codec && (codec->encode_sub || codec->encode2);
    
    int av_codec_is_decoder(const AVCodec *codec)
    
    av_cold void avcodec_register(AVCodec *codec)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
        AVCodec **p;
    
    
        while(*p || avpriv_atomic_ptr_cas((void * volatile *)p, NULL, codec))
    
    
        if (codec->init_static_data)
            codec->init_static_data(codec);
    
    unsigned avcodec_get_edge_width(void)
    {
        return EDGE_WIDTH;
    }
    
    Luca Barbato's avatar
    Luca Barbato committed
    void avcodec_set_dimensions(AVCodecContext *s, int width, int height)
    {
    
        int ret = ff_set_dimensions(s, width, height);
        if (ret < 0) {
            av_log(s, AV_LOG_WARNING, "Failed to set dimensions %d %d\n", width, height);
        }
    
    }
    #endif
    
    int ff_set_dimensions(AVCodecContext *s, int width, int height)
    {
        int ret = av_image_check_size(width, height, 0, s);
    
        if (ret < 0)
            width = height = 0;
    
    Luca Barbato's avatar
    Luca Barbato committed
        s->coded_width  = width;
        s->coded_height = height;
    
        s->width        = AV_CEIL_RSHIFT(width,  s->lowres);
        s->height       = AV_CEIL_RSHIFT(height, s->lowres);
    
    int ff_set_sar(AVCodecContext *avctx, AVRational sar)
    {
        int ret = av_image_check_sar(avctx->width, avctx->height, sar);
    
        if (ret < 0) {
    
            av_log(avctx, AV_LOG_WARNING, "ignoring invalid SAR: %d/%d\n",
    
                   sar.num, sar.den);
            avctx->sample_aspect_ratio = (AVRational){ 0, 1 };
            return ret;
        } else {
            avctx->sample_aspect_ratio = sar;
        }
        return 0;
    }
    
    
    int ff_side_data_update_matrix_encoding(AVFrame *frame,
                                            enum AVMatrixEncoding matrix_encoding)
    {
        AVFrameSideData *side_data;
        enum AVMatrixEncoding *data;
    
        side_data = av_frame_get_side_data(frame, AV_FRAME_DATA_MATRIXENCODING);
        if (!side_data)
            side_data = av_frame_new_side_data(frame, AV_FRAME_DATA_MATRIXENCODING,
                                               sizeof(enum AVMatrixEncoding));
    
        if (!side_data)
            return AVERROR(ENOMEM);
    
        data  = (enum AVMatrixEncoding*)side_data->data;
        *data = matrix_encoding;
    
        return 0;
    }
    
    
    void avcodec_align_dimensions2(AVCodecContext *s, int *width, int *height,
                                   int linesize_align[AV_NUM_DATA_POINTERS])
    {
        int i;
    
    Luca Barbato's avatar
    Luca Barbato committed
        int w_align = 1;
        int h_align = 1;
    
        AVPixFmtDescriptor const *desc = av_pix_fmt_desc_get(s->pix_fmt);
    
    
        if (desc) {
            w_align = 1 << desc->log2_chroma_w;
            h_align = 1 << desc->log2_chroma_h;
        }
    
    Luca Barbato's avatar
    Luca Barbato committed
        switch (s->pix_fmt) {
    
        case AV_PIX_FMT_YUV420P:
        case AV_PIX_FMT_YUYV422:
    
        case AV_PIX_FMT_YVYU422:
    
        case AV_PIX_FMT_UYVY422:
        case AV_PIX_FMT_YUV422P:
        case AV_PIX_FMT_YUV440P:
        case AV_PIX_FMT_YUV444P:
        case AV_PIX_FMT_GBRP:
    
        case AV_PIX_FMT_GBRAP:
    
        case AV_PIX_FMT_GRAY8:
        case AV_PIX_FMT_GRAY16BE:
        case AV_PIX_FMT_GRAY16LE:
        case AV_PIX_FMT_YUVJ420P:
        case AV_PIX_FMT_YUVJ422P:
        case AV_PIX_FMT_YUVJ440P:
        case AV_PIX_FMT_YUVJ444P:
        case AV_PIX_FMT_YUVA420P:
    
        case AV_PIX_FMT_YUVA422P:
        case AV_PIX_FMT_YUVA444P:
    
        case AV_PIX_FMT_YUV420P9LE:
        case AV_PIX_FMT_YUV420P9BE:
        case AV_PIX_FMT_YUV420P10LE:
        case AV_PIX_FMT_YUV420P10BE:
    
        case AV_PIX_FMT_YUV420P12LE:
        case AV_PIX_FMT_YUV420P12BE:
        case AV_PIX_FMT_YUV420P14LE:
        case AV_PIX_FMT_YUV420P14BE:
    
        case AV_PIX_FMT_YUV420P16LE:
        case AV_PIX_FMT_YUV420P16BE:
    
        case AV_PIX_FMT_YUVA420P9LE:
        case AV_PIX_FMT_YUVA420P9BE:
        case AV_PIX_FMT_YUVA420P10LE:
        case AV_PIX_FMT_YUVA420P10BE:
        case AV_PIX_FMT_YUVA420P16LE:
        case AV_PIX_FMT_YUVA420P16BE:
    
        case AV_PIX_FMT_YUV422P9LE:
        case AV_PIX_FMT_YUV422P9BE:
        case AV_PIX_FMT_YUV422P10LE:
        case AV_PIX_FMT_YUV422P10BE:
    
        case AV_PIX_FMT_YUV422P12LE:
        case AV_PIX_FMT_YUV422P12BE:
        case AV_PIX_FMT_YUV422P14LE:
        case AV_PIX_FMT_YUV422P14BE:
    
        case AV_PIX_FMT_YUV422P16LE:
        case AV_PIX_FMT_YUV422P16BE:
    
        case AV_PIX_FMT_YUVA422P9LE:
        case AV_PIX_FMT_YUVA422P9BE:
    
        case AV_PIX_FMT_YUVA422P10LE:
        case AV_PIX_FMT_YUVA422P10BE:
    
        case AV_PIX_FMT_YUVA422P16LE:
        case AV_PIX_FMT_YUVA422P16BE:
    
        case AV_PIX_FMT_YUV440P10LE:
        case AV_PIX_FMT_YUV440P10BE:
        case AV_PIX_FMT_YUV440P12LE:
        case AV_PIX_FMT_YUV440P12BE:
    
        case AV_PIX_FMT_YUV444P9LE:
        case AV_PIX_FMT_YUV444P9BE:
        case AV_PIX_FMT_YUV444P10LE:
        case AV_PIX_FMT_YUV444P10BE:
    
        case AV_PIX_FMT_YUV444P12LE:
        case AV_PIX_FMT_YUV444P12BE:
        case AV_PIX_FMT_YUV444P14LE:
        case AV_PIX_FMT_YUV444P14BE:
    
        case AV_PIX_FMT_YUV444P16LE:
        case AV_PIX_FMT_YUV444P16BE:
    
        case AV_PIX_FMT_YUVA444P9LE:
        case AV_PIX_FMT_YUVA444P9BE:
        case AV_PIX_FMT_YUVA444P10LE:
        case AV_PIX_FMT_YUVA444P10BE:
    
        case AV_PIX_FMT_YUVA444P16LE:
        case AV_PIX_FMT_YUVA444P16BE:
    
        case AV_PIX_FMT_GBRP9LE:
        case AV_PIX_FMT_GBRP9BE:
        case AV_PIX_FMT_GBRP10LE:
        case AV_PIX_FMT_GBRP10BE:
    
        case AV_PIX_FMT_GBRP12LE:
        case AV_PIX_FMT_GBRP12BE:
        case AV_PIX_FMT_GBRP14LE:
        case AV_PIX_FMT_GBRP14BE:
    
        case AV_PIX_FMT_GBRP16LE:
        case AV_PIX_FMT_GBRP16BE:
    
    Kieran Kunhya's avatar
    Kieran Kunhya committed
        case AV_PIX_FMT_GBRAP12LE:
        case AV_PIX_FMT_GBRAP12BE:
    
        case AV_PIX_FMT_GBRAP16LE:
        case AV_PIX_FMT_GBRAP16BE:
    
            w_align = 16; //FIXME assume 16 pixel per macroblock
            h_align = 16 * 2; // interlaced needs 2 macroblocks height
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        case AV_PIX_FMT_YUVJ411P:
    
    Luca Barbato's avatar
    Luca Barbato committed
            w_align = 32;
    
    Luca Barbato's avatar
    Luca Barbato committed
            if (s->codec_id == AV_CODEC_ID_SVQ1) {
                w_align = 64;
                h_align = 64;
    
    Luca Barbato's avatar
    Luca Barbato committed
            if (s->codec_id == AV_CODEC_ID_RPZA) {
                w_align = 4;
                h_align = 4;
    
        case AV_PIX_FMT_PAL8:
        case AV_PIX_FMT_BGR8:
        case AV_PIX_FMT_RGB8:
    
            if (s->codec_id == AV_CODEC_ID_SMC ||
                s->codec_id == AV_CODEC_ID_CINEPAK) {
    
    Luca Barbato's avatar
    Luca Barbato committed
                w_align = 4;
                h_align = 4;
    
            if (s->codec_id == AV_CODEC_ID_JV) {
                w_align = 8;
                h_align = 8;
            }
    
    Luca Barbato's avatar
    Luca Barbato committed
            if ((s->codec_id == AV_CODEC_ID_MSZH) ||
                (s->codec_id == AV_CODEC_ID_ZLIB)) {
                w_align = 4;
                h_align = 4;
    
            break;
        case AV_PIX_FMT_RGB24:
            if (s->codec_id == AV_CODEC_ID_CINEPAK) {
                w_align = 4;
                h_align = 4;
            }
    
        if (s->codec_id == AV_CODEC_ID_IFF_ILBM) {
    
            w_align = FFMAX(w_align, 8);
    
    Luca Barbato's avatar
    Luca Barbato committed
        *width  = FFALIGN(*width, w_align);
        *height = FFALIGN(*height, h_align);
    
        if (s->codec_id == AV_CODEC_ID_H264 || s->lowres) {
    
    Luca Barbato's avatar
    Luca Barbato committed
            // some of the optimized chroma MC reads one line too much
    
            // which is also done in mpeg decoders with lowres > 0
    
    Luca Barbato's avatar
    Luca Barbato committed
            *height += 2;
    
    
            // H.264 uses edge emulation for out of frame motion vectors, for this
            // it requires a temporary area large enough to hold a 21x21 block,
            // increasing witdth ensure that the temporary area is large enough,
            // the next rounded up width is 32
    
    Luca Barbato's avatar
    Luca Barbato committed
    void avcodec_align_dimensions(AVCodecContext *s, int *width, int *height)
    {
    
        const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(s->pix_fmt);
        int chroma_shift = desc->log2_chroma_w;
    
        int linesize_align[AV_NUM_DATA_POINTERS];
    
        avcodec_align_dimensions2(s, width, height, linesize_align);
    
    Luca Barbato's avatar
    Luca Barbato committed
        align               = FFMAX(linesize_align[0], linesize_align[3]);
    
        linesize_align[1] <<= chroma_shift;
        linesize_align[2] <<= chroma_shift;
    
    Luca Barbato's avatar
    Luca Barbato committed
        align               = FFMAX3(align, linesize_align[1], linesize_align[2]);
        *width              = FFALIGN(*width, align);
    
    int avcodec_enum_to_chroma_pos(int *xpos, int *ypos, enum AVChromaLocation pos)
    {
        if (pos <= AVCHROMA_LOC_UNSPECIFIED || pos >= AVCHROMA_LOC_NB)
            return AVERROR(EINVAL);
        pos--;
    
        *xpos = (pos&1) * 128;
        *ypos = ((pos>>1)^(pos<4)) * 128;
    
        return 0;
    }
    
    
    enum AVChromaLocation avcodec_chroma_pos_to_enum(int xpos, int ypos)
    {
        int pos, xout, yout;
    
        for (pos = AVCHROMA_LOC_UNSPECIFIED + 1; pos < AVCHROMA_LOC_NB; pos++) {
            if (avcodec_enum_to_chroma_pos(&xout, &yout, pos) == 0 && xout == xpos && yout == ypos)
                return pos;
        }
        return AVCHROMA_LOC_UNSPECIFIED;
    }
    
    
    int avcodec_fill_audio_frame(AVFrame *frame, int nb_channels,
                                 enum AVSampleFormat sample_fmt, const uint8_t *buf,
                                 int buf_size, int align)
    {
        int ch, planar, needed_size, ret = 0;
    
        needed_size = av_samples_get_buffer_size(NULL, nb_channels,
                                                 frame->nb_samples, sample_fmt,
                                                 align);
        if (buf_size < needed_size)
            return AVERROR(EINVAL);
    
        planar = av_sample_fmt_is_planar(sample_fmt);
        if (planar && nb_channels > AV_NUM_DATA_POINTERS) {
    
            if (!(frame->extended_data = av_mallocz_array(nb_channels,
    
                                                    sizeof(*frame->extended_data))))
                return AVERROR(ENOMEM);
        } else {
            frame->extended_data = frame->data;
        }
    
        if ((ret = av_samples_fill_arrays(frame->extended_data, &frame->linesize[0],
    
                                          (uint8_t *)(intptr_t)buf, nb_channels, frame->nb_samples,
    
                                          sample_fmt, align)) < 0) {
            if (frame->extended_data != frame->data)
    
                av_freep(&frame->extended_data);
    
            return ret;
        }
        if (frame->extended_data != frame->data) {
            for (ch = 0; ch < AV_NUM_DATA_POINTERS; ch++)
                frame->data[ch] = frame->extended_data[ch];
        }
    
        return ret;
    }
    
    
    static int update_frame_pool(AVCodecContext *avctx, AVFrame *frame)
    
        FramePool *pool = avctx->internal->pool;
        int i, ret;
    
        switch (avctx->codec_type) {
        case AVMEDIA_TYPE_VIDEO: {
    
            uint8_t *data[4];
            int linesize[4];
    
            int size[4] = { 0 };
            int w = frame->width;
            int h = frame->height;
            int tmpsize, unaligned;
    
            if (pool->format == frame->format &&
                pool->width == frame->width && pool->height == frame->height)
                return 0;
    
            avcodec_align_dimensions2(avctx, &w, &h, pool->stride_align);
    
            do {
                // NOTE: do not align linesizes individually, this breaks e.g. assumptions
                // that linesize[0] == 2*linesize[1] in the MPEG-encoder for 4:2:2
    
                ret = av_image_fill_linesizes(linesize, avctx->pix_fmt, w);
                if (ret < 0)
                    return ret;
    
                // increase alignment of w for next try (rhs gives the lowest bit set in w)
    
    Luca Barbato's avatar
    Luca Barbato committed
                w += w & ~(w - 1);
    
    Luca Barbato's avatar
    Luca Barbato committed
                for (i = 0; i < 4; i++)
    
                    unaligned |= linesize[i] % pool->stride_align[i];
    
            tmpsize = av_image_fill_pointers(data, avctx->pix_fmt, h,
                                             NULL, linesize);
    
            for (i = 0; i < 3 && data[i + 1]; i++)
                size[i] = data[i + 1] - data[i];
            size[i] = tmpsize - (data[i] - data[0]);
    
            for (i = 0; i < 4; i++) {
                av_buffer_pool_uninit(&pool->pools[i]);
    
                pool->linesize[i] = linesize[i];
    
                    pool->pools[i] = av_buffer_pool_init(size[i] + 16 + STRIDE_ALIGN - 1,
    
                                                         CONFIG_MEMORY_POISONING ?
                                                            NULL :
                                                            av_buffer_allocz);
    
                    if (!pool->pools[i]) {
                        ret = AVERROR(ENOMEM);
                        goto fail;
                    }
                }
            }
            pool->format = frame->format;
            pool->width  = frame->width;
            pool->height = frame->height;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    
    
            break;
            }
        case AVMEDIA_TYPE_AUDIO: {
    
            int ch     = av_frame_get_channels(frame); //av_get_channel_layout_nb_channels(frame->channel_layout);
    
            int planar = av_sample_fmt_is_planar(frame->format);
            int planes = planar ? ch : 1;
    
            if (pool->format == frame->format && pool->planes == planes &&
                pool->channels == ch && frame->nb_samples == pool->samples)
                return 0;
    
            av_buffer_pool_uninit(&pool->pools[0]);
            ret = av_samples_get_buffer_size(&pool->linesize[0], ch,
                                             frame->nb_samples, frame->format, 0);
            if (ret < 0)
                goto fail;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    
    
            pool->pools[0] = av_buffer_pool_init(pool->linesize[0], NULL);
            if (!pool->pools[0]) {
                ret = AVERROR(ENOMEM);
                goto fail;
            }
    
            pool->format     = frame->format;
            pool->planes     = planes;
            pool->channels   = ch;
            pool->samples = frame->nb_samples;
            break;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            }
    
        default: av_assert0(0);
        }
        return 0;
    fail:
        for (i = 0; i < 4; i++)
            av_buffer_pool_uninit(&pool->pools[i]);
        pool->format = -1;
        pool->planes = pool->channels = pool->samples = 0;
        pool->width  = pool->height = 0;
        return ret;
    }
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    
    
    static int audio_get_buffer(AVCodecContext *avctx, AVFrame *frame)
    {
        FramePool *pool = avctx->internal->pool;
        int planes = pool->planes;
        int i;
    
        frame->linesize[0] = pool->linesize[0];
    
        if (planes > AV_NUM_DATA_POINTERS) {
    
            frame->extended_data = av_mallocz_array(planes, sizeof(*frame->extended_data));
    
            frame->nb_extended_buf = planes - AV_NUM_DATA_POINTERS;
    
            frame->extended_buf  = av_mallocz_array(frame->nb_extended_buf,
    
                                              sizeof(*frame->extended_buf));
            if (!frame->extended_data || !frame->extended_buf) {
                av_freep(&frame->extended_data);
                av_freep(&frame->extended_buf);
                return AVERROR(ENOMEM);
    
            frame->extended_data = frame->data;
    
            av_assert0(frame->nb_extended_buf == 0);
        }
    
    
        for (i = 0; i < FFMIN(planes, AV_NUM_DATA_POINTERS); i++) {
            frame->buf[i] = av_buffer_pool_get(pool->pools[0]);
            if (!frame->buf[i])
                goto fail;
            frame->extended_data[i] = frame->data[i] = frame->buf[i]->data;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        }
    
        for (i = 0; i < frame->nb_extended_buf; i++) {
            frame->extended_buf[i] = av_buffer_pool_get(pool->pools[0]);
            if (!frame->extended_buf[i])
                goto fail;
            frame->extended_data[i + AV_NUM_DATA_POINTERS] = frame->extended_buf[i]->data;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        }
    
    
        if (avctx->debug & FF_DEBUG_BUFFERS)
            av_log(avctx, AV_LOG_DEBUG, "default_get_buffer called on frame %p", frame);
    
        return 0;
    fail:
        av_frame_unref(frame);
        return AVERROR(ENOMEM);
    }
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    
    
    static int video_get_buffer(AVCodecContext *s, AVFrame *pic)
    {
        FramePool *pool = s->internal->pool;
    
        const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(pic->format);
    
        if (pic->data[0] || pic->data[1] || pic->data[2] || pic->data[3]) {
            av_log(s, AV_LOG_ERROR, "pic->data[*]!=NULL in avcodec_default_get_buffer\n");
    
        if (!desc) {
            av_log(s, AV_LOG_ERROR,
                "Unable to get pixel format descriptor for format %s\n",
                av_get_pix_fmt_name(pic->format));
            return AVERROR(EINVAL);
        }
    
    
        memset(pic->data, 0, sizeof(pic->data));
    
        pic->extended_data = pic->data;
    
    
        for (i = 0; i < 4 && pool->pools[i]; i++) {
            pic->linesize[i] = pool->linesize[i];
    
            pic->buf[i] = av_buffer_pool_get(pool->pools[i]);
            if (!pic->buf[i])
                goto fail;
    
    
            pic->data[i] = pic->buf[i]->data;
    
        }
        for (; i < AV_NUM_DATA_POINTERS; i++) {
            pic->data[i] = NULL;
            pic->linesize[i] = 0;
        }
    
        if (desc->flags & AV_PIX_FMT_FLAG_PAL ||
            desc->flags & AV_PIX_FMT_FLAG_PSEUDOPAL)
            avpriv_set_systematic_pal2((uint32_t *)pic->data[1], pic->format);
    
    Luca Barbato's avatar
    Luca Barbato committed
        if (s->debug & FF_DEBUG_BUFFERS)
    
            av_log(s, AV_LOG_DEBUG, "default_get_buffer called on pic %p\n", pic);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        return 0;
    
    fail:
        av_frame_unref(pic);
        return AVERROR(ENOMEM);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    }
    
    
    void ff_color_frame(AVFrame *frame, const int c[4])
    
    {
        const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(frame->format);
        int p, y, x;
    
    
        av_assert0(desc->flags & AV_PIX_FMT_FLAG_PLANAR);
    
    
        for (p = 0; p<desc->nb_components; p++) {
            uint8_t *dst = frame->data[p];
            int is_chroma = p == 1 || p == 2;
    
            int bytes  = is_chroma ? AV_CEIL_RSHIFT(frame->width,  desc->log2_chroma_w) : frame->width;
            int height = is_chroma ? AV_CEIL_RSHIFT(frame->height, desc->log2_chroma_h) : frame->height;
    
            for (y = 0; y < height; y++) {
    
                    for (x = 0; x<bytes; x++)
                        ((uint16_t*)dst)[x] = c[p];
                }else
                    memset(dst, c[p], bytes);
                dst += frame->linesize[p];
            }
        }
    }
    
    
    int avcodec_default_get_buffer2(AVCodecContext *avctx, AVFrame *frame, int flags)
    
        int ret;
    
        if ((ret = update_frame_pool(avctx, frame)) < 0)
            return ret;
    
    
        switch (avctx->codec_type) {
        case AVMEDIA_TYPE_VIDEO:
            return video_get_buffer(avctx, frame);
        case AVMEDIA_TYPE_AUDIO:
            return audio_get_buffer(avctx, frame);
        default:
            return -1;
        }
    }
    
    
    static int add_metadata_from_side_data(AVPacket *avpkt, AVFrame *frame)
    {
        int size;
        const uint8_t *side_metadata;
    
        AVDictionary **frame_md = avpriv_frame_get_metadatap(frame);
    
        side_metadata = av_packet_get_side_data(avpkt,
                                                AV_PKT_DATA_STRINGS_METADATA, &size);
        return av_packet_unpack_dictionary(side_metadata, size, frame_md);
    }
    
    int ff_init_buffer_info(AVCodecContext *avctx, AVFrame *frame)
    
        AVPacket *pkt = avctx->internal->pkt;
    
            enum AVPacketSideDataType packet;
            enum AVFrameSideDataType frame;
        } sd[] = {
    
            { AV_PKT_DATA_REPLAYGAIN ,                AV_FRAME_DATA_REPLAYGAIN },
            { AV_PKT_DATA_DISPLAYMATRIX,              AV_FRAME_DATA_DISPLAYMATRIX },
            { AV_PKT_DATA_STEREO3D,                   AV_FRAME_DATA_STEREO3D },
            { AV_PKT_DATA_AUDIO_SERVICE_TYPE,         AV_FRAME_DATA_AUDIO_SERVICE_TYPE },
            { AV_PKT_DATA_MASTERING_DISPLAY_METADATA, AV_FRAME_DATA_MASTERING_DISPLAY_METADATA },
    
    
        if (pkt) {
            frame->pkt_pts = pkt->pts;
            av_frame_set_pkt_pos     (frame, pkt->pos);
            av_frame_set_pkt_duration(frame, pkt->duration);
            av_frame_set_pkt_size    (frame, pkt->size);
    
            for (i = 0; i < FF_ARRAY_ELEMS(sd); i++) {
                int size;
                uint8_t *packet_sd = av_packet_get_side_data(pkt, sd[i].packet, &size);
                if (packet_sd) {
                    AVFrameSideData *frame_sd = av_frame_new_side_data(frame,
                                                                       sd[i].frame,
                                                                       size);
                    if (!frame_sd)
                        return AVERROR(ENOMEM);
    
                    memcpy(frame_sd->data, packet_sd, size);
                }
    
            add_metadata_from_side_data(pkt, frame);
    
        } else {
            frame->pkt_pts = AV_NOPTS_VALUE;
    
            av_frame_set_pkt_pos     (frame, -1);
            av_frame_set_pkt_duration(frame, 0);
            av_frame_set_pkt_size    (frame, -1);
    
        frame->reordered_opaque = avctx->reordered_opaque;
    
        if (frame->color_primaries == AVCOL_PRI_UNSPECIFIED)
            frame->color_primaries = avctx->color_primaries;
        if (frame->color_trc == AVCOL_TRC_UNSPECIFIED)
            frame->color_trc = avctx->color_trc;
    
        if (av_frame_get_colorspace(frame) == AVCOL_SPC_UNSPECIFIED)
            av_frame_set_colorspace(frame, avctx->colorspace);
        if (av_frame_get_color_range(frame) == AVCOL_RANGE_UNSPECIFIED)
            av_frame_set_color_range(frame, avctx->color_range);
    
        if (frame->chroma_location == AVCHROMA_LOC_UNSPECIFIED)
            frame->chroma_location = avctx->chroma_sample_location;
    
        switch (avctx->codec->type) {
    
        case AVMEDIA_TYPE_VIDEO:
    
            frame->format              = avctx->pix_fmt;
    
            if (!frame->sample_aspect_ratio.num)
                frame->sample_aspect_ratio = avctx->sample_aspect_ratio;
    
            if (frame->width && frame->height &&
                av_image_check_sar(frame->width, frame->height,
    
                                   frame->sample_aspect_ratio) < 0) {
                av_log(avctx, AV_LOG_WARNING, "ignoring invalid SAR: %u/%u\n",
                       frame->sample_aspect_ratio.num,
                       frame->sample_aspect_ratio.den);
                frame->sample_aspect_ratio = (AVRational){ 0, 1 };
            }
    
    
            break;
        case AVMEDIA_TYPE_AUDIO:
    
            if (!frame->sample_rate)
                frame->sample_rate    = avctx->sample_rate;
            if (frame->format < 0)
                frame->format         = avctx->sample_fmt;
    
            if (!frame->channel_layout) {
                if (avctx->channel_layout) {
                     if (av_get_channel_layout_nb_channels(avctx->channel_layout) !=
                         avctx->channels) {
                         av_log(avctx, AV_LOG_ERROR, "Inconsistent channel "
                                "configuration.\n");
                         return AVERROR(EINVAL);
                     }
    
                    frame->channel_layout = avctx->channel_layout;
                } else {
                    if (avctx->channels > FF_SANE_NB_CHANNELS) {
                        av_log(avctx, AV_LOG_ERROR, "Too many channels: %d.\n",
                               avctx->channels);
                        return AVERROR(ENOSYS);
                    }
                }
            }
    
            av_frame_set_channels(frame, avctx->channels);
    
    int ff_decode_frame_props(AVCodecContext *avctx, AVFrame *frame)
    {
    
        return ff_init_buffer_info(avctx, frame);
    
    static void validate_avframe_allocation(AVCodecContext *avctx, AVFrame *frame)
    {
        if (avctx->codec_type == AVMEDIA_TYPE_VIDEO) {
            int i;
            int num_planes = av_pix_fmt_count_planes(frame->format);
            const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(frame->format);
            int flags = desc ? desc->flags : 0;
            if (num_planes == 1 && (flags & AV_PIX_FMT_FLAG_PAL))
                num_planes = 2;
            for (i = 0; i < num_planes; i++) {
                av_assert0(frame->data[i]);
            }
            // For now do not enforce anything for palette of pseudopal formats
            if (num_planes == 1 && (flags & AV_PIX_FMT_FLAG_PSEUDOPAL))
                num_planes = 2;
            // For formats without data like hwaccel allow unused pointers to be non-NULL.
            for (i = num_planes; num_planes > 0 && i < FF_ARRAY_ELEMS(frame->data); i++) {
                if (frame->data[i])
                    av_log(avctx, AV_LOG_ERROR, "Buffer returned by get_buffer2() did not zero unused plane pointers\n");
                frame->data[i] = NULL;
            }
        }
    }
    
    
    static int get_buffer_internal(AVCodecContext *avctx, AVFrame *frame, int flags)
    
        const AVHWAccel *hwaccel = avctx->hwaccel;
    
        if (avctx->codec_type == AVMEDIA_TYPE_VIDEO) {
            if ((ret = av_image_check_size(avctx->width, avctx->height, 0, avctx)) < 0 || avctx->pix_fmt<0) {
                av_log(avctx, AV_LOG_ERROR, "video_get_buffer: image parameters invalid\n");
                return AVERROR(EINVAL);
            }
    
            if (frame->width <= 0 || frame->height <= 0) {
    
                frame->width  = FFMAX(avctx->width,  AV_CEIL_RSHIFT(avctx->coded_width,  avctx->lowres));
                frame->height = FFMAX(avctx->height, AV_CEIL_RSHIFT(avctx->coded_height, avctx->lowres));
    
    
            if (frame->data[0] || frame->data[1] || frame->data[2] || frame->data[3]) {
                av_log(avctx, AV_LOG_ERROR, "pic->data[*]!=NULL in get_buffer_internal\n");
                return AVERROR(EINVAL);
            }
    
        ret = ff_decode_frame_props(avctx, frame);
        if (ret < 0)
            return ret;
    
        if (hwaccel) {
            if (hwaccel->alloc_frame) {
                ret = hwaccel->alloc_frame(avctx, frame);
                goto end;
            }
        } else
            avctx->sw_pix_fmt = avctx->pix_fmt;
    
        ret = avctx->get_buffer2(avctx, frame, flags);
    
        if (ret >= 0)
            validate_avframe_allocation(avctx, frame);
    
        if (avctx->codec_type == AVMEDIA_TYPE_VIDEO && !override_dimensions) {
    
            frame->width  = avctx->width;
            frame->height = avctx->height;
        }
    
        return ret;
    
    int ff_get_buffer(AVCodecContext *avctx, AVFrame *frame, int flags)
    {
        int ret = get_buffer_internal(avctx, frame, flags);
    
            av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
    
        return ret;
    }
    
    static int reget_buffer_internal(AVCodecContext *avctx, AVFrame *frame)
    
        av_assert0(avctx->codec_type == AVMEDIA_TYPE_VIDEO);
    
        if (frame->data[0] && (frame->width != avctx->width || frame->height != avctx->height || frame->format != avctx->pix_fmt)) {
            av_log(avctx, AV_LOG_WARNING, "Picture changed from size:%dx%d fmt:%s to size:%dx%d fmt:%s in reget buffer()\n",
                   frame->width, frame->height, av_get_pix_fmt_name(frame->format), avctx->width, avctx->height, av_get_pix_fmt_name(avctx->pix_fmt));
            av_frame_unref(frame);
    
        ff_init_buffer_info(avctx, frame);
    
        if (!frame->data[0])
            return ff_get_buffer(avctx, frame, AV_GET_BUFFER_FLAG_REF);
    
        if (av_frame_is_writable(frame))
            return ff_decode_frame_props(avctx, frame);
    
        tmp = av_frame_alloc();
        if (!tmp)
            return AVERROR(ENOMEM);
    
        av_frame_move_ref(tmp, frame);
    
    
        ret = ff_get_buffer(avctx, frame, AV_GET_BUFFER_FLAG_REF);
        if (ret < 0) {
    
        av_frame_copy(frame, tmp);
        av_frame_free(&tmp);
    
    int ff_reget_buffer(AVCodecContext *avctx, AVFrame *frame)
    {
        int ret = reget_buffer_internal(avctx, frame);
        if (ret < 0)
            av_log(avctx, AV_LOG_ERROR, "reget_buffer() failed\n");
        return ret;
    }
    
    
    Luca Barbato's avatar
    Luca Barbato committed
    int avcodec_default_execute(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2), void *arg, int *ret, int count, int size)
    {
    
    Luca Barbato's avatar
    Luca Barbato committed
        for (i = 0; i < count; i++) {
            int r = func(c, (char *)arg + i * size);
            if (ret)
                ret[i] = r;
    
    Luca Barbato's avatar
    Luca Barbato committed
    int avcodec_default_execute2(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int jobnr, int threadnr), void *arg, int *ret, int count)
    {