ffmpeg 中av_rescale_rnd 的含义

原创 2016年05月31日 15:31:05

一、函数声明:

  1. int64_t av_rescale_rnd(int64_t a, int64_t b, int64_t c, enum AVRounding rnd);


直接看代码, 它的作用是计算 "a * b / c" 的值并分五种方式来取整.
用在FFmpeg中, 
则是将以 "时钟基c" 表示的 数值a 转换成以 "时钟基b" 来表示。


一共有5种方式:

  1. AV_ROUND_ZERO     = 0, // Round toward zero.      趋近于0
  2. AV_ROUND_INF      = 1, // Round away from zero.   趋远于0
  3. AV_ROUND_DOWN     = 2, // Round toward -infinity. 趋于更小的整数
  4. AV_ROUND_UP       = 3, // Round toward +infinity. 趋于更大的整数
  5. AV_ROUND_NEAR_INF = 5, // Round to nearest and halfway cases away from zero.
  6.                        //                         四舍五入,小于0.5取值趋向0,大于0.5取值趋远于0


二、函数定义(见于libavutil/mathematics.c):

int64_t av_rescale_rnd(int64_t a, int64_t b, int64_t c, enum AVRounding rnd)
{
  int64_t r=0;
  assert(c > 0);
  assert(b >=0);
  assert((unsigned)rnd<=5 && rnd!=4);

  /* 将小于0的整数,转换成大于0的整数来计算 */
  if (a<0 && a != INT64_MIN) 
    return -av_rescale_rnd(-a, b, c, rnd ^ ((rnd>>1)&1));

  if (rnd==AV_ROUND_NEAR_INF) 
    r= c / 2;
  else if (rnd&1) 
    r= c - 1;

  if (b<=INT_MAX && c<=INT_MAX)
  {
    /* 处理b和c 小于32位整数的情况 */
    if (a<=INT_MAX)
      return (a * b + r)/c;
    else
      return a/c*b + (a%c*b + r)/c;
  }
  else
  {
    /* 处理b或c 大于64位整数的算法 */
    uint64_t a0= a&0xFFFFFFFF;
    uint64_t a1= a>>32;
    uint64_t b0= b&0xFFFFFFFF;
    uint64_t b1= b>>32;
    uint64_t t1= a0*b1 + a1*b0;
    uint64_t t1a= t1<<32;
    int i;

    a0 = a0*b0 + t1a;
    a1 = a1*b1 + (t1>>32) + (a0<t1a);
    a0 += r;
    a1 += a0<r;

    for (i=63; i>=0; i--)
    {
      a1+= a1 + ((a0>>i)&1);
      t1+=t1;
      if (c <= a1)
      {
        a1 -= c;
        t1++;
      }
    }
  }
  return t1;
}

三、实例分析
将以"1MHz时钟基" 表示的 "PTS/DTS值a" 转换成以 "90kHz时钟基" 表示。
av_rescale_q(a=-10949117256, 
             bq={num=1, den=1000000}, 
             cq={num=1, den=90000))
{
  int64_t b= bq.num * (int64_t)cq.den; // = 1 * 90000 = 90000;
  int64_t c= cq.num * (int64_t)bq.den; // = 1 * 1000000 = 1000000
  return av_rescale_rnd(a, b, c, 5); 
}

av_rescale_rnd(a=10949117256, b=90000, c=1000000, rnd=5)
{
  if (rnd==5) 
    r = c / 2; // r =500000;
  
  if (b<=INT_MAX && c<=INT_MAX)
  {
    if (a<=INT_MAX)
      return (a * b + r)/c;
    else
      return a/c*b + (a%c*b + r)/c; // = 10949117256 / 1000000 * 90000 + 
                                       // (10949117256 % 1000000 * 90000 + 500000) / 1000000
                                       // = 985420553
  }
  else
  {
    ...
  } 
}



版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

ffmpeg 中av_rescale_rnd 的含义

一、函数声明: int64_t av_rescale_rnd(int64_t a, int64_t b, int64_t c, enum AVRounding rnd); 直接看代码, 它的作用是...

ffmpeg 源代码简单分析 : av_read_frame()

ffmpeg中的av_read_frame()的作用是读取码流中的音频若干帧或者视频一帧。例如,解码视频的时候,每解码一个视频帧,需要先调用 av_read_frame()获得一帧视频的压缩数据,然后...

ffmpeg框架阅读笔记一:读取数据帧函数 int av_read_frame(AVFormatContext *s, AVPacket *pkt)

做音频播放,数据源从完整音频文件变成了网络拉取音频流到缓冲区,实现片段流的播放。分析av_read_frame过程,自定义获取数据源的功能。重点查找如何读入数据源。首先分析函数av_read_fram...

[9]【ffmpeg源码分析 1】av_register_all()和avcodec_register_all()

日期:2016.10.18 作者:isshe github:github.com/isshe 邮箱:i.sshe@outlook.com 前言接下来打算学习一下编解码,不过好...

ffmpeg 源代码简单分析 : av_register_all()

此前写了好几篇ffmpeg源代码分析文章,列表如下: 图解FFMPEG打开媒体的函数avformat_open_input ffmpeg 源代码简单分析 : av_register_all(...

FFmpeg源代码简单分析:av_write_frame()

打算写两篇文章简单分析FFmpeg的写文件用到的3个函数avformat_write_header(),av_write_frame()以及av_write_trailer()。上篇文章已经分析了av...

FFmpeg总结(三)AV系列结构体之AVCodecContext

位置: libavcodec/avcodec.h 描述:主要扩展API的结构体New fields can be added to the end with minor version bumps.

ffmpeg 源代码简单分析 : av_register_all()

前一阵子看了一下ffmpeg的源代码,并且做了一些注释,在此贴出来以作备忘。 本文分析一下ffmpeg注册复用器,编码器等的函数av_register_all()。该函数在所有基于ffmpeg的应用程...

ffmpeg 源代码简单分析 : av_read_frame()

此前写了好几篇ffmpeg源代码分析文章,列表如下: 图解FFMPEG打开媒体的函数avformat_open_input ffmpeg 源代码简单分析 : av_register_all(...

ffmpeg源码简析(九)av_log(),AVClass,AVOption

1.av_log()av_log()是FFmpeg中输出日志的函数。随便打开一个FFmpeg的源代码文件,就会发现其中遍布着av_log()函数。一般情况下FFmpeg类库的源代码中是不允许使用pri...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)