FFMPEG 之 AVFormat

1 描述

它是一个I/O 和muxing/demuxing库,处理各种媒体容器格式库,主要包括两大功能:demuxing ,如将一个媒体文件拆分成音视频流, muxing,如将音视频数据写入一个媒体文件中。同时它还具有I/O module 可支持各种协议,如文件,tcp,http等等。muing 由AVInputFormat 结构体去描述,相反地,demuxing 由AVOutputFormat 去描述。

  AVFormatContext 它是整个AVFormat 的上下文, 所有有关demuxing/muxing 都包含这结构体中。其中最重要的三个成员是:

1 struct AVInputFormat *iformat;/ struct AVOutputFormat *oformat;//muxing/demuxing 上下文

2 AVStream **streams;//存在音视频流

3 AVIOContext *pb;//读取媒体数据。

2 解复用器 

       解复用器读取媒体文件,并将其分割成数据块(视频/音频/字幕),a packet 包含一个或者多个已编码音视频单元,通过avformat_open_input() 去打开这文件,通过av_read_frame() 去读取数据包,最后通过avformat_close_input去关闭清除。如下表为FFMPEG Demuxing 部分开出来的API ,其定义是在avformat.h 文件中,实现在utils.c 文件中。如下表为demuxing 部分开出的API。在avformat_open_input() 之后,也能读出一些mediainfo 信息,但是要获取更多更完整的mediainfo 信息需要通过avformat_find_stream_info(), 前者只会读取文件头,后者还会解码一些帧。

int av_read_frame(AVFormatContext *s, AVPacket *ptk)  这函数作用是得到stream 的下一帧,如果读取成功将返回0 ,失败或者流结束返回负值, 其中流结束返回的负值是AVERROR_EOF。

函数原型

作用

AVInputFormat *av_find_input_format(char *name)

获得一个基于后缀名的AVInputFormat结构体

AVInputFormat*

av_probe_input_format(AVProbeData*pd , is_opened)

根据Probe data 去获得一个AVInputFormat结构体

AVInputFormat *

av_probe_input_format2(*pd, is_opend,*score_max)

根据Probe dat 去获得一个AVInputFormat结构体,当且只有获得的分数大于指定的分数才行

AVInputFormat*

av_probe_input_format3(pd,is_opend,*score_ret)

根据probe data去获得一个AVInPutFormat 结构体,并返回获得score

av_probe_input_buffer2(AVIOContext*,AVInputFormat*,char *url,logctx,offset,max_probe)

探测字节流,决定AVInputFormat

int avformat_open_input(AVFormatContext **, char*url,AVInputFormat *,AVDictionnary*)

打开一个流,并读取文件头

int avformat_find_stream_info(AVFormatContext*,AVDictionary*)

读取packets, 并获得一些media information

av_find_best_stream(AVFormatContext*,AVMediaType,AVCodec**)

根据输入参数找解码器

avProgram*av_find_program_from_stream(AVFormatContext *,AVProgram *last,int s)

获得program

int av_read_frame(AVFormatContext *s,AVPacket*pkt)

获取stream next frame

int av_seek_frame(AVFormatContext *s,int stream_index,int64_t timestamp,int flag)

seek to the keyframestamp

int avformat_seek_file(AVFormatContext *s,int stream_index,int min_ts,int ts,int max_ts)

通过指定min_ts,max_ts,去找最接近target ts timestamp

int avformat_flush(AVFormatContext *s)

flush all buffered data

int av_read_play(AVFormatContext *s)

开始播放一个网络流

int av_read_pause(AVFormatContext *s)

暂停一个网络流

void avformat_close_input(AVFormatContext**)

关闭一个打开的AVFormatContext

3 复用器

    复用器接收AVPacket 数据,并将其写入到指定的容器格式,avformat_write_header() 用于写文件头,av_write_frame()/av_interleaved_write_frame() 写数据packet, av_write_tailer() 用于完成文件。

Muxers 以AVPackets 形式接收数据,并写入指定的容器格式。其API 定义与avformat.c,实现位于util.c 中.如下表为muxing 部分开出的API。

函数原型

作用

int  avformat_write_header(AVFormatContext *s,AVDictionary **options)

allocate  stream private data,并将stream 头写入container

int  avformat_init_output (AVFormatContext*,AVDictionary **options)

allocate stream private data,但不写入streame head

int av_write_frame(AVFomatContext *s,AVPacker*pkt)

写packet 到输出的文件

int av_interleaved_write_frame(AVFormatContext *s,AVPacket *pkt)

写入packet 到输出文件,其data 必须是interleaved

int av_write_uncoded_frame(AVFomartContext *s,int stream_index ,AVFrame *frame)

写未编码的data到输出文件, 其数据格式可以不是interleaved

int av_interleaved_write_uncoded_frame(AVFormatContext *s,int stream_index,AVFrame *frame)

写入未编码的data, 其数据格式必须是interleaved

int av_write_uncoded_frame_query(AVFormatContext *s,int stream_index)

测试muxer 是否支持写入uncoded 数据

int av_write_tailer(AVFormatContext *s)

将stream 尾写入输出文件,并free private data

AVOutputFormat *av_guess_format(char *short_name,char *filename,char *mime_type)

根据输入参数,获得AVOutputForamt

AVCodecID av_guess_codec(AVOutputFormat *fmt,char *short_name,char*filename,char*mine_type,AVMediaType type)

根据输入参数获得CodecID

int av_get_output_timestamp(struct AVFormatContext *s,int stream,int64_t * dts,int64* wall)

获取当前数据的时序信息

 

4 metadata信息

    metadata是描述媒体文件的一些特有信息,比如音乐专辑信息,艺术家等等。被允许在muxing/demuxing 去设置/获取。AVFormatContext,AVStream,AVChapter,和AVProgram         可通过AVDictionary API 设置获取。

5 I/O read/write

ffmpeg 不但可以读取本地文件,同时它还可以读取网络文件,市面上常见的流媒体协议它都支持(如rtmp,rtsp,http等)。同各种容器的媒体文件demuxers一样,不同的协议底层都需有相应的文件去实现,ffmpeg 有对各种协议进行封装,avio.h开出的API 我们可以无感的读取各种协议的数据,如果我们使用ffmpeg 去播放media 的话,关于I/O 操作API 我们都接触不到,因为ffmpeg 内部已经帮我们做好了。AVFormatContext 中有个AVIOContext 结构体,它里面增加I/O读取buff 的管理,在AVIOContext 中的URLContext 成员,对应的就是具体的文件协议,比如rtmp,http,file hls,tcp 等等。它们之间的关系如下图所示:

6 core functions

列出一些查询libavformat 功能,申请core结构体的API。

函数原型

作用

unsigned avformat_version(void)

获取ffmpeg 的版本号

const char* avformat_configuration()

返回ffmpeg 编译的配置参数

int avformat_network_init(void)

对网络库进行全局初始化

int avformat_network_deinit(void)

撤销由avformat_network_init()完成的初始化

const AVOutputFormat *av_muxer_iterate(void **opaque)

 遍历所有注册的muxers

const AVInputFormat *av_demux_iterate(void **opaque)

遍历所有注册的demuxers

AVFormatContext *avformat_alloc_context()

allocate an AVFormatContext

avformat_free_context(AVFormatContext *)

free an AVFormatContext

const AVClass *avformat_get_class(void)

获取AVFormatContext 的AVClass

AVStream *avformat_new_stream(AVFormatContext *s,const AVCodec *s)

demuxing 被ffmpeg 内部调用.

在muxing 时, 需要在avformat_write_header() 之前由用户call, 用于创建media stream

uint8 * av_stream_new_side_date(AVStream *stream,enum AVPacketSideDateType type,int size)

申请附加信息空间

int av_stream_add_side_data(AVStream *st,enum AVPacketSideDataType type, uint8 *data,size_t size)

在stream 添加side data ,比如解码显示等附加信息

int8_t *av_stream_get_side_data(const AVStream *stream ,enum AVPacketSideDateType type,int size)

获取附加信息

AVProgram *av_new_program(AVFormatContext *s, int id )

根据AVFormatContext 申请AVprogram 结构体

 

7 utility functions

  描述一些与muxing/demuxing 有关的一些工具函数。在实际编程中,av_dump_format()还是用的比较多的, 比如demux 编程中, 我们初始化程序后,通过av_dump_format() 获取获取当前media 的info ,还是检查是否有初始化成功。

 

函数原型

作用

void av_hex_dump(FILE *f,const uint8_t *buf,int size )

将一段十六进制的数据包输入到一个文件流中

void av_hex_dump_log(void *avcl,int level,const uint8_t *buf,int size)

将一段十六进制的数据包在log 中输出

void av_pkt_dump2(FILE *f,const AVPacket *pkt,int dump_payload,const AVStream *st)

打印AVPacket信息

void av_pkt_dump_log2(void *avcl,int level,const AVPacket *pkt,int dump_payload,const AVStream *st)

AVPacket 信息,在log 中输出

enum AVCodecID av_codec_get_id(const struct AVCodecTag const *tag, enum AVCodecID id )

通过codec tag 寻找codecID

unsigned int av_codec_get_tag(const struct AVCodecTag *const *tag,enum AVCodecID id)

通过Codec Id 找AVCodec Tag

int av_index_search_timestamp(AVStream *st,int64_t timestamp, int flags)

获取特定时间撮的索引

int av_add_index_entry(AVStream *st,int64_t pos, int64_t timestamp,int size,int distance,int flag)

将索引条目添加到存储列表中

void av_url_split()

将url拆分成各个组件

void av_dump_format(AVFormatContext *ic,int index ,char *url,int is_output)

打印文件的详细信息,如duration,bitrate,streams,programs

int av_get_frame_filename2(char *url,int buf_size,)

return in “buf” the path with “d”,replaced by a number

int av_filename_number_test(char *filename)

检查文件名是否是一个编号序列生成

int av_sdp_create(AVFormatContext *ac[],int n_flies,char *buf,int size)

为rtp 会话生成sdp

int av_match_ext(const char *filename ,const char *extensions)

检查文件名是否与给定的扩展名相匹配

int avformat_query_codec(const AVOutputFormat *ofmt,enum AVCodecId codec_id,int std)

测试给定的容器是否可以存储指定的编码格式数据

AVRational av_guess_sample_aspect_ratio(AVFormatContext *format,AVStream *stream,AVFrame *frame)

根据stream 和 frame 的比例,猜测样本的比例

AVRational av_guess_frame_rate(AVFormatContext *ctx,AVStream *stream ,AVFrame *frame)

根据容器和编码器猜测帧速率

int avformat_match_stream_specifier(AVFormatContext *s,AVStream *st,char*spec)

检查流是否与流说明的spec相匹配

int avformat_transfer_internal_stream_timing_info(const AVOutputFormat,AVStream *ost,const AVStream *ist)

将内部时序信息从一个流传输到另一个流中

AVRational av_stream_get_codec_timebase(const AVStream *st)

从流内部获得编码器时基

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值