#include "parser/parser.hpp" #include namespace hex::lang { Parser::Parser() { } using TokenIter = std::vector::const_iterator; std::vector parseTillToken(TokenIter &curr, Token::Type endTokenType); bool tryConsume(TokenIter &curr, std::initializer_list tokenTypes) { std::vector::const_iterator originalPosition = curr; for (const auto& type : tokenTypes) { if (curr->type != type) { curr = originalPosition; return false; } curr++; } return true; } ASTNode* parseBuiltinVariableDecl(TokenIter &curr) { return new ASTNodeVariableDecl(curr[-3].typeToken.type, curr[-2].identifierToken.identifier); } ASTNode* parseCustomTypeVariableDecl(TokenIter &curr) { return new ASTNodeVariableDecl(Token::TypeToken::Type::CustomType, curr[-2].identifierToken.identifier, curr[-3].identifierToken.identifier); } std::optional parseStruct(TokenIter &curr) { const std::string &structName = curr[-2].identifierToken.identifier; std::vector nodes; std::optional offset = { }; while (!tryConsume(curr, {Token::Type::ScopeClose})) { if (tryConsume(curr, {Token::Type::Type, Token::Type::Identifier, Token::Type::EndOfExpression})) nodes.push_back(parseBuiltinVariableDecl(curr)); else if (tryConsume(curr, {Token::Type::Identifier, Token::Type::Identifier, Token::Type::EndOfExpression})) nodes.push_back(parseCustomTypeVariableDecl(curr)); else break; } if (tryConsume(curr, {Token::Type::Operator})) { if (curr[-1].operatorToken.op == Token::OperatorToken::Operator::AtDeclaration) { if (tryConsume(curr, {Token::Type::Integer})) { offset = curr[-1].integerToken.integer; } else { for(auto &node : nodes) delete node; return { }; } } else { for(auto &node : nodes) delete node; return { }; } } if (!tryConsume(curr, {Token::Type::EndOfExpression})) { for(auto &node : nodes) delete node; return { }; } return new ASTNodeStruct(structName, nodes, offset); } ASTNode *parseScope(TokenIter &curr) { return new ASTNodeScope(parseTillToken(curr, Token::Type::ScopeClose)); } std::optional parseUsingDeclaration(TokenIter &curr) { auto keyword = curr[-5].keywordToken; auto name = curr[-4].identifierToken; auto op = curr[-3].operatorToken; if (keyword.keyword != Token::KeywordToken::Keyword::Using) return { }; if (op.op != Token::OperatorToken::Operator::Assignment) return { }; if (curr[-2].type == Token::Type::Type) { auto type = curr[-2].typeToken; return new ASTNodeTypeDecl(type.type, name.identifier); } else if (curr[-2].type == Token::Type::Identifier) { auto customType = curr[-2].identifierToken; return new ASTNodeTypeDecl(Token::TypeToken::Type::CustomType, name.identifier, customType.identifier); } return { }; } std::optional> parseStatement(TokenIter &curr) { std::vector program; if (tryConsume(curr, { Token::Type::Keyword, Token::Type::Identifier, Token::Type::ScopeOpen })) { if (curr[-3].keywordToken.keyword == Token::KeywordToken::Keyword::Struct) { auto structAst = parseStruct(curr); if (!structAst.has_value()) { for(auto &node : program) delete node; return { }; } program.push_back(structAst.value()); } return program; } else if (tryConsume(curr, { Token::Type::ScopeOpen })) { program.push_back(parseScope(curr)); return program; } else if (tryConsume(curr, { Token::Type::Keyword, Token::Type::Identifier, Token::Type::Operator, Token::Type::Type, Token::Type::EndOfExpression})) { auto usingDecl = parseUsingDeclaration(curr); if (!usingDecl.has_value()) { for(auto &node : program) delete node; return { }; } program.push_back(usingDecl.value()); return program; } else if (tryConsume(curr, { Token::Type::Keyword, Token::Type::Identifier, Token::Type::Operator, Token::Type::Identifier, Token::Type::EndOfExpression})) { auto usingDecl = parseUsingDeclaration(curr); if (!usingDecl.has_value()) { for(auto &node : program) delete node; return { }; } program.push_back(usingDecl.value()); return program; } else { for(auto &node : program) delete node; return { }; } } std::vector parseTillToken(TokenIter &curr, Token::Type endTokenType) { std::vector program; while (curr->type != endTokenType) { auto newTokens = parseStatement(curr); if (!newTokens.has_value()) break; program.insert(program.end(), newTokens->begin(), newTokens->end()); } curr++; return program; } std::pair> Parser::parse(const std::vector &tokens) { auto currentToken = tokens.begin(); auto program = parseTillToken(currentToken, Token::Type::EndOfProgram); if (program.empty()) return { ResultParseError, { } }; return { ResultSuccess, program }; } }