diff options
author | Michael Niedermayer <michaelni@gmx.at> | 2003-10-20 22:33:53 +0000 |
---|---|---|
committer | Michael Niedermayer <michaelni@gmx.at> | 2003-10-20 22:33:53 +0000 |
commit | 5ed9f2e5aaa922b423e87a1b812ca5489238d08e (patch) | |
tree | 6cba9ce51d8310ac6ded1b8b107d341129390d8b | |
parent | 1c0dcc391a302068199385ecc158aae52b5d3e69 (diff) | |
download | ffmpeg-5ed9f2e5aaa922b423e87a1b812ca5489238d08e.tar.gz |
use continued fractions to approximate a fraction if its numerator or denominator is too large
Originally committed as revision 2405 to svn://svn.ffmpeg.org/ffmpeg/trunk
-rw-r--r-- | libavcodec/utils.c | 40 |
1 files changed, 26 insertions, 14 deletions
diff --git a/libavcodec/utils.c b/libavcodec/utils.c index f9ed4f5ab6..34051ad3a4 100644 --- a/libavcodec/utils.c +++ b/libavcodec/utils.c @@ -684,7 +684,7 @@ char av_get_pict_type_char(int pict_type){ int av_reduce(int *dst_nom, int *dst_den, int64_t nom, int64_t den, int64_t max){ int exact=1, sign=0; - int64_t gcd, larger; + int64_t gcd; assert(den != 0); @@ -698,22 +698,34 @@ int av_reduce(int *dst_nom, int *dst_den, int64_t nom, int64_t den, int64_t max) sign= 1; } - for(;;){ //note is executed 1 or 2 times - gcd = ff_gcd(nom, den); - nom /= gcd; - den /= gcd; + gcd = ff_gcd(nom, den); + nom /= gcd; + den /= gcd; - larger= FFMAX(nom, den); - - if(larger > max){ - int64_t div= (larger + max - 1) / max; - nom = (nom + div/2)/div; - den = (den + div/2)/div; - exact=0; - }else - break; + if(nom > max || den > max){ + AVRational a0={0,1}, a1={1,0}; + exact=0; + + for(;;){ + int64_t x= nom / den; + int64_t a2n= x*a1.num + a0.num; + int64_t a2d= x*a1.den + a0.den; + + if(a2n > max || a2d > max) break; + + nom %= den; + + a0= a1; + a1= (AVRational){a2n, a2d}; + if(nom==0) break; + x= nom; nom=den; den=x; + } + nom= a1.num; + den= a1.den; } + assert(ff_gcd(nom, den) == 1); + if(sign) nom= -nom; *dst_nom = nom; |