2 Star 6 Fork 0

秩启/Lox++

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
该仓库未声明开源许可证文件(LICENSE),使用请关注具体项目描述及其代码上游依赖。
克隆/下载
Resolver.h 8.24 KB
一键复制 编辑 原始数据 按行查看 历史
秩启 提交于 2022-05-07 15:56 . Support "super" error handling
//
// Created by Gorun on 2022/5/4.
//
#ifndef LOXPP_RESOLVER_H
#define LOXPP_RESOLVER_H
#include <any>
#include <memory>
#include <vector>
#include <map>
#include "Error.h"
#include "Expr.h"
#include "Stmt.h"
#include "Interpreter.h"
using namespace std;
class Resolver: public ExprVisitor, public StmtVisitor {
enum class FunctionType {
NONE,
FUNCTION,
INITIALIZER,
METHOD
};
enum class ClassType {
NONE,
CLASS,
SUBCLASS
};
Interpreter& interpreter;
vector<map<string, bool>> scopes;
FunctionType currentFunction = FunctionType::NONE;
ClassType currentClass = ClassType::NONE;
public:
explicit Resolver(Interpreter& interpreter)
:interpreter{interpreter}
{}
//These are key functions to resolving:
void resolve(const vector<shared_ptr<Stmt>>& statements) {
for (const shared_ptr<Stmt>& statement: statements) {
resolve(statement);
}
}
void resolve(shared_ptr<Stmt> stmt) {
stmt->accept(*this);
}
void resolve(shared_ptr<Expr> expr) {
expr->accept(*this);
}
//This produce a new scope
any visitBlockStmt(shared_ptr<Block> stmt) override {
beginScope();
resolve(stmt->statements);
endScope();
return {};
}
//This produce a new variable name
any visitVarStmt(shared_ptr<Var> stmt) override {
declare(stmt->name);
if (stmt->initializer!=nullptr) {
resolve(stmt->initializer);
}
define(stmt->name);
return {};
}
//This references to a variable name, so it needs to be resolved
any visitVariableExpr(shared_ptr<Variable> expr) override {
if (!scopes.empty()) {
map<string, bool>& scope = scopes.back();
auto element = scope.find(expr->name.lexeme);
if (element!=scope.end() && !element->second) {
error(expr->name, "Can't read local variable in its own initializer");
}
}
resolveLocal(expr, expr->name);
return {};
}
//This references to a variable name, so it needs to be resolved, too
any visitAssignExpr(shared_ptr<Assign> expr) override {
resolve(expr->value);
resolveLocal(expr, expr->name);
return {};
}
//This produce a new function name and a list of parameters to resolve
any visitFunctionStmt(shared_ptr<Function> stmt) override {
declare(stmt->name);
define(stmt->name);
resolveFunction(stmt, FunctionType::FUNCTION);
return {};
}
//Fairly useless functions below:
//Checking every place that can contain expressions/statements:
//Resolving statements:
any visitExpressionStmt(shared_ptr<Expression> stmt) override {
resolve(stmt->expression);
return {};
}
any visitIfStmt(shared_ptr<If> stmt) override {
resolve(stmt->condition);
resolve(stmt->thenBranch);
if (stmt->elseBranch!=nullptr)
resolve(stmt->elseBranch);
return {};
}
any visitPrintStmt(shared_ptr<Print> stmt) override {
resolve(stmt->expression);
return {};
}
any visitReturnStmt(shared_ptr<Return> stmt) override {
if (currentFunction == FunctionType::NONE) {
error(stmt->keyword, "Can't return from top-level code.");
}
if (stmt->value!=nullptr) {
if (currentFunction==FunctionType::INITIALIZER) {
error(stmt->keyword, "Can't return a value from an initializer.");
}
resolve(stmt->value);
}
return {};
}
any visitWhileStmt(shared_ptr<While> stmt) override {
resolve(stmt->condition);
resolve(stmt->body);
return {};
}
any visitInputStmt(shared_ptr<Input> stmt) override {
//Input statement must have a variable on the right
return {};
}
any visitClassStmt(shared_ptr<Class> stmt) override {
ClassType enclosingClass = currentClass;
currentClass = ClassType::CLASS;
declare(stmt->name);
define(stmt->name);
if (stmt->superclass!=nullptr&&stmt->name.lexeme==stmt->superclass->name.lexeme) {
error(stmt->superclass->name, "A class can't inherit from itself.");
}
if (stmt->superclass!=nullptr) {
currentClass=ClassType::SUBCLASS;
resolve(stmt->superclass);
}
if (stmt->superclass!=nullptr) {
beginScope();
scopes.back()["super"]=true;
}
beginScope();
scopes.back()["this"]=true;
for (shared_ptr<Function> method: stmt->methods) {
FunctionType declaration=FunctionType::METHOD;
if (method->name.lexeme=="init") {
declaration = FunctionType::INITIALIZER;
}
resolveFunction(method, declaration);
}
endScope();
if (stmt->superclass!=nullptr)
endScope();
currentClass = enclosingClass;
return {};
}
//Resolving expressions:
any visitBinaryExpr(shared_ptr<Binary> expr) override {
resolve(expr->left);
resolve(expr->right);
return {};
}
any visitCallExpr(shared_ptr<Call> expr) override {
resolve(expr->callee);
for (const auto& argument: expr->arguments) {
resolve(argument);
}
return {};
}
any visitGroupingExpr(shared_ptr<Grouping> expr) override {
resolve(expr->expression);
return {};
}
any visitLiteralExpr(shared_ptr<Literal> expr) override {
return {};
}
any visitLogicalExpr(shared_ptr<Logical> expr) override {
resolve(expr->left);
resolve(expr->right);
return {};
}
any visitUnaryExpr(shared_ptr<Unary> expr) override {
resolve(expr->right);
return {};
}
any visitTernaryExpr(shared_ptr<Ternary> expr) override {
resolve(expr->condition);
resolve(expr->elseBranch);
resolve(expr->thenBranch);
return {};
}
any visitGetExpr(shared_ptr<Get> expr) override {
resolve(expr->object);
return {};
}
any visitSetExpr(shared_ptr<Set> expr) override {
resolve(expr->value);
resolve(expr->object);
return {};
}
any visitThisExpr(shared_ptr<This> expr) override {
if (currentClass==ClassType::NONE) {
error(expr->keyword, "Can't use 'this' outside of a class.");
return {};
}
resolveLocal(expr, expr->keyword);
return {};
}
any visitSuperExpr(shared_ptr<Super> expr) override {
if (currentClass==ClassType::NONE) {
error(expr->keyword, "Can't use 'super' outside of a class.");
} else if (currentClass!=ClassType::SUBCLASS) {
error(expr->keyword, "Can't use 'super' in a class with no superclass.");
}
resolveLocal(expr, expr->keyword);
return {};
}
//Tool functions below:
void beginScope() {
scopes.emplace_back();
}
void endScope() {
scopes.pop_back();
}
void declare(const Token& name) {
if (scopes.empty()) return;
map<string, bool>& scope = scopes.back();
if (scope.find(name.lexeme)!=scope.end()) {
error(name, "Already a variable with this name in this scope.");
}
scopes.back()[name.lexeme]=false;
}
void define(const Token& name) {
if (scopes.empty()) return;
scopes.back()[name.lexeme]=true;
}
void resolveLocal(shared_ptr<Expr> expr, const Token& name) {
for (int i=scopes.size()-1; i>=0; i--) {
if (scopes[i].find(name.lexeme)!=scopes[i].end()) {
interpreter.resolve(expr, scopes.size()-1-i);
return;
}
}
}
void resolveFunction(shared_ptr<Function> function, FunctionType type) {
FunctionType enclosingfunction = currentFunction;
currentFunction = type;
beginScope();
for (const Token& param: function->params) {
declare(param);
define(param);
}
resolve(function->body);
endScope();
currentFunction = enclosingfunction;
}
};
#endif //LOXPP_RESOLVER_H
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
C++
1
https://gitee.com/rankrev/loxpp.git
git@gitee.com:rankrev/loxpp.git
rankrev
loxpp
Lox++
master

搜索帮助