#pragma once namespace nall::Eval { inline auto whitespace(char n) -> bool { return n == ' ' || n == '\t' || n == '\r' || n == '\n'; } inline auto parse(Node*& node, const char*& s, uint depth) -> void { auto unaryPrefix = [&](Node::Type type, uint seek, uint depth) { auto parent = new Node(type); parse(parent->link(0) = new Node, s += seek, depth); node = parent; }; auto unarySuffix = [&](Node::Type type, uint seek, uint depth) { auto parent = new Node(type); parent->link(0) = node; parse(parent, s += seek, depth); node = parent; }; auto binary = [&](Node::Type type, uint seek, uint depth) { auto parent = new Node(type); parent->link(0) = node; parse(parent->link(1) = new Node, s += seek, depth); node = parent; }; auto ternary = [&](Node::Type type, uint seek, uint depth) { auto parent = new Node(type); parent->link(0) = node; parse(parent->link(1) = new Node, s += seek, depth); if(s[0] != ':') throw "mismatched ternary"; parse(parent->link(2) = new Node, s += seek, depth); node = parent; }; auto separator = [&](Node::Type type, uint seek, uint depth) { if(node->type != Node::Type::Separator) return binary(type, seek, depth); uint n = node->link.size(); parse(node->link(n) = new Node, s += seek, depth); }; while(whitespace(s[0])) s++; if(!s[0]) return; if(s[0] == '(' && !node->link) { parse(node, s += 1, 1); if(*s++ != ')') throw "mismatched group"; } if(isLiteral(s)) { node->type = Node::Type::Literal; node->literal = literal(s); } #define p() (!node->literal && !node->link) while(true) { while(whitespace(s[0])) s++; if(!s[0]) return; if(depth >= 13) break; if(s[0] == '(' && !p()) { binary(Node::Type::Function, 1, 1); if(*s++ != ')') throw "mismatched function"; continue; } if(s[0] == '[') { binary(Node::Type::Subscript, 1, 1); if(*s++ != ']') throw "mismatched subscript"; continue; } if(s[0] == '.') { binary(Node::Type::Member, 1, 13); continue; } if(s[0] == '+' && s[1] == '+' && !p()) { unarySuffix(Node::Type::SuffixIncrement, 2, 13); continue; } if(s[0] == '-' && s[1] == '-' && !p()) { unarySuffix(Node::Type::SuffixDecrement, 2, 13); continue; } if(s[0] == '&' && p()) { unaryPrefix(Node::Type::Reference, 1, 12); continue; } if(s[0] == '*' && p()) { unaryPrefix(Node::Type::Dereference, 1, 12); continue; } if(s[0] == '!' && p()) { unaryPrefix(Node::Type::LogicalNot, 1, 12); continue; } if(s[0] == '~' && p()) { unaryPrefix(Node::Type::BitwiseNot, 1, 12); continue; } if(s[0] == '+' && s[1] != '+' && p()) { unaryPrefix(Node::Type::Positive, 1, 12); continue; } if(s[0] == '-' && s[1] != '-' && p()) { unaryPrefix(Node::Type::Negative, 1, 12); continue; } if(s[0] == '+' && s[1] == '+' && p()) { unaryPrefix(Node::Type::PrefixIncrement, 2, 12); continue; } if(s[0] == '-' && s[1] == '-' && p()) { unaryPrefix(Node::Type::PrefixDecrement, 2, 12); continue; } if(depth >= 12) break; if(depth >= 11) break; if(s[0] == '*' && s[1] != '=') { binary(Node::Type::Multiply, 1, 11); continue; } if(s[0] == '/' && s[1] != '=') { binary(Node::Type::Divide, 1, 11); continue; } if(s[0] == '%' && s[1] != '=') { binary(Node::Type::Modulo, 1, 11); continue; } if(depth >= 10) break; if(s[0] == '+' && s[1] != '=') { binary(Node::Type::Add, 1, 10); continue; } if(s[0] == '-' && s[1] != '=') { binary(Node::Type::Subtract, 1, 10); continue; } if(depth >= 9) break; if(s[0] == '<' && s[1] == '<' && s[2] == '<' && s[3] != '=') { binary(Node::Type::RotateLeft, 3, 9); continue; } if(s[0] == '>' && s[1] == '>' && s[2] == '>' && s[3] != '=') { binary(Node::Type::RotateRight, 3, 9); continue; } if(s[0] == '<' && s[1] == '<' && s[2] != '=') { binary(Node::Type::ShiftLeft, 2, 9); continue; } if(s[0] == '>' && s[1] == '>' && s[2] != '=') { binary(Node::Type::ShiftRight, 2, 9); continue; } if(depth >= 8) break; if(s[0] == '&' && s[1] != '&' && s[1] != '=') { binary(Node::Type::BitwiseAnd, 1, 8); continue; } if(s[0] == '|' && s[1] != '|' && s[1] != '=') { binary(Node::Type::BitwiseOr, 1, 8); continue; } if(s[0] == '^' && s[1] != '^' && s[1] != '=') { binary(Node::Type::BitwiseXor, 1, 8); continue; } if(depth >= 7) break; if(s[0] == '~' && s[1] != '=') { binary(Node::Type::Concatenate, 1, 7); continue; } if(depth >= 6) break; if(s[0] == '=' && s[1] == '=') { binary(Node::Type::Equal, 2, 6); continue; } if(s[0] == '!' && s[1] == '=') { binary(Node::Type::NotEqual, 2, 6); continue; } if(s[0] == '<' && s[1] == '=') { binary(Node::Type::LessThanEqual, 2, 6); continue; } if(s[0] == '>' && s[1] == '=') { binary(Node::Type::GreaterThanEqual, 2, 6); continue; } if(s[0] == '<') { binary(Node::Type::LessThan, 1, 6); continue; } if(s[0] == '>') { binary(Node::Type::GreaterThan, 1, 6); continue; } if(depth >= 5) break; if(s[0] == '&' && s[1] == '&') { binary(Node::Type::LogicalAnd, 2, 5); continue; } if(s[0] == '|' && s[1] == '|') { binary(Node::Type::LogicalOr, 2, 5); continue; } if(s[0] == '?' && s[1] == '?') { binary(Node::Type::Coalesce, 2, 4); continue; } if(s[0] == '?' && s[1] != '?') { ternary(Node::Type::Condition, 1, 4); continue; } if(depth >= 4) break; if(s[0] == '=') { binary(Node::Type::Assign, 1, 3); continue; } if(s[0] == ':' && s[1] == '=') { binary(Node::Type::Create, 2, 3); continue; } if(s[0] == '*' && s[1] == '=') { binary(Node::Type::AssignMultiply, 2, 3); continue; } if(s[0] == '/' && s[1] == '=') { binary(Node::Type::AssignDivide, 2, 3); continue; } if(s[0] == '%' && s[1] == '=') { binary(Node::Type::AssignModulo, 2, 3); continue; } if(s[0] == '+' && s[1] == '=') { binary(Node::Type::AssignAdd, 2, 3); continue; } if(s[0] == '-' && s[1] == '=') { binary(Node::Type::AssignSubtract, 2, 3); continue; } if(s[0] == '<' && s[1] == '<' && s[2] == '<' && s[3] == '=') { binary(Node::Type::AssignRotateLeft, 4, 3); continue; } if(s[0] == '>' && s[1] == '>' && s[2] == '>' && s[3] == '=') { binary(Node::Type::AssignRotateRight, 4, 3); continue; } if(s[0] == '<' && s[1] == '<' && s[2] == '=') { binary(Node::Type::AssignShiftLeft, 3, 3); continue; } if(s[0] == '>' && s[1] == '>' && s[2] == '=') { binary(Node::Type::AssignShiftRight, 3, 3); continue; } if(s[0] == '&' && s[1] == '=') { binary(Node::Type::AssignBitwiseAnd, 2, 3); continue; } if(s[0] == '|' && s[1] == '=') { binary(Node::Type::AssignBitwiseOr, 2, 3); continue; } if(s[0] == '^' && s[1] == '=') { binary(Node::Type::AssignBitwiseXor, 2, 3); continue; } if(s[0] == '~' && s[1] == '=') { binary(Node::Type::AssignConcatenate, 2, 3); continue; } if(depth >= 3) break; if(depth >= 2) break; if(s[0] == ',') { separator(Node::Type::Separator, 1, 2); continue; } if(depth >= 1 && (s[0] == ')' || s[0] == ']')) break; while(whitespace(s[0])) s++; if(!s[0]) break; throw "unrecognized terminal"; } #undef p } inline auto parse(const string& expression) -> Node* { auto result = new Node; const char* p = expression; parse(result, p, 0); return result; } }