From 29e98f6bd6ba9ebeb1fb9353b3a6910ca8e3362a Mon Sep 17 00:00:00 2001 From: Max Horn Date: Fri, 31 Aug 2018 10:40:10 +0200 Subject: [PATCH] WIP: add TOKENIZE_STREAM --- lib/tokenizer.g | 190 ++++++++++++++++++++++++++++++++++++++++++++++++ src/intrprtr.c | 8 +- src/read.c | 44 ++++++++++- src/read.h | 8 ++ src/scanner.c | 60 ++++++++++++--- src/scanner.h | 4 + src/streams.c | 110 ++++++++++++++++++++++++++++ 7 files changed, 408 insertions(+), 16 deletions(-) create mode 100644 lib/tokenizer.g diff --git a/lib/tokenizer.g b/lib/tokenizer.g new file mode 100644 index 00000000000..e00565975d7 --- /dev/null +++ b/lib/tokenizer.g @@ -0,0 +1,190 @@ +ID_TO_SYMBOL_LIST:=[]; +SYMBOL_COLORS:=rec(); +# We split the scanner ids into the lower 3 bits, plus PValuation(id>>3, 2) +# compressing them to 3+5 = 8 bits +COMPRESS_SCANNER_ID := function(id) + local lo, hi; + lo := (id mod 8); + hi := QuoInt(id, 8); + if hi > 0 then + hi := PValuation(hi, 2) + 1; + fi; + return lo + hi * 8 + 1; +end; + +for name in RecNames(SCANNER_SYMBOLS) do + SYMBOL_COLORS.(name) := TextAttr.reset; + id := SCANNER_SYMBOLS.(name); + #Print(name, ": ",id, " -> ", [ id mod 8, QuoInt(id,8), COMPRESS_SCANNER_ID(id)], "\n"); + ID_TO_SYMBOL_LIST[COMPRESS_SCANNER_ID(id)] := name; +od; +ID_TO_SYMBOL := id -> ID_TO_SYMBOL_LIST[COMPRESS_SCANNER_ID(id)]; + +# identifiers and keywords +SYMBOL_COLORS.S_IDENT := TextAttr.1; +SYMBOL_COLORS.S_UNBIND := TextAttr.1; +SYMBOL_COLORS.S_ISBOUND := TextAttr.1; +SYMBOL_COLORS.S_TRYNEXT := TextAttr.1; +SYMBOL_COLORS.S_INFO := TextAttr.1; +SYMBOL_COLORS.S_ASSERT := TextAttr.1; +SYMBOL_COLORS.S_READWRITE := TextAttr.1; +SYMBOL_COLORS.S_READONLY := TextAttr.1; +SYMBOL_COLORS.S_ASSERT := TextAttr.1; +SYMBOL_COLORS.S_REC := TextAttr.1; +SYMBOL_COLORS.S_FUNCTION := TextAttr.1; +SYMBOL_COLORS.S_LOCAL := TextAttr.1; +SYMBOL_COLORS.S_END := TextAttr.1; +SYMBOL_COLORS.S_IF := TextAttr.1; +SYMBOL_COLORS.S_FOR := TextAttr.1; +SYMBOL_COLORS.S_WHILE := TextAttr.1; +SYMBOL_COLORS.S_REPEAT := TextAttr.1; +SYMBOL_COLORS.S_ATOMIC := TextAttr.1; +SYMBOL_COLORS.S_THEN := TextAttr.1; +SYMBOL_COLORS.S_ELIF := TextAttr.1; +SYMBOL_COLORS.S_ELSE := TextAttr.1; +SYMBOL_COLORS.S_FI := TextAttr.1; +SYMBOL_COLORS.S_DO := TextAttr.1; +SYMBOL_COLORS.S_OD := TextAttr.1; +SYMBOL_COLORS.S_UNTIL := TextAttr.1; +SYMBOL_COLORS.S_BREAK := TextAttr.1; +SYMBOL_COLORS.S_RETURN := TextAttr.1; +SYMBOL_COLORS.S_QUIT := TextAttr.1; +SYMBOL_COLORS.S_QQUIT := TextAttr.1; +SYMBOL_COLORS.S_CONTINUE := TextAttr.1; + +SYMBOL_COLORS.S_MOD := TextAttr.1; +SYMBOL_COLORS.S_IN := TextAttr.1; +SYMBOL_COLORS.S_NOT := TextAttr.1; +SYMBOL_COLORS.S_AND := TextAttr.1; +SYMBOL_COLORS.S_OR := TextAttr.1; + +# brackets, parens, ... +SYMBOL_COLORS.S_LBRACK := TextAttr.5; +SYMBOL_COLORS.S_LBRACE := TextAttr.5; +SYMBOL_COLORS.S_BLBRACK := TextAttr.5; +SYMBOL_COLORS.S_RBRACK := TextAttr.5; +SYMBOL_COLORS.S_RBRACE := TextAttr.5; +SYMBOL_COLORS.S_DOT := TextAttr.5; +SYMBOL_COLORS.S_BDOT := TextAttr.5; +SYMBOL_COLORS.S_LPAREN := TextAttr.5; +SYMBOL_COLORS.S_RPAREN := TextAttr.5; +SYMBOL_COLORS.S_COMMA := TextAttr.5; +SYMBOL_COLORS.S_DOTDOT := TextAttr.5; +SYMBOL_COLORS.S_COLON := TextAttr.5; +SYMBOL_COLORS.S_DOTDOTDOT := TextAttr.5; +SYMBOL_COLORS.S_SEMICOLON := TextAttr.5; +SYMBOL_COLORS.S_DUALSEMICOLON := TextAttr.5; + + +# constants +SYMBOL_COLORS.S_INT := TextAttr.4; +SYMBOL_COLORS.S_FLOAT := TextAttr.4; +SYMBOL_COLORS.S_TRUE := TextAttr.4; +SYMBOL_COLORS.S_FALSE := TextAttr.4; +SYMBOL_COLORS.S_CHAR := TextAttr.4; + +# strings +SYMBOL_COLORS.S_STRING := TextAttr.3; + +# operators +SYMBOL_COLORS.S_MULT := TextAttr.2; +SYMBOL_COLORS.S_MULT := TextAttr.2; +SYMBOL_COLORS.S_DIV := TextAttr.2; +SYMBOL_COLORS.S_POW := TextAttr.2; +SYMBOL_COLORS.S_PLUS := TextAttr.2; +SYMBOL_COLORS.S_MINUS := TextAttr.2; +SYMBOL_COLORS.S_EQ := TextAttr.2; +SYMBOL_COLORS.S_LT := TextAttr.2; +SYMBOL_COLORS.S_GT := TextAttr.2; +SYMBOL_COLORS.S_NE := TextAttr.2; +SYMBOL_COLORS.S_LE := TextAttr.2; +SYMBOL_COLORS.S_GE := TextAttr.2; +SYMBOL_COLORS.S_ASSIGN := TextAttr.2; + + +ExtractRangeFromLines := function(lines, startline, startpos, endline, endpos) + local data, tmp, i; + if startline = endline then + return lines[startline]{[startpos+1 .. endpos]}; + fi; + tmp := lines[startline]; + data := tmp{[startpos+1 .. Length(tmp)]}; + Add(data, '\n'); + for i in [startline+1 .. endline-1] do + Append(data, lines[i]); + Add(data, '\n'); + od; + tmp := lines[endline]; + Append(data, tmp{[1 .. endpos]}); + return data; +end; + + +TOKENIZE_STRING:=function(str) + local res, stat, token, symbol, lines, text, sep1, sep2; + Add(str, '\n'); + sep1 := ""; + sep2 := ""; + #sep1 := "<"; + #sep2 := ">"; +# Print("Input:\n", str, "\n"); +# Print("Output:\n"); + lines := SplitString(str, "\n"); + res := TOKENIZE_STREAM(InputTextString(str)); + for stat in res do + if not IsList(stat) then continue; fi; + for token in stat do + if not IsList(token) then continue; fi; + if token[1] = "ERROR" then + Print("\nEncountered an error: ", token[2], "\n"); + continue; + fi; + symbol := Remove(token); + Add(token, ID_TO_SYMBOL(symbol)); + + # extract symbol + if Length(token) <> 8 then continue; fi; + if symbol = SCANNER_SYMBOLS.S_EOF then + Print("\n\n-- EOF --\n"); + continue; + fi; + text := ExtractRangeFromLines(lines, token[2], token[3], token[4], token[5]); + if Length(text) > 0 then + Print(TextAttr.6, sep1, text, sep2, TextAttr.reset); + #Print(TextAttr.b6, sep1, text, sep2, TextAttr.reset); + fi; + text := ExtractRangeFromLines(lines, token[4], token[5], token[6], token[7]); + if Length(text) > 0 then + Print(SYMBOL_COLORS.(ID_TO_SYMBOL(symbol)), sep1, text, sep2, TextAttr.reset); + fi; + Add(token, text); + od; + od; + Print("\n"); + return res; +end; + + +SetPrintFormattingStatus("*stdout*", false); + +l:=TOKENIZE_STRING("1;"); + +l:=TOKENIZE_STRING("1"); + +l:=TOKENIZE_STRING("1-;"); + + +l:=TOKENIZE_STRING("1+1;"); + +l:=TOKENIZE_STRING("123 + 456;"); + +l:=TOKENIZE_STRING("1+1; x:=y-3;"); + +l:=TOKENIZE_STRING("x:=0123 + 1234; xxxx+777777;"); + +l:=TOKENIZE_STRING(""" +1+1; x:=y-3; +# This is a little test program +f := x -> x+1; # increment function +f(2); +"""); diff --git a/src/intrprtr.c b/src/intrprtr.c index 8190eb69502..a128b44b21d 100644 --- a/src/intrprtr.c +++ b/src/intrprtr.c @@ -241,7 +241,7 @@ void IntrBegin(IntrState * intr) intr->StackObj = NEW_PLIST(T_PLIST, 64); /* must be in immediate (non-ignoring, non-coding) mode */ - GAP_ASSERT(intr->ignoring == 0); +// GAP_ASSERT(intr->ignoring == 0); GAP_ASSERT(intr->coding == 0); /* no return-statement was yet interpreted */ @@ -254,13 +254,13 @@ ExecStatus IntrEnd(IntrState * intr, BOOL error, Obj * result) if ( ! error ) { /* must be back in immediate (non-ignoring, non-coding) mode */ - GAP_ASSERT(intr->ignoring == 0); +// GAP_ASSERT(intr->ignoring == 0); GAP_ASSERT(intr->coding == 0); /* and the stack must contain the result value (which may be void) */ - GAP_ASSERT(LEN_PLIST(intr->StackObj) == 1); +// GAP_ASSERT(LEN_PLIST(intr->StackObj) == 1); if (result) - *result = PopVoidObj(intr); + *result = (LEN_PLIST(intr->StackObj) == 0) ? 0 : PopVoidObj(intr); return intr->returning; } diff --git a/src/read.c b/src/read.c index 66bc84191b4..1a7be1516f5 100644 --- a/src/read.c +++ b/src/read.c @@ -36,6 +36,12 @@ #endif +static ExecStatus ReadCommand(Obj context, + TypInputFile * input, + BOOL justTokenize, + Obj * evalResult, + BOOL * dualSemicolon); + /**************************************************************************** ** *S TRY_IF_NO_ERROR @@ -2510,6 +2516,32 @@ ExecStatus ReadEvalCommand(Obj context, TypInputFile * input, Obj * evalResult, BOOL * dualSemicolon) +{ + return ReadCommand(context, input, FALSE, evalResult, dualSemicolon); +} + + +/**************************************************************************** +** +*F ReadTokenizeCommand() +** +*/ +ExecStatus ReadTokenizeCommand(Obj context, TypInputFile * input, Obj * tokens) +{ + return ReadCommand(context, input, TRUE, tokens, 0); +} + + +/**************************************************************************** +** +*F ReadCommand() +** +*/ +static ExecStatus ReadCommand(Obj context, + TypInputFile * input, + BOOL justTokenize, + Obj * evalResult, + BOOL * dualSemicolon) { volatile ExecStatus status; volatile Obj tilde; @@ -2528,6 +2560,16 @@ ExecStatus ReadEvalCommand(Obj context, ClearError(); + if (justTokenize) { + // HACK / TODO: explain this + rs->intr.returning = STATUS_RETURN; + rs->intr.ignoring = 1; + if (*evalResult) + rs->s.tokens = *evalResult; + else + rs->s.tokens = *evalResult = NEW_PLIST(T_PLIST, 16); + } + /* get the first symbol from the input */ Match_(rs, rs->s.Symbol, "", 0); @@ -2607,7 +2649,7 @@ ExecStatus ReadEvalCommand(Obj context, *dualSemicolon = (rs->s.Symbol == S_DUALSEMICOLON); // end the interpreter - status = IntrEnd(&rs->intr, rs->s.NrError > 0, evalResult); + status = IntrEnd(&rs->intr, rs->s.NrError > 0, justTokenize ? 0 : evalResult); // restore the execution environment SWITCH_TO_OLD_LVARS(oldLVars); diff --git a/src/read.h b/src/read.h index 242346a12c6..6b9dee4bf61 100644 --- a/src/read.h +++ b/src/read.h @@ -40,6 +40,14 @@ ExecStatus ReadEvalCommand(Obj context, BOOL * dualSemicolon); +/**************************************************************************** +** +*F ReadTokenizeCommand() +** +*/ +ExecStatus ReadTokenizeCommand(Obj context, TypInputFile * input, Obj * tokens); + + /**************************************************************************** ** *F ReadEvalFile() . . . . . . . . . . . . . . . . . . . . . . . read a file diff --git a/src/scanner.c b/src/scanner.c index 48a304e6785..99d8c103d87 100644 --- a/src/scanner.c +++ b/src/scanner.c @@ -19,6 +19,7 @@ #include "error.h" #include "gapstate.h" #include "gaputils.h" +#include "integer.h" #include "io.h" #include "lists.h" #include "plist.h" @@ -43,8 +44,18 @@ static void SyntaxErrorOrWarning(ScannerState * s, Int tokenoffset) { GAP_ASSERT(tokenoffset >= 0 && tokenoffset <= 2); + + + if (s->tokens) { + // if tokenization mode, don't print the error, instead + // add it to tokens + PushPlist(s->tokens, NewPlistFromArgs( + MakeImmString("ERROR"), MakeImmString(msg), + INTOBJ_INT(error), INTOBJ_INT(tokenoffset))); + } + // do not print a message if we found one already on the current line - if (s->input->lastErrorLine != s->input->number) { + else if (s->input->lastErrorLine != s->input->number) { // open error output TypOutputFile output = { 0 }; @@ -887,6 +898,8 @@ static UInt NextSymbol(ScannerState * s) // Record end of previous symbol's position StoreSymbolPosition(s); + Obj tokenEntry = 0; + Char c = PEEK_CURR_CHAR(s->input); // skip over , , and comments @@ -896,6 +909,7 @@ static UInt NextSymbol(ScannerState * s) if (c == '%') { // we have encountered a pragma GetPragma(s, c); + // TODO: record token positions before/after return S_PRAGMA; } @@ -907,13 +921,21 @@ static UInt NextSymbol(ScannerState * s) // Record start of this symbol's position StoreSymbolPosition(s); - // switch according to the character - if (IsAlpha(c)) { - return GetIdent(s, 0, c); - } + UInt symbol = S_ILLEGAL; - UInt symbol; + if (s->tokens) { + // in tokenizer mode, record the current position + tokenEntry = NEW_PLIST(T_PLIST, 3); + PushPlist(tokenEntry, MakeImmString("token")); + PushPlist(tokenEntry, INTOBJ_INT(s->SymbolStartLine[1])); + PushPlist(tokenEntry, INTOBJ_INT(s->SymbolStartPos[1])); + PushPlist(tokenEntry, INTOBJ_INT(s->SymbolStartLine[0])); + PushPlist(tokenEntry, INTOBJ_INT(s->SymbolStartPos[0])); + // push the token now, so that SyntaxError can modify it if necessary + PushPlist(s->tokens, tokenEntry); + } + // switch according to the character switch (c) { case '.': symbol = S_DOT; c = GET_NEXT_CHAR(); if (c == '.') { symbol = S_DOTDOT; c = GET_NEXT_CHAR(); @@ -962,17 +984,33 @@ static UInt NextSymbol(ScannerState * s) case '?': symbol = S_HELP; GetHelp(s); break; case '"': symbol = S_STRING; GetString(s); break; case '\'': symbol = S_CHAR; GetChar(s); break; - case '\\': return GetIdent(s, 0, c); - case '_': return GetIdent(s, 0, c); - case '@': return GetIdent(s, 0, c); + case '\\': symbol = GetIdent(s, 0, c); break; + case '_': symbol = GetIdent(s, 0, c); break; + case '@': symbol = GetIdent(s, 0, c); break; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': - return GetNumber(s, 0, c); + symbol = GetNumber(s, 0, c); break; case '\377': symbol = S_EOF; FlushRestOfInputLine(s->input); break; - default: symbol = S_ILLEGAL; GET_NEXT_CHAR(); break; + default: + if (IsAlpha(c)) { + symbol = GetIdent(s, 0, c); + } + else { + symbol = S_ILLEGAL; + GET_NEXT_CHAR(); + } + break; + } + + if (s->tokens) { + // in tokenizer mode, record the current position + PushPlist(tokenEntry, INTOBJ_INT(GetInputLineNumber(s->input))); + PushPlist(tokenEntry, INTOBJ_INT(GetInputLinePosition(s->input))); + PushPlist(tokenEntry, ObjInt_UInt((UInt4)symbol)); } + return symbol; } diff --git a/src/scanner.h b/src/scanner.h index a2c57ad34ef..c8a2b701d52 100644 --- a/src/scanner.h +++ b/src/scanner.h @@ -234,6 +234,10 @@ typedef struct { // error occurred. UInt NrError; + // HACK: if non-null this points at a plist in which information about the + // tokens in an input stream are stored. + Obj tokens; + } ScannerState; diff --git a/src/streams.c b/src/streams.c index 0f4a8276f78..f76858e7762 100644 --- a/src/streams.c +++ b/src/streams.c @@ -30,6 +30,8 @@ #include "precord.h" #include "read.h" #include "records.h" +#include "scanner.h" +#include "stats.h" #include "stringobj.h" #include "sysfiles.h" #include "sysopt.h" @@ -317,6 +319,31 @@ static void READ_INNER(TypInputFile * input) } +Obj FuncTOKENIZE_STREAM(Obj self, Obj stream) +{ + TypInputFile input = { 0 }; + if (!OpenInputStream(&input, stream, FALSE)) { + return Fail; + } + + Obj result = NEW_PLIST(T_PLIST, 16); + while (1) { + ClearError(); + Obj tokens = 0; + ExecStatus status = ReadTokenizeCommand(0, &input, &tokens); + if (status == STATUS_EOF) + break; + GAP_ASSERT(tokens != 0); + PushPlist(result, tokens); + PushPlist(result, INTOBJ_INT(status)); + } + + CloseInput(&input); + ClearError(); + return result; +} + + /**************************************************************************** ** *F READ_AS_FUNC() . . . . . . . . . . . . . read current input as function @@ -1746,6 +1773,9 @@ static StructGVarFunc GVarFuncs[] = { GVAR_FUNC_1ARGS(READ_AS_FUNC, input), GVAR_FUNC_1ARGS(READ_GAP_ROOT, filename), GVAR_FUNC_3ARGS(CALL_WITH_STREAM, stream, func, args), + + GVAR_FUNC_1ARGS(TOKENIZE_STREAM, stream), + GVAR_FUNC_1ARGS(LOG_TO, filename), GVAR_FUNC_1ARGS(LOG_TO_STREAM, filename), GVAR_FUNC_0ARGS(CLOSE_LOG_TO), @@ -1844,6 +1874,86 @@ static Int InitLibrary ( /* init filters and functions */ InitGVarFuncsFromTable( GVarFuncs ); +#define EXPORT_IN_PREC(r, symbol) \ + AssPRec(r, RNamName(#symbol), ObjInt_UInt((UInt4)symbol)) + + Obj scannerSymbols = NEW_PREC(80); + AssGVar(GVarName("SCANNER_SYMBOLS"), scannerSymbols); + + EXPORT_IN_PREC(scannerSymbols, S_AND); + EXPORT_IN_PREC(scannerSymbols, S_ASSERT); + EXPORT_IN_PREC(scannerSymbols, S_ASSIGN); + EXPORT_IN_PREC(scannerSymbols, S_ATOMIC); + EXPORT_IN_PREC(scannerSymbols, S_BDOT); + EXPORT_IN_PREC(scannerSymbols, S_BLBRACK); + EXPORT_IN_PREC(scannerSymbols, S_BREAK); + EXPORT_IN_PREC(scannerSymbols, S_CHAR); + EXPORT_IN_PREC(scannerSymbols, S_COLON); + EXPORT_IN_PREC(scannerSymbols, S_COMMA); + EXPORT_IN_PREC(scannerSymbols, S_CONTINUE); + EXPORT_IN_PREC(scannerSymbols, S_DIV); + EXPORT_IN_PREC(scannerSymbols, S_DO); + EXPORT_IN_PREC(scannerSymbols, S_DOT); + EXPORT_IN_PREC(scannerSymbols, S_DOTDOT); + EXPORT_IN_PREC(scannerSymbols, S_DOTDOTDOT); + EXPORT_IN_PREC(scannerSymbols, S_DUALSEMICOLON); + EXPORT_IN_PREC(scannerSymbols, S_ELIF); + EXPORT_IN_PREC(scannerSymbols, S_ELSE); + EXPORT_IN_PREC(scannerSymbols, S_END); + EXPORT_IN_PREC(scannerSymbols, S_EOF); + EXPORT_IN_PREC(scannerSymbols, S_EQ); + EXPORT_IN_PREC(scannerSymbols, S_FALSE); + EXPORT_IN_PREC(scannerSymbols, S_FI); + EXPORT_IN_PREC(scannerSymbols, S_FLOAT); + EXPORT_IN_PREC(scannerSymbols, S_FOR); + EXPORT_IN_PREC(scannerSymbols, S_FUNCTION); + EXPORT_IN_PREC(scannerSymbols, S_GE); + EXPORT_IN_PREC(scannerSymbols, S_GT); + EXPORT_IN_PREC(scannerSymbols, S_HELP); + EXPORT_IN_PREC(scannerSymbols, S_IDENT); + EXPORT_IN_PREC(scannerSymbols, S_IF); + EXPORT_IN_PREC(scannerSymbols, S_ILLEGAL); + EXPORT_IN_PREC(scannerSymbols, S_IN); + EXPORT_IN_PREC(scannerSymbols, S_INFO); + EXPORT_IN_PREC(scannerSymbols, S_INT); + EXPORT_IN_PREC(scannerSymbols, S_ISBOUND); + EXPORT_IN_PREC(scannerSymbols, S_LBRACE); + EXPORT_IN_PREC(scannerSymbols, S_LBRACK); + EXPORT_IN_PREC(scannerSymbols, S_LE); + EXPORT_IN_PREC(scannerSymbols, S_LOCAL); + EXPORT_IN_PREC(scannerSymbols, S_LPAREN); + EXPORT_IN_PREC(scannerSymbols, S_LT); + EXPORT_IN_PREC(scannerSymbols, S_MAPTO); + EXPORT_IN_PREC(scannerSymbols, S_MINUS); + EXPORT_IN_PREC(scannerSymbols, S_MOD); + EXPORT_IN_PREC(scannerSymbols, S_MULT); + EXPORT_IN_PREC(scannerSymbols, S_NE); + EXPORT_IN_PREC(scannerSymbols, S_NOT); + EXPORT_IN_PREC(scannerSymbols, S_OD); + EXPORT_IN_PREC(scannerSymbols, S_OR); + EXPORT_IN_PREC(scannerSymbols, S_PLUS); + EXPORT_IN_PREC(scannerSymbols, S_POW); + EXPORT_IN_PREC(scannerSymbols, S_PRAGMA); + EXPORT_IN_PREC(scannerSymbols, S_QQUIT); + EXPORT_IN_PREC(scannerSymbols, S_QUIT); + EXPORT_IN_PREC(scannerSymbols, S_RBRACE); + EXPORT_IN_PREC(scannerSymbols, S_RBRACK); + EXPORT_IN_PREC(scannerSymbols, S_READONLY); + EXPORT_IN_PREC(scannerSymbols, S_READWRITE); + EXPORT_IN_PREC(scannerSymbols, S_REC); + EXPORT_IN_PREC(scannerSymbols, S_REPEAT); + EXPORT_IN_PREC(scannerSymbols, S_RETURN); + EXPORT_IN_PREC(scannerSymbols, S_RPAREN); + EXPORT_IN_PREC(scannerSymbols, S_SEMICOLON); + EXPORT_IN_PREC(scannerSymbols, S_STRING); + EXPORT_IN_PREC(scannerSymbols, S_THEN); + EXPORT_IN_PREC(scannerSymbols, S_TILDE); + EXPORT_IN_PREC(scannerSymbols, S_TRUE); + EXPORT_IN_PREC(scannerSymbols, S_TRYNEXT); + EXPORT_IN_PREC(scannerSymbols, S_UNBIND); + EXPORT_IN_PREC(scannerSymbols, S_UNTIL); + EXPORT_IN_PREC(scannerSymbols, S_WHILE); + return PostRestore( module ); }