JM8.6解码端是如何从配置文件decoder.cfg获取数据的? (init_conf函数)

2024-02-06 16:38

本文主要是介绍JM8.6解码端是如何从配置文件decoder.cfg获取数据的? (init_conf函数),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

      在ldecod.c文件的main中调用了init_conf函数,这个函数实际上就实现了程序从配置文件读数据的过程. 之前说过,在JM8.6编码端,Configure函数起着做配置文件encoder_baseline.cfg数据的作用,现在来看看解码端的init_conf函数.

      main中是这样调用的:

init_conf(input, argv[1]);


     其中input是一个结构体指针,字符串指针argv[1]实际上就指向了字符串"decoder.cfg", 进入init_conf函数看看:

void init_conf(struct inp_par *inp,char *config_filename)
{FILE *fd;int NAL_mode;// read the decoder configuration fileif((fd=fopen(config_filename,"r")) == NULL){snprintf(errortext, ET_SIZE, "Error: Control file %s not found\n",config_filename);error(errortext, 300);}fscanf(fd,"%s",inp->infile);                // H.264 compressed input bitsream// 过滤配置文件中的“注释”fscanf(fd,"%*[^\n]");fscanf(fd,"%s",inp->outfile);               // YUV 4:2:2 input formatfscanf(fd,"%*[^\n]");fscanf(fd,"%s",inp->reffile);               // reference filefscanf(fd,"%*[^\n]");// Frame buffer sizefscanf(fd,"%d,",&inp->dpb_size);   // may be overwritten in case of RTP NALfscanf(fd,"%*[^\n]");if (inp->dpb_size < 1){snprintf(errortext, ET_SIZE, "Decoded Picture Buffer Size is %d. It has to be at least 1",inp->dpb_size);error(errortext,1);}fscanf(fd,"%d",&(NAL_mode));                // NAL modefscanf(fd,"%*[^\n]");switch(NAL_mode){case 0:inp->FileFormat = PAR_OF_ANNEXB;break;case 1:inp->FileFormat = PAR_OF_RTP;break;default:snprintf(errortext, ET_SIZE, "NAL mode %i is not supported", NAL_mode);error(errortext,400);}fscanf(fd,"%d,",&inp->ref_offset);   // offset used for SNR computationfscanf(fd,"%*[^\n]");fscanf(fd,"%d,",&inp->poc_scale);   // offset used for SNR computationfscanf(fd,"%*[^\n]");if (inp->poc_scale < 1 || inp->poc_scale > 2){snprintf(errortext, ET_SIZE, "Poc Scale is %d. It has to be 1 or 2",inp->poc_scale);error(errortext,1);}#ifdef _LEAKYBUCKET_fscanf(fd,"%ld,",&inp->R_decoder);             // Decoder ratefscanf(fd, "%*[^\n]");fscanf(fd,"%ld,",&inp->B_decoder);             // Decoder buffer sizefscanf(fd, "%*[^\n]");fscanf(fd,"%ld,",&inp->F_decoder);             // Decoder initial delayfscanf(fd, "%*[^\n]"); fscanf(fd,"%s",inp->LeakyBucketParamFile);    // file where Leaky Bucket params (computed by encoder) are storedfscanf(fd,"%*[^\n]");
#endiffclose (fd);#if TRACEif ((p_trace=fopen(TRACEFILE,"w"))==0)             // append new statistic at the end{snprintf(errortext, ET_SIZE, "Error open file %s!",TRACEFILE);error(errortext,500);}
#endifif ((p_out=fopen(inp->outfile,"wb"))==0){snprintf(errortext, ET_SIZE, "Error open file %s ",inp->outfile);error(errortext,500);}
/*  if ((p_out2=fopen("out.yuv","wb"))==0){snprintf(errortext, ET_SIZE, "Error open file %s ",inp->outfile);error(errortext,500);}*/fprintf(stdout,"--------------------------------------------------------------------------\n");fprintf(stdout," Decoder config file                    : %s \n",config_filename);fprintf(stdout,"--------------------------------------------------------------------------\n");fprintf(stdout," Input H.264 bitstream                  : %s \n",inp->infile);fprintf(stdout," Output decoded YUV 4:2:0               : %s \n",inp->outfile);fprintf(stdout," Output status file                     : %s \n",LOGFILE);if ((p_ref=fopen(inp->reffile,"rb"))==0){fprintf(stdout," Input reference file                   : %s does not exist \n",inp->reffile);fprintf(stdout,"                                          SNR values are not available\n");}elsefprintf(stdout," Input reference file                   : %s \n",inp->reffile);fprintf(stdout,"--------------------------------------------------------------------------\n");
#ifdef _LEAKYBUCKET_fprintf(stdout," Rate_decoder        : %8ld \n",inp->R_decoder);fprintf(stdout," B_decoder           : %8ld \n",inp->B_decoder);fprintf(stdout," F_decoder           : %8ld \n",inp->F_decoder);fprintf(stdout," LeakyBucketParamFile: %s \n",inp->LeakyBucketParamFile); // Leaky Bucket Param filecalc_buffer(inp);fprintf(stdout,"--------------------------------------------------------------------------\n");
#endiffprintf(stdout,"POC must = frame# or field# for SNRs to be correct\n");fprintf(stdout,"Frame    POC   QP  SnrY    SnrU    SnrV   Time(ms)\n");
}


       可见,通过调用init_conf函数,实际上就是把argv[1]对应的配置文件的数据往input对应的结构体中倾倒,像倒茶一般, 这样,程序中的全局的指针变量对应的结构体就获得了配置文件encoder.cfg中的数据.

      

       最后,附带给出配置文件decoder.cfg中的内容:

test.264                 ........H.264 coded bitstream
test_dec.yuv             ........Output file, YUV 4:2:0 format
test_rec.yuv             ........Ref sequence (for SNR)
10                       ........Decoded Picture Buffer size
0                        ........NAL mode (0=Annex B, 1: RTP packets)
0                        ........SNR computation offset
1                        ........Poc Scale (1 or 2)
500000                   ........Rate_Decoder
104000                   ........B_decoder
73000                    ........F_decoder
leakybucketparam.cfg     ........LeakyBucket Params

This is a file containing input parameters to the JVT H.264/AVC decoder.
The text line following each parameter is discarded by the decoder.

 

 

 

这篇关于JM8.6解码端是如何从配置文件decoder.cfg获取数据的? (init_conf函数)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/684916

相关文章

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

利用matlab bar函数绘制较为复杂的柱状图,并在图中进行适当标注

示例代码和结果如下:小疑问:如何自动选择合适的坐标位置对柱状图的数值大小进行标注?😂 clear; close all;x = 1:3;aa=[28.6321521955954 26.2453660695847 21.69102348512086.93747104431360 6.25442246899816 3.342835958564245.51365061796319 4.87

OpenCV结构分析与形状描述符(11)椭圆拟合函数fitEllipse()的使用

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C++11 算法描述 围绕一组2D点拟合一个椭圆。 该函数计算出一个椭圆,该椭圆在最小二乘意义上最好地拟合一组2D点。它返回一个内切椭圆的旋转矩形。使用了由[90]描述的第一个算法。开发者应该注意,由于数据点靠近包含的 Mat 元素的边界,返回的椭圆/旋转矩形数据

web群集--nginx配置文件location匹配符的优先级顺序详解及验证

文章目录 前言优先级顺序优先级顺序(详解)1. 精确匹配(Exact Match)2. 正则表达式匹配(Regex Match)3. 前缀匹配(Prefix Match) 匹配规则的综合应用验证优先级 前言 location的作用 在 NGINX 中,location 指令用于定义如何处理特定的请求 URI。由于网站往往需要不同的处理方式来适应各种请求,NGINX 提供了多种匹

Unity3D 运动之Move函数和translate

CharacterController.Move 移动 function Move (motion : Vector3) : CollisionFlags Description描述 A more complex move function taking absolute movement deltas. 一个更加复杂的运动函数,每次都绝对运动。 Attempts to

✨机器学习笔记(二)—— 线性回归、代价函数、梯度下降

1️⃣线性回归(linear regression) f w , b ( x ) = w x + b f_{w,b}(x) = wx + b fw,b​(x)=wx+b 🎈A linear regression model predicting house prices: 如图是机器学习通过监督学习运用线性回归模型来预测房价的例子,当房屋大小为1250 f e e t 2 feet^

JavaSE(十三)——函数式编程(Lambda表达式、方法引用、Stream流)

函数式编程 函数式编程 是 Java 8 引入的一个重要特性,它允许开发者以函数作为一等公民(first-class citizens)的方式编程,即函数可以作为参数传递给其他函数,也可以作为返回值。 这极大地提高了代码的可读性、可维护性和复用性。函数式编程的核心概念包括高阶函数、Lambda 表达式、函数式接口、流(Streams)和 Optional 类等。 函数式编程的核心是Lambda

PHP APC缓存函数使用教程

APC,全称是Alternative PHP Cache,官方翻译叫”可选PHP缓存”。它为我们提供了缓存和优化PHP的中间代码的框架。 APC的缓存分两部分:系统缓存和用户数据缓存。(Linux APC扩展安装) 系统缓存 它是指APC把PHP文件源码的编译结果缓存起来,然后在每次调用时先对比时间标记。如果未过期,则使用缓存的中间代码运行。默认缓存 3600s(一小时)。但是这样仍会浪费大量C