123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- #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;
- }
- }
|