简介

avpicture_fill函数将ptr指向的数据填充到picture内,但并没有拷贝,只是将picture结构内的data指针指向了ptr的数据。其实现如下:

avpiture_fill

avpiture_fill直接调用av_image_fill_arrays函数。

// libavcodec/avpicture.c
int avpicture_fill(AVPicture *picture, const uint8_t *ptr,
enum AVPixelFormat pix_fmt, int width, int height)
{
return av_image_fill_arrays(picture->data, picture->linesize,
ptr, pix_fmt, width, height, );
}

av_image_fill_arrays

// libavutil/imgutils.c
int av_image_fill_arrays(uint8_t *dst_data[], int dst_linesize[],
const uint8_t *src, enum AVPixelFormat pix_fmt,
int width, int height, int align)
{
int ret, i; ret = av_image_check_size(width, height, , NULL);
if (ret < )
return ret; ret = av_image_fill_linesizes(dst_linesize, pix_fmt, width);
if (ret < )
return ret; for (i = ; i < ; i++)
dst_linesize[i] = FFALIGN(dst_linesize[i], align); return av_image_fill_pointers(dst_data, pix_fmt, height, (uint8_t *)src, dst_linesize);
}

其中av_image_check_size用来检测输入的widht和height是否可用,判断条件如下:

if ((int)w>0 && (int)h>0 && (w+128)*(uint64_t)(h+128) < INT_MAX/8)
return 0;

av_image_fill_linesizes

int av_image_fill_linesizes(int linesizes[], enum AVPixelFormat pix_fmt, int width)
{
int i, ret;
const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(pix_fmt);
int max_step []; /* max pixel step for each plane */
int max_step_comp[]; /* the component for each plane which has the max pixel step */ memset(linesizes, , *sizeof(linesizes[])); if (!desc || desc->flags & AV_PIX_FMT_FLAG_HWACCEL)
return AVERROR(EINVAL); av_image_fill_max_pixsteps(max_step, max_step_comp, desc);
for (i = ; i < ; i++) {
if ((ret = image_get_linesize(width, i, max_step[i], max_step_comp[i], desc)) < ) return ret;
linesizes[i] = ret;
} return ;
}
  1. 将linsizes数组的内容置为0;
  2. 利用av_pix_fmt_desc_get函数得到输入格式的AVPixFmtDescriptor指针;
  3. 最后利用image_get_linesize函数获得linesizes数组中每个元素的值;

    FFALIGN

    由于在afpicture_fill中填充的align为1, 故该宏返回的值还是linesizes[i];

    // libavutil/common.h 
    
    #define FFALIGN(x, a) (((x)+(a)-1)&~((a)-1))

    av_image_fill_pointers

    int av_image_fill_pointers(uint8_t *data[], enum AVPixelFormat pix_fmt, int height,
    uint8_t *ptr, const int linesizes[])
    {
    int i, total_size, size[] = { }, has_plane[] = { }; const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(pix_fmt);
    memset(data , , sizeof(data[])*); if (!desc || desc->flags & AV_PIX_FMT_FLAG_HWACCEL)
    return AVERROR(EINVAL); data[] = ptr;
    if (linesizes[] > (INT_MAX - ) / height)
    return AVERROR(EINVAL);
    size[] = linesizes[] * height; if (desc->flags & AV_PIX_FMT_FLAG_PAL ||
    desc->flags & AV_PIX_FMT_FLAG_PSEUDOPAL) {
    size[] = (size[] + ) & ~;
    data[] = ptr + size[]; /* palette is stored here as 256 32 bits words */
    return size[] + * ;
    } for (i = ; i < ; i++)
    has_plane[desc->comp[i].plane] = ; total_size = size[];
    for (i = ; i < && has_plane[i]; i++) {
    int h, s = (i == || i == ) ? desc->log2_chroma_h : ;
    data[i] = data[i-] + size[i-];
    h = (height + ( << s) - ) >> s;
    if (linesizes[i] > INT_MAX / h)
    return AVERROR(EINVAL);
    size[i] = h * linesizes[i];
    if (total_size > INT_MAX - size[i])
    return AVERROR(EINVAL);
    total_size += size[i];
    } return total_size;
    }

    将data数组内的指针分别指向ptr内的数据。

转自 http://www.voidcn.com/article/p-aquvaett-zg.html

avpicture_fill的实现的更多相关文章

  1. ffmpeg avpicture_fill的一些使用

    标签: ffmpegavpicture_fill 2013-05-17 10:03 4713人阅读 评论(1) 收藏 举报  分类: ffmpeg(3)  这个FFMPEG我没找到详细的中文教程,只有 ...

  2. 关于avpicture_fill 和 sws_scale的关系

    avpicture_fill((AVPicture *) pFrameRGB, buffer, PIX_FMT_RGB565, pCodecCtx->width, pCodecCtx->h ...

  3. 海康网络摄像机YV12转换为BGR,由opencv Mat显示 (转)

    我使用的是海康DS-2CD852MF-E, 200万,网络摄像机,已经比较老了,不过SDK在海康官网下载的,开发流程都差不多. 海康摄像机回调解码后的视频数据格式为YV12,顺便说一下YV12的数据格 ...

  4. FFmpeg学习5:多线程播放视音频

    在前面的学习中,视频和音频的播放是分开进行的.这主要是为了学习的方便,经过一段时间的学习,对FFmpeg的也有了一定的了解,本文就介绍了 如何使用多线程同时播放音频和视频(未实现同步),并对前面的学习 ...

  5. FFmpeg学习2:解码数据结构及函数总结

    在上一篇文章中,对FFmpeg的视频解码过程做了一个总结.由于才接触FFmpeg,还是挺陌生的,这里就解码过程再做一个总结. 本文的总结分为以下两个部分: 数据读取,主要关注在解码过程中所用到的FFm ...

  6. 【视频处理】YUV与RGB格式转换

    YUV格式具有亮度信息和色彩信息分离的特点,但大多数图像处理操作都是基于RGB格式. 因此当要对图像进行后期处理显示时,需要把YUV格式转换成RGB格式. RGB与YUV的变换公式如下: YUV(25 ...

  7. 简单播放器(增加sdl事件控制)

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

  8. ffmpeg 和 SDL 的结合使用

    FFmpeg是一套可以用来记录.转换数字音频.视频,并能将其转化为流的开源计算机程序.采用LGPL或GPL许可证.它提供了录制.转换以及流化音视 频的完整解决方案.它包含了非常先进的音频/视频编解码库 ...

  9. 学习FFmpeg API

    ffmpeg是编解码的利器,用了很久,以前看过dranger 的教程,非常精彩,受益颇多,是学习ffmpeg api很好的材料.可惜的是其针对的ffmpeg版本已经比较老了,而ffmpeg的更新又很快 ...

随机推荐

  1. UIImageView - BNR

    继续上节UINavigationController - BNR. 打开BNRDetailViewController.xib文件,向view中添加UIImageView对象,选中该对象,通过Attr ...

  2. 【精编重制版】JavaWeb 入门级项目实战 -- 文章发布系统 (第二节)

    说明 本教程是,原文章发布系统教程的精编重制版,会包含每一节的源码,以及修正之前的一些错误.因为之前的教程只做到了评论模块,很多地方还不完美,因此重制版会修复之前的一些谬误和阐述不清的地方,而且,后期 ...

  3. 【教程】switch上的Human Fall Flat如何设置本地双人?

    1. 保证两个手柄已插入主机上 2. 进入游戏至游戏开始界面 3. 将主机插入拓展坞,等待电视显示 4. 稍等数秒,电视上会提示使用手柄方式 5. 此时按照多人游戏的手柄操作方法即可

  4. 一次linux服务器黑客入侵后处理

     场景: 周一上班centos服务器ssh不可用,web和数据库等应用不响应.好在vnc可以登录 使用last命令查询,2号之前的登录信息已被清空,并且sshd文件在周六晚上被修改,周日晚上2点服务器 ...

  5. Python学习第二篇

    list_num=list(range(1,1000001)) print(min(list_num)) print(max(list_num)) print(sum(list_num)) print ...

  6. 安装配置JDK和Eclipse的步骤

    导读 作为Java程序员,需要在Linux系统上安装Eclipse,很多人不知要如何安装,在安装Eclipse前,还需安装JDK,Linux下如何安装JDK和Eclipse呢?下面跟朋友们介绍下Lin ...

  7. JMeter学习FTP测试计划(转)

    FTP服务主要提供上传和下载功能.有时间需要我们测试服务器上传和下载的性能.在这里我通过JMeter做一个FTP测试计划的例子. 1.创建一个线程组 2.线程组--->添加--->配置元件 ...

  8. [转帖]buffer与cache的区别

    作者:沈万马链接:https://www.zhihu.com/question/26190832/answer/146259979来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注 ...

  9. sass变量引入全局

    https://www.jianshu.com/p/ab9ab999344b(copy) 本文以Sass做案例,Less的参考,基本配置大同小异. 假如我们有一个Sass的全局变量common.scs ...

  10. leetcode:Roman to Integer and Integer to Roman

    2015-06-03 罗马数字以前接触过I到VIII比较多,直到遇见这个题目才知道更详细.阿拉伯数字和罗马数字之间的转换最重的是了解罗马数字的规则. 罗马数字规则:(总结) 1, 罗马数字共有7个,即 ...