首先,我们查看一下后面要用到的枚举变量AVRounding。

它定义了舍入法,即如何将含有小数部分的数字转化为整数的方法。

enum AVRounding {
    AV_ROUND_ZERO     = 0, ///< Round toward zero. 
    AV_ROUND_INF      = 1, ///< Round away from zero. 
    AV_ROUND_DOWN     = 2, ///< Round toward -infinity. 
    AV_ROUND_UP       = 3, ///< Round toward +infinity.
    AV_ROUND_NEAR_INF = 5, ///< Round to nearest and halfway cases away from zero.
    AV_ROUND_PASS_MINMAX = 8192,
};
  • AV_ROUND_ZERO
    朝向零点舍入
  • AV_ROUND_INF
    背向零点方向舍入
  • AV_ROUND_DOWN
    朝向负轴方向舍入
  • AV_ROUND_UP
    朝向正轴方向舍入
  • AV_ROUND_NEAR_INF
    四舍五入法,舍入到离当前最近的整数
  • AV_ROUND_PASS_MINMAX
    这个Flag并不表示一个舍入法,而是针对 INT64_MIN 和 INT64_MIN 这两个特殊值设立的,表示当针对这两个特殊值做舍入时,直接返回这两个特殊值即可,不做舍入。
    AV_NOPTS_VALUE 等同于 INT64_MIN
    以上5个Flag是互斥的,同时只能使用一个,而想要使用这个Flag时,要使用 or 位运算符来组合上述Flag使用。
    示例代码:
av_rescale_rnd(3, 1, 2, AV_ROUND_UP | AV_ROUND_PASS_MINMAX);             
   	// Rescaling 3:                              
	//     Calculating 3 * 1 / 2                         
	//     3 / 2 is rounded up to 2                     
	//     => 2                                                        
	av_rescale_rnd(AV_NOPTS_VALUE, 1, 2, AV_ROUND_UP | AV_ROUND_PASS_MINMAX);
	// Rescaling AV_NOPTS_VALUE:                         
	//     AV_NOPTS_VALUE == INT64_MIN                   
	//     AV_NOPTS_VALUE is passed through             
	//     => AV_NOPTS_VALUE

 

我们介绍一下AVRational类型。我们知道,在计算机中,我们是无法准确表示一个有小数的有理数的。而在FFmpeg中,则需要精确地表示有理数,为了避免这个问题,提供了AVRational以分数的形式来保存有理数,这样就可以准确表示要表达的值。

如AVRational{1,3},就表示三分之一,如果转而用浮点数表示,那么就只能表示为 0.33333 这样的只有有限位的数,并不准确。

FFmpeg还提供了针对AVRational诸多计算函数,我们下面要介绍的函数在底层可能就使用了它们。

 

下面我们详细了解一些这些API:

int64_t av_const av_gcd(int64_t a, int64_t b);

/*
	求整数a和整数b的最大公约数。
	当a和b均大于0时,返回值也大于0;当a和b有一个为0是,返回0。
	注意:负数没有最大公约数。
*/
int64_t av_rescale(int64_t a, int64_t b, int64_t c) av_const;
int64_t av_rescale_rnd(int64_t a, int64_t b, int64_t c, enum AVRounding rnd) av_const;
int64_t av_rescale_q(int64_t a, AVRational bq, AVRational cq) av_const;
int64_t av_rescale_q_rnd(int64_t a, AVRational bq, AVRational cq,enum AVRounding rnd) av_const;

/*
	以上4个函数是非常类似的,都是对整数a做一个缩放,准确来说是:
		a * b / c
	不同的是:
	1、b和c的类型不同,可以是整数,也可以是AVRational
	2、由于计算结果不是整数,而返回值是整数,因此需要指定舍入法。不能指定舍入法的函数使用四舍五入法。
	
	为什么要提供这几个函数?它们所做的事情貌似非常的简单,这是因为我们在计算 a*b/c 
        这个表达式时,先执行 a*b ,此时可能会导致值超出了int64_t的取值范围而导致错误,
        而这个函数则规避了这个隐藏的问题。
*/
int av_compare_ts(int64_t ts_a, AVRational tb_a, int64_t ts_b, AVRational tb_b);

/*
	比较两个以不同时基为单位的时间戳。
	如果 ts_a 在 ts_b 前,返回-1
	    ts_a 在 ts_b 后,返回1
	    ts_a 和 ts_b 表示同一个时间点,返回0
*/
int64_t av_compare_mod(uint64_t a, uint64_t b, uint64_t mod);

/*
	比较两个数的余数。
	如果 `a % mod < b % mod`,返回负值
	    `a % mod > b % mod`,返回正值
	    `a % mod == b % mod`,返回0
	    
	注意:除数mod,必须是2的幂,不能是任意整数。
*/
int64_t av_rescale_delta(AVRational in_tb, int64_t in_ts,  AVRational fs_tb, int duration, int64_t *last, AVRational out_tb);

/*
	和av_rescale_*()的类似,返回 in_ts * in_tb / out_tb。
	
	但其使用的场景却不同,这个函数被设计为在如下情况使用:当一个流中的每个音频packet需要将input时间戳转化到其他不同的时基时,
        就使用这个函数。与简单的av_rescale_q()相比,这个函数对可能的非一致frame duration
        (即同一个流中获取的帧中保存的采样数可能会变化)具有更好的健壮性。

	参数last是一个私有的状态值,同一个流上的按照顺序获取的packet必须使用同一个last。在第一次调用时,last应被初始化为AV_NOPTS_VALUE。
	
	参数fs_tb:duration的时基,一般要比in_tb和out_tb更加的细粒度化。
	参数duration:直到下次调用这个函数的时长,也就是当前packet或frame的duration
	参数last:指向一个时间戳,时基是fs_tb,是一个私有变量
	
	注意:在这个函数的上下文中,duration是以采样为单位的,而非秒。
*/
int64_t av_add_stable(AVRational ts_tb, int64_t ts, AVRational inc_tb, int64_t inc);

/*
	为时间戳添加一个值。
	可以注意到添加的值是以另一个时基为单位的。
	
	注意:这个函数保证,当同一个时间戳被重复添加值时不会累积舍入误差。
	代码:
	AVRational inRat = (AVRational){1,5};
        AVRational outRat = (AVRational){1,15};
        nRet = 0;
        for(int i = 0;i < 6;i++)
        {
            nRet = av_add_stable(inRat,nRet,outRat,4);
            printf("ret = %d.\n",nRet);
            //值依次为:1,3,4,5,7,8
        }
        printf("ret = %d.\n",nRet); //最终值为8而非6
*/