int main(int argc, char **argv)
{
int ret;
BenchmarkTimeStamps ti;
/* 初始化动态加载 */
init_dynload();
/* 注册退出回调函数 */
register_exit(ffmpeg_cleanup);
/* 设置stderr的缓冲模式(win32运行时需要) */
setvbuf(stderr, NULL, _IONBF, 0);
/* 设置日志打印选项 */
av_log_set_flags(AV_LOG_SKIP_REPEATED);
parse_loglevel(argc, argv, options);
#if CONFIG_AVDEVICE
/* 注册音视频设备 */
avdevice_register_all();
#endif
/* 初始化网络模块 */
avformat_network_init();
/* 显示ffmpeg的banner信息 */
show_banner(argc, argv, options);
/* 解析命令行选项并打开所有的输入/输出文件 */
ret = ffmpeg_parse_options(argc, argv);
if (ret < 0)
exit_program(1);
/* 检查是否没有指定输出文件并且没有输入文件 */
if (nb_output_files <= 0 && nb_input_files == 0) {
show_usage();
av_log(NULL, AV_LOG_WARNING, "Use -h to get full help or, even better, run 'man %s'\n", program_name);
exit_program(1);
}
/* 检查是否至少指定一个输出文件 */
if (nb_output_files <= 0) {
av_log(NULL, AV_LOG_FATAL, "At least one output file must be specified\n");
exit_program(1);
}
current_time = ti = get_benchmark_time_stamps();
/* 文件转码或抓取 */
if (transcode() < 0)
exit_program(1);
if (do_benchmark) {
int64_t utime, stime, rtime;
current_time = get_benchmark_time_stamps();
utime = current_time.user_usec - ti.user_usec;
stime = current_time.sys_usec - ti.sys_usec;
rtime = current_time.real_usec - ti.real_usec;
av_log(NULL, AV_LOG_INFO,
"bench: utime=%0.3fs stime=%0.3fs rtime=%0.3fs\n",
utime / 1000000.0, stime / 1000000.0, rtime / 1000000.0);
}
av_log(NULL, AV_LOG_DEBUG,
"%"PRIu64" frames successfully decoded, %"PRIu64" decoding errors\n",
decode_error_stat[0], decode_error_stat[1]);
/* 检查解码错误是否超过了指定的错误率 */
if ((decode_error_stat[0] + decode_error_stat[1]) * max_error_rate < decode_error_stat[1])
exit_program(69);
/* 根据是否接收到信号确定程序的返回码,并终止程序 */
exit_program(received_nb_signals ? 255 : main_return_code);
return main_return_code;
}
下面是对每个步骤的功能的详细解释:
针对main函数中几个重要的函数,下面将逐步解析他们的具体实现
// 解析命令行参数并设置选项
int ffmpeg_parse_options(int argc, char **argv)
{
// 定义选项解析上下文和错误信息
OptionParseContext octx;
uint8_t error[128];
int ret;
memset(&octx, 0, sizeof(octx));
/* split the commandline into an internal representation */
// 将命令行参数拆分为内部表示形式
ret = split_commandline(&octx, argc, argv, options, groups,
FF_ARRAY_ELEMS(groups));
if (ret < 0) {
av_log(NULL, AV_LOG_FATAL, "Error splitting the argument list: ");
goto fail;
}
/* apply global options */
// 应用全局选项
ret = parse_optgroup(NULL, &octx.global_opts);
if (ret < 0) {
av_log(NULL, AV_LOG_FATAL, "Error parsing global options: ");
goto fail;
}
/* configure terminal and setup signal handlers */
// 配置终端并设置信号处理程序
term_init();
/* open input files */
// 打开输入文件
ret = open_files(&octx.groups[GROUP_INFILE], "input", open_input_file);
if (ret < 0) {
av_log(NULL, AV_LOG_FATAL, "Error opening input files: ");
goto fail;
}
// 应用同步偏移量
apply_sync_offsets();
/* create the complex filtergraphs */
// 创建复杂的滤波器图
ret = init_complex_filters();
if (ret < 0) {
av_log(NULL, AV_LOG_FATAL, "Error initializing complex filters.\n");
goto fail;
}
/* open output files */
// 打开输出文件
ret = open_files(&octx.groups[GROUP_OUTFILE], "output", open_output_file);
if (ret < 0) {
av_log(NULL, AV_LOG_FATAL, "Error opening output files: ");
goto fail;
}
// 检查滤波器的输出
check_filter_outputs();
fail:
// 反初始化选项解析上下文
uninit_parse_context(&octx);
if (ret < 0) {
// 如果有错误,将错误信息输出到日志
av_strerror(ret, error, sizeof(error));
av_log(NULL, AV_LOG_FATAL, "%s\n", error);
}
return ret;
}
函数功能:该函数用于解析命令行参数并进行相应处理,包括拆分命令行参数为内部表示形式、应用全局选项、配置终端和设置信号处理程序、打开输入文件、应用同步偏移量、创建复杂的滤波器图、打开输出文件、检查滤波器的输出。如果出现错误,会将错误信息输出到日志。
static int transcode(void)
{
int ret, i;
OutputStream *ost;
InputStream *ist;
int64_t timer_start;
int64_t total_packets_written = 0;
//执行转码初始化操作,并将返回值赋给变量ret。
ret = transcode_init();
if (ret < 0)
goto fail;
//检查是否启用了从标准输入进行交互的选项。
if (stdin_interaction) {
av_log(NULL, AV_LOG_INFO, "Press [q] to stop, [?] for help\n");
}
// 记录转码开始的时间,用于计算整个过程的耗时。
timer_start = av_gettime_relative();
//执行输入线程的初始化,启动输入线程来读取输入文件。
if ((ret = init_input_threads()) < 0)
goto fail;
//持续循环进行转码操作,直到接收到停止信号。
while (!received_sigterm) {
int64_t cur_time= av_gettime_relative(); //记录当前时间,用于计算转码过程中的耗时。
/* if 'q' pressed, exits */
/* 检查是否启用了从标准输入进行交互的选项。如果启用了与用户交互的选项,
将会检查是否需要检查键盘交互来停止转码。*/
if (stdin_interaction)
if (check_keyboard_interaction(cur_time) < 0)
break;
/* check if there's any stream where output is still needed */
// 检查是否还需要写入输出流
if (!need_output()) {
av_log(NULL, AV_LOG_VERBOSE, "No more output streams to write to, finishing.\n");
break;
}
// 执行转码的一步操作,比如从输入文件中读取数据,应用滤镜,写入输出文件等等。
ret = transcode_step();
if (ret < 0 && ret != AVERROR_EOF) {
av_log(NULL, AV_LOG_ERROR, "Error while filtering: %s\n", av_err2str(ret));
break;
}
/* dump report by using the output first video and audio streams */
// 根据输出流的第一个视频和音频流打印转码报告。显示转码的进度和统计信息。
print_report(0, timer_start, cur_time);
}
free_input_threads(); //释放输入线程的资源。
/* at the end of stream, we must flush the decoder buffers */
//循环处理每个输入流的解码器缓冲区,以确保在流结束时进行刷新。
for (i = 0; i < nb_input_streams; i++) {
ist = input_streams[i];
if (!input_files[ist->file_index]->eof_reached) {
process_input_packet(ist, NULL, 0);
}
}
flush_encoders(); // 刷新编码器的缓冲区,将剩余的数据编码为输出帧。
term_exit(); // 执行终结操作,例如释放资源和关闭设备。
/* write the trailer if needed */
// 循环写入每个输出文件的尾部。
for (i = 0; i < nb_output_files; i++) {
ret = of_write_trailer(output_files[i]);
if (ret < 0 && exit_on_error)
exit_program(1);
}
/* dump report by using the first video and audio streams */
// 根据第一个视频和音频流打印转码报告,显示转码的最终进度和统计信息。
print_report(1, timer_start, av_gettime_relative());
/* close each encoder */
// 循环关闭每个输出流的编码器。
for (i = 0; i < nb_output_streams; i++) {
uint64_t packets_written;
ost = output_streams[i];
packets_written = atomic_load(&ost->packets_written);
total_packets_written += packets_written;
if (!packets_written && (abort_on_flags & ABORT_ON_FLAG_EMPTY_OUTPUT_STREAM)) {
av_log(NULL, AV_LOG_FATAL, "Empty output on stream %d.\n", i);
exit_program(1);
}
}
// 如果输出为空且设置了对空输出流终止转码的标志,将会退出程序。
if (!total_packets_written && (abort_on_flags & ABORT_ON_FLAG_EMPTY_OUTPUT)) {
av_log(NULL, AV_LOG_FATAL, "Empty output\n");
exit_program(1);
}
/* close each decoder */
// 循环关闭每个输入流的解码器。
for (i = 0; i < nb_input_streams; i++) {
ist = input_streams[i];
if (ist->decoding_needed) {
avcodec_close(ist->dec_ctx);
}
}
// 释放所有的硬件设备。
hw_device_free_all();
/* finished ! */
ret = 0;
fail:
free_input_threads(); // 释放输入线程所占用的资源。
// 如果存在输出流,将会释放输出流相关的资源。
if (output_streams) {
for (i = 0; i < nb_output_streams; i++) {
ost = output_streams[i];
if (ost) {
if (ost->logfile) {
if (fclose(ost->logfile))
av_log(NULL, AV_LOG_ERROR,
"Error closing logfile, loss of information possible: %s\n",
av_err2str(AVERROR(errno)));
ost->logfile = NULL;
}
av_freep(&ost->forced_kf_pts);
av_freep(&ost->apad);
av_freep(&ost->disposition);
av_dict_free(&ost->encoder_opts);
av_dict_free(&ost->sws_dict);
av_dict_free(&ost->swr_opts);
}
}
}
return ret;
}
通过以上的流程,transcode()函数实现了将输入数据进行转码处理,并将结果写入输出流中,最终完成转码操作。每个流程均有特定的功能,按照一定的顺序进行处理,达到转码的目的。每个流程的操作都与上下文关系密切,根据上一个流程的处理结果和状态来决定下一个流程的操作和流程逻辑。整个转码过程在循环中进行,直到所有的输入数据处理完毕、输出流刷新完毕,并将输出文件的文件尾写入,最后退出程序。
static int transcode_init(void)
{
int ret = 0, i, j, k;
OutputStream *ost;
InputStream *ist;
char error[1024] = {0};
// 遍历 nb_filtergraphs 列表中的 FilterGraph 结构体。
for (i = 0; i < nb_filtergraphs; i++) {
FilterGraph *fg = filtergraphs[i];
for (j = 0; j < fg->nb_outputs; j++) {
OutputFilter *ofilter = fg->outputs[j];
if (!ofilter->ost || ofilter->ost->source_index >= 0) // 如果输出的 ost 为空或 source_index 大于等于0,则跳过
continue;
if (fg->nb_inputs != 1) // 如果 FilterGraph 的输入数不等于1,则跳过。
continue;
for (k = nb_input_streams-1; k >= 0 ; k--) // 从后往前遍历输入流列表,找到与 FilterGraph 的第一个输入匹配的输入流。
if (fg->inputs[0]->ist == input_streams[k])
break;
ofilter->ost->source_index = k; // 将找到的输入流的索引赋值给 ost 的 source_index 属性
}
}
/* init framerate emulation */
// 初始化帧率模拟
for (i = 0; i < nb_input_files; i++) { // 遍历输入文件列表。
InputFile *ifile = input_files[i];
if (ifile->readrate || ifile->rate_emu) // 如果输入文件的 readrate 或 rate_emu 不为0,则对其所有流进行帧率模拟的初始化。
for (j = 0; j < ifile->nb_streams; j++)
input_streams[j + ifile->ist_index]->start = av_gettime_relative(); // 初始化方法是设置输入流的 start 属性为当前系统时间。
}
/* init input streams */
// 初始化输入流
for (i = 0; i < nb_input_streams; i++) // 遍历输入流列表。
if ((ret = init_input_stream(i, error, sizeof(error))) < 0) // 调用 init_input_stream() 对输入流进行初始化。
goto dump_format;
/*
* initialize stream copy and subtitle/data streams.
* Encoded AVFrame based streams will get initialized as follows:
* - when the first AVFrame is received in do_video_out
* - just before the first AVFrame is received in either transcode_step
* or reap_filters due to us requiring the filter chain buffer sink
* to be configured with the correct audio frame size, which is only
* known after the encoder is initialized.
*/
//初始化流复制和字幕/数据流:
for (i = 0; i < nb_output_streams; i++) { // 遍历输出流列表。
// 对于每个输出流,检查其是否为编码的 AVFrame 流。如果是,则跳过初始化。
if (output_streams[i]->enc_ctx &&
(output_streams[i]->st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO ||
output_streams[i]->st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO))
continue;
ret = init_output_stream_wrapper(output_streams[i], NULL, 0); // 初始化输出流。
if (ret < 0)
goto dump_format;
}
/* discard unused programs */
// 丢弃未使用的程序:
for (i = 0; i < nb_input_files; i++) { // 遍历输入文件列表。
InputFile *ifile = input_files[i];
for (j = 0; j < ifile->ctx->nb_programs; j++) { // 对于每个输入文件,遍历其所有程序。
AVProgram *p = ifile->ctx->programs[j];
int discard = AVDISCARD_ALL;
// 检查每个程序关联的流是否需要丢弃。
for (k = 0; k < p->nb_stream_indexes; k++)
if (!input_streams[ifile->ist_index + p->stream_index[k]]->discard) {
discard = AVDISCARD_DEFAULT; // 如果有未被丢弃的流,则将程序的丢弃标志设置为默认丢弃。
break;
}
p->discard = discard;
}
}
dump_format:
/* dump the stream mapping */
// 打印流映射信息:
av_log(NULL, AV_LOG_INFO, "Stream mapping:\n");
for (i = 0; i < nb_input_streams; i++) {
ist = input_streams[i];
for (j = 0; j < ist->nb_filters; j++) {
// 对于每个输入流的滤镜图,如果不是简单滤镜图,则打印其相关信息。
if (!filtergraph_is_simple(ist->filters[j]->graph)) {
av_log(NULL, AV_LOG_INFO, " Stream #%d:%d (%s) -> %s",
ist->file_index, ist->st->index, ist->dec ? ist->dec->name : "?",
ist->filters[j]->name);
if (nb_filtergraphs > 1)
av_log(NULL, AV_LOG_INFO, " (graph %d)", ist->filters[j]->graph->index);
av_log(NULL, AV_LOG_INFO, "\n");
}
}
}
// 打印输出流映射信息
for (i = 0; i < nb_output_streams; i++) { // 遍历输出流列表。
ost = output_streams[i];
// 如果输出流是附加的文件流,则打印其文件名和关联的流信息。
if (ost->attachment_filename) {
/* an attached file */
av_log(NULL, AV_LOG_INFO, " File %s -> Stream #%d:%d\n",
ost->attachment_filename, ost->file_index, ost->index);
continue;
}
// 如果输出流是来自复杂滤镜图的输出流,则打印滤镜图和关联的流信息。
if (ost->filter && !filtergraph_is_simple(ost->filter->graph)) {
/* output from a complex graph */
av_log(NULL, AV_LOG_INFO, " %s", ost->filter->name);
if (nb_filtergraphs > 1)
av_log(NULL, AV_LOG_INFO, " (graph %d)", ost->filter->graph->index);
av_log(NULL, AV_LOG_INFO, " -> Stream #%d:%d (%s)\n", ost->file_index,
ost->index, ost->enc ? ost->enc->name : "?");
continue;
}
// 否则,打印输入流和输出流的对应关系。
av_log(NULL, AV_LOG_INFO, " Stream #%d:%d -> #%d:%d",
input_streams[ost->source_index]->file_index,
input_streams[ost->source_index]->st->index,
ost->file_index,
ost->index);
if (ost->enc_ctx) {
const AVCodec *in_codec = input_streams[ost->source_index]->dec;
const AVCodec *out_codec = ost->enc;
const char *decoder_name = "?";
const char *in_codec_name = "?";
const char *encoder_name = "?";
const char *out_codec_name = "?";
const AVCodecDescriptor *desc;
if (in_codec) {
decoder_name = in_codec->name;
desc = avcodec_descriptor_get(in_codec->id);
if (desc)
in_codec_name = desc->name;
if (!strcmp(decoder_name, in_codec_name))
decoder_name = "native";
}
if (out_codec) {
encoder_name = out_codec->name;
desc = avcodec_descriptor_get(out_codec->id);
if (desc)
out_codec_name = desc->name;
if (!strcmp(encoder_name, out_codec_name))
encoder_name = "native";
}
av_log(NULL, AV_LOG_INFO, " (%s (%s) -> %s (%s))",
in_codec_name, decoder_name,
out_codec_name, encoder_name);
} else
av_log(NULL, AV_LOG_INFO, " (copy)");
av_log(NULL, AV_LOG_INFO, "\n");
}
if (ret) {
av_log(NULL, AV_LOG_ERROR, "%s\n", error);
return ret;
}
atomic_store(&transcode_init_done, 1);
return 0;
}
static int transcode_step(void)
{
OutputStream *ost; // 输出流指针
InputStream *ist = NULL; // 输入流指针
int ret;
// 选择输出流
ost = choose_output();
if (!ost) {
// 如果没有输出流
if (got_eagain()) { // 检查是否发生 EAGAIN 错误
reset_eagain();
av_usleep(10000); // 延时等待一段时间
return 0;
}
av_log(NULL, AV_LOG_VERBOSE, "No more inputs to read from, finishing.\n");
return AVERROR_EOF;
}
if (ost->filter && !ost->filter->graph->graph) {
// 如果输出流有滤镜,并且滤镜图没有构建,则进行滤镜图的配置
if (ifilter_has_all_input_formats(ost->filter->graph)) {
ret = configure_filtergraph(ost->filter->graph);
if (ret < 0) {
av_log(NULL, AV_LOG_ERROR, "Error reinitializing filters!\n");
return ret;
}
}
}
if (ost->filter && ost->filter->graph->graph) {
// 如果输出流有滤镜,并且滤镜图已经构建,则进行相关操作
if (av_buffersink_get_type(ost->filter->filter) == AVMEDIA_TYPE_AUDIO)
init_output_stream_wrapper(ost, NULL, 1); // 音频特殊处理,进行输出流的初始化
ret = transcode_from_filter(ost->filter->graph, &ist); // 从滤镜开始转码
if (ret < 0)
return ret;
if (!ist)
return 0;
} else if (ost->filter) {
// 如果输出流有滤镜,但滤镜图没有构建,则尝试找到输入流
int i;
for (i = 0; i < ost->filter->graph->nb_inputs; i++) {
InputFilter *ifilter = ost->filter->graph->inputs[i];
if (!ifilter->ist->got_output && !input_files[ifilter->ist->file_index]->eof_reached) {
ist = ifilter->ist;
break;
}
}
if (!ist) {
ost->inputs_done = 1; // 输入流处理完毕
return 0;
}
} else {
// 如果输出流没有滤镜,则直接使用输入流
av_assert0(ost->source_index >= 0);
ist = input_streams[ost->source_index];
}
ret = process_input(ist->file_index); // 处理输入
if (ret == AVERROR(EAGAIN)) {
// 如果返回 EAGAIN 错误
if (input_files[ist->file_index]->eagain)
ost->unavailable = 1;
return 0;
}
if (ret < 0)
return ret == AVERROR_EOF ? 0 : ret;
return reap_filters(0); // 回收滤镜
}