|
- /***************************************************************
-
- bwbasic.c Main Program File
- for Bywater BASIC Interpreter
-
- Copyright (c) 1993, Ted A. Campbell
- Bywater Software
-
- email: tcamp@delphi.com
-
- Copyright and Permissions Information:
-
- All U.S. and international rights are claimed by the author,
- Ted A. Campbell.
-
- This software is released under the terms of the GNU General
- Public License (GPL), which is distributed with this software
- in the file "COPYING". The GPL specifies the terms under
- which users may copy and use the software in this distribution.
-
- A separate license is available for commercial distribution,
- for information on which you should contact the author.
-
- ***************************************************************/
-
- /*---------------------------------------------------------------*/
- /* NOTE: Modifications marked "JBV" were made by Jon B. Volkoff, */
- /* 11/1995 (eidetics@cerf.net). */
- /* */
- /* Those additionally marked with "DD" were at the suggestion of */
- /* Dale DePriest (daled@cadence.com). */
- /* */
- /* Version 3.00 by Howard Wulf, AF5NE */
- /* */
- /* Version 3.10 by Howard Wulf, AF5NE */
- /* */
- /* Version 3.20 by Howard Wulf, AF5NE */
- /* */
- /* Version 3.20d by Ken Martin */
- /* */
- /*---------------------------------------------------------------*/
-
-
-
- #include "bwbasic.h"
-
- static void break_handler (void);
- static void break_mes (int x);
- static int bwb_init (void);
- static void bwb_initialize_warnings (void);
- static void bwb_interact (void);
- static int bwb_ladd (char *buffer, LineType * p, int IsUser);
- static void bwb_single_step (char *buffer);
- static void bwb_xtxtline (char *buffer);
- static int bwx_signon (void);
- static void execute_profile (char *FileName);
- static void execute_program (char *FileName);
- static char *FindClassicStatementEnd (char *C);
- static int FixQuotes (char *tbuf);
- static void ImportClassicIfThenElse (char *InBuffer);
- static int is_ln (char *buffer);
- static int is_numconst (char *buffer);
- static void mark_preset_variables (void);
- static FILE *nice_open (char *BaseName);
- static void process_basic_line (char *buffer);
-
- GlobalType *My = NULL;
-
- static char *Banner[] = {
- "######## ## ## ## ## ### ######## ######## ######## ",
- "## ## ## ## ## ## ## ## ## ## ## ## ## ",
- "## ## #### ## ## ## ## ## ## ## ## ## ",
- "######## ## ## ## ## ## ## ## ###### ######## ",
- "## ## ## ## ## ## ######### ## ## ## ## ",
- "## ## ## ## ## ## ## ## ## ## ## ## ",
- "######## ## ### ### ## ## ## ######## ## ## ",
- " ",
- " ######## ### ###### #### ###### ",
- " ## ## ## ## ## ## ## ## ##",
- " ## ## ## ## ## ## ## ",
- " ######## ## ## ###### ## ## ",
- " ## ## ######### ## ## ## ",
- " ## ## ## ## ## ## ## ## ##",
- " ######## ## ## ###### #### ###### ",
- " ",
- "Bywater BASIC Interpreter, version 3.20d ",
- "Copyright (c) 1993, Ted A. Campbell ",
- "Copyright (c) 1995-1997 , Jon B. Volkoff ",
- "Copyright (c) 2014-2017 , Howard Wulf, AF5NE ",
- " 2019-2020 , Ken Martin ",
- " ",
- NULL
- };
-
- #define NUM_WARNINGS 80
-
- static char *ERROR4[NUM_WARNINGS];
-
- static void
- bwb_initialize_warnings (void)
- {
- int i;
- for (i = 0; i < NUM_WARNINGS; i++)
- {
- ERROR4[i] = NULL;
- }
- /* Error code tree */
- ERROR4[1] = "NEXT without FOR";
- ERROR4[2] = "Syntax error";
- ERROR4[3] = "RETURN without GOSUB";
- ERROR4[4] = "Out of DATA";
- ERROR4[5] = "Illegal function call";
- ERROR4[6] = "Overflow";
- ERROR4[7] = "Out of memory";
- ERROR4[8] = "Undefined line";
- ERROR4[9] = "Subscript out of range";
- ERROR4[10] = "Redimensioned array";
- ERROR4[11] = "Division by zero";
- ERROR4[12] = "Illegal direct";
- ERROR4[13] = "Type mismatch";
- ERROR4[14] = "Out of string space";
- ERROR4[15] = "String too long";
- ERROR4[16] = "String formula too complex";
- ERROR4[17] = "Can't continue";
- ERROR4[18] = "Undefined user function";
- ERROR4[19] = "No RESUME";
- ERROR4[20] = "RESUME without error";
- ERROR4[21] = "Unprintable error";
- ERROR4[22] = "Missing operand";
- ERROR4[23] = "Line buffer overflow";
- ERROR4[26] = "FOR without NEXT";
- ERROR4[27] = "Bad DATA";
- ERROR4[29] = "WHILE without WEND";
- ERROR4[30] = "WEND without WHILE";
- ERROR4[31] = "EXIT FUNCTION without FUNCTION";
- ERROR4[32] = "END FUNCTION without FUNCTION";
- ERROR4[33] = "EXIT SUB without SUB";
- ERROR4[34] = "END SUB without SUB";
- ERROR4[35] = "EXIT FOR without FOR";
- ERROR4[50] = "Field overflow";
- ERROR4[51] = "Internal error";
- ERROR4[52] = "Bad file number";
- ERROR4[53] = "File not found";
- ERROR4[54] = "Bad file mode";
- ERROR4[55] = "File already open";
- ERROR4[57] = "Disk I/O error";
- ERROR4[58] = "File already exists";
- ERROR4[61] = "Disk full";
- ERROR4[62] = "Input past end";
- ERROR4[63] = "Bad record number";
- ERROR4[64] = "Bad file name";
- ERROR4[66] = "Direct statement in file";
- ERROR4[67] = "Too many files";
- ERROR4[70] = "Variable Not Declared";
- ERROR4[73] = "Advanced Feature";
- }
-
- /***************************************************************
-
- FUNCTION: bwx_terminate()
-
- DESCRIPTION: This function terminates program execution.
-
- ***************************************************************/
-
- void
- bwx_terminate (void)
- {
-
- exit (0);
- }
-
-
-
- /***************************************************************
-
- FUNCTION: break_handler()
-
- DESCRIPTION: This function is called by break_mes()
- and handles program interruption by break
- (or by the STOP command).
-
- ***************************************************************/
- static void
- break_handler (void)
- {
- /*
- **
- **
- */
- assert( My != NULL );
-
- My->AutomaticLineNumber = 0;
- My->AutomaticLineIncrement = 0;
-
- if (My->IsInteractive)
- {
- /* INTERACTIVE: terminate program */
-
- /* reset all stack counters */
- bwb_clrexec ();
- SetOnError (0);
-
- My->ERR = -1; /* in break_handler() */
-
-
- /* reset the break handler */
- signal (SIGINT, break_mes);
-
- longjmp (My->mark, -1);
-
-
- return;
- }
- /* NOT INTERACTIVE: terminate immediately */
- bwx_terminate ();
- }
-
- /***************************************************************
-
- FUNCTION: break_mes()
-
- DESCRIPTION: This function is called (a) by a SIGINT
- signal or (b) by bwb_STOP via bwx_STOP.
- It prints an error message then calls
- break_handler() to terminate the program.
-
- ***************************************************************/
-
- static void
- break_mes (int x /* Parameter 'x' is never used */ )
- {
- /*
- **
- ** break_mes is FATAL.
- ** x == SIGINT for control-C
- ** x == 0 for bwx_STOP
- **
- */
- assert( My != NULL );
- assert( My->SYSOUT != NULL );
- assert( My->SYSOUT->cfp != NULL );
- assert( My->CurrentVersion != NULL );
-
-
- if (My->ERR < 0) /* in break_mes(), do not make a bad situation worse */
- {
- /* an error has already ben reported */
- }
- else
- {
- fputc ('\n', My->SYSOUT->cfp);
- ResetConsoleColumn ();
- if (My->CurrentVersion->OptionVersionValue & (C77))
- {
- if (is_empty_string (My->ProgramFilename) == FALSE)
- {
- fprintf (My->SYSOUT->cfp, "FILE:%s, ", My->ProgramFilename);
- }
- }
- fprintf (My->SYSOUT->cfp, "Program interrupted");
- if (My->ThisLine) /* break_mes */
- {
- if (My->ThisLine->LineFlags & (LINE_USER)) /* break_mes */
- {
- /* don't print the line number */
- }
- else
- {
- fprintf (My->SYSOUT->cfp, " at line %d", My->ThisLine->number); /* break_mes */
- }
- }
- fputc ('\n', My->SYSOUT->cfp);
- ResetConsoleColumn ();
- fflush (My->SYSOUT->cfp);
- }
- break_handler ();
- }
-
- extern void
- bwx_STOP (int IsShowMessage)
- {
-
- if (IsShowMessage)
- {
- break_mes (0);
- }
- else
- {
- break_handler ();
- }
- }
-
- static int
- bwx_signon (void)
- {
- /*
- **
- ** Display a sign-on banner.
- ** NOT called if a file is provided on the command line.
- **
- */
- int i;
- assert( My != NULL );
- assert( My->SYSOUT != NULL );
- assert( My->SYSOUT->cfp != NULL );
-
-
- for (i = 0; Banner[i] != NULL; i++)
- {
- fprintf (My->SYSOUT->cfp, "%s\n", Banner[i]);
- }
- ResetConsoleColumn ();
- return TRUE;
- }
-
-
-
- DoubleType
- bwx_TIMER (DoubleType Seconds)
- {
- /*
- **
- ** Return a number representing Seconds in the future. Seconds >= 0.
- ** Seconds may be non-integer, such as 0.001 or 86399.999. The
- ** maximum allowed Seconds is MAXDBL. This is used two ways:
- **
- ** 1) in bwb_TIMER(), the ON TIMER count (>0) is used to determine
- ** the expiration time. In this case, simply return what the value
- ** will be in the future. Note that ON TIMER enforces
- ** Seconds > (1 / CLOCKS_PER_SEC), and
- **
- ** 2) in bwb_execline(), zero (=0) is used to determine the current
- ** time and compare it to #1. In this case, simply return what the
- ** value is now.
- **
- ** Both the resolution of the timer, and the frequency of update,
- ** are implementation defined.
- **
- */
-
- if (Seconds < 0)
- {
- WARN_INTERNAL_ERROR;
- return 0;
- }
- else
- {
- DoubleType Result;
- Result = clock ();
- assert (CLOCKS_PER_SEC > 0);
- Result /= CLOCKS_PER_SEC;
- if (Seconds > 0)
- {
- Result += Seconds;
- }
- return Result;
- }
- }
-
- void
- CleanLine (char *buffer)
- {
- /*
- **
- ** cleanup the line, so it is easier to parse
- **
- */
- char *newbuffer;
-
-
-
- if (is_empty_string (buffer))
- {
- /* do nothing */
- return;
- }
-
- /* remove CR/LF */
- newbuffer = bwb_strchr (buffer, '\r');
- if (newbuffer != NULL)
- {
- *newbuffer = NulChar;
- }
- newbuffer = bwb_strchr (buffer, '\n');
- if (newbuffer != NULL)
- {
- *newbuffer = NulChar;
- }
-
- /* remove ALL embedded control characters */
- /* if you want a control character, then use CHR$ */
- newbuffer = buffer;
- while (*newbuffer != NulChar)
- {
- if (bwb_isprint (*newbuffer))
- {
- /* OK */
- }
- else
- {
- *newbuffer = ' ';
- }
- newbuffer++;
- }
- /* LTRIM$ */
- newbuffer = buffer;
- if (*newbuffer != NulChar)
- {
- /* not an empty line, so remove one (or more) leading spaces */
- while (*newbuffer == ' ')
- {
- newbuffer++;
- }
- if (newbuffer > buffer)
- {
- bwb_strcpy (buffer, newbuffer);
- }
- }
- /* RTRIM$ */
- newbuffer = buffer;
- if (*newbuffer != NulChar)
- {
- /* not an empty line, so remove one (or more) trailing spaces */
- char *E;
-
- E = bwb_strchr (newbuffer, NulChar);
- E--;
- while (E >= newbuffer && *E == ' ')
- {
- *E = NulChar;
- E--;
- }
- }
- }
-
- static int
- bwb_init (void)
- {
- /*
- **
- ** initialize Bywater BASIC
- **
- */
- int n;
- static char start_buf[] = "";
- static char end_buf[] = "";
-
- /* Memory allocation */
- if ((My = (GlobalType *) calloc (1, sizeof (GlobalType))) == NULL)
- {
- return FALSE;
- }
- if ((My->MaxLenBuffer =
- (char *) calloc (MAXLEN + 1 /* NulChar */ , sizeof (char))) == NULL)
- {
- return FALSE;
- }
- if ((My->NumLenBuffer =
- (char *) calloc (NUMLEN + 1 /* NulChar */ , sizeof (char))) == NULL)
- {
- return FALSE;
- }
- if ((My->ConsoleOutput =
- (char *) calloc (MAX_LINE_LENGTH + 1 /* NulChar */ ,
- sizeof (char))) == NULL)
- {
- return FALSE;
- }
- if ((My->ConsoleInput =
- (char *) calloc (MAX_LINE_LENGTH + 1 /* NulChar */ ,
- sizeof (char))) == NULL)
- {
- return FALSE;
- }
- if ((My->SYSIN = (FileType *) calloc (1, sizeof (FileType))) == NULL)
- {
- return FALSE;
- }
- if ((My->SYSOUT = (FileType *) calloc (1, sizeof (FileType))) == NULL)
- {
- return FALSE;
- }
- if ((My->SYSPRN = (FileType *) calloc (1, sizeof (FileType))) == NULL)
- {
- return FALSE;
- }
- if ((My->StartMarker = (LineType *) calloc (1, sizeof (LineType))) == NULL)
- {
- return FALSE;
- }
- if ((My->UserMarker = (LineType *) calloc (1, sizeof (LineType))) == NULL)
- {
- return FALSE;
- }
- if ((My->EndMarker = (LineType *) calloc (1, sizeof (LineType))) == NULL)
- {
- return FALSE;
- }
- if ((My->EndMarker = (LineType *) calloc (1, sizeof (LineType))) == NULL)
- {
- return FALSE;
- }
- if ((My->ERROR4 =
- (char *) calloc (MAX_ERR_LENGTH + 1 /* NulChar */ ,
- sizeof (char))) == NULL)
- {
- return FALSE;
- }
-
- My->CurrentVersion = &bwb_vertable[0];
- My->IsInteractive = TRUE;
- My->OptionSleepDouble = 1;
- My->OptionIndentInteger = 2;
- My->OptionTerminalType = C_OPTION_TERMINAL_NONE;
- My->OptionRoundType = C_OPTION_ROUND_BANK;
- My->NextValidLineNumber = MINLIN;
- My->IncludeLevel = 0; /* %INCLUDE */
- My->IsCommandLineFile = FALSE;
- My->ExternalInputFile = NULL; /* for automated testing, --TAPE command line parameter */
- My->IsPrinter = FALSE; /* CBASIC-II: CONSOLE and LPRINTER commands */
- My->OptionPromptString = DEF_PROMPT;
- My->OptionEditString = DEF_EDITOR;
- My->OptionFilesString = DEF_FILES;
- My->OptionRenumString = DEF_RENUM;
- My->OptionExtensionString = DEF_EXTENSION;
- My->OptionScaleInteger = 0;
- My->OptionDigitsInteger = SIGNIFICANT_DIGITS;
- My->OptionZoneInteger = ZONE_WIDTH;
- My->UseParameterString = NULL;
- My->ProgramFilename = NULL;
-
- My->StartMarker->number = MINLIN - 1;
- My->StartMarker->next = My->EndMarker;
- My->StartMarker->position = 0;
- My->StartMarker->buffer = start_buf;
-
- My->EndMarker->number = MAXLIN + 1;
- My->EndMarker->next = My->EndMarker;
- My->EndMarker->position = 0;
- My->EndMarker->buffer = end_buf;
-
- My->UserMarker->number = MINLIN - 1;
- My->UserMarker->next = My->EndMarker;
- My->UserMarker->position = 0;
- My->UserMarker->buffer = NULL;
-
- My->DataLine = My->EndMarker;
- My->DataPosition = 0;
-
- My->StackHead = NULL;
- My->StackDepthInteger = 0;
-
- My->FieldHead = NULL;
-
- My->VirtualHead = NULL;
- My->FileHead = NULL;
- My->ThisLine = My->StartMarker; /* bwb_init */
-
- My->SYSIN->DevMode = DEVMODE_INPUT;
- My->SYSIN->width = 80;
- My->SYSIN->col = 1;
- My->SYSIN->row = 1;
- My->SYSIN->delimit = ',';
- My->SYSIN->cfp = stdin;
-
- My->SYSOUT->DevMode = DEVMODE_OUTPUT;
- My->SYSOUT->width = 80;
- My->SYSOUT->col = 1;
- My->SYSOUT->row = 1;
- My->SYSOUT->delimit = ',';
- My->SYSOUT->cfp = stdout;
-
- My->SYSPRN->DevMode = DEVMODE_OUTPUT;
- My->SYSPRN->width = 80;
- My->SYSPRN->col = 1;
- My->SYSPRN->row = 1;
- My->SYSPRN->delimit = ',';
- My->SYSPRN->cfp = stderr;
-
- My->LPRINT_NULLS = 0;
- My->SCREEN_ROWS = 24;
-
- /* OPEN #0 is an ERROR. */
- /* CLOSE #0 is an ERROR. */
- /* WIDTH #0, 80 is the same as WIDTH 80. */
- /* LPRINT and LLIST are sent to bwx_LPRINT() */
-
- /* default variable type */
- for (n = 0; n < 26; n++)
- {
- My->DefaultVariableType[n] = DoubleTypeCode;
- }
- /* default COMMAND$(0-9) */
- for (n = 0; n < 10; n++)
- {
- My->COMMAND4[n] = NULL;
- }
-
- /* initialize tables of variables, functions */
- bwb_initialize_warnings ();
- SortAllCommands ();
- SortAllFunctions ();
- SortAllOperators ();
- var_init ();
- IntrinsicFunction_init ();
- UserFunction_init ();
- OptionVersionSet (0);
- /* OK */
- return TRUE;
- }
-
- /***************************************************************
-
- FUNCTION: main()
-
- DESCRIPTION: As in any C program, main() is the basic
- function from which the rest of the
- program is called. Some environments,
- however, provide their own main() functions
- (Microsoft Windows (tm) is an example).
- In these cases, the following code will
- have to be included in the initialization
- function that is called by the environment.
-
- ***************************************************************/
-
- static void
- process_basic_line (char *buffer)
- {
- CleanLine (buffer);
- if (is_empty_string (buffer))
- {
- /* empty -- do nothing */
- }
- else if (is_ln (buffer) == FALSE)
- {
- /* If there is no line number, then execute the line as received */
- /* RUN */
- WARN_CLEAR; /* process_basic_line */
- SetOnError (0);
- bwb_xtxtline (buffer); /* process_basic_line */
- }
- else if (is_numconst (buffer) == TRUE)
- {
- /*-----------------------------------------------------------------*/
- /* Another possibility: if buffer is a numeric constant, */
- /* then delete the indicated line number (JBV) */
- /*-----------------------------------------------------------------*/
- /* 100 */
- int LineNumber;
- LineNumber = atoi (buffer);
- WARN_CLEAR; /* process_basic_line */
- SetOnError (0);
- sprintf (buffer, "delete %d", LineNumber);
- bwb_xtxtline (buffer); /* process_basic_line */
- }
- else
- {
- /* If there is a line number, then add it to the BASIC program */
- /* 100 REM */
- bwb_ladd (buffer, My->StartMarker, FALSE);
- }
- }
- static void
- bwb_single_step (char *buffer)
- {
- assert( My != NULL );
- assert (buffer != NULL);
-
- process_basic_line (buffer);
- while (My->StackHead != NULL)
- {
- bwb_execline ();
- }
- }
-
- static void
- mark_preset_variables (void)
- {
- /* mark all existing variables as preset */
- /* this includes all variables declared in any PROFILE */
- VariableType *v;
- assert( My != NULL );
-
-
- for (v = My->VariableHead; v != NULL; v = v->next)
- {
- v->VariableFlags |= VARIABLE_PRESET;
- v->VariableFlags |= VARIABLE_COMMON;
- }
- }
-
-
- static void
- execute_profile (char *FileName)
- {
- FILE *profile;
- assert( My != NULL );
- assert (FileName != NULL);
-
- My->NextValidLineNumber = MINLIN;
-
- #ifdef LINUX
- /* Begin 20200806 ChipMaster@YeOlPiShack.net Patch --
- *
- * on *nix check the current folder, user's home folder and then /etc for
- * the profile script
- *
- * WARN: I assume that $HOME and FileName are reasonably sized. If one
- * or the other are oversized then an attempt to load a weird-named
- * script will happen.
- *
- * 1 to open [profile.bas]
- * 2 to open [/home/$USER/profile.bas]
- * 3 to open [/etc/profile.bas]
- * */
- char pname[1024];
- pname[0]=pname[sizeof(pname)-1]=0;
- profile = fopen (FileName, "r");
- if(profile == NULL) {
- strncpy(pname, getenv("HOME"), sizeof(pname)-1);
- if(pname[0]) {
- strncat(strcat(pname, "/"), FileName, sizeof(pname)-strlen(pname)-1);
- profile = fopen (pname, "r");
- }
- }
- if(profile == NULL) {
- strncat(strcpy(pname, "/etc/"), FileName, sizeof(pname)-6);
- profile = fopen(pname, "r");
- }
- #else
- profile = fopen (FileName, "r");
- #endif
- /* End 20200806 Patch */
-
- if (profile == NULL)
- {
- /* NOT FOUND */
- /* OPTIONAL */
- return;
- }
- /* FOUND */
- if (My->IsInteractive)
- {
- /*
- **
- ** set a buffer for jump: program execution returns to this point in
- ** case of a jump (error, interrupt, or finish program)
- **
- */
- My->program_run = 0;
- signal (SIGINT, break_mes);
- setjmp (My->mark);
- if (My->program_run > 0)
- {
- /* error in PROFILE */
- exit (1);
- }
- My->program_run++;
- }
-
- /*
- The profile only exists to allow executing OPTION ... commands.
- No other use is supported.
- */
-
- {
- char *tbuf;
- int tlen;
-
- tbuf = My->ConsoleInput;
- tlen = MAX_LINE_LENGTH;
- while (fgets (tbuf, tlen, profile)) /* execute_profile */
- {
- tbuf[tlen] = NulChar;
- bwb_single_step (tbuf); /* in execute_profile() */
- }
- bwb_fclose (profile);
- mark_preset_variables ();
- }
- }
-
-
-
- static void
- execute_program (char *FileName)
- {
- assert( My != NULL );
- assert( My->SYSOUT != NULL );
- assert( My->SYSOUT->cfp != NULL );
-
- assert (FileName != NULL);
-
- My->NextValidLineNumber = MINLIN;
- My->IsCommandLineFile = TRUE;
- if (bwb_fload (FileName) == FALSE)
- {
- fprintf (My->SYSOUT->cfp, "Failed to open file %s\n", FileName);
- /* the file load has failed, so do NOT run the program */
- exit (1);
- }
- if (My->ERR < 0 /* in execute_program(), file load failed */ )
- {
- /* the file load has failed, so do NOT run the program */
- exit (1);
- }
- bwb_RUN (My->StartMarker);
- }
-
- extern int
- main (int argc, char **argv)
- {
- int i;
- assert (argc >= 0);
- assert (argv != NULL);
-
- if (bwb_init () == FALSE)
- {
- /* FATAL */
- puts ("Out of memory");
- return 1;
- }
- assert( My != NULL );
- assert( My->SYSOUT != NULL );
- assert( My->SYSOUT->cfp != NULL );
- assert( My->SYSIN != NULL );
- assert( My->SYSIN->cfp != NULL );
-
- /* Signon message banner */
- if (argc < 2)
- {
- /* no parameters */
- if (My->IsInteractive)
- {
- bwx_signon ();
- }
- else
- {
- /* if INTERACTIVE is FALSE, then we must have a program file */
- fputs ("Program file not specified\n", My->SYSOUT->cfp);
- return 1;
- }
- }
-
- if (My->IsInteractive)
- {
- /*
- **
- ** set a buffer for jump: program execution returns to this point in
- ** case of a jump (error, interrupt, or finish program)
- **
- */
- My->program_run = 0;
- signal (SIGINT, break_mes);
- setjmp (My->mark);
- if (My->program_run > 0)
- {
- /* error in profile */
- return 1;
- }
- My->program_run++;
- }
-
- #if PROFILE
- execute_profile (PROFILENAME);
- #endif
-
- /* check to see if there is a program file: but do
- * this only the first time around! */
- for (i = 1; i < argc; i++)
- {
- /*
- SYNTAX:
- bwbasic [ --profile profile.bas ] [ --tape tapefile.inp ] [ program.bas ]
- */
- if (bwb_stricmp (argv[i], "--profile") == 0
- || bwb_stricmp (argv[i], "-p") == 0
- || bwb_stricmp (argv[i], "/profile") == 0
- || bwb_stricmp (argv[i], "/p") == 0)
- {
- i++;
- if (i < argc)
- {
- /* --profile profile.bas */
- execute_profile (argv[i]);
- }
- }
- else
- if (bwb_stricmp (argv[i], "--tape") == 0
- || bwb_stricmp (argv[i], "-t") == 0
- || bwb_stricmp (argv[i], "/tape") == 0
- || bwb_stricmp (argv[i], "/t") == 0)
- {
- i++;
- if (i < argc)
- {
- /* --tape tapefile.inp */
- My->ExternalInputFile = fopen (argv[i], "r");
- }
- }
- else
- {
- /* program.bas */
- {
- int j;
- int n;
-
- j = i;
- for (n = 0; n < 10 && j < argc; n++, j++)
- {
- My->COMMAND4[n] = argv[j];
- }
- }
- execute_program (argv[i]);
- break;
- }
- }
-
- if (My->IsInteractive)
- {
- /*
- **
- ** set a buffer for jump: program execution returns to this point in
- ** case of a jump (error, interrupt, or finish program)
- **
- */
- My->program_run = 0;
- signal (SIGINT, break_mes);
- setjmp (My->mark);
- if (My->program_run > 0)
- {
- /* error in console mode */
- }
- My->program_run++;
- }
-
- /* main program loop */
- My->NextValidLineNumber = MINLIN;
- while (!feof (My->SYSIN->cfp)) /* condition !feof( My->SYSIN->cfp ) added in v1.11 */
- {
- bwb_mainloop ();
- }
-
- bwx_terminate (); /* allow ^D (Unix) exit with grace */
-
- return 0;
- }
-
-
-
- /***************************************************************
-
- FUNCTION: bwb_interact()
-
- DESCRIPTION: This function gets a line from the user
- and processes it.
-
- ***************************************************************/
-
- static void
- bwb_interact (void)
- {
- char *tbuf;
- int tlen;
- assert( My != NULL );
-
-
- tbuf = My->ConsoleInput;
- tlen = MAX_LINE_LENGTH;
- My->NextValidLineNumber = MINLIN;
- /* take input from keyboard */
- if (My->AutomaticLineNumber > 0 && My->AutomaticLineIncrement > 0)
- {
- /* AUTO 100, 10 */
- char *zbuf; /* end of the prompt, start of the response */
- int zlen; /* length of the prompt */
- char LineExists;
- LineType *l;
-
- LineExists = ' ';
- for (l = My->StartMarker->next; l != My->EndMarker; l = l->next)
- {
- if (l->number == My->AutomaticLineNumber)
- {
- /* FOUND */
- LineExists = '*';
- break;
- }
- else if (l->number > My->AutomaticLineNumber)
- {
- /* NOT FOUND */
- LineExists = ' ';
- break;
- }
- }
- sprintf (tbuf, "%d%c", My->AutomaticLineNumber, LineExists);
- zbuf = bwb_strchr (tbuf, NulChar);
- zlen = bwb_strlen (tbuf);
- bwx_input (tbuf, FALSE, zbuf, tlen - zlen);
- zbuf[-1] = ' '; /* remove LineExists indicator */
- CleanLine (zbuf); /* JBV */
- if (is_empty_string (zbuf))
- {
- /* empty response */
- if (LineExists == '*')
- {
- /*
- An empty response with an existing line,
- causes AUTO to continue with the next line,
- leaving the current line intact.
- */
- My->AutomaticLineNumber += My->AutomaticLineIncrement;
- }
- else
- {
- /*
- An empty response with a non-existing line,
- causes AUTO to terminate.
- */
- My->AutomaticLineNumber = 0;
- My->AutomaticLineIncrement = 0;
- }
- }
- else
- {
- /* non-empty response */
- if (bwb_stricmp (zbuf, "MAN") == 0)
- {
- /* MAN terminates AUTO mode */
- My->AutomaticLineNumber = 0;
- My->AutomaticLineIncrement = 0;
- }
- else
- {
- /* overwrite any existing line */
- bwb_ladd (tbuf, My->StartMarker, FALSE);
- My->AutomaticLineNumber += My->AutomaticLineIncrement;
- }
- }
- }
- else
- {
- bwx_input (My->OptionPromptString, FALSE, tbuf, tlen);
- process_basic_line (tbuf);
- }
- }
-
-
-
- /***************************************************************
-
- FUNCTION: bwb_fload()
-
- DESCRIPTION: This function loads a BASIC program
- file into memory given a FILE pointer.
-
- ***************************************************************/
-
- static int
- FixQuotes (char *tbuf)
- {
- /* fix unbalanced quotes */
- /* 'tbuf' shall be declared "char tbuf[ tlen + 1]". */
- int p;
- int QuoteCount;
- int tlen;
- assert( My != NULL );
- assert( My->CurrentVersion != NULL );
-
- assert (tbuf != NULL);
-
- QuoteCount = 0;
- tlen = MAX_LINE_LENGTH;
- tbuf[tlen] = NulChar;
- p = 0;
- while (tbuf[p])
- {
- if (tbuf[p] == My->CurrentVersion->OptionQuoteChar)
- {
- QuoteCount++;
- }
- p++;
- }
- if (QuoteCount & 1)
- {
- /* odd == missing trailing quote */
- if (p < tlen)
- {
- /* we have room to put the missig quote */
- tbuf[p] = My->CurrentVersion->OptionQuoteChar;
- p++;
- tbuf[p] = NulChar;
- }
- else
- {
- /* we cannot fix it */
- return FALSE;
- }
- }
- /* OK */
- return TRUE;
- }
-
- static FILE *
- nice_open (char *BaseName)
- {
- FILE *file;
- assert( My != NULL );
-
- if (BaseName == NULL)
- {
- BaseName = My->ProgramFilename;
- }
- if (is_empty_string (BaseName))
- {
- WARN_BAD_FILE_NAME;
- return NULL;
- }
-
- file = fopen (BaseName, "r");
- if (file == NULL)
- if (is_empty_string (My->OptionExtensionString) == FALSE)
- {
- char *FileName;
-
- FileName = bwb_strdup2 (BaseName, My->OptionExtensionString);
- if (FileName == NULL)
- {
- WARN_OUT_OF_MEMORY;
- return NULL;
- }
- file = fopen (FileName, "r");
- if (file == NULL)
- {
- free (FileName);
- }
- else if (BaseName == My->ProgramFilename)
- {
- if (My->ProgramFilename != NULL)
- {
- free (My->ProgramFilename);
- }
- My->ProgramFilename = FileName;
- }
- }
- return file;
- }
-
- extern int
- bwb_fload (char *FileName)
- {
- /*
- **
- ** Load a BASIC program from the specified 'FileName'.
- ** If 'FileName' is NULL, then load from My->ProgramFilename,
- **
- */
- char *Magic_Word; /* SYNTAX: %INCLUDE literal.file.name */
- int Magic_Length;
- FILE *file;
- char *tbuf;
- int tlen;
-
- Magic_Word = "%INCLUDE "; /* SYNTAX: %INCLUDE literal.file.name */
- Magic_Length = bwb_strlen (Magic_Word);
- tbuf = My->MaxLenBuffer;
- tlen = MAXLEN;
-
- /*
- Just in case you are wondering...
- Although this handles the most common cases, it does not
- handle all possible cases.
- The correct solution is to provide the actual filename (with extension),
- as it exists in the operating system.
- */
-
- file = nice_open (FileName);
- if (file == NULL)
- {
- WARN_BAD_FILE_NAME;
- return FALSE;
- }
- My->NextValidLineNumber = MINLIN;
- while (fgets (tbuf, tlen, file)) /* bwb_fload */
- {
- tbuf[tlen] = NulChar;
- CleanLine (tbuf);
- if (is_empty_string (tbuf))
- {
- /* ignore */
- }
- else if (bwb_strnicmp (tbuf, Magic_Word, Magic_Length) == 0)
- {
- /* %iNCLUDE */
- int Result;
- int p;
- char varname[NameLengthMax + 1];
-
- p = Magic_Length;
- if (buff_read_varname (tbuf, &p, varname) == FALSE)
- {
- fprintf (My->SYSOUT->cfp, "%s Filename\n", Magic_Word);
- ResetConsoleColumn ();
- return FALSE;
- }
- My->IncludeLevel++; /* %INCLUDE */
- Result = bwb_fload (varname);
- My->IncludeLevel--; /* %INCLUDE */
- if (Result == FALSE)
- {
- fprintf (My->SYSOUT->cfp, "%s %s Failed\n", Magic_Word, varname);
- ResetConsoleColumn ();
- return FALSE;
- }
- }
- else
- {
- /* normal */
- bwb_ladd (tbuf, My->StartMarker, FALSE);
- }
- }
-
- /* close file stream */
-
- bwb_fclose (file); /* file != NULL */
-
- My->NextValidLineNumber = MINLIN;
-
- return TRUE;
- }
-
-
- static char *
- FindClassicStatementEnd (char *C)
- {
- /*
- ** find the end of the current statement
- */
- assert( My != NULL );
- assert( My->CurrentVersion != NULL );
-
- assert (C != NULL);
-
-
- if (My->CurrentVersion->OptionStatementChar == NulChar
- && My->CurrentVersion->OptionCommentChar == NulChar)
- {
- /* DO NOTHING: Multi-statment lines are not possible */
- return NULL;
- }
- /* skip line number */
- while (bwb_isdigit (*C))
- {
- C++;
- }
- /* skip spaces */
- while (*C == ' ')
- {
- C++;
- }
- if (IS_CHAR (*C, My->CurrentVersion->OptionCommentChar))
- {
- /* The entire line is a comment */
- return NULL;
- }
- if (bwb_strnicmp (C, "REM", 3) == 0)
- {
- /* The entire line is a comment */
- return NULL;
- }
- if ((My->CurrentVersion->OptionFlags & OPTION_LABELS_ON)
- && (My->CurrentVersion->OptionStatementChar != NulChar))
- {
- /* determine if this line is a LABEL */
- int p;
- char label[NameLengthMax + 1];
-
- p = 0;
- if (buff_read_label (C, &p, label))
- {
- if (buff_skip_char (C, &p, My->CurrentVersion->OptionStatementChar))
- {
- if (buff_is_eol (C, &p))
- {
- /* The entire line is a label */
- /* LABEL : \0 */
- return NULL;
- }
- }
- }
- }
- /* not a special case, so split on the first unquoted
- * OptionCommentChar or OptionStatementChar */
- while (*C != NulChar)
- {
- if (*C == My->CurrentVersion->OptionQuoteChar)
- {
- /* skip leading quote */
- C++;
- while (*C != NulChar && *C != My->CurrentVersion->OptionQuoteChar)
- {
- /* skip string constant */
- C++;
- }
- if (*C == My->CurrentVersion->OptionQuoteChar)
- {
- /* skip trailing quote */
- C++;
- }
- }
- else if (IS_CHAR (*C, My->CurrentVersion->OptionCommentChar) /* ', ! */ )
- {
- /* FOUND */
- return C;
- }
- else
- if (IS_CHAR (*C, My->CurrentVersion->OptionStatementChar) /* :, \ */ )
- {
- /* FOUND */
- return C;
- }
- else
- {
- C++;
- }
- }
- /* NOT FOUND */
- return NULL;
- }
-
-
- static void
- ImportClassicIfThenElse (char *InBuffer)
- {
- /*
- **
- ** Determine the type of IF command:
- **
- ** a) STANDARD:
- ** IF x THEN line ELSE line
- **
- ** b) CLASSIC:
- ** IF x THEN stmt(s) ELSE stmt(s)
- **
- ** c) STRUCTURED:
- ** IF x THEN
- ** stmts
- ** ELSE
- ** stmts
- ** END IF
- **
- ** The STANDARD and STRUCTURED forms
- ** are natively supported.
- **
- ** The CLASSIC form is converted to
- ** the STRUCTURED form.
- **
- */
-
- int i;
-
- int nIF = 0;
- int nTHEN = 0;
- int nELSE = 0;
- int nENDIF = 0;
-
- #define NO_COMMAND 0
- #define IF_COMMAND 1
- #define THEN_COMMAND 2
- #define ELSE_COMMAND 3
- #define ENDIF_COMMAND 4
- int LastCommand = NO_COMMAND;
-
- const char *REM = "REM ";
- const char *IF = "IF ";
- const char *THEN = "THEN ";
- const char *THEN2 = "THEN";
- const char *ELSE = "ELSE ";
- const char *ENDIF = "END IF";
- const char *GOTO = "GOTO ";
- const char *DATA = "DATA ";
- const char *CASE = "CASE ";
- char *OutBuffer = My->ConsoleOutput;
- char *Input;
- char *Output;
- char LastChar = My->CurrentVersion->OptionStatementChar;
-
- int REM_len = bwb_strlen (REM);
- int IF_len = bwb_strlen (IF);
- int THEN_len = bwb_strlen (THEN);
- int THEN2_len = bwb_strlen (THEN2);
- int ELSE_len = bwb_strlen (ELSE);
- int ENDIF_len = bwb_strlen (ENDIF);
- int GOTO_len = bwb_strlen (GOTO);
- int DATA_len = bwb_strlen (DATA);
- int CASE_len = bwb_strlen (CASE);
-
- #define OUTPUTCHAR( c ) { *Output = c; Output++; }
- #define COPYCHAR { LastChar = *Input; *Output = *Input; Output++; Input++; }
- #define COPY_LINENUMBER while( bwb_isdigit( *Input ) ) COPYCHAR;
- #define COPY_SPACES while( *Input == ' ' ) COPYCHAR;
- #define COPY_IF for( i = 0; i < IF_len; i++ ) COPYCHAR;
- #define COPY_THEN for( i = 0; i < THEN_len; i++ ) COPYCHAR;
- #define COPY_THEN2 for( i = 0; i < THEN2_len; i++ ) COPYCHAR;
- #define COPY_ELSE for( i = 0; i < ELSE_len; i++ ) COPYCHAR;
- #define COPY_ENDIF for( i = 0; i < ENDIF_len; i++ ) COPYCHAR;
- #define FORCE_ENDIF for( i = 0; i < ENDIF_len; i++ ) OUTPUTCHAR( ENDIF[ i ] );
- #define FORCE_GOTO for( i = 0; i < GOTO_len; i++ ) OUTPUTCHAR( GOTO[ i ] );
- #define FORCE_COLON if( LastChar != My->CurrentVersion->OptionStatementChar ) OUTPUTCHAR( My->CurrentVersion->OptionStatementChar );
-
- assert( My != NULL );
- assert( My->CurrentVersion != NULL );
- assert (InBuffer != NULL);
-
-
- Input = InBuffer;
- Output = OutBuffer;
-
- if (My->CurrentVersion->OptionStatementChar == NulChar)
- {
- /* DO NOTHING: All IFs must be STANDARD or STRUCTURED */
- return;
- }
-
-
- COPY_LINENUMBER;
- COPY_SPACES;
- LastChar = My->CurrentVersion->OptionStatementChar;
-
-
- while (*Input != NulChar)
- {
- if (*Input == My->CurrentVersion->OptionCommentChar)
- {
- /* comment */
- break;
- }
- else if (*Input == My->CurrentVersion->OptionQuoteChar)
- {
- /* string constant */
- COPYCHAR;
- while (*Input != NulChar
- && *Input != My->CurrentVersion->OptionQuoteChar)
- {
- COPYCHAR;
- }
- if (*Input == My->CurrentVersion->OptionQuoteChar)
- {
- COPYCHAR;
- }
- else
- {
- /* add missing Quote */
- OUTPUTCHAR (My->CurrentVersion->OptionQuoteChar);
- }
- COPY_SPACES;
- }
- else if (bwb_isalnum (LastChar))
- {
- /* can NOT be the start of a command */
- COPYCHAR;
- }
- else if (!bwb_isalpha (*Input))
- {
- /* can NOT be the start of a command */
- COPYCHAR;
- }
- else if (bwb_strnicmp (Input, REM, REM_len) == 0)
- {
- break;
- }
- else if (bwb_strnicmp (Input, DATA, DATA_len) == 0)
- {
- /* DATA ... */
- break;
- }
- else if (bwb_strnicmp (Input, CASE, CASE_len) == 0)
- {
- /* CASE ... */
- break;
- }
- else if (bwb_strnicmp (Input, IF, IF_len) == 0)
- {
- /* IF */
- LastCommand = IF_COMMAND;
- nIF++;
- COPY_IF;
- COPY_SPACES;
- }
- else if (bwb_strnicmp (Input, GOTO, GOTO_len) == 0 && nIF > nTHEN)
- {
- /* IF x GOTO line ELSE line */
- LastCommand = THEN_COMMAND;
- nTHEN++;
- COPY_THEN;
- COPY_SPACES;
- COPY_LINENUMBER;
- COPY_SPACES;
- if (bwb_strnicmp (Input, ELSE, ELSE_len) == 0)
- {
- /* ELSE line */
- COPY_ELSE;
- COPY_SPACES;
- COPY_LINENUMBER;
- COPY_SPACES;
- }
- /* IS STANDARD, NOT CLASSIC */
- nENDIF++;
- LastCommand = ENDIF_COMMAND;
- }
- else if (bwb_strnicmp (Input, THEN, THEN_len) == 0)
- {
- /* THEN */
- LastCommand = THEN_COMMAND;
- nTHEN++;
- COPY_THEN;
- COPY_SPACES;
- if (bwb_isdigit (*Input))
- {
- /* STANDARD: IF x THEN line ELSE line */
- char *SavedInput;
- char *SavedOutput;
- SavedInput = Input;
- SavedOutput = Output;
-
- COPY_LINENUMBER;
- COPY_SPACES;
- if (bwb_strnicmp (Input, ELSE, ELSE_len) == 0)
- {
- /* ELSE line */
- COPY_ELSE;
- COPY_SPACES;
- if (bwb_isdigit (*Input))
- {
- COPY_LINENUMBER;
- COPY_SPACES;
- /* IS STANDARD, NOT CLASSIC */
- nENDIF++;
- LastCommand = ENDIF_COMMAND;
- }
- else
- {
- /* IF x THEN line ELSE stmts */
- Input = SavedInput;
- Output = SavedOutput;
- FORCE_COLON;
- FORCE_GOTO;
- COPY_LINENUMBER;
- COPY_SPACES;
- }
- }
- else
- {
- /* IS STANDARD, NOT CLASSIC */
- nENDIF++;
- LastCommand = ENDIF_COMMAND;
- }
- }
- else
- if (*Input == My->CurrentVersion->OptionCommentChar
- || *Input == NulChar)
- {
- /* IS STRUCTURED, NOT CLASSIC */
- nENDIF++;
- LastCommand = ENDIF_COMMAND;
- }
- else
- {
- /* CLASSIC: IF x THEN stmts ELSE stmts */
- }
- FORCE_COLON;
- }
- else if (bwb_strnicmp (Input, THEN2, THEN2_len) == 0)
- {
- /* trailing THEN ? */
- char *PeekInput;
-
- PeekInput = Input;
- PeekInput += THEN2_len;
- while (*PeekInput == ' ')
- {
- PeekInput++;
- }
- if (*PeekInput == My->CurrentVersion->OptionCommentChar
- || *PeekInput == NulChar)
- {
- /* IS STRUCTURED, NOT CLASSIC */
- nTHEN++;
- COPY_THEN2;
- nENDIF++;
- LastCommand = ENDIF_COMMAND;
- FORCE_COLON;
- }
- else
- {
- /* THEN line */
- }
- }
- else if (bwb_strnicmp (Input, ELSE, ELSE_len) == 0)
- {
- /* ELSE */
- if (LastCommand == ELSE_COMMAND)
- {
- /* we need an ENDIF here */
- FORCE_COLON;
- FORCE_ENDIF;
- FORCE_COLON;
- nENDIF++;
- }
- LastCommand = ELSE_COMMAND;
- nELSE++;
- FORCE_COLON;
- COPY_ELSE;
- FORCE_COLON;
- COPY_SPACES;
- if (bwb_isdigit (*Input))
- {
- /* IF x THEN stmts ELSE line */
- FORCE_GOTO;
- COPY_LINENUMBER;
- COPY_SPACES;
- }
- FORCE_COLON;
- }
- else if (bwb_strnicmp (Input, ENDIF, ENDIF_len) == 0)
- {
- /* ENDIF */
- LastCommand = ENDIF_COMMAND;
- nENDIF++;
- COPY_ENDIF;
- FORCE_COLON;
- }
- else
- {
- /* was NOT the start of a command */
- COPYCHAR;
- }
- }
- /* end of input */
- if (nENDIF < nIF)
- {
- while (nENDIF < nIF)
- {
- /* we need trailing ENDIF's */
- nENDIF++;
- FORCE_COLON;
- FORCE_ENDIF;
- }
- }
- /* fixup trailing REMark command */
- if (bwb_strnicmp (Input, REM, REM_len) == 0)
- {
- /* REMark */
- /* 100 A=1 REMark */
- /* 100 A=1:REMark */
- /* 100 A=1'REMark */
- FORCE_COLON;
- }
- /* copy the comments */
- while (*Input != NulChar)
- {
- COPYCHAR;
- /* cppcheck: (style) Variable 'LastChar' is assigned a value that is never used. */
- }
- OUTPUTCHAR (NulChar);
- bwb_strcpy (InBuffer, OutBuffer);
- }
-
-
- /***************************************************************
-
- FUNCTION: bwb_ladd()
-
- DESCRIPTION: This function adds a new line (in the
- buffer) to the program in memory.
-
- ***************************************************************/
- static int
- bwb_ladd (char *buffer, LineType * p, int IsUser)
- {
- LineType *l;
- LineType *previous;
- char *newbuffer;
- char *NextStatement;
- char *ThisStatement;
- int Replace;
- char BreakChar;
-
- assert( My != NULL );
- assert( My->SYSOUT != NULL );
- assert( My->SYSOUT->cfp != NULL );
- assert( My->CurrentVersion != NULL );
- assert (buffer != NULL);
- assert (p != NULL);
-
-
- Replace = TRUE;
- BreakChar = NulChar;
- CleanLine (buffer);
- if (is_empty_string (buffer))
- {
- /* silengtly ignore blank lines */
- return FALSE;
- }
- /*
- from here, the line WILL be added so the user can EDIT it,
- we just complain and refuse to run if any error is detected.
- */
- My->IsScanRequired = TRUE; /* program needs to be scanned again */
-
- /* AUTO-FIX UNBALANCED QUOTES */
- if (FixQuotes (buffer) == FALSE)
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp, "UNBALANCED QUOTES: %s\n", buffer);
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, UNBALANCED QUOTES */
- }
-
- if (IS_CHAR (*buffer, My->CurrentVersion->OptionStatementChar))
- {
- /* part of a multi-statement line */
- }
- else if (IS_CHAR (*buffer, My->CurrentVersion->OptionCommentChar))
- {
- /* part of a multi-statement line */
- }
- else
- {
- ImportClassicIfThenElse (buffer);
- }
- ThisStatement = buffer;
- NextStatement = NULL;
- BreakChar = NulChar;
-
- do
- {
- if (BreakChar == NulChar)
- {
- /* first pass thru the do{} while loop, do nothing */
- }
- else if (IS_CHAR (BreakChar, My->CurrentVersion->OptionCommentChar))
- {
- /* ThisStatment will turn out to be the last 08082020 Ken ? */
- * ThisStatement = My->CurrentVersion->OptionCommentChar;
- }
- else if (IS_CHAR (BreakChar, My->CurrentVersion->OptionStatementChar))
- {
- /* we are NOT the last statement, skip over the OptionStatementChar */
- ThisStatement++;
- }
- else
- {
- /* Internal Error */
- }
-
- if (BreakChar == NulChar
- && IS_CHAR (*buffer, My->CurrentVersion->OptionStatementChar))
- {
- /* first pass thru and line begins with colon */
- /* part of a multi-statement line */
- NextStatement = NULL;
- if (My->NextValidLineNumber > 1)
- {
- My->NextValidLineNumber--;
- }
- Replace = FALSE;
- }
- else
- if (BreakChar == NulChar
- && IS_CHAR (*buffer, My->CurrentVersion->OptionCommentChar))
- {
- /* first pass thru and line begins with apostrophe */
- /* part of a multi-statement line */
- NextStatement = NULL;
- if (My->NextValidLineNumber > 1)
- {
- My->NextValidLineNumber--;
- }
- Replace = FALSE;
- }
- else
- {
- NextStatement = FindClassicStatementEnd (ThisStatement);
- }
-
- if (NextStatement == NULL)
- {
- /* we are the last statement */
- }
- else
- {
- /* another statement follows */
- BreakChar = *NextStatement;
- *NextStatement = NulChar;
- }
- CleanLine (ThisStatement);
- if (is_empty_string (ThisStatement) == FALSE)
- {
-
- /* get memory for this line */
- if ((l = (LineType *) calloc (1, sizeof (LineType))) == NULL)
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp, "Out of memory\n");
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, Out of memory */
- return FALSE;
- }
-
- /* this line has not been executed or numbered */
- l->LineFlags = 0;
- if (IsUser)
- {
- l->LineFlags |= LINE_USER;
- }
- l->IncludeLevel = My->IncludeLevel; /* %INCLUDE */
- l->position = 0;
-
- /*
- **
- ** ALL lines have a line number.
- ** If a line number is not provided,
- ** then the next available line number is assigned.
- **
- */
- newbuffer = ThisStatement;
- l->number = My->NextValidLineNumber;
-
- if (buff_read_line_number (newbuffer, &(l->position), &l->number))
- {
- if (l->number < My->NextValidLineNumber)
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp, "%d < %d - LINE OUT OF ORDER: %s\n",
- l->number, My->NextValidLineNumber, buffer);
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, LINE OUT OF ORDER */
- l->number = My->NextValidLineNumber; /* sane default */
- }
- else if (l->number < MINLIN || l->number > MAXLIN)
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp, "INVALID LINE NUMBER: %s\n", buffer);
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, INVALID LINE NUMBER */
- l->number = My->NextValidLineNumber; /* sane default */
- }
- else
- {
- /* OK */
- My->NextValidLineNumber = l->number;
- l->LineFlags |= LINE_NUMBERED; /* line was manually numbered */
- }
- /* A SPACE IS REQUIRED AFTER THE LINE NUMBER -- NO EXCEPTIONS */
- if (newbuffer[l->position] != ' ')
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp, "MISSING SPACE AFTER LINE NUMBER: %s\n",
- buffer);
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, MISSING SPACE AFTER LINE NUMBER */
- }
- /* newuffer does NOT contain the line number */
- newbuffer += l->position;
- }
- /* the next valid line number is this line number plus one */
- CleanLine (newbuffer);
- if (*newbuffer != NulChar
- && *newbuffer == My->CurrentVersion->OptionStatementChar)
- {
- /* this is part of a multi-statement line */
- newbuffer++;
- CleanLine (newbuffer);
- }
- /*
- **
- ** copy into the line buffer
- **
- */
- if (l->buffer != NULL)
- {
- free (l->buffer);
- l->buffer = NULL; /* JBV */
- }
- if ((l->buffer =
- (char *) calloc (bwb_strlen (newbuffer) + 1 /* NulChar */ ,
- sizeof (char))) == NULL)
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp, "Out of memory\n");
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, Out of memory */
- return FALSE; /* we cannot determine the command */
- }
- bwb_strcpy (l->buffer, newbuffer);
- /*
- **
- ** determine command
- **
- */
- line_start (l);
- if (l->cmdnum)
- {
- /* OK */
- }
- else
- {
- /* ERROR */
- fprintf (My->SYSOUT->cfp,
- "ILLEGAL COMMAND AFTER LINE NUMBER: %d %s\n", l->number,
- l->buffer);
- ResetConsoleColumn ();
- My->ERR = -1; /* bwb_ladd, ILLEGAL COMMAND AFTER LINE NUMBER */
- }
- /*
- **
- ** add the line to the linked-list of lines
- **
- */
- for (previous = p; previous != My->EndMarker; previous = previous->next)
- {
- if (previous->number == l->number)
- {
- if (Replace == TRUE)
- {
- /* REPLACE 'previous' */
- while (previous->number == l->number)
- {
- LineType *z;
-
- z = previous;
- previous = previous->next;
- bwb_freeline (z);
- }
- l->next = previous;
- p->next = l;
-
- }
- else
- {
- /* APPEND after 'previous' */
- while (previous->next->number == l->number)
- {
- previous = previous->next;
- }
- l->next = previous->next;
- previous->next = l;
- }
- break;
- }
- else
- if (previous->number < l->number
- && l->number < previous->next->number)
- {
- /* INSERT BETWEEN 'previous' AND 'previous->next' */
- l->next = previous->next;
- previous->next = l;
- break;
- }
- p = previous;
- }
-
- }
- /* another statement may follow */
- ThisStatement = NextStatement;
- Replace = FALSE;
- }
- while (ThisStatement != NULL);
- My->NextValidLineNumber++;
- return TRUE;
- }
-
- /***************************************************************
-
- FUNCTION: bwb_xtxtline()
-
- DESCRIPTION: This function executes a text line, i.e.,
- places it in memory and then relinquishes
- control.
-
- ***************************************************************/
-
- static void
- bwb_xtxtline (char *buffer)
- {
- assert( My != NULL );
- assert (buffer != NULL);
-
- /* remove old program from memory */
- bwb_xnew (My->UserMarker);
-
- CleanLine (buffer);
- if (is_empty_string (buffer))
- {
- /* silengtly ignore blank lines */
- return;
- }
-
- /* add to the user line list */
- bwb_ladd (buffer, My->UserMarker, TRUE);
-
- /* execute the line as BASIC command line */
- if (bwb_incexec ())
- {
- My->StackHead->line = My->UserMarker->next; /* and set current line in it */
- My->StackHead->ExecCode = EXEC_NORM;
- }
- }
-
- /***************************************************************
-
- FUNCTION: bwb_incexec()
-
- DESCRIPTION: This function increments the EXEC
- stack counter.
-
- ***************************************************************/
-
- int
- bwb_incexec (void)
- {
- StackType *StackItem;
- assert( My != NULL );
-
-
- if (My->StackDepthInteger >= EXECLEVELS)
- {
- WARN_OUT_OF_MEMORY;
- return FALSE;
- }
- if ((StackItem = (StackType *) calloc (1, sizeof (StackType))) == NULL)
- {
- WARN_OUT_OF_MEMORY;
- return FALSE;
- }
- StackItem->ExecCode = EXEC_NORM; /* sane default */
- StackItem->line = My->ThisLine; /* bwb_incexec */
- StackItem->LoopTopLine = NULL;
- StackItem->local_variable = NULL;
- StackItem->OnErrorGoto = 0;
- StackItem->next = My->StackHead;
- My->StackHead = StackItem;
- My->StackDepthInteger++;
- return TRUE;
- }
-
- /***************************************************************
-
- FUNCTION: bwb_decexec()
-
- DESCRIPTION: This function decrements the EXEC
- stack counter.
-
- ***************************************************************/
- void
- bwb_clrexec (void)
- {
- assert( My != NULL );
-
- while (My->StackHead != NULL)
- {
- bwb_decexec ();
- }
- }
-
- void
- bwb_decexec (void)
- {
- StackType *StackItem;
- assert( My != NULL );
-
-
- if (My->StackHead == NULL)
- {
- WARN_RETURN_WITHOUT_GOSUB;
- return;
- }
- StackItem = My->StackHead;
- My->StackHead = StackItem->next;
- free (StackItem);
- My->StackDepthInteger--;
- }
-
- /***************************************************************
-
- FUNCTION: bwb_mainloop()
-
- DESCRIPTION: This C function performs one iteration
- of the interpreter. In a non-preemptive
- scheduler, this function should be called
- by the scheduler, not by bwBASIC code.
-
- ***************************************************************/
-
- void
- bwb_mainloop (void)
- {
- assert( My != NULL );
-
-
- if (My->StackHead)
- {
- /* BASIC program running */
- bwb_execline (); /* execute one line of program */
- return;
- }
- /* BASIC program completed */
-
- if (My->ExternalInputFile != NULL)
- {
- /* for automated testing, --TAPE command line parameter */
- if (bwb_is_eof (My->ExternalInputFile) == FALSE)
- {
- /* --TAPE command line parameter is active */
- bwb_interact (); /* get user interaction */
- return;
- }
- }
- /* TAPE command inactive or completed */
-
- if (My->IsCommandLineFile)
- {
- /* BASIC program was started from command line */
- bwx_terminate ();
- return;
- }
- /* BASIC program was not started from command line */
-
- if (My->IsInteractive)
- {
- /* interactive */
- bwb_interact (); /* get user interaction */
- return;
- }
- /* non-interactive */
-
- bwx_terminate ();
- }
-
- /***************************************************************
-
- FUNCTION: bwb_execline()
-
- DESCRIPTION: This function executes a single line of
- a program in memory. This function is
- called by bwb_mainloop().
-
- ***************************************************************/
-
- extern int
- bwx_Error (int ERR, char *ErrorMessage)
- {
- /*
- ERR is the error number
- ErrorMessage is used to override the default error
- message, and is usually NULL
- */
- assert( My != NULL );
-
- switch (ERR)
- {
- case 0:
- /*
- **
- ** Clear any existing error
- **
- */
- My->IsErrorPending = FALSE; /* bwx_Error, ERR == 0 */
- My->ERR = 0; /* bwx_Error, ERR == 0 */
- My->ERL = NULL; /* bwx_Error, ERR == 0 */
- bwb_strcpy (My->ERROR4, ""); /* bwx_Error, ERR == 0 */
- return FALSE;
- case 6: /* WARN_OVERFLOW */
- case 11: /* WARN_DIVISION_BY_ZERO */
- case 15: /* WARN_STRING_TOO_LONG */
- /*
- **
- ** Behavior depends upon whether an Error handler is active.
- **
- */
- if (GetOnError () == 0)
- {
- /*
- **
- ** Error handler is NOT active.
- ** Do NOT set ERL, ERR, and ERROR$.
- ** Continue processing.
- **
- */
- if (ErrorMessage == NULL)
- {
- /* use the default error message */
- if (ERR > 0 && ERR < NUM_WARNINGS)
- {
- ErrorMessage = ERROR4[ERR];
- }
- }
- if (ErrorMessage != NULL)
- {
- if (bwb_strlen (ErrorMessage) > 0)
- {
- fprintf (My->SYSOUT->cfp, "%s\n", ErrorMessage);
- ResetConsoleColumn ();
- }
- }
- return FALSE; /* continue processing */
- }
- /*
- **
- ** Error handler IS active.
- ** Fall-thru to set ERL, ERR, and ERROR$.
- ** Abort processing.
- **
- */
- }
- if (My->IsErrorPending == FALSE) /* no errors pending */
- {
- /*
- **
- ** only keep the first pending error to occur
- **
- */
- My->IsErrorPending = TRUE; /* bwx_Error, ERR != 0 */
- My->ERR = ERR; /* bwx_Error, ERR != 0 */
- My->ERL = NULL; /* bwx_Error, ERR != 0 */
- bwb_strcpy (My->ERROR4, ""); /* bwx_Error, ERR != 0 */
- if (My->StackHead)
- {
- My->ERL = My->StackHead->line;
- }
- if (ErrorMessage == NULL)
- {
- /* use the default error message */
- if (ERR > 0 && ERR < NUM_WARNINGS)
- {
- ErrorMessage = ERROR4[ERR];
- }
- }
- if (ErrorMessage != NULL)
- {
- if (bwb_strlen (ErrorMessage) > MAX_ERR_LENGTH)
- {
- ErrorMessage[MAX_ERR_LENGTH] = NulChar;
- }
- bwb_strcpy (My->ERROR4, ErrorMessage);
- }
- }
- return TRUE; /* abort processing */
- }
-
- void
- bwb_execline (void)
- {
- LineType *r, *l;
- assert( My != NULL );
- assert( My->SYSOUT != NULL );
- assert( My->SYSOUT->cfp != NULL );
- assert( My->CurrentVersion != NULL );
-
- if (My->StackHead == NULL) /* in bwb_execline(), FATAL ERROR PENDING */
- {
- return;
- }
-
- l = My->StackHead->line;
-
- /* if the line is My->EndMarker, then break out of EXEC loops */
- if (l == NULL || l == My->EndMarker || My->ERR < 0) /* in bwb_execline(), FATAL ERROR PENDING */
- {
- bwb_clrexec ();
- return;
- }
-
- My->ThisLine = l; /* bwb_execline */
-
- /* Print line number if trace is on */
- if (My->IsTraceOn == TRUE)
- {
- if (l->LineFlags & (LINE_USER))
- {
- /* USER line in console */
- }
- else if (l->number > 0)
- {
- fprintf (My->SYSOUT->cfp, "[ %d %s ]", l->number, l->buffer);
- }
- }
- l->position = l->Startpos;
-
- /* if there is a BASIC command in the line, execute it here */
- if (l->cmdnum)
- {
- /* OK */
- }
- else
- {
- WARN_ILLEGAL_DIRECT;
- l->cmdnum = C_REM;
- }
- /* l->cmdnum != 0 */
-
- if (l->LineFlags & LINE_BREAK)
- {
- /* BREAK line */
- l->LineFlags &= ~LINE_BREAK;
- My->ContinueLine = l;
- bwx_STOP (TRUE);
- return;
- }
-
- /* advance beyond whitespace */
- line_skip_spaces (l); /* keep this */
-
- /* execute the command vector */
- if (My->CurrentVersion->OptionFlags & (OPTION_COVERAGE_ON))
- {
- /* We do this here so "END" and "STOP" are marked */
- if (l->cmdnum == C_DATA)
- {
- /* DATA lines are marked when they are READ */
- }
- else
- {
- /* this line was executed */
- l->LineFlags |= LINE_EXECUTED;
- }
- }
- r = bwb_vector (l);
- if (r == NULL)
- {
- WARN_INTERNAL_ERROR;
- return;
- }
- assert (r != NULL);
-
- if (My->ERR < 0) /* in bwb_execline(), FATAL ERROR PENDING */
- {
- /* FATAL */
- bwb_clrexec ();
- return;
- }
-
- if (My->IsErrorPending /* Keep This */ )
- {
- /* we are probably not at the end-of-the-line */
- }
- else if (r == l)
- {
- /* we should be at the end-of-the-line */
- if (line_is_eol (l))
- {
- /* OK */
- }
- else
- {
- WARN_SYNTAX_ERROR;
- return;
- }
- }
- else
- {
- /* we are probably not at the end-of-the-line */
- }
-
- if (My->IsErrorPending /* Keep This */ )
- {
- /*
- **
- ** a NON-FATAL ERROR has occurred. ERR, ERL, and ERROR$ were
- ** already set using bwb_warning(ERR,ERROR$)
- **
- */
- int err_gotol;
- My->IsErrorPending = FALSE; /* Error Recognized */
- err_gotol = GetOnError ();
- if (l->LineFlags & (LINE_USER))
- {
- /*
- **
- ** ---------------------------------------------------------------------
- ** USER line in console
- ** ---------------------------------------------------------------------
- **
- ** fall thru to the DEFAULT ERROR HANDLER
- **
- */
- }
- else if (l->number == 0)
- {
- /* fall thru to the DEFAULT ERROR HANDLER */
- }
- else if (My->ERL == NULL)
- {
- /* fall thru to the DEFAULT ERROR HANDLER */
- }
- else if (My->ERL->number == 0)
- {
- /* fall thru to the DEFAULT ERROR HANDLER */
- }
- else if (err_gotol == -1)
- {
- /*
- **
- ** ---------------------------------------------------------------------
- ** ON ERROR RESUME NEXT
- ** ---------------------------------------------------------------------
- **
- */
- assert (r != NULL);
- assert (r->next != NULL);
-
- r->next->position = 0;
- assert (My->StackHead != NULL);
- My->StackHead->line = r->next;
- return;
- }
- else if (err_gotol == 0)
- {
- /*
- **
- ** ---------------------------------------------------------------------
- ** ON ERROR GOTO 0
- ** ---------------------------------------------------------------------
- **
- ** fall thru to the DEFAULT ERROR HANDLER
- **
- */
- }
- else if (err_gotol == My->ERL->number)
- {
- /*
- **
- ** ---------------------------------------------------------------------
- ** RECURSION
- ** ---------------------------------------------------------------------
- **
- ** For example:
- ** 10 ON ERROR GOTO 20
- ** 20 ERROR 1
- **
- ** fall thru to the DEFAULT ERROR HANDLER
- **
- */
- }
- else
- {
- /* USER ERROR HANDLER SPECIFIED */
- /* find the user-specified error handler and jump there */
- LineType *x;
- for (x = My->StartMarker->next; x != My->EndMarker; x = x->next)
- {
- if (x->number == err_gotol)
- {
- /* FOUND */
- if (My->CurrentVersion->OptionFlags & (OPTION_ERROR_GOSUB))
- {
- /*
- **
- ** ---------------------------------------------------------------
- ** OPTION ERROR GOSUB
- ** ---------------------------------------------------------------
- **
- ** RETURN should act like RESUME NEXT...
- ** Execution resumes at the statement immediately following
- ** the one which caused the error.
- ** For structured commands, this is the bottom line of the structure.
- **
- */
- switch (l->cmdnum)
- {
- case C_IF8THEN:
- /* skip to END_IF */
- assert (l->OtherLine != NULL);
- for (l = l->OtherLine; l->cmdnum != C_END_IF; l = l->OtherLine);
- break;
- case C_SELECT_CASE:
- /* skip to END_SELECT */
- assert (l->OtherLine != NULL);
- for (l = l->OtherLine; l->cmdnum != C_END_SELECT;
- l = l->OtherLine);
- break;
- default:
- l = l->next;
- }
- l->position = 0;
- My->StackHead->line = l;
- if (bwb_incexec ())
- {
- x->position = 0;
- assert (My->StackHead != NULL);
- My->StackHead->line = x;
- My->StackHead->ExecCode = EXEC_GOSUB;
- }
- else
- {
- /* ERROR -- OUT OF MEMORY */
- assert (My->StackHead != NULL);
- My->StackHead->line = My->EndMarker;
- }
- }
- else
- {
- /*
- **
- ** ---------------------------------------------------------------
- ** OPTION ERROR GOTO
- ** ---------------------------------------------------------------
- **
- */
- x->position = 0; /* start of line */
- assert (My->StackHead != NULL);
- My->StackHead->line = x;
- }
- return;
- }
- }
- /* NOT FOUND */
- /* fall thru to the DEFAULT ERROR HANDLER */
- }
- /*
- **
- ** -----------------------------------------------------------------------
- ** DEFAULT ERROR HANDLER (FATAL)
- ** -----------------------------------------------------------------------
- **
- */
- /*
- **
- ** display error message
- **
- */
- if (l->LineFlags & (LINE_USER) || l->number <= 0)
- {
- /* USER line in console */
- fprintf (My->SYSOUT->cfp, "\nERROR: %s\n", My->ERROR4);
- ResetConsoleColumn ();
- }
- else
- {
- /* BASIC program line */
- fprintf (My->SYSOUT->cfp, "\nERROR in line %d: %s\n", l->number,
- My->ERROR4);
- ResetConsoleColumn ();
- /*
- **
- ** display stack trace
- **
- */
- if (My->CurrentVersion->OptionFlags & (OPTION_TRACE_ON))
- {
- /*
- ** Dump the BASIC stack trace when a FATAL error occurs.
- ** First line is the TOP of the stack.
- ** Last line is the BOTTOM of the stack.
- */
- StackType *StackItem;
- fprintf (My->SYSOUT->cfp, "\nSTACK TRACE:\n");
- for (StackItem = My->StackHead; StackItem != NULL;
- StackItem = StackItem->next)
- {
- LineType *l;
-
- l = StackItem->line;
- if (l != NULL)
- {
- if (MINLIN <= l->number && l->number <= MAXLIN)
- {
- /* BASIC program line */
- if (l->buffer == NULL)
- {
- fprintf (My->SYSOUT->cfp, "%d\n", l->number);
- }
- else
- {
- fprintf (My->SYSOUT->cfp, "%d:%s\n", l->number, l->buffer);
- }
- }
- }
- }
- ResetConsoleColumn ();
- }
- }
-
- My->AutomaticLineNumber = 0;
- My->AutomaticLineIncrement = 0;
-
- if (My->IsInteractive)
- {
- /* INTERACTIVE: terminate program */
-
- /* reset all stack counters */
- bwb_clrexec ();
- SetOnError (0);
-
- My->ERR = -1; /* in bwb_execline(), default error handler */
-
-
- /* reset the break handler */
- signal (SIGINT, break_mes);
-
-
- return;
- }
- /* NOT INTERACTIVE: terminate immediately */
- bwx_terminate ();
- return; /* never reached */
- }
- if (l->number > 0)
- {
- /* These events only occur in running programs */
- if (My->IsTimerOn)
- {
- /* TIMER ON */
- if (My->OnTimerCount > 0)
- {
- if (bwx_TIMER (0) > My->OnTimerExpires)
- {
- My->IsTimerOn = FALSE;
- if (My->OnTimerLineNumber > 0)
- {
- /* ON TIMER( My->OnTimerCount ) GOSUB My->OnTimerLineNumber */
- LineType *x;
-
- for (x = My->StartMarker->next; x != My->EndMarker; x = x->next)
- {
- if (x->number == My->OnTimerLineNumber)
- {
- /* FOUND */
- /* save current stack level */
- assert (My->StackHead != NULL);
- My->StackHead->line = r;
- /* increment exec stack */
- if (bwb_incexec ())
- {
- /* set the new position to x and return x */
- x->position = 0;
- assert (My->StackHead != NULL);
- My->StackHead->line = x;
- My->StackHead->ExecCode = EXEC_GOSUB;
- }
- else
- {
- /* ERROR */
- assert (My->StackHead != NULL);
- My->StackHead->line = My->EndMarker;
- }
- return;
- }
- }
- /* NOT FOUND */
- }
- }
- }
- }
- }
- /* check for end of line: if so, advance to next line and return */
- if (r == l)
- {
- /* advance to the next line */
- l->next->position = 0;
- r = l->next;
- }
- else if (line_is_eol (r))
- {
- /* we could be at the end-of-the-line, added for RETURN */
- /* advance to the next line */
- r->next->position = 0;
- r = r->next;
- }
- /* else reset with the value in r */
- assert (My->StackHead != NULL);
- My->StackHead->line = r;
- }
-
- /***************************************************************
-
- FUNCTION: is_ln()
-
- DESCRIPTION: This function determines whether a program
- line (in buffer) begins with a line number.
-
- ***************************************************************/
-
- static int
- is_ln (char *buffer)
- {
- int position;
-
- assert (buffer != NULL);
-
- position = 0;
- buff_skip_spaces (buffer, &position); /* keep this */
- if (bwb_isdigit (buffer[position]))
- {
- return TRUE;
- }
- return FALSE;
- }
-
- /***************************************************************
-
- FUNCTION: is_numconst()
-
- DESCRIPTION: This function reads the string in <buffer>
- and returns TRUE if it is a numerical
- constant and FALSE if it is not. At
- this point, only decimal (base 10)
- constants are detected.
-
- ***************************************************************/
-
- static int
- is_numconst (char *buffer)
- {
- char *p;
-
- assert (buffer != NULL);
-
- /* Return FALSE for empty buffer */
-
- if (buffer[0] == NulChar)
- {
- return FALSE;
- }
- /* else check digits */
-
- p = buffer;
- while (*p != NulChar)
- {
- switch (*p)
- {
- case '0':
- case '1':
- case '2':
- case '3':
- case '4':
- case '5':
- case '6':
- case '7':
- case '8':
- case '9':
- break;
- default:
- return FALSE;
- }
- p++;
- }
-
- /* only numerical characters detected */
-
- return TRUE;
-
- }
-
- /* SWITCH */
- LineType *
- bwb_vector( LineType *l )
- {
- LineType *r;
- assert (l != NULL);
- switch( l->cmdnum )
- {
- case C_DEF8LBL:
- r = bwb_DEF8LBL( l );
- break;
- case C_APPEND:
- r = bwb_APPEND( l );
- break;
- case C_AS:
- r = bwb_AS( l );
- break;
- case C_AUTO:
- r = bwb_AUTO( l );
- break;
- case C_BACKSPACE:
- r = bwb_BACKSPACE( l );
- break;
- case C_BREAK:
- r = bwb_BREAK( l );
- break;
- case C_BUILD:
- r = bwb_BUILD( l );
- break;
- case C_BYE:
- r = bwb_BYE( l );
- break;
- case C_CALL:
- r = bwb_CALL( l );
- break;
- case C_CASE:
- r = bwb_CASE( l );
- break;
- case C_CASE_ELSE:
- r = bwb_CASE_ELSE( l );
- break;
- case C_CHAIN:
- r = bwb_CHAIN( l );
- break;
- case C_CHANGE:
- r = bwb_CHANGE( l );
- break;
- case C_CLEAR:
- r = bwb_CLEAR( l );
- break;
- case C_CLOAD:
- r = bwb_CLOAD( l );
- break;
- case C_CLOAD8:
- r = bwb_CLOAD8( l );
- break;
- case C_CLOSE:
- r = bwb_CLOSE( l );
- break;
- case C_CLR:
- r = bwb_CLR( l );
- break;
- case C_CMDS:
- r = bwb_CMDS( l );
- break;
- case C_COMMON:
- r = bwb_COMMON( l );
- break;
- case C_CONSOLE:
- r = bwb_CONSOLE( l );
- break;
- case C_CONST:
- r = bwb_CONST( l );
- break;
- case C_CONT:
- r = bwb_CONT( l );
- break;
- case C_CONTINUE:
- r = bwb_CONTINUE( l );
- break;
- case C_COPY:
- r = bwb_COPY( l );
- break;
- case C_CREATE:
- r = bwb_CREATE( l );
- break;
- case C_CSAVE:
- r = bwb_CSAVE( l );
- break;
- case C_CSAVE8:
- r = bwb_CSAVE8( l );
- break;
- case C_DATA:
- r = bwb_DATA( l );
- break;
- case C_DEC:
- r = bwb_DEC( l );
- break;
- case C_DEF:
- r = bwb_DEF( l );
- break;
- case C_DEFBYT:
- r = bwb_DEFBYT( l );
- break;
- case C_DEFCUR:
- r = bwb_DEFCUR( l );
- break;
- case C_DEFDBL:
- r = bwb_DEFDBL( l );
- break;
- case C_DEFINT:
- r = bwb_DEFINT( l );
- break;
- case C_DEFLNG:
- r = bwb_DEFLNG( l );
- break;
- case C_DEFSNG:
- r = bwb_DEFSNG( l );
- break;
- case C_DEFSTR:
- r = bwb_DEFSTR( l );
- break;
- case C_DELETE:
- r = bwb_DELETE( l );
- break;
- case C_DELIMIT:
- r = bwb_DELIMIT( l );
- break;
- case C_DIM:
- r = bwb_DIM( l );
- break;
- case C_DISPLAY:
- r = bwb_DISPLAY( l );
- break;
- case C_DO:
- r = bwb_DO( l );
- break;
- case C_DOS:
- r = bwb_DOS( l );
- break;
- case C_DSP:
- r = bwb_DSP( l );
- break;
- case C_EDIT:
- r = bwb_EDIT( l );
- break;
- case C_ELSE:
- r = bwb_ELSE( l );
- break;
- case C_ELSEIF:
- r = bwb_ELSEIF( l );
- break;
- case C_END:
- r = bwb_END( l );
- break;
- case C_END_FUNCTION:
- r = bwb_END_FUNCTION( l );
- break;
- case C_END_IF:
- r = bwb_END_IF( l );
- break;
- case C_END_SELECT:
- r = bwb_END_SELECT( l );
- break;
- case C_END_SUB:
- r = bwb_END_SUB( l );
- break;
- case C_ERASE:
- r = bwb_ERASE( l );
- break;
- case C_EXCHANGE:
- r = bwb_EXCHANGE( l );
- break;
- case C_EXIT:
- r = bwb_EXIT( l );
- break;
- case C_EXIT_DO:
- r = bwb_EXIT_DO( l );
- break;
- case C_EXIT_FOR:
- r = bwb_EXIT_FOR( l );
- break;
- case C_EXIT_FUNCTION:
- r = bwb_EXIT_FUNCTION( l );
- break;
- case C_EXIT_REPEAT:
- r = bwb_EXIT_REPEAT( l );
- break;
- case C_EXIT_SUB:
- r = bwb_EXIT_SUB( l );
- break;
- case C_EXIT_WHILE:
- r = bwb_EXIT_WHILE( l );
- break;
- case C_FEND:
- r = bwb_FEND( l );
- break;
- case C_FIELD:
- r = bwb_FIELD( l );
- break;
- case C_FILE:
- r = bwb_FILE( l );
- break;
- case C_FILES:
- r = bwb_FILES( l );
- break;
- case C_FLEX:
- r = bwb_FLEX( l );
- break;
- case C_FNCS:
- r = bwb_FNCS( l );
- break;
- case C_FNEND:
- r = bwb_FNEND( l );
- break;
- case C_FOR:
- r = bwb_FOR( l );
- break;
- case C_FUNCTION:
- r = bwb_FUNCTION( l );
- break;
- case C_GET:
- r = bwb_GET( l );
- break;
- case C_GO:
- r = bwb_GO( l );
- break;
- case C_GO_SUB:
- r = bwb_GO_SUB( l );
- break;
- case C_GO_TO:
- r = bwb_GO_TO( l );
- break;
- case C_GOODBYE:
- r = bwb_GOODBYE( l );
- break;
- case C_GOSUB:
- r = bwb_GOSUB( l );
- break;
- case C_GOTO:
- r = bwb_GOTO( l );
- break;
- case C_HELP:
- r = bwb_HELP( l );
- break;
- case C_IF:
- r = bwb_IF( l );
- break;
- case C_IF_END:
- r = bwb_IF_END( l );
- break;
- case C_IF_MORE:
- r = bwb_IF_MORE( l );
- break;
- case C_IF8THEN:
- r = bwb_IF8THEN( l );
- break;
- case C_IMAGE:
- r = bwb_IMAGE( l );
- break;
- case C_INC:
- r = bwb_INC( l );
- break;
- case C_INPUT:
- r = bwb_INPUT( l );
- break;
- case C_INPUT_LINE:
- r = bwb_INPUT_LINE( l );
- break;
- case C_LET:
- r = bwb_LET( l );
- break;
- case C_LINE:
- r = bwb_LINE( l );
- break;
- case C_LINE_INPUT:
- r = bwb_LINE_INPUT( l );
- break;
- case C_LIST:
- r = bwb_LIST( l );
- break;
- case C_LISTNH:
- r = bwb_LISTNH( l );
- break;
- case C_LLIST:
- r = bwb_LLIST( l );
- break;
- case C_LOAD:
- r = bwb_LOAD( l );
- break;
- case C_LOCAL:
- r = bwb_LOCAL( l );
- break;
- case C_LOOP:
- r = bwb_LOOP( l );
- break;
- case C_LPRINT:
- r = bwb_LPRINT( l );
- break;
- case C_LPRINTER:
- r = bwb_LPRINTER( l );
- break;
- case C_LPT:
- r = bwb_LPT( l );
- break;
- case C_LSET:
- r = bwb_LSET( l );
- break;
- case C_MAINTAINER:
- r = bwb_MAINTAINER( l );
- break;
- case C_MAINTAINER_CMDS:
- r = bwb_MAINTAINER_CMDS( l );
- break;
- case C_MAINTAINER_CMDS_HTML:
- r = bwb_MAINTAINER_CMDS_HTML( l );
- break;
- case C_MAINTAINER_CMDS_ID:
- r = bwb_MAINTAINER_CMDS_ID( l );
- break;
- case C_MAINTAINER_CMDS_MANUAL:
- r = bwb_MAINTAINER_CMDS_MANUAL( l );
- break;
- case C_MAINTAINER_CMDS_SWITCH:
- r = bwb_MAINTAINER_CMDS_SWITCH( l );
- break;
- case C_MAINTAINER_CMDS_TABLE:
- r = bwb_MAINTAINER_CMDS_TABLE( l );
- break;
- case C_MAINTAINER_DEBUG:
- r = bwb_MAINTAINER_DEBUG( l );
- break;
- case C_MAINTAINER_DEBUG_OFF:
- r = bwb_MAINTAINER_DEBUG_OFF( l );
- break;
- case C_MAINTAINER_DEBUG_ON:
- r = bwb_MAINTAINER_DEBUG_ON( l );
- break;
- case C_MAINTAINER_FNCS:
- r = bwb_MAINTAINER_FNCS( l );
- break;
- case C_MAINTAINER_FNCS_HTML:
- r = bwb_MAINTAINER_FNCS_HTML( l );
- break;
- case C_MAINTAINER_FNCS_ID:
- r = bwb_MAINTAINER_FNCS_ID( l );
- break;
- case C_MAINTAINER_FNCS_MANUAL:
- r = bwb_MAINTAINER_FNCS_MANUAL( l );
- break;
- case C_MAINTAINER_FNCS_SWITCH:
- r = bwb_MAINTAINER_FNCS_SWITCH( l );
- break;
- case C_MAINTAINER_FNCS_TABLE:
- r = bwb_MAINTAINER_FNCS_TABLE( l );
- break;
- case C_MAINTAINER_MANUAL:
- r = bwb_MAINTAINER_MANUAL( l );
- break;
- case C_MAINTAINER_STACK:
- r = bwb_MAINTAINER_STACK( l );
- break;
- case C_MARGIN:
- r = bwb_MARGIN( l );
- break;
- case C_MAT:
- r = bwb_MAT( l );
- break;
- case C_MAT_GET:
- r = bwb_MAT_GET( l );
- break;
- case C_MAT_INPUT:
- r = bwb_MAT_INPUT( l );
- break;
- case C_MAT_PRINT:
- r = bwb_MAT_PRINT( l );
- break;
- case C_MAT_PUT:
- r = bwb_MAT_PUT( l );
- break;
- case C_MAT_READ:
- r = bwb_MAT_READ( l );
- break;
- case C_MAT_WRITE:
- r = bwb_MAT_WRITE( l );
- break;
- case C_MERGE:
- r = bwb_MERGE( l );
- break;
- case C_MID4:
- r = bwb_MID4( l );
- break;
- case C_MON:
- r = bwb_MON( l );
- break;
- case C_NAME:
- r = bwb_NAME( l );
- break;
- case C_NEW:
- r = bwb_NEW( l );
- break;
- case C_NEXT:
- r = bwb_NEXT( l );
- break;
- case C_OF:
- r = bwb_OF( l );
- break;
- case C_OLD:
- r = bwb_OLD( l );
- break;
- case C_ON:
- r = bwb_ON( l );
- break;
- case C_ON_ERROR:
- r = bwb_ON_ERROR( l );
- break;
- case C_ON_ERROR_GOSUB:
- r = bwb_ON_ERROR_GOSUB( l );
- break;
- case C_ON_ERROR_GOTO:
- r = bwb_ON_ERROR_GOTO( l );
- break;
- case C_ON_ERROR_RESUME:
- r = bwb_ON_ERROR_RESUME( l );
- break;
- case C_ON_ERROR_RESUME_NEXT:
- r = bwb_ON_ERROR_RESUME_NEXT( l );
- break;
- case C_ON_ERROR_RETURN:
- r = bwb_ON_ERROR_RETURN( l );
- break;
- case C_ON_ERROR_RETURN_NEXT:
- r = bwb_ON_ERROR_RETURN_NEXT( l );
- break;
- case C_ON_TIMER:
- r = bwb_ON_TIMER( l );
- break;
- case C_OPEN:
- r = bwb_OPEN( l );
- break;
- case C_OPTION:
- r = bwb_OPTION( l );
- break;
- case C_OPTION_ANGLE:
- r = bwb_OPTION_ANGLE( l );
- break;
- case C_OPTION_ANGLE_DEGREES:
- r = bwb_OPTION_ANGLE_DEGREES( l );
- break;
- case C_OPTION_ANGLE_GRADIANS:
- r = bwb_OPTION_ANGLE_GRADIANS( l );
- break;
- case C_OPTION_ANGLE_RADIANS:
- r = bwb_OPTION_ANGLE_RADIANS( l );
- break;
- case C_OPTION_ARITHMETIC:
- r = bwb_OPTION_ARITHMETIC( l );
- break;
- case C_OPTION_ARITHMETIC_DECIMAL:
- r = bwb_OPTION_ARITHMETIC_DECIMAL( l );
- break;
- case C_OPTION_ARITHMETIC_FIXED:
- r = bwb_OPTION_ARITHMETIC_FIXED( l );
- break;
- case C_OPTION_ARITHMETIC_NATIVE:
- r = bwb_OPTION_ARITHMETIC_NATIVE( l );
- break;
- case C_OPTION_BASE:
- r = bwb_OPTION_BASE( l );
- break;
- case C_OPTION_BUGS:
- r = bwb_OPTION_BUGS( l );
- break;
- case C_OPTION_BUGS_BOOLEAN:
- r = bwb_OPTION_BUGS_BOOLEAN( l );
- break;
- case C_OPTION_BUGS_OFF:
- r = bwb_OPTION_BUGS_OFF( l );
- break;
- case C_OPTION_BUGS_ON:
- r = bwb_OPTION_BUGS_ON( l );
- break;
- case C_OPTION_COMPARE:
- r = bwb_OPTION_COMPARE( l );
- break;
- case C_OPTION_COMPARE_BINARY:
- r = bwb_OPTION_COMPARE_BINARY( l );
- break;
- case C_OPTION_COMPARE_DATABASE:
- r = bwb_OPTION_COMPARE_DATABASE( l );
- break;
- case C_OPTION_COMPARE_TEXT:
- r = bwb_OPTION_COMPARE_TEXT( l );
- break;
- case C_OPTION_COVERAGE:
- r = bwb_OPTION_COVERAGE( l );
- break;
- case C_OPTION_COVERAGE_OFF:
- r = bwb_OPTION_COVERAGE_OFF( l );
- break;
- case C_OPTION_COVERAGE_ON:
- r = bwb_OPTION_COVERAGE_ON( l );
- break;
- case C_OPTION_DATE:
- r = bwb_OPTION_DATE( l );
- break;
- case C_OPTION_DIGITS:
- r = bwb_OPTION_DIGITS( l );
- break;
- case C_OPTION_DISABLE:
- r = bwb_OPTION_DISABLE( l );
- break;
- case C_OPTION_DISABLE_COMMAND:
- r = bwb_OPTION_DISABLE_COMMAND( l );
- break;
- case C_OPTION_DISABLE_FUNCTION:
- r = bwb_OPTION_DISABLE_FUNCTION( l );
- break;
- case C_OPTION_DISABLE_OPERATOR:
- r = bwb_OPTION_DISABLE_OPERATOR( l );
- break;
- case C_OPTION_EDIT:
- r = bwb_OPTION_EDIT( l );
- break;
- case C_OPTION_ENABLE:
- r = bwb_OPTION_ENABLE( l );
- break;
- case C_OPTION_ENABLE_COMMAND:
- r = bwb_OPTION_ENABLE_COMMAND( l );
- break;
- case C_OPTION_ENABLE_FUNCTION:
- r = bwb_OPTION_ENABLE_FUNCTION( l );
- break;
- case C_OPTION_ENABLE_OPERATOR:
- r = bwb_OPTION_ENABLE_OPERATOR( l );
- break;
- case C_OPTION_ERROR:
- r = bwb_OPTION_ERROR( l );
- break;
- case C_OPTION_ERROR_GOSUB:
- r = bwb_OPTION_ERROR_GOSUB( l );
- break;
- case C_OPTION_ERROR_GOTO:
- r = bwb_OPTION_ERROR_GOTO( l );
- break;
- case C_OPTION_EXPLICIT:
- r = bwb_OPTION_EXPLICIT( l );
- break;
- case C_OPTION_EXTENSION:
- r = bwb_OPTION_EXTENSION( l );
- break;
- case C_OPTION_FILES:
- r = bwb_OPTION_FILES( l );
- break;
- case C_OPTION_IMPLICIT:
- r = bwb_OPTION_IMPLICIT( l );
- break;
- case C_OPTION_INDENT:
- r = bwb_OPTION_INDENT( l );
- break;
- case C_OPTION_LABELS:
- r = bwb_OPTION_LABELS( l );
- break;
- case C_OPTION_LABELS_OFF:
- r = bwb_OPTION_LABELS_OFF( l );
- break;
- case C_OPTION_LABELS_ON:
- r = bwb_OPTION_LABELS_ON( l );
- break;
- case C_OPTION_PROMPT:
- r = bwb_OPTION_PROMPT( l );
- break;
- case C_OPTION_PUNCT:
- r = bwb_OPTION_PUNCT( l );
- break;
- case C_OPTION_PUNCT_AT:
- r = bwb_OPTION_PUNCT_AT( l );
- break;
- case C_OPTION_PUNCT_BYTE:
- r = bwb_OPTION_PUNCT_BYTE( l );
- break;
- case C_OPTION_PUNCT_COMMENT:
- r = bwb_OPTION_PUNCT_COMMENT( l );
- break;
- case C_OPTION_PUNCT_CURRENCY:
- r = bwb_OPTION_PUNCT_CURRENCY( l );
- break;
- case C_OPTION_PUNCT_DOUBLE:
- r = bwb_OPTION_PUNCT_DOUBLE( l );
- break;
- case C_OPTION_PUNCT_FILENUM:
- r = bwb_OPTION_PUNCT_FILENUM( l );
- break;
- case C_OPTION_PUNCT_IMAGE:
- r = bwb_OPTION_PUNCT_IMAGE( l );
- break;
- case C_OPTION_PUNCT_INPUT:
- r = bwb_OPTION_PUNCT_INPUT( l );
- break;
- case C_OPTION_PUNCT_INTEGER:
- r = bwb_OPTION_PUNCT_INTEGER( l );
- break;
- case C_OPTION_PUNCT_LONG:
- r = bwb_OPTION_PUNCT_LONG( l );
- break;
- case C_OPTION_PUNCT_LPAREN:
- r = bwb_OPTION_PUNCT_LPAREN( l );
- break;
- case C_OPTION_PUNCT_PRINT:
- r = bwb_OPTION_PUNCT_PRINT( l );
- break;
- case C_OPTION_PUNCT_QUOTE:
- r = bwb_OPTION_PUNCT_QUOTE( l );
- break;
- case C_OPTION_PUNCT_RPAREN:
- r = bwb_OPTION_PUNCT_RPAREN( l );
- break;
- case C_OPTION_PUNCT_SINGLE:
- r = bwb_OPTION_PUNCT_SINGLE( l );
- break;
- case C_OPTION_PUNCT_STATEMENT:
- r = bwb_OPTION_PUNCT_STATEMENT( l );
- break;
- case C_OPTION_PUNCT_STRING:
- r = bwb_OPTION_PUNCT_STRING( l );
- break;
- case C_OPTION_RECLEN:
- r = bwb_OPTION_RECLEN( l );
- break;
- case C_OPTION_RENUM:
- r = bwb_OPTION_RENUM( l );
- break;
- case C_OPTION_ROUND:
- r = bwb_OPTION_ROUND( l );
- break;
- case C_OPTION_ROUND_BANK:
- r = bwb_OPTION_ROUND_BANK( l );
- break;
- case C_OPTION_ROUND_MATH:
- r = bwb_OPTION_ROUND_MATH( l );
- break;
- case C_OPTION_ROUND_TRUNCATE:
- r = bwb_OPTION_ROUND_TRUNCATE( l );
- break;
- case C_OPTION_SCALE:
- r = bwb_OPTION_SCALE( l );
- break;
- case C_OPTION_SLEEP:
- r = bwb_OPTION_SLEEP( l );
- break;
- case C_OPTION_STDERR:
- r = bwb_OPTION_STDERR( l );
- break;
- case C_OPTION_STDIN:
- r = bwb_OPTION_STDIN( l );
- break;
- case C_OPTION_STDOUT:
- r = bwb_OPTION_STDOUT( l );
- break;
- case C_OPTION_STRICT:
- r = bwb_OPTION_STRICT( l );
- break;
- case C_OPTION_STRICT_OFF:
- r = bwb_OPTION_STRICT_OFF( l );
- break;
- case C_OPTION_STRICT_ON:
- r = bwb_OPTION_STRICT_ON( l );
- break;
- case C_OPTION_TERMINAL:
- r = bwb_OPTION_TERMINAL( l );
- break;
- case C_OPTION_TERMINAL_ADM:
- r = bwb_OPTION_TERMINAL_ADM( l );
- break;
- case C_OPTION_TERMINAL_ANSI:
- r = bwb_OPTION_TERMINAL_ANSI( l );
- break;
- case C_OPTION_TERMINAL_NONE:
- r = bwb_OPTION_TERMINAL_NONE( l );
- break;
- case C_OPTION_TIME:
- r = bwb_OPTION_TIME( l );
- break;
- case C_OPTION_TRACE:
- r = bwb_OPTION_TRACE( l );
- break;
- case C_OPTION_TRACE_OFF:
- r = bwb_OPTION_TRACE_OFF( l );
- break;
- case C_OPTION_TRACE_ON:
- r = bwb_OPTION_TRACE_ON( l );
- break;
- case C_OPTION_USING:
- r = bwb_OPTION_USING( l );
- break;
- case C_OPTION_USING_ALL:
- r = bwb_OPTION_USING_ALL( l );
- break;
- case C_OPTION_USING_COMMA:
- r = bwb_OPTION_USING_COMMA( l );
- break;
- case C_OPTION_USING_DIGIT:
- r = bwb_OPTION_USING_DIGIT( l );
- break;
- case C_OPTION_USING_DOLLAR:
- r = bwb_OPTION_USING_DOLLAR( l );
- break;
- case C_OPTION_USING_EXRAD:
- r = bwb_OPTION_USING_EXRAD( l );
- break;
- case C_OPTION_USING_FILLER:
- r = bwb_OPTION_USING_FILLER( l );
- break;
- case C_OPTION_USING_FIRST:
- r = bwb_OPTION_USING_FIRST( l );
- break;
- case C_OPTION_USING_LENGTH:
- r = bwb_OPTION_USING_LENGTH( l );
- break;
- case C_OPTION_USING_LITERAL:
- r = bwb_OPTION_USING_LITERAL( l );
- break;
- case C_OPTION_USING_MINUS:
- r = bwb_OPTION_USING_MINUS( l );
- break;
- case C_OPTION_USING_PERIOD:
- r = bwb_OPTION_USING_PERIOD( l );
- break;
- case C_OPTION_USING_PLUS:
- r = bwb_OPTION_USING_PLUS( l );
- break;
- case C_OPTION_VERSION:
- r = bwb_OPTION_VERSION( l );
- break;
- case C_OPTION_ZONE:
- r = bwb_OPTION_ZONE( l );
- break;
- case C_PAUSE:
- r = bwb_PAUSE( l );
- break;
- case C_PDEL:
- r = bwb_PDEL( l );
- break;
- case C_POP:
- r = bwb_POP( l );
- break;
- case C_PRINT:
- r = bwb_PRINT( l );
- break;
- case C_PTP:
- r = bwb_PTP( l );
- break;
- case C_PTR:
- r = bwb_PTR( l );
- break;
- case C_PUT:
- r = bwb_PUT( l );
- break;
- case C_QUIT:
- r = bwb_QUIT( l );
- break;
- case C_READ:
- r = bwb_READ( l );
- break;
- case C_RECALL:
- r = bwb_RECALL( l );
- break;
- case C_REM:
- r = bwb_REM( l );
- break;
- case C_RENAME:
- r = bwb_RENAME( l );
- break;
- case C_RENUM:
- r = bwb_RENUM( l );
- break;
- case C_RENUMBER:
- r = bwb_RENUMBER( l );
- break;
- case C_REPEAT:
- r = bwb_REPEAT( l );
- break;
- case C_REPLACE:
- r = bwb_REPLACE( l );
- break;
- case C_RESET:
- r = bwb_RESET( l );
- break;
- case C_RESTORE:
- r = bwb_RESTORE( l );
- break;
- case C_RESUME:
- r = bwb_RESUME( l );
- break;
- case C_RETURN:
- r = bwb_RETURN( l );
- break;
- case C_RSET:
- r = bwb_RSET( l );
- break;
- case C_RUN:
- r = bwb_RUN( l );
- break;
- case C_RUNNH:
- r = bwb_RUNNH( l );
- break;
- case C_SAVE:
- r = bwb_SAVE( l );
- break;
- case C_SCRATCH:
- r = bwb_SCRATCH( l );
- break;
- case C_SELECT:
- r = bwb_SELECT( l );
- break;
- case C_SELECT_CASE:
- r = bwb_SELECT_CASE( l );
- break;
- case C_STEP:
- r = bwb_STEP( l );
- break;
- case C_STOP:
- r = bwb_STOP( l );
- break;
- case C_STORE:
- r = bwb_STORE( l );
- break;
- case C_SUB:
- r = bwb_SUB( l );
- break;
- case C_SUB_END:
- r = bwb_SUB_END( l );
- break;
- case C_SUB_EXIT:
- r = bwb_SUB_EXIT( l );
- break;
- case C_SUBEND:
- r = bwb_SUBEND( l );
- break;
- case C_SUBEXIT:
- r = bwb_SUBEXIT( l );
- break;
- case C_SWAP:
- r = bwb_SWAP( l );
- break;
- case C_SYSTEM:
- r = bwb_SYSTEM( l );
- break;
- case C_TEXT:
- r = bwb_TEXT( l );
- break;
- case C_THEN:
- r = bwb_THEN( l );
- break;
- case C_TIMER:
- r = bwb_TIMER( l );
- break;
- case C_TIMER_OFF:
- r = bwb_TIMER_OFF( l );
- break;
- case C_TIMER_ON:
- r = bwb_TIMER_ON( l );
- break;
- case C_TIMER_STOP:
- r = bwb_TIMER_STOP( l );
- break;
- case C_TLOAD:
- r = bwb_TLOAD( l );
- break;
- case C_TO:
- r = bwb_TO( l );
- break;
- case C_TRACE:
- r = bwb_TRACE( l );
- break;
- case C_TRACE_OFF:
- r = bwb_TRACE_OFF( l );
- break;
- case C_TRACE_ON:
- r = bwb_TRACE_ON( l );
- break;
- case C_TSAVE:
- r = bwb_TSAVE( l );
- break;
- case C_TTY:
- r = bwb_TTY( l );
- break;
- case C_TTY_IN:
- r = bwb_TTY_IN( l );
- break;
- case C_TTY_OUT:
- r = bwb_TTY_OUT( l );
- break;
- case C_UNTIL:
- r = bwb_UNTIL( l );
- break;
- case C_USE:
- r = bwb_USE( l );
- break;
- case C_VARS:
- r = bwb_VARS( l );
- break;
- case C_WEND:
- r = bwb_WEND( l );
- break;
- case C_WHILE:
- r = bwb_WHILE( l );
- break;
- case C_WRITE:
- r = bwb_WRITE( l );
- break;
- default:
- WARN_INTERNAL_ERROR;
- r = l;
- break;
- }
- return r;
- }
-
- /* EOF */
|