博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
FFmpeg数据结构AVBuffer
阅读量:6118 次
发布时间:2019-06-21

本文共 4575 字,大约阅读时间需要 15 分钟。

本文为作者原创,转载请注明出处:

AVBuffer是FFmpeg中很常用的一种缓冲区,缓冲区使用引用计数(reference-counted)机制。

AVBufferRef则对AVBuffer缓冲区提供了一层封装,最主要的是作引用计数处理,实现了一种安全机制。用户不应直接访问AVBuffer,应通过AVBufferRef来访问AVBuffer,以保证安全。
FFmpeg中很多基础的数据结构都包含了AVBufferRef成员,来间接使用AVBuffer缓冲区。
本文使用的FFmpeg版本号为FFmpeg 4.1。

AVBuffer和AVBufferRef结构体定义及操作函数位于libavutil中的buffer.h、buffer_internal.h、buffer.c三个文件中。需要关注的要点是AVBufferRef和AVBuffer的关系以及缓冲区引用计数的概念

1. 数据结构定义

1.1 struct AVBuffer

struct AVBuffer定义于“libavutil/buffer_internal.h”,buffer_internal.h位于FFmpeg工程源码中,而FFmpeg提供的开发库头文件中并无此文件,因此这是一个内部数据结构,不向用户开放,用户不应直接访问AVBuffer,应通过AVBufferRef来访问AVBuffer,以保证安全。

struct AVBuffer {    uint8_t *data; /**< data described by this buffer */    int      size; /**< size of data in bytes */    /**     *  number of existing AVBufferRef instances referring to this buffer     */    atomic_uint refcount;    /**     * a callback for freeing the data     */    void (*free)(void *opaque, uint8_t *data);    /**     * an opaque pointer, to be used by the freeing callback     */    void *opaque;    /**     * A combination of BUFFER_FLAG_*     */    int flags;};
  • data: 缓冲区地址
  • size: 缓冲区大小
  • refcount: 引用计数值
  • free: 用于释放缓冲区内存的回调函数
  • opaque: 提供给free回调函数的参数
  • flags: 缓冲区标志

1.2 struct AVBufferRef

struct AVBufferRef定义于buffer.h中:

/** * A reference to a data buffer. * * The size of this struct is not a part of the public ABI and it is not meant * to be allocated directly. */typedef struct AVBufferRef {    AVBuffer *buffer;    /**     * The data buffer. It is considered writable if and only if     * this is the only reference to the buffer, in which case     * av_buffer_is_writable() returns 1.     */    uint8_t *data;    /**     * Size of data in bytes.     */    int      size;} AVBufferRef;
  • buffer: AVBuffer
  • data: 缓冲区地址,实际等于buffer->data
  • size: 缓冲区大小,实际等于buffer->size

2. 关键函数实现

2.1 av_buffer_alloc()

AVBufferRef *av_buffer_alloc(int size){    AVBufferRef *ret = NULL;    uint8_t    *data = NULL;    data = av_malloc(size);    if (!data)        return NULL;    ret = av_buffer_create(data, size, av_buffer_default_free, NULL, 0);    if (!ret)        av_freep(&data);    return ret;}

av_buffer_alloc()作了如下处理:

a) 使用av_malloc分配缓冲区

b) 调用av_buffer_create()创建AVBuffer AVBufferRef::*buffer成员,用于管理AVBuffer缓冲区

c) 返回AVBufferRef *对象

2.2 av_buffer_create()

AVBufferRef *av_buffer_create(uint8_t *data, int size,                              void (*free)(void *opaque, uint8_t *data),                              void *opaque, int flags){    AVBufferRef *ref = NULL;    AVBuffer    *buf = NULL;    buf = av_mallocz(sizeof(*buf));    if (!buf)        return NULL;    buf->data     = data;    buf->size     = size;    buf->free     = free ? free : av_buffer_default_free;    buf->opaque   = opaque;    atomic_init(&buf->refcount, 1);    if (flags & AV_BUFFER_FLAG_READONLY)        buf->flags |= BUFFER_FLAG_READONLY;    ref = av_mallocz(sizeof(*ref));    if (!ref) {        av_freep(&buf);        return NULL;    }    ref->buffer = buf;    ref->data   = data;    ref->size   = size;    return ref;}

av_buffer_create()是一个比较核心的函数,从其实现代码很容易看出AVBufferRef和AVBuffer这间的关系。

函数主要功能就是初始化AVBuffer AVBufferRef::*buffer成员,即为上述清单ref->buffer各字段赋值,最终,AVBufferRef *ref全部构造完毕,将之返回。

其中void (*free)(void *opaque, uint8_t *data)参数赋值为av_buffer_default_free,实现如下。其实就是直接调用了av_free回收内存。

void av_buffer_default_free(void *opaque, uint8_t *data){    av_free(data);}

2.3 av_buffer_ref()

AVBufferRef *av_buffer_ref(AVBufferRef *buf){    AVBufferRef *ret = av_mallocz(sizeof(*ret));    if (!ret)        return NULL;    *ret = *buf;    atomic_fetch_add_explicit(&buf->buffer->refcount, 1, memory_order_relaxed);    return ret;}

av_buffer_ref()处理如下:

a) *ret = *buf;一句将buf各成员值赋值给ret中对应成员,buf和ret将共用同一份AVBuffer缓冲区

b) atomic_fetch_add_explicit(...);一句将AVBuffer缓冲区引用计数加1

注意此处的关键点:共用缓冲区(缓冲区不拷贝),缓冲区引用计数加1

2.4 av_buffer_unref()

static void buffer_replace(AVBufferRef **dst, AVBufferRef **src){    AVBuffer *b;    b = (*dst)->buffer;    if (src) {        **dst = **src;        av_freep(src);    } else        av_freep(dst);    if (atomic_fetch_add_explicit(&b->refcount, -1, memory_order_acq_rel) == 1) {        b->free(b->opaque, b->data);        av_freep(&b);    }}void av_buffer_unref(AVBufferRef **buf){    if (!buf || !*buf)        return;    buffer_replace(buf, NULL);}

av_buffer_unref()处理如下:

a) 回收AVBufferRef **buf内存

b) 将(*buf)->buffer(即AVBAVBufferRef的成员AVBuffer)的引用计数减1,若引用计数为0,则通过b->free(b->opaque, b->data);调用回调函数回收AVBuffer缓冲区内存

注意此处的关键点:销毁一个AVBufferRef时,将其AVBuffer缓冲区引用计数减1,若缓冲区引用计数变为0,则将缓冲区也回收,这很容易理解,只有当缓冲区不被任何对象引用时,缓冲区才能被销毁

3. 修改记录

2018-12-13 V1.0 初稿

转载于:https://www.cnblogs.com/leisure_chn/p/10399048.html

你可能感兴趣的文章
Unity LineRenderer制作画版
查看>>
JAVA学习笔记
查看>>
接口—学习
查看>>
vivo手机执行input命令提示killed
查看>>
072:【Django数据库】ORM聚合函数详解-aggregate和annotate
查看>>
html5学习笔记1
查看>>
Linux命令——procinfo
查看>>
go标准库的学习-io
查看>>
html 富文本编辑器相关--主动选择文字-setSelectionRange-控制光标位置
查看>>
iOS-旧项目中手动内存管理(MRC)转ARC
查看>>
转:建立索引的原则 - 以innodb为例
查看>>
Spring深入理解(三)
查看>>
单据状态图
查看>>
2.1 Spring容器的基本实现--Spring源码深度解析
查看>>
ASN.1 key structures in DER and PEM
查看>>
20165226 学习基础和C语言基础调查
查看>>
iOS开发中的小技巧 - 多张图合成一张
查看>>
javascript实现各种排序算法
查看>>
centos 安装 配置
查看>>
Mac 安装mysql
查看>>