Mercurial > ~dholland > hg > ag > index.cgi
diff examples/dsl/dsl.syn @ 0:13d2b8934445
Import AnaGram (near-)release tree into Mercurial.
author | David A. Holland |
---|---|
date | Sat, 22 Dec 2007 17:52:45 -0500 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/examples/dsl/dsl.syn Sat Dec 22 17:52:45 2007 -0500 @@ -0,0 +1,1461 @@ +{ // C Prologue +/***** + + AnaGram Programming Examples + + A Dos Script Language + + Copyright 1993 Parsifal Software. All Rights Reserved. + + This software is provided 'as-is', without any express or implied + warranty. In no event will the authors be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + 3. This notice may not be removed or altered from any source distribution. + +*****/ + +#include "stack.h" +#include "charsink.h" +#include "strdict.h" +#include "array.h" +#include "symbol.h" +#include "query.h" + +#ifdef __BCPLUSPLUS__ +extern unsigned _stklen = 0x4000; // set stack size +#endif + + +// Define stacks for temporary storage + +stack <action_pointer> as(25); // Stack actions +stack <int> is(100); // Stack string indices +stack <char *> ps(1000,20); // Stack parameter strings +stack <query_item> qs(23); // Stack query items + + +// Define data structures for symbol table + +#define N_STRINGS 2000 + +string_accumulator sa(64000U,500); +string_dictionary sd(N_STRINGS); +array <symbol_table_entry> st(N_STRINGS); + +} // End of C Prologue + + +// Character Set Definitions + +digit = '0-9' +eof = 0 + ^Z +letter = 'a-z' + 'A-Z' + '_' +not double quote = ~eof - ('"' + '\\' + '\n') +not eol = ~(eof + '\n') +not paren = ~(eof + '(' + ')') +not single quote = ~eof - ('\'' + '\\' + '\n') +operator = '#' + '=' + '<' + '>' + '|' +punctuation = '(' + ')' + '{' + '}' + '[' + ']' + '"' + '\n' +text char = ~(eof + operator + white + punctuation + '@') +white = ' ' + '\t' + '\v' + '\f' + '\r' + + +// Configuration Section + +[ + // White space control + disregard ws + lexeme {literal, integer constant, string literal, paren string, + character constant, eol, literals, name} + distinguish lexemes + + // parser configuration + pointer input + context type = action_pointer + distinguish keywords {letter + digit} + parser file name = "#.cpp" + far tables + + // Debugging options + line numbers + test file mask = "*.dsl" +] + + +// White Space Definitions + +ws + -> white | comment + +comment + -> comment head, "*/" + +comment head + -> "/*" + -> comment head, ~eof + +// Comment out one of the following two productions to determine whether +// comments nest or not + +comment head + -> comment head, comment // comments nest + +/* +comment + -> comment head, comment // comments do not nest +*/ + +eol + -> '\n' + -> "//", not eol?..., '\n' // C++ style comments + + + +// Script File Description + +script file $ + -> [execution block | declaration | eol]..., eof + + +// Dos Command and Parameter Recognition + +word + -> paren string + -> string literal + -> integer variable:v =++sa << sd[v], lookup(); + -> string variable:v =++sa << sd[v], lookup(); + -> undeclared variable:v =++sa << sd[v], lookup(); + -> word, '[', !sa << '[';, + parameter string, ']' =concat(sa) << ']', lookup(); + +string + -> word + -> string, '#', word =concat(sa); + +parameter string + -> param word + -> parameter string, '#', + param word =concat(sa); + +literal + -> text char:c =++sa << c; + -> literal, text char:c =sa << c; + +param word + -> word + -> action text =action_string(); + + +// Gather, but do not execute, the text of an action block + +action text + -> action text head, '}' + +action text head + -> '{' =as << CONTEXT; + -> action text head, action word + -> action text head, eol + -> action text head, action text ={action_pointer a; as >> a;} + -> action text head, operator+'@' + + +action word + -> paren string =--sa; + -> string literal =--sa; + -> literal =--sa; + -> action word, '[', action parameter string, ']' + +action parameter string + -> action param word + -> action parameter string, '#', + action param word + +action param word + -> action word + -> action text + + +/***** + + Parenthesized string + + May contain any characters inside balanced parentheses. If parentheses + are included, they must balance. Outer parentheses are stripped before + use. + +*****/ + +paren string + -> paren string chars, ')' + +paren string chars + -> '(' =++sa; + -> paren string chars, paren string char + +paren string char + -> not paren:c =sa << c; + -> !sa << '(';, paren string chars, ')' =concat(sa) << ')'; + + +/***** + + String Literal + + Follows the same rules as for string literals in C and C++ + +*****/ + +string literal + -> string chars, '"' + +string chars + -> '"' =++sa; + -> string chars, string char + +string char + -> not double quote:c =sa << c; + -> escape sequence:c =sa << c; + +(int) escape sequence + -> "\\a" ='\a'; + -> "\\b" ='\b'; + -> "\\f" ='\f'; + -> "\\n" ='\n'; + -> "\\r" ='\r'; + -> "\\t" ='\t'; + -> "\\v" ='\v'; + -> "\\\\" ='\\'; + -> "\\?" = '\?'; + -> "\\'" ='\''; + -> "\\\"" ='"'; + -> octal escape + -> hex escape + +(int) octal escape + -> one octal | two octal | three octal + +(int) one octal + -> '\\', '0-7':d =d-'0'; + +(int) two octal + -> one octal:n, '0-7':d =8*n + d-'0'; + +(int) three octal + -> two octal:n, '0-7':d =8*n + d-'0'; + +(int) hex escape + -> "\\x", hex number:n =(int) n; + +(long) hex number + -> hex digit + -> hex number:n, hex digit:d =16*n + d; + +[ + sticky {one octal, two octal, hex number} +] + + +/***** + + Command Line Interpretation + + The identifier may be the name of a DOS command, internal or external, + a path name of an arbitrary executable, or an internal commmand of the + scripting language. It may appear literally, or may be the result of + string concatenation and substitution. + + command is used in the program logic section, below. + +*****/ + +command + -> identifier, parameters? =exec(); + -> identifier, parameters?, + '<', parameter string =exec_redirect_in(); + -> piped command:file, '|', + identifier, parameters? =exec_pipe_in(file); + -> piped command:file, '>', + parameter string =grab_output(file); + -> piped command:file, ">>", + parameter string =append_output(file); + +(char *) piped command + -> identifier, parameters? =exec_pipe_out(); + -> identifier, parameters?, + '<', parameter string =exec_redirect_in_pipe_out(); + -> piped command:file, '|', + identifier, parameters? =exec_pipe_in_pipe_out(file); + +identifier + -> string =sa << 0, ++ps << (sa.top()); + +parameters + -> parameter string =ps << (sa.top()), sa << 0; + -> parameters, parameter string =ps << (sa.top()), sa << 0; + + +/***** + + Program logic. + + This section of syntax controls the interpretation of a sequence + of commands enclosed within braces. + +*****/ + +execution block + -> '{', eol?...,[command sequence, eol?... | if sequence, eol?...], '}' + + +/***** + + A command sequence is any sequence of statements and + if statements that ends with a statement. + +*****/ + +command sequence + -> statement + -> command sequence, eol..., statement + -> if sequence, eol..., statement + -> if sequence:pc, eol?..., + "else", action text =do_if(pc,1); + +/***** + + An if sequence is any direct sequence of statements and if statements + that ends with an if statement. The difference between an "if sequence" and + a "command sequence" is that an else clause may follow the "if sequence". + +*****/ + +(int) if sequence + -> if statement + -> if sequence, eol..., if statement:cc =cc; + -> command sequence, eol..., if statement:cc =cc; + -> if sequence:pc, eol?..., "else", eol?..., + if condition:cc, action text =do_if(pc,cc!=0); + +(int) if condition + -> "if", + '(', conditional exp:cc, ')' =(int) cc; + +(int) if statement + -> if condition:cc, action text =do_if(0,cc != 0); + +/***** + + A statement is any command that isn't an if or else statement. + The iteration on the while statement is a ruse. + +*****/ + +statement + -> command + -> assignment + -> for statement + -> declaration + -> screen description + -> while statement... + + +/***** + + Assignment statements + + There are four varieties of assignment statement depending on whether + or how the variable on the left hand side has been previously declared. + +*****/ + +assignment + -> undeclared variable:v, '=', + parameter string =assign_value(v); + -> integer variable:v, '=', + conditional exp:x, ';'? =st[v].data.integer = (int) x; + -> string variable:v, '=', + string exp, ';'? =st[v].data.text = copy(sa--); + -> string variable:v, '@', + primary exp:n, '=', + conditional exp:x, ';'? =st[v].data.text[(unsigned)n] = (char) x; + + +/***** + + Semantically determined production to determine treatment of variable + on left side of assignment statement. + +*****/ + +(unsigned) integer variable, + string variable, + undeclared variable + -> literal =check_integer(); + + +/***** + + While Statement + + The While statement loops by simply resetting the input pointer for + the parser back to the beginning of the while loop. This is the reason + for the iteration of the while statement in the production for "simple + command". + +*****/ + +while statement + -> "while", '(', conditional exp:cc, ')', + action text =do_while(cc != 0); + + +/***** + + + + For Statement + + This for statement corresponds to the for statement in the DOS batch + programming language, not the for statement in C. + +*****/ + +for statement + -> "for", name, // !++sa << '(';, + "in", parameter string, + "do"?, action text =do_for_loop(); + + +/***** + + Declaration statements + +*****/ + +declaration + -> "action", literals:n, + action text =define_action(n); + -> "int", + name, '=', conditional exp:x, ';'? =define_integer((sa--).top(), x); + -> "string", + name, '=', string exp, ';'? =define_string(); + +(int) literals + -> literal =0; + -> literals:n, ws..., literal =n+1; + + +name + -> letter:c =++sa << c; + -> name, letter:c =sa << c; + -> name, digit:c =sa << c; + + +/***** + + Integer and String Expression Logic + + The syntax for expressions is essentially that of C, with the addition + of string comparison. The only missing operators are ++, --, and comma. + +*****/ + + +(long) conditional exp + -> logical or exp + -> logical or exp:c, '?', conditional exp:x, ':', + conditional exp:y = c != 0 ? x : y; + +(long) logical or exp + -> logical and exp + -> logical or exp:x, "||", + logical and exp:y =x != 0 || y!=0; + +(long) logical and exp + -> inclusive or exp + -> logical and exp:x, "&&", + inclusive or exp:y =x != 0 && y !=0; + +(long) inclusive or exp + -> exclusive or exp + -> inclusive or exp:x, '|', + exclusive or exp:y =x | y; + +(long) exclusive or exp + -> and exp + -> exclusive or exp:x, '^', and exp:y =x ^ y; + +(long) and exp + -> equality exp + -> and exp:x, '&', equality exp:y =x & y; + +(long) equality exp + -> relational exp + -> equality exp:x, "==", relational exp:y =x == y; + -> equality exp:x, "!=", relational exp:y =x != y; + -> string exp, "==", string exp =string_comp() == 0; + -> string exp, "!=", string exp =string_comp() != 0; + + +(long) relational exp + -> shift exp + -> relational exp:x, '<', shift exp:y =x < y; + -> relational exp:x, '>', shift exp:y =x > y; + -> relational exp:x, "<=", shift exp:y =x <= y; + -> relational exp:x, ">=", shift exp:y =x >= y; + -> string exp, '<', string exp =string_comp() < 0; + -> string exp, '>', string exp =string_comp() > 0; + -> string exp, "<=", string exp =string_comp() <= 0; + -> string exp, ">=", string exp =string_comp() >= 0; + +(long) shift exp + -> additive exp + -> shift exp:x, "<<", additive exp:y =x << (int) y; + -> shift exp:x, ">>", additive exp:y =x >> (int) y; + +(long) additive exp + -> multiplicative exp + -> additive exp:x, '+', + multiplicative exp:y =x + y; + -> additive exp:x, '-', + multiplicative exp:y =x - y; + +(long) multiplicative exp + -> unary exp + -> multiplicative exp:x, '*', unary exp:y =x * y; + -> multiplicative exp:x, '/', nonzero:y =x / y; + -> multiplicative exp:x, '%', nonzero:y =x % y; + +(long) nonzero + -> unary exp: x ={ + assert(x); + return x; + } + +(long) unary exp + -> primary exp + -> '+', unary exp:x =x; + -> '-', unary exp:x =-x; + -> '~', unary exp:x =~x; + -> '!', unary exp:x =!x; + +(long) primary exp + -> integer constant:x =x; + -> character constant:x =x; + -> string term,'@', primary exp:n =((unsigned char *) (sa--).top())[(int) n]; + -> '#', string element ={ + long temp; + sscanf((sa--).top(), "%ld", &temp); + return temp; + } + -> numeric name + -> '(', conditional exp:x, ')' =x; + -> built_in name:x, built_in argument =(*st[(unsigned)x].data.func)(); + +built_in argument + -> '(', parameter string, ')' + +(long) numeric name, + string name, + built_in name, + undefined name + -> name =name_type(); + + +/***** + + String Expressions + +*****/ + +string exp + -> string term + -> string exp, '#', string term =concat(sa); + +string term + -> string element + -> string term, '@', '(', + conditional exp:first, "..", + conditional exp:last, ')' =extract((unsigned)first, (unsigned) last); + -> string term, '[', !sa << '[';, + parameter string, ']' =concat(sa) << ']', lookup(); + +string element + -> string literal + -> string name:x =++sa << st[(unsigned)x].data.text; + -> undefined name:x =++sa << sd[(unsigned)x]; + -> action text =action_string(); + -> '=', primary exp:x =++sa,sa.printf("%ld",x); + -> '(', string exp, ')' + +/***** + + Integer constants + + The syntax for integer constants is identical to that in C. + +*****/ + +(long) integer constant + -> hex constant + -> octal constant + -> decimal constant + +(long) hex constant + -> {"0x" | "0X"} =0; + -> hex constant:x, hex digit:d =16*x + d; + +(long) hex digit + -> '0-9':d =d - '0'; + -> 'a-f' + 'A-F':d =(d&7) + 9; + +(long) octal constant + -> '0' =0; + -> octal constant:n, '0-7':d =8*n + d-'0'; + +(long) decimal constant + -> '1-9':d =d-'0'; + -> decimal constant:n, '0-9':d =10*n + d-'0'; + + +/***** + + Character Constant + + The rules for character constant are the same as in C. + +*****/ + +(int) character constant + -> '\'', char constant element:c, '\'' =c; + +(int) char constant element + -> not single quote + -> escape sequence + + +/***** + + Screen Display + +*****/ + +screen description + -> screen items:scd, '}' =display_queries(scd); + +(screen_descriptor *) screen items + -> "screen", '{' =reset(qs), new screen_descriptor; + -> screen items, eol + -> screen items:scd, + "title", '=', + formula, eol =scd->title = formula(), scd; + -> screen items:scd, + color spec:c, eol =scd->color = (char) c, scd; + -> screen items:scd, + "entry", color spec:c, eol =scd->entry_color = (char) c, scd; + -> screen items:scd, + "highlight", color spec:c, eol =scd->highlight_color = (char) c, scd; + -> screen items:scd, + "size", '=', conditional exp:w, + ',', conditional exp:h =scd->width = (unsigned)w, scd->height = (unsigned) h, scd; + -> screen items:scd, + "location", '=', conditional exp:px, + ',', conditional exp:py =scd->pos.x = (unsigned) px,scd->pos.y = (unsigned) py, scd; + -> screen items:scd, + query line:q, '}', eol =qs << *q, delete q, scd; + -> screen items:scd, + button line:q, '}', eol =qs << *q, delete q, scd; + +(int) color spec + -> "color", '=', conditional exp:fg, + ',', conditional exp:bg =COLOR((unsigned)fg,(unsigned)bg); + + +(query_item *) query line + -> "field", '{' =clear(new query_item); + -> query line, eol + -> query line:q, + "variable", '=', + literal, eol =q->id = sd << (sa--).top(), q; + -> query line:q, + "default", '=', + formula, eol =q->value = formula(), q; + -> query line:q, + "prompt", '=', + formula, eol =q->prompt = formula(), q; + -> query line:q, + "explanation", '=', + formula, eol =q->explanation = formula(),q; + +(query_item *) button line + -> "button", '{' =clear(new query_item); + -> button line, eol + -> button line:q, + "prompt", '=', formula, eol =q->prompt = formula(), q; + -> button line:q, + "explanation", '=', + formula, eol =q->explanation = formula(),q; + -> button line:q, + action text, eol =q->action = copy_action(), q; + +formula + -> formula element =reset(is) << (sd << (sa--).top()); + -> formula, '#', formula element =is << (sd << (sa--).top()); + +formula element + -> paren string + -> string literal + -> literal + -> formula element, '[', !sa << '[';, + parameter string, ']' =concat(sa) << ']'; + +{ + +#include <stdlib.h> +#include <ctype.h> +#include <sys/stat.h> +#include <fcntl.h> +#include <time.h> +#include <assert.h> +#include <errno.h> + +#if defined(__MSDOS__) || defined(__WINDOWS__) +#include <io.h> +#include <conio.h> +#include <process.h> +#else +#include <unistd.h> + +/* This is only meant to compile, not run. (Unix) */ +static int kbhit(void) { return 0; } +static int getch(void) { return '?'; } +static void ungetch(int) { } +static void strupr(char *) { } +static int spawnvp(int, const char *, char *const *) { return -1; } +#define O_TEXT 0 +#define O_BINARY 0 +#define P_WAIT 0 +#define _MAX_PATH 128 +#define _MAX_DRIVE 4 +#define _MAX_DIR 128 +#define _MAX_FNAME 16 +#define _MAX_EXT 4 +static void _splitpath(const char *, char *, char *, char *, char *) {} +static void _makepath(char *, const char *, const char *, const char *, + const char *) {} + +#endif + +#ifdef __MSDOS__ +#include <dos.h> +#else + +/* This is only meant to compile, not run. (Windows, Unix) */ + +#define far +struct find_t { int attrib; char name[16]; }; +int _dos_findfirst(const char *, int, struct find_t *) { return -1; } +int _dos_findnext(struct find_t *) { return -1; } +#define _A_SUBDIR 1 +struct diskfree_t { + unsigned avail_clusters, bytes_per_sector, sectors_per_cluster; +}; +void _dos_getdiskfree(int, struct diskfree_t *f) { + f->avail_clusters = 0; + f->bytes_per_sector = 512; + f->sectors_per_cluster = 4; +} +void _dos_getftime(int, unsigned short *d, unsigned short *t) { *d=*t=0; } + +#endif /* not MSDOS */ + +#include "edit.h" + + +#include "screen.h" +#include "util.h" +#include "redirect.h" + +#define GET_CONTEXT CONTEXT.pointer = PCB.pointer;\ + CONTEXT.line=PCB.line;\ + CONTEXT.column = PCB.column; + + +int debug_switch = 0; +char *error_msg = NULL; +unsigned errorlevel_index; +int errorlevel; +int exitcode = 0; +int exitflag = 0; +int first_line = 1; +int first_column = 1; +unsigned stderr_index; + +void display_queries(screen_descriptor *); + +#define FIRST_LINE first_line +#define FIRST_COLUMN first_column + + +/***** + + Internal Functions + +*****/ + +long file_exists(void) { + FILE *f = fopen((sa--).top(),"r"); + if (f != NULL) fclose(f); + return f != NULL; +} + +long directory_exists(void) { + struct find_t ff; + int result; + + sa << "\\*.*"; + result = _dos_findfirst((sa--).top(),_A_SUBDIR,&ff); + return result == 0; +} + +long string_length(void) { + return size(sa--); +} + +long get_file_length(void) { + int handle = open((sa--).top(), O_RDONLY); + long length; + if (handle < 0) return 0; + length = filelength(handle); + close(handle); + return length; +} + +long disk_space(void) { + struct diskfree_t free; + int drive = toupper(*(sa--).top()) - 64; + long avail; + + _dos_getdiskfree(drive, &free); + avail = (long) free.avail_clusters + * (long) free.bytes_per_sector + * (long) free.sectors_per_cluster; + return avail; +} + +long file_time(void) { + int handle = open((sa--).top(), O_RDONLY); +#ifdef __BCPLUSPLUS__ + unsigned date, time; +#else + unsigned short date, time; +#endif + struct tm t; + + if (handle < 0) return 0; + _dos_getftime(handle, &date, &time); + close(handle); + t.tm_year = ((date & 0xfe00) >> 9) + 80; + t.tm_mon = ((date & 0x1e00) >> 5) - 1; + t.tm_mday = date & 0x001f; + ; + t.tm_hour = (time & 0xf800) >> 11; + t.tm_min = (time & 0x07e0) >> 5; + t.tm_sec = (time & 0x001f) << 1; + return mktime(&t); +} + + +// Support for reduction procecures + +// Compare top strings on string accumulator + +/* + pops top two strings from string accumulator using strcmp + and returns + -1 if first string is less than top string + 0 if strings match + +1 if top string is greater than first string +*/ + +int string_comp(void) { + int n = size(sa); + array<char> right_string((sa--).top(), n+1); + return strcmp((sa--).top(),right_string); +} + +/* + replace the top string on the stack, with a substring where the index + of the first character in the substring is given by "first" and the index + of the last character is given by "last" +*/ + +void extract(unsigned first, unsigned last) { + int n = last - first + 1; + assert (last >= first); + array <char> x( (sa--).top() + first, n+1 ); + x[n] = 0; + ++sa << x; +} + +/* + Look up the top string on the accumulator stack in the string dictionary. + If it has a value in the symbol table, replace it with the symbol table + value. If the value is numeric, convert it to integer. Otherwise, leave the + string untouched on the stack. +*/ + +void lookup(void) { + unsigned index = sd[sa.top()]; + if (index == 0) return; + switch (st[index].type) { + case string_type: + case value_type: { + --sa; // discard name + ++sa << st[index].data.text; // stack value + break; + } + case integer_type: { + --sa; // discard name + (++sa).printf("%ld", st[index].data.integer); // convert to ascii + break; + } + default: + /* not supposed to happen? */ + break; + } +} + +/* + Find the data type of a symbol and change the reduction accordingly. + Return the dictionary index for strings, and the value itself for integers. +*/ + +long name_type(void) { + unsigned index = sd << (sa--).top(); + switch (st[index].type) { + case value_type: + case string_type: { + CHANGE_REDUCTION(string_name); + return index; + } + case built_in_function_type: { + CHANGE_REDUCTION(built_in_name); + return index; + } + case undefined_type: { + CHANGE_REDUCTION(undefined_name); + return index; + } + case integer_type: return st[index].data.integer; + default: + /* not supposed to happen? */ + break; + } + return 0; +} + +/* + Store a string formula. A string formula is a sequence of string identifiers + the values of which are to be concatenated. The parser has accumulated the + identifiers on the integer_stack, is. The formula is terminated by a zero + entry. +*/ + +int *formula(void) { + int n = size(is << 0); + int *f = new int[n]; + while (n--) is >> f[n]; + return f; +} + +/* + Make a copy of an action that has been identified in the text stream. + An action pointer was stacked at the beginning of the action text on the + action stack, as. +*/ + +action_pointer copy_action(void) { + action_pointer ap; + as >> ap; // pop action descriptor + unsigned length = (unsigned) (PCB.pointer - ap.pointer); + unsigned char *action = memdup(ap.pointer,length + 1); + action[length] = 0; + ap.pointer = action; + return ap; +} + + +// Internal Commands + +int echo(int, char *args[]) { + int i; + const char *cs = ""; + for (i = 1; args[i]; i++) printf("%s%s", cs, args[i]), cs = " "; + printf("\n"); + fflush(stdout); + return 0; +} + +int pause(int, char *[]) { + int c; + while (kbhit()) getch(); // Empty buffer + printf("Press any key to continue . . .\n"); + c = getch(); + if (c == 3) exit(1); + return c; +} + +int exit_script(int n_args, char *args[]) { + if (n_args > 1) sscanf(args[1], "%d", &exitcode); + exit(exitcode); + return exitcode; +} + +/* +int return_script(int n_args, char *args[]) { + if (n_args > 1) sscanf(args[1], "%d", &exitcode); + PCB.exit_flag = AG_SUCCESS_CODE; + return exitcode; +} +*/ + +int subdirs(int, char *args[]) { + struct find_t file_block; + int flag; + int length = strlen(args[1]); + array <char> name(args[1],length + 5); + + strcat(name, "\\*.*"); + for (flag = _dos_findfirst(name, _A_SUBDIR, &file_block); + flag == 0; flag = _dos_findnext(&file_block)) { + if ((file_block.attrib & _A_SUBDIR) == 0) continue; + if (strcmp(file_block.name, ".") == 0) continue; + if (strcmp(file_block.name, "..") == 0) continue; + puts(file_block.name); + } + return 0; +} + +int files(int, char *args[]) { + struct find_t file_block; + int flag; + int length = strlen(args[1]); + array<char> name(args[1],length + 5); + + strcat(name, "\\*.*"); + for (flag = _dos_findfirst(name, 0, &file_block); + flag == 0; flag = _dos_findnext(&file_block)) { + puts(file_block.name); + } + return 0; +} + + +/***** + + Execute Command Line + +*****/ + + +void perform_action(action_pointer ap) { + dsl_pcb_type save_pcb = PCB; + + PCB.pointer = ap.pointer; + first_line = ap.line; + first_column = ap.column; + dsl(); + exitflag = PCB.exit_flag != AG_SUCCESS_CODE; + PCB = save_pcb; + if (exitflag) PCB.exit_flag = AG_SEMANTIC_ERROR_CODE; +} + +void exec(void) { + int n = size(ps << (char *) NULL); + int n_args = n - 1; + unsigned index; + unsigned uc_index; + int i; + + array <char *> args(n); + while (n--) ps >> args[n]; + + index = sd[args[0]]; + while (index && st[index].type == string_type) { + args[0] = st[index].data.text; // stack value + index = sd[args[0]]; + } + if (debug_switch) { + for (i = 0; args[i]; i++) fprintf(stderr, "%s ", args[i]); + fprintf(stderr,"\nPress any key to continue\n"); + while (!kbhit()); + getch(); + } + strupr(args[0]); + uc_index = sd[args[0]]; + if (n_args == 1 && strlen(args[0]) == 2 && args[0][1] == ':') { + errorlevel = system(args[0]); + } + else if ( *args[0] && uc_index) switch (st[uc_index].type) { + case internal_type: { + errorlevel = (*st[uc_index].data.proc)(n_args, args); + break; + } + case dos_type: { + int i; + for (i = 1; args[i]; i++) args[i][-1] = ' '; + errorlevel = system(args[0]); + if (errorlevel == -1) { + fprintf(stderr,"Error invoking %s: %s\n", args[0], strerror(errno)); + exit(1); + } + break; + } + default: + /* not supposed to happen? */ + break; + } + else if ( *args[0] && index) switch (st[index].type) { + case action_type: { + action_descriptor d = *st[index].data.action; + stack <symbol_table_entry> old_entries(d.n_args); + for (i = 0; i < d.n_args && args[i+1]; i++) { + old_entries << st[d.args[i]]; + st[d.args[i]].type = value_type; + st[d.args[i]].data.text = memdup(args[i+1], 1 + strlen(args[i+1])); + } + perform_action(d.ap); + for (i = d.n_args; i--;) { + release(st[d.args[i]]); + old_entries >> st[d.args[i]]; + } + break; + } + default: { + char **tmpargs = args; + errorlevel = spawnvp(P_WAIT,args[0], tmpargs); + if (errorlevel == -1) { + fprintf(stderr,"Error invoking %s: %s\n", args[0], strerror(errno)); + exit(1); + } + } + } + else { + } + st[errorlevel_index].data.integer = errorlevel; + while (n_args--) --sa; + --ps; + if (kbhit()) { + int c = getch(); + if (c == 3) exit(1); + ungetch(c); + } +} + +void discard_temp_file(char *file_name) { + unlink(file_name); // Delete file + delete [] file_name; // Free storage for name +} + + +/***** + + Execute Command with piped input + +*****/ + + +void exec_pipe_in(char *file_name) { + { + redirect sin(STDIN, file_name); + exec(); + } + discard_temp_file(file_name); +} + + +/***** + + Execute Command with redirected I/O + +*****/ + +void exec_redirect_in(void) { + redirect sin(STDIN, (sa--).top()); + exec(); +} + +char *exec_pipe_out(void) { + fflush(stdout); + redirect sout(STDOUT); + exec(); + fflush(stdout); + return save_file(sout); +} + +char *exec_pipe_in_pipe_out(char *file_name) { + char *result; + { + redirect sin(STDIN, file_name); + fflush(stdout); + redirect sout(STDOUT); + exec(); + fflush(stdout); + result = save_file(sout); + } + discard_temp_file(file_name); + return result; +} + +char *exec_redirect_in_pipe_out(void) { + fflush(stdout); + redirect sout(STDOUT); + exec_redirect_in(); + fflush(stdout); + return save_file(sout); +} + +unsigned check_integer(void) { + unsigned index = sd << (sa--).top(); + if (st[index].type == integer_type) return index; + CHANGE_REDUCTION(undeclared_variable); + if (st[index].type == string_type) CHANGE_REDUCTION(string_variable); + return index; +} + +void assign_value(unsigned index) { + char *text = copy(sa--); + release(st[index]); + st[index].type = value_type; + st[index].data.text = text; +} + +void grab_output(char *temp_name) { + unlink(sa.top()); // delete old file + rename(temp_name, (sa--).top()); // rename temp file + delete [] temp_name; // discard name string +} + +void append_output(char *temp_name) { + fflush(stdout); + redirect sout(STDOUT, (sa--).top(), 1); // append to file named on sa + redirect sin(STDIN, temp_name); + char buf[2000]; + int n; + while (1) { + n = read(STDIN, buf, 2000); + if (n == 0) break; + write(STDOUT, buf, n); + } + fflush(stdout); + unlink(temp_name); + delete [] temp_name; +} + +void action_string(void) { + action_pointer ap; + as >> ap; + unsigned length = (unsigned)(PCB.pointer - ap.pointer); + array <unsigned char> action(ap.pointer,length + 1); + action[length] = 0; + fflush(stdout); + redirect sout(STDOUT); + char *result; + + ap.pointer = action; + perform_action(ap); + fflush(stdout); + result = content(sout); + ++sa << result; + delete [] result; +} + + +// Program Control functions + +// If/else statement + +int do_if(int pc, int cc) { + action_pointer ap; + as >> ap; + if (!pc && cc && exitflag == 0) { + unsigned length = (unsigned) (PCB.pointer - ap.pointer); + array<unsigned char> q(ap.pointer, length+1); + q[length] = 0; + ap.pointer = q; + perform_action(ap); + } + return pc || cc; +} + +// While statement + +void do_while(int cc) { + unsigned length; + action_pointer ap; + as >> ap; + if (cc == 0) return; + length = (unsigned) (PCB.pointer - ap.pointer); + array<unsigned char> q(ap.pointer, length+1); + q[length] = 0; + ap.pointer = q; + perform_action(ap); + if (exitflag) return; + PCB.pointer = CONTEXT.pointer; + PCB.line = CONTEXT.line; + PCB.column = CONTEXT.column; +} + + +// For Statement +// Note that this is the for statement in the DOS batch languange for, not C + +void do_for_loop(void) { + int n,k; + char *q; + const char *seps = " \t\v\f\r\n"; + action_pointer ap; + as >> ap; + unsigned length = (unsigned)(PCB.pointer - ap.pointer); + array <unsigned char> action(ap.pointer, length + 1); + action[length] = 0; + + ap.pointer = action; + n = size(sa); + array<char> text((sa--).top(), n + 1); + + + unsigned index = sd << (sa--).top(); + + ++ps; + for (q = strtok(text, seps); q != NULL; q = strtok(NULL,seps)) { + if (*q == '(') { + int k = strlen(q) - 1; + assert(q[k] == ')'); + q[k] = 0; + q++; + } + else if (*q == '"') { + int k = strlen(q) - 1; + assert(q[k] == '"'); + q[k] = 0; + q++; + } + ps << q; + } + k = n = size(ps); + array<char *> args(n); + while (k--) ps >> args[k]; + --ps; + symbol_table_entry save_table_entry = st[index]; + st[index].type = value_type; + + for (k = 0; k < n && exitflag == 0; k++) { + st[index].data.text = args[k]; + perform_action(ap); + } + st[index] = save_table_entry; +} + +void invoke_script(void) { + int handle = open(sa.top(), O_TEXT | O_RDONLY); + long size; + unsigned n; + action_pointer ap; + + if (handle < 0) { + fprintf(stderr,"Cannot open %s\n", (sa--).top()); + exit(1); + } + --sa; + size = filelength(handle); + assert(size < 65536L); + array <unsigned char> data((unsigned) size+1); + n = (unsigned) read(handle,data,(unsigned) size); + data[n] = 0; + close(handle); + exitflag = 0; + ap.pointer = data; + ap.line = ap.column = 1; + perform_action(ap); + st[errorlevel_index].data.integer = exitcode; + exitflag = exitcode = 0; + return; +} + +internal_commands_descriptor internal_commands[] = { + {"ECHO", echo}, + {"EXIT", exit_script}, + {"FILES", files}, + {"PAUSE", pause}, +// {"RETURN", return_script}, + {"SUBDIRS", subdirs}, + {NULL, NULL} +}; + +struct built_ins_descriptor built_ins[] = { + {"file_exists", file_exists}, + {"directory_exists", directory_exists}, + {"string_length", string_length}, + {"file_length", get_file_length}, + {"disk_space", disk_space}, + {"file_time", file_time}, + {NULL, NULL} +}; + +void set_extension(char *path, const char *e) { + char s[_MAX_PATH]; + char drive[_MAX_DRIVE]; + char dir[_MAX_DIR]; + char file[_MAX_FNAME]; + char ext[_MAX_EXT]; + + _splitpath(path,drive,dir,file,ext); + _makepath(s, drive, dir, file, e); + ++sa << s; +} + +/* + Note that if this program is called without any arguments, it looks for a + script with the same name as the executable. Thus, to make an install + program that picks up the install script without any arguments, you simply + rename DSL.EXE to INSTALL.EXE. Then when you run it without any arguments + it will run the INSTALL.DSL script. +*/ + +int main(int argc, char *argv[]) { + int arg_number = 0; + int i = 1; + int j = 0; + + init_dos_internals(); + set_arg(j++, argv[0]); + if (argc > i && (argv[i][0] == '/' || argv[i][0] == '-')) { + if (toupper(argv[i][1]) != 'D') { + printf("Unrecognized switch -- /%c\n", argv[i][1]); + return 1; + } + debug_switch = 1; + i++; + } + if (argc > i) arg_number = i++; + set_extension(argv[arg_number], "DSL"); + set_arg(j++,copy(sa)); + while (i < argc) set_arg(j++, argv[i++]); + define_integer("argc", j); + invoke_script(); // Takes file name from sa + return exitcode; +} +} // End Embedded C