eval.c
上传用户:hjq518
上传日期:2021-12-09
资源大小:5084k
文件大小:6k
源码类别:

Audio

开发平台:

Visual C++

  1. /*
  2.  * simple arithmetic expression evaluator
  3.  *
  4.  * Copyright (c) 2002 Michael Niedermayer <michaelni@gmx.at>
  5.  *
  6.  * This library is free software; you can redistribute it and/or
  7.  * modify it under the terms of the GNU Lesser General Public
  8.  * License as published by the Free Software Foundation; either
  9.  * version 2 of the License, or (at your option) any later version.
  10.  *
  11.  * This library is distributed in the hope that it will be useful,
  12.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  14.  * Lesser General Public License for more details.
  15.  *
  16.  * You should have received a copy of the GNU Lesser General Public
  17.  * License along with this library; if not, write to the Free Software
  18.  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  19.  *
  20.  */
  21. /**
  22.  * @file eval.c
  23.  * simple arithmetic expression evaluator.
  24.  *
  25.  * see http://joe.hotchkiss.com/programming/eval/eval.html
  26.  */
  27. #include <stdio.h>
  28. #include <stdlib.h>
  29. #include <string.h>
  30. #include <math.h>
  31. #ifndef NAN
  32.   #define NAN 0
  33. #endif
  34. #ifndef M_PI
  35. #define M_PI 3.14159265358979323846
  36. #endif
  37. #define STACK_SIZE 100
  38. typedef struct Parser{
  39.     double stack[STACK_SIZE];
  40.     int stack_index;
  41.     char *s;
  42.     double *const_value;
  43.     const char **const_name;          // NULL terminated
  44.     double (**func1)(void *, double a); // NULL terminated
  45.     const char **func1_name;          // NULL terminated
  46.     double (**func2)(void *, double a, double b); // NULL terminated
  47.     char **func2_name;          // NULL terminated
  48.     void *opaque;
  49. } Parser;
  50. static void evalExpression(Parser *p);
  51. static void push(Parser *p, double d){
  52.     if(p->stack_index+1>= STACK_SIZE){
  53.         fprintf(stderr, "stack overflow in the parsern");
  54.         return;
  55.     }
  56.     p->stack[ p->stack_index++ ]= d;
  57. //printf("push %fn", d); fflush(stdout);
  58. }
  59. static double pop(Parser *p){
  60.     if(p->stack_index<=0){
  61.         fprintf(stderr, "stack underflow in the parsern");
  62.         return NAN;
  63.     }
  64. //printf("popn"); fflush(stdout);
  65.     return p->stack[ --p->stack_index ];
  66. }
  67. static int strmatch(const char *s, const char *prefix){
  68.     int i;
  69.     for(i=0; prefix[i]; i++){
  70.         if(prefix[i] != s[i]) return 0;
  71.     }
  72.     return 1;
  73. }
  74. static void evalPrimary(Parser *p){
  75.     double d, d2=NAN;
  76.     char *next= p->s;
  77.     int i;
  78.     /* number */
  79.     d= strtod(p->s, &next);
  80.     if(next != p->s){
  81.         push(p, d);
  82.         p->s= next;
  83.         return;
  84.     }
  85.     /* named constants */
  86.     for(i=0; p->const_name[i]; i++){
  87.         if(strmatch(p->s, p->const_name[i])){
  88.             push(p, p->const_value[i]);
  89.             p->s+= strlen(p->const_name[i]);
  90.             return;
  91.         }
  92.     }
  93.     p->s= strchr(p->s, '(');
  94.     if(p->s==NULL){
  95.         fprintf(stderr, "Parser: missing ( in "%s"n", next);
  96.         return;
  97.     }
  98.     p->s++; // "("
  99.     evalExpression(p);
  100.     d= pop(p);
  101.     if(p->s[0]== ','){
  102.         p->s++; // ","
  103.         evalExpression(p);
  104.         d2= pop(p);
  105.     }
  106.     if(p->s[0] != ')'){
  107.         fprintf(stderr, "Parser: missing ) in "%s"n", next);
  108.         return;
  109.     }
  110.     p->s++; // ")"
  111.          if( strmatch(next, "sinh"  ) ) d= sinh(d);
  112.     else if( strmatch(next, "cosh"  ) ) d= cosh(d);
  113.     else if( strmatch(next, "tanh"  ) ) d= tanh(d);
  114.     else if( strmatch(next, "sin"   ) ) d= sin(d);
  115.     else if( strmatch(next, "cos"   ) ) d= cos(d);
  116.     else if( strmatch(next, "tan"   ) ) d= tan(d);
  117.     else if( strmatch(next, "exp"   ) ) d= exp(d);
  118.     else if( strmatch(next, "log"   ) ) d= log(d);
  119.     else if( strmatch(next, "squish") ) d= 1/(1+exp(4*d));
  120.     else if( strmatch(next, "gauss" ) ) d= exp(-d*d/2)/sqrt(2*M_PI);
  121.     else if( strmatch(next, "abs"   ) ) d= fabs(d);
  122.     else if( strmatch(next, "max"   ) ) d= d > d2 ? d : d2;
  123.     else if( strmatch(next, "min"   ) ) d= d < d2 ? d : d2;
  124.     else if( strmatch(next, "gt"    ) ) d= d > d2 ? 1.0 : 0.0;
  125.     else if( strmatch(next, "gte"    ) ) d= d >= d2 ? 1.0 : 0.0;
  126.     else if( strmatch(next, "lt"    ) ) d= d > d2 ? 0.0 : 1.0;
  127.     else if( strmatch(next, "lte"    ) ) d= d >= d2 ? 0.0 : 1.0;
  128.     else if( strmatch(next, "eq"    ) ) d= d == d2 ? 1.0 : 0.0;
  129. //    else if( strmatch(next, "l1"    ) ) d= 1 + d2*(d - 1);
  130. //    else if( strmatch(next, "sq01"  ) ) d= (d >= 0.0 && d <=1.0) ? 1.0 : 0.0;
  131.     else{
  132.         int error=1;
  133.         for(i=0; p->func1_name && p->func1_name[i]; i++){
  134.             if(strmatch(next, p->func1_name[i])){
  135.                 d= p->func1[i](p->opaque, d);
  136.                 error=0;
  137.                 break;
  138.             }
  139.         }
  140.         for(i=0; p->func2_name && p->func2_name[i]; i++){
  141.             if(strmatch(next, p->func2_name[i])){
  142.                 d= p->func2[i](p->opaque, d, d2);
  143.                 error=0;
  144.                 break;
  145.             }
  146.         }
  147.         if(error){
  148.             fprintf(stderr, "Parser: unknown function in "%s"n", next);
  149.             return;
  150.         }
  151.     }
  152.     push(p, d);
  153. }
  154. static void evalPow(Parser *p){
  155.     int neg= 0;
  156.     if(p->s[0]=='+') p->s++;
  157.     if(p->s[0]=='-'){
  158.         neg= 1;
  159.         p->s++;
  160.     }
  161.     if(p->s[0]=='('){
  162.         p->s++;;
  163.         evalExpression(p);
  164.         if(p->s[0]!=')')
  165.             fprintf(stderr, "Parser: missing )n");
  166.         p->s++;
  167.     }else{
  168.         evalPrimary(p);
  169.     }
  170.     if(neg) push(p, -pop(p));
  171. }
  172. static void evalFactor(Parser *p){
  173.     evalPow(p);
  174.     while(p->s[0]=='^'){
  175.         double d;
  176.         p->s++;
  177.         evalPow(p);
  178.         d= pop(p);
  179.         push(p, pow(pop(p), d));
  180.     }
  181. }
  182. static void evalTerm(Parser *p){
  183.     evalFactor(p);
  184.     while(p->s[0]=='*' || p->s[0]=='/'){
  185.         int inv= p->s[0]=='/';
  186.         double d;
  187.         p->s++;
  188.         evalFactor(p);
  189.         d= pop(p);
  190.         if(inv) d= 1.0/d;
  191.         push(p, d * pop(p));
  192.     }
  193. }
  194. static void evalExpression(Parser *p){
  195.     evalTerm(p);
  196.     while(p->s[0]=='+' || p->s[0]=='-'){
  197.         int sign= p->s[0]=='-';
  198.         double d;
  199.         p->s++;
  200.         evalTerm(p);
  201.         d= pop(p);
  202.         if(sign) d= -d;
  203.         push(p, d + pop(p));
  204.     }
  205. }
  206. double x264_eval(char *s, double *const_value, const char **const_name,
  207.                  double (**func1)(void *, double), const char **func1_name,
  208.                  double (**func2)(void *, double, double), char **func2_name,
  209.                  void *opaque){
  210.     Parser p;
  211.     p.stack_index=0;
  212.     p.s= s;
  213.     p.const_value= const_value;
  214.     p.const_name = const_name;
  215.     p.func1      = func1;
  216.     p.func1_name = func1_name;
  217.     p.func2      = func2;
  218.     p.func2_name = func2_name;
  219.     p.opaque     = opaque;
  220.     evalExpression(&p);
  221.     return pop(&p);
  222. }