ChipMaster's bwBASIC This also includes history going back to v2.10. *WARN* some binary files might have been corrupted by CRLF.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1226 lines
37 KiB

  1. /****************************************************************
  2. bwb_elx.c Parse Elements of Expressions
  3. for Bywater BASIC Interpreter
  4. Copyright (c) 1993, Ted A. Campbell
  5. Bywater Software
  6. email: tcamp@delphi.com
  7. Copyright and Permissions Information:
  8. All U.S. and international rights are claimed by the author,
  9. Ted A. Campbell.
  10. This software is released under the terms of the GNU General
  11. Public License (GPL), which is distributed with this software
  12. in the file "COPYING". The GPL specifies the terms under
  13. which users may copy and use the software in this distribution.
  14. A separate license is available for commercial distribution,
  15. for information on which you should contact the author.
  16. ****************************************************************/
  17. /*---------------------------------------------------------------*/
  18. /* NOTE: Modifications marked "JBV" were made by Jon B. Volkoff, */
  19. /* 11/1995 (eidetics@cerf.net). */
  20. /* */
  21. /* Those additionally marked with "DD" were at the suggestion of */
  22. /* Dale DePriest (daled@cadence.com). */
  23. /*---------------------------------------------------------------*/
  24. #include <stdio.h>
  25. #include <ctype.h>
  26. #include <math.h>
  27. #include "bwbasic.h"
  28. #include "bwb_mes.h"
  29. /***************************************************************
  30. FUNCTION: exp_paren()
  31. DESCRIPTION: This function interprets a parenthetical
  32. expression, calling bwb_exp() (recursively)
  33. to resolve the internal expression.
  34. ***************************************************************/
  35. #if ANSI_C
  36. int
  37. exp_paren( char *expression )
  38. #else
  39. int
  40. exp_paren( expression )
  41. char *expression;
  42. #endif
  43. {
  44. struct exp_ese *e;
  45. int s_pos; /* position in build buffer */
  46. int loop;
  47. int paren_level;
  48. /* find a string enclosed by parentheses */
  49. CURTASK exps[ CURTASK expsc ].pos_adv = 1; /* start beyond open paren */
  50. s_pos = 0;
  51. loop = TRUE;
  52. paren_level = 1;
  53. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  54. while( loop == TRUE )
  55. {
  56. /* check the current character */
  57. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  58. {
  59. case '\r': /* these tests added v1.11 */
  60. case '\n':
  61. case '\0':
  62. bwb_error( err_incomplete );
  63. loop = FALSE;
  64. break;
  65. case '(':
  66. ++paren_level;
  67. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  68. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  69. ++s_pos;
  70. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  71. break;
  72. case ')':
  73. --paren_level;
  74. if ( paren_level == 0 )
  75. {
  76. loop = FALSE;
  77. }
  78. else
  79. {
  80. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  81. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  82. ++s_pos;
  83. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  84. }
  85. break;
  86. case '\"': /* embedded string constant */
  87. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  88. while ( ( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] != '\"' )
  89. && ( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] != '\0' ) )
  90. {
  91. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  92. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  93. ++s_pos;
  94. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  95. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  96. }
  97. break;
  98. default:
  99. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  100. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  101. ++s_pos;
  102. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  103. break;
  104. }
  105. /* advance the counter */
  106. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  107. }
  108. #if INTENSIVE_DEBUG
  109. sprintf( bwb_ebuf, "in exp_paren() found internal string <%s>",
  110. CURTASK exps[ CURTASK expsc ].string );
  111. bwb_debug( bwb_ebuf );
  112. #endif
  113. /* call bwb_exp() recursively to interpret this expression */
  114. CURTASK exps[ CURTASK expsc ].rec_pos = 0;
  115. e = bwb_exp( CURTASK exps[ CURTASK expsc ].string, FALSE,
  116. &( CURTASK exps[ CURTASK expsc ].rec_pos ) );
  117. /* assign operation and value at this level */
  118. CURTASK exps[ CURTASK expsc ].type = e->type;
  119. switch ( e->type )
  120. {
  121. case STRING:
  122. CURTASK exps[ CURTASK expsc ].operation = CONST_STRING;
  123. str_btob( exp_getsval( &( CURTASK exps[ CURTASK expsc ] )), exp_getsval( e ) );
  124. break;
  125. default:
  126. CURTASK exps[ CURTASK expsc ].operation = NUMBER;
  127. CURTASK exps[ CURTASK expsc ].nval = exp_getnval( e );
  128. break;
  129. }
  130. return TRUE;
  131. }
  132. /***************************************************************
  133. FUNCTION: exp_strconst()
  134. DESCRIPTION: This function interprets a string
  135. constant.
  136. ***************************************************************/
  137. #if ANSI_C
  138. int
  139. exp_strconst( char *expression )
  140. #else
  141. int
  142. exp_strconst( expression )
  143. char *expression;
  144. #endif
  145. {
  146. int e_pos, s_pos;
  147. /* assign values to structure */
  148. CURTASK exps[ CURTASK expsc ].type = STRING;
  149. CURTASK exps[ CURTASK expsc ].operation = CONST_STRING;
  150. /* set counters */
  151. s_pos = 0;
  152. CURTASK exps[ CURTASK expsc ].pos_adv = e_pos = 1;
  153. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  154. /* read the string up until the next double quotation mark */
  155. /* While yer at it, check for a null terminator too (JBV, found by DD) */
  156. while(( expression[ e_pos ] != '\"') && ( expression[ e_pos ] != '\0' ))
  157. {
  158. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ e_pos ];
  159. ++e_pos;
  160. ++s_pos;
  161. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  162. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  163. if ( s_pos >= ( MAXSTRINGSIZE - 1 ) )
  164. {
  165. #if PROG_ERRORS
  166. sprintf( bwb_ebuf, "string <%s> exceeds maximum size (%d) for string constant.",
  167. expression, MAXSTRINGSIZE );
  168. bwb_error( bwb_ebuf );
  169. #else
  170. bwb_error( err_overflow );
  171. #endif
  172. return OP_NULL;
  173. }
  174. }
  175. /* now write string over to bstring */
  176. str_ctob( &( CURTASK exps[ CURTASK expsc ].sval ), CURTASK exps[ CURTASK expsc ].string );
  177. /* advance past last double quotation mark */
  178. /*-------------------------------------------------------------*/
  179. /* Of course, it doesn't hurt to make sure it's really a quote */
  180. /* (JBV, found by DD) */
  181. /*-------------------------------------------------------------*/
  182. if ( expression[ e_pos ] == '\"' ) ++CURTASK exps[ CURTASK expsc ].pos_adv;
  183. /* return */
  184. return TRUE;
  185. }
  186. /***************************************************************
  187. FUNCTION: exp_numconst()
  188. DESCRIPTION: This function interprets a numerical
  189. constant.
  190. ***************************************************************/
  191. #if ANSI_C
  192. int
  193. exp_numconst( char *expression )
  194. #else
  195. int
  196. exp_numconst( expression )
  197. char *expression;
  198. #endif
  199. {
  200. int base; /* numerical base for the constant */
  201. static struct bwb_variable mantissa; /* mantissa of floating-point number */
  202. static int init = FALSE; /* is mantissa variable initialized? */
  203. int exponent; /* exponent for floating point number */
  204. int man_start; /* starting point of mantissa */
  205. int s_pos; /* position in build string */
  206. int build_loop;
  207. int need_pm;
  208. int i;
  209. bnumber d;
  210. #if CHECK_RECURSION
  211. static int in_use = FALSE; /* boolean: is function in use? */
  212. /* check recursion status */
  213. if ( in_use == TRUE )
  214. {
  215. sprintf( bwb_ebuf, "Recursion error in bwb_exp.c:exp_findop(): recursion violation." );
  216. bwb_error( bwb_ebuf );
  217. }
  218. /* reset recursion status indicator */
  219. else
  220. {
  221. in_use = TRUE;
  222. }
  223. #endif
  224. /* initialize the variable if necessary */
  225. #if INTENSIVE_DEBUG
  226. strcpy( mantissa.name, "(mantissa)" );
  227. #endif
  228. if ( init == FALSE )
  229. {
  230. init = TRUE;
  231. var_make( &mantissa, NUMBER );
  232. }
  233. /* be sure that the array_pos[ 0 ] for mantissa is set to dim_base;
  234. this is necessary because mantissa might be used before dim_base
  235. is set */
  236. mantissa.array_pos[ 0 ] = dim_base;
  237. #if INTENSIVE_DEBUG
  238. sprintf( bwb_ebuf, "in exp_numconst(): received <%s>, eval <%c>",
  239. expression, expression[ 0 ] );
  240. bwb_debug( bwb_ebuf );
  241. #endif
  242. need_pm = FALSE;
  243. CURTASK exps[ CURTASK expsc ].nval = (bnumber) 0;
  244. /* check the first character(s) to determine numerical base
  245. and starting point of the mantissa */
  246. switch( expression[ 0 ] )
  247. {
  248. case '-':
  249. case '+':
  250. case '0':
  251. case '1':
  252. case '2':
  253. case '3':
  254. case '4':
  255. case '5':
  256. case '6':
  257. case '7':
  258. case '8':
  259. case '9':
  260. case '.':
  261. base = 10; /* decimal constant */
  262. man_start = 0; /* starts at position 0 */
  263. need_pm = FALSE;
  264. break;
  265. case '&': /* hex or octal constant */
  266. if ( ( expression[ 1 ] == 'H' ) || ( expression[ 1 ] == 'h' ))
  267. {
  268. base = 16; /* hexadecimal constant */
  269. man_start = 2; /* starts at position 2 */
  270. }
  271. else
  272. {
  273. base = 8; /* octal constant */
  274. if ( ( expression[ 1 ] == 'O' ) || ( expression[ 1 ] == 'o' ))
  275. {
  276. man_start = 2; /* starts at position 2 */
  277. }
  278. else
  279. {
  280. man_start = 1; /* starts at position 1 */
  281. }
  282. }
  283. break;
  284. default:
  285. #if PROG_ERRORS
  286. sprintf( bwb_ebuf, "expression <%s> is not a numerical constant.",
  287. expression );
  288. bwb_error( bwb_ebuf );
  289. #else
  290. bwb_error( err_syntax );
  291. #endif
  292. return OP_NULL;
  293. }
  294. /* now build the mantissa according to the numerical base */
  295. switch( base )
  296. {
  297. case 10: /* decimal constant */
  298. /* initialize counters */
  299. CURTASK exps[ CURTASK expsc ].pos_adv = man_start;
  300. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  301. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  302. s_pos = 0;
  303. exponent = OP_NULL;
  304. build_loop = TRUE;
  305. /* loop to build the string */
  306. while ( build_loop == TRUE )
  307. {
  308. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  309. {
  310. case '-': /* prefixed plus or minus */
  311. case '+':
  312. /* in the first position, a plus or minus sign can
  313. be added to the beginning of the string to be
  314. scanned */
  315. if ( CURTASK exps[ CURTASK expsc ].pos_adv == man_start )
  316. {
  317. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  318. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  319. ++s_pos;
  320. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  321. }
  322. /* but in any other position, the plus or minus sign
  323. must be taken as an operator and thus as terminating
  324. the string to be scanned */
  325. else
  326. {
  327. build_loop = FALSE;
  328. }
  329. break;
  330. case '.': /* note at least single precision */
  331. case '0': /* or ordinary digit */
  332. case '1':
  333. case '2':
  334. case '3':
  335. case '4':
  336. case '5':
  337. case '6':
  338. case '7':
  339. case '8':
  340. case '9':
  341. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  342. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  343. ++s_pos;
  344. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  345. break;
  346. case '#': /* Microsoft-type precision indicator; ignored but terminates */
  347. case '!': /* Microsoft-type precision indicator; ignored but terminates */
  348. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  349. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  350. exponent = FALSE;
  351. build_loop = FALSE;
  352. break;
  353. case 'E': /* exponential, single precision */
  354. case 'e':
  355. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  356. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  357. exponent = TRUE;
  358. build_loop = FALSE;
  359. break;
  360. case 'D': /* exponential, double precision */
  361. case 'd':
  362. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  363. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  364. exponent = TRUE;
  365. build_loop = FALSE;
  366. break;
  367. default: /* anything else, terminate */
  368. build_loop = FALSE;
  369. break;
  370. }
  371. }
  372. /* assign the value to the mantissa variable */
  373. #if NUMBER_DOUBLE
  374. sscanf( CURTASK exps[ CURTASK expsc ].string, "%lf",
  375. var_findnval( &mantissa, mantissa.array_pos ));
  376. #else
  377. sscanf( CURTASK exps[ CURTASK expsc ].string, "%f",
  378. var_findnval( &mantissa, mantissa.array_pos ));
  379. #endif
  380. #if INTENSIVE_DEBUG
  381. sprintf( bwb_ebuf, "in exp_numconst(): read mantissa, string <%s> val <%lf>",
  382. CURTASK exps[ CURTASK expsc ].string, var_getnval( &mantissa ) );
  383. bwb_debug( bwb_ebuf );
  384. #endif
  385. /* test if integer bounds have been exceeded */
  386. if ( CURTASK exps[ CURTASK expsc ].type == NUMBER )
  387. {
  388. i = (int) var_getnval( &mantissa );
  389. d = (bnumber) i;
  390. if ( d != var_getnval( &mantissa ))
  391. {
  392. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  393. #if INTENSIVE_DEBUG
  394. sprintf( bwb_ebuf, "in exp_numconst(): integer bounds violated, promote to NUMBER" );
  395. bwb_debug( bwb_ebuf );
  396. #endif
  397. }
  398. }
  399. /* read the exponent if there is one */
  400. if ( exponent == TRUE )
  401. {
  402. /* allow a plus or minus once at the beginning */
  403. need_pm = TRUE;
  404. /* initialize counters */
  405. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  406. s_pos = 0;
  407. build_loop = TRUE;
  408. /* loop to build the string */
  409. while ( build_loop == TRUE )
  410. {
  411. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  412. {
  413. case '-': /* prefixed plus or minus */
  414. case '+':
  415. if ( need_pm == TRUE ) /* only allow once */
  416. {
  417. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  418. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  419. ++s_pos;
  420. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  421. }
  422. else
  423. {
  424. build_loop = FALSE;
  425. }
  426. break;
  427. case '0': /* or ordinary digit */
  428. case '1':
  429. case '2':
  430. case '3':
  431. case '4':
  432. case '5':
  433. case '6':
  434. case '7':
  435. case '8':
  436. case '9':
  437. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  438. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  439. ++s_pos;
  440. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  441. need_pm = FALSE;
  442. break;
  443. default: /* anything else, terminate */
  444. build_loop = FALSE;
  445. break;
  446. }
  447. } /* end of build loop for exponent */
  448. /* assign the value to the user variable */
  449. #if NUMBER_DOUBLE
  450. sscanf( CURTASK exps[ CURTASK expsc ].string, "%lf",
  451. &( CURTASK exps[ CURTASK expsc ].nval ) );
  452. #else
  453. sscanf( CURTASK exps[ CURTASK expsc ].string, "%f",
  454. &( CURTASK exps[ CURTASK expsc ].nval ) );
  455. #endif
  456. #if INTENSIVE_DEBUG
  457. sprintf( bwb_ebuf, "in exp_numconst(): exponent is <%d>",
  458. (int) CURTASK exps[ CURTASK expsc ].nval );
  459. bwb_debug( bwb_ebuf );
  460. #endif
  461. } /* end of exponent search */
  462. if ( CURTASK exps[ CURTASK expsc ].nval == (bnumber) 0 )
  463. {
  464. CURTASK exps[ CURTASK expsc ].nval = var_getnval( &mantissa );
  465. }
  466. else
  467. {
  468. CURTASK exps[ CURTASK expsc ].nval = var_getnval( &mantissa )
  469. * pow( (bnumber) 10.0, (bnumber) CURTASK exps[ CURTASK expsc ].nval );
  470. }
  471. break;
  472. case 8: /* octal constant */
  473. /* initialize counters */
  474. CURTASK exps[ CURTASK expsc ].pos_adv = man_start;
  475. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  476. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  477. s_pos = 0;
  478. exponent = OP_NULL;
  479. build_loop = TRUE;
  480. /* loop to build the string */
  481. while ( build_loop == TRUE )
  482. {
  483. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  484. {
  485. case '0': /* or ordinary digit */
  486. case '1':
  487. case '2':
  488. case '3':
  489. case '4':
  490. case '5':
  491. case '6':
  492. case '7':
  493. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  494. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  495. ++s_pos;
  496. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  497. break;
  498. default: /* anything else, terminate */
  499. build_loop = FALSE;
  500. break;
  501. }
  502. }
  503. /* now scan the string to determine the number */
  504. sscanf( CURTASK exps[ CURTASK expsc ].string, "%o", &i );
  505. CURTASK exps[ CURTASK expsc ].nval = (bnumber) i;
  506. break;
  507. case 16: /* hexadecimal constant */
  508. /* initialize counters */
  509. CURTASK exps[ CURTASK expsc ].pos_adv = man_start;
  510. CURTASK exps[ CURTASK expsc ].type = NUMBER;
  511. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  512. s_pos = 0;
  513. exponent = OP_NULL;
  514. build_loop = TRUE;
  515. /* loop to build the string */
  516. while ( build_loop == TRUE )
  517. {
  518. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  519. {
  520. case '0': /* or ordinary digit */
  521. case '1':
  522. case '2':
  523. case '3':
  524. case '4':
  525. case '5':
  526. case '6':
  527. case '7':
  528. case '8':
  529. case '9':
  530. case 'A':
  531. case 'a':
  532. case 'B':
  533. case 'b':
  534. case 'C':
  535. case 'c':
  536. case 'D':
  537. case 'd':
  538. case 'E':
  539. case 'e':
  540. case 'F': /* Don't forget these! (JBV) */
  541. case 'f':
  542. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  543. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance to next character */
  544. ++s_pos;
  545. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  546. break;
  547. default: /* anything else, terminate */
  548. build_loop = FALSE;
  549. break;
  550. }
  551. }
  552. /* now scan the string to determine the number */
  553. sscanf( CURTASK exps[ CURTASK expsc ].string, "%x", &i );
  554. CURTASK exps[ CURTASK expsc ].nval = (bnumber) i;
  555. break;
  556. }
  557. /* note that the operation at this level is now a determined NUMBER */
  558. CURTASK exps[ CURTASK expsc ].operation = NUMBER;
  559. #if INTENSIVE_DEBUG
  560. sprintf( bwb_ebuf, "in exp_numconst(): exit level <%d> precision <%c> value <%lf>",
  561. CURTASK expsc, CURTASK exps[ CURTASK expsc ].type, exp_getnval( &( CURTASK exps[ CURTASK expsc ] ) ) );
  562. bwb_debug( bwb_ebuf );
  563. #endif
  564. #if CHECK_RECURSION
  565. in_use = FALSE;
  566. #endif
  567. return TRUE;
  568. }
  569. /***************************************************************
  570. FUNCTION: exp_function()
  571. DESCRIPTION: This function interprets a function,
  572. calling bwb_exp() (recursively) to resolve any
  573. arguments to the function.
  574. ***************************************************************/
  575. #if ANSI_C
  576. int
  577. exp_function( char *expression )
  578. #else
  579. int
  580. exp_function( expression )
  581. char *expression;
  582. #endif
  583. {
  584. struct exp_ese *e;
  585. int s_pos; /* position in build buffer */
  586. int loop;
  587. int paren_level;
  588. int n_args;
  589. struct bwb_variable *v;
  590. /* struct bwb_variable argv[ MAX_FARGS ]; */ /* Removed by JBV */
  591. struct bwb_variable *argv; /* Added by JBV */
  592. bstring *b;
  593. #if INTENSIVE_DEBUG
  594. char tbuf[ MAXSTRINGSIZE + 1 ];
  595. sprintf( bwb_ebuf, "in exp_function(): entered function, expression <%s>",
  596. expression );
  597. bwb_debug( bwb_ebuf );
  598. #endif
  599. /*-----------------------------------------------------------*/
  600. /* Added by JBV */
  601. /* Required because adding a simple "static" modifier in the */
  602. /* argv declaration doesn't work for recursive calls! */
  603. /*-----------------------------------------------------------*/
  604. if ( ( argv = (struct bwb_variable *) CALLOC( MAX_FARGS,
  605. sizeof( struct bwb_variable ), "exp_function" )) == NULL )
  606. {
  607. bwb_error( err_getmem );
  608. return NULL;
  609. }
  610. /* assign pointers to argument stack */
  611. /* get the function name */
  612. exp_getvfname( expression, CURTASK exps[ CURTASK expsc ].string );
  613. #if INTENSIVE_DEBUG
  614. sprintf( bwb_ebuf, "in exp_function(): name is <%s>.",
  615. CURTASK exps[ CURTASK expsc ].string );
  616. bwb_debug( bwb_ebuf );
  617. #endif
  618. /* now find the function itself */
  619. CURTASK exps[ CURTASK expsc ].function = fnc_find( CURTASK exps[ CURTASK expsc ].string );
  620. /* check to see if it is valid */
  621. if ( CURTASK exps[ CURTASK expsc ].function == NULL )
  622. {
  623. #if PROG_ERRORS
  624. sprintf( bwb_ebuf, "Failed to find function <%s>.",
  625. CURTASK exps[ CURTASK expsc ].string );
  626. bwb_error( bwb_ebuf );
  627. #else
  628. bwb_error( err_uf );
  629. #endif
  630. return OP_ERROR;
  631. }
  632. /* note that this level is a function */
  633. CURTASK exps[ CURTASK expsc ].operation = FUNCTION;
  634. CURTASK exps[ CURTASK expsc ].pos_adv = strlen( CURTASK exps[ CURTASK expsc ].string );
  635. /* check for begin parenthesis */
  636. loop = TRUE;
  637. while( loop == TRUE )
  638. {
  639. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  640. {
  641. case ' ': /* whitespace */
  642. case '\t':
  643. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance */
  644. break;
  645. case '(': /* begin paren */
  646. #if INTENSIVE_DEBUG
  647. sprintf( bwb_ebuf, "in exp_function(): found begin parenthesis." );
  648. bwb_debug( bwb_ebuf );
  649. #endif
  650. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance beyond it */
  651. paren_level = 1; /* set paren_level */
  652. loop = FALSE; /* and break out */
  653. break;
  654. default: /* anything else */
  655. loop = FALSE;
  656. paren_level = 0; /* do not look for arguments */
  657. break;
  658. }
  659. }
  660. /* find arguments within parentheses */
  661. /* for each argument, find a string ending with ',' or with end parenthesis */
  662. n_args = 0;
  663. s_pos = 0;
  664. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  665. while( paren_level > 0 )
  666. {
  667. /* check the current character */
  668. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  669. {
  670. case ',': /* end of an argument */
  671. if ( paren_level == 1 ) /* ignore ',' within parentheses */
  672. {
  673. /* call bwb_exp() recursively to resolve the argument */
  674. if ( exp_validarg( CURTASK exps[ CURTASK expsc ].string ) == TRUE )
  675. {
  676. #if INTENSIVE_DEBUG
  677. sprintf( bwb_ebuf,
  678. "in exp_function(): valid argument (not last)." );
  679. bwb_debug( bwb_ebuf );
  680. #endif
  681. CURTASK exps[ CURTASK expsc ].rec_pos = 0;
  682. e = bwb_exp( CURTASK exps[ CURTASK expsc ].string, FALSE,
  683. &( CURTASK exps[ CURTASK expsc ].rec_pos ) );
  684. /* assign operation and value at this level */
  685. var_make( &( argv[ n_args ] ), e->type );
  686. switch( argv[ n_args ].type )
  687. {
  688. case NUMBER:
  689. * var_findnval( &( argv[ n_args ] ), argv[ n_args ].array_pos )
  690. = exp_getnval( e );
  691. break;
  692. case STRING:
  693. str_btob( var_findsval( &( argv[ n_args ] ),
  694. argv[ n_args ].array_pos ), exp_getsval( e ) );
  695. break;
  696. }
  697. ++n_args; /* increment number of arguments */
  698. }
  699. s_pos = 0; /* reset counter */
  700. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  701. }
  702. else
  703. {
  704. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  705. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  706. ++s_pos;
  707. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  708. }
  709. break;
  710. case '(':
  711. ++paren_level;
  712. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  713. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  714. ++s_pos;
  715. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  716. break;
  717. case ')':
  718. --paren_level;
  719. #if INTENSIVE_DEBUG
  720. sprintf( bwb_ebuf,
  721. "in exp_function(): hit close parenthesis." );
  722. bwb_debug( bwb_ebuf );
  723. #endif
  724. if ( paren_level == 0 )
  725. {
  726. #if INTENSIVE_DEBUG
  727. sprintf( bwb_ebuf,
  728. "in exp_function(): paren level 0." );
  729. bwb_debug( bwb_ebuf );
  730. #endif
  731. /* call bwb_exp() recursively to resolve the argument */
  732. if ( exp_validarg( CURTASK exps[ CURTASK expsc ].string ) == TRUE )
  733. {
  734. #if INTENSIVE_DEBUG
  735. sprintf( bwb_ebuf,
  736. "in exp_function(): valid argument (last)." );
  737. bwb_debug( bwb_ebuf );
  738. #endif
  739. CURTASK exps[ CURTASK expsc ].rec_pos = 0;
  740. e = bwb_exp( CURTASK exps[ CURTASK expsc ].string, FALSE,
  741. &( CURTASK exps[ CURTASK expsc ].rec_pos ) );
  742. #if INTENSIVE_DEBUG
  743. sprintf( bwb_ebuf,
  744. "in exp_function(): return from bwb_exp(), last arg, type <%c> op <%d>",
  745. e->type, e->operation );
  746. bwb_debug( bwb_ebuf );
  747. #endif
  748. /* assign operation and value at this level */
  749. var_make( &( argv[ n_args ] ), e->type );
  750. switch( argv[ n_args ].type )
  751. {
  752. case NUMBER:
  753. * var_findnval( &( argv[ n_args ] ), argv[ n_args ].array_pos )
  754. = exp_getnval( e );
  755. break;
  756. case STRING:
  757. str_btob( var_findsval( &( argv[ n_args ] ),
  758. argv[ n_args ].array_pos ), exp_getsval( e ) );
  759. break;
  760. }
  761. ++n_args; /* increment number of arguments */
  762. }
  763. s_pos = 0; /* reset counter */
  764. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  765. }
  766. else
  767. {
  768. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  769. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  770. ++s_pos;
  771. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  772. }
  773. break;
  774. case '\"': /* embedded string constant */
  775. /* add the initial quotation mark */
  776. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  777. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  778. ++s_pos;
  779. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  780. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  781. /* add intervening characters */
  782. while ( ( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] != '\"' )
  783. && ( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] != '\0' ) )
  784. {
  785. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  786. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  787. ++s_pos;
  788. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  789. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  790. }
  791. /* add the concluding quotation mark */
  792. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  793. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  794. ++s_pos;
  795. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  796. /* the following bracketed out 14 July 1992; since this counter */
  797. /* incremented at the end of the switch statement, this may */
  798. /* increment it past the next character needed */
  799. /* ++CURTASK exps[ CURTASK expsc ].pos_adv; */
  800. break;
  801. default:
  802. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  803. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  804. ++s_pos;
  805. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  806. #if INTENSIVE_DEBUG
  807. sprintf( bwb_ebuf, "in exp_function(): new char <%d>=<%c>",
  808. expression[ CURTASK exps[ CURTASK expsc ].pos_adv ],
  809. expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] );
  810. bwb_debug( bwb_ebuf );
  811. sprintf( bwb_ebuf, "in exp_function(): building <%s>.",
  812. CURTASK exps[ CURTASK expsc ].string );
  813. bwb_debug( bwb_ebuf );
  814. #endif
  815. break;
  816. }
  817. /* advance the counter */
  818. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  819. }
  820. #if INTENSIVE_DEBUG
  821. sprintf( bwb_ebuf, "in exp_function(): ready to call function vector" );
  822. bwb_debug( bwb_ebuf );
  823. #endif
  824. /* call the function vector */
  825. #if INTENSIVE_DEBUG
  826. sprintf( bwb_ebuf, "in exp_function(): calling preset function" );
  827. bwb_debug( bwb_ebuf );
  828. #endif
  829. v = CURTASK exps[ CURTASK expsc ].function->vector ( n_args, &( argv[ 0 ] ),
  830. CURTASK exps[ CURTASK expsc ].function->id );
  831. /*-------------------------------------------------*/
  832. /* Now free the argv memory */
  833. /* (some other less fortunate routine may need it) */
  834. /* JBV, 10/95 */
  835. /*-------------------------------------------------*/
  836. FREE( argv, "exp_function" );
  837. #if INTENSIVE_DEBUG
  838. sprintf( bwb_ebuf, "in exp_function(): return from function vector, type <%c>",
  839. v->type );
  840. bwb_debug( bwb_ebuf );
  841. #endif
  842. /* assign the value at this level */
  843. CURTASK exps[ CURTASK expsc ].type = (char) v->type;
  844. switch( v->type )
  845. {
  846. case STRING:
  847. CURTASK exps[ CURTASK expsc ].operation = CONST_STRING;
  848. #if INTENSIVE_DEBUG
  849. sprintf( bwb_ebuf, "in exp_function(): ready to assign STRING" );
  850. bwb_debug( bwb_ebuf );
  851. #endif
  852. b = var_findsval( v, v->array_pos );
  853. str_btob( exp_getsval( &( CURTASK exps[ CURTASK expsc ] )), b );
  854. #if INTENSIVE_DEBUG
  855. str_btoc( tbuf, b );
  856. sprintf( bwb_ebuf, "in exp_function(): string assigned <%s>", tbuf );
  857. bwb_debug( bwb_ebuf );
  858. #endif
  859. break;
  860. default:
  861. CURTASK exps[ CURTASK expsc ].operation = NUMBER;
  862. CURTASK exps[ CURTASK expsc ].nval = var_getnval( v );
  863. break;
  864. }
  865. #if INTENSIVE_DEBUG
  866. sprintf( bwb_ebuf, "in exp_function(): end of function" );
  867. bwb_debug( bwb_ebuf );
  868. #endif
  869. /* return */
  870. return TRUE;
  871. }
  872. /***************************************************************
  873. FUNCTION: exp_variable()
  874. DESCRIPTION: This function interprets a variable.
  875. ***************************************************************/
  876. #if ANSI_C
  877. int
  878. exp_variable( char *expression )
  879. #else
  880. int
  881. exp_variable( expression )
  882. char *expression;
  883. #endif
  884. {
  885. int pos;
  886. int *pp;
  887. int n_params;
  888. register int n;
  889. struct bwb_variable *v;
  890. bstring *b;
  891. int p;
  892. #if INTENSIVE_DEBUG
  893. sprintf( bwb_ebuf, "in exp_variable(): entered function." );
  894. bwb_debug( bwb_ebuf );
  895. #endif
  896. /* get the variable name */
  897. exp_getvfname( expression, CURTASK exps[ CURTASK expsc ].string );
  898. /* now find the variable itself */
  899. v = CURTASK exps[ CURTASK expsc ].xvar = var_find( CURTASK exps[ CURTASK expsc ].string );
  900. #if INTENSIVE_DEBUG
  901. sprintf( bwb_ebuf, "in exp_variable(): level <%d>, found variable name <%s>",
  902. CURTASK expsc, CURTASK exps[ CURTASK expsc ].xvar->name );
  903. bwb_debug( bwb_ebuf );
  904. #endif
  905. /* note that this level is a variable */
  906. CURTASK exps[ CURTASK expsc ].operation = VARIABLE;
  907. /* read subscripts */
  908. pos = strlen( CURTASK exps[ CURTASK expsc ].string );
  909. if ( ( v->dimensions == 1 ) && ( v->array_sizes[ 0 ] == 1 ))
  910. {
  911. #if INTENSIVE_DEBUG
  912. sprintf( bwb_ebuf, "in exp_variable(): variable <%s> has 1 dimension",
  913. CURTASK exps[ CURTASK expsc ].xvar->name );
  914. bwb_debug( bwb_ebuf );
  915. #endif
  916. pos = strlen( v->name );
  917. n_params = 1;
  918. pp = &p;
  919. pp[ 0 ] = dim_base;
  920. }
  921. else
  922. {
  923. #if INTENSIVE_DEBUG
  924. sprintf( bwb_ebuf, "in exp_variable(): variable <%s> has > 1 dimensions",
  925. CURTASK exps[ CURTASK expsc ].xvar->name );
  926. bwb_debug( bwb_ebuf );
  927. #endif
  928. dim_getparams( expression, &pos, &n_params, &pp );
  929. }
  930. CURTASK exps[ CURTASK expsc ].pos_adv = pos;
  931. for ( n = 0; n < v->dimensions; ++n )
  932. {
  933. CURTASK exps[ CURTASK expsc ].array_pos[ n ] = v->array_pos[ n ] = pp[ n ];
  934. }
  935. #if INTENSIVE_DEBUG
  936. for ( n = 0; n < v->dimensions; ++ n )
  937. {
  938. sprintf( bwb_ebuf, "in exp_variable(): var <%s> array_pos element <%d> is <%d>.",
  939. v->name, n, v->array_pos[ n ] );
  940. bwb_debug( bwb_ebuf );
  941. }
  942. #endif
  943. /* assign the type and value at this level */
  944. CURTASK exps[ CURTASK expsc ].type = (char) v->type;
  945. switch( v->type )
  946. {
  947. case STRING:
  948. b = var_findsval( v, v->array_pos );
  949. #if TEST_BSTRING
  950. sprintf( bwb_ebuf, "in exp_variable(): b string name is <%s>",
  951. b->name );
  952. bwb_debug( bwb_ebuf );
  953. #endif
  954. #if OLDWAY
  955. CURTASK exps[ CURTASK expsc ].sval.length = b->length;
  956. CURTASK exps[ CURTASK expsc ].sval.sbuffer = b->sbuffer;
  957. #endif
  958. str_btob( &( CURTASK exps[ CURTASK expsc ].sval ), b );
  959. break;
  960. default:
  961. CURTASK exps[ CURTASK expsc ].nval = var_getnval( v );
  962. break;
  963. }
  964. #if INTENSIVE_DEBUG
  965. sprintf( bwb_ebuf, "in exp_variable(): exit, name <%s>, level <%d>, op <%d>",
  966. v->name, CURTASK expsc, CURTASK exps[ CURTASK expsc ].operation );
  967. bwb_debug( bwb_ebuf );
  968. #endif
  969. /* return */
  970. return TRUE;
  971. }