diff --git a/src/syntaxtree.c b/src/syntaxtree.c index 2cbfa707e5..0b6298cbf0 100644 --- a/src/syntaxtree.c +++ b/src/syntaxtree.c @@ -31,56 +31,127 @@ #include #include +enum { MAX_ARITY = 4 }; + typedef Obj (*CompileFuncT)(Obj node, Expr expr); typedef Obj (*CompileArgT)(Expr expr); +typedef Expr (*CodeFuncT)(Obj node); +typedef Expr (*CodeArgT)(Obj node); + typedef struct { const Char * argname; CompileArgT argcomp; + CodeArgT argcode; + Int isStat; } ArgT; typedef struct { UInt1 tnum; CompileFuncT compile; + CodeFuncT code; const Char * name; - UInt1 arity; - ArgT args[4]; /* The maximum compiler arity is currently 4 */ + UInt arity; + ArgT args[MAX_ARITY]; } CompilerT; // We put compilers for statements and expressions into the same static array, // assuming that the set of their respective ids are disjoint. static const CompilerT Compilers[]; #define COMPILER_ARITY(...) (sizeof((ArgT[]){ __VA_ARGS__ }) / sizeof(ArgT)) -#define COMPILER_GENERIC(tnum, compiler, ...) \ +#define COMPILER_GENERIC(tnum, compiler, coder, ...) \ [tnum] = { \ - tnum, compiler, #tnum, COMPILER_ARITY(__VA_ARGS__), { __VA_ARGS__ } \ + tnum, compiler, coder, #tnum, COMPILER_ARITY(__VA_ARGS__), \ + { __VA_ARGS__ } \ } -#define COMPILER(tnum, compiler) COMPILER_GENERIC(tnum, compiler) +#define COMPILER(tnum, compiler, coder) \ + COMPILER_GENERIC(tnum, compiler, coder) #define COMPILER_(tnum, ...) \ - COMPILER_GENERIC(tnum, SyntaxTreeDefaultCompiler, __VA_ARGS__) + COMPILER_GENERIC(tnum, SyntaxTreeDefaultCompiler, \ + SyntaxTreeDefaultCoder, __VA_ARGS__) -#define ARG(name, func) \ +#define ARG(name, func, coder, isStat) \ { \ - name, func \ + name, func, coder, isStat \ } -#define ARG_(name) ARG(name, SyntaxTreeCompiler) -#define ARGS(name) ARG(name, 0) +// ARG_EXPR indicates an "expression argument", i.e., that the corresponding +// slot of the statement or expression type it belongs to is meant to +// reference an expression. This information is used for validation when +// coding an AST back into a T_BODY object. The expression is stored in resp. +// read from the record entry . +#define ARG_EXPR(name, func, coder) ARG(name, func, coder, 0) + +// ARG_STAT is like ARG_EXPR, except that it indicates that the corresponding +// slot should contain a statement instead of an expression. There is no +// ARG_STAT, because it is never used. + + +// ARG_EXPR_ can be used instead of ARG_EXPR if the default compiler +// and coder should be used to compile the corresponding expression or code +// the corresponding subnode. +#define ARG_EXPR_(name) \ + ARG(name, SyntaxTreeCompiler, SyntaxTreeDefaultExprCoder, 0) +#define ARG_STAT_(name) \ + ARG(name, SyntaxTreeCompiler, SyntaxTreeDefaultStatCoder, 1) + +// ARGS_EXPR can be used instead of ARG_EXPR as the last argument to the +// COMPILER_ macro, and indicates that a variable number of arguments follow; +// all are supposed to be expressions; and all of them are to be stored into a +// plist which is stored in resp. read from the record entry . +#define ARGS_EXPR(name) ARG(name, 0, 0, 0) +#define ARGS_STAT(name) ARG(name, 0, 0, 1) + +// plain list mapping statement tnums to strings +// like "T_INTEXPR" +static Obj typeStrings; +// plain record mapping strings like "T_INTEXPR" to the +// corresponding statement tnum +static Obj typeRec; -static Obj typeStrings; +static Obj ElmRecST(UInt1 tnum, Obj node, const char * name) +{ + RequirePlainRec("ElmRecST", node); + UInt rnam = RNamName(name); + if (!IsbPRec(node, rnam)) { + const char * typename = Compilers[tnum].name; + ErrorQuit("while coding %s: node field <%s> must be present", + (Int) typename, (Int)name); + } + return ElmPRec(node, rnam); +} static Obj SyntaxTreeFunc(Obj result, Obj func); +static UInt1 FIND_TNUM_FROM_STRING(Obj typestr) +{ + UInt rnam = RNamObj(typestr); + if (IsbPRec(typeRec, rnam)) { + return (UInt1)UInt_ObjInt(ElmPRec(typeRec, rnam)); + } + ErrorQuit("Unrecognized syntax tree node type %g", (Int)typestr, 0); +} + +static UInt1 GetTypeTNum(Obj node) +{ + RequirePlainRec("GetTypeTNum", node); + UInt type = RNamName("type"); + if (!IsbPRec(node, type)) { + ErrorQuit("while coding: field is not present", 0, 0); + } + return FIND_TNUM_FROM_STRING(ElmPRec(node, type)); +} + static inline Obj NewSyntaxTreeNode(UInt1 tnum) { Obj result; Obj typestr; - typestr = ELM_PLIST(typeStrings, tnum + 1); + typestr = ELM_LIST(typeStrings, tnum + 1); result = NEW_PREC(2); AssPRec(result, RNamName("type"), typestr); @@ -102,22 +173,11 @@ static Obj SyntaxTreeCompiler(Expr expr) return comp.compile(result, expr); } -static Obj SyntaxTreeArgcompInt(UInt i) -{ - return INTOBJ_INT(i); -} - static Obj SyntaxTreeRNam(Expr expr) { return NAME_RNAM(expr); } -static Obj SyntaxTreeRefLVar(Obj result, Expr expr) -{ - AssPRec(result, RNamName("lvar"), INTOBJ_INT(LVAR_REFLVAR(expr))); - return result; -} - static Obj SyntaxTreeDefaultCompiler(Obj result, Expr expr) { int i; @@ -159,6 +219,109 @@ static Obj SyntaxTreeDefaultCompiler(Obj result, Expr expr) return result; } +static Stat SyntaxTreeDefaultStatCoder(Obj); +static Stat SyntaxTreeDefaultExprCoder(Obj); + +static Expr SyntaxTreeDefaultCoder(Obj node) +{ + + RequirePlainRec("SyntaxTreeDefaultCoder", node); + + UInt1 tnum = GetTypeTNum(node); + + CompilerT comp = Compilers[tnum]; + + if (comp.code != SyntaxTreeDefaultCoder) { + return comp.code(node); + } + + // compute how big the statement/expression we are generating needs to + // be, i.e., how many Obj-sized "slots" we need; this is usually + // comp.arity, but we need to take into account variadic argument lists + UInt slots = comp.arity; + UInt arity = comp.arity; + + UInt isvararg = comp.arity > 0 && comp.args[comp.arity - 1].argcomp == 0; + Obj vararglist; + + if (isvararg) { + arity--; + vararglist = ElmRecST(tnum, node, comp.args[arity].argname); + slots = arity + LEN_LIST(vararglist); + } + + // reserve space for the statement or expressions + Expr expr = NewStatOrExpr(tnum, slots * sizeof(Expr), 0); + + UInt i; + + for (i = 0; i < arity; i++) { + Obj subast = ElmRecST(tnum, node, comp.args[i].argname); + WRITE_EXPR(expr, i, comp.args[i].argcode(subast)); + } + + if (isvararg) { + for (i = arity; i < slots; i++) { + Obj elem = ELM0_LIST(vararglist, i - arity + 1); + // Deal with empty entries in list expressions + if (elem == 0) { + WRITE_EXPR(expr, i, 0); + } + else if (comp.args[arity].isStat) { + WRITE_EXPR(expr, i, SyntaxTreeDefaultStatCoder(elem)); + } + else { + WRITE_EXPR(expr, i, SyntaxTreeDefaultExprCoder(elem)); + } + } + } + + return expr; +} + + +// Checks if node codes a statement +static Stat SyntaxTreeDefaultStatCoder(Obj node) +{ + RequirePlainRec("SyntaxTreeDefaultStatCoder", node); + Int tnum = GetTypeTNum(node); + if (!(FIRST_STAT_TNUM <= tnum && tnum <= LAST_STAT_TNUM)) + ErrorQuit("tried to code expression %s as statement", + (Int)Compilers[tnum].name, 0); + return (Stat)SyntaxTreeDefaultCoder(node); +} + +// Checks if node codes an expression +static Expr SyntaxTreeDefaultExprCoder(Obj node) +{ + RequirePlainRec("SyntaxTreeDefaultExprCoder", node); + UInt1 tnum = GetTypeTNum(node); + if (!(FIRST_EXPR_TNUM <= tnum && tnum <= LAST_EXPR_TNUM)) + ErrorQuit("tried to code statement %s as expression", + (Int)Compilers[tnum].name, 0); + return (Expr)SyntaxTreeDefaultCoder(node); +} + +static Expr SyntaxTreeCodeGVar(Obj name) +{ + RequireStringRep("SyntaxTreeCodeGVar", name); + return GVarName(CONST_CSTR_STRING(name)); +} + +static Obj SyntaxTreeRefLVar(Obj result, Expr expr) +{ + AssPRec(result, RNamName("lvar"), INTOBJ_INT(LVAR_REFLVAR(expr))); + return result; +} + +static Expr SyntaxTreeCodeRefLVar(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeRefLVar", node); + Obj lvar = ElmRecST(T_REFLVAR, node, "lvar"); + RequireSmallInt("SyntaxTreeCodeRefLVar", lvar, ""); + return REFLVAR_LVAR(INT_INTOBJ(lvar)); +} + static Obj SyntaxTreeEvalCompiler(Obj result, Expr expr) { AssPRec(result, RNamName("value"), EVAL_EXPR(expr)); @@ -200,6 +363,26 @@ static Obj SyntaxTreeRangeExpr(Obj result, Expr expr) return result; } +static Expr SyntaxTreeCodeRangeExpr(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeRangeExpr", node); + UInt hassecond = ISB_REC(node, RNamName("second")); + UInt size = hassecond ? 3 : 2; + Expr result = NewStatOrExpr(T_RANGE_EXPR, size * sizeof(Expr), 0); + WRITE_EXPR( + result, 0, + SyntaxTreeDefaultExprCoder(ElmRecST(T_RANGE_EXPR, node, "first"))); + WRITE_EXPR( + result, size - 1, + SyntaxTreeDefaultExprCoder(ElmRecST(T_RANGE_EXPR, node, "last"))); + if (hassecond) { + WRITE_EXPR(result, 1, + SyntaxTreeDefaultExprCoder( + ElmRecST(T_RANGE_EXPR, node, "second"))); + } + return result; +} + static Obj SyntaxTreeRecExpr(Obj result, Expr expr) { Obj key; @@ -236,15 +419,69 @@ static Obj SyntaxTreeRecExpr(Obj result, Expr expr) return result; } -static Obj SyntaxTreeFloatLazy(Obj result, Expr expr) +static Expr SyntaxTreeCodeRecExpr(Obj node) { - Obj string; + RequirePlainRec("SyntaxTreeCodeRecExpr", node); + UInt1 tnum = GetTypeTNum(node); + Obj keyvalue = ElmRecST(tnum, node, "keyvalue"); + UInt len = LEN_LIST(keyvalue); + Expr record = NewStatOrExpr(tnum, 2 * len * sizeof(Expr), 0); + for (int i = 0; i < len; i++) { + Obj keyvaluepair = ELM_LIST(keyvalue, i + 1); + Obj keynode = ElmRecST(tnum, keyvaluepair, "key"); + Obj valuenode = ElmRecST(tnum, keyvaluepair, "value"); + Expr key; + if (IS_STRING(keynode)) { + key = INTEXPR_INT(RNamObj(keynode)); + } + else { + key = SyntaxTreeDefaultExprCoder(keynode); + } + Expr value = SyntaxTreeDefaultExprCoder(valuenode); + WRITE_EXPR(record, 2 * i, key); + WRITE_EXPR(record, 2 * i + 1, value); + } + return record; +} - string = GET_VALUE_FROM_CURRENT_BODY(READ_EXPR(expr, 1)); +static Obj SyntaxTreeFloatLazy(Obj result, Expr expr) +{ + Obj string = GET_VALUE_FROM_CURRENT_BODY(READ_EXPR(expr, 1)); AssPRec(result, RNamName("value"), string); return result; } +static Obj SyntaxTreeFloatEager(Obj result, Expr expr) +{ + Obj value = GET_VALUE_FROM_CURRENT_BODY(READ_EXPR(expr, 0)); + Obj string = GET_VALUE_FROM_CURRENT_BODY(READ_EXPR(expr, 1)); + UChar chr = (UChar)READ_EXPR(expr, 2); + AssPRec(result, RNamName("value"), value); + AssPRec(result, RNamName("string"), string); + AssPRec(result, RNamName("mark"), ObjsChar[chr]); + return result; +} + +static Expr SyntaxTreeCodeFloatLazy(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeFloatLazy", node); + Obj value = ElmRecST(T_FLOAT_EXPR_LAZY, node, "value"); + return CodeLazyFloatExpr(value, 0); +} + +static Expr SyntaxTreeCodeFloatEager(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeFloatEager", node); + Obj value = ElmRecST(T_FLOAT_EXPR_EAGER, node, "value"); + Obj string = ElmRecST(T_FLOAT_EXPR_EAGER, node, "string"); + Obj mark = ElmRecST(T_FLOAT_EXPR_EAGER, node, "mark"); + Expr fl = NewStatOrExpr(T_FLOAT_EXPR_EAGER, 3 * sizeof(UInt), 0); + WRITE_EXPR(fl, 0, AddValueToBody(value)); + WRITE_EXPR(fl, 1, AddValueToBody(string)); + WRITE_EXPR(fl, 2, (UInt)CHAR_VALUE(mark)); + return fl; +} + static Obj SyntaxTreeIf(Obj result, Stat stat) { Obj cond; @@ -279,6 +516,53 @@ static Obj SyntaxTreeCompilePragma(Obj result, Stat stat) return result; } +static Expr SyntaxTreeCodeIf(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeIf", node); + UInt1 tnum = GetTypeTNum(node); + Obj branches = ElmRecST(tnum, node, "branches"); + UInt len = LEN_LIST(branches); + Expr ifexpr = NewStatOrExpr(tnum, 2 * len * sizeof(Expr), 0); + for (int i = 0; i < len; i++) { + Obj condbodypair = ELM_LIST(branches, i + 1); + Obj conditionnode = ElmRecST(tnum, condbodypair, "condition"); + Obj bodynode = ElmRecST(tnum, condbodypair, "body"); + Expr condition = SyntaxTreeDefaultExprCoder(conditionnode); + Stat body = SyntaxTreeDefaultStatCoder(bodynode); + WRITE_EXPR(ifexpr, 2 * i, condition); + WRITE_EXPR(ifexpr, 2 * i + 1, body); + } + return ifexpr; +} + +static Expr SyntaxTreeCodeImmediateInteger(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeImmediateInteger", node); + Obj value = ElmRecST(T_INTEXPR, node, "value"); + RequireSmallInt("SyntaxTreeCodeImmediateInteger", value, ".value"); + return INTEXPR_INT(INT_INTOBJ(value)); +} + +static Expr SyntaxTreeCodeValue(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeValue", node); + UInt1 tnum = GetTypeTNum(node); + Obj value = ElmRecST(tnum, node, "value"); + Expr expr = NewStatOrExpr(tnum, sizeof(UInt), 0); + Int ix = AddValueToBody(value); + WRITE_EXPR(expr, 0, ix); + return expr; +} + +static Expr SyntaxTreeCodeChar(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeChar", node); + Obj chr = ElmRecST(T_CHAR_EXPR, node, "value"); + Char currchar = CHAR_VALUE(chr); + Expr lit = NewStatOrExpr(T_CHAR_EXPR, sizeof(UChar), 0); + WRITE_EXPR(lit, 0, currchar); + return lit; +} static Obj SyntaxTreeFunc(Obj result, Obj func) { @@ -317,191 +601,295 @@ static Obj SyntaxTreeFunc(Obj result, Obj func) return result; } -static const CompilerT Compilers[] = { - COMPILER_(T_PROCCALL_0ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_1ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_2ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_3ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_4ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_5ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_6ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_PROCCALL_XARGS, ARG_("funcref"), ARGS("args")), - - COMPILER_(T_PROCCALL_OPTS, ARG_("opts"), ARG_("call")), +static UInt SyntaxTreeCodeFunc_Internal(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeFunc_Internal", node); + Int narg = INT_INTOBJ(ElmRecST(T_FUNC_EXPR, node, "narg")); + Int nloc = INT_INTOBJ(ElmRecST(T_FUNC_EXPR, node, "nloc")); + Obj nams = ElmRecST(T_FUNC_EXPR, node, "nams"); + Obj variadic = ElmRecST(T_FUNC_EXPR, node, "variadic"); + if (variadic == True) { + narg = -narg; + } + CodeFuncExprBegin(narg, nloc, nams, 0); + Obj stat_rec = ElmRecST(T_FUNC_EXPR, node, "stats"); + Obj body_stats = ElmRecST(T_FUNC_EXPR, stat_rec, "statements"); + UInt nr_stats = LEN_LIST(body_stats); + for (int i = 1; i <= nr_stats; i++) { + Expr current = SyntaxTreeDefaultStatCoder(ELM_LIST(body_stats, i)); + PushStat(current); + } + return nr_stats; +} - COMPILER_(T_EMPTY), +static Expr SyntaxTreeCodeFunc(Obj node) +{ + RequirePlainRec("SyntaxTreeCodeFunc", node); + UInt nr_stats = SyntaxTreeCodeFunc_Internal(node); + Expr funcexpr = CodeFuncExprEnd(nr_stats, 0); + return funcexpr; +} - COMPILER_(T_SEQ_STAT, ARGS("statements")), - COMPILER_(T_SEQ_STAT2, ARGS("statements")), - COMPILER_(T_SEQ_STAT3, ARGS("statements")), - COMPILER_(T_SEQ_STAT4, ARGS("statements")), - COMPILER_(T_SEQ_STAT5, ARGS("statements")), - COMPILER_(T_SEQ_STAT6, ARGS("statements")), - COMPILER_(T_SEQ_STAT7, ARGS("statements")), +Obj FuncSYNTAX_TREE_CODE(Obj self, Obj tree) +{ + RequirePlainRec("SYNTAX_TREE_CODE", tree); + CodeBegin(); + UInt nr_stats = SyntaxTreeCodeFunc_Internal(tree); + CodeFuncExprEnd(nr_stats, 0); + Obj func = CodeEnd(0); + if (IsbPRec(tree, RNamName("name"))) { + Obj name = ELM_REC(tree, RNamName("name")); + SET_NAME_FUNC(func, name); + } + return func; +} - COMPILER(T_IF, SyntaxTreeIf), - COMPILER(T_IF_ELSE, SyntaxTreeIf), - COMPILER(T_IF_ELIF, SyntaxTreeIf), - COMPILER(T_IF_ELIF_ELSE, SyntaxTreeIf), +static const CompilerT Compilers[] = { + COMPILER_(T_PROCCALL_0ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_1ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_2ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_3ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_4ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_5ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_6ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_PROCCALL_XARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), - COMPILER_(T_FOR, ARG_("variable"), ARG_("collection"), ARGS("body")), - COMPILER_(T_FOR2, ARG_("variable"), ARG_("collection"), ARGS("body")), - COMPILER_(T_FOR3, ARG_("variable"), ARG_("collection"), ARGS("body")), + COMPILER_(T_PROCCALL_OPTS, ARG_EXPR_("opts"), ARG_STAT_("call")), - COMPILER_( - T_FOR_RANGE, ARG_("variable"), ARG_("collection"), ARGS("body")), - COMPILER_( - T_FOR_RANGE2, ARG_("variable"), ARG_("collection"), ARGS("body")), - COMPILER_( - T_FOR_RANGE3, ARG_("variable"), ARG_("collection"), ARGS("body")), - - COMPILER_(T_WHILE, ARG_("condition"), ARGS("body")), - COMPILER_(T_WHILE2, ARG_("condition"), ARGS("body")), - COMPILER_(T_WHILE3, ARG_("condition"), ARGS("body")), + COMPILER_(T_EMPTY), - COMPILER_(T_REPEAT, ARG_("condition"), ARGS("body")), - COMPILER_(T_REPEAT2, ARG_("condition"), ARGS("body")), - COMPILER_(T_REPEAT3, ARG_("condition"), ARGS("body")), + COMPILER_(T_SEQ_STAT, ARGS_STAT("statements")), + COMPILER_(T_SEQ_STAT2, ARGS_STAT("statements")), + COMPILER_(T_SEQ_STAT3, ARGS_STAT("statements")), + COMPILER_(T_SEQ_STAT4, ARGS_STAT("statements")), + COMPILER_(T_SEQ_STAT5, ARGS_STAT("statements")), + COMPILER_(T_SEQ_STAT6, ARGS_STAT("statements")), + COMPILER_(T_SEQ_STAT7, ARGS_STAT("statements")), + + COMPILER(T_IF, SyntaxTreeIf, SyntaxTreeCodeIf), + COMPILER(T_IF_ELSE, SyntaxTreeIf, SyntaxTreeCodeIf), + COMPILER(T_IF_ELIF, SyntaxTreeIf, SyntaxTreeCodeIf), + COMPILER(T_IF_ELIF_ELSE, SyntaxTreeIf, SyntaxTreeCodeIf), + + COMPILER_(T_FOR, + ARG_EXPR_("variable"), + ARG_EXPR_("collection"), + ARGS_STAT("body")), + COMPILER_(T_FOR2, + ARG_EXPR_("variable"), + ARG_EXPR_("collection"), + ARGS_STAT("body")), + COMPILER_(T_FOR3, + ARG_EXPR_("variable"), + ARG_EXPR_("collection"), + ARGS_STAT("body")), + + COMPILER_(T_FOR_RANGE, + ARG_EXPR_("variable"), + ARG_EXPR_("collection"), + ARGS_STAT("body")), + COMPILER_(T_FOR_RANGE2, + ARG_EXPR_("variable"), + ARG_EXPR_("collection"), + ARGS_STAT("body")), + COMPILER_(T_FOR_RANGE3, + ARG_EXPR_("variable"), + ARG_EXPR_("collection"), + ARGS_STAT("body")), + + COMPILER_(T_WHILE, ARG_EXPR_("condition"), ARGS_STAT("body")), + COMPILER_(T_WHILE2, ARG_EXPR_("condition"), ARGS_STAT("body")), + COMPILER_(T_WHILE3, ARG_EXPR_("condition"), ARGS_STAT("body")), + + COMPILER_(T_REPEAT, ARG_EXPR_("condition"), ARGS_STAT("body")), + COMPILER_(T_REPEAT2, ARG_EXPR_("condition"), ARGS_STAT("body")), + COMPILER_(T_REPEAT3, ARG_EXPR_("condition"), ARGS_STAT("body")), #ifdef HPCGAP - COMPILER_( - T_ATOMIC, ARG_("body"), ARGS("locks")), // TODO: better output + COMPILER_(T_ATOMIC, + ARG_STAT_("body"), + ARGS_EXPR("locks")), // TODO: better output #endif COMPILER_(T_BREAK), COMPILER_(T_CONTINUE), - COMPILER_(T_RETURN_OBJ, ARG_("obj")), + COMPILER_(T_RETURN_OBJ, ARG_EXPR_("obj")), COMPILER_(T_RETURN_VOID), - COMPILER_(T_ASS_LVAR, ARG("lvar", SyntaxTreeArgcompInt), ARG_("rhs")), - COMPILER_(T_UNB_LVAR, ARG("lvar", SyntaxTreeArgcompInt)), + COMPILER_(T_ASS_LVAR, + ARG_EXPR("lvar", ObjInt_UInt, UInt_ObjInt), + ARG_EXPR_("rhs")), + COMPILER_(T_UNB_LVAR, ARG_EXPR("lvar", ObjInt_UInt, UInt_ObjInt)), - COMPILER_(T_ASS_HVAR, ARG("hvar", SyntaxTreeArgcompInt), ARG_("rhs")), - COMPILER_(T_UNB_HVAR, ARG("hvar", SyntaxTreeArgcompInt)), + COMPILER_(T_ASS_HVAR, + ARG_EXPR("hvar", ObjInt_UInt, UInt_ObjInt), + ARG_EXPR_("rhs")), + COMPILER_(T_UNB_HVAR, ARG_EXPR("hvar", ObjInt_UInt, UInt_ObjInt)), - COMPILER_(T_ASS_GVAR, ARG("gvar", NameGVar), ARG_("rhs")), - COMPILER_(T_UNB_GVAR, ARG("gvar", NameGVar)), + COMPILER_(T_ASS_GVAR, + ARG_EXPR("gvar", NameGVar, SyntaxTreeCodeGVar), + ARG_EXPR_("rhs")), + COMPILER_(T_UNB_GVAR, ARG_EXPR("gvar", NameGVar, SyntaxTreeCodeGVar)), - COMPILER_(T_ASS_LIST, ARG_("list"), ARG_("pos"), ARG_("rhs")), - COMPILER_(T_ASS2_LIST, ARG_("list"), ARG_("pos"), ARG_("rhs")), - COMPILER_(T_ASSS_LIST, ARG_("list"), ARG_("poss"), ARG_("rhss")), + COMPILER_( + T_ASS_LIST, ARG_EXPR_("list"), ARG_EXPR_("pos"), ARG_EXPR_("rhs")), + COMPILER_( + T_ASS2_LIST, ARG_EXPR_("list"), ARG_EXPR_("pos"), ARG_EXPR_("rhs")), + COMPILER_( + T_ASSS_LIST, ARG_EXPR_("list"), ARG_EXPR_("poss"), ARG_EXPR_("rhss")), COMPILER_(T_ASS_LIST_LEV, - ARG_("lists"), - ARG_("pos"), - ARG_("rhss"), - ARG("level", SyntaxTreeArgcompInt)), + ARG_EXPR_("lists"), + ARG_EXPR_("pos"), + ARG_EXPR_("rhss"), + ARG_EXPR("level", ObjInt_UInt, UInt_ObjInt)), COMPILER_(T_ASSS_LIST_LEV, - ARG_("lists"), - ARG_("poss"), - ARG_("rhss"), - ARG("level", SyntaxTreeArgcompInt)), - COMPILER_(T_UNB_LIST, ARG_("list"), ARG_("pos")), + ARG_EXPR_("lists"), + ARG_EXPR_("poss"), + ARG_EXPR_("rhss"), + ARG_EXPR("level", ObjInt_UInt, UInt_ObjInt)), + COMPILER_(T_UNB_LIST, ARG_EXPR_("list"), ARG_EXPR_("pos")), COMPILER_(T_ASS_REC_NAME, - ARG_("record"), - ARG("rnam", SyntaxTreeRNam), - ARG_("rhs")), - COMPILER_( - T_ASS_REC_EXPR, ARG_("record"), ARG_("expression"), ARG_("rhs")), - COMPILER_(T_UNB_REC_NAME, ARG_("record"), ARG("rnam", SyntaxTreeRNam)), - COMPILER_(T_UNB_REC_EXPR, ARG_("record"), ARG_("expression")), - - COMPILER_(T_ASS_POSOBJ, ARG_("posobj"), ARG_("pos"), ARG_("rhs")), - COMPILER_(T_UNB_POSOBJ, ARG_("posobj"), ARG_("pos")), - - COMPILER_(T_ASS_COMOBJ_NAME, ARG_("comobj"), ARG("rnam", SyntaxTreeRNam)), + ARG_EXPR_("record"), + ARG_EXPR("rnam", SyntaxTreeRNam, RNamObj), + ARG_EXPR_("rhs")), + COMPILER_(T_ASS_REC_EXPR, + ARG_EXPR_("record"), + ARG_EXPR_("expression"), + ARG_EXPR_("rhs")), + COMPILER_(T_UNB_REC_NAME, + ARG_EXPR_("record"), + ARG_EXPR("rnam", SyntaxTreeRNam, RNamObj)), + COMPILER_(T_UNB_REC_EXPR, ARG_EXPR_("record"), ARG_EXPR_("expression")), + + COMPILER_(T_ASS_POSOBJ, + ARG_EXPR_("posobj"), + ARG_EXPR_("pos"), + ARG_EXPR_("rhs")), + COMPILER_(T_UNB_POSOBJ, ARG_EXPR_("posobj"), ARG_EXPR_("pos")), + + COMPILER_(T_ASS_COMOBJ_NAME, + ARG_EXPR_("comobj"), + ARG_EXPR("rnam", SyntaxTreeRNam, RNamObj), + ARG_EXPR_("rhs")), + COMPILER_(T_ASS_COMOBJ_EXPR, + ARG_EXPR_("comobj"), + ARG_EXPR_("expression"), + ARG_EXPR_("rhs")), + COMPILER_(T_UNB_COMOBJ_NAME, + ARG_EXPR_("comobj"), + ARG_EXPR("rnam", SyntaxTreeRNam, RNamObj)), COMPILER_( - T_ASS_COMOBJ_EXPR, ARG_("comobj"), ARG_("expression"), ARG_("rhs")), - COMPILER_(T_UNB_COMOBJ_NAME, ARG_("comobj"), ARG("rnam", SyntaxTreeRNam)), - COMPILER_(T_UNB_COMOBJ_EXPR, ARG_("comobj"), ARG_("expression")), + T_UNB_COMOBJ_EXPR, ARG_EXPR_("comobj"), ARG_EXPR_("expression")), - COMPILER_(T_INFO, ARG_("sel"), ARG_("lev"), ARGS("args")), - COMPILER_(T_ASSERT_2ARGS, ARG_("level"), ARG_("condition")), - COMPILER_( - T_ASSERT_3ARGS, ARG_("level"), ARG_("condition"), ARG_("message")), + COMPILER_(T_INFO, ARG_EXPR_("sel"), ARG_EXPR_("lev"), ARGS_EXPR("args")), + COMPILER_(T_ASSERT_2ARGS, ARG_EXPR_("level"), ARG_EXPR_("condition")), + COMPILER_(T_ASSERT_3ARGS, + ARG_EXPR_("level"), + ARG_EXPR_("condition"), + ARG_EXPR_("message")), + COMPILER(T_PRAGMA, SyntaxTreeCompilePragma, SyntaxTreeCodeValue), - COMPILER(T_PRAGMA, SyntaxTreeCompilePragma), /* Statements */ - COMPILER_(T_FUNCCALL_0ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_1ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_2ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_3ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_4ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_5ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_6ARGS, ARG_("funcref"), ARGS("args")), - COMPILER_(T_FUNCCALL_XARGS, ARG_("funcref"), ARGS("args")), - - COMPILER(T_FUNC_EXPR, SyntaxTreeFuncExpr), - - COMPILER_(T_FUNCCALL_OPTS, ARG_("opts"), ARG_("call")), - - - COMPILER_(T_OR, ARG_("left"), ARG_("right")), - COMPILER_(T_AND, ARG_("left"), ARG_("right")), - COMPILER_(T_NOT, ARG_("op")), - COMPILER_(T_EQ, ARG_("left"), ARG_("right")), - COMPILER_(T_NE, ARG_("left"), ARG_("right")), - COMPILER_(T_LT, ARG_("left"), ARG_("right")), - COMPILER_(T_GE, ARG_("left"), ARG_("right")), - COMPILER_(T_GT, ARG_("left"), ARG_("right")), - COMPILER_(T_LE, ARG_("left"), ARG_("right")), - COMPILER_(T_IN, ARG_("left"), ARG_("right")), - COMPILER_(T_SUM, ARG_("left"), ARG_("right")), - COMPILER_(T_AINV, ARG_("op")), - COMPILER_(T_DIFF, ARG_("left"), ARG_("right")), - COMPILER_(T_PROD, ARG_("left"), ARG_("right")), - COMPILER_(T_QUO, ARG_("left"), ARG_("right")), - COMPILER_(T_MOD, ARG_("left"), ARG_("right")), - COMPILER_(T_POW, ARG_("left"), ARG_("right")), - - COMPILER(T_INTEXPR, SyntaxTreeEvalCompiler), - COMPILER(T_INT_EXPR, SyntaxTreeEvalCompiler), + COMPILER_(T_FUNCCALL_0ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_1ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_2ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_3ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_4ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_5ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_6ARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + COMPILER_(T_FUNCCALL_XARGS, ARG_EXPR_("funcref"), ARGS_EXPR("args")), + + COMPILER(T_FUNC_EXPR, SyntaxTreeFuncExpr, SyntaxTreeCodeFunc), + + COMPILER_(T_FUNCCALL_OPTS, ARG_EXPR_("opts"), ARG_EXPR_("call")), + + + COMPILER_(T_OR, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_AND, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_NOT, ARG_EXPR_("op")), + COMPILER_(T_EQ, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_NE, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_LT, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_GE, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_GT, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_LE, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_IN, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_SUM, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_AINV, ARG_EXPR_("op")), + COMPILER_(T_DIFF, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_PROD, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_QUO, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_MOD, ARG_EXPR_("left"), ARG_EXPR_("right")), + COMPILER_(T_POW, ARG_EXPR_("left"), ARG_EXPR_("right")), + + COMPILER( + T_INTEXPR, SyntaxTreeEvalCompiler, SyntaxTreeCodeImmediateInteger), + COMPILER(T_INT_EXPR, SyntaxTreeEvalCompiler, SyntaxTreeCodeValue), COMPILER_(T_TRUE_EXPR), COMPILER_(T_FALSE_EXPR), COMPILER_(T_TILDE_EXPR), - COMPILER(T_CHAR_EXPR, SyntaxTreeEvalCompiler), - COMPILER_(T_PERM_EXPR, ARGS("cycles")), - COMPILER_(T_PERM_CYCLE, ARGS("points")), - COMPILER_(T_LIST_EXPR, ARGS("list")), - COMPILER_(T_LIST_TILDE_EXPR, ARGS("list")), - COMPILER(T_RANGE_EXPR, SyntaxTreeRangeExpr), - COMPILER(T_STRING_EXPR, SyntaxTreeEvalCompiler), - COMPILER(T_REC_EXPR, SyntaxTreeRecExpr), + COMPILER(T_CHAR_EXPR, SyntaxTreeEvalCompiler, SyntaxTreeCodeChar), + COMPILER_(T_PERM_EXPR, ARGS_EXPR("cycles")), + COMPILER_(T_PERM_CYCLE, ARGS_EXPR("points")), + COMPILER_(T_LIST_EXPR, ARGS_EXPR("list")), + COMPILER_(T_LIST_TILDE_EXPR, ARGS_EXPR("list")), + COMPILER(T_RANGE_EXPR, SyntaxTreeRangeExpr, SyntaxTreeCodeRangeExpr), + COMPILER(T_STRING_EXPR, SyntaxTreeEvalCompiler, SyntaxTreeCodeValue), + COMPILER(T_REC_EXPR, SyntaxTreeRecExpr, SyntaxTreeCodeRecExpr), COMPILER_(T_REC_TILDE_EXPR), - COMPILER(T_FLOAT_EXPR_EAGER, SyntaxTreeEvalCompiler), - COMPILER(T_FLOAT_EXPR_LAZY, SyntaxTreeFloatLazy), + COMPILER( + T_FLOAT_EXPR_EAGER, SyntaxTreeFloatEager, SyntaxTreeCodeFloatEager), + COMPILER(T_FLOAT_EXPR_LAZY, SyntaxTreeFloatLazy, SyntaxTreeCodeFloatLazy), // T_REFLVAR is encoded differently from all other // references to LVARs, so we have to treat // them specially here - COMPILER(T_REFLVAR, SyntaxTreeRefLVar), - COMPILER_(T_ISB_LVAR, ARG("lvar", SyntaxTreeArgcompInt)), + COMPILER(T_REFLVAR, SyntaxTreeRefLVar, SyntaxTreeCodeRefLVar), + COMPILER_(T_ISB_LVAR, ARG_EXPR("lvar", ObjInt_UInt, UInt_ObjInt)), - COMPILER_(T_REF_HVAR, ARG("hvar", SyntaxTreeArgcompInt)), - COMPILER_(T_ISB_HVAR, ARG("hvar", SyntaxTreeArgcompInt)), + COMPILER_(T_REF_HVAR, ARG_EXPR("hvar", ObjInt_UInt, UInt_ObjInt)), + COMPILER_(T_ISB_HVAR, ARG_EXPR("hvar", ObjInt_UInt, UInt_ObjInt)), - COMPILER_(T_REF_GVAR, ARG("gvar", NameGVar)), - COMPILER_(T_ISB_GVAR, ARG("gvar", NameGVar)), + COMPILER_(T_REF_GVAR, ARG_EXPR("gvar", NameGVar, SyntaxTreeCodeGVar)), + COMPILER_(T_ISB_GVAR, ARG_EXPR("gvar", NameGVar, SyntaxTreeCodeGVar)), // TODO: can this be unified? - COMPILER_(T_ELM_LIST, ARG_("list"), ARG_("pos")), - COMPILER_(T_ELM2_LIST, ARG_("list"), ARG_("pos1"), ARG_("pos2")), - COMPILER_(T_ELMS_LIST, ARG_("list"), ARG_("poss")), - COMPILER_(T_ELM_LIST_LEV, ARG_("lists"), ARG_("pos"), ARG_("level")), - COMPILER_(T_ELMS_LIST_LEV, ARG_("lists"), ARG_("poss"), ARG_("level")), - COMPILER_(T_ISB_LIST, ARG_("list"), ARG_("pos")), - COMPILER_(T_ELM_REC_NAME, ARG_("record"), ARG("name", SyntaxTreeRNam)), - COMPILER_(T_ELM_REC_EXPR, ARG_("record"), ARG_("expression")), - COMPILER_(T_ISB_REC_NAME, ARG_("record"), ARG("name", SyntaxTreeRNam)), - COMPILER_(T_ISB_REC_EXPR, ARG_("record"), ARG_("expression")), - COMPILER_(T_ELM_POSOBJ, ARG_("posobj"), ARG_("pos")), - COMPILER_(T_ISB_POSOBJ, ARG_("posobj"), ARG_("pos")), - COMPILER_(T_ELM_COMOBJ_NAME, ARG_("comobj"), ARG("name", SyntaxTreeRNam)), - COMPILER_(T_ELM_COMOBJ_EXPR, ARG_("comobj"), ARG_("expression")), - COMPILER_(T_ISB_COMOBJ_NAME, ARG_("comobj"), ARG("name", SyntaxTreeRNam)), - COMPILER_(T_ISB_COMOBJ_EXPR, ARG_("comobj"), ARG_("expression")), + COMPILER_(T_ELM_LIST, ARG_EXPR_("list"), ARG_EXPR_("pos")), + COMPILER_( + T_ELM2_LIST, ARG_EXPR_("list"), ARG_EXPR_("pos1"), ARG_EXPR_("pos2")), + COMPILER_(T_ELMS_LIST, ARG_EXPR_("list"), ARG_EXPR_("poss")), + COMPILER_(T_ELM_LIST_LEV, + ARG_EXPR_("lists"), + ARG_EXPR_("pos"), + ARG_EXPR_("level")), + COMPILER_(T_ELMS_LIST_LEV, + ARG_EXPR_("lists"), + ARG_EXPR_("poss"), + ARG_EXPR_("level")), + COMPILER_(T_ISB_LIST, ARG_EXPR_("list"), ARG_EXPR_("pos")), + COMPILER_(T_ELM_REC_NAME, + ARG_EXPR_("record"), + ARG_EXPR("name", SyntaxTreeRNam, RNamObj)), + COMPILER_(T_ELM_REC_EXPR, ARG_EXPR_("record"), ARG_EXPR_("expression")), + COMPILER_(T_ISB_REC_NAME, + ARG_EXPR_("record"), + ARG_EXPR("name", SyntaxTreeRNam, RNamObj)), + COMPILER_(T_ISB_REC_EXPR, ARG_EXPR_("record"), ARG_EXPR_("expression")), + COMPILER_(T_ELM_POSOBJ, ARG_EXPR_("posobj"), ARG_EXPR_("pos")), + COMPILER_(T_ISB_POSOBJ, ARG_EXPR_("posobj"), ARG_EXPR_("pos")), + COMPILER_(T_ELM_COMOBJ_NAME, + ARG_EXPR_("comobj"), + ARG_EXPR("name", SyntaxTreeRNam, RNamObj)), + COMPILER_( + T_ELM_COMOBJ_EXPR, ARG_EXPR_("comobj"), ARG_EXPR_("expression")), + COMPILER_(T_ISB_COMOBJ_NAME, + ARG_EXPR_("comobj"), + ARG_EXPR("name", SyntaxTreeRNam, RNamObj)), + COMPILER_( + T_ISB_COMOBJ_EXPR, ARG_EXPR_("comobj"), ARG_EXPR_("expression")), }; static Obj FuncSYNTAX_TREE(Obj self, Obj func) @@ -517,6 +905,7 @@ static Obj FuncSYNTAX_TREE(Obj self, Obj func) } static StructGVarFunc GVarFuncs[] = { GVAR_FUNC(SYNTAX_TREE, 1, "func"), + GVAR_FUNC(SYNTAX_TREE_CODE, 1, "tree"), { 0, 0, 0, 0, 0 } }; static Int InitKernel(StructInitInfo * module) @@ -525,6 +914,7 @@ static Int InitKernel(StructInitInfo * module) InitHdlrFuncsFromTable(GVarFuncs); InitGlobalBag(&typeStrings, "typeStrings"); + InitGlobalBag(&typeRec, "typeRec"); return 0; } @@ -535,12 +925,16 @@ static Int InitLibrary(StructInitInfo * module) InitGVarFuncsFromTable(GVarFuncs); typeStrings = NEW_PLIST(T_PLIST, ARRAY_SIZE(Compilers)); + typeRec = NEW_PREC(0); for (UInt tnum = 0; tnum < ARRAY_SIZE(Compilers); tnum++) { const char * str = Compilers[tnum].name; - if (str) + if (str) { + AssPRec(typeRec, RNamName(str), ObjInt_UInt(tnum)); ASS_LIST(typeStrings, tnum + 1, MakeImmString(str)); + } } + /* return success */ return 0; } diff --git a/tst/testinstall/syntaxtree.tst b/tst/testinstall/syntaxtree.tst index fb8f728a64..2307e39559 100644 --- a/tst/testinstall/syntaxtree.tst +++ b/tst/testinstall/syntaxtree.tst @@ -2854,6 +2854,8 @@ rec( stats := rec( statements := [ rec( obj := rec( + mark := '\000', + string := "1.0", type := "T_FLOAT_EXPR_EAGER", value := 1. ), type := "T_RETURN_OBJ" ) ],