[FFmpeg-devel] [PATCH] eval: add bitor and bitand functions

Stefano Sabatini stefasab at gmail.com
Wed Mar 13 22:36:45 CET 2013


TODO: bump micro
---
 doc/eval.texi       |    8 ++++++++
 libavutil/eval.c    |    9 ++++++++-
 tests/ref/fate/eval |    9 +++++++++
 3 files changed, 25 insertions(+), 1 deletion(-)

diff --git a/doc/eval.texi b/doc/eval.texi
index 3b7964c..1dca2da 100644
--- a/doc/eval.texi
+++ b/doc/eval.texi
@@ -32,6 +32,14 @@ Compute arcsine of @var{x}.
 @item atan(x)
 Compute arctangent of @var{x}.
 
+ at item bitand(x, y)
+ at item bitor(x, y)
+Compute bitwise and/or operation on @var{x} and @var{y}.
+
+ at var{x} and @var{y} are converted to integers before executing the
+operation. Operation result is undefined if the integer representation
+of @var{x} or @var{y} takes more than 64 bits.
+
 @item ceil(expr)
 Round the value of expression @var{expr} upwards to the nearest
 integer. For example, "ceil(1.5)" is "2.0".
diff --git a/libavutil/eval.c b/libavutil/eval.c
index 4875725..9a688ae 100644
--- a/libavutil/eval.c
+++ b/libavutil/eval.c
@@ -145,7 +145,7 @@ struct AVExpr {
         e_pow, e_mul, e_div, e_add,
         e_last, e_st, e_while, e_taylor, e_root, e_floor, e_ceil, e_trunc,
         e_sqrt, e_not, e_random, e_hypot, e_gcd,
-        e_if, e_ifnot, e_print,
+        e_if, e_ifnot, e_print, e_bitand, e_bitor,
     } type;
     double value; // is sign in other types
     union {
@@ -284,6 +284,8 @@ static double eval_expr(Parser *p, AVExpr *e)
                 case e_last:return e->value * d2;
                 case e_st : return e->value * (p->var[av_clip(d, 0, VARS-1)]= d2);
                 case e_hypot:return e->value * (sqrt(d*d + d2*d2));
+                case e_bitand: return isnan(d) || isnan(d2) ? NAN : e->value * ((long int)d & (long int)d2);
+                case e_bitor:  return isnan(d) || isnan(d2) ? NAN : e->value * ((long int)d | (long int)d2);
             }
         }
     }
@@ -424,6 +426,8 @@ static int parse_primary(AVExpr **e, Parser *p)
     else if (strmatch(next, "gcd"   )) d->type = e_gcd;
     else if (strmatch(next, "if"    )) d->type = e_if;
     else if (strmatch(next, "ifnot" )) d->type = e_ifnot;
+    else if (strmatch(next, "bitand")) d->type = e_bitand;
+    else if (strmatch(next, "bitor" )) d->type = e_bitor;
     else {
         for (i=0; p->func1_names && p->func1_names[i]; i++) {
             if (strmatch(next, p->func1_names[i])) {
@@ -809,6 +813,9 @@ int main(int argc, char **argv)
         "gauss(0.1)",
         "hypot(4,3)",
         "gcd(30,55)*print(min(9,1))",
+        "bitor(42, 12)",
+        "bitand(42, 12)",
+        "bitand(NAN, 1)",
         NULL
     };
 
diff --git a/tests/ref/fate/eval b/tests/ref/fate/eval
index 59e3fe4..4a8e4e3 100644
--- a/tests/ref/fate/eval
+++ b/tests/ref/fate/eval
@@ -252,3 +252,12 @@ Evaluating 'gcd(30,55)*print(min(9,1))'
 
 12.700000 == 12.7
 0.931323 == 0.931322575
+
+Evaluating 'bitor(42, 12)'
+'bitor(42, 12)' -> 46.000000
+
+Evaluating 'bitand(42, 12)'
+'bitand(42, 12)' -> 8.000000
+
+Evaluating 'bitand(NAN, 1)'
+'bitand(NAN, 1)' -> nan
-- 
1.7.9.5



More information about the ffmpeg-devel mailing list