ffmpeg architecture(下)

第3章-转码

TLDR;给我看代码和执行。

$ make run_transcoding

我们将跳过一些细节,但是请放心:源代码可在github上找到

在本章中,我们将创建一个用C编写的极简代码转换器,可以使用FFmpeg / libav库(尤其是libavcodec,libavformat和libavutil)将H264编码的视频转换为H265。

只是快速回顾一下:AVFormatContext是媒体文件格式的抽象,又名容器(例如:MKV,MP4,WEBM,TS)。的AV表示每种类型的用于给定格式(例如:音频,视频,字幕,元数据)的数据。所述AVPacket是从所获得的压缩数据的切片AVStream可以由一个解码avcodec:产生称为原始数据(例如AV1,H264,VP9,HEVC)AVFrame

转码

让我们从简单的转换操作开始,然后我们可以基于此代码构建,第一步是加载输入文件

//分配一个AVFormatContext
avfc = avformat_alloc_context();
//打开输入流并读取标题。
avformat_open_input(avfc,in_filename, NULL, NULL);
//读取媒体文件的数据包以获取流信息。
avformat_find_stream_info(avfc, NULL);

现在,我们要设置解码器,它AVFormatContext将使我们能够访问所有AVStream组件,并且对于每个组件,我们都可以获取它们AVCodec并创建特定组件AVCodecContext,最后我们可以打开给定的编解码器,以便继续进行解码处理。

所述AVCodecContext持有如比特率,帧速率,采样率,信道,高度,和许多其它有关的媒体数据的配置等。

对于(int i = 0 ; i <avfc-> nb_streams; i ++)
{
  AVStream * avs = avfc-> 流 [i];
  AVCodec * avc = avcodec_find_decoder(avs-> codecpar- > codec_id);
  AVCodecContext * avcc = avcodec_alloc_context3(* avc);
  avcodec_parameters_to_context(* avcc,avs-> codecpar);
  avcodec_open2(* avcc,* avc,NULL);
}

我们还需要准备输出媒体文件以进行多路传输,我们首先为输出分配内存AVFormatContext。我们以输出格式创建每个流。为了正确打包流,我们从解码器复制编解码器参数

我们设置标志 AV_CODEC_FLAG_GLOBAL_HEADER告诉编码器它可以使用全局头文件,最后打开输出文件进行写入并保留头文件。

avformat_alloc_output_context2(&encoder_avfc,NULL,NULL,out_filename);
AVStream * avs = avformat_new_stream(encoder_avfc,NULL);
avcodec_parameters_copy(avs-> codecpar,解码器_avs-> codecpar);
如果(encoder_avfc-> oformat-> flags和AVFMT_GLOBALHEADER)
  encoder_avfc-> flags | = AV_CODEC_FLAG_GLOBAL_HEADER;
avio_open(&encoder_avfc-> pb,编码器->文件名,AVIO_FLAG_WRITE);
avformat_write_header(encoder- > avfc,&muxer_opts);

我们AVPacket从解码器获取,调整时间戳,并将数据包正确写入输出文件。即使函数av_interleaved_write_frame说“写帧”,我们仍在存储数据包。通过将流预告片写入文件中,我们完成了转换过程。

AVFrame * input_frame = av_frame_alloc();
AVPacket * input_packet = av_packet_alloc();
而(av_read_frame(decoder_avfc,input_packet)> = 0)
{
  av_packet_rescale_ts(input_packet,解码器_视频_avs-> time_base,编码器 _视频_avs- > time_base);
  av_interleaved_write_frame(* avfc,input_packet)< 0));
}
av_write_trailer(encoder_avfc);

转码

上一节展示了一个简单的transmuxer程序,现在我们将添加对文件进行编码的功能,尤其是使它能够将视频从转换h264h265

在准备好解码器之后,但在安排输出媒体文件之前,我们将设置编码器。

AVRational input_framerate = av_guess_frame_rate(decoder_avfc,decoder_video_avs,NULL);
AVStream * video_avs = avformat_new_stream(encoder_avfc,NULL);
char * codec_name = “ libx265 ” ;
char * codec_priv_key = “ x265-params ” ;
//我们将为x265使用内部选项
// //禁用场景更改检测,然后修复
// 60帧的GOP。
char * codec_priv_value = “ keyint = 60:min-keyint = 60:scenecut = 0 ” ;
AVCodec * video_avc = avcodec_find_encoder_by_name(codec_name);
AVCodecContext * video_avcc = avcodec_alloc_context3(video_avc);
//编码器编解码器参数
av_opt_set(sc-> video_avcc-> priv_data,codec_priv_key,codec_priv_value, 0);
video_avcc-> height =解码器_ctx-> height;
video_avcc-> width =解码器_ctx-> width;
video_avcc-> pix_fmt = video_avc-> pix_fmts [ 0 ];
//控制率 
video_avcc-> bit_rate = 2 * 1000 * 1000 ;
video_avcc-> rc_buffer_size = 4 * 1000 * 1000 ;
video_avcc-> rc_max_rate = 2 * 1000 * 1000 ;
video_avcc-> rc_min_rate = 2.5 * 1000 * 1000 ;
//时基
video_avcc-> time_base = av_inv_q(input_framerate);
video_avs-> time_base = sc-> video_avcc-> time_base;
avcodec_open2(sc-> video_avcc,sc-> video_avc,NULL);
avcodec_parameters_from_context(sc-> video_avs-> codecpar,sc-> video_avcc);

我们需要扩展解码循环以进行视频流转码:

AVFrame * input_frame = av_frame_alloc();
AVPacket * input_packet = av_packet_alloc();
而(av_read_frame(decoder_avfc,input_packet)> = 0)
{
  int响应= avcodec_send_packet(decoder_video_avcc,input_packet);
  while(响应> = 0){
    响应= avcodec_receive_frame(decoder_video_avcc,input_frame);
    if(响应== AVERROR(EAGAIN)|| response == AVERROR_EOF){
       中断 ;
    } else  if(response < 0){
       返回响应;
    }
    if(响应> = 0){
       编码(encoder_avfc,decoder_video_avs,encoder_video_avs,decoder_video_avcc,input_packet-> stream_index);
    }
    av_frame_unref(input_frame);
  }
  av_packet_unref(input_packet);
}
av_write_trailer(encoder_avfc);
//使用的函数
int  编码(AVFormatContext * avfc,AVStream * dec_video_avs,AVStream * enc_video_avs,AVCodecContext video_avcc int索引){
  AVPacket * output_packet = av_packet_alloc();
  int response = avcodec_send_frame(video_avcc,input_frame);
  while(响应> = 0){
    响应= avcodec_receive_packet(video_avcc,output_packet);
    if(响应== AVERROR(EAGAIN)|| response == AVERROR_EOF){
       中断 ;
    } else  if(响应< 0){
       return - 1 ;
    }
    output_packet-> stream_index = index ;
    output_packet-> 持续时间 = enc_video_avs-> 那么time_base。den / enc_video_avs-> time_base。num / dec_video_avs-> avg_frame_rate。num * dec_video_avs-> avg_frame_rate。巢穴 ;
    av_packet_rescale_ts(output_packet,dec_video_avs-> time_base,enc_video_avs-> time_base);
    响应= av_interleaved_write_frame(avfc,output_packet);
  }
  av_packet_unref(output_packet);
  av_packet_free(&output_packet);
  返回 0 ;
}

我们将媒体流从转换h264h265,正如预期的那样h265,媒体文件的版本小于,h264创建的程序能够:

  / *
    * H264-> H265 
   *音频->重混合(未触摸)
   * MP4-MP4 
* / 
  StreamingParams sp = { 0 };   
  sp.copy_audio = 1 ;
  sp.copy_video = 0 ;
  sp.video_codec = “ libx265 ”;
  sp.codec_priv_key = “ x265-params ” ;
  sp.codec_priv_value = “ keyint = 60:min-keyint = 60:scenecut = 0 ” ;
  / *
    * H264-> H264(固定gop)
   *音频->重混合(未触摸)
   * MP4-MP4 
* / 
  StreamingParams sp = { 0 };   
  sp.copy_audio = 1 ;
  sp.copy_video = 0 ;
  sp.video_codec = “ libx264 ”;
  sp.codec_priv_key = “ x264参数” ;
  sp.codec_priv_value = “ keyint = 60:min-keyint = 60:scenecut = 0:force-cfr = 1 ” ;
  / *
    * H264-> H264(固定gop)
   *音频->重混合(
未修改)* MP4-    片段MP4 
* / 
  StreamingParams sp = { 0 };   
  sp.copy_audio = 1 ;
  sp.copy_video = 0 ;
  sp.video_codec = “ libx264 ”;
  sp.codec_priv_key = “ x264参数” ;
  sp.codec_priv_value = “ keyint = 60:min-keyint = 60:scenecut = 0:force-cfr = 1 ” ;
  sp.muxer_opt_key = “ movflags ”;
  sp.muxer_opt_value = “ frag_keyframe + empty_moov + default_base_moof ” ;
  / *
    * H264-> H264(固定gop)
   *音频-> AAC 
   * MP4-MPEG-TS 
* / 
  StreamingParams sp = { 0 };   
  sp.copy_audio = 0 ;
  sp.copy_video = 0 ;
  sp.video_codec = “ libx264 ”;
  sp.codec_priv_key = “ x264参数” ;
  sp.codec_priv_value = “ keyint = 60:min-keyint = 60:scenecut = 0:force-cfr = 1 ” ;
  sp.audio_codec = “ aac ” ;
  sp.output_extension = “ .ts ” ;
  / * WIP:P->它不在VLC上播放,最终比特率很高
   * H264-> VP9 
   *音频-> Vorbis 
   * MP4-WebM 
* / // StreamingParams sp = {0}; // sp.copy_audio = 0; // sp.copy_video = 0; // sp.video_codec =“ libvpx-vp9”; // sp.audio_codec =“ libvorbis”; // sp.output_extension =“ .webm”;   
现在,说实话,这是难度比我想这会是我不得不挖掘到FFmpeg的命令行的源代码和测试了很多,我觉得我失去了一些东西,因为我不得不执行force-cfrh264工作而且我仍然看到一些警告消息,例如warning messages (forced frame type (5) at 80 was changed to frame type (3))

ffmpeg architecture(下)的更多相关文章

  1. ffmpeg architecture(中)

    ffmpeg architecture(中) 艰苦学习FFmpeg libav 您是否不奇怪有时会发出声音和视觉? 由于FFmpeg作为命令行工具非常有用,可以对媒体文件执行基本任务,因此如何在程序中 ...

  2. ffmpeg architecture(上)

    ffmpeg architecture(上) 目录 介绍 视频-您看到的是什么! 音频-您在听什么! 编解码器-缩小数据 容器-音频和视频的舒适场所 FFmpeg-命令行 FFmpeg命令行工具101 ...

  3. ffmpeg windows下编译ffmpeg

    windows下编译ffmpeg 今天由于工作需求需重新编译ffmpeg,百度,goole了一大堆,看眼花缭乱的,但几乎都是三种方案,大部分都是直接转发,一字不漏,错误的缺文件的还是照转,可是问题都大 ...

  4. FFmpeg Windows下安装与测试

    FFmpeg 简介 FFmpeg的名称来自MPEG视频编码标准,前面的"FF"代表"Fast Forward",FFmpeg是一套可以用来记录.转换数字音频.视 ...

  5. FFmpeg: mac下手动编译android上使用的FFmpeg(支持x86、armeabi-v7a、arm64-v8a)

    之前一直在linux下编译FFmpeg,最近换电脑了,尝试了下在mac下编译ffmpeg,特记录之. 一. 准备工作 1. 下载FFmpeg.(http://ffmpeg.org/download.h ...

  6. ffmpeg windows下编译安装

    安装msys2 更新源使下载速度更快 进入msys64/etc/pacman.d/目录中,分别在三个文件中增加mirrorlist.mingw32Server = http://mirrors.ust ...

  7. ffmpeg Windows下采集摄像头一帧数据,并保存为bmp图片

    这里请注意,在编译ffmpeg时,不要使用--disable-devices选项. 使用 --enable-encoder=rawvideo --enable-decoder=rawvideo 启用r ...

  8. [转载] ffmpeg Windows下采集摄像头一帧数据,并保存为bmp图片

    这里请注意,在编译ffmpeg时,不要使用--disable-devices选项. 使用 --enable-encoder=rawvideo --enable-decoder=rawvideo 启用r ...

  9. Linux下使用NDK编译FFMPEG(libstagefright)

    这个月要负责一个项目,使用FFMPEG渲染视频,主要是Android端的,由于性能要求,要使用硬解码,但网上大多数教程都是没有libstagefright的,所以个人觉得,生成的so库文件也是没有开启 ...

随机推荐

  1. POJ 2396 构造矩阵(上下流)

    题意:       要求构造一个矩阵,给你行和,列和,还有一些点的上下范围,输出一个满足题意的矩阵. 思路:       这个题目很经典,这是自己看上下流后接触的第一道题,感觉很基础的一道题目,现在我 ...

  2. Win64 驱动内核编程-29.强制解锁文件

    强制解锁文件 强制解锁因其他进程占用而无法删除的文件. 1.调用 ZwQuerySystemInformation 的 16 功能号来枚举系统里的句柄 2.打开拥有此句柄的进程并把此句柄复制到自己的进 ...

  3. 学习Canvas绘图与动画基础 制作弧和圆(五)

    1 <!DOCTYPE html> 2 <html> 3 <head lang="en"> 4 <meta charset="U ...

  4. <JVM上篇:内存与垃圾回收篇>02-类加载子系统

    笔记来源:尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) 同步更新:https://gitee.com/vectorx/NOTE_JVM https://codechina.cs ...

  5. thinkphp 中区块block和模板继承extend用法举例

    1.介绍 模板继承其实并不难理解,就好比类的继承一样,模板也可以定义一个基础模板(或者是布局),并且其中定义相关的区块(block),然后继承(extend)该基础模板的子模板中就可以对基础模板中定义 ...

  6. ColyseusJS 轻量级多人游戏服务器开发框架 - 中文手册(下)

    快速上手多人游戏服务器开发.后续会基于 Google Agones,更新相关 K8S 运维.大规模快速扩展专用游戏服务器的文章.拥抱️原生 Cloud-Native! 系列 ColyseusJS 轻量 ...

  7. Ubuntu部署项目

    一.Ubuntu目录结构 目录 说明 bin 存放二进制可执行文件(ls,cat,mkdir等) .exe== boot 存放用于系统引导时使用的各种文件 开机引导 dev 用于存放设备文件 打印机啥 ...

  8. MySQL DDL详情揭露

    前言: MySQL中DDL语句,即数据定义语言,用于创建.删除.修改.库或表结构,对数据库或表的结构操作.常见的有create,alter,drop等.这类语句通常会耗费很大代价,特别是对于大表做表结 ...

  9. 自带的 print 函数居然会报错?

    前言 最近用 Python 写了几个简单的脚本来处理一些数据,因为只是简单功能所以我就直接使用 print 来打印日志. 任务运行时偶尔会出现一些异常: 因为我在不同地方都有打印日志,导致每次报错的地 ...

  10. Redis6.x学习笔记(四)复制

    复制概述 Redis支持复制的功能,以实现当一台服务器的数据更新后,自动将新的数据异步同步到其它数据库. Redis复制实现中,把数据库分为主数据库master和从数据库slave,主数据库可以进行读 ...