使用ffmpeg的filter处理yuv数据包括split filter(分流)、crop filter(裁剪)、vflip filter(垂直向上的翻转)、overlay filter(合成)

使用ffmpeg的filter处理yuv数据包括split filter(分流)、crop filter(裁剪)、vflip filter(垂直向上的翻转)、overlay filter(合成)

#include <stdio.h>#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libavfilter/avfilter.h>
#include <libavfilter/buffersink.h>
#include <libavfilter/buffersrc.h>
#include <libavutil/opt.h>
#include <libavutil/imgutils.h>int main()
{printf("Hello video mark!\n");int ret = 0;FILE* infile = NULL;const char* infileName = "768x320.yuv";fopen_s(&infile, infileName, "rb+");if(!infile){printf("fopen_s() infile failed!\n");return -1;}int in_width = 768;int in_height = 320;FILE* outfile = NULL;const char* outfileName = "out_mark.yuv";fopen_s(&outfile, outfileName, "wb");if(!outfile){printf("fopen_s() outfile failed!\n");return -1;}//注册初始化所有过滤器avfilter_register_all();//用于整个过滤流程的一个封装AVFilterGraph* filter_grah = avfilter_graph_alloc();if(!filter_grah){printf("avfilter_graph_alloc() failed!\n");return -1;}char args[512];sprintf(args,"video_size=%dx%d:pix_fmt=%d:time_base=%d/%d:pixel_aspect=%d/%d",in_width, in_height, AV_PIX_FMT_YUV420P,1, 25, 1, 1);//获取一个用于AVFilterGraph输入的过滤器AVFilter* buffersSrc = avfilter_get_by_name("buffer");AVFilterContext* bufferSrc_ctx;//将bufferSrc添加到AVFilterGraph中//args是用在bufferSrc的参数ret = avfilter_graph_create_filter(&bufferSrc_ctx, buffersSrc,"in", args, NULL, filter_grah);if(ret < 0){printf("avfilter_graph_create_filter() buffersSrc failed!\n");return -1;}AVBufferSinkParams* bufferSinkParams;AVFilterContext* bufferSink_ctx;//获取一个用于AVFilterGraph输出的过滤器AVFilter* bufferSink = avfilter_get_by_name("buffersink");enum AVPixelFormat pix_fmts[] = {AV_PIX_FMT_YUV420P, AV_PIX_FMT_NONE};bufferSinkParams = av_buffersink_params_alloc();bufferSinkParams->pixel_fmts = pix_fmts;ret = avfilter_graph_create_filter(&bufferSink_ctx, bufferSink,"out", NULL, bufferSinkParams, filter_grah);if(ret < 0){printf("avfilter_graph_create_filter() bufferSink failed!\n");return -1;}//split filter(分流)AVFilter* splitFilter = avfilter_get_by_name("split");AVFilterContext* splitFilter_ctx;//outputs=2 分流2通道ret = avfilter_graph_create_filter(&splitFilter_ctx, splitFilter, "split","outputs=2", NULL, filter_grah);if(ret < 0){printf("avfilter_graph_create_filter() splitFilter failed!\n");return -1;}//crop filter(裁剪)AVFilter* cropFilter = avfilter_get_by_name("crop");AVFilterContext* cropFilter_ctx;ret = avfilter_graph_create_filter(&cropFilter_ctx, cropFilter, "crop","out_w=iw:out_h=ih/2:x=0:y=0", NULL, filter_grah);if(ret < 0){printf("avfilter_graph_create_filter() cropFilter failed!\n");return -1;}//vflip filter(垂直向上的翻转)AVFilter* vflipFilter = avfilter_get_by_name("vflip");AVFilterContext* vflipFilter_ctx;ret = avfilter_graph_create_filter(&vflipFilter_ctx, vflipFilter, "vflip",NULL, NULL, filter_grah);if(ret < 0){printf("avfilter_graph_create_filter() vflipFilter failed!\n");return -1;}//overlay filter(合成)AVFilter* overlayFilter = avfilter_get_by_name("overlay");AVFilterContext* overlayFilter_ctx;ret = avfilter_graph_create_filter(&overlayFilter_ctx, overlayFilter, "overlay","y=0:H/2", NULL, filter_grah);if(ret < 0){printf("avfilter_graph_create_filter() overlayFilter failed!\n");return -1;}//srcFilter -> splitFilter//srcpad、dstpad是一个索引,通道的索引ret = avfilter_link(bufferSrc_ctx, 0, splitFilter_ctx, 0);if(ret != 0){printf("avfilter_link() srcFilter -> splitFilter failed!\n");return -1;}//splitFilter[0] -> overlayfilter[0]ret = avfilter_link(splitFilter_ctx, 0, overlayFilter_ctx, 0);if(ret != 0){printf("avfilter_link() splitFilter[0] -> overlayfilter[0] failed!\n");return -1;}//splitFilter[1] -> cropFilterret = avfilter_link(splitFilter_ctx, 1, cropFilter_ctx, 0);if(ret != 0){printf("avfilter_link() splitFilter[1] -> cropFilter failed!\n");return -1;}
\//cropFilter -> vflipFilterret = avfilter_link(cropFilter_ctx, 0, vflipFilter_ctx, 0);if(ret != 0){printf("avfilter_link() cropFilter -> vflipFilter failed!\n");return -1;}//vflipFilter -> overlayfilter[1]ret = avfilter_link(vflipFilter_ctx, 0, overlayFilter_ctx, 1);if(ret != 0){printf("avfilter_link() vflipFilter -> overlayfilter[1] failed!\n");return -1;}//overlayfilter -> bufferSinkret = avfilter_link(overlayFilter_ctx, 0, bufferSink_ctx, 0);if(ret != 0){printf("avfilter_link() overlayfilter -> bufferSink failed!\n");return -1;}//确认所有过滤器的连接ret = avfilter_graph_config(filter_grah, NULL);if(ret < 0){printf("avfilter_graph_config() failed!\n");return -1;}//打印filtergraph的信息char* graph_str = avfilter_graph_dump(filter_grah, NULL);printf("\n%s\n", graph_str);av_free(graph_str);//输入帧AVFrame* frame_in = av_frame_alloc();unsigned char* frame_buffer_in = (unsigned char*)av_malloc(av_image_get_buffer_size(AV_PIX_FMT_YUV420P, in_width, in_height, 1));//输出帧AVFrame* frame_out = av_frame_alloc();frame_in->width = in_width;frame_in->height = in_height;frame_in->format = AV_PIX_FMT_YUV420P;uint32_t frame_size = in_width * in_height * 3 / 2;while (1){//读取yuv数据if(fread(frame_buffer_in, 1, frame_size, infile) != frame_size){break;}av_image_fill_arrays(frame_in->data, frame_in->linesize, frame_buffer_in, AV_PIX_FMT_YUV420P, in_width, in_height, 1);//添加帧数据到过滤器if(av_buffersrc_add_frame(bufferSrc_ctx, frame_in) < 0){printf("av_buffersrc_add_frame() failed!\n");break;}//获取输出帧数据ret = av_buffersink_get_frame(bufferSink_ctx, frame_out);if(ret < 0){printf("av_buffersink_get_frame() failed!\n");break;}//输出文件if(frame_out->format == AV_PIX_FMT_YUV420P){for (int i = 0; i < frame_out->height; i++) {fwrite(frame_out->data[0] + frame_out->linesize[0] * i, 1, frame_out->width, outfile);}for (int i = 0; i < frame_out->height / 2; i++) {fwrite(frame_out->data[1] + frame_out->linesize[1] * i, 1, frame_out->width / 2, outfile);}for (int i = 0; i < frame_out->height / 2; i++) {fwrite(frame_out->data[2] + frame_out->linesize[2] * i, 1, frame_out->width / 2, outfile);}}av_frame_unref(frame_out);}fclose(infile);fclose(outfile);av_frame_free(&frame_in);av_frame_free(&frame_out);avfilter_graph_free(&filter_grah);printf("end video mark!\n");return 0;
}```

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/379083.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vc++ 6.0 堆栈_在C ++中使用链接列表实现堆栈

vc 6.0 堆栈To implement a stack using a linked list, basically we need to implement the push() and pop() operations of a stack using linked list. 要使用链接列表实现堆栈 &#xff0c;基本上&#xff0c;我们需要使用链接列表实现堆栈的push()和pop()操作。 Exampl…

烟雨小书店

烟雨小书店演示视频 源码

协议地址结构_TCP/IP 协议 讲解

计算机网络体系结构分层太厉害了&#xff0c;终于有人能把TCP/IP 协议讲的明明白白了计算机网络体系结构分层不难看出&#xff0c;TCP/IP 与 OSI 在分层模块上稍有区别。OSI 参考模型注重“通信协议必要的功能是什么”&#xff0c;而 TCP/IP 则更强调“在计算机上实现协议应该开…

ffmpeg进行混音,将两路音频pcm数据合成一路输出

ffmpeg进行混音&#xff0c;将两路音频pcm数据合成一路输出 audiomixer.h #ifndef AUDIOMIXER_H #define AUDIOMIXER_H#include <map> #include <mutex> #include <cstdio> #include <cstdint> #include <string> #include <memory>exter…

python sep函数_Python中带有print()函数的sep参数

python sep函数sep parameter stands for separator, it uses with the print() function to specify the separator between the arguments. sep参数代表分隔符&#xff0c;它与print()函数一起使用以指定参数之间的分隔符。 The default value is space i.e. if we dont us…

关于 MySQL 主从复制的配置(转)

来源&#xff1a;http://www.oschina.net/bbs/thread/10388设置Mysql的主从设置很重要&#xff0c;有如下几点用处&#xff1a;1 做备份机器&#xff0c;一旦主服务器崩溃&#xff0c;可以直接启用从服务器作为主服务器2 可以直接锁定从服务器的表只读&#xff0c;然后做备份数…

Silverlight 同域WCF免跨域文件

在sl3使用wcf时常常会因为sl中调用了不同域的wcf服务而导至调用服务失败&#xff0c;记得在很久以前sl当是只支持同域的访问&#xff0c;那么让我有一个想法&#xff0c;就是在sl引用时可以动态地取得当前sl所在的域&#xff0c;而wcf服务也必须同时部署到这个域下边&#xff0…

使用ffmpeg 的 filter 给图片添加水印

使用ffmpeg 的 filter 给图片添加水印。 main.c #include <stdio.h>#include <libavfilter/avfilter.h> #include <libavfilter/buffersrc.h> #include <libavfilter/buffersink.h> #include <libavformat/avformat.h> #include <libavcodec…

程序崩溃 分析工具_程序分析工具| 软件工程

程序崩溃 分析工具A program analysis tool implies an automatic tool that takes the source code or the executable code of a program as information and produces reports with respect to a few significant attributes of the program, for example, its size, multif…

28335接两个spi设备_IIC和SPI如此流行,谁才是嵌入式工程师的必备工具?

IICvs SPI现今&#xff0c;在低端数字通信应用领域&#xff0c;我们随处可见 IIC (Inter-Integrated Circuit) 和 SPI (Serial Peripheral Interface)的身影。原因是这两种通信协议非常适合近距离低速芯片间通信。Philips(for IIC)和 Motorola(for SPI) 出于不同背景和市场需求…

线性表15|魔术师发牌问题和拉丁方阵 - 数据结构和算法20

线性表15 : 魔术师发牌问题和拉丁方阵 让编程改变世界 Change the world by program 题外话 今天小甲鱼看到到微博有朋友在问&#xff0c;这个《数据结构和算法》系列课程有木有JAVA版本的&#xff1f; 因为这个问题之前也有一些朋友问过&#xff0c;所以咱在这里统一说下哈…

[ZT]Three ways to tell if a .NET Assembly is Strongly Named (or has Strong Name)

Here are several convenient ways to tell whether a .NET assembly is strongly named. (English language note: I assume the form “strongly named” is preferred over “strong named” since that’s the form used in the output of the sn.exe tool shown immediat…

最佳页面置换算法

在一个请求分页系统中&#xff0c;采用最佳页面置换算法时&#xff0c;假如一个作业的页面走向为4、3、2、1、4、3、5、4、3、2、1、5&#xff0c;当分配给该作业的物理块数M分别为3和4时&#xff0c;试计算在访问过程中所发生的缺页次数和缺页率。请给出分析过程。 解析&…

网络名称 转换 网络地址_网络地址转换| 计算机网络

网络名称 转换 网络地址At the time of classful addressing, the number of household users and small businesses that want to use the Internet kept increasing. In the beginning, a user was connected to the Internet with a dial-up line, for a specific period of…

rstudio 修改代码间距_第一章 R和RStudio

R与RStudioR是一种统计学编程语言&#xff0c;在科学计算领域非常流行。它是由Ross Ihaka和Robert Gentleman开发的&#xff0c;是 "S "编程语言的开源实现。R也是使用这种语言进行统计计算的软件的名字。它有一个庞大的在线支持社区和专门的软件包&#xff0c;可以为…

ubuntu下最稳定的QQ

一、安装好 Wine 1.2&#xff08;1.2 版安装好就支持中文界面的了&#xff09; 当然得有WINE 了 当然我的有 如果没有可以如下方法得到&#xff1a; 第一种方法&#xff1a;如果你已经安装过 Wine 的老版本&#xff0c;那么只要添加 Wine 1.2 的软件源&#xff0c;然后去新立得…

字体Times New Roman

Windows系统中的字体是Monotype公司为微软公司制作的Times New Roman PS&#xff08;TrueType字体&#xff09;&#xff0c;视窗系统从3.1版本开始就一直附带这个字体。而在苹果电脑公司的麦金塔系统中使用的是Linotype公司的 Times Roman (在Macintosh系统中直接简称为‘Times…

最近最久未使用页面置换算法

在一个请求分页系统中&#xff0c;采用最近最久未使用页面置换算法时&#xff0c;假如一个作业的页面走向为4、3、2、1、4、3、5、4、3、2、1、5&#xff0c;当分配给该作业的物理块数M分别为3和4时&#xff0c;试计算在访问过程中所发生的缺页次数和缺页率。请给出分析过程。 …