parser.c
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:63k
源码类别:

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * parser.c:
  3.  *****************************************************************************
  4.  * Copyright (C) 2004 VideoLAN
  5.  * $Id: parser.c 8205 2004-07-17 13:55:48Z asmax $
  6.  *
  7.  * Authors: Cyril Deguet <asmax@videolan.org>
  8.  *          code from projectM http://xmms-projectm.sourceforge.net
  9.  *
  10.  * This program is free software; you can redistribute it and/or modify
  11.  * it under the terms of the GNU General Public License as published by
  12.  * the Free Software Foundation; either version 2 of the License, or
  13.  * (at your option) any later version.
  14.  *
  15.  * This program is distributed in the hope that it will be useful,
  16.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  18.  * GNU General Public License for more details.
  19.  *
  20.  * You should have received a copy of the GNU General Public License
  21.  * along with this program; if not, write to the Free Software
  22.  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  23.  *****************************************************************************/
  24. /* parser.c */
  25. #include <stdio.h>
  26. #include <string.h>
  27. #include <stdlib.h>
  28. #include "common.h"
  29. #include "fatal.h"
  30. #include "splaytree_types.h"
  31. #include "splaytree.h"
  32. #include "tree_types.h"
  33. #include "expr_types.h"
  34. #include "eval.h"
  35. #include "param_types.h"
  36. #include "param.h"
  37. #include "func_types.h"
  38. #include "func.h"
  39. #include "preset_types.h"
  40. #include "builtin_funcs.h"
  41. #include "per_pixel_eqn_types.h"
  42. #include "per_pixel_eqn.h"
  43. #include "init_cond_types.h"
  44. #include "init_cond.h"
  45. #include "per_frame_eqn_types.h"
  46. #include "per_frame_eqn.h"
  47. #include "parser.h"
  48. #include "engine_vars.h"
  49. #include "custom_wave_types.h"
  50. #include "custom_wave.h"
  51. #include "custom_shape_types.h"
  52. #include "custom_shape.h"
  53. /* Strings that prefix (and denote the type of) equations */
  54. #define PER_FRAME_STRING "per_frame_"
  55. #define PER_FRAME_STRING_LENGTH 10
  56. #define PER_PIXEL_STRING "per_pixel_"
  57. #define PER_PIXEL_STRING_LENGTH 10
  58. #define PER_FRAME_INIT_STRING "per_frame_init_"
  59. #define PER_FRAME_INIT_STRING_LENGTH 15
  60. #define WAVECODE_STRING "wavecode_"
  61. #define WAVECODE_STRING_LENGTH 9
  62. #define WAVE_STRING "wave_"
  63. #define WAVE_STRING_LENGTH 5
  64. #define PER_POINT_STRING "per_point"
  65. #define PER_POINT_STRING_LENGTH 9
  66. #define PER_FRAME_STRING_NO_UNDERSCORE "per_frame"
  67. #define PER_FRAME_STRING_NO_UNDERSCORE_LENGTH 9
  68. #define SHAPECODE_STRING "shapecode_"
  69. #define SHAPECODE_STRING_LENGTH 10
  70. #define SHAPE_STRING "shape_"
  71. #define SHAPE_STRING_LENGTH 6
  72. #define SHAPE_INIT_STRING "init"
  73. #define SHAPE_INIT_STRING_LENGTH 4
  74. #define WAVE_INIT_STRING "init"
  75. #define WAVE_INIT_STRING_LENGTH 4
  76. /* Stores a line of a file as its being parsed */
  77. char string_line_buffer[STRING_LINE_SIZE]; 
  78. /* The current position of the string line buffer (see above) */
  79. int string_line_buffer_index = 0;
  80. /* All infix operators (except '=') are prototyped here */
  81. extern infix_op_t * infix_add, * infix_minus, * infix_div, * infix_mult,
  82.   * infix_or, * infix_and, * infix_mod, * infix_positive, * infix_negative;
  83. /* If the parser reads a line with a custom wave, this pointer is set to
  84.    the custom wave of concern */
  85. custom_wave_t * current_wave = NULL;
  86. custom_shape_t * current_shape = NULL;
  87. /* Counts the number of lines parsed */
  88. unsigned int line_count = 1;
  89. int per_frame_eqn_count  = 0;
  90. int per_frame_init_eqn_count = 0;
  91. typedef enum {
  92.   NORMAL_LINE_MODE,
  93.   PER_FRAME_LINE_MODE,
  94.   PER_PIXEL_LINE_MODE,
  95.   INIT_COND_LINE_MODE,
  96.   CUSTOM_WAVE_PER_POINT_LINE_MODE,
  97.   CUSTOM_WAVE_PER_FRAME_LINE_MODE,
  98.   CUSTOM_WAVE_WAVECODE_LINE_MODE,
  99.   CUSTOM_SHAPE_SHAPECODE_LINE_MODE,
  100. } line_mode_t;
  101. line_mode_t line_mode = NORMAL_LINE_MODE;
  102. /* Token enumeration type */
  103. typedef enum {
  104.   tEOL,   /* end of a line, usually a '/n' or '/r' */
  105.   tEOF,   /* end of file */
  106.   tLPr,   /* ( */
  107.   tRPr,   /* ) */
  108.   tLBr,   /* [ */
  109.   tRBr,   /* ] */
  110.   tEq,    /* = */
  111.   tPlus,  /* + */
  112.   tMinus, /* - */
  113.   tMult,  /* * */
  114.   tMod,   /* % */
  115.   tDiv,   /* / */
  116.   tOr,    /* | */
  117.   tAnd,   /* & */
  118.   tComma, /* , */
  119.   tPositive, /* + as a prefix operator */
  120.   tNegative, /* - as a prefix operator */
  121.   tSemiColon, /* ; */
  122.   tStringTooLong, /* special token to indicate an invalid string length */
  123.   tStringBufferFilled /* the string buffer for this line is maxed out */
  124. } token_t;
  125. int get_string_prefix_len(char * string);
  126. tree_expr_t * insert_gen_expr(gen_expr_t * gen_expr, tree_expr_t ** root);
  127. tree_expr_t * insert_infix_op(infix_op_t * infix_op, tree_expr_t ** root);
  128. token_t parseToken(FILE * fs, char * string);
  129. gen_expr_t ** parse_prefix_args(FILE * fs, int num_args, struct PRESET_T * preset);
  130. gen_expr_t * parse_infix_op(FILE * fs, token_t token, tree_expr_t * tree_expr, struct PRESET_T * preset);
  131. gen_expr_t * parse_sign_arg(FILE * fs);
  132. int parse_float(FILE * fs, double * float_ptr);
  133. int parse_int(FILE * fs, int * int_ptr);
  134. int insert_gen_rec(gen_expr_t * gen_expr, tree_expr_t * root);
  135. int insert_infix_rec(infix_op_t * infix_op, tree_expr_t * root);
  136. gen_expr_t * parse_gen_expr(FILE * fs, tree_expr_t * tree_expr, struct PRESET_T * preset);
  137. per_frame_eqn_t * parse_implicit_per_frame_eqn(FILE * fs, char * param_string, int index, struct PRESET_T * preset);
  138. init_cond_t * parse_per_frame_init_eqn(FILE * fs, struct PRESET_T * preset, splaytree_t * database);
  139. int parse_wavecode_prefix(char * token, int * id, char ** var_string);
  140. int parse_wavecode(char * token, FILE * fs, preset_t * preset);
  141. int parse_wave_prefix(char * token, int * id, char ** eqn_string);
  142. int parse_shapecode(char * eqn_string, FILE * fs, preset_t * preset);
  143. int parse_shapecode_prefix(char * token, int * id, char ** var_string);
  144. int parse_wave(char * eqn_string, FILE * fs, preset_t * preset);
  145. int parse_shape(char * eqn_string, FILE * fs, preset_t * preset);
  146. int parse_shape_prefix(char * token, int * id, char ** eqn_string);
  147. int update_string_buffer(char * buffer, int * index);
  148. int string_to_float(char * string, double * float_ptr);
  149. /* Grabs the next token from the file. The second argument points
  150.    to the raw string */
  151. token_t parseToken(FILE * fs, char * string) {
  152.   
  153.   char c;
  154.   int i;
  155.   
  156.   if (string != NULL)
  157.     memset(string, 0, MAX_TOKEN_SIZE);
  158.   
  159.   /* Loop until a delimiter is found, or the maximum string size is found */
  160.   for (i = 0; i < MAX_TOKEN_SIZE;i++) {
  161.     c = fgetc(fs);
  162.     
  163.     /* If the string line buffer is full, quit */
  164.     if (string_line_buffer_index == (STRING_LINE_SIZE - 1))
  165.       return tStringBufferFilled;
  166.     
  167.     /* Otherwise add this character to the string line buffer */
  168.     string_line_buffer[string_line_buffer_index++] = c;
  169.     /* Now interpret the character */
  170.     switch (c) {
  171.       
  172.     case '+':
  173.       return tPlus; 
  174.     case '-':
  175.       return tMinus;
  176.     case '%':
  177.       return tMod;
  178.     case '/':
  179.       
  180.       /* check for line comment here */
  181.       if ((c = fgetc(fs)) == '/') {
  182. while(1) {
  183.   c = fgetc(fs);
  184.   if (c == EOF) {
  185.     line_mode = NORMAL_LINE_MODE;
  186.     return tEOF;
  187.   }
  188.   if (c == 'n') {
  189.     line_mode = NORMAL_LINE_MODE;
  190.     return tEOL;
  191.   }
  192. }
  193.       }
  194.       
  195.       /* Otherwise, just a regular division operator */
  196.       ungetc(c, fs);
  197.       return tDiv;
  198.       
  199.     case '*':
  200.       return tMult;
  201.     case '|':
  202.       return tOr;
  203.     case '&':
  204.       return tAnd;
  205.     case '(': 
  206.       return tLPr;
  207.     case ')':
  208.       return tRPr;
  209.     case '[': 
  210.       return tLBr;
  211.     case ']':
  212.       return tRBr;
  213.     case '=': 
  214.       return tEq;
  215.       //    case 'r':
  216.       //break;
  217.     case 'n':
  218.       line_count++;
  219.       line_mode = NORMAL_LINE_MODE;
  220.       return tEOL;
  221.     case ',':
  222.       return tComma;
  223.     case ';':
  224.       return tSemiColon;
  225.     case ' ': /* space, skip the character */
  226.       i--;
  227.       break;
  228.     case EOF:
  229.       line_count = 1;
  230.       line_mode = NORMAL_LINE_MODE;
  231.       return tEOF;
  232.       
  233.     default: 
  234.       if (string != NULL)
  235. string[i] = c;
  236.     } 
  237.     
  238.   }
  239.   
  240.  /* String reached maximum length, return special token error */ 
  241.   return tStringTooLong;
  242.   
  243. }
  244. /* Parse input in the form of "exp, exp, exp, ...)" 
  245.    Returns a general expression list */
  246. gen_expr_t ** parse_prefix_args(FILE * fs, int num_args, struct PRESET_T * preset) {
  247.   int i, j;
  248.   gen_expr_t ** expr_list; /* List of arguments to function */
  249.   gen_expr_t * gen_expr;
  250.   
  251.   /* Malloc the expression list */
  252.   expr_list =  (gen_expr_t**)malloc(sizeof(gen_expr_t*)*num_args);
  253.   
  254.   /* Malloc failed */
  255.   if (expr_list == NULL)
  256.     return NULL;
  257.   
  258.   
  259.   i = 0;
  260.   while (i < num_args) {
  261.     //if (PARSE_DEBUG) printf("parse_prefix_args: parsing argument %d...n", i+1);
  262.     /* Parse the ith expression in the list */
  263.     if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  264.       //if (PARSE_DEBUG) printf("parse_prefix_args: failed to get parameter # %d for function (LINE %d)n", i+1, line_count);
  265.       for (j = 0; j < i; j++) 
  266. free_gen_expr(expr_list[j]);
  267.       free(expr_list);
  268.       return NULL;
  269.     }
  270.     /* Assign entry in expression list */
  271.     expr_list[i++] = gen_expr;
  272.   }
  273.   
  274.   //if (PARSE_DEBUG) printf("parse_prefix_args: finished parsing %d arguments (LINE %d)n", num_args, line_count);
  275.   /* Finally, return the resulting expression list */
  276.   return expr_list;
  277.   
  278. }
  279. /* Parses a comment at the top of the file. Stops when left bracket is found */
  280. int parse_top_comment(FILE * fs) {
  281.   char string[MAX_TOKEN_SIZE];
  282.   token_t token;
  283.   
  284.   /* Process tokens until left bracket is found */
  285.   while ((token = parseToken(fs, string)) != tLBr) {
  286.     if (token == tEOF) 
  287.       return PARSE_ERROR;
  288.   }
  289.  /* Done, return success */
  290.  return SUCCESS; 
  291. }
  292. /* Right Bracket is parsed by this function.
  293.    puts a new string into name */
  294. int parse_preset_name(FILE * fs, char * name) {
  295.   token_t token;
  296.   if (name == NULL)
  297. return FAILURE;
  298.   if ((token = parseToken(fs, name)) != tRBr)
  299.     return PARSE_ERROR;
  300.  
  301.   //if (PARSE_DEBUG) printf("parse_preset_name: parsed preset (name = "%s")n", name);
  302.   
  303.   return SUCCESS;
  304. }
  305. /* Parses per pixel equations */
  306. int parse_per_pixel_eqn(FILE * fs, preset_t * preset) {
  307.   char string[MAX_TOKEN_SIZE];
  308.   gen_expr_t * gen_expr;
  309.   if (PARSE_DEBUG) printf("parse_per_pixel: per_pixel equation parsing start...(LINE %d)n", line_count);
  310.   if (parseToken(fs, string) != tEq) { /* parse per pixel operator  name */
  311.     if (PARSE_DEBUG) printf("parse_per_pixel: equal operator expected after per pixel operator "%s", but not found (LINE %d)n", 
  312.          string, line_count);
  313.     return PARSE_ERROR;
  314.   }
  315.   
  316.   /* Parse right side of equation as an expression */
  317.   if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  318.     if (PARSE_DEBUG) printf("parse_per_pixel: equation evaluated to null? (LINE %d)n", line_count);
  319.     return PARSE_ERROR;
  320.   }
  321.   
  322.   /* Add the per pixel equation */
  323.   if (add_per_pixel_eqn(string, gen_expr, preset) < 0) {
  324.     free_gen_expr(gen_expr);
  325.     return PARSE_ERROR;
  326.   }
  327.   return SUCCESS;
  328. }
  329. /* Parses an equation line, this function is way too big, should add some helper functions */
  330. int parse_line(FILE * fs, struct PRESET_T * preset) {
  331.   char eqn_string[MAX_TOKEN_SIZE];
  332.   token_t token;
  333.   init_cond_t * init_cond;
  334.   per_frame_eqn_t * per_frame_eqn;
  335.   
  336.   /* Clear the string line buffer */
  337.   memset(string_line_buffer, 0, STRING_LINE_SIZE);
  338.   string_line_buffer_index = 0;
  339.   
  340.   
  341.   switch (token = parseToken(fs, eqn_string)) {
  342.     
  343.     /* Invalid Cases */
  344.   case tRBr:
  345.   case tLPr:
  346.   case tRPr:
  347.   case tComma:
  348.   case tLBr:
  349.   case tPlus:
  350.   case tMinus:
  351.   case tMod:
  352.   case tMult:
  353.   case tOr:
  354.   case tAnd:
  355.   case tDiv:
  356.     
  357.     //    if (PARSE_DEBUG) printf("parse_line: invalid token found at start of line (LINE %d)n", line_count);
  358.     /* Invalid token found, return a parse error */
  359.     return PARSE_ERROR;
  360.     
  361.     
  362.   case tEOL:  /* Empty line */
  363.     line_mode = NORMAL_LINE_MODE;
  364.     return SUCCESS;
  365.     
  366.   case tEOF: /* End of File */
  367.     line_mode = NORMAL_LINE_MODE;
  368.     line_count = 1;
  369.     return EOF;
  370.     
  371.   case tSemiColon: /* Indicates end of expression */
  372.     return SUCCESS;
  373.     
  374.     /* Valid Case, either an initial condition or equation should follow */
  375.   case tEq:
  376.     
  377.     /* CASE: PER FRAME INIT EQUATION */     
  378.     if (!strncmp(eqn_string, PER_FRAME_INIT_STRING, PER_FRAME_INIT_STRING_LENGTH)) {
  379.       
  380.       //if (PARSE_DEBUG) printf("parse_line: per frame init equation found...(LINE %d)n", line_count);
  381.       
  382.       /* Set the line mode to normal */
  383.       line_mode = NORMAL_LINE_MODE;
  384.       
  385.       /* Parse the per frame equation */
  386.       if ((init_cond = parse_per_frame_init_eqn(fs, preset, NULL)) == NULL) {
  387. //if (PARSE_DEBUG) printf("parse_line: per frame init equation parsing failed (LINE %d)n", line_count);
  388. return PARSE_ERROR;
  389.       }
  390.       
  391.       /* Insert the equation in the per frame equation tree */
  392.       if (splay_insert(init_cond, init_cond->param->name, preset->per_frame_init_eqn_tree) < 0) {
  393. //if (PARSE_DEBUG) printf("parse_line: failed to add a perframe equation (ERROR)n");
  394. free_init_cond(init_cond); /* will free the gen expr too */
  395. return ERROR;
  396.       }
  397.       
  398.      
  399.       if (update_string_buffer(preset->per_frame_init_eqn_string_buffer, 
  400.        &preset->per_frame_init_eqn_string_index) < 0)
  401. { return FAILURE;}
  402.       
  403.       return SUCCESS;
  404.       
  405.     }
  406.     /* Per frame equation case */     
  407.     if (!strncmp(eqn_string, PER_FRAME_STRING, PER_FRAME_STRING_LENGTH)) {
  408.       
  409.       /* Sometimes per frame equations are implicitly defined without the
  410.  per_frame_ prefix. This informs the parser that one could follow */
  411.       line_mode = PER_FRAME_LINE_MODE;
  412.       
  413.       //if (PARSE_DEBUG) printf("parse_line: per frame equation found...(LINE %d)n", line_count);
  414.       
  415.       /* Parse the per frame equation */
  416.       if ((per_frame_eqn = parse_per_frame_eqn(fs, ++per_frame_eqn_count, preset)) == NULL) {
  417. if (PARSE_DEBUG) printf("parse_line: per frame equation parsing failed (LINE %d)n", line_count);
  418. return PARSE_ERROR;
  419.       }
  420.       
  421.       /* Insert the equation in the per frame equation tree */
  422.       if (splay_insert(per_frame_eqn, &per_frame_eqn_count, preset->per_frame_eqn_tree) < 0) {
  423. if (PARSE_DEBUG) printf("parse_line: failed to add a perframe equation (ERROR)n");
  424. free_per_frame_eqn(per_frame_eqn); /* will free the gen expr too */
  425. return ERROR;
  426.       }
  427.     
  428.       if (update_string_buffer(preset->per_frame_eqn_string_buffer, 
  429.        &preset->per_frame_eqn_string_index) < 0)
  430. return FAILURE;
  431.       
  432.       
  433.       
  434.       return SUCCESS;
  435.       
  436.     }
  437.     
  438.     /* Wavecode initial condition case */
  439.     if (!strncmp(eqn_string, WAVECODE_STRING, WAVECODE_STRING_LENGTH)) {
  440.       
  441.           line_mode = CUSTOM_WAVE_WAVECODE_LINE_MODE;
  442.       //if (PARSE_DEBUG) 
  443.       //      printf("parse_line: wavecode prefix found: "%s"n", eqn_string);
  444.   //      printf("string:%dn", 5);
  445.   //SUPER MYSTERIO-BUG - Don't Remove
  446.   printf("");
  447.   
  448.       return parse_wavecode(eqn_string, fs, preset);
  449.     }
  450.     
  451.     /* Custom Wave Prefix */
  452.     if ((!strncmp(eqn_string, WAVE_STRING, WAVE_STRING_LENGTH)) && 
  453. ((eqn_string[5] >= 48) && (eqn_string[5] <= 57))) {
  454.       
  455.       //    if (PARSE_DEBUG) printf("parse_line wave prefix found: "%s"n", eqn_string);
  456.       
  457.       return parse_wave(eqn_string, fs, preset);
  458.       
  459.     }
  460.     
  461.     
  462.     /* Shapecode initial condition case */
  463.     if (!strncmp(eqn_string, SHAPECODE_STRING, SHAPECODE_STRING_LENGTH)) {
  464.       
  465.       line_mode = CUSTOM_SHAPE_SHAPECODE_LINE_MODE;
  466.       
  467.       if (PARSE_DEBUG) printf("parse_line: shapecode prefix found: "%s"n", eqn_string);
  468.       
  469.       return parse_shapecode(eqn_string, fs, preset);
  470.     }
  471.     
  472.     /* Custom Shape Prefix */
  473.     if ((!strncmp(eqn_string, SHAPE_STRING, SHAPE_STRING_LENGTH)) && 
  474. ((eqn_string[6] >= 48) && (eqn_string[6] <= 57))) {
  475.       
  476.       if (PARSE_DEBUG) printf("parse_line shape prefix found: "%s"n", eqn_string);
  477.       return parse_shape(eqn_string, fs, preset);
  478.       
  479.     }
  480.     
  481.     /* Per pixel equation case */
  482.     if (!strncmp(eqn_string, PER_PIXEL_STRING, PER_PIXEL_STRING_LENGTH)) {
  483.       line_mode = PER_PIXEL_LINE_MODE;
  484.       
  485.       if (parse_per_pixel_eqn(fs, preset) < 0)
  486. return PARSE_ERROR;
  487.       
  488.       
  489.       if (update_string_buffer(preset->per_pixel_eqn_string_buffer, 
  490.        &preset->per_pixel_eqn_string_index) < 0)
  491. return FAILURE;
  492.       
  493.       if (PARSE_DEBUG) printf("parse_line: finished parsing per pixel equation (LINE %d)n", line_count);
  494.       return SUCCESS;
  495.     } 
  496.     
  497.     /* Sometimes equations are written implicitly in milkdrop files, in the form
  498.        
  499.     per_frame_1 = p1 = eqn1; p2 = eqn2; p3 = eqn3;..; 
  500.     
  501.     which is analagous to:
  502.     
  503.     per_frame_1 = p1 = eqn1; per_frame_2 = p2 = eqn2; per_frame_3 = p3 = eqn3; ...;
  504.     
  505.     The following line mode hack allows such implicit declaration of the 
  506.     prefix that specifies the equation type. An alternative method
  507.     may be to associate each equation line as list of equations separated
  508.     by semicolons (and a new line ends the list). Instead, however, a global
  509.     variable called "line_mode" specifies the last type of equation found,
  510.     and bases any implicitly typed input on this fact
  511.     
  512.     Note added by Carmelo Piccione (cep@andrew.cmu.edu) 10/19/03
  513.     */
  514.     
  515.     /* Per frame line mode previously, try to parse the equation implicitly */
  516.     if (line_mode == PER_FRAME_LINE_MODE) {
  517.       if ((per_frame_eqn = parse_implicit_per_frame_eqn(fs, eqn_string, ++per_frame_eqn_count, preset)) == NULL)
  518. return PARSE_ERROR;
  519.       
  520.       /* Insert the equation in the per frame equation tree */
  521.       if (splay_insert(per_frame_eqn, &per_frame_eqn_count, preset->per_frame_eqn_tree) < 0) {
  522. if (PARSE_DEBUG) printf("parse_line: failed to add a perframe equation (ERROR)n");
  523. free_per_frame_eqn(per_frame_eqn); /* will free the gen expr too */
  524. return ERROR;
  525.       }
  526.       
  527.       
  528.       if (update_string_buffer(preset->per_frame_eqn_string_buffer, 
  529.        &preset->per_frame_eqn_string_index) < 0)
  530. return FAILURE;
  531.       
  532.       
  533.       
  534.       return SUCCESS;
  535.     }
  536.     
  537.     //if (PARSE_DEBUG) printf("parse_line: found initial condition: name = "%s" (LINE %d)n", eqn_string, line_count);
  538.     
  539.     /* Evaluate the initial condition */
  540.     if ((init_cond = parse_init_cond(fs, eqn_string, preset)) == NULL) {
  541.        if (PARSE_DEBUG) printf("parse_line: failed to parse initial condition (LINE %d)n", line_count);
  542.       return PARSE_ERROR; 
  543.     }
  544.     
  545.     /* Add equation to initial condition tree */
  546.     if (splay_insert(init_cond, init_cond->param->name, preset->init_cond_tree) < 0) {
  547.       if (PARSE_DEBUG) printf("parse_line: failed to add initial condition "%s" to equation tree (LINE %d)n", 
  548.              init_cond->param->name, line_count);
  549.       free_init_cond(init_cond);
  550.       return FAILURE;
  551.     }
  552.     
  553.     /* Finished with initial condition line */
  554.     //    if (PARSE_DEBUG) printf("parse_line: initial condition parsed successfullyn");
  555.     
  556.     return SUCCESS;
  557.     
  558.     /* END INITIAL CONDITIONING PARSING */
  559.     
  560.     
  561.   default: /* an uncaught type or an error has occurred */
  562.     if (PARSE_DEBUG) printf("parse_line: uncaught case, token val = %dn", token); 
  563.     return PARSE_ERROR;
  564.   }
  565.   
  566.   /* Because of the default in the case statement, 
  567.      control flow should never actually reach here */ 
  568.   return PARSE_ERROR;
  569. }
  570. /* Parses a general expression, this function is the meat of the parser */
  571. gen_expr_t * parse_gen_expr (FILE * fs, tree_expr_t * tree_expr, struct PRESET_T * preset) {
  572.   
  573.   int i;
  574.   char string[MAX_TOKEN_SIZE];
  575.   token_t token;
  576.   gen_expr_t * gen_expr;
  577.   double val;
  578.   param_t * param = NULL;
  579.   func_t * func;
  580.   gen_expr_t ** expr_list;
  581.   switch (token = parseToken(fs,string)) {
  582.   /* Left Parentice Case */
  583.   case tLPr:
  584.     
  585.     /* CASE 1 (Left Parentice): See if the previous string before this parentice is a function name */
  586.     if ((func = find_func(string)) != NULL) {
  587.         if (PARSE_DEBUG) printf("parse_gen_expr: found prefix function (name = %s) (LINE %d)n", func->name, line_count);
  588.       
  589.       /* Parse the functions arguments */
  590.       if ((expr_list = parse_prefix_args(fs, func->num_args, preset)) == NULL) {
  591. if (PARSE_DEBUG) printf("parse_prefix_args: failed to generate an expresion list! (LINE %d) n", line_count);
  592. free_tree_expr(tree_expr);
  593. return NULL;
  594.       }
  595.       
  596.       /* Convert function to expression */
  597.       if ((gen_expr = prefun_to_expr((void*)func->func_ptr, expr_list, func->num_args)) == NULL)  { 
  598.   if (PARSE_DEBUG) printf("parse_prefix_args: failed to convert prefix function to general expression (LINE %d) n", 
  599.    line_count);
  600. free_tree_expr(tree_expr);
  601. for (i = 0; i < func->num_args;i++)
  602.   free_gen_expr(expr_list[i]);
  603. free(expr_list);
  604. return NULL;
  605.       }
  606.     
  607.       
  608.       
  609.       token = parseToken(fs, string);
  610.       if (*string != 0) {
  611. if (PARSE_DEBUG) printf("parse_prefix_args: empty string expected, but not found...(LINE %d)n", line_count);
  612. /* continue anyway for now, could be implicit multiplication */
  613.       }
  614.       
  615.       return parse_infix_op(fs, token, insert_gen_expr(gen_expr, &tree_expr), preset);
  616.     }
  617.      
  618.     
  619.     /* Case 2: (Left Parentice), a string coupled with a left parentice. Either an error or implicit 
  620.        multiplication operator. For now treat it as an error */
  621.     if (*string != 0) {
  622.       if (PARSE_DEBUG) printf("parse_gen_expr: implicit multiplication case unimplemented!n");
  623.       free_tree_expr(tree_expr);
  624.       return NULL;
  625.     }
  626.     
  627.     /* CASE 3 (Left Parentice): the following is enclosed parentices to change order
  628.        of operations. So we create a new expression tree */
  629.     
  630.     if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  631.       //if (PARSE_DEBUG) printf("parse_gen_expr:  found left parentice, but failed to create new expression tree n");
  632.       free_tree_expr(tree_expr);
  633.       return NULL;
  634.     }
  635.     
  636.     if (PARSE_DEBUG) printf("parse_gen_expr: finished enclosed expression tree...n");
  637.     token = parseToken(fs, string);
  638.     return parse_infix_op(fs, token, insert_gen_expr(gen_expr, &tree_expr), preset);
  639.     /* Plus is a prefix operator check */
  640.   case tPlus:
  641.     if (*string == 0) {
  642.       
  643.       //if (PARSE_DEBUG) printf("parse_gen_expr: plus used as prefix (LINE %d)n", line_count);
  644.   /* Treat prefix plus as implict 0 preceding operator */
  645.       gen_expr = const_to_expr(0);
  646.       return parse_infix_op(fs, tPositive, insert_gen_expr(gen_expr, &tree_expr), preset);
  647.     }
  648.     
  649.     /* Minus is a prefix operator check */
  650.   case tMinus:
  651.     if (*string == 0) {
  652.      
  653.       /* Use the negative infix operator, but first add an implicit zero to the operator tree */
  654.       gen_expr = const_to_expr(0);
  655.       //return parse_gen_expr(fs, insert_gen_expr(gen_expr, &tree_expr), preset);
  656. return parse_infix_op(fs, tNegative, insert_gen_expr(gen_expr, &tree_expr), preset);
  657.     }
  658.     
  659.     /* All the following cases are strings followed by an infix operator or terminal */
  660.   case tRPr:
  661.   case tEOL: 
  662.   case tEOF:
  663.   case tSemiColon:
  664.   case tComma:
  665.     
  666.     /* CASE 1 (terminal): string is empty, but not null. Not sure if this will actually happen
  667.        any more. */
  668.     if (*string == 0) {
  669.       //if (PARSE_DEBUG) printf("parse_gen_expr: empty string coupled with terminal (LINE %d) n", line_count);
  670.       return parse_infix_op(fs, token, tree_expr, preset);
  671.       
  672.     }
  673.     
  674.   default:  
  675.     /* CASE 0: Empty string, parse error */
  676.     if (*string == 0) {
  677.       if (PARSE_DEBUG) printf("parse_gen_expr: empty string coupled with infix op (ERROR!) (LINE %d) n", line_count);
  678.       free_tree_expr(tree_expr);
  679.       return NULL;
  680.     }
  681.     /* CASE 1: Check if string is a just a floating point number */
  682.     if (string_to_float(string, &val) != PARSE_ERROR) {
  683.       if ((gen_expr = const_to_expr(val)) == NULL) {
  684. free_tree_expr(tree_expr);
  685. return NULL;
  686.       }
  687.       
  688.       /* Parse the rest of the line */
  689.       return parse_infix_op(fs, token, insert_gen_expr(gen_expr, &tree_expr), preset);          
  690.     
  691.     }
  692.       
  693.     /* CASE 4: custom shape variable */
  694.     if (current_shape != NULL) {
  695.       if ((param = find_param_db(string, current_shape->param_tree, FALSE)) == NULL) {
  696. if ((param = find_builtin_param(string)) == NULL)
  697.   if ((param = find_param_db(string, current_shape->param_tree, TRUE)) == NULL) {
  698.     free_tree_expr(tree_expr);
  699.     return NULL;
  700.   }
  701.       }
  702.       
  703.       if (PARSE_DEBUG) {
  704. printf("parse_gen_expr: custom shape parameter (name = %s)... ", param->name);
  705. fflush(stdout);
  706.       }  
  707.       
  708.       /* Convert parameter to an expression */
  709.       if ((gen_expr = param_to_expr(param)) == NULL) {
  710. free_tree_expr(tree_expr);
  711. return NULL;
  712.       }
  713.       
  714.       //if (PARSE_DEBUG) printf("converted to expression (LINE %d)n", line_count);
  715.       
  716.       /* Parse the rest of the line */
  717.       return parse_infix_op(fs, token, insert_gen_expr(gen_expr, &tree_expr), preset);
  718.     }
  719.     
  720.     /* CASE 5: custom wave variable */
  721.     if (current_wave != NULL) {
  722.       if ((param = find_param_db(string, current_wave->param_tree, FALSE)) == NULL) {
  723. if ((param = find_builtin_param(string)) == NULL) 
  724.   if ((param = find_param_db(string, current_wave->param_tree, TRUE)) == NULL) {
  725.     free_tree_expr(tree_expr);
  726.     return NULL;
  727.   }
  728.         
  729.       }
  730.       if (PARSE_DEBUG) {
  731. printf("parse_gen_expr: custom wave parameter (name = %s)... ", param->name);
  732. fflush(stdout);
  733.       }
  734. /* Convert parameter to an expression */
  735. if ((gen_expr = param_to_expr(param)) == NULL) {
  736.   free_tree_expr(tree_expr);
  737.   return NULL;
  738. }
  739. if (PARSE_DEBUG) printf("converted to expression (LINE %d)n", line_count);
  740. /* Parse the rest of the line */
  741. return parse_infix_op(fs, token, insert_gen_expr(gen_expr, &tree_expr), preset);
  742.       
  743.     }
  744.     /* CASE 6: regular parameter. Will be created if necessary and the string has no invalid characters */
  745.     if ((param = find_param(string, preset, P_CREATE)) != NULL) {
  746.       
  747.       if (PARSE_DEBUG) {
  748. printf("parse_gen_expr: parameter (name = %s)... ", param->name);
  749. fflush(stdout);
  750.       }  
  751.     
  752. /* Convert parameter to an expression */
  753.       if ((gen_expr = param_to_expr(param)) == NULL) {
  754. free_tree_expr(tree_expr);
  755. return NULL;
  756.       }
  757.       
  758.       if (PARSE_DEBUG) printf("converted to expression (LINE %d)n", line_count);
  759.       
  760.       /* Parse the rest of the line */
  761.       return parse_infix_op(fs, token, insert_gen_expr(gen_expr, &tree_expr), preset);
  762.           
  763.     }
  764.    
  765.     /* CASE 7: Bad string, give up */
  766.     if (PARSE_DEBUG) printf("parse_gen_expr: syntax error [string = "%s"] (LINE %d)n", string, line_count);
  767.     free_tree_expr(tree_expr);
  768.     return NULL;
  769.   }
  770. }
  771.   
  772. /* Inserts expressions into tree according to operator precedence.
  773.    If root is null, a new tree is created, with gen_expr as only element */
  774. tree_expr_t * insert_infix_op(infix_op_t * infix_op, tree_expr_t **root) {
  775.   tree_expr_t * new_root;
  776.   
  777.   /* Sanity check */
  778.   if (infix_op == NULL)
  779.     return NULL;
  780.   
  781.   /* The root is null, so make this operator
  782.      the new root */
  783.   
  784.   if (*root == NULL) {
  785.     new_root = new_tree_expr(infix_op, NULL, NULL, NULL);
  786.     *root = new_root;
  787.     return new_root;
  788.   }
  789.   
  790.   /* The root node is not an infix function,
  791.      so we make this infix operator the new root  */ 
  792.   
  793.   if ((*root)->infix_op == NULL) {
  794.     new_root = new_tree_expr(infix_op, NULL, *root, NULL);
  795.     (*root) = new_root;
  796.     return new_root;
  797.   }
  798.   
  799.   /* The root is an infix function. If the precedence
  800.      of the item to be inserted is greater than the root's
  801.      precedence, then make gen_expr the root */
  802.   
  803.   if (infix_op->precedence > (*root)->infix_op->precedence) {
  804.     new_root = new_tree_expr(infix_op, NULL, *root, NULL);
  805.     (*root) = new_root;
  806.       return new_root;
  807.   }
  808.   
  809.   /* If control flow reaches here, use a recursive helper
  810.      with the knowledge that the root is higher precedence
  811.      than the item to be inserted */
  812.   
  813.   insert_infix_rec(infix_op, *root);
  814.   return *root;
  815.   
  816. }
  817. tree_expr_t * insert_gen_expr(gen_expr_t * gen_expr, tree_expr_t ** root) {
  818.   tree_expr_t * new_root;
  819.   
  820.   /* If someone foolishly passes a null
  821.      pointer to insert, return the original tree */
  822.   
  823.   if (gen_expr == NULL) {
  824.     return *root;
  825.   }
  826.   /* If the root is null, generate a new expression tree,
  827.      using the passed expression as the root element */
  828.   
  829.   if (*root == NULL) {
  830.     new_root = new_tree_expr(NULL, gen_expr, NULL, NULL);
  831.     *root = new_root;
  832.     return new_root;
  833.   }
  834.   
  835.   
  836.   /* Otherwise. the new element definitely will not replace the current root.
  837.      Use a recursive helper function to do insertion */
  838.   insert_gen_rec(gen_expr, *root);
  839.   return *root;
  840. }
  841. /* A recursive helper function to insert general expression elements into the operator tree */
  842. int insert_gen_rec(gen_expr_t * gen_expr, tree_expr_t * root) {
  843.   
  844.   /* Trivial Case: root is null */
  845.   
  846.   if (root == NULL) {
  847.     ////if (PARSE_DEBUG) printf("insert_gen_rec: root is null, returning failuren");
  848.     return FAILURE;
  849.   }
  850.   
  851.   
  852.   /* The current node's left pointer is null, and this
  853.      current node is an infix operator, so insert the
  854.      general expression at the left pointer */
  855.   
  856.   if ((root->left == NULL) && (root->infix_op != NULL)) {
  857.     root->left = new_tree_expr(NULL, gen_expr, NULL, NULL);
  858.     return SUCCESS;
  859.   }
  860.   
  861.   /* The current node's right pointer is null, and this
  862.      current node is an infix operator, so insert the
  863.      general expression at the right pointer */
  864.   
  865.   if ((root->right == NULL) && (root->infix_op != NULL)) {
  866.     root->right = new_tree_expr(NULL, gen_expr, NULL, NULL);
  867.     return SUCCESS;
  868.   }
  869.   
  870.   /* Otherwise recurse down to the left. If
  871.      this succeeds then return. If it fails, try
  872.      recursing down to the right */
  873.   
  874.   if (insert_gen_rec(gen_expr, root->left) == FAILURE) 
  875.     return insert_gen_rec(gen_expr, root->right);
  876.   /* Impossible for control flow to reach here, but in
  877.      the world of C programming, who knows... */
  878.   //if (PARSE_DEBUG) printf("insert_gen_rec: should never reach here!n");  
  879.   return FAILURE;
  880. }
  881. /* A recursive helper function to insert infix arguments by operator precedence */
  882. int insert_infix_rec(infix_op_t * infix_op, tree_expr_t * root) {
  883.   /* Shouldn't happen, implies a parse error */
  884.   if (root == NULL)
  885.     return FAILURE;
  886.   
  887.   /* Also shouldn't happen, also implies a (different) parse error */
  888.   if (root->infix_op == NULL)
  889.     return FAILURE;
  890.   /* Left tree is empty, attach this operator to it. 
  891.      I don't think this will ever happen */
  892.   if (root->left == NULL) {
  893.     root->left = new_tree_expr(infix_op, NULL, root->left, NULL);
  894.     return SUCCESS;
  895.   }
  896.  
  897.   /* Right tree is empty, attach this operator to it */
  898.   if (root->right == NULL) {
  899.     root->right = new_tree_expr(infix_op, NULL, root->right, NULL);
  900.     return SUCCESS;
  901.   }
  902.   /* The left element can now be ignored, since there is no way for this
  903.      operator to use those expressions */
  904.   /* If the right element is not an infix operator,
  905.      then insert the expression here, attaching the old right branch
  906.      to the left of the new expression */
  907.   if (root->right->infix_op == NULL) {
  908.     root->right = new_tree_expr(infix_op, NULL, root->right, NULL);
  909.     return SUCCESS;
  910.   }
  911.   
  912.   /* Traverse deeper if the inserting operator precedence is less than the
  913.      the root's right operator precedence */
  914.   if (infix_op->precedence < root->right->infix_op->precedence) 
  915.     return insert_infix_rec(infix_op, root->right);
  916.   /* Otherwise, insert the operator here */
  917.   
  918.   root->right = new_tree_expr(infix_op, NULL, root->right, NULL);
  919.   return SUCCESS;
  920. }
  921. /* Parses an infix operator */
  922. gen_expr_t * parse_infix_op(FILE * fs, token_t token, tree_expr_t * tree_expr, struct PRESET_T * preset) {
  923.   gen_expr_t * gen_expr;
  924.   switch (token) {
  925.   /* All the infix operators */
  926.   case tPlus:
  927.     //if (PARSE_DEBUG) printf("parse_infix_op: found addition operator (LINE %d)n", line_count);
  928.     return parse_gen_expr(fs, insert_infix_op(infix_add, &tree_expr), preset);
  929.   case tMinus:
  930.     //if (PARSE_DEBUG) printf("parse_infix_op: found subtraction operator (LINE %d)n", line_count);
  931.     return parse_gen_expr(fs, insert_infix_op(infix_minus, &tree_expr), preset);
  932.   case tMult:
  933.     //if (PARSE_DEBUG) printf("parse_infix_op: found multiplication operator (LINE %d)n", line_count);
  934.     return parse_gen_expr(fs, insert_infix_op(infix_mult, &tree_expr), preset);
  935.   case tDiv:
  936.     //if (PARSE_DEBUG) printf("parse_infix_op: found division operator (LINE %d)n", line_count);  
  937.     return parse_gen_expr(fs, insert_infix_op(infix_div, &tree_expr), preset);
  938.   case tMod:
  939.     //if (PARSE_DEBUG) printf("parse_infix_op: found modulo operator (LINE %d)n", line_count);  
  940.     return parse_gen_expr(fs, insert_infix_op(infix_mod, &tree_expr), preset);
  941.   case tOr:  
  942.     //if (PARSE_DEBUG) printf("parse_infix_op: found bitwise or operator (LINE %d)n", line_count);     
  943.     return parse_gen_expr(fs, insert_infix_op(infix_or, &tree_expr), preset);
  944.   case tAnd:    
  945.     //if (PARSE_DEBUG) printf("parse_infix_op: found bitwise and operator (LINE %d)n", line_count);     
  946.     return parse_gen_expr(fs, insert_infix_op(infix_and, &tree_expr), preset);
  947.   case tPositive:
  948.     //if (PARSE_DEBUG) printf("parse_infix_op: found positive operator (LINE %d)n", line_count);     
  949.     return parse_gen_expr(fs, insert_infix_op(infix_positive, &tree_expr), preset);
  950.   case tNegative:
  951.     //if (PARSE_DEBUG) printf("parse_infix_op: found negative operator (LINE %d)n", line_count);     
  952.     return parse_gen_expr(fs, insert_infix_op(infix_negative, &tree_expr), preset);
  953.   case tEOL:
  954.   case tEOF:
  955.   case tSemiColon:
  956.   case tRPr:
  957.   case tComma:   
  958. //if (PARSE_DEBUG) printf("parse_infix_op: terminal found (LINE %d)n", line_count);
  959.    gen_expr = new_gen_expr(TREE_T, (void*)tree_expr);
  960.    return gen_expr;
  961.   default:
  962.     //if (PARSE_DEBUG) printf("parse_infix_op: operator or terminal expected, but not found (LINE %d)n", line_count);
  963.     free_tree_expr(tree_expr);
  964.     return NULL;
  965.   }  
  966.   /* Will never happen */
  967.   return NULL;
  968.   
  969. }
  970. /* Parses an integer, checks for +/- prefix */
  971. int parse_int(FILE * fs, int * int_ptr) {
  972. char string[MAX_TOKEN_SIZE];
  973.   token_t token;
  974.   int sign;
  975.   char * end_ptr = " ";
  976.   token = parseToken(fs, string);
  977.  
  978.   switch (token) {
  979.   case tMinus:
  980.     sign = -1;
  981.     token = parseToken(fs, string); 
  982.     break;
  983.   case tPlus:
  984.     sign = 1;
  985.     token = parseToken(fs, string);
  986.     break;
  987.   default: 
  988.     sign = 1;
  989.     break;
  990.   }
  991.  
  992.   if (string[0] == 0) 
  993.     return PARSE_ERROR;
  994.   
  995.   /* Convert the string to an integer. *end_ptr
  996.      should end up pointing to null terminator of 'string' 
  997.      if the conversion was successful. */
  998.   //  printf("STRING: "%s"n", string);
  999.   (*int_ptr) = sign*strtol(string, &end_ptr, 10);
  1000.   /* If end pointer is a return character or null terminator, all is well */
  1001.   if ((*end_ptr == 'r') || (*end_ptr == '')) 
  1002.     return SUCCESS;
  1003.     return PARSE_ERROR;
  1004.   
  1005. }
  1006. /* Parses a floating point number */
  1007. int string_to_float(char * string, double * float_ptr) {
  1008.   char ** error_ptr;
  1009.   if (*string == 0)
  1010.     return PARSE_ERROR;
  1011.   error_ptr = malloc(sizeof(char**));
  1012.   
  1013.   (*float_ptr) = strtod(string, error_ptr);
  1014.  
  1015.   /* These imply a succesful parse of the string */
  1016.   if ((**error_ptr == '') || (**error_ptr == 'r')) {
  1017.     free(error_ptr);
  1018.     return SUCCESS;
  1019.   }
  1020.     
  1021.   (*float_ptr) = 0;
  1022.   free(error_ptr);
  1023.   return PARSE_ERROR;  
  1024. }
  1025. /* Parses a floating point number */
  1026. int parse_float(FILE * fs, double * float_ptr) {
  1027.   char string[MAX_TOKEN_SIZE];
  1028.   char ** error_ptr;
  1029.   token_t token;
  1030.   int sign;
  1031.   
  1032.   error_ptr = malloc(sizeof(char**));
  1033.   token = parseToken(fs, string);
  1034.   switch (token) {
  1035.   case tMinus:
  1036.   sign = -1;
  1037.   token = parseToken(fs, string); 
  1038.   break;
  1039.   case tPlus:
  1040.   sign = 1;
  1041.   token = parseToken(fs, string);
  1042.   break;
  1043.   default: 
  1044.     sign = 1;  
  1045.   }
  1046.  
  1047.   if (string[0] == 0) {
  1048.     free(error_ptr);
  1049.     return PARSE_ERROR;
  1050.   }
  1051.   (*float_ptr) = sign*strtod(string, error_ptr);
  1052.  
  1053.   /* No conversion was performed */
  1054.   if ((**error_ptr == '') || (**error_ptr == 'r')) {
  1055.     free(error_ptr);
  1056.     return SUCCESS;
  1057.   }
  1058.     
  1059.   //if (PARSE_DEBUG) printf("parse_float: double conversion failed for string "%s"n", string);
  1060.   (*float_ptr) = 0;
  1061.   free(error_ptr);
  1062.   return PARSE_ERROR;
  1063.   
  1064.   
  1065. }
  1066. /* Parses a per frame equation. That is, interprets a stream of data as a per frame equation */
  1067. per_frame_eqn_t * parse_per_frame_eqn(FILE * fs, int index, struct PRESET_T * preset) {
  1068.   
  1069.   char string[MAX_TOKEN_SIZE];
  1070.   param_t * param;
  1071.   per_frame_eqn_t * per_frame_eqn;
  1072.   gen_expr_t * gen_expr;
  1073.   
  1074.   if (parseToken(fs, string) != tEq) {
  1075.     //if (PARSE_DEBUG) printf("parse_per_frame_eqn: no equal sign after string "%s" (LINE %d)n", string, line_count);
  1076.     return NULL;
  1077.   }
  1078.   
  1079.   /* Find the parameter associated with the string, create one if necessary */
  1080.   if ((param = find_param(string, preset, P_CREATE)) == NULL) { 
  1081.     return NULL;
  1082.   }
  1083.   
  1084.   /* Make sure parameter is writable */
  1085.   if (param->flags & P_FLAG_READONLY) {
  1086.       //if (PARSE_DEBUG) printf("parse_per_frame_eqn: parameter %s is marked as read only (LINE %d)n", param->name, line_count);  
  1087.       return NULL;
  1088.   }
  1089.   
  1090.   /* Parse right side of equation as an expression */
  1091.   if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  1092.     //if (PARSE_DEBUG) printf("parse_per_frame_eqn: equation evaluated to null (LINE %d)n", line_count);
  1093.     return NULL;
  1094.   }
  1095.   
  1096.   //if (PARSE_DEBUG) printf("parse_per_frame_eqn: finished per frame equation evaluation (LINE %d)n", line_count);
  1097.   
  1098.   /* Create a new per frame equation */
  1099.   if ((per_frame_eqn = new_per_frame_eqn(index, param, gen_expr)) == NULL) {
  1100.     //if (PARSE_DEBUG) printf("parse_per_frame_eqn: failed to create a new per frame eqn, out of memory?n");
  1101.     free_gen_expr(gen_expr);
  1102.     return NULL;
  1103.   }
  1104.   
  1105.   //if (PARSE_DEBUG) printf("parse_per_frame_eqn: per_frame eqn parsed succesfullyn");
  1106.   
  1107.   return per_frame_eqn;
  1108. }
  1109. /* Parses an 'implicit' per frame equation. That is, interprets a stream of data as a per frame equation without a prefix */
  1110. per_frame_eqn_t * parse_implicit_per_frame_eqn(FILE * fs, char * param_string, int index, struct PRESET_T * preset) {
  1111.   
  1112.   param_t * param;
  1113.   per_frame_eqn_t * per_frame_eqn;
  1114.   gen_expr_t * gen_expr;
  1115.   
  1116.   if (fs == NULL)
  1117.     return NULL;
  1118.   if (param_string == NULL)
  1119.     return NULL;
  1120.   if (preset == NULL)
  1121.     return NULL;
  1122.   //rintf("param string: %sn", param_string);
  1123.   /* Find the parameter associated with the string, create one if necessary */
  1124.   if ((param = find_param(param_string, preset, P_CREATE)) == NULL) { 
  1125.     return NULL;
  1126.   }
  1127.   
  1128.   //printf("parse_implicit_per_frame_eqn: param is %sn", param->name);
  1129.   /* Make sure parameter is writable */
  1130.   if (param->flags & P_FLAG_READONLY) {
  1131.     //if (PARSE_DEBUG) printf("parse_implicit_per_frame_eqn: parameter %s is marked as read only (LINE %d)n", param->name, line_count);  
  1132.     return NULL;
  1133.   }
  1134.   
  1135.   /* Parse right side of equation as an expression */
  1136.   if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  1137.     //if (PARSE_DEBUG) printf("parse_implicit_per_frame_eqn: equation evaluated to null (LINE %d)n", line_count);
  1138.     return NULL;
  1139.   }
  1140.   
  1141.   //if (PARSE_DEBUG) printf("parse_implicit_per_frame_eqn: finished per frame equation evaluation (LINE %d)n", line_count);
  1142.   
  1143.   /* Create a new per frame equation */
  1144.   if ((per_frame_eqn = new_per_frame_eqn(index, param, gen_expr)) == NULL) {
  1145.     //if (PARSE_DEBUG) printf("parse_implicit_per_frame_eqn: failed to create a new per frame eqn, out of memory?n");
  1146.     free_gen_expr(gen_expr);
  1147.     return NULL;
  1148.   }
  1149.   
  1150.   //if (PARSE_DEBUG) printf("parse_implicit_per_frame_eqn: per_frame eqn parsed succesfullyn");
  1151.   
  1152.   return per_frame_eqn;
  1153. }
  1154. /* Parses an initial condition */
  1155. init_cond_t * parse_init_cond(FILE * fs, char * name, struct PRESET_T * preset) {
  1156.   param_t * param;
  1157.   value_t init_val;
  1158.   init_cond_t * init_cond;
  1159.   if (name == NULL)
  1160.     return NULL;
  1161.   if (preset == NULL)
  1162.     return NULL;
  1163.   
  1164.   /* Search for the paramater in the database, creating it if necessary */
  1165.   if ((param = find_param(name, preset, P_CREATE)) == NULL) {
  1166.     return NULL;
  1167.   }
  1168.   
  1169.   //if (PARSE_DEBUG) printf("parse_init_cond: parameter = "%s" (LINE %d)n", param->name, line_count);
  1170.   
  1171.   if (param->flags & P_FLAG_READONLY) {
  1172.     //if (PARSE_DEBUG) printf("parse_init_cond: builtin parameter "%s" marked as read only!n", param->name);
  1173.     return NULL;
  1174.   }
  1175.   
  1176.   /* At this point, a parameter has been created or was found
  1177.      in the database. */
  1178.   
  1179.   //if (PARSE_DEBUG) printf("parse_init_cond: parsing initial condition value... (LINE %d)n", line_count);
  1180.   
  1181.   /* integer value (boolean is an integer in C) */
  1182.   if ((param->type == P_TYPE_INT) || (param->type == P_TYPE_BOOL)) {
  1183.     if ((parse_int(fs, (int*)&init_val.int_val)) == PARSE_ERROR) {
  1184.       //if (PARSE_DEBUG) printf("parse_init_cond: error parsing integer!n");
  1185.       return NULL;
  1186.     }
  1187.   }
  1188.   
  1189.   /* double value */
  1190.   else if (param->type == P_TYPE_DOUBLE) {
  1191.     if ((parse_float(fs, (double*)&init_val.double_val)) == PARSE_ERROR) {
  1192.       //if (PARSE_DEBUG) printf("parse_init_cond: error parsing double!n");
  1193.       return NULL;
  1194.     }
  1195.   }
  1196.   
  1197.   /* Unknown value */
  1198.   else {
  1199.     //if (PARSE_DEBUG) printf("parse_init_cond: unknown parameter type!n");
  1200.     return NULL;
  1201.   }
  1202.   
  1203.   /* Create new initial condition */
  1204.   if ((init_cond = new_init_cond(param, init_val)) == NULL) {
  1205.       //if (PARSE_DEBUG) printf("parse_init_cond: new_init_cond failed!n");
  1206.       return NULL;
  1207.   }
  1208.   
  1209.   /* Finished */
  1210.   return init_cond;
  1211. }
  1212. /* Parses a per frame init equation, not sure if this works right now */
  1213. init_cond_t * parse_per_frame_init_eqn(FILE * fs, struct PRESET_T * preset, splaytree_t * database) {
  1214.   
  1215.   char name[MAX_TOKEN_SIZE];
  1216.   param_t * param = NULL;
  1217.   value_t init_val;
  1218.   init_cond_t * init_cond;
  1219.   gen_expr_t * gen_expr;
  1220.   double val;
  1221.   token_t token;
  1222.   if (preset == NULL)
  1223.     return NULL;
  1224.   if (fs == NULL)
  1225.     return NULL;
  1226.   if ((token = parseToken(fs, name)) != tEq)
  1227.     return NULL;
  1228.   
  1229.   /* If a database was specified,then use find_param_db instead */
  1230.   if ((database != NULL) && ((param = find_param_db(name, database, TRUE)) == NULL)) {
  1231.     return NULL;
  1232.   }
  1233.   /* Otherwise use the builtin parameter and user databases. This is confusing. Sorry. */
  1234.   if ((param == NULL) && ((param = find_param(name, preset, P_CREATE)) == NULL)) {
  1235.     return NULL;
  1236.   }
  1237.   
  1238.   //if (PARSE_DEBUG) printf("parse_per_frame_init_eqn: parameter = "%s" (LINE %d)n", param->name, line_count);
  1239.   
  1240.   if (param->flags & P_FLAG_READONLY) {
  1241.     //if (PARSE_DEBUG) printf("pars_per_frame_init_eqn: builtin parameter "%s" marked as read only!n", param->name);
  1242.     return NULL;
  1243.   }
  1244.   
  1245.   /* At this point, a parameter has been created or was found
  1246.      in the database. */
  1247.   
  1248.   //if (PARSE_DEBUG) printf("parse_per_frame_init_eqn: parsing right hand side of per frame init equation.. (LINE %d)n", line_count);
  1249.   
  1250.   if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  1251.     //if (PARSE_DEBUG) printf("parse_per_frame_init_eqn: failed to parse general expresion!n");
  1252.     return NULL;
  1253.   }
  1254.  
  1255.   /* Compute initial condition value */
  1256.   val = eval_gen_expr(gen_expr);
  1257.   
  1258.   /* Free the general expression now that we are done with it */
  1259.   free_gen_expr(gen_expr);
  1260.   /* integer value (boolean is an integer in C) */
  1261.   if ((param->type == P_TYPE_INT) || (param->type == P_TYPE_BOOL)) {
  1262.     init_val.int_val = (int)val;
  1263.   }
  1264.   
  1265.   /* double value */
  1266.   else if (param->type == P_TYPE_DOUBLE) {
  1267.     init_val.double_val = val;
  1268.   }
  1269.   
  1270.   /* Unknown value */
  1271.   else {
  1272.     //if (PARSE_DEBUG) printf("parse_per_frame_init_eqn: unknown parameter type!n");
  1273.     return NULL;
  1274.   }
  1275.   
  1276.   /* Create new initial condition */
  1277.   if ((init_cond = new_init_cond(param, init_val)) == NULL) {
  1278.       //if (PARSE_DEBUG) printf("parse_per_frame_init_eqn: new_init_cond failed!n");
  1279.       return NULL;
  1280.   }
  1281.   /* Finished */
  1282.   return init_cond;
  1283. }
  1284. int parse_wavecode(char * token, FILE * fs, preset_t * preset) {
  1285.   char * var_string;
  1286.   init_cond_t * init_cond;
  1287.   custom_wave_t * custom_wave;
  1288.   int id;
  1289.   value_t init_val;
  1290.   param_t * param;
  1291.   /* Null argument checks */
  1292.   if (preset == NULL)
  1293.     return FAILURE;
  1294.   if (fs == NULL)
  1295.     return FAILURE;
  1296.   if (token == NULL)
  1297.     return FAILURE;
  1298.   /* token should be in the form wavecode_N_var, such as wavecode_1_samples */
  1299.   
  1300.   /* Get id and variable name from token string */
  1301.   if (parse_wavecode_prefix(token, &id, &var_string) < 0)   
  1302.     return PARSE_ERROR;
  1303.   
  1304.   //if (PARSE_DEBUG) printf("parse_wavecode: wavecode id = %d, parameter = "%s"n", id, var_string);
  1305.   /* Retrieve custom wave information from preset. The 3rd argument
  1306.      if true creates a custom wave if one does not exist */
  1307.   if ((custom_wave = find_custom_wave(id, preset, TRUE)) == NULL) {
  1308.     //if (PARSE_DEBUG) printf("parse_wavecode: failed to load (or create) custom wave (id = %d)!n", id);
  1309.     return FAILURE;
  1310.   }
  1311.   //if (PARSE_DEBUG) printf("parse_wavecode: custom wave found (id = %d)n", custom_wave->id);
  1312.   /* Retrieve parameter from this custom waves parameter db */
  1313.   if ((param = find_param_db(var_string, custom_wave->param_tree, TRUE)) == NULL)
  1314.     return FAILURE;
  1315.   //if (PARSE_DEBUG) printf("parse_wavecode: custom wave parameter found (name = %s)n", param->name);
  1316.   /* integer value (boolean is an integer in C) */
  1317.   if ((param->type == P_TYPE_INT) || (param->type == P_TYPE_BOOL)) {
  1318.     if ((parse_int(fs, (int*)&init_val.int_val)) == PARSE_ERROR) {
  1319.       //if (PARSE_DEBUG) printf("parse_wavecode: error parsing integer!n");
  1320.       return PARSE_ERROR;
  1321.     }
  1322.   }
  1323.   
  1324.   /* double value */
  1325.   else if (param->type == P_TYPE_DOUBLE) {
  1326.     if ((parse_float(fs, (double*)&init_val.double_val)) == PARSE_ERROR) {
  1327.       //if (PARSE_DEBUG) printf("parse_wavecode: error parsing double!n");
  1328.       return PARSE_ERROR;
  1329.     }
  1330.   }
  1331.   
  1332.   /* Unknown value */
  1333.   else {
  1334.     //if (PARSE_DEBUG) printf("parse_wavecode: unknown parameter type!n");
  1335.     return PARSE_ERROR;
  1336.   }
  1337.   
  1338.   /* Create new initial condition */
  1339.   if ((init_cond = new_init_cond(param, init_val)) == NULL) {
  1340.       //if (PARSE_DEBUG) printf("parse_wavecode: new_init_cond failed!n");
  1341.       return FAILURE;
  1342.   }
  1343.   
  1344.   if (splay_insert(init_cond, param->name, custom_wave->init_cond_tree) < 0) {
  1345.     free_init_cond(init_cond);
  1346.     return PARSE_ERROR;
  1347.   }
  1348.   //if (PARSE_DEBUG) printf("parse_wavecode: [success]n");
  1349.   return SUCCESS;
  1350. }
  1351. int parse_shapecode(char * token, FILE * fs, preset_t * preset) {
  1352.   char * var_string;
  1353.   init_cond_t * init_cond;
  1354.   custom_shape_t * custom_shape;
  1355.   int id;
  1356.   value_t init_val;
  1357.   param_t * param;
  1358.   /* Null argument checks */
  1359.   if (preset == NULL)
  1360.     return FAILURE;
  1361.   if (fs == NULL)
  1362.     return FAILURE;
  1363.   if (token == NULL)
  1364.     return FAILURE;
  1365.   /* token should be in the form shapecode_N_var, such as shapecode_1_samples */
  1366.   
  1367.   /* Get id and variable name from token string */
  1368.   if (parse_shapecode_prefix(token, &id, &var_string) < 0)   
  1369.     return PARSE_ERROR;
  1370.   
  1371.   //if (PARSE_DEBUG) printf("parse_shapecode: shapecode id = %d, parameter = "%s"n", id, var_string);
  1372.   /* Retrieve custom shape information from preset. The 3rd argument
  1373.      if true creates a custom shape if one does not exist */
  1374.   if ((custom_shape = find_custom_shape(id, preset, TRUE)) == NULL) {
  1375.     //if (PARSE_DEBUG) printf("parse_shapecode: failed to load (or create) custom shape (id = %d)!n", id);
  1376.     return FAILURE;
  1377.   }
  1378.   //if (PARSE_DEBUG) printf("parse_shapecode: custom shape found (id = %d)n", custom_shape->id);
  1379.   /* Retrieve parameter from this custom shapes parameter db */
  1380.   if ((param = find_param_db(var_string, custom_shape->param_tree, TRUE)) == NULL) {
  1381.     //if (PARSE_DEBUG) printf("parse_shapecode: failed to create parameter.n");
  1382.     return FAILURE;
  1383.   }
  1384.   //if (PARSE_DEBUG) printf("parse_shapecode: custom shape parameter found (name = %s)n", param->name);
  1385.   /* integer value (boolean is an integer in C) */
  1386.   if ((param->type == P_TYPE_INT) || (param->type == P_TYPE_BOOL)) {
  1387.     if ((parse_int(fs, (int*)&init_val.int_val)) == PARSE_ERROR) {
  1388.       //if (PARSE_DEBUG) printf("parse_shapecode: error parsing integer!n");
  1389.       return PARSE_ERROR;
  1390.     }
  1391.   }
  1392.   
  1393.   /* double value */
  1394.   else if (param->type == P_TYPE_DOUBLE) {
  1395.     if ((parse_float(fs, (double*)&init_val.double_val)) == PARSE_ERROR) {
  1396.       //if (PARSE_DEBUG) printf("parse_shapecode: error parsing double!n");
  1397.       return PARSE_ERROR;
  1398.     }
  1399.   }
  1400.   
  1401.   /* Unknown value */
  1402.   else {
  1403.     //if (PARSE_DEBUG) printf("parse_shapecode: unknown parameter type!n");
  1404.     return PARSE_ERROR;
  1405.   }
  1406.   
  1407.   /* Create new initial condition */
  1408.   if ((init_cond = new_init_cond(param, init_val)) == NULL) {
  1409.       //if (PARSE_DEBUG) printf("parse_shapecode: new_init_cond failed!n");
  1410.       return FAILURE;
  1411.   }
  1412.  
  1413.   if (splay_insert(init_cond, param->name, custom_shape->init_cond_tree) < 0) {
  1414.     free_init_cond(init_cond);
  1415.     //if (PARSE_DEBUG) printf("parse_shapecode: initial condition already set, not reinserting it (param = "%s")n", param->name);
  1416.     return PARSE_ERROR;
  1417.   }
  1418.   //if (PARSE_DEBUG) printf("parse_shapecode: [success]n");
  1419.   return SUCCESS;
  1420. }
  1421. int parse_wavecode_prefix(char * token, int * id, char ** var_string) {
  1422.   int len, i, j;
  1423.   
  1424.   if (token == NULL)
  1425.     return FAILURE;
  1426.   if (*var_string == NULL)
  1427.     return FAILURE;
  1428.   if (id == NULL)
  1429.     return FAILURE;
  1430.   
  1431.   len = strlen(token);
  1432.   /* Move pointer passed "wavecode_" prefix */
  1433.   if (len <= WAVECODE_STRING_LENGTH)
  1434.     return FAILURE;
  1435.   i = WAVECODE_STRING_LENGTH;
  1436.   j = 0;
  1437.   (*id) = 0;
  1438.   
  1439.   /* This loop grabs the integer id for this custom wave */
  1440.   while ((i < len) && (token[i] >=  48) && (token[i] <= 57)) {
  1441.     if (j >= MAX_TOKEN_SIZE)
  1442.       return FAILURE;
  1443.     
  1444.     (*id) = 10*(*id) + (token[i]-48);
  1445.     j++;
  1446.     i++;
  1447.   }
  1448.  
  1449.   if (i > (len - 2))
  1450.     return FAILURE;
  1451.   
  1452.   *var_string = token + i + 1;
  1453.  
  1454.   return SUCCESS;
  1455. }
  1456. int parse_shapecode_prefix(char * token, int * id, char ** var_string) {
  1457.   int len, i, j;
  1458.   
  1459.   if (token == NULL)
  1460.     return FAILURE;
  1461.   if (*var_string == NULL)
  1462.     return FAILURE;
  1463.   if (id == NULL)
  1464.     return FAILURE;
  1465.   
  1466.   len = strlen(token);
  1467.   /* Move pointer passed "shapecode_" prefix */
  1468.   if (len <= SHAPECODE_STRING_LENGTH)
  1469.     return FAILURE;
  1470.   i = SHAPECODE_STRING_LENGTH;
  1471.   j = 0;
  1472.   (*id) = 0;
  1473.   
  1474.   /* This loop grabs the integer id for this custom shape */
  1475.   while ((i < len) && (token[i] >=  48) && (token[i] <= 57)) {
  1476.     if (j >= MAX_TOKEN_SIZE)
  1477.       return FAILURE;
  1478.     
  1479.     (*id) = 10*(*id) + (token[i]-48);
  1480.     j++;
  1481.     i++;
  1482.   }
  1483.  
  1484.   if (i > (len - 2))
  1485.     return FAILURE;
  1486.   
  1487.   *var_string = token + i + 1;
  1488.  
  1489.   return SUCCESS;
  1490. }
  1491. int parse_wave_prefix(char * token, int * id, char ** eqn_string) {
  1492.   int len, i, j;
  1493.   
  1494.   if (token == NULL)
  1495.     return FAILURE;
  1496.   if (eqn_string == NULL)
  1497.     return FAILURE;
  1498.   if (id == NULL)
  1499.     return FAILURE;
  1500.   
  1501.   len = strlen(token);
  1502.  
  1503.   if (len <= WAVE_STRING_LENGTH)
  1504.     return FAILURE;
  1505.   i = WAVE_STRING_LENGTH;
  1506.   j = 0;
  1507.   (*id) = 0;
  1508.   
  1509.   /* This loop grabs the integer id for this custom wave */
  1510.   while ((i < len) && (token[i] >=  48) && (token[i] <= 57)) {
  1511.     if (j >= MAX_TOKEN_SIZE)
  1512.       return FAILURE;
  1513.     
  1514.     (*id) = 10*(*id) + (token[i]-48);
  1515.     j++;
  1516.     i++;
  1517.   }
  1518.   if (i > (len - 2))
  1519.     return FAILURE;
  1520.  
  1521.   *eqn_string = token + i + 1;
  1522.  
  1523.   return SUCCESS;
  1524. }
  1525. int parse_shape_prefix(char * token, int * id, char ** eqn_string) {
  1526.   int len, i, j;
  1527.   
  1528.   if (token == NULL)
  1529.     return FAILURE;
  1530.   if (eqn_string == NULL)
  1531.     return FAILURE;
  1532.   if (id == NULL)
  1533.     return FAILURE;
  1534.   
  1535.   len = strlen(token);
  1536.  
  1537.   if (len <= SHAPE_STRING_LENGTH)
  1538.     return FAILURE;
  1539.   i = SHAPE_STRING_LENGTH;
  1540.   j = 0;
  1541.   (*id) = 0;
  1542.   
  1543.   /* This loop grabs the integer id for this custom wave */
  1544.   while ((i < len) && (token[i] >=  48) && (token[i] <= 57)) {
  1545.     if (j >= MAX_TOKEN_SIZE)
  1546.       return FAILURE;
  1547.     
  1548.     (*id) = 10*(*id) + (token[i]-48);
  1549.     j++;
  1550.     i++;
  1551.   }
  1552.   if (i > (len - 2))
  1553.     return FAILURE;
  1554.  
  1555.   *eqn_string = token + i + 1;
  1556.  
  1557.   return SUCCESS;
  1558. }
  1559. /* Parses custom wave equations */
  1560. int parse_wave(char * token, FILE * fs, preset_t * preset) {
  1561.   
  1562.   int id;
  1563.   char * eqn_type;
  1564.   char string[MAX_TOKEN_SIZE];
  1565.   param_t * param;
  1566.   per_frame_eqn_t * per_frame_eqn;
  1567.   gen_expr_t * gen_expr;
  1568.   custom_wave_t * custom_wave;
  1569.   init_cond_t * init_cond;
  1570.   if (token == NULL)
  1571.     return FAILURE;
  1572.   if (fs == NULL)
  1573.     return FAILURE;
  1574.   if (preset == NULL)
  1575.     return FAILURE;
  1576.   
  1577.   /* Grab custom wave id and equation type (per frame or per point) from string token */
  1578.   if (parse_wave_prefix(token, &id, &eqn_type) < 0) {
  1579.     //if (PARSE_DEBUG) printf("parse_wave: syntax error in custom wave prefix!n");
  1580.     return FAILURE;
  1581.   }
  1582.   /* Retrieve custom wave associated with this id */
  1583.   if ((custom_wave = find_custom_wave(id, preset, TRUE)) == NULL)
  1584.     return FAILURE;
  1585.   /* per frame init equation case */     
  1586.   if (!strncmp(eqn_type, WAVE_INIT_STRING, WAVE_INIT_STRING_LENGTH)) {
  1587.     //if (PARSE_DEBUG) printf("parse_wave (per frame init): [begin] (LINE %d)n", line_count);
  1588.     /* Parse the per frame equation */
  1589.     if ((init_cond = parse_per_frame_init_eqn(fs, preset, custom_wave->param_tree)) == NULL) {
  1590.       //if (PARSE_DEBUG) printf("parse_wave (per frame init): equation parsing failed (LINE %d)n", line_count);
  1591.       return PARSE_ERROR;
  1592.     }
  1593.     /* Insert the equation in the per frame equation tree */
  1594.     if (splay_insert(init_cond, init_cond->param->name, custom_wave->per_frame_init_eqn_tree) < 0) {
  1595.       //if (PARSE_DEBUG) printf("parse_wave (per frame init): failed to add equation (ERROR)n");
  1596.        free_init_cond(init_cond); /* will free the gen expr too */
  1597.       return FAILURE;
  1598.     }
  1599.    
  1600.     if (update_string_buffer(custom_wave->per_frame_init_eqn_string_buffer, 
  1601.      &custom_wave->per_frame_init_eqn_string_index) < 0)
  1602.       return FAILURE;
  1603.     return SUCCESS;
  1604.   
  1605.   }
  1606.   /* per frame equation case */
  1607.   if (!strncmp(eqn_type, PER_FRAME_STRING_NO_UNDERSCORE, PER_FRAME_STRING_NO_UNDERSCORE_LENGTH)) {
  1608.     //if (PARSE_DEBUG) printf("parse_wave (per_frame): [start] (custom wave id = %d)n", custom_wave->id);
  1609.     
  1610.     if (parseToken(fs, string) != tEq) {
  1611.       //if (PARSE_DEBUG) printf("parse_wave (per_frame): no equal sign after string "%s" (LINE %d)n", string, line_count);
  1612.       return PARSE_ERROR;
  1613.     }
  1614.   
  1615.     /* Find the parameter associated with the string in the custom wave database */
  1616.     if ((param = find_param_db(string, custom_wave->param_tree, TRUE)) == NULL) { 
  1617.       //if (PARSE_DEBUG) printf("parse_wave (per_frame): parameter "%s" not found or cannot be malloc'ed!!n", string);
  1618.       return FAILURE;
  1619.     }
  1620.   
  1621.     
  1622.     /* Make sure parameter is writable */
  1623.     if (param->flags & P_FLAG_READONLY) {
  1624.       //if (PARSE_DEBUG) printf("parse_wave (per_frame): parameter %s is marked as read only (LINE %d)n", param->name, line_count);  
  1625.       return FAILURE;
  1626.     }
  1627.   
  1628.     /* Parse right side of equation as an expression */
  1629.     current_wave = custom_wave;
  1630.     if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  1631.       //if (PARSE_DEBUG) printf("parse_wave (per_frame): equation evaluated to null (LINE %d)n", line_count);
  1632.       current_wave = NULL;
  1633.       return PARSE_ERROR;
  1634.     }
  1635.     current_wave = NULL;
  1636.     //if (PARSE_DEBUG) printf("parse_wave (per_frame): [finished parsing equation] (LINE %d)n", line_count);
  1637.   
  1638.     /* Create a new per frame equation */
  1639.     if ((per_frame_eqn = new_per_frame_eqn(custom_wave->per_frame_count++, param, gen_expr)) == NULL) {
  1640.       //if (PARSE_DEBUG) printf("parse_wave (per_frame): failed to create a new per frame eqn, out of memory?n");
  1641.       free_gen_expr(gen_expr);
  1642.       return FAILURE;
  1643.     }
  1644.  
  1645.     if (splay_insert(per_frame_eqn, &per_frame_eqn->index, custom_wave->per_frame_eqn_tree) < 0) {
  1646.       free_per_frame_eqn(per_frame_eqn);
  1647.       return FAILURE;
  1648.     }
  1649.        
  1650.     //if (PARSE_DEBUG) printf("parse_wave (per_frame): equation %d associated with custom wave %d [success]n", 
  1651.     //     per_frame_eqn->index, custom_wave->id);
  1652.     
  1653.     /* Need to add stuff to string buffer so the editor can read the equations. 
  1654.        Why not make a nice little helper function for this? - here it is: */
  1655.    
  1656.     if (update_string_buffer(custom_wave->per_frame_eqn_string_buffer, &custom_wave->per_frame_eqn_string_index) < 0)
  1657.       return FAILURE;
  1658.  
  1659.     return SUCCESS;
  1660.   }
  1661.   /* per point equation case */
  1662.   if (!strncmp(eqn_type, PER_POINT_STRING, PER_POINT_STRING_LENGTH)) {
  1663.     //if (PARSE_DEBUG) printf("parse_wave (per_point): per_pixel equation parsing start...(LINE %d)n", line_count);
  1664.     if (parseToken(fs, string) != tEq) { /* parse per pixel operator  name */
  1665.       //if (PARSE_DEBUG) printf("parse_wave (per_point): equal operator missing after per pixel operator! (LINE %d)n", line_count);
  1666.       return PARSE_ERROR;
  1667.     }
  1668.     
  1669.     /* Parse right side of equation as an expression */
  1670.     current_wave = custom_wave;
  1671.     if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  1672.       //if (PARSE_DEBUG) printf("parse_wave (per_point): equation evaluated to null? (LINE %d)n", line_count);
  1673.       return PARSE_ERROR;
  1674.     }
  1675.     current_wave = NULL;
  1676.     /* Add the per point equation */
  1677.     if (add_per_point_eqn(string, gen_expr, custom_wave) < 0) {
  1678.       free_gen_expr(gen_expr);
  1679.       return PARSE_ERROR;
  1680.     }
  1681.    
  1682.     if (update_string_buffer(custom_wave->per_point_eqn_string_buffer, &custom_wave->per_point_eqn_string_index) < 0)
  1683.       return FAILURE;
  1684.     //if (PARSE_DEBUG) printf("parse_wave (per_point): [finished] (custom wave id = %d)n", custom_wave->id);
  1685.     return SUCCESS;
  1686.   }
  1687.   /* Syntax error, return parse error */
  1688.   return PARSE_ERROR;
  1689. }
  1690. /* Parses custom shape equations */
  1691. int parse_shape(char * token, FILE * fs, preset_t * preset) {
  1692.   
  1693.   int id;
  1694.   char * eqn_type;
  1695.   char string[MAX_TOKEN_SIZE];
  1696.   param_t * param;
  1697.   per_frame_eqn_t * per_frame_eqn;
  1698.   gen_expr_t * gen_expr;
  1699.   custom_shape_t * custom_shape;
  1700.   init_cond_t * init_cond;
  1701.   if (token == NULL)
  1702.     return FAILURE;
  1703.   if (fs == NULL)
  1704.     return FAILURE;
  1705.   if (preset == NULL)
  1706.     return FAILURE;
  1707.   
  1708.   /* Grab custom shape id and equation type (per frame or per point) from string token */
  1709.   if (parse_shape_prefix(token, &id, &eqn_type) < 0) {
  1710.     //if (PARSE_DEBUG) printf("parse_shape: syntax error in custom shape prefix!n");
  1711.     return PARSE_ERROR;
  1712.   }
  1713.   /* Retrieve custom shape associated with this id */
  1714.   if ((custom_shape = find_custom_shape(id, preset, TRUE)) == NULL)
  1715.     return FAILURE;
  1716.   /* per frame init equation case */     
  1717.   if (!strncmp(eqn_type, SHAPE_INIT_STRING, SHAPE_INIT_STRING_LENGTH)) {
  1718.     //if (PARSE_DEBUG) printf("parse_shape (per frame init): [begin] (LINE %d)n", line_count);
  1719.     /* Parse the per frame equation */
  1720.     if ((init_cond = parse_per_frame_init_eqn(fs, preset, custom_shape->param_tree)) == NULL) {
  1721.       //if (PARSE_DEBUG) printf("parse_shape (per frame init): equation parsing failed (LINE %d)n", line_count);
  1722.       return PARSE_ERROR;
  1723.     }
  1724.     
  1725.     /* Insert the equation in the per frame equation tree */
  1726.     if (splay_insert(init_cond, init_cond->param->name, custom_shape->per_frame_init_eqn_tree) < 0) {
  1727.       //if (PARSE_DEBUG) printf("parse_shape (per frame init): failed to add equation (ERROR)n");
  1728.       free_init_cond(init_cond); /* will free the gen expr too */
  1729.       return ERROR;
  1730.     }
  1731.     if (update_string_buffer(custom_shape->per_frame_init_eqn_string_buffer, 
  1732.      &custom_shape->per_frame_init_eqn_string_index) < 0)
  1733.       return FAILURE;
  1734.     return SUCCESS;
  1735.   
  1736.   }
  1737.   /* per frame equation case */
  1738.   if (!strncmp(eqn_type, PER_FRAME_STRING_NO_UNDERSCORE, PER_FRAME_STRING_NO_UNDERSCORE_LENGTH)) {
  1739.     //if (PARSE_DEBUG) printf("parse_shape (per_frame): [start] (custom shape id = %d)n", custom_shape->id);
  1740.     
  1741.     if (parseToken(fs, string) != tEq) {
  1742.       //if (PARSE_DEBUG) printf("parse_shape (per_frame): no equal sign after string "%s" (LINE %d)n", string, line_count);
  1743.       return PARSE_ERROR;
  1744.     }
  1745.   
  1746.     /* Find the parameter associated with the string in the custom shape database */
  1747.     if ((param = find_param_db(string, custom_shape->param_tree, TRUE)) == NULL) { 
  1748.       //if (PARSE_DEBUG) printf("parse_shape (per_frame): parameter "%s" not found or cannot be malloc'ed!!n", string);
  1749.       return FAILURE;
  1750.     }
  1751.   
  1752.     
  1753.     /* Make sure parameter is writable */
  1754.     if (param->flags & P_FLAG_READONLY) {
  1755.       //if (PARSE_DEBUG) printf("parse_shape (per_frame): parameter %s is marked as read only (LINE %d)n", param->name, line_count);  
  1756.       return FAILURE;
  1757.     }
  1758.   
  1759.     /* Parse right side of equation as an expression */
  1760.     current_shape = custom_shape;
  1761.     if ((gen_expr = parse_gen_expr(fs, NULL, preset)) == NULL) {
  1762.       //if (PARSE_DEBUG) printf("parse_shape (per_frame): equation evaluated to null (LINE %d)n", line_count);
  1763.       current_shape = NULL;
  1764.       return PARSE_ERROR;
  1765.     }
  1766.     current_shape = NULL;
  1767.     //if (PARSE_DEBUG) printf("parse_shape (per_frame): [finished parsing equation] (LINE %d)n", line_count);
  1768.   
  1769.     /* Create a new per frame equation */
  1770.     if ((per_frame_eqn = new_per_frame_eqn(custom_shape->per_frame_count++, param, gen_expr)) == NULL) {
  1771.       //if (PARSE_DEBUG) printf("parse_shape (per_frame): failed to create a new per frame eqn, out of memory?n");
  1772.       free_gen_expr(gen_expr);
  1773.       return FAILURE;
  1774.     }
  1775.  
  1776.     if (splay_insert(per_frame_eqn, &per_frame_eqn->index, custom_shape->per_frame_eqn_tree) < 0) {
  1777.       free_per_frame_eqn(per_frame_eqn);
  1778.       return FAILURE;
  1779.     }
  1780.        
  1781.     //if (PARSE_DEBUG) printf("parse_shape (per_frame): equation %d associated with custom shape %d [success]n", 
  1782.     //     per_frame_eqn->index, custom_shape->id);
  1783.     
  1784.     /* Need to add stuff to string buffer so the editor can read the equations.
  1785.        Why not make a nice little helper function for this? - here it is: */
  1786.     
  1787.     if (update_string_buffer(custom_shape->per_frame_eqn_string_buffer, &custom_shape->per_frame_eqn_string_index) < 0)
  1788.       return FAILURE;
  1789.  
  1790.     return SUCCESS;
  1791.   }
  1792.   /* Syntax error, return parse error */
  1793.   return PARSE_ERROR;
  1794. }
  1795. /* Helper function to update the string buffers used by the editor */
  1796. int update_string_buffer(char * buffer, int * index) {
  1797.   int string_length;
  1798.   int skip_size;
  1799.   if (!buffer)
  1800.     return FAILURE;
  1801.   if (!index)
  1802.     return FAILURE;
  1803.   
  1804.   /* If the string line buffer used by the parser is already full then quit */
  1805.   if (string_line_buffer_index == (STRING_LINE_SIZE-1))
  1806.     return FAILURE;
  1807.   if ((skip_size = get_string_prefix_len(string_line_buffer)) == FAILURE)
  1808.     return FAILURE;
  1809.   string_line_buffer[string_line_buffer_index++] = 'n';
  1810.   //  string_length = strlen(string_line_buffer + strlen(eqn_string)+1);
  1811.   if (skip_size >= STRING_LINE_SIZE)
  1812.     return FAILURE;
  1813.   string_length = strlen(string_line_buffer + skip_size);
  1814.   if (skip_size > (STRING_LINE_SIZE-1))
  1815.     return FAILURE;
  1816.   /* Add line to string buffer */
  1817.   strncpy(buffer + (*index), 
  1818.   string_line_buffer + skip_size, string_length);
  1819.   
  1820.   /* Buffer full, quit */
  1821.   if ((*index) > (STRING_BUFFER_SIZE - 1)) {
  1822.     //if (PARSE_DEBUG) printf("update_string_buffer: string buffer full!n");
  1823.     return FAILURE;
  1824.   }
  1825.   
  1826.   /* Otherwise, increment string index by the added string length */
  1827.   (*index)+=string_length;
  1828.     
  1829.   return SUCCESS;
  1830.   
  1831. }
  1832. /* Helper function: returns the length of the prefix portion in the line
  1833.    buffer (the passed string here). In other words, given
  1834.    the string 'per_frame_1 = x = ....', return the length of 'per_frame_1 = '
  1835.    Returns -1 if syntax error
  1836. */
  1837. int get_string_prefix_len(char * string) {
  1838.   
  1839.   int i = 0;
  1840.   /* Null argument check */
  1841.   if (string == NULL)
  1842.     return FAILURE;
  1843.   
  1844.   /* First find the equal sign */
  1845.   while (string[i] != '=') {
  1846.     if (string[i] == 0)
  1847.       return FAILURE;
  1848.     i++;
  1849.   }
  1850.   /* If the string already ends at the next char then give up */
  1851.   if (string[i+1] == 0)
  1852.     return FAILURE;
  1853.   /* Move past the equal sign */
  1854.   i++;
  1855.   /* Now found the start of the LHS variable, ie skip the spaces */
  1856.   while(string[i] == ' ') {
  1857.     i++;
  1858.   }
  1859.   /* If this is the end of the string then its a syntax error */
  1860.   if (string[i] == 0)
  1861.     return FAILURE;
  1862.   /* Finished succesfully, return the length */
  1863.   return i;
  1864. }