398 lines
14 KiB
398 lines
14 KiB
#include <stdio.h>
|
|
#include <iostream>
|
|
#include <vector>
|
|
#include <map>
|
|
#include "util.h"
|
|
#include "parser.h"
|
|
#include "tokenize.h"
|
|
|
|
// Extended BEDMAS precedence order
|
|
int precedence(Node tok) {
|
|
std::string v = tok.val;
|
|
if (v == "!" || v == "not") return 0;
|
|
else if (v=="^" || v == "**") return 1;
|
|
else if (v=="*" || v=="/" || v=="@/" || v=="%" || v=="@%") return 2;
|
|
else if (v=="+" || v=="-") return 3;
|
|
else if (v=="<" || v==">" || v=="<=" || v==">=") return 4;
|
|
else if (v=="@<" || v=="@>" || v=="@<=" || v=="@>=") return 4;
|
|
else if (v=="&" || v=="|" || v=="xor" || v=="==" || v == "!=") return 5;
|
|
else if (v=="&&" || v=="and") return 6;
|
|
else if (v=="||" || v=="or") return 7;
|
|
else if (v==":") return 8;
|
|
else if (v=="=") return 10;
|
|
else if (v=="+=" || v=="-=" || v=="*=" || v=="/=" || v=="%=") return 10;
|
|
else if (v=="@/=" || v=="@%=") return 10;
|
|
else return -1;
|
|
}
|
|
|
|
// Token classification for shunting-yard purposes
|
|
int toktype(Node tok) {
|
|
if (tok.type == ASTNODE) return COMPOUND;
|
|
std::string v = tok.val;
|
|
if (v == "(" || v == "[" || v == "{") return LPAREN;
|
|
else if (v == ")" || v == "]" || v == "}") return RPAREN;
|
|
else if (v == ",") return COMMA;
|
|
else if (v == "!" || v == "not" || v == "neg") return UNARY_OP;
|
|
else if (precedence(tok) >= 0) return BINARY_OP;
|
|
if (tok.val[0] != '"' && tok.val[0] != '\'') {
|
|
for (unsigned i = 0; i < tok.val.length(); i++) {
|
|
if (chartype(tok.val[i]) == SYMB) {
|
|
err("Invalid symbol: "+tok.val, tok.metadata);
|
|
}
|
|
}
|
|
}
|
|
return ALPHANUM;
|
|
}
|
|
|
|
|
|
// Converts to reverse polish notation
|
|
std::vector<Node> shuntingYard(std::vector<Node> tokens) {
|
|
std::vector<Node> iq;
|
|
for (int i = tokens.size() - 1; i >= 0; i--) {
|
|
iq.push_back(tokens[i]);
|
|
}
|
|
std::vector<Node> oq;
|
|
std::vector<Node> stack;
|
|
Node prev, tok;
|
|
int prevtyp = 0, toktyp = 0;
|
|
|
|
while (iq.size()) {
|
|
prev = tok;
|
|
prevtyp = toktyp;
|
|
tok = iq.back();
|
|
toktyp = toktype(tok);
|
|
iq.pop_back();
|
|
// Alphanumerics go straight to output queue
|
|
if (toktyp == ALPHANUM) {
|
|
oq.push_back(tok);
|
|
}
|
|
// Left parens go on stack and output queue
|
|
else if (toktyp == LPAREN) {
|
|
if (prevtyp != ALPHANUM && prevtyp != RPAREN) {
|
|
oq.push_back(token("id", tok.metadata));
|
|
}
|
|
stack.push_back(tok);
|
|
oq.push_back(tok);
|
|
}
|
|
// If rparen, keep moving from stack to output queue until lparen
|
|
else if (toktyp == RPAREN) {
|
|
while (stack.size() && toktype(stack.back()) != LPAREN) {
|
|
oq.push_back(stack.back());
|
|
stack.pop_back();
|
|
}
|
|
if (stack.size()) {
|
|
stack.pop_back();
|
|
}
|
|
oq.push_back(tok);
|
|
}
|
|
else if (toktyp == UNARY_OP) {
|
|
stack.push_back(tok);
|
|
}
|
|
// If binary op, keep popping from stack while higher bedmas precedence
|
|
else if (toktyp == BINARY_OP) {
|
|
if (tok.val == "-" && prevtyp != ALPHANUM && prevtyp != RPAREN) {
|
|
stack.push_back(token("neg", tok.metadata));
|
|
}
|
|
else {
|
|
int prec = precedence(tok);
|
|
while (stack.size()
|
|
&& (toktype(stack.back()) == BINARY_OP
|
|
|| toktype(stack.back()) == UNARY_OP)
|
|
&& precedence(stack.back()) <= prec) {
|
|
oq.push_back(stack.back());
|
|
stack.pop_back();
|
|
}
|
|
stack.push_back(tok);
|
|
}
|
|
}
|
|
// Comma means finish evaluating the argument
|
|
else if (toktyp == COMMA) {
|
|
while (stack.size() && toktype(stack.back()) != LPAREN) {
|
|
oq.push_back(stack.back());
|
|
stack.pop_back();
|
|
}
|
|
}
|
|
}
|
|
while (stack.size()) {
|
|
oq.push_back(stack.back());
|
|
stack.pop_back();
|
|
}
|
|
return oq;
|
|
}
|
|
|
|
// Converts reverse polish notation into tree
|
|
Node treefy(std::vector<Node> stream) {
|
|
std::vector<Node> iq;
|
|
for (int i = stream.size() -1; i >= 0; i--) {
|
|
iq.push_back(stream[i]);
|
|
}
|
|
std::vector<Node> oq;
|
|
while (iq.size()) {
|
|
Node tok = iq.back();
|
|
iq.pop_back();
|
|
int typ = toktype(tok);
|
|
// If unary, take node off end of oq and wrap it with the operator
|
|
// If binary, do the same with two nodes
|
|
if (typ == UNARY_OP || typ == BINARY_OP) {
|
|
std::vector<Node> args;
|
|
int rounds = (typ == BINARY_OP) ? 2 : 1;
|
|
for (int i = 0; i < rounds; i++) {
|
|
if (oq.size() == 0) {
|
|
err("Line malformed, not enough args for "+tok.val,
|
|
tok.metadata);
|
|
}
|
|
args.push_back(oq.back());
|
|
oq.pop_back();
|
|
}
|
|
std::vector<Node> args2;
|
|
while (args.size()) {
|
|
args2.push_back(args.back());
|
|
args.pop_back();
|
|
}
|
|
oq.push_back(astnode(tok.val, args2, tok.metadata));
|
|
}
|
|
// If rparen, keep grabbing until we get to an lparen
|
|
else if (typ == RPAREN) {
|
|
std::vector<Node> args;
|
|
while (1) {
|
|
if (toktype(oq.back()) == LPAREN) break;
|
|
args.push_back(oq.back());
|
|
oq.pop_back();
|
|
if (!oq.size()) err("Bracket without matching", tok.metadata);
|
|
}
|
|
oq.pop_back();
|
|
args.push_back(oq.back());
|
|
oq.pop_back();
|
|
// We represent a[b] as (access a b)
|
|
if (tok.val == "]")
|
|
args.push_back(token("access", tok.metadata));
|
|
if (args.back().type == ASTNODE)
|
|
args.push_back(token("fun", tok.metadata));
|
|
std::string fun = args.back().val;
|
|
args.pop_back();
|
|
// We represent [1,2,3] as (array_lit 1 2 3)
|
|
if (fun == "access" && args.size() && args.back().val == "id") {
|
|
fun = "array_lit";
|
|
args.pop_back();
|
|
}
|
|
std::vector<Node> args2;
|
|
while (args.size()) {
|
|
args2.push_back(args.back());
|
|
args.pop_back();
|
|
}
|
|
// When evaluating 2 + (3 * 5), the shunting yard algo turns that
|
|
// into 2 ( id 3 5 * ) +, effectively putting "id" as a dummy
|
|
// function where the algo was expecting a function to call the
|
|
// thing inside the brackets. This reverses that step
|
|
if (fun == "id" && args2.size() == 1) {
|
|
oq.push_back(args2[0]);
|
|
}
|
|
else {
|
|
oq.push_back(astnode(fun, args2, tok.metadata));
|
|
}
|
|
}
|
|
else oq.push_back(tok);
|
|
// This is messy, but has to be done. Import/inset other files here
|
|
std::string v = oq.back().val;
|
|
if ((v == "inset" || v == "import" || v == "create")
|
|
&& oq.back().args.size() == 1
|
|
&& oq.back().args[0].type == TOKEN) {
|
|
int lastSlashPos = tok.metadata.file.rfind("/");
|
|
std::string root;
|
|
if (lastSlashPos >= 0)
|
|
root = tok.metadata.file.substr(0, lastSlashPos) + "/";
|
|
else
|
|
root = "";
|
|
std::string filename = oq.back().args[0].val;
|
|
filename = filename.substr(1, filename.length() - 2);
|
|
if (!exists(root + filename))
|
|
err("File does not exist: "+root + filename, tok.metadata);
|
|
oq.back().args.pop_back();
|
|
oq.back().args.push_back(parseSerpent(root + filename));
|
|
}
|
|
//Useful for debugging
|
|
//for (int i = 0; i < oq.size(); i++) {
|
|
// std::cerr << printSimple(oq[i]) << " ";
|
|
//}
|
|
//std::cerr << " <-\n";
|
|
}
|
|
// Output must have one argument
|
|
if (oq.size() == 0) {
|
|
err("Output blank", Metadata());
|
|
}
|
|
else if (oq.size() > 1) {
|
|
err("Multiple expressions or unclosed bracket", oq[1].metadata);
|
|
}
|
|
|
|
return oq[0];
|
|
}
|
|
|
|
|
|
// Parses one line of serpent
|
|
Node parseSerpentTokenStream(std::vector<Node> s) {
|
|
return treefy(shuntingYard(s));
|
|
}
|
|
|
|
|
|
// Count spaces at beginning of line
|
|
int spaceCount(std::string s) {
|
|
unsigned pos = 0;
|
|
while (pos < s.length() && (s[pos] == ' ' || s[pos] == '\t'))
|
|
pos++;
|
|
return pos;
|
|
}
|
|
|
|
// Is this a command that takes an argument on the same line?
|
|
bool bodied(std::string tok) {
|
|
return tok == "if" || tok == "elif" || tok == "while"
|
|
|| tok == "with" || tok == "def";
|
|
}
|
|
|
|
// Is this a command that takes an argument as a child block?
|
|
bool childBlocked(std::string tok) {
|
|
return tok == "if" || tok == "elif" || tok == "else"
|
|
|| tok == "code" || tok == "shared" || tok == "init"
|
|
|| tok == "while" || tok == "repeat" || tok == "for"
|
|
|| tok == "with" || tok == "def";
|
|
}
|
|
|
|
// Are the two commands meant to continue each other?
|
|
bool bodiedContinued(std::string prev, std::string tok) {
|
|
return (prev == "if" && tok == "elif")
|
|
|| (prev == "elif" && tok == "else")
|
|
|| (prev == "elif" && tok == "elif")
|
|
|| (prev == "if" && tok == "else")
|
|
|| (prev == "init" && tok == "code")
|
|
|| (prev == "shared" && tok == "code")
|
|
|| (prev == "shared" && tok == "init");
|
|
}
|
|
|
|
// Is a line of code empty?
|
|
bool isLineEmpty(std::string line) {
|
|
std::vector<Node> tokens = tokenize(line);
|
|
if (!tokens.size() || tokens[0].val == "#" || tokens[0].val == "//")
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
// Parse lines of serpent (helper function)
|
|
Node parseLines(std::vector<std::string> lines, Metadata metadata, int sp) {
|
|
std::vector<Node> o;
|
|
int origLine = metadata.ln;
|
|
unsigned i = 0;
|
|
while (i < lines.size()) {
|
|
metadata.ln = origLine + i;
|
|
std::string main = lines[i];
|
|
if (isLineEmpty(main)) {
|
|
i += 1;
|
|
continue;
|
|
}
|
|
int spaces = spaceCount(main);
|
|
if (spaces != sp) {
|
|
err("Indent mismatch", metadata);
|
|
}
|
|
// Tokenize current line
|
|
std::vector<Node> tokens = tokenize(main.substr(sp), metadata);
|
|
// Remove extraneous tokens, including if / elif
|
|
std::vector<Node> tokens2;
|
|
for (unsigned j = 0; j < tokens.size(); j++) {
|
|
if (tokens[j].val == "#" || tokens[j].val == "//") break;
|
|
if (j >= 1 || !bodied(tokens[j].val)) {
|
|
tokens2.push_back(tokens[j]);
|
|
}
|
|
}
|
|
if (tokens2.size() > 0 && tokens2.back().val == ":")
|
|
tokens2.pop_back();
|
|
// Parse current line
|
|
Node out = parseSerpentTokenStream(tokens2);
|
|
// Parse child block
|
|
int childIndent = 999999;
|
|
std::vector<std::string> childBlock;
|
|
while (1) {
|
|
i++;
|
|
if (i >= lines.size())
|
|
break;
|
|
bool ile = isLineEmpty(lines[i]);
|
|
if (!ile) {
|
|
int spaces = spaceCount(lines[i]);
|
|
if (spaces <= sp) break;
|
|
childBlock.push_back(lines[i]);
|
|
if (spaces < childIndent) childIndent = spaces;
|
|
}
|
|
else childBlock.push_back("");
|
|
}
|
|
// Child block empty?
|
|
bool cbe = true;
|
|
for (unsigned i = 0; i < childBlock.size(); i++) {
|
|
if (childBlock[i].length() > 0) { cbe = false; break; }
|
|
}
|
|
// Bring back if / elif into AST
|
|
if (bodied(tokens[0].val)) {
|
|
std::vector<Node> args;
|
|
args.push_back(out);
|
|
out = astnode(tokens[0].val, args, out.metadata);
|
|
}
|
|
// Add child block to AST
|
|
if (childBlocked(tokens[0].val)) {
|
|
if (cbe)
|
|
err("Expected indented child block!", out.metadata);
|
|
out.type = ASTNODE;
|
|
metadata.ln += 1;
|
|
out.args.push_back(parseLines(childBlock, metadata, childIndent));
|
|
metadata.ln -= 1;
|
|
}
|
|
else if (!cbe)
|
|
err("Did not expect indented child block!", out.metadata);
|
|
if (o.size() == 0 || o.back().type == TOKEN) {
|
|
o.push_back(out);
|
|
continue;
|
|
}
|
|
// This is a little complicated. Basically, the idea here is to build
|
|
// constructions like [if [< x 5] [a] [elif [< x 10] [b] [else [c]]]]
|
|
std::vector<Node> u;
|
|
u.push_back(o.back());
|
|
if (bodiedContinued(o.back().val, out.val)) {
|
|
while (1) {
|
|
if (!bodiedContinued(u.back().val, out.val)) {
|
|
u.pop_back();
|
|
break;
|
|
}
|
|
if (!u.back().args.size()
|
|
|| !bodiedContinued(u.back().val, u.back().args.back().val)) {
|
|
break;
|
|
}
|
|
u.push_back(u.back().args.back());
|
|
}
|
|
u.back().args.push_back(out);
|
|
while (u.size() > 1) {
|
|
Node v = u.back();
|
|
u.pop_back();
|
|
u.back().args.pop_back();
|
|
u.back().args.push_back(v);
|
|
}
|
|
o.pop_back();
|
|
o.push_back(u[0]);
|
|
}
|
|
else o.push_back(out);
|
|
}
|
|
if (o.size() == 1)
|
|
return o[0];
|
|
else if (o.size())
|
|
return astnode("seq", o, o[0].metadata);
|
|
else
|
|
return astnode("seq", o, Metadata());
|
|
}
|
|
|
|
// Parses serpent code
|
|
Node parseSerpent(std::string s) {
|
|
std::string input = s;
|
|
std::string file = "main";
|
|
if (exists(s)) {
|
|
file = s;
|
|
input = get_file_contents(s);
|
|
}
|
|
return parseLines(splitLines(input), Metadata(file, 0, 0), 0);
|
|
}
|
|
|
|
|
|
using namespace std;
|
|
|