#include #include #include #include #include "util.h" #include "lllparser.h" #include "tokenize.h" struct _parseOutput { Node node; int newpos; }; // Helper, returns subtree and position of start of next node _parseOutput _parse(std::vector inp, int pos) { Metadata met = inp[pos].metadata; _parseOutput o; // Bracket: keep grabbing tokens until we get to the // corresponding closing bracket if (inp[pos].val == "(" || inp[pos].val == "[") { std::string fun, rbrack; std::vector args; pos += 1; if (inp[pos].val == "[") { fun = "access"; rbrack = "]"; } else rbrack = ")"; // First argument is the function while (inp[pos].val != ")") { _parseOutput po = _parse(inp, pos); if (fun.length() == 0 && po.node.type == 1) { std::cerr << "Error: first arg must be function\n"; fun = po.node.val; } else if (fun.length() == 0) { fun = po.node.val; } else { args.push_back(po.node); } pos = po.newpos; } o.newpos = pos + 1; o.node = astnode(fun, args, met); } // Normal token, return it and advance to next token else { o.newpos = pos + 1; o.node = token(inp[pos].val, met); } return o; } // stream of tokens -> lisp parse tree Node parseLLLTokenStream(std::vector inp) { _parseOutput o = _parse(inp, 0); return o.node; } // Parses LLL Node parseLLL(std::string s, bool allowFileRead) { std::string input = s; std::string file = "main"; if (exists(s) && allowFileRead) { file = s; input = get_file_contents(s); } return parseLLLTokenStream(tokenize(s, Metadata(file, 0, 0), true)); }