2024-12-24 17:46:47 +01:00

252 lines
6.8 KiB
C++

#pragma once
#include <any>
#include <boost/json/serialize.hpp>
#include <exception>
#include <iostream>
#include <memory>
#include <sstream>
#include <stdexcept>
#include <string>
#include <utility>
#include <boost/json.hpp>
#include "Terminal.h"
#define LOC_ARG const SourceRange &sourceRange
namespace plsm {
namespace ast {
class BinExpr;
class CallExpr;
class CastExpr;
class Identifier;
class LambdaExpr;
class UnExpr;
class IntValue;
class FloatValue;
class Import;
class Module;
class AssignStmt;
class Block;
class ExprStmt;
class FnParam;
class FnDecl;
class IfStmt;
class RetStmt;
class VarDecl;
class WhileStmt;
class NamedTypeName;
class ASTVisitor {
public:
virtual ~ASTVisitor() = default;
virtual std::any visit(BinExpr &binExpr, std::any param) = 0;
virtual std::any visit(CallExpr &callExpr, std::any param) = 0;
virtual std::any visit(CastExpr &castExpr, std::any param) = 0;
virtual std::any visit(Identifier &identifier, std::any param) = 0;
virtual std::any visit(LambdaExpr &lambdaExpr, std::any param) = 0;
virtual std::any visit(UnExpr &unExpr, std::any param) = 0;
virtual std::any visit(IntValue &intValue, std::any param) = 0;
virtual std::any visit(FloatValue &floatValue, std::any param) = 0;
virtual std::any visit(Import &import, std::any param) = 0;
virtual std::any visit(Module &module, std::any param) = 0;
virtual std::any visit(AssignStmt &assignStmt, std::any param) = 0;
virtual std::any visit(Block &block, std::any param) = 0;
virtual std::any visit(ExprStmt &exprStmt, std::any param) = 0;
virtual std::any visit(FnParam &fnParam, std::any param) = 0;
virtual std::any visit(FnDecl &fnDecl, std::any param) = 0;
virtual std::any visit(IfStmt &ifStmt, std::any param) = 0;
virtual std::any visit(InlineAsmConstraint &inlineAsmConstraint,
std::any param) = 0;
virtual std::any visit(InlineAsm &inlineAsm, std::any param) = 0;
virtual std::any visit(RetStmt &retStmt, std::any param) = 0;
virtual std::any visit(VarDecl &varDecl, std::any param) = 0;
virtual std::any visit(WhileStmt &whileStmt, std::any param) = 0;
virtual std::any visit(NamedTypeName &namedTypeName, std::any param) = 0;
};
} // namespace ast
} // namespace plsm
namespace plsm {
namespace ast {
class Jsonable {
public:
virtual ~Jsonable() = default;
virtual boost::json::value toJson() const = 0;
protected:
template <class CurrNode>
static inline boost::json::value getJsonProperty(boost::json::value json,
std::string property) {
boost::json::value prop;
try {
if (!json.as_object().contains(property))
throw std::exception();
prop = json.as_object().at(property);
} catch (...) {
std::cout << boost::json::serialize(json) << std::endl;
throw std::runtime_error("missing property '" + property + "' in " +
typeid(CurrNode).name() + "::fromJson");
}
return prop;
}
template <class CurrNode, typename T>
static inline T getJsonValue(boost::json::value json, std::string property) {
auto prop = getJsonProperty<CurrNode>(json, property);
try {
return boost::json::value_to<T>(prop);
} catch (...) {
throw std::runtime_error("invalid value for property '" + property +
"' in " + typeid(CurrNode).name() +
"::fromJson");
}
}
template <class CurrNode, class SubNode>
static inline auto fromJsonProperty(boost::json::value json,
std::string property) {
return std::unique_ptr<SubNode>(
SubNode::fromJson(getJsonProperty<CurrNode>(json, property)));
}
template <class CurrNode, class SubNode>
static inline auto fromJsonVector(boost::json::value json,
std::string property) {
auto arr = getJsonProperty<CurrNode>(json, property).as_array();
std::vector<std::unique_ptr<SubNode>> result;
for (auto &el : arr) {
result.push_back(std::unique_ptr<SubNode>(SubNode::fromJson(el)));
}
return result;
}
};
class SourceRange {
public:
SourceRange(const std::string &file, std::string text,
std::pair<size_t, size_t> start, std::pair<size_t, size_t> end)
: file(file), text(text), start(start), end(end) {}
const std::string file, text;
const std::pair<size_t, size_t> start, end;
static SourceRange unknown() {
return SourceRange("<?>", "<?>", {0, 0}, {0, 0});
};
static SourceRange json() {
return SourceRange("<json>", "<json>", {0, 0}, {0, 0});
};
};
class TypeName;
class Type : public Jsonable {
public:
Type() : Jsonable() {}
virtual ~Type() = default;
virtual TypeName *toTypeName() = 0;
virtual bool operator==(const Type &other) = 0;
virtual bool operator!=(const Type &other) { return !(*this == other); }
static Type *fromJson(boost::json::value json);
};
class Symbol {
public:
const std::string name;
std::shared_ptr<Type> type;
Symbol(const std::string &name) : name(name) {}
Symbol(const std::string &name, Type *type) : name(name), type(type) {}
};
class ASTNode : public Jsonable {
public:
ASTNode(LOC_ARG) : Jsonable(), sourceRange(sourceRange) {}
virtual ~ASTNode() = default;
const SourceRange sourceRange;
virtual std::string toJsonString() const {
return boost::json::serialize(toJson(), {});
}
static ASTNode *fromJson(boost::json::value json);
virtual bool isExpr() const { return false; }
virtual bool isStmt() const { return false; }
virtual bool isTypeName() const { return false; }
virtual std::string error(const std::string &message) const {
std::stringstream ss;
ss << "In file " << sourceRange.file << ":" << sourceRange.start.first
<< ":" << sourceRange.start.second + 1 << "\n"
<< terminal::cyan << sourceRange.text << terminal::reset << "\n"
<< terminal::red << message << terminal::reset;
return ss.str();
}
virtual std::any accept(ASTVisitor *visitor, std::any param) = 0;
};
class Expr : public ASTNode {
public:
std::shared_ptr<Type> type;
Expr(LOC_ARG) : ASTNode(sourceRange) {}
virtual ~Expr() = default;
static Expr *fromJson(boost::json::value json);
virtual bool isExpr() const override { return true; }
};
class Stmt : public ASTNode {
public:
Stmt(LOC_ARG) : ASTNode(sourceRange) {}
virtual ~Stmt() = default;
static Stmt *fromJson(boost::json::value json);
virtual bool alywasReturns() const = 0;
virtual bool isStmt() const override { return true; }
};
class TypeName : public ASTNode {
public:
std::shared_ptr<Type> type;
TypeName(LOC_ARG) : ASTNode(sourceRange) {}
virtual ~TypeName() = default;
static TypeName *fromJson(boost::json::value json);
virtual bool isTypeName() const override { return true; }
};
} // namespace ast
} // namespace plsm