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.
 
 
 
 
 
 

1265 lines
38 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. register int i, j; /* JBV */
  594. #if INTENSIVE_DEBUG
  595. char tbuf[ MAXSTRINGSIZE + 1 ];
  596. sprintf( bwb_ebuf, "in exp_function(): entered function, expression <%s>",
  597. expression );
  598. bwb_debug( bwb_ebuf );
  599. #endif
  600. /*-----------------------------------------------------------*/
  601. /* Added by JBV */
  602. /* Required because adding a simple "static" modifier in the */
  603. /* argv declaration doesn't work for recursive calls! */
  604. /*-----------------------------------------------------------*/
  605. if ( ( argv = (struct bwb_variable *) CALLOC( MAX_FARGS,
  606. sizeof( struct bwb_variable ), "exp_function" )) == NULL )
  607. {
  608. bwb_error( err_getmem );
  609. return OP_ERROR;
  610. }
  611. /* assign pointers to argument stack */
  612. /* get the function name */
  613. exp_getvfname( expression, CURTASK exps[ CURTASK expsc ].string );
  614. #if INTENSIVE_DEBUG
  615. sprintf( bwb_ebuf, "in exp_function(): name is <%s>.",
  616. CURTASK exps[ CURTASK expsc ].string );
  617. bwb_debug( bwb_ebuf );
  618. #endif
  619. /* now find the function itself */
  620. CURTASK exps[ CURTASK expsc ].function = fnc_find( CURTASK exps[ CURTASK expsc ].string );
  621. /* check to see if it is valid */
  622. if ( CURTASK exps[ CURTASK expsc ].function == NULL )
  623. {
  624. #if PROG_ERRORS
  625. sprintf( bwb_ebuf, "Failed to find function <%s>.",
  626. CURTASK exps[ CURTASK expsc ].string );
  627. bwb_error( bwb_ebuf );
  628. #else
  629. bwb_error( err_uf );
  630. #endif
  631. return OP_ERROR;
  632. }
  633. /* note that this level is a function */
  634. CURTASK exps[ CURTASK expsc ].operation = FUNCTION;
  635. CURTASK exps[ CURTASK expsc ].pos_adv = strlen( CURTASK exps[ CURTASK expsc ].string );
  636. /* check for begin parenthesis */
  637. loop = TRUE;
  638. while( loop == TRUE )
  639. {
  640. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  641. {
  642. case ' ': /* whitespace */
  643. case '\t':
  644. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance */
  645. break;
  646. case '(': /* begin paren */
  647. #if INTENSIVE_DEBUG
  648. sprintf( bwb_ebuf, "in exp_function(): found begin parenthesis." );
  649. bwb_debug( bwb_ebuf );
  650. #endif
  651. ++CURTASK exps[ CURTASK expsc ].pos_adv; /* advance beyond it */
  652. paren_level = 1; /* set paren_level */
  653. loop = FALSE; /* and break out */
  654. break;
  655. default: /* anything else */
  656. loop = FALSE;
  657. paren_level = 0; /* do not look for arguments */
  658. break;
  659. }
  660. }
  661. /* find arguments within parentheses */
  662. /* for each argument, find a string ending with ',' or with end parenthesis */
  663. n_args = 0;
  664. s_pos = 0;
  665. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  666. while( paren_level > 0 )
  667. {
  668. /* check the current character */
  669. switch( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] )
  670. {
  671. case ',': /* end of an argument */
  672. if ( paren_level == 1 ) /* ignore ',' within parentheses */
  673. {
  674. /* call bwb_exp() recursively to resolve the argument */
  675. if ( exp_validarg( CURTASK exps[ CURTASK expsc ].string ) == TRUE )
  676. {
  677. #if INTENSIVE_DEBUG
  678. sprintf( bwb_ebuf,
  679. "in exp_function(): valid argument (not last)." );
  680. bwb_debug( bwb_ebuf );
  681. #endif
  682. CURTASK exps[ CURTASK expsc ].rec_pos = 0;
  683. e = bwb_exp( CURTASK exps[ CURTASK expsc ].string, FALSE,
  684. &( CURTASK exps[ CURTASK expsc ].rec_pos ) );
  685. /* assign operation and value at this level */
  686. var_make( &( argv[ n_args ] ), e->type );
  687. switch( argv[ n_args ].type )
  688. {
  689. case NUMBER:
  690. * var_findnval( &( argv[ n_args ] ), argv[ n_args ].array_pos )
  691. = exp_getnval( e );
  692. break;
  693. case STRING:
  694. str_btob( var_findsval( &( argv[ n_args ] ),
  695. argv[ n_args ].array_pos ), exp_getsval( e ) );
  696. break;
  697. }
  698. ++n_args; /* increment number of arguments */
  699. }
  700. s_pos = 0; /* reset counter */
  701. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  702. }
  703. else
  704. {
  705. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  706. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  707. ++s_pos;
  708. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  709. }
  710. break;
  711. case '(':
  712. ++paren_level;
  713. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  714. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  715. ++s_pos;
  716. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  717. break;
  718. case ')':
  719. --paren_level;
  720. #if INTENSIVE_DEBUG
  721. sprintf( bwb_ebuf,
  722. "in exp_function(): hit close parenthesis." );
  723. bwb_debug( bwb_ebuf );
  724. #endif
  725. if ( paren_level == 0 )
  726. {
  727. #if INTENSIVE_DEBUG
  728. sprintf( bwb_ebuf,
  729. "in exp_function(): paren level 0." );
  730. bwb_debug( bwb_ebuf );
  731. #endif
  732. /* call bwb_exp() recursively to resolve the argument */
  733. if ( exp_validarg( CURTASK exps[ CURTASK expsc ].string ) == TRUE )
  734. {
  735. #if INTENSIVE_DEBUG
  736. sprintf( bwb_ebuf,
  737. "in exp_function(): valid argument (last)." );
  738. bwb_debug( bwb_ebuf );
  739. #endif
  740. CURTASK exps[ CURTASK expsc ].rec_pos = 0;
  741. e = bwb_exp( CURTASK exps[ CURTASK expsc ].string, FALSE,
  742. &( CURTASK exps[ CURTASK expsc ].rec_pos ) );
  743. #if INTENSIVE_DEBUG
  744. sprintf( bwb_ebuf,
  745. "in exp_function(): return from bwb_exp(), last arg, type <%c> op <%d>",
  746. e->type, e->operation );
  747. bwb_debug( bwb_ebuf );
  748. #endif
  749. /* assign operation and value at this level */
  750. var_make( &( argv[ n_args ] ), e->type );
  751. switch( argv[ n_args ].type )
  752. {
  753. case NUMBER:
  754. * var_findnval( &( argv[ n_args ] ), argv[ n_args ].array_pos )
  755. = exp_getnval( e );
  756. break;
  757. case STRING:
  758. str_btob( var_findsval( &( argv[ n_args ] ),
  759. argv[ n_args ].array_pos ), exp_getsval( e ) );
  760. break;
  761. }
  762. ++n_args; /* increment number of arguments */
  763. }
  764. s_pos = 0; /* reset counter */
  765. CURTASK exps[ CURTASK expsc ].string[ 0 ] = '\0';
  766. }
  767. else
  768. {
  769. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  770. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  771. ++s_pos;
  772. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  773. }
  774. break;
  775. case '\"': /* embedded string constant */
  776. /* add the initial quotation mark */
  777. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  778. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  779. ++s_pos;
  780. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  781. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  782. /* add intervening characters */
  783. while ( ( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] != '\"' )
  784. && ( expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] != '\0' ) )
  785. {
  786. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  787. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  788. ++s_pos;
  789. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  790. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  791. }
  792. /* add the concluding quotation mark */
  793. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  794. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  795. ++s_pos;
  796. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  797. /* the following bracketed out 14 July 1992; since this counter */
  798. /* incremented at the end of the switch statement, this may */
  799. /* increment it past the next character needed */
  800. /* ++CURTASK exps[ CURTASK expsc ].pos_adv; */
  801. break;
  802. default:
  803. CURTASK exps[ CURTASK expsc ].string[ s_pos ]
  804. = expression[ CURTASK exps[ CURTASK expsc ].pos_adv ];
  805. ++s_pos;
  806. CURTASK exps[ CURTASK expsc ].string[ s_pos ] = '\0';
  807. #if INTENSIVE_DEBUG
  808. sprintf( bwb_ebuf, "in exp_function(): new char <%d>=<%c>",
  809. expression[ CURTASK exps[ CURTASK expsc ].pos_adv ],
  810. expression[ CURTASK exps[ CURTASK expsc ].pos_adv ] );
  811. bwb_debug( bwb_ebuf );
  812. sprintf( bwb_ebuf, "in exp_function(): building <%s>.",
  813. CURTASK exps[ CURTASK expsc ].string );
  814. bwb_debug( bwb_ebuf );
  815. #endif
  816. break;
  817. }
  818. /* advance the counter */
  819. ++CURTASK exps[ CURTASK expsc ].pos_adv;
  820. }
  821. #if INTENSIVE_DEBUG
  822. sprintf( bwb_ebuf, "in exp_function(): ready to call function vector" );
  823. bwb_debug( bwb_ebuf );
  824. #endif
  825. /* call the function vector */
  826. #if INTENSIVE_DEBUG
  827. sprintf( bwb_ebuf, "in exp_function(): calling preset function" );
  828. bwb_debug( bwb_ebuf );
  829. #endif
  830. v = CURTASK exps[ CURTASK expsc ].function->vector ( n_args, &( argv[ 0 ] ),
  831. CURTASK exps[ CURTASK expsc ].function->id );
  832. /*-------------------------------------------------*/
  833. /* Now free the argv memory */
  834. /* (some other less fortunate routine may need it) */
  835. /* JBV, 10/95 */
  836. /*-------------------------------------------------*/
  837. /* First kleanup the joint (JBV) */
  838. for ( i = 0; i < n_args; ++i )
  839. {
  840. if ( argv[ i ].memnum != NULL )
  841. {
  842. /* Revised to FREE pass-thru call by JBV */
  843. FREE(argv[ i ].memnum, "exp_function");
  844. argv[ i ].memnum = NULL;
  845. }
  846. if ( argv[ i ].memstr != NULL )
  847. {
  848. /* Remember to deallocate those far-flung branches! (JBV) */
  849. for ( j = 0; j < (int) argv[ i ].array_units; ++j )
  850. {
  851. if ( argv[ i ].memstr[ j ].sbuffer != NULL )
  852. {
  853. /* Revised to FREE pass-thru call by JBV */
  854. FREE( argv[ i ].memstr[ j ].sbuffer, "exp_function" );
  855. argv[ i ].memstr[ j ].sbuffer = NULL;
  856. }
  857. argv[ i ].memstr[ j ].rab = FALSE;
  858. argv[ i ].memstr[ j ].length = 0;
  859. }
  860. /* Revised to FREE pass-thru call by JBV */
  861. FREE( argv[ i ].memstr, "exp_function" );
  862. argv[ i ].memstr = NULL;
  863. }
  864. /* Revised to FREE pass-thru calls by JBV */
  865. if (argv[ i ].array_sizes != NULL)
  866. {
  867. FREE( argv[ i ].array_sizes, "exp_function" );
  868. argv[ i ].array_sizes = NULL; /* JBV */
  869. }
  870. if (argv[ i ].array_pos != NULL)
  871. {
  872. FREE( argv[ i ].array_pos, "exp_function" );
  873. argv[ i ].array_pos = NULL; /* JBV */
  874. }
  875. }
  876. FREE( argv, "exp_function" );
  877. #if INTENSIVE_DEBUG
  878. sprintf( bwb_ebuf, "in exp_function(): return from function vector, type <%c>",
  879. v->type );
  880. bwb_debug( bwb_ebuf );
  881. #endif
  882. /* assign the value at this level */
  883. CURTASK exps[ CURTASK expsc ].type = (char) v->type;
  884. switch( v->type )
  885. {
  886. case STRING:
  887. CURTASK exps[ CURTASK expsc ].operation = CONST_STRING;
  888. #if INTENSIVE_DEBUG
  889. sprintf( bwb_ebuf, "in exp_function(): ready to assign STRING" );
  890. bwb_debug( bwb_ebuf );
  891. #endif
  892. b = var_findsval( v, v->array_pos );
  893. str_btob( exp_getsval( &( CURTASK exps[ CURTASK expsc ] )), b );
  894. #if INTENSIVE_DEBUG
  895. str_btoc( tbuf, b );
  896. sprintf( bwb_ebuf, "in exp_function(): string assigned <%s>", tbuf );
  897. bwb_debug( bwb_ebuf );
  898. #endif
  899. break;
  900. default:
  901. CURTASK exps[ CURTASK expsc ].operation = NUMBER;
  902. CURTASK exps[ CURTASK expsc ].nval = var_getnval( v );
  903. break;
  904. }
  905. #if INTENSIVE_DEBUG
  906. sprintf( bwb_ebuf, "in exp_function(): end of function" );
  907. bwb_debug( bwb_ebuf );
  908. #endif
  909. /* return */
  910. return TRUE;
  911. }
  912. /***************************************************************
  913. FUNCTION: exp_variable()
  914. DESCRIPTION: This function interprets a variable.
  915. ***************************************************************/
  916. #if ANSI_C
  917. int
  918. exp_variable( char *expression )
  919. #else
  920. int
  921. exp_variable( expression )
  922. char *expression;
  923. #endif
  924. {
  925. int pos;
  926. int *pp;
  927. int n_params;
  928. register int n;
  929. struct bwb_variable *v;
  930. bstring *b;
  931. int p;
  932. #if INTENSIVE_DEBUG
  933. sprintf( bwb_ebuf, "in exp_variable(): entered function." );
  934. bwb_debug( bwb_ebuf );
  935. #endif
  936. /* get the variable name */
  937. exp_getvfname( expression, CURTASK exps[ CURTASK expsc ].string );
  938. /* now find the variable itself */
  939. v = CURTASK exps[ CURTASK expsc ].xvar = var_find( CURTASK exps[ CURTASK expsc ].string );
  940. #if INTENSIVE_DEBUG
  941. sprintf( bwb_ebuf, "in exp_variable(): level <%d>, found variable name <%s>",
  942. CURTASK expsc, CURTASK exps[ CURTASK expsc ].xvar->name );
  943. bwb_debug( bwb_ebuf );
  944. #endif
  945. /* note that this level is a variable */
  946. CURTASK exps[ CURTASK expsc ].operation = VARIABLE;
  947. /* read subscripts */
  948. pos = strlen( CURTASK exps[ CURTASK expsc ].string );
  949. if ( ( v->dimensions == 1 ) && ( v->array_sizes[ 0 ] == 1 ))
  950. {
  951. #if INTENSIVE_DEBUG
  952. sprintf( bwb_ebuf, "in exp_variable(): variable <%s> has 1 dimension",
  953. CURTASK exps[ CURTASK expsc ].xvar->name );
  954. bwb_debug( bwb_ebuf );
  955. #endif
  956. pos = strlen( v->name );
  957. n_params = 1;
  958. pp = &p;
  959. pp[ 0 ] = dim_base;
  960. }
  961. else
  962. {
  963. #if INTENSIVE_DEBUG
  964. sprintf( bwb_ebuf, "in exp_variable(): variable <%s> has > 1 dimensions",
  965. CURTASK exps[ CURTASK expsc ].xvar->name );
  966. bwb_debug( bwb_ebuf );
  967. #endif
  968. dim_getparams( expression, &pos, &n_params, &pp );
  969. }
  970. CURTASK exps[ CURTASK expsc ].pos_adv = pos;
  971. for ( n = 0; n < v->dimensions; ++n )
  972. {
  973. CURTASK exps[ CURTASK expsc ].array_pos[ n ] = v->array_pos[ n ] = pp[ n ];
  974. }
  975. #if INTENSIVE_DEBUG
  976. for ( n = 0; n < v->dimensions; ++ n )
  977. {
  978. sprintf( bwb_ebuf, "in exp_variable(): var <%s> array_pos element <%d> is <%d>.",
  979. v->name, n, v->array_pos[ n ] );
  980. bwb_debug( bwb_ebuf );
  981. }
  982. #endif
  983. /* assign the type and value at this level */
  984. CURTASK exps[ CURTASK expsc ].type = (char) v->type;
  985. switch( v->type )
  986. {
  987. case STRING:
  988. b = var_findsval( v, v->array_pos );
  989. #if TEST_BSTRING
  990. sprintf( bwb_ebuf, "in exp_variable(): b string name is <%s>",
  991. b->name );
  992. bwb_debug( bwb_ebuf );
  993. #endif
  994. #if OLDWAY
  995. CURTASK exps[ CURTASK expsc ].sval.length = b->length;
  996. CURTASK exps[ CURTASK expsc ].sval.sbuffer = b->sbuffer;
  997. #endif
  998. str_btob( &( CURTASK exps[ CURTASK expsc ].sval ), b );
  999. break;
  1000. default:
  1001. CURTASK exps[ CURTASK expsc ].nval = var_getnval( v );
  1002. break;
  1003. }
  1004. #if INTENSIVE_DEBUG
  1005. sprintf( bwb_ebuf, "in exp_variable(): exit, name <%s>, level <%d>, op <%d>",
  1006. v->name, CURTASK expsc, CURTASK exps[ CURTASK expsc ].operation );
  1007. bwb_debug( bwb_ebuf );
  1008. #endif
  1009. /* return */
  1010. return TRUE;
  1011. }