/*************************************************************** bwb_prn.c Print and Error-Handling Commands 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 */ /* */ /*---------------------------------------------------------------*/ #include "bwbasic.h" static int buff_read_using (char *buffer, int *position, char *format_string, int format_length); static LineType *bwb_mat_dump (LineType * l, int IsWrite); static int bwb_print_at (LineType * l); static void CleanNumericString (char *prnbuf, int RemoveDot); static int CountDigits (char *Buffer); static LineType *D71_PUT (LineType * l); static LineType *file_write_matrix (LineType * l, char delimit); static LineType *H14_PUT (LineType * Line); static void internal_print (LineType * l, int IsCSV); static int is_magic_number (char *buffer); static int is_magic_string (char *buffer); static int line_read_using (LineType * l, char *format_string, int format_length); static void next_zone (void); static int parse_file_number (LineType * l); static void print_using_number (char *buffer, int *position, VariantType * e); static void print_using_string (char *buffer, int *position, VariantType * e); static void print_using_variant (char *buffer, int *position, VariantType * e, int IsCSV); static LineType *S70_PUT (LineType * l); static void xputc1 (char c); static void xputc2 (char c); static void xputs (char *buffer); /* We try to allow as many legacy PRINT USING formats as reasonable. Many legacy PRINT USING formats are incompatible with one another. For example: 1) some use '%' for strings, others use '%' for numbers, others consider '%' as a lieral. 2) some count a leading or traling signs in the width, while others do not. 3) when a value requires more digits than the assigned width: a) some truncate the displayed value to the width, b) some expand the width, c) some print a number of '%' or '*', and d) some halt processing. There is no perfect solution that will work for all possible dialects. */ #define PrintUsingNumberDigit My->CurrentVersion->OptionUsingDigit /* Digit placeholder, usually '#' */ #define PrintUsingNumberComma My->CurrentVersion->OptionUsingComma /* Comma, such as thousands, usually ',' */ #define PrintUsingNumberPeriod My->CurrentVersion->OptionUsingPeriod /* Period, such as dollars and cents, usually '.' */ #define PrintUsingNumberPlus My->CurrentVersion->OptionUsingPlus /* Plus sign, positive value, usually '+' */ #define PrintUsingNumberMinus My->CurrentVersion->OptionUsingMinus /* Minus sign, negative value, usually '-' */ #define PrintUsingNumberExponent My->CurrentVersion->OptionUsingExrad /* Exponential format, usually '^' */ #define PrintUsingNumberDollar My->CurrentVersion->OptionUsingDollar /* Currency symbol, usually '$' */ #define PrintUsingNumberFiller My->CurrentVersion->OptionUsingFiller /* Print filler, such as checks, usually '*' */ #define PrintUsingLiteral My->CurrentVersion->OptionUsingLiteral /* The next char is a literal, usually '_' */ #define PrintUsingStringFirst My->CurrentVersion->OptionUsingFirst /* The first character of the string, usually '!' */ #define PrintUsingStringAll My->CurrentVersion->OptionUsingAll /* Print the entire string, usually '&' */ #define PrintUsingStringLength My->CurrentVersion->OptionUsingLength /* Print a substring, usually '%' */ /* ** ** ZoneChar is a MAGIC character code used by file_write_matrix() to request printing by zones. ** ZoneChar can be any character, other than NulChar, that the user will not use as a literal delimiter. ** The user is allowed to specify CHR$(9), '\t', as a literal delimiter. ** */ #define ZoneChar 0x01 /* an unlikely literal delimiter */ int is_empty_string (char *Buffer) { if (Buffer == NULL) { return TRUE; } while (*Buffer == ' ') { Buffer++; } if (*Buffer == NulChar) { return TRUE; } return FALSE; } FileType * find_file_by_name (char *FileName) { FileType *F; if (is_empty_string (FileName)) { /* the rules for Console and Printer vary by command */ return NULL; } /* search the list of OPEN files */ assert( My != NULL ); for (F = My->FileHead; F != NULL; F = F->next) { assert( F != NULL ); if (F->DevMode == DEVMODE_CLOSED) { } else if (F->FileName == NULL) { } else if (bwb_stricmp (F->FileName, FileName) == 0) { /* FOUND */ return F; } } /* NOT FOUND */ return NULL; } FileType * find_file_by_number (int FileNumber) { FileType *F; /* handle MAGIC file numbers */ if (FileNumber <= 0) { /* the rules for Console and Printer vary by command */ return NULL; } /* search the list of OPEN files */ assert( My != NULL ); for (F = My->FileHead; F != NULL; F = F->next) { assert( F != NULL ); if (F->DevMode != DEVMODE_CLOSED) { if (F->FileNumber == FileNumber) { /* FOUND */ return F; } } } /* NOT FOUND */ return NULL; } FileType * file_new (void) { /* search for an empty slot. If not found, add a new slot. */ FileType *F; assert( My != NULL ); for (F = My->FileHead; F != NULL; F = F->next) { assert( F != NULL ); if (F->DevMode == DEVMODE_CLOSED) { /* FOUND */ return F; } } /* NOT FOUND */ if ((F = (FileType *) calloc (1, sizeof (FileType))) == NULL) { WARN_OUT_OF_MEMORY; return NULL; } assert( F != NULL ); F->next = My->FileHead; My->FileHead = F; return F; } void file_clear (FileType * F) { /* clean up a file slot that is no longer needed */ assert (F != NULL); clear_virtual_by_file (F->FileNumber); F->FileNumber = 0; F->DevMode = DEVMODE_CLOSED; /* DEVMODE_ item */ F->width = 0; /* width for OUTPUT and APPEND; reclen for RANDOM; not used for INPUT or BINARY */ F->col = 0; /* current column for OUTPUT and APPEND */ F->row = 0; /* current row for OUTPUT and APPEND */ F->EOF_LineNumber = 0; /* CBASIC-II: IF END # filenumber THEN linenumber */ F->delimit = NulChar; /* DELIMIT for READ and WRITE */ if (F->FileName != NULL) { free (F->FileName); F->FileName = NULL; } if (F->cfp != NULL) { bwb_fclose (F->cfp); F->cfp = NULL; } if (F->buffer != NULL) { free (F->buffer); F->buffer = NULL; } } int file_next_number (void) { int FileNumber; FileType *F; FileNumber = 0; assert( My != NULL ); for (F = My->FileHead; F != NULL; F = F->next) { assert( F != NULL ); if (F->DevMode != DEVMODE_CLOSED) { if (F->FileNumber > FileNumber) { FileNumber = F->FileNumber; } } } /* 'FileNumber' is the highest FileNumber that is currently open */ FileNumber++; return FileNumber; } /*************************************************************** FUNCTION: bwx_putc() DESCRIPTION: This function outputs a single character to the default output device. ***************************************************************/ static void CleanNumericString (char *prnbuf, int RemoveDot) { /* remove trailing zeroes */ char *E; char *D; assert (prnbuf != NULL); E = bwb_strchr (prnbuf, 'E'); if (E == NULL) { E = bwb_strchr (prnbuf, 'e'); } if (E) { /* SCIENTIFIC == SCALED notation */ /* trim leading zeroes in exponent */ char *F; char *G; F = E; while (bwb_isalpha (*F)) { F++; } while (*F == '+' || *F == '-') { /* skip sign */ F++; } G = F; while (*G == '0' || *G == ' ') { /* skip leading zeroes or spaces */ G++; } if (G > F) { bwb_strcpy (F, G); } G = NULL; /* no longer valid */ *E = NulChar; /* for bwb_strlen() */ } D = bwb_strchr (prnbuf, '.'); if (D) { int N; N = bwb_strlen (D); if (N > 1) { int M; N--; M = N; while (D[N] == '0') { /* remove trailing zeroes */ D[N] = '_'; N--; } if (RemoveDot) { if (E) { /* SCIENTIFIC == SCALED notation */ /* do NOT remove '.' */ } else { /* NORMAL == UNSCALED notation */ /* remove trailing '.' */ /* this will only occur for integer values */ while (D[N] == '.') { /* _###. POSITIVE INTEGER */ /* -###. NEGATIVE INTEGER */ D[N] = '_'; N--; } } } if (N < M) { if (E) { /* SCIENTIFIC == SCALED notation */ *E = 'E'; E = NULL; } N++; /* if INTEGER, then N == 0, else N > 0 */ M++; /* if SCIENTIFIC, then *M == 'E' else *M == NulChar */ bwb_strcpy (&(D[N]), &(D[M])); } } } if (E) { /* SCIENTIFIC == SCALED notation */ *E = 'E'; E = NULL; } if (prnbuf[1] == '0' && prnbuf[2] == '.') { /* _0.### POSITIVE FRACTION ==> _.### */ /* -0.### NEGATIVE FRACTION ==> -.### */ bwb_strcpy (&(prnbuf[1]), &(prnbuf[2])); } if (prnbuf[1] == '.' && prnbuf[2] == 'E') { /* _.E POSITIVE ZERO ==> _0 */ /* -.E NEGATIVE ZERO ==> _0 */ bwb_strcpy (prnbuf, " 0"); } } static int CountDigits (char *Buffer) { int NumDigits; char *P; assert (Buffer != NULL); /* determine the number of significant digits */ NumDigits = 0; P = Buffer; while (*P) { if (bwb_isalpha (*P)) { /* 'E', 'e', and so on. */ break; } if (bwb_isdigit (*P)) { NumDigits++; } P++; } return NumDigits; } extern void FormatBasicNumber (DoubleType Input, char *Output /* [ NUMLEN ] */ ) { /******************************************************************************* This is essentially sprintf( Output, "%g", Input ), except the rules for selecting between "%e", "%f", and "%d" are different. The C rules depend upon the value of the exponent. The BASIC rules depend upon the number of significant digits. The results of this routine have been verified by the NBS2 test suite, so... THINK VERY CAREFULLY BEFORE MAKING ANY CHANGES TO THIS ROUTINE. *******************************************************************************/ char *E; assert (Output != NULL); assert( My != NULL ); if (My->OptionScaleInteger >= 1 && My->OptionScaleInteger <= My->OptionDigitsInteger) { /* round */ DoubleType Scale; Scale = pow (10, My->OptionScaleInteger); assert( Scale != 0 ); Input = bwb_rint (Input * Scale) / Scale; } /* print in scientific form first, to determine exponent and significant digits */ sprintf (Output, "% 1.*E", My->OptionDigitsInteger - 1, Input); E = bwb_strchr (Output, 'E'); if (E == NULL) { E = bwb_strchr (Output, 'e'); } if (E) { /* valid */ int Exponent; int NumDigits; int DisplayDigits; int zz; char *F; /* pointer to the exponent's value */ F = E; while (bwb_isalpha (*F)) { F++; } Exponent = atoi (F); CleanNumericString (Output, FALSE); NumDigits = CountDigits (Output); DisplayDigits = MIN (NumDigits, My->OptionDigitsInteger); zz = MAX (Exponent, DisplayDigits - Exponent - 2); if (zz >= My->OptionDigitsInteger) { /* SCIENTIFIC */ sprintf (Output, "%# 1.*E", DisplayDigits - 1, Input); } else if (Input == (int) Input) { /* INTEGER */ sprintf (Output, "% *d", DisplayDigits, (int) Input); } else { /* FLOAT */ int Before; /* number of digits before the '.' */ int After; /* number of digits after the '.' */ Before = Exponent + 1; if (Before < 0) { Before = 0; } After = My->OptionDigitsInteger - Before; if (After < 0) { After = 0; } sprintf (Output, "%# *.*f", Before, After, Input); } CleanNumericString (Output, FALSE); } else { /* ERROR, NAN, INFINITY, ETC. */ } } LineType * bwb_LPRINT (LineType * l) { int IsCSV; assert (l != NULL); assert( My != NULL ); assert( My->SYSPRN != NULL ); My->CurrentFile = My->SYSPRN; IsCSV = FALSE; internal_print (l, IsCSV); return (l); } /*************************************************************** FUNCTION: bwb_print() DESCRIPTION: This function implements the BASIC PRINT command. SYNTAX: PRINT [# device-number,][USING format-string$;] expressions... ***************************************************************/ static int bwb_print_at (LineType * l) { int position; int r; int c; assert (l != NULL); position = 0; r = 0; c = 0; if (line_read_integer_expression (l, &position)) { /* OK */ } else { WARN_SYNTAX_ERROR; return FALSE; } if (line_skip_seperator (l)) { /* OK */ } else { WARN_SYNTAX_ERROR; return FALSE; } if (position < 0) { WARN_SYNTAX_ERROR; return FALSE; } assert( My != NULL ); assert( My->SYSOUT != NULL ); if (My->SYSOUT->width <= 0) { WARN_SYNTAX_ERROR; return FALSE; } if (My->SCREEN_ROWS <= 0) { WARN_SYNTAX_ERROR; return FALSE; } assert( My->CurrentFile == My->SYSOUT ); /* position is 0-based. 0 is top left, */ assert( My->CurrentFile != NULL ); assert( My->CurrentFile->width != 0 ); r = position / My->CurrentFile->width; c = position - r * My->CurrentFile->width; while (r >= My->SCREEN_ROWS) { r -= My->SCREEN_ROWS; } r++; /* 0-based to 1-based */ c++; /* 0-based to 1-based */ bwx_LOCATE (r, c); return TRUE; } static int parse_file_number (LineType * l) { /* ... # FileNumber , ... */ int FileNumber; assert (l != NULL); if (line_read_integer_expression (l, &FileNumber) == FALSE) { WARN_SYNTAX_ERROR; return FALSE; } assert( My != NULL ); assert( My->CurrentVersion != NULL ); if (My->CurrentVersion->OptionVersionValue & (C77)) { /* CBASIC-II: SERIAL & RANDOM file writes PRINT # file_number ; expression [, expression] ' SERIAL write PRINT # file_number , record_number ; expression [, expression] ' RANDOM write */ if (FileNumber <= 0) { WARN_BAD_FILE_NUMBER; return FALSE; } /* normal file */ My->CurrentFile = find_file_by_number (FileNumber); if (My->CurrentFile == NULL) { WARN_BAD_FILE_NUMBER; return FALSE; } if (line_skip_CommaChar (l) /* comma specific */ ) { /* PRINT # file_number , record_number ; expression [, expression] ' RANDOM write */ /* get the RecordNumber */ int RecordNumber; if ((My->CurrentFile->DevMode & DEVMODE_RANDOM) == 0) { WARN_BAD_FILE_MODE; return FALSE; } if (My->CurrentFile->width <= 0) { WARN_FIELD_OVERFLOW; return FALSE; } if (line_read_integer_expression (l, &RecordNumber) == FALSE) { WARN_SYNTAX_ERROR; return FALSE; } if (RecordNumber <= 0) { WARN_BAD_RECORD_NUMBER; return FALSE; } RecordNumber--; /* BASIC to C */ /* if( TRUE ) */ { long offset; offset = RecordNumber; offset *= My->CurrentFile->width; fseek (My->CurrentFile->cfp, offset, SEEK_SET); } } if (line_is_eol (l)) { /* PRINT # filenum */ /* PRINT # filenum , recnum */ } else if (line_skip_SemicolonChar (l) /* semicolon specific */ ) { /* PRINT # filenum ; */ /* PRINT # filenum , recnum ; */ } else { WARN_SYNTAX_ERROR; return FALSE; } return TRUE; } /* SERIAL file writes: PRINT # file_number PRINT # file_number [, expression] */ if (FileNumber < 0) { My->CurrentFile = My->SYSPRN; } else if (FileNumber == 0) { My->CurrentFile = My->SYSOUT; } else { /* normal file */ My->CurrentFile = find_file_by_number (FileNumber); } if (My->CurrentFile == NULL) { WARN_BAD_FILE_NUMBER; return FALSE; } if ((My->CurrentFile->DevMode & DEVMODE_WRITE) == 0) { WARN_BAD_FILE_NUMBER; return FALSE; } if (line_is_eol (l)) { /* PRINT # 2 */ } else if (line_skip_seperator (l)) { /* PRINT # 2 , ... */ } else { WARN_SYNTAX_ERROR; return FALSE; } return TRUE; } LineType * bwb_PRINT (LineType * l) { int IsCSV; assert (l != NULL); IsCSV = FALSE; assert( My != NULL ); if (My->IsPrinter == TRUE) { My->CurrentFile = My->SYSPRN; } else { My->CurrentFile = My->SYSOUT; } internal_print (l, IsCSV); return (l); } /*************************************************************** FUNCTION: internal_print() DESCRIPTION: This function implements the PRINT command, utilizing a specified file our output device. ***************************************************************/ static int buff_read_using (char *buffer, int *position, char *format_string, int format_length) { int p; assert (buffer != NULL); assert (position != NULL); assert (format_string != NULL); p = *position; if (buff_skip_word (buffer, &p, "USING")) { buff_skip_spaces (buffer, &p); /* keep this */ if (bwb_isdigit (buffer[p])) { /* PRINT USING ### */ int n; int LineNumber; LineType *x; char *C; char *F; n = 0; LineNumber = 0; x = NULL; if (buff_read_line_number (buffer, &p, &LineNumber) == FALSE) { WARN_SYNTAX_ERROR; return FALSE; } /* check for target label */ x = find_line_number (LineNumber); /* USING 100 */ if (x == NULL) { WARN_UNDEFINED_LINE; return FALSE; } /* line exists */ if (x->cmdnum != C_IMAGE) { WARN_UNDEFINED_LINE; return FALSE; } /* line contains IMAGE command */ C = x->buffer; C += x->Startpos; F = format_string; /* look for leading quote in IMAGE "..." */ while (*C == ' ') { C++; } assert( My != NULL ); assert( My->CurrentVersion != NULL ); if (*C == My->CurrentVersion->OptionQuoteChar) { /* QUOTED */ /* skip leading quote */ C++; while (*C != NulChar && *C != My->CurrentVersion->OptionQuoteChar) { /* copy format string, but not the trailing quote */ if (n == format_length) { WARN_STRING_TOO_LONG; break; } *F = *C; C++; F++; n++; } /* skip trailing quote */ } else { /* UNQUOTED */ while (*C) { /* copy format string verbatim */ if (n == format_length) { WARN_STRING_TOO_LONG; break; } *F = *C; C++; F++; n++; } } /* terminate format string */ *F = NulChar; if (buff_skip_seperator (buffer, &p) == FALSE) { WARN_SYNTAX_ERROR; return FALSE; } } else { { char *Value; Value = NULL; if (buff_read_string_expression (buffer, &p, &Value) == FALSE) { WARN_SYNTAX_ERROR; return FALSE; } if (Value == NULL) { WARN_SYNTAX_ERROR; return FALSE; } if (bwb_strlen (Value) > format_length) { WARN_STRING_TOO_LONG; Value[format_length] = NulChar; } bwb_strcpy (format_string, Value); free (Value); Value = NULL; } if (buff_skip_seperator (buffer, &p) == FALSE) { WARN_SYNTAX_ERROR; return FALSE; } } *position = p; return TRUE; } return FALSE; } static int line_read_using (LineType * l, char *format_string, int format_length) { assert (l != NULL); assert (format_string != NULL); return buff_read_using (l->buffer, &(l->position), format_string, format_length); } static void internal_print (LineType * l, int IsCSV) { /* if no arguments, simply print CR and return */ /* 1980 PRINT , , ,"A" */ int OutputCR; char *format_string; int format_length; int format_position; assert (l != NULL); OutputCR = TRUE; assert( My != NULL ); assert( My->ConsoleOutput != NULL ); assert( MAX_LINE_LENGTH > 1 ); format_string = My->ConsoleOutput; format_length = MAX_LINE_LENGTH; format_position = 0; format_string[0] = NulChar; if (line_skip_FilenumChar (l)) { /* PRINT # file, ... */ if (parse_file_number (l) == FALSE) { return; } assert( My->CurrentVersion != NULL ); if (My->CurrentVersion->OptionVersionValue & (C77) && My->CurrentFile->FileNumber > 0) { /* ** ** CBASIC-II files are CSV files. ** ** Strings are quoted other than PRINT USING. ** Comma seperator writes a literal comma. ** Semicolon seperator writes a literal comma. ** Numbers do NOT have leading or trailing spaces. ** */ IsCSV = TRUE; } OutputCR = TRUE; } else if (line_skip_AtChar (l)) { /* PRINT @ position, ... */ assert( My->SYSOUT != NULL ); My->CurrentFile = My->SYSOUT; if (bwb_print_at (l) == FALSE) { return; } OutputCR = TRUE; } else if (My->CurrentVersion->OptionVersionValue & (B15|T80|HB1|HB2) && line_skip_word (l, "AT")) { /* PRINT AT position, ... */ assert( My->SYSOUT != NULL ); My->CurrentFile = My->SYSOUT; if (bwb_print_at (l) == FALSE) { return; } OutputCR = TRUE; } assert( My->CurrentFile != NULL ); while (line_is_eol (l) == FALSE) { /* LOOP THROUGH PRINT ELEMENTS */ VariantType e; VariantType *E; E = &e; CLEAR_VARIANT (E); if (line_skip_CommaChar (l) /* comma-specific */ ) { if (format_string[0]) { /* PRINT USING active */ } else if (IsCSV) { xputc1 (','); } else { /* tab over */ next_zone (); } OutputCR = FALSE; } else if (line_skip_SemicolonChar (l) /* semicolon-specific */ ) { if (format_string[0]) { /* PRINT USING active */ } else if (IsCSV) { xputc1 (','); } else { /* concatenate strings */ } OutputCR = FALSE; } else if (line_read_using (l, format_string, format_length)) { format_position = 0; OutputCR = TRUE; } else if (line_read_expression (l, E)) /* internal_print */ { /* resolve the string */ if (My->IsErrorPending /* Keep This */ ) { /* ** ** this might look odd... ** but we want to abort printing on the first warning. ** The expression list could include a function with side-effects, ** so any error should immediately halt further evaluation. ** */ RELEASE_VARIANT (E); return; } print_using_variant (format_string, &format_position, E, IsCSV); RELEASE_VARIANT (E); OutputCR = TRUE; } else { WARN_SYNTAX_ERROR; return; } } if (OutputCR == TRUE) { /* did not end with ',' or ';' */ xputc1 ('\n'); } if (My->CurrentFile == My->SYSOUT) { /* FOR I = 1 TO 1000: PRINT "."; : NEXT I : PRINT */ fflush (My->SYSOUT->cfp); } } /*************************************************************** FUNCTION: print_using_variant() DESCRIPTION: This function gets the PRINT USING format string, returning a structure to the format. ***************************************************************/ static void print_using_number (char *buffer, int *position, VariantType * e) { /* Format a NUMBER. 'buffer' points to the beginning of a PRINT USING format string, such as "###.##". 'position' is the current offset in 'buffer'. 'e' is the current expression to print. */ int width; int precision; int exponent; char HeadChar; char FillChar; char CurrChar; char ComaChar; char TailChar; int p; char *tbuf; assert (buffer != NULL); assert (position != NULL); assert (e != NULL); width = 0; precision = 0; exponent = 0; HeadChar = ' '; FillChar = ' '; CurrChar = ' '; ComaChar = ' '; TailChar = ' '; assert( My != NULL ); assert( My->ConsoleInput != NULL ); tbuf = My->ConsoleInput; p = *position; while (IS_CHAR (buffer[p], PrintUsingNumberPlus) || IS_CHAR (buffer[p], PrintUsingNumberMinus)) { HeadChar = buffer[p]; width++; p++; } while (IS_CHAR (buffer[p], PrintUsingNumberFiller) || IS_CHAR (buffer[p], PrintUsingNumberDollar)) { if (IS_CHAR (buffer[p], PrintUsingNumberFiller)) { FillChar = PrintUsingNumberFiller; } else if (IS_CHAR (buffer[p], PrintUsingNumberDollar)) { CurrChar = PrintUsingNumberDollar; } width++; p++; } while (IS_CHAR (buffer[p], PrintUsingNumberDigit) || IS_CHAR (buffer[p], PrintUsingNumberComma)) { if (IS_CHAR (buffer[p], PrintUsingNumberComma)) { ComaChar = PrintUsingNumberComma; } width++; p++; } if (IS_CHAR (buffer[p], PrintUsingNumberPeriod)) { while (IS_CHAR (buffer[p], PrintUsingNumberPeriod)) { width++; p++; } while (IS_CHAR (buffer[p], PrintUsingNumberDigit)) { precision++; width++; p++; } } while (IS_CHAR (buffer[p], PrintUsingNumberExponent)) { exponent++; precision++; width++; p++; } while (IS_CHAR (buffer[p], PrintUsingNumberPlus) || IS_CHAR (buffer[p], PrintUsingNumberMinus)) { TailChar = buffer[p]; width++; p++; } /* format the number */ /* displaying both a Heading and a Trailing sign is NOT supported */ if (TailChar == ' ') { /* do nothing */ } else if (IS_CHAR (TailChar, PrintUsingNumberPlus) || IS_CHAR (TailChar, PrintUsingNumberMinus)) { /* force the sign to be printed, so we can move it */ HeadChar = TailChar; } else { WARN_INTERNAL_ERROR; return; } if (HeadChar == ' ') { /* only display a '-' sign */ if (exponent > 0) { sprintf (tbuf, "%*.*e", width, precision, e->Number); } else { sprintf (tbuf, "%*.*f", width, precision, e->Number); } } else if (IS_CHAR (HeadChar, PrintUsingNumberPlus) || IS_CHAR (HeadChar, PrintUsingNumberMinus)) { /* force a leading sign '+' or '-' */ if (exponent > 0) { sprintf (tbuf, "%+*.*e", width, precision, e->Number); } else { sprintf (tbuf, "%+*.*f", width, precision, e->Number); } } else { WARN_INTERNAL_ERROR; return; } if (TailChar == ' ') { /* do nothing */ } else if (IS_CHAR (TailChar, PrintUsingNumberPlus) || IS_CHAR (TailChar, PrintUsingNumberMinus)) { /* move sign '+' or '-' to end */ int i; int n; n = bwb_strlen (tbuf); for (i = 0; i < n; i++) { if (tbuf[i] != ' ') { if (IS_CHAR (tbuf[i], PrintUsingNumberPlus)) { tbuf[i] = ' '; if (IS_CHAR (TailChar, PrintUsingNumberPlus)) { /* TailChar of '+' does print a '+' */ bwb_strcat (tbuf, "+"); } else if (IS_CHAR (TailChar, PrintUsingNumberMinus)) { /* TailChar of '-' does NOT print a '+' */ bwb_strcat (tbuf, " "); } } else if (IS_CHAR (tbuf[i], PrintUsingNumberMinus)) { tbuf[i] = ' '; bwb_strcat (tbuf, "-"); } break; } } if (tbuf[0] == ' ') { n = bwb_strlen (tbuf); /* n > 0 */ for (i = 1; i < n; i++) { tbuf[i - 1] = tbuf[i]; } tbuf[n - 1] = NulChar; } } else { WARN_INTERNAL_ERROR; return; } if (CurrChar == ' ') { /* do nothing */ } else if (IS_CHAR (CurrChar, PrintUsingNumberDollar)) { int i; int n; n = bwb_strlen (tbuf); for (i = 0; i < n; i++) { if (tbuf[i] != ' ') { if (i > 0) { if (bwb_isdigit (tbuf[i])) { tbuf[i - 1] = CurrChar; } else { /* sign char */ tbuf[i - 1] = tbuf[i]; tbuf[i] = CurrChar; } } break; } } } else { WARN_INTERNAL_ERROR; return; } if (FillChar == ' ') { /* do nothing */ } else if (IS_CHAR (FillChar, PrintUsingNumberFiller)) { int i; int n; n = bwb_strlen (tbuf); for (i = 0; i < n; i++) { if (tbuf[i] != ' ') { break; } tbuf[i] = PrintUsingNumberFiller; } } else { WARN_INTERNAL_ERROR; return; } if (ComaChar == ' ') { xputs (tbuf); } else if (IS_CHAR (ComaChar, PrintUsingNumberComma)) { int dig_pos; int dec_pos; int i; int n; int commas; dig_pos = -1; dec_pos = -1; n = bwb_strlen (tbuf); for (i = 0; i < n; i++) { if ((bwb_isdigit (tbuf[i]) != 0) && (dig_pos == -1)) { dig_pos = i; } if ((tbuf[i] == PrintUsingNumberPeriod) && (dec_pos == -1)) { dec_pos = i; } if ((dig_pos != -1) && (dec_pos != -1)) { break; } } if (dig_pos == -1) { dec_pos = n; } if (dec_pos == -1) { dec_pos = n; } /* count the number of commas */ commas = 0; for (i = 0; i < n; i++) { if (((dec_pos - i) % 3 == 0) && (i > dig_pos) && (i < dec_pos)) { commas++; } } /* now, actually print */ for (i = 0; i < n; i++) { if (i < commas && tbuf[i] == FillChar) { /* Ignore the same number of leading spaces as there are commas. While not perfect for all possible cases, it is usually good enough for practical purposes. */ } else { if (((dec_pos - i) % 3 == 0) && (i > dig_pos) && (i < dec_pos)) { xputc1 (PrintUsingNumberComma); } xputc1 (tbuf[i]); } } } else { WARN_INTERNAL_ERROR; return; } *position = p; } static void print_using_string (char *buffer, int *position, VariantType * e) { /* Format a STRING. 'buffer' points to the beginning of a PRINT USING format string, such as "###.##". 'position' is the current offset in 'buffer'. 'e' is the current expression to print. */ int p; char *tbuf; assert (buffer != NULL); assert (position != NULL); assert (e != NULL); assert( My != NULL ); assert( My->NumLenBuffer != NULL ); p = *position; if (e->VariantTypeCode == StringTypeCode) { tbuf = e->Buffer; } else { tbuf = My->NumLenBuffer; FormatBasicNumber (e->Number, tbuf); } if (IS_CHAR (buffer[p], PrintUsingStringFirst)) { /* print first character only */ int i; i = 0; if (tbuf[i] == NulChar) { xputc1 (' '); } else { xputc1 (tbuf[i]); i++; } p++; } else if (IS_CHAR (buffer[p], PrintUsingStringAll)) { /* print entire string */ p++; xputs (tbuf); } else if (IS_CHAR (buffer[p], PrintUsingStringLength)) { /* print N characters or spaces */ int i; i = 0; if (tbuf[i] == NulChar) { xputc1 (' '); } else { xputc1 (tbuf[i]); i++; } p++; while (buffer[p] != NulChar && buffer[p] != PrintUsingStringLength) { if (tbuf[i] == NulChar) { xputc1 (' '); } else { xputc1 (tbuf[i]); i++; } p++; } if (IS_CHAR (buffer[p], PrintUsingStringLength)) { if (tbuf[i] == NulChar) { xputc1 (' '); } else { xputc1 (tbuf[i]); i++; } p++; } } *position = p; } static int is_magic_string (char *buffer) { /* for the character string pointed to 'buffer': return TRUE if it is a MagicString sequence, return FALSE otherwise. */ assert (buffer != NULL); /* 1 character sequences */ if (IS_CHAR (buffer[0], PrintUsingStringFirst)) { /* "!" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingStringAll)) { /* "&" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingStringLength)) { /* "%...%" */ return TRUE; } /* 2 character sequences */ /* 3 character sequences */ return FALSE; } static int is_magic_number (char *buffer) { /* for the character string pointed to 'buffer': return TRUE if it is a MagicNumber sequence, return FALSE otherwise. */ assert (buffer != NULL); /* 1 character sequences */ if (IS_CHAR (buffer[0], PrintUsingNumberDigit)) { /* "#" */ return TRUE; } /* 2 character sequences */ if (IS_CHAR (buffer[0], PrintUsingNumberFiller)) if (IS_CHAR (buffer[1], PrintUsingNumberFiller)) { /* "**" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingNumberDollar)) if (IS_CHAR (buffer[1], PrintUsingNumberDollar)) { /* "$$" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingNumberPlus)) if (IS_CHAR (buffer[1], PrintUsingNumberDigit)) { /* "+#" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingNumberMinus)) if (IS_CHAR (buffer[1], PrintUsingNumberDigit)) { /* "-#" */ return TRUE; } /* 3 character sequences */ if (IS_CHAR (buffer[0], PrintUsingNumberPlus)) if (IS_CHAR (buffer[1], PrintUsingNumberFiller)) if (IS_CHAR (buffer[2], PrintUsingNumberFiller)) { /* "+**" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingNumberPlus)) if (IS_CHAR (buffer[1], PrintUsingNumberDollar)) if (IS_CHAR (buffer[2], PrintUsingNumberDollar)) { /* "+$$" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingNumberMinus)) if (IS_CHAR (buffer[1], PrintUsingNumberFiller)) if (IS_CHAR (buffer[2], PrintUsingNumberFiller)) { /* "-**" */ return TRUE; } if (IS_CHAR (buffer[0], PrintUsingNumberMinus)) if (IS_CHAR (buffer[1], PrintUsingNumberDollar)) if (IS_CHAR (buffer[2], PrintUsingNumberDollar)) { /* "-$$" */ return TRUE; } return FALSE; } static void print_using_variant (char *buffer, int *position, VariantType * e, int IsCSV) { /* Format an EXPRESSION. 'buffer' points to the beginning of a PRINT USING format string, such as "###.##". 'position' is the current offset in 'buffer'. 'e' is the current expression to print. */ int IsUsed; assert (buffer != NULL); assert (position != NULL); assert (e != NULL); assert( My != NULL ); assert( My->NumLenBuffer != NULL ); /* PRINT A, B, C */ /* PRINT USING "", A, B, C */ /* PRINT USING "#", A, B, C */ IsUsed = FALSE; if (buffer[0]) { /* we have a format string */ int p; p = *position; if (p > 0 && buffer[p] == NulChar) { /* recycle the format string */ p = 0; } while (buffer[p]) { if (is_magic_string (&buffer[p])) { if (IsUsed) { /* stop here, ready for next string value */ break; } if (e->VariantTypeCode != StringTypeCode) { /* we are a number value, so we cannot match a magic string */ break; } /* magic and value are both string */ print_using_string (buffer, &p, e); IsUsed = TRUE; } else if (is_magic_number (&buffer[p])) { if (IsUsed) { /* stop here, ready for next number value */ break; } if (e->VariantTypeCode == StringTypeCode) { /* we are a string value, so we cannot match a magic number */ break; } /* magic and value are both number */ print_using_number (buffer, &p, e); IsUsed = TRUE; } else if (IS_CHAR (buffer[p], PrintUsingLiteral)) { /* print next character as literal */ p++; if (buffer[p] == NulChar) { /* PRINT USING "_" */ xputc1 (' '); } else { /* PRINT USING "_%" */ xputc1 (buffer[p]); p++; } } else { /* print this character as literal */ /* PRINT USING "A" */ xputc1 (buffer[p]); p++; } } *position = p; } if (IsUsed == FALSE) { /* we did not actually print the vlue */ if (e->VariantTypeCode == StringTypeCode) { /* ** ** PRINT A$ ** PRINT USING "";A$ ** PRINT USING "ABC";A$ ** */ if (IsCSV) { xputc1 ('\"'); xputs (e->Buffer); xputc1 ('\"'); } else { xputs (e->Buffer); } } else { /* ** ** PRINT X ** PRINT USING "";X ** PRINT USING "ABC";X ** ** [space]number[space] POSITIVE or ZERO ** [minus]number[space] NEGATIVE ** **/ char *tbuf; tbuf = My->NumLenBuffer; FormatBasicNumber (e->Number, tbuf); if (IsCSV) { char *P; P = tbuf; while (*P == ' ') { P++; } xputs (P); } else { xputs (tbuf); xputc1 (' '); } } } } /*************************************************************** FUNCTION: xputs() DESCRIPTION: This function outputs a null-terminated string to a specified file or output device. ***************************************************************/ static void xputs (char *buffer) { assert (buffer != NULL); assert( My != NULL ); assert (My->CurrentFile != NULL); if (My->CurrentFile->width > 0) { /* check to see if the width will be exceeded */ int n; n = My->CurrentFile->col + bwb_strlen (buffer) - 1; if (n > My->CurrentFile->width) { xputc1 ('\n'); } } /* output the string */ while (*buffer) { xputc1 (*buffer); buffer++; } } /*************************************************************** FUNCTION: next_zone() DESCRIPTION: Advance to the next print zone. ***************************************************************/ static void next_zone (void) { assert( My != NULL ); assert (My->CurrentFile != NULL); if (My->CurrentFile->width > 0) { /* ** ** check to see if width will be exceeded ** */ int LastZoneColumn; LastZoneColumn = 1; while (LastZoneColumn < My->CurrentFile->width) { LastZoneColumn += My->OptionZoneInteger; } LastZoneColumn -= My->OptionZoneInteger; if (My->CurrentFile->col >= LastZoneColumn) { /* ** ** width will be exceeded, so advance to a new line ** */ xputc1 ('\n'); return; } } /* ** ** advance to the next print zone ** */ if ((My->CurrentFile->col % My->OptionZoneInteger) == 1) { xputc1 (' '); } while ((My->CurrentFile->col % My->OptionZoneInteger) != 1) { xputc1 (' '); } } /*************************************************************** FUNCTION: xputc1() DESCRIPTION: This function outputs a character to a specified file or output device, checking to be sure the PRINT width is within the bounds specified for that device. ***************************************************************/ static void xputc1 (char c) { assert( My != NULL ); assert (My->CurrentFile != NULL); if (My->CurrentFile->width > 0) { /* ** ** check to see if width has been exceeded ** */ if (c != '\n') { /* ** ** REM this should print one line, not two lines ** WIDTH 80 ** PRINT SPACE$( 80 ) ** */ if (My->CurrentFile->col > My->CurrentFile->width) { xputc2 ('\n'); /* output LF */ } } } /* ** ** output the character ** */ xputc2 (c); } /*************************************************************** FUNCTION: xputc2() DESCRIPTION: This function sends a character to a specified file or output device. ***************************************************************/ static void xputc2 (char c) { assert( My != NULL ); assert (My->CurrentFile != NULL); assert (My->CurrentFile->cfp != NULL); assert( My->CurrentVersion != NULL ); if (c == '\n') { /* ** ** CBASIC-II: RANDOM files are padded on the right with spaces ** */ if (My->CurrentVersion->OptionVersionValue & (C77)) if (My->CurrentFile->DevMode & DEVMODE_RANDOM) if (My->CurrentFile->width > 0) { #if HAVE_MSDOS /* "\n" is converted to "\r\n" */ while (My->CurrentFile->col < (My->CurrentFile->width - 1)) #else /* ! HAVE_MSDOS */ while (My->CurrentFile->col < My->CurrentFile->width) #endif /* ! HAVE_MSDOS */ { fputc (' ', My->CurrentFile->cfp); My->CurrentFile->col++; } } /* ** ** output the character ** */ fputc (c, My->CurrentFile->cfp); /* ** ** NULLS ** */ if (My->LPRINT_NULLS > 0) if (My->CurrentFile == My->SYSPRN) if (My->CurrentFile->width > 0) { int i; for (i = 0; i < My->LPRINT_NULLS; i++) { fputc (NulChar, My->SYSPRN->cfp); } } /* ** ** update current column position ** */ My->CurrentFile->col = 1; My->CurrentFile->row++; return; } /* ** ** output the character ** */ fputc (c, My->CurrentFile->cfp); /* ** ** update current column position ** */ My->CurrentFile->col++; } extern void ResetConsoleColumn (void) { assert( My != NULL ); assert (My->SYSOUT != NULL); My->SYSOUT->col = 1; } static LineType * S70_PUT (LineType * l) { /* PUT filename$ , value [, ...] */ VariantType e; VariantType *E; assert (l != NULL); assert( My != NULL ); assert( My->CurrentVersion != NULL ); assert( My->NumLenBuffer != NULL ); E = &e; CLEAR_VARIANT (E); if (line_read_expression (l, E) == FALSE) /* bwb_PUT */ { goto EXIT; } if (E->VariantTypeCode == StringTypeCode) { /* STRING */ /* PUT filename$ ... */ if (is_empty_string (E->Buffer)) { /* PUT "" ... is an error */ WARN_BAD_FILE_NAME; goto EXIT; } My->CurrentFile = find_file_by_name (E->Buffer); if (My->CurrentFile == NULL) { /* implicitly OPEN for writing */ My->CurrentFile = file_new (); My->CurrentFile->cfp = fopen (E->Buffer, "w"); if (My->CurrentFile->cfp == NULL) { WARN_BAD_FILE_NAME; goto EXIT; } My->CurrentFile->FileNumber = file_next_number (); My->CurrentFile->DevMode = DEVMODE_OUTPUT; My->CurrentFile->width = 0; /* WIDTH == RECLEN */ My->CurrentFile->col = 1; My->CurrentFile->row = 1; My->CurrentFile->delimit = ','; My->CurrentFile->buffer = NULL; if (My->CurrentFile->FileName != NULL) { free (My->CurrentFile->FileName); My->CurrentFile->FileName = NULL; } My->CurrentFile->FileName = E->Buffer; E->Buffer = NULL; } } else { /* NUMBER -- file must already be OPEN */ /* PUT filenumber ... */ if (E->Number < 0) { /* "PUT # -1" is an error */ WARN_BAD_FILE_NUMBER; goto EXIT; } if (E->Number == 0) { /* "PUT # 0" is an error */ WARN_BAD_FILE_NUMBER; goto EXIT; } /* normal file */ My->CurrentFile = find_file_by_number ((int) bwb_rint (E->Number)); if (My->CurrentFile == NULL) { /* file not OPEN */ WARN_BAD_FILE_NUMBER; goto EXIT; } } if (My->CurrentFile == NULL) { WARN_BAD_FILE_NUMBER; goto EXIT; } if ((My->CurrentFile->DevMode & DEVMODE_WRITE) == 0) { WARN_BAD_FILE_NUMBER; goto EXIT; } if (line_is_eol (l)) { /* PUT F$ */ /* PUT #1 */ xputc1 ('\n'); goto EXIT; } else if (line_skip_seperator (l)) { /* OK */ } else { WARN_SYNTAX_ERROR; goto EXIT; } /* loop through elements */ while (line_is_eol (l) == FALSE) { while (line_skip_seperator (l)) { /* PUT F$, ,,,A,,,B,,, */ /* PUT #1, ,,,A,,,B,,, */ xputc1 (My->CurrentFile->delimit); } if (line_is_eol (l) == FALSE) { /* print this item */ CLEAR_VARIANT (E); if (line_read_expression (l, E) == FALSE) /* bwb_PUT */ { goto EXIT; } if (E->VariantTypeCode == StringTypeCode) { /* STRING */ xputc1 (My->CurrentVersion->OptionQuoteChar); xputs (E->Buffer); xputc1 (My->CurrentVersion->OptionQuoteChar); } else { /* NUMBER */ char *tbuf; tbuf = My->NumLenBuffer; FormatBasicNumber (E->Number, tbuf); xputs (tbuf); } RELEASE_VARIANT (E); } } /* print LF */ xputc1 ('\n'); /* OK */ EXIT: RELEASE_VARIANT (E); return (l); } static LineType * D71_PUT (LineType * l) { /* PUT # file_number [ , RECORD record_number ] */ int file_number; assert (l != NULL); assert( My != NULL ); file_number = 0; if (line_skip_FilenumChar (l)) { /* OPTIONAL */ } if (line_read_integer_expression (l, &file_number) == FALSE) { WARN_BAD_FILE_NUMBER; return (l); } if (file_number < 1) { WARN_BAD_FILE_NUMBER; return (l); } My->CurrentFile = find_file_by_number (file_number); if (My->CurrentFile == NULL) { WARN_BAD_FILE_NUMBER; return (l); } if (My->CurrentFile->DevMode != DEVMODE_RANDOM) { WARN_BAD_FILE_NUMBER; return (l); } if (My->CurrentFile->width <= 0) { WARN_BAD_FILE_NUMBER; return (l); } if (line_is_eol (l)) { /* PUT # file_number */ } else { /* PUT # file_number , RECORD record_number */ int record_number; long offset; record_number = 0; offset = 0; if (line_skip_seperator (l) == FALSE) { WARN_SYNTAX_ERROR; return (l); } if (line_skip_word (l, "RECORD") == FALSE) { WARN_SYNTAX_ERROR; return (l); } if (line_read_integer_expression (l, &record_number) == FALSE) { WARN_BAD_RECORD_NUMBER; return (l); } if (record_number <= 0) { WARN_BAD_RECORD_NUMBER; return (l); } record_number--; /* BASIC to C */ offset = record_number; offset *= My->CurrentFile->width; if (fseek (My->CurrentFile->cfp, offset, SEEK_SET) != 0) { WARN_BAD_RECORD_NUMBER; return (l); } } field_put (My->CurrentFile); /* if( TRUE ) */ { int i; for (i = 0; i < My->CurrentFile->width; i++) { char c; c = My->CurrentFile->buffer[i]; fputc (c, My->CurrentFile->cfp); } } /* OK */ return (l); } static LineType * H14_PUT (LineType * Line) { /* PUT # FileNumber [ , RecordNumber ] ' RANDOM */ /* PUT # FileNumber , [ BytePosition ] , scalar [,...] ' BINARY */ int file_number; assert (Line != NULL); assert( My != NULL ); file_number = 0; if (line_skip_FilenumChar (Line)) { /* OPTIONAL */ } if (line_read_integer_expression (Line, &file_number) == FALSE) { WARN_BAD_FILE_NUMBER; return (Line); } if (file_number < 1) { WARN_BAD_FILE_NUMBER; return (Line); } My->CurrentFile = find_file_by_number (file_number); if (My->CurrentFile == NULL) { WARN_BAD_FILE_NUMBER; return (Line); } if (My->CurrentFile->DevMode == DEVMODE_RANDOM) { /* PUT # FileNumber [ , RecordNumber ] ' RANDOM */ if (My->CurrentFile->width <= 0) { WARN_BAD_FILE_NUMBER; return (Line); } if (line_is_eol (Line)) { /* PUT # file_number */ } else { /* PUT # FileNumber , RecordNumber ' RANDOM */ int record_number; long offset; record_number = 0; offset = 0; if (line_skip_seperator (Line) == FALSE) { WARN_SYNTAX_ERROR; return (Line); } if (line_read_integer_expression (Line, &record_number) == FALSE) { WARN_BAD_RECORD_NUMBER; return (Line); } if (record_number <= 0) { WARN_BAD_RECORD_NUMBER; return (Line); } record_number--; /* BASIC to C */ offset = record_number; offset *= My->CurrentFile->width; if (fseek (My->CurrentFile->cfp, offset, SEEK_SET) != 0) { WARN_BAD_RECORD_NUMBER; return (Line); } } field_put (My->CurrentFile); /* if( TRUE ) */ { int i; for (i = 0; i < My->CurrentFile->width; i++) { char c; c = My->CurrentFile->buffer[i]; fputc (c, My->CurrentFile->cfp); } } /* OK */ return (Line); } else if (My->CurrentFile->DevMode == DEVMODE_BINARY) { /* PUT # FileNumber , [ BytePosition ] , scalar [,...] ' BINARY */ if (line_skip_seperator (Line) == FALSE) { WARN_SYNTAX_ERROR; return (Line); } if (line_skip_seperator (Line)) { /* BytePosition not provided */ } else { int RecordNumber; long offset; RecordNumber = 0; offset = 0; if (line_read_integer_expression (Line, &RecordNumber) == FALSE) { WARN_BAD_RECORD_NUMBER; return (Line); } if (RecordNumber <= 0) { WARN_BAD_RECORD_NUMBER; return (Line); } RecordNumber--; /* BASIC to C */ offset = RecordNumber; if (fseek (My->CurrentFile->cfp, offset, SEEK_SET) != 0) { WARN_BAD_RECORD_NUMBER; return (Line); } if (line_skip_seperator (Line) == FALSE) { WARN_SYNTAX_ERROR; return (Line); } } do { VariableType *v; if ((v = line_read_scalar (Line)) == NULL) { WARN_SYNTAX_ERROR; return (Line); } if (binary_get_put (v, TRUE) == FALSE) { WARN_SYNTAX_ERROR; return (Line); } } while (line_skip_seperator (Line)); /* OK */ return (Line); } WARN_BAD_FILE_MODE; return (Line); } extern LineType * bwb_PUT (LineType * Line) { assert (Line != NULL); assert( My != NULL ); assert( My->CurrentVersion != NULL ); if (My->CurrentVersion->OptionVersionValue & (S70 | I70 | I73)) { return S70_PUT (Line); } if (My->CurrentVersion->OptionVersionValue & (D71 | R86)) { return D71_PUT (Line); } if (My->CurrentVersion->OptionVersionValue & (H14)) { return H14_PUT (Line); } WARN_INTERNAL_ERROR; return (Line); } /*************************************************************** FUNCTION: bwb_write() DESCRIPTION: This C function implements the BASIC WRITE command. SYNTAX: WRITE [# device-number,] element [, element ].... ***************************************************************/ extern LineType * bwb_WRITE (LineType * l) { int IsCSV; assert (l != NULL); IsCSV = TRUE; assert( My != NULL ); assert( My->SYSOUT != NULL ); My->CurrentFile = My->SYSOUT; internal_print (l, IsCSV); return (l); } static LineType * file_write_matrix (LineType * l, char delimit) { /* MAT PRINT [ # filenumber , ] matrix [;|,] ... */ /* MAT WRITE [ # filenumber , ] matrix [;|,] ... */ /* MAT PUT filename$ , matrix [;|,] ... */ /* MAT PUT filenumber , matrix [;|,] ... */ /* Array must be 1, 2 or 3 dimensions */ /* Array may be either NUMBER or STRING */ assert (l != NULL); do { VariableType *v; char ItemSeperator; /* get matrix name */ if ((v = line_read_matrix (l)) == NULL) { WARN_SUBSCRIPT_OUT_OF_RANGE; return (l); } /* variable MUST be an array of 1, 2 or 3 dimensions */ if (v->dimensions < 1) { WARN_SUBSCRIPT_OUT_OF_RANGE; return (l); } if (v->dimensions > 3) { WARN_SUBSCRIPT_OUT_OF_RANGE; return (l); } /* ** ** This may look odd, but MAT PRINT is special. ** The variable seperator AFTER the variable determines how the variable's values are printed. ** The number of dimension determines: ** a) the meaning of comma (,) and semicolon (;) ** b) the default of row-by-row or col-by-col ** */ ItemSeperator = NulChar; /* concatenate the columns */ if (line_skip_CommaChar (l) /* comma-specific */ ) { /* ** ** force printing with the specified delimiter, ** which is usually a Comma but can be any character. ** */ ItemSeperator = delimit; /* for MAT PRINT this is forced to be a ZoneChar */ } else if (line_skip_SemicolonChar (l) /* semicolon-specific */ ) { /* ** ** force concatenating the columns, ** ignoring the specified delimiter. ** */ ItemSeperator = NulChar; } else { /* ** ** default the item seperator based upon variable's dimensions ** */ switch (v->dimensions) { case 1: /* by default, a one dimension array is printed row-by-row */ ItemSeperator = '\n'; break; case 2: /* by default, a two dimension array is printed col-by-col */ ItemSeperator = delimit; break; case 3: /* by default, a three dimension array is printed col-by-col */ ItemSeperator = delimit; break; } } /* print array */ switch (v->dimensions) { case 1: { /* OPTION BASE 0 DIM A(5) ... MAT PRINT A ... FOR I = 0 TO 5 PRINT A(I) NEXT I ... */ for (v->VINDEX[0] = v->LBOUND[0]; v->VINDEX[0] <= v->UBOUND[0]; v->VINDEX[0]++) { VariantType variant; CLEAR_VARIANT (&variant); if (v->VINDEX[0] > v->LBOUND[0]) { switch (ItemSeperator) { case NulChar: break; case ZoneChar: next_zone (); break; default: xputc1 (ItemSeperator); } } if (var_get (v, &variant) == FALSE) { WARN_VARIABLE_NOT_DECLARED; return (l); } if (variant.VariantTypeCode == StringTypeCode) { xputs (variant.Buffer); } else { char *tbuf; tbuf = My->NumLenBuffer; FormatBasicNumber (variant.Number, tbuf); xputs (tbuf); } } xputc1 ('\n'); } break; case 2: { /* OPTION BASE 0 DIM B(2,3) ... MAT PRINT B ... FOR I = 0 TO 2 FOR J = 0 TO 3 PRINT B(I,J), NEXT J PRINT NEXT I ... */ for (v->VINDEX[0] = v->LBOUND[0]; v->VINDEX[0] <= v->UBOUND[0]; v->VINDEX[0]++) { for (v->VINDEX[1] = v->LBOUND[1]; v->VINDEX[1] <= v->UBOUND[1]; v->VINDEX[1]++) { VariantType variant; CLEAR_VARIANT (&variant); if (v->VINDEX[1] > v->LBOUND[1]) { switch (ItemSeperator) { case NulChar: break; case ZoneChar: next_zone (); break; default: xputc1 (ItemSeperator); } } if (var_get (v, &variant) == FALSE) { WARN_VARIABLE_NOT_DECLARED; return (l); } if (variant.VariantTypeCode == StringTypeCode) { xputs (variant.Buffer); } else { char *tbuf; tbuf = My->NumLenBuffer; FormatBasicNumber (variant.Number, tbuf); xputs (tbuf); } } xputc1 ('\n'); } } break; case 3: { /* OPTION BASE 0 DIM C(2,3,4) ... MAT PRINT C ... FOR I = 0 TO 2 FOR J = 0 TO 3 FOR K = 0 TO 4 PRINT C(I,J,K), NEXT K PRINT NEXT J PRINT NEXT I ... */ for (v->VINDEX[0] = v->LBOUND[0]; v->VINDEX[0] <= v->UBOUND[0]; v->VINDEX[0]++) { for (v->VINDEX[1] = v->LBOUND[1]; v->VINDEX[1] <= v->UBOUND[1]; v->VINDEX[1]++) { for (v->VINDEX[2] = v->LBOUND[2]; v->VINDEX[2] <= v->UBOUND[2]; v->VINDEX[2]++) { VariantType variant; CLEAR_VARIANT (&variant); if (v->VINDEX[2] > v->LBOUND[2]) { switch (ItemSeperator) { case NulChar: break; case ZoneChar: next_zone (); break; default: xputc1 (ItemSeperator); } } if (var_get (v, &variant) == FALSE) { WARN_VARIABLE_NOT_DECLARED; return (l); } if (variant.VariantTypeCode == StringTypeCode) { xputs (variant.Buffer); } else { char *tbuf; tbuf = My->NumLenBuffer; FormatBasicNumber (variant.Number, tbuf); xputs (tbuf); } } xputc1 ('\n'); } xputc1 ('\n'); } } break; } /* process the next variable, if any */ } while (line_is_eol (l) == FALSE); return (l); } extern LineType * bwb_MAT_PUT (LineType * l) { /* MAT PUT filename$ , matrix [;|,] ... */ /* MAT PUT filenumber , matrix [;|,] ... */ /* Array must be 1, 2 or 3 dimensions */ /* Array may be either NUMBER or STRING */ VariantType x; VariantType *X; assert (l != NULL); assert( My != NULL ); assert( My->SYSOUT != NULL ); My->CurrentFile = My->SYSOUT; X = &x; CLEAR_VARIANT (X); if (line_read_expression (l, X) == FALSE) /* bwb_MAT_PUT */ { goto EXIT; } if (X->VariantTypeCode == StringTypeCode) { /* STRING */ /* MAT PUT filename$ ... */ if (is_empty_string (X->Buffer)) { /* MAT PUT "" ... is an error */ WARN_BAD_FILE_NAME; goto EXIT; } My->CurrentFile = find_file_by_name (X->Buffer); if (My->CurrentFile == NULL) { /* implicitly OPEN for writing */ My->CurrentFile = file_new (); My->CurrentFile->cfp = fopen (X->Buffer, "w"); if (My->CurrentFile->cfp == NULL) { WARN_BAD_FILE_NAME; goto EXIT; } My->CurrentFile->FileNumber = file_next_number (); My->CurrentFile->DevMode = DEVMODE_OUTPUT; My->CurrentFile->width = 0; /* WIDTH == RECLEN */ My->CurrentFile->col = 1; My->CurrentFile->row = 1; My->CurrentFile->delimit = ','; My->CurrentFile->buffer = NULL; if (My->CurrentFile->FileName != NULL) { free (My->CurrentFile->FileName); My->CurrentFile->FileName = NULL; } My->CurrentFile->FileName = X->Buffer; X->Buffer = NULL; } } else { /* NUMBER -- file must already be OPEN */ /* MAT PUT filenumber ... */ if (X->Number < 0) { /* "MAT PUT # -1" is an error */ WARN_BAD_FILE_NUMBER; goto EXIT; } if (X->Number == 0) { /* "MAT PUT # 0" is an error */ WARN_BAD_FILE_NUMBER; goto EXIT; } /* normal file */ My->CurrentFile = find_file_by_number ((int) bwb_rint (X->Number)); if (My->CurrentFile == NULL) { /* file not OPEN */ WARN_BAD_FILE_NUMBER; goto EXIT; } } RELEASE_VARIANT (X); if (My->CurrentFile == NULL) { WARN_BAD_FILE_NUMBER; goto EXIT; } if ((My->CurrentFile->DevMode & DEVMODE_WRITE) == 0) { WARN_BAD_FILE_NUMBER; goto EXIT; } if (line_skip_seperator (l)) { /* OK */ } else { WARN_SYNTAX_ERROR; goto EXIT; } return file_write_matrix (l, My->CurrentFile->delimit); EXIT: RELEASE_VARIANT (X); return (l); } static LineType * bwb_mat_dump (LineType * l, int IsWrite) { /* MAT PRINT [ # filenumber , ] matrix [;|,] ... */ /* MAT WRITE [ # filenumber , ] matrix [;|,] ... */ /* Array must be 1, 2 or 3 dimensions */ /* Array may be either NUMBER or STRING */ char delimit; assert (l != NULL); assert( My != NULL ); assert( My->SYSOUT != NULL ); My->CurrentFile = My->SYSOUT; if (line_skip_FilenumChar (l)) { /* ... # file, ... */ if (parse_file_number (l) == FALSE) { return (l); } if (line_is_eol (l)) { WARN_SYNTAX_ERROR; return (l); } } if (IsWrite) { /* MAT WRITE */ delimit = My->CurrentFile->delimit; } else { /* MAT PRINT */ delimit = ZoneChar; } return file_write_matrix (l, delimit); } extern LineType * bwb_MAT_WRITE (LineType * l) { assert (l != NULL); return bwb_mat_dump (l, TRUE); } extern LineType * bwb_MAT_PRINT (LineType * l) { assert (l != NULL); return bwb_mat_dump (l, FALSE); } /* EOF */