Skip to content
Snippets Groups Projects
ast_expr2.y 41.1 KiB
Newer Older
  • Learn to ignore specific revisions
  • %{
    /* Written by Pace Willisson (pace@blitz.com) 
     * and placed in the public domain.
     *
     * Largely rewritten by J.T. Conklin (jtc@wimsey.com)
     *
    
     * And then overhauled twice by Steve Murphy (murf@digium.com)
    
     * to add double-quoted strings, allow mult. spaces, improve
     * error messages, and then to fold in a flex scanner for the 
     * yylex operation.
     *
     * $FreeBSD: src/bin/expr/expr.y,v 1.16 2000/07/22 10:59:36 se Exp $
     */
    
    
    #include <sys/types.h>
    #include <stdio.h>
    
    #if !defined(STANDALONE) && !defined(STANDALONE2)	\
    	
    
    #ifndef __USE_ISOC99
    #define __USE_ISOC99 1
    #endif
    #endif
    
    #define FP___PRINTF "%.18Lg"
    #define FP___TYPE    long double
    
    #else
    #define FP___PRINTF "%.16g"
    #define FP___TYPE    double
    #endif
    
    #elif defined(HAVE_COS)
    
    #define FUNC_COS	(long double)cos
    
    #ifdef HAVE_SINL
    #define FUNC_SIN   sinl
    
    #elif defined(HAVE_SIN)
    
    #define FUNC_SIN	(long double)sin
    #endif
    
    #ifdef HAVE_TANL
    #define FUNC_TAN   tanl
    
    #elif defined(HAVE_TAN)
    
    #define FUNC_TAN	(long double)tan
    #endif
    
    #ifdef HAVE_ACOSL
    #define FUNC_ACOS   acosl
    
    #elif defined(HAVE_ACOS)
    
    #define FUNC_ACOS	(long double)acos
    #endif
    
    #ifdef HAVE_ASINL
    #define FUNC_ASIN   asinl
    
    #elif defined(HAVE_ASIN)
    
    #define FUNC_ASIN	(long double)asin
    #endif
    
    #ifdef HAVE_ATANL
    #define FUNC_ATAN   atanl
    
    #elif defined(HAVE_ATAN)
    
    #define FUNC_ATAN	(long double)atan
    #endif
    
    #ifdef HAVE_ATAN2L
    #define FUNC_ATAN2   atan2l
    
    #elif defined(HAVE_ATAN2)
    
    #define FUNC_ATAN2	(long double)atan2
    #endif
    
    #ifdef HAVE_POWL
    #define FUNC_POW   powl
    
    #elif defined(HAVE_POW)
    
    #define FUNC_POW	(long double)pow
    #endif
    
    #ifdef HAVE_SQRTL
    #define FUNC_SQRT   sqrtl
    
    #elif defined(HAVE_SQRT)
    
    #define FUNC_SQRT	(long double)sqrt
    #endif
    
    #ifdef HAVE_RINTL
    #define FUNC_RINT   rintl
    
    #elif defined(HAVE_RINT)
    
    #define FUNC_RINT	(long double)rint
    #endif
    
    #ifdef HAVE_EXPL
    #define FUNC_EXP   expl
    
    #elif defined(HAVE_EXP)
    
    #define FUNC_EXP	(long double)exp
    #endif
    
    #ifdef HAVE_LOGL
    #define FUNC_LOG   logl
    
    #elif defined(HAVE_LOG)
    
    #ifdef HAVE_REMAINDERL
    #define FUNC_REMAINDER   remainderl
    #elif defined(HAVE_REMAINDER)
    #define FUNC_REMAINDER	(long double)remainder
    
    #endif
    
    #ifdef HAVE_FMODL
    #define FUNC_FMOD   fmodl
    
    #elif defined(HAVE_FMOD)
    
    #define FUNC_FMOD	(long double)fmod
    #endif
    
    #ifdef HAVE_STRTOLD
    #define FUNC_STRTOD  strtold
    
    #elif defined(HAVE_STRTOD)
    
    #define FUNC_STRTOD  (long double)strtod
    #endif
    
    #ifdef HAVE_FLOORL
    
    #elif defined(HAVE_FLOOR)
    
    #define FUNC_FLOOR	(long double)floor
    
    #elif defined(HAVE_CEIL)
    
    #define FUNC_CEIL	(long double)ceil
    
    #elif defined(HAVE_ROUND)
    
    #define FUNC_ROUND     (long double)round
    
    #elif defined(HAVE_TRUNC)
    
    #define FUNC_TRUNC     (long double)trunc
    
    
    /*! \note
     * Oddly enough, some platforms have some ISO C99 functions, but not others, so
     * we define the missing functions in terms of their mathematical identities.
     */
    
    #elif (defined(HAVE_EXPL) && defined(HAVE_LOGL))
    
    #define	FUNC_EXP2(x)	expl((x) * logl(2.0))
    
    #elif (defined(HAVE_EXP) && defined(HAVE_LOG))
    
    #define	FUNC_EXP2(x)	(long double)exp((x) * log(2.0))
    
    #elif (defined(HAVE_EXPL) && defined(HAVE_LOGL))
    
    #define	FUNC_EXP10(x)	expl((x) * logl(10.0))
    
    #elif (defined(HAVE_EXP) && defined(HAVE_LOG))
    
    #define	FUNC_EXP10(x)	(long double)exp((x) * log(10.0))
    
    #elif defined(HAVE_LOGL)
    
    #define	FUNC_LOG2(x)	(logl(x) / logl(2.0))
    
    #elif defined(HAVE_LOG10L)
    #define	FUNC_LOG2(x)	(log10l(x) / log10l(2.0))
    #elif defined(HAVE_LOG2)
    #define FUNC_LOG2       (long double)log2
    #elif defined(HAVE_LOG)
    
    #define	FUNC_LOG2(x)	((long double)log(x) / log(2.0))
    
    #elif defined(HAVE_LOGL)
    
    #define	FUNC_LOG10(x)	(logl(x) / logl(10.0))
    
    #elif defined(HAVE_LOG2L)
    #define	FUNC_LOG10(x)	(log2l(x) / log2l(10.0))
    #elif defined(HAVE_LOG10)
    #define	FUNC_LOG10(x)	(long double)log10(x)
    #elif defined(HAVE_LOG)
    
    #define	FUNC_LOG10(x)	((long double)log(x) / log(10.0))
    
    #include <stdlib.h>
    
    #ifndef _GNU_SOURCE
    #define _GNU_SOURCE
    #endif
    
    #include <string.h>
    
    #include <math.h>
    
    #include <locale.h>
    
    #include <ctype.h>
    
    Kevin P. Fleming's avatar
    Kevin P. Fleming committed
    #if !defined(SOLARIS) && !defined(__CYGWIN__)
    
    #include <errno.h>
    #include <regex.h>
    #include <limits.h>
    
    
    #include "asterisk/ast_expr.h"
    #include "asterisk/logger.h"
    
    #if !defined(STANDALONE) && !defined(STANDALONE2)
    
    #if defined(LONG_LONG_MIN) && !defined(QUAD_MIN)
    
    #define QUAD_MIN LONG_LONG_MIN
    #endif
    
    #if defined(LONG_LONG_MAX) && !defined(QUAD_MAX)
    
    #define QUAD_MAX LONG_LONG_MAX
    #endif
    
    #  if ! defined(QUAD_MIN)
    
    #   define QUAD_MIN     (-0x7fffffffffffffffLL-1)
    
    #  endif
    #  if ! defined(QUAD_MAX)
    
    #define YYENABLE_NLS 0
    
    #define YYPARSE_PARAM parseio
    #define YYLEX_PARAM ((struct parse_io *)parseio)->scanner
    #define YYERROR_VERBOSE 1
    
    extern char extra_error_message[4095];
    extern int extra_error_message_supplied;
    
    	AST_EXPR_number, AST_EXPR_numeric_string, AST_EXPR_string
    
    #if defined(STANDALONE) || defined(STANDALONE2)
    
    void ast_log(int level, const char *file, int line, const char *function, const char *fmt, ...) __attribute__ ((format (printf,5,6)));
    #endif
    
    
    struct val {
    	enum valtype type;
    	union {
    		char *s;
    
    		FP___TYPE i; /* either long double, or just double, on a bad day */
    
    enum node_type {
    	AST_EXPR_NODE_COMMA, AST_EXPR_NODE_STRING, AST_EXPR_NODE_VAL
    } ;
    
    struct expr_node 
    {
    	enum node_type type;
    	struct val *val;
    	struct expr_node *left;
    	struct expr_node *right;
    };
    
    
    
    typedef void *yyscan_t;
    
    struct parse_io
    {
    	char *string;
    	struct val *val;
    	yyscan_t scanner;
    
    static int		chk_div __P((FP___TYPE, FP___TYPE));
    static int		chk_minus __P((FP___TYPE, FP___TYPE, FP___TYPE));
    static int		chk_plus __P((FP___TYPE, FP___TYPE, FP___TYPE));
    static int		chk_times __P((FP___TYPE, FP___TYPE, FP___TYPE));
    
    static void		free_value __P((struct val *));
    static int		is_zero_or_null __P((struct val *));
    static int		isstring __P((struct val *));
    
    static struct val	*make_number __P((FP___TYPE));
    
    static struct val	*make_str __P((const char *));
    static struct val	*op_and __P((struct val *, struct val *));
    static struct val	*op_colon __P((struct val *, struct val *));
    static struct val	*op_eqtilde __P((struct val *, struct val *));
    
    Steve Murphy's avatar
    Steve Murphy committed
    static struct val	*op_tildetilde __P((struct val *, struct val *));
    
    static struct val	*op_div __P((struct val *, struct val *));
    static struct val	*op_eq __P((struct val *, struct val *));
    static struct val	*op_ge __P((struct val *, struct val *));
    static struct val	*op_gt __P((struct val *, struct val *));
    static struct val	*op_le __P((struct val *, struct val *));
    static struct val	*op_lt __P((struct val *, struct val *));
    
    static struct val	*op_cond __P((struct val *, struct val *, struct val *));
    
    static struct val	*op_minus __P((struct val *, struct val *));
    static struct val	*op_negate __P((struct val *));
    static struct val	*op_compl __P((struct val *));
    static struct val	*op_ne __P((struct val *, struct val *));
    static struct val	*op_or __P((struct val *, struct val *));
    static struct val	*op_plus __P((struct val *, struct val *));
    static struct val	*op_rem __P((struct val *, struct val *));
    static struct val	*op_times __P((struct val *, struct val *));
    
    static struct val   *op_func(struct val *funcname, struct expr_node *arglist, struct ast_channel *chan);
    
    static int		to_number __P((struct val *));
    
    static void		to_string __P((struct val *));
    
    static struct expr_node *alloc_expr_node(enum node_type);
    static void destroy_arglist(struct expr_node *arglist);
    
    
    /* uh, if I want to predeclare yylex with a YYLTYPE, I have to predeclare the yyltype... sigh */
    typedef struct yyltype
    {
      int first_line;
      int first_column;
    
      int last_line;
      int last_column;
    } yyltype;
    
    # define YYLTYPE yyltype
    # define YYLTYPE_IS_TRIVIAL 1
    
    /* we will get warning about no prototype for yylex! But we can't
       define it here, we have no definition yet for YYSTYPE. */
    
    int		ast_yyerror(const char *,YYLTYPE *, struct parse_io *);
     
    /* I wanted to add args to the yyerror routine, so I could print out
       some useful info about the error. Not as easy as it looks, but it
       is possible. */
    #define ast_yyerror(x) ast_yyerror(x,&yyloc,parseio)
    
    #define DESTROY(x) {if((x)->type == AST_EXPR_numeric_string || (x)->type == AST_EXPR_string) free((x)->u.s); (x)->u.s = 0; free(x);}
    
    %}
     
    %pure-parser
    %locations
    /* %debug  for when you are having big problems */
    
    /* %name-prefix="ast_yy" */
    
    %union
    {
    	struct val *val;
    
    extern int		ast_yylex __P((YYSTYPE *, YYLTYPE *, yyscan_t));
    
    %left <val> TOK_OR
    %left <val> TOK_AND
    %left <val> TOK_EQ TOK_GT TOK_LT TOK_GE TOK_LE TOK_NE
    %left <val> TOK_PLUS TOK_MINUS
    
    %left <val> TOK_MULT TOK_DIV TOK_MOD 
    
    Steve Murphy's avatar
    Steve Murphy committed
    %left <val> TOK_COLON TOK_EQTILDE TOK_TILDETILDE
    
    %left <val> TOK_RP TOK_LP
    
    %token <val> TOKEN
    
    %type <val> start expr
    
    
    %destructor {  free_value($$); }  expr TOKEN TOK_COND TOK_COLONCOLON TOK_OR TOK_AND TOK_EQ 
                                     TOK_GT TOK_LT TOK_GE TOK_LE TOK_NE TOK_PLUS TOK_MINUS TOK_MULT TOK_DIV TOK_MOD TOK_COMPL TOK_COLON TOK_EQTILDE 
    
    Steve Murphy's avatar
    Steve Murphy committed
                                     TOK_RP TOK_LP TOK_TILDETILDE
    
    %%
    
    start: expr { ((struct parse_io *)parseio)->val = (struct val *)calloc(sizeof(struct val),1);
    
                  ((struct parse_io *)parseio)->val->type = $1->type;
    
                  if( $1->type == AST_EXPR_number )
    
    				  ((struct parse_io *)parseio)->val->u.i = $1->u.i;
    
    				  ((struct parse_io *)parseio)->val->u.s = $1->u.s; 
    			  free($1);
    			}
    
    	| {/* nothing */ ((struct parse_io *)parseio)->val = (struct val *)calloc(sizeof(struct val),1);
                  ((struct parse_io *)parseio)->val->type = AST_EXPR_string;
    			  ((struct parse_io *)parseio)->val->u.s = strdup(""); 
    			}
    
    
    arglist: expr { $$ = alloc_expr_node(AST_EXPR_NODE_VAL); $$->val = $1;}
    
           | arglist TOK_COMMA expr %prec TOK_RP {struct expr_node *x = alloc_expr_node(AST_EXPR_NODE_VAL);
    
                                     struct expr_node *t;
    								 DESTROY($2);
                                     for (t=$1;t->right;t=t->right)
    						         	  ;
                                     $$ = $1; t->right = x; x->val = $3;}
    
           | arglist TOK_COMMA %prec TOK_RP {struct expr_node *x = alloc_expr_node(AST_EXPR_NODE_VAL);
                                     struct expr_node *t;  /* NULL args should OK */
    								 DESTROY($2);
                                     for (t=$1;t->right;t=t->right)
    						         	  ;
                                     $$ = $1; t->right = x; x->val = make_str("");}
    
           ;
    
    expr: 
          TOKEN TOK_LP arglist TOK_RP { $$ = op_func($1,$3, ((struct parse_io *)parseio)->chan);
    		                            DESTROY($2);
    									DESTROY($4);
    									DESTROY($1);
    									destroy_arglist($3);
                                      }
        | TOKEN {$$ = $1;}
    	| TOK_LP expr TOK_RP { $$ = $2;
    
    	                       @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    
    						   @$.first_line=0; @$.last_line=0;
    							DESTROY($1); DESTROY($3); }
    
    	| expr TOK_OR expr { $$ = op_or ($1, $3);
    
                             @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						 @$.first_line=0; @$.last_line=0;}
    	| expr TOK_AND expr { $$ = op_and ($1, $3); 
    
    	                      @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
                              @$.first_line=0; @$.last_line=0;}
    
    	| expr TOK_EQ expr { $$ = op_eq ($1, $3);
    
    	                     @$.first_column = @1.first_column; @$.last_column = @3.last_column;
    
    						 @$.first_line=0; @$.last_line=0;}
    	| expr TOK_GT expr { $$ = op_gt ($1, $3);
    
                             @$.first_column = @1.first_column; @$.last_column = @3.last_column;
    
    						 @$.first_line=0; @$.last_line=0;}
    	| expr TOK_LT expr { $$ = op_lt ($1, $3); 
    
    	                     @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						 @$.first_line=0; @$.last_line=0;}
    	| expr TOK_GE expr  { $$ = op_ge ($1, $3); 
    
    	                      @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						  @$.first_line=0; @$.last_line=0;}
    	| expr TOK_LE expr  { $$ = op_le ($1, $3); 
    
    	                      @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						  @$.first_line=0; @$.last_line=0;}
    	| expr TOK_NE expr  { $$ = op_ne ($1, $3); 
    
    	                      @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						  @$.first_line=0; @$.last_line=0;}
    	| expr TOK_PLUS expr { $$ = op_plus ($1, $3); 
    
    	                       @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						   @$.first_line=0; @$.last_line=0;}
    	| expr TOK_MINUS expr { $$ = op_minus ($1, $3); 
    
    	                        @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    
    	| TOK_MINUS expr %prec TOK_COMPL { $$ = op_negate ($2); 
    
    	                        @$.first_column = @1.first_column; @$.last_column = @2.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    
    	                        @$.first_column = @1.first_column; @$.last_column = @2.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    	| expr TOK_MULT expr { $$ = op_times ($1, $3); 
    
    	                       @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						   @$.first_line=0; @$.last_line=0;}
    	| expr TOK_DIV expr { $$ = op_div ($1, $3); 
    
    	                      @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						  @$.first_line=0; @$.last_line=0;}
    	| expr TOK_MOD expr { $$ = op_rem ($1, $3); 
    
    	                      @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    						  @$.first_line=0; @$.last_line=0;}
    	| expr TOK_COLON expr { $$ = op_colon ($1, $3); 
    
    	                        @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    	| expr TOK_EQTILDE expr { $$ = op_eqtilde ($1, $3); 
    
    	                        @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    
    	| expr TOK_COND expr TOK_COLONCOLON expr  { $$ = op_cond ($1, $3, $5); 
    
    						DESTROY($2);	
    						DESTROY($4);	
    
    	                        @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    
    Steve Murphy's avatar
    Steve Murphy committed
    	| expr TOK_TILDETILDE expr { $$ = op_tildetilde ($1, $3); 
    						DESTROY($2);	
    	                        @$.first_column = @1.first_column; @$.last_column = @3.last_column; 
    							@$.first_line=0; @$.last_line=0;}
    
    static struct expr_node *alloc_expr_node(enum node_type nt)
    {
    	struct expr_node *x = calloc(1,sizeof(struct expr_node));
    	if (!x) {
    		ast_log(LOG_ERROR, "Allocation for expr_node FAILED!!\n");
    		return 0;
    	}
    	x->type = nt;
    	return x;
    }
    
    
    
    
    static struct val *
    
    make_number (FP___TYPE i)
    
    {
    	struct val *vp;
    
    	vp = (struct val *) malloc (sizeof (*vp));
    	if (vp == NULL) {
    		ast_log(LOG_WARNING, "malloc() failed\n");
    		return(NULL);
    	}
    
    
    	vp->type = AST_EXPR_number;
    
    	vp->u.i  = i;
    	return vp; 
    }
    
    static struct val *
    make_str (const char *s)
    {
    	struct val *vp;
    	size_t i;
    
    	int isint; /* this started out being a test for an integer, but then ended up being a test for a float */
    
    
    	vp = (struct val *) malloc (sizeof (*vp));
    	if (vp == NULL || ((vp->u.s = strdup (s)) == NULL)) {
    
    		if (vp) {
    			free(vp);
    		}
    
    		ast_log(LOG_WARNING,"malloc() failed\n");
    		return(NULL);
    	}
    
    
    	for (i = 0, isint = (isdigit(s[0]) || s[0] == '-' || s[0]=='.'); isint && i < strlen(s); i++)
    
    		if (!isdigit(s[i]) && s[i] != '.') {
    			isint = 0;
    			break;
    		}
    
    	}
    	if (isint)
    		vp->type = AST_EXPR_numeric_string;
    	else	
    		vp->type = AST_EXPR_string;
    
    	return vp;
    }
    
    
    static void
    free_value (struct val *vp)
    {	
    	if (vp==NULL) {
    		return;
    	}
    	if (vp->type == AST_EXPR_string || vp->type == AST_EXPR_numeric_string)
    		free (vp->u.s);	
    
    static int
    to_number (struct val *vp)
    
    	if (vp == NULL) {
    
    		ast_log(LOG_WARNING,"vp==NULL in to_number()\n");
    
    	if (vp->type == AST_EXPR_number)
    
    		return 1;
    
    	if (vp->type == AST_EXPR_string)
    		return 0;
    
    	/* vp->type == AST_EXPR_numeric_string, make it numeric */
    	errno = 0;
    
    	i  = FUNC_STRTOD(vp->u.s, (char**)0); /* either strtod, or strtold on a good day */
    
    	if (errno != 0) {
    
    		ast_log(LOG_WARNING,"Conversion of %s to number under/overflowed!\n", vp->u.s);
    
    		return(0);
    	}
    	free (vp->u.s);
    	vp->u.i = i;
    
    	vp->type = AST_EXPR_number;
    
    	return 1;
    }
    
    static void
    strip_quotes(struct val *vp)
    {
    	if (vp->type != AST_EXPR_string && vp->type != AST_EXPR_numeric_string)
    		return;
    	
    	if( vp->u.s[0] == '"' && vp->u.s[strlen(vp->u.s)-1] == '"' )
    	{
    		char *f, *t;
    		f = vp->u.s;
    		t = vp->u.s;
    		
    		while( *f )
    		{
    			if( *f  && *f != '"' )
    				*t++ = *f++;
    			else
    				f++;
    		}
    		*t = *f;
    	}
    }
    
    static void
    to_string (struct val *vp)
    {
    	char *tmp;
    
    	if (vp->type == AST_EXPR_string || vp->type == AST_EXPR_numeric_string)
    		return;
    
    	tmp = malloc ((size_t)25);
    	if (tmp == NULL) {
    		ast_log(LOG_WARNING,"malloc() failed\n");
    		return;
    	}
    
    
    	sprintf(tmp, FP___PRINTF, vp->u.i);
    
    	vp->type = AST_EXPR_string;
    	vp->u.s  = tmp;
    }
    
    
    static int
    isstring (struct val *vp)
    {
    
    	/* only TRUE if this string is not a valid number */
    
    	return (vp->type == AST_EXPR_string);
    }
    
    
    static int
    is_zero_or_null (struct val *vp)
    {
    
    	if (vp->type == AST_EXPR_number) {
    
    		return (vp->u.i == 0);
    	} else {
    
    		return (*vp->u.s == 0 || (to_number(vp) && vp->u.i == 0));
    
    #ifdef STANDALONE2
    
    void ast_log(int level, const char *file, int line, const char *function, const char *fmt, ...)
    {
    	va_list vars;
    	va_start(vars,fmt);
    	
            printf("LOG: lev:%d file:%s  line:%d func: %s  ",
                       level, file, line, function);
    	vprintf(fmt, vars);
    	fflush(stdout);
    	va_end(vars);
    }
    
    
    int main(int argc,char **argv) {
    	char s[4096];
    	char out[4096];
    	FILE *infile;
    	
    	if( !argv[1] )
    		exit(20);
    	
    	if( access(argv[1],F_OK)== 0 )
    	{
    		int ret;
    		
    		infile = fopen(argv[1],"r");
    		if( !infile )
    		{
    			printf("Sorry, couldn't open %s for reading!\n", argv[1]);
    			exit(10);
    		}
    		while( fgets(s,sizeof(s),infile) )
    		{
    			if( s[strlen(s)-1] == '\n' )
    				s[strlen(s)-1] = 0;
    			
    			ret = ast_expr(s, out, sizeof(out), NULL);
    			printf("Expression: %s    Result: [%d] '%s'\n",
    				   s, ret, out);
    		}
    		fclose(infile);
    	}
    	else
    	{
    		if (ast_expr(argv[1], s, sizeof(s), NULL))
    			printf("=====%s======\n",s);
    		else
    			printf("No result\n");
    	}
    	return 0;
    }
    
    #endif
    
    
    #undef ast_yyerror
    #define ast_yyerror(x) ast_yyerror(x, YYLTYPE *yylloc, struct parse_io *parseio)
    
    /* I put the ast_yyerror func in the flex input file,
       because it refers to the buffer state. Best to
       let it access the BUFFER stuff there and not trying
       define all the structs, macros etc. in this file! */
    
    
    static void destroy_arglist(struct expr_node *arglist)
    {
    	struct expr_node *arglist_next;
    	
    	while (arglist)
    	{
    		arglist_next = arglist->right;
    		if (arglist->val)
    			free_value(arglist->val);
    		arglist->val = 0;
    		arglist->right = 0;
    		free(arglist);
    		arglist = arglist_next;
    	}
    }
    
    
    #if !defined(STANDALONE) && !defined(STANDALONE2)
    
    static char *compose_func_args(struct expr_node *arglist)
    {
    	struct expr_node *t = arglist;
    	char *argbuf;
    	int total_len = 0;
    	
    	while (t) {
    		if (t != arglist)
    			total_len += 1; /* for the sep */
    		if (t->val) {
    			if (t->val->type == AST_EXPR_number)
    				total_len += 25; /* worst case */
    			else
    				total_len += strlen(t->val->u.s);
    		}
    		
    		t = t->right;
    	}
    	total_len++; /* for the null */
    	ast_log(LOG_NOTICE,"argbuf allocated %d bytes;\n", total_len);
    	argbuf = malloc(total_len);
    	argbuf[0] = 0;
    	t = arglist;
    	while (t) {
    		char numbuf[30];
    		
    		if (t != arglist)
    
    		
    		if (t->val) {
    			if (t->val->type == AST_EXPR_number) {
    				sprintf(numbuf,FP___PRINTF,t->val->u.i);
    				strcat(argbuf,numbuf);
    			} else
    				strcat(argbuf,t->val->u.s);
    		}
    		t = t->right;
    	}
    
    	ast_log(LOG_NOTICE,"argbuf uses %d bytes;\n", (int) strlen(argbuf));
    
    	return argbuf;
    }
    
    static int is_really_num(char *str)
    {
    	if ( strspn(str,"-0123456789. 	") == strlen(str))
    		return 1;
    	else
    		return 0;
    }
    
    
    static struct val *op_func(struct val *funcname, struct expr_node *arglist, struct ast_channel *chan)
    {
    	if (strspn(funcname->u.s,"ABCDEFGHIJKLMNOPQRSTUVWXYZ_0123456789") == strlen(funcname->u.s))
    	{
    		struct val *result;
    
    		if (0) {
    #ifdef FUNC_COS
    		} else if (strcmp(funcname->u.s,"COS") == 0) {
    
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_COS(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"SIN") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_SIN(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"TAN") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_TAN(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"ACOS") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_ACOS(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"ASIN") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_ASIN(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"ATAN") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_ATAN(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"ATAN2") == 0) {
    			if (arglist && arglist->right && !arglist->right->right && arglist->val && arglist->right->val){
    				to_number(arglist->val);
    				to_number(arglist->right->val);
    				result = make_number(FUNC_ATAN2(arglist->val->u.i, arglist->right->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"POW") == 0) {
    			if (arglist && arglist->right && !arglist->right->right && arglist->val && arglist->right->val){
    				to_number(arglist->val);
    				to_number(arglist->right->val);
    				result = make_number(FUNC_POW(arglist->val->u.i, arglist->right->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"SQRT") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_SQRT(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"FLOOR") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_FLOOR(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"CEIL") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_CEIL(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"ROUND") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_ROUND(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"RINT") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_RINT(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"TRUNC") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_TRUNC(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"EXP") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_EXP(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"EXP2") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_EXP2(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"EXP10") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_EXP10(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}
    
    		} else if (strcmp(funcname->u.s,"LOG") == 0) {
    			if (arglist && !arglist->right && arglist->val){
    				to_number(arglist->val);
    				result = make_number(FUNC_LOG(arglist->val->u.i));
    				return result;
    			} else {
    				ast_log(LOG_WARNING,"Wrong args to %s() function\n",funcname->u.s);
    				return make_number(0.0);
    			}