mirror of
https://codeberg.org/ProgramSnail/lang_2023.git
synced 2025-12-05 22:48:42 +00:00
typeclass tree start
This commit is contained in:
parent
c1dec6a0d1
commit
173d50672a
15 changed files with 1475 additions and 35 deletions
|
|
@ -20,6 +20,10 @@ add_executable(lang_interpreter src/main.cpp
|
||||||
src/visitor.cpp
|
src/visitor.cpp
|
||||||
src/build_visitor.cpp
|
src/build_visitor.cpp
|
||||||
src/print_visitor.cpp
|
src/print_visitor.cpp
|
||||||
|
src/find_symbols_visitor.cpp
|
||||||
|
src/link_symbols_visitor.cpp
|
||||||
|
src/type_check_visitor.cpp
|
||||||
|
src/typed_print_visitor.cpp
|
||||||
lang-parser/src/parser.c
|
lang-parser/src/parser.c
|
||||||
lang-parser/src/tree_sitter/parser.h
|
lang-parser/src/tree_sitter/parser.h
|
||||||
tree-sitter/lib/src/lib.c)
|
tree-sitter/lib/src/lib.c)
|
||||||
|
|
|
||||||
|
|
@ -15,6 +15,7 @@
|
||||||
|
|
||||||
namespace info {
|
namespace info {
|
||||||
|
|
||||||
|
// TODO: fix dfs, etc.
|
||||||
// TODO: optimize recalc
|
// TODO: optimize recalc
|
||||||
|
|
||||||
class TypeGraph {
|
class TypeGraph {
|
||||||
|
|
@ -69,6 +70,7 @@ public:
|
||||||
|
|
||||||
size_t cluster_id = verticles_[id].cluster.value();
|
size_t cluster_id = verticles_[id].cluster.value();
|
||||||
|
|
||||||
|
typeclasses_[id].methods[method.first].definition = method.second.definition;
|
||||||
std::vector<utils::IdType> typeclasses;
|
std::vector<utils::IdType> typeclasses;
|
||||||
|
|
||||||
typeclasses.reserve(cluster_requirements_[cluster_id].typeclasses.size());
|
typeclasses.reserve(cluster_requirements_[cluster_id].typeclasses.size());
|
||||||
|
|
@ -174,20 +176,20 @@ private:
|
||||||
std::vector<std::vector<size_t>> FindClusters() {
|
std::vector<std::vector<size_t>> FindClusters() {
|
||||||
std::vector<std::vector<size_t>> clusters;
|
std::vector<std::vector<size_t>> clusters;
|
||||||
|
|
||||||
auto sorted_verticles = TopSort();
|
auto sorted_verticles = BackTopSort();
|
||||||
|
|
||||||
std::vector<size_t> marks(sorted_verticles.size(), 0);
|
std::vector<size_t> marks(sorted_verticles.size(), 0);
|
||||||
for (size_t i = 0; i < sorted_verticles.size(); ++i) {
|
for (size_t i = 0; i < sorted_verticles.size(); ++i) {
|
||||||
if (marks[i] == 0) {
|
if (marks[i] == 0) {
|
||||||
clusters.emplace_back();
|
clusters.emplace_back();
|
||||||
VisitDfs(i, clusters[i], marks, back_edges_, clusters.size());
|
BackVisitDfs(i, clusters[i], marks, back_edges_, clusters.size());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return clusters;
|
return clusters;
|
||||||
}
|
}
|
||||||
|
|
||||||
void VisitDfs(size_t id,
|
void BackVisitDfs(size_t id,
|
||||||
std::vector<size_t>& verticles,
|
std::vector<size_t>& verticles,
|
||||||
std::vector<size_t>& marks,
|
std::vector<size_t>& marks,
|
||||||
const std::vector<std::vector<size_t>>& edges,
|
const std::vector<std::vector<size_t>>& edges,
|
||||||
|
|
@ -197,19 +199,20 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
marks[id] = mark;
|
marks[id] = mark;
|
||||||
verticles.push_back(id);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < edges[id].size(); ++i) {
|
for (size_t i = 0; i < edges[id].size(); ++i) {
|
||||||
VisitDfs(id, verticles, marks, edges, mark);
|
BackVisitDfs(id, verticles, marks, edges, mark);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<size_t> TopSort() {
|
verticles.push_back(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<size_t> BackTopSort() {
|
||||||
std::vector<size_t> sorted_verticles;
|
std::vector<size_t> sorted_verticles;
|
||||||
std::vector<size_t> marks(verticles_.size(), 0);
|
std::vector<size_t> marks(verticles_.size(), 0);
|
||||||
|
|
||||||
for (size_t i = 0; i < marks.size(); ++i) {
|
for (size_t i = 0; i < marks.size(); ++i) {
|
||||||
VisitDfs(i, sorted_verticles, marks, edges_, 1);
|
BackVisitDfs(i, sorted_verticles, marks, edges_, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
return sorted_verticles;
|
return sorted_verticles;
|
||||||
|
|
|
||||||
BIN
include/.type_check_visitor.hpp.kate-swp
Normal file
BIN
include/.type_check_visitor.hpp.kate-swp
Normal file
Binary file not shown.
|
|
@ -14,6 +14,8 @@ namespace interpreter::tokens {
|
||||||
struct BaseNode {
|
struct BaseNode {
|
||||||
std::pair<size_t, size_t> start_position;
|
std::pair<size_t, size_t> start_position;
|
||||||
std::pair<size_t, size_t> end_position;
|
std::pair<size_t, size_t> end_position;
|
||||||
|
|
||||||
|
std::optional<utils::IdType> type_;
|
||||||
};
|
};
|
||||||
|
|
||||||
// ----------------- Declarations -----------------
|
// ----------------- Declarations -----------------
|
||||||
|
|
|
||||||
|
|
@ -1,9 +1,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
#include <string>
|
#include <string>
|
||||||
////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Temporary frozen, TODO
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
|
@ -12,41 +10,63 @@
|
||||||
// for calngd
|
// for calngd
|
||||||
#include "utils.hpp"
|
#include "utils.hpp"
|
||||||
#include "interpreter_tree.hpp"
|
#include "interpreter_tree.hpp"
|
||||||
#include "error_handling.hpp"
|
|
||||||
|
|
||||||
namespace info {
|
namespace info {
|
||||||
|
|
||||||
class TypeclassGraph {
|
class TypeclassGraph {
|
||||||
public:
|
public:
|
||||||
struct TypeclassMethod {
|
struct MethodInfo {
|
||||||
std::string name;
|
interpreter::tokens::FunctionDeclaration* declaration = nullptr;
|
||||||
interpreter::tokens::FunctionDeclaration* definition = nullptr;
|
std::optional<interpreter::tokens::FunctionDefinition*> definition;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ParametrizedTypeclass {
|
struct ParametrizedTypeclass {
|
||||||
utils::IdType typeclass;
|
std::string typeclass;
|
||||||
std::vector<utils::IdType> parameter_ids; // ??
|
// TODO: parameters
|
||||||
};
|
};
|
||||||
|
|
||||||
struct TypeclassVertex {
|
struct TypeclassVertex {
|
||||||
std::vector<std::pair<std::string, std::vector<ParametrizedTypeclass>>> parameters;
|
std::string name;
|
||||||
std::vector<TypeclassMethod> methods;
|
|
||||||
std::vector<ParametrizedTypeclass> dependencies;
|
|
||||||
interpreter::tokens::TypeclassDefinitionStatement* definition = nullptr;
|
interpreter::tokens::TypeclassDefinitionStatement* definition = nullptr;
|
||||||
|
std::unordered_map<std::string, MethodInfo> methods;
|
||||||
|
std::vector<ParametrizedTypeclass> dependencies;
|
||||||
|
|
||||||
|
// TODO: std::vector<std::pair<std::string, std::vector<ParametrizedTypeclass>>> parameters;
|
||||||
};
|
};
|
||||||
|
|
||||||
utils::IdType AddTypeclass(const TypeclassVertex& typeclass) { // TODO: universal reference
|
std::optional<utils::IdType> AddTypeclass(
|
||||||
for (auto& method : typeclass.methods) {
|
const std::string& name,
|
||||||
if (method_to_typeclass_.count(method.name) != 0) {
|
interpreter::tokens::TypeclassDefinitionStatement* definition,
|
||||||
error_handling::HandleTypecheckError("");
|
const std::vector<ParametrizedTypeclass>& dependencies,
|
||||||
|
const std::vector<std::pair<std::string, interpreter::tokens::FunctionDeclaration*>>& method_declarations,
|
||||||
|
const std::vector<std::pair<std::string, interpreter::tokens::FunctionDefinition*>>& method_definitions) {
|
||||||
|
for (auto& method : method_declarations) {
|
||||||
|
if (method_to_typeclass_.count(method.first) != 0) {
|
||||||
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (auto& method : typeclass.methods) {
|
|
||||||
method_to_typeclass_[method.name] = typeclasses_.size();
|
is_calculated_ = false;
|
||||||
|
|
||||||
|
typeclasses_.emplace_back();
|
||||||
|
TypeclassVertex& typeclass = typeclasses_.back();
|
||||||
|
typeclass.name = name;
|
||||||
|
typeclass.definition = definition;
|
||||||
|
typeclass.dependencies = dependencies;
|
||||||
|
|
||||||
|
for (auto& method : method_declarations) {
|
||||||
|
method_to_typeclass_[method.first] = typeclasses_.size();
|
||||||
|
typeclass.methods[method.first].declaration = method.second;
|
||||||
}
|
}
|
||||||
typeclasses_.push_back(typeclass);
|
name_to_typeclass_[name] = typeclasses_.size() - 1;
|
||||||
|
|
||||||
|
for (auto& method : method_definitions) {
|
||||||
|
typeclass.methods[method.first].definition = method.second;
|
||||||
|
}
|
||||||
|
|
||||||
return typeclasses_.size() - 1;
|
return typeclasses_.size() - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<utils::IdType> FindMethodTypeclass(const std::string& name) {
|
std::optional<utils::IdType> FindMethodTypeclass(const std::string& name) {
|
||||||
auto method_iter = method_to_typeclass_.find(name);
|
auto method_iter = method_to_typeclass_.find(name);
|
||||||
if (method_iter == method_to_typeclass_.end()) {
|
if (method_iter == method_to_typeclass_.end()) {
|
||||||
|
|
@ -55,12 +75,62 @@ public:
|
||||||
return method_iter->second;
|
return method_iter->second;
|
||||||
}
|
}
|
||||||
|
|
||||||
const TypeclassVertex& GetTypeclass(utils::IdType id) {
|
const TypeclassVertex& GetTypeclass(utils::IdType id) { // check, if calculated ??
|
||||||
return typeclasses_.at(id);
|
return typeclasses_.at(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool IsCalculated() {
|
||||||
|
return is_calculated_;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CalculateGraph() {
|
||||||
|
if (is_calculated_) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::vector<size_t>> edges(typeclasses_.size());
|
||||||
|
|
||||||
|
for (size_t i = 0; i < typeclasses_.size(); ++i) {
|
||||||
|
edges[i].resize(typeclasses_[i].dependencies.size());
|
||||||
|
for (size_t j = 0; j < edges[i].size(); ++j) {
|
||||||
|
auto dependency_iter = name_to_typeclass_.find(typeclasses_[i].dependencies[j].typeclass);
|
||||||
|
|
||||||
|
if (dependency_iter == name_to_typeclass_.end()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
edges[i][j] = dependency_iter->second;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<size_t> sorted_verticles = utils::BackTopSort(edges);
|
||||||
|
std::reverse(sorted_verticles.begin(), sorted_verticles.end());
|
||||||
|
|
||||||
|
for (auto& id : sorted_verticles) {
|
||||||
|
for (auto& dependency : typeclasses_[id].dependencies) {
|
||||||
|
for (auto& method : typeclasses_[name_to_typeclass_[dependency.typeclass]].methods) {
|
||||||
|
auto method_iter = typeclasses_[id].methods.find(method.first);
|
||||||
|
if (method_iter == typeclasses_[id].methods.end()) {
|
||||||
|
typeclasses_[id].methods[method.first] = method.second;
|
||||||
|
} else {
|
||||||
|
if (!method_iter->second.definition.has_value()) {
|
||||||
|
method_iter->second.definition = method.second.definition;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
is_calculated_ = true;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
private:
|
private:
|
||||||
std::unordered_map<std::string, utils::IdType> method_to_typeclass_;
|
std::unordered_map<std::string, utils::IdType> method_to_typeclass_;
|
||||||
|
std::unordered_map<std::string, utils::IdType> name_to_typeclass_;
|
||||||
std::vector<TypeclassVertex> typeclasses_;
|
std::vector<TypeclassVertex> typeclasses_;
|
||||||
|
|
||||||
|
bool is_calculated_ = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace info
|
} // namespace info
|
||||||
|
|
|
||||||
128
include/typed_print_visitor.hpp
Normal file
128
include/typed_print_visitor.hpp
Normal file
|
|
@ -0,0 +1,128 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <ostream>
|
||||||
|
|
||||||
|
// for clangd
|
||||||
|
#include "type_info_contexts.hpp"
|
||||||
|
#include "visitor.hpp"
|
||||||
|
|
||||||
|
namespace interpreter {
|
||||||
|
|
||||||
|
class TypedPrintVisitor : public Visitor {
|
||||||
|
public:
|
||||||
|
explicit TypedPrintVisitor(std::ostream& out,
|
||||||
|
info::TypeInfoContextManager& context_manager)
|
||||||
|
: out_(out), context_manager_(context_manager) {}
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Sources -----------------
|
||||||
|
|
||||||
|
void Visit(SourceFile* node) override;
|
||||||
|
|
||||||
|
// Namespaces, partitions -----------------
|
||||||
|
|
||||||
|
void Visit(PartitionSources* node) override;
|
||||||
|
void Visit(Partition* node) override;
|
||||||
|
void Visit(NamespaceSources* node) override;
|
||||||
|
void Visit(Namespace* node) override;
|
||||||
|
|
||||||
|
// Definitions -----------------
|
||||||
|
|
||||||
|
void Visit(ImportStatement* node) override;
|
||||||
|
void Visit(AliasDefinitionStatement* node) override;
|
||||||
|
void Visit(VariableDefinitionStatement* node) override;
|
||||||
|
void Visit(FunctionDeclaration* node) override;
|
||||||
|
void Visit(FunctionDefinitionStatement* node) override;
|
||||||
|
void Visit(TypeDefinitionStatement* node) override;
|
||||||
|
void Visit(AbstractTypeDefinitionStatement* node) override;
|
||||||
|
void Visit(TypeclassDefinitionStatement* node) override;
|
||||||
|
|
||||||
|
// Definition parts
|
||||||
|
|
||||||
|
void Visit(FunctionDefinition* node) override;
|
||||||
|
void Visit(TypeDefinition* node) override;
|
||||||
|
void Visit(AnyAnnotatedType* node) override;
|
||||||
|
|
||||||
|
// Flow control -----------------
|
||||||
|
|
||||||
|
void Visit(TypeConstructorPatternParameter* node) override;
|
||||||
|
void Visit(TypeConstructorPattern* node) override;
|
||||||
|
void Visit(MatchCase* node) override;
|
||||||
|
void Visit(Match* node) override;
|
||||||
|
void Visit(Condition* node) override;
|
||||||
|
void Visit(DoWhileLoop* node) override;
|
||||||
|
void Visit(WhileLoop* node) override;
|
||||||
|
void Visit(ForLoop* node) override;
|
||||||
|
void Visit(LoopLoop* node) override;
|
||||||
|
|
||||||
|
// Statements, expressions, blocks, etc. -----------------
|
||||||
|
|
||||||
|
void Visit(Block* node) override;
|
||||||
|
|
||||||
|
void Visit(ScopedStatement* node) override;
|
||||||
|
|
||||||
|
// Operators
|
||||||
|
|
||||||
|
void Visit(BinaryOperatorExpression* node) override;
|
||||||
|
void Visit(UnaryOperatorExpression* node) override;
|
||||||
|
void Visit(ReferenceExpression* node) override;
|
||||||
|
void Visit(AccessExpression* node) override;
|
||||||
|
|
||||||
|
// Simple Expressions
|
||||||
|
|
||||||
|
void Visit(FunctionCallExpression* node) override;
|
||||||
|
|
||||||
|
void Visit(TupleExpression* node) override;
|
||||||
|
void Visit(VariantExpression* node) override;
|
||||||
|
void Visit(ReturnExpression* node) override;
|
||||||
|
void Visit(TypeConstructorParameter* node) override;
|
||||||
|
void Visit(TypeConstructor* node) override;
|
||||||
|
void Visit(LambdaFunction* node) override;
|
||||||
|
void Visit(ArrayExpression* node) override;
|
||||||
|
|
||||||
|
void Visit(LoopControlExpression& node) override; // enum
|
||||||
|
|
||||||
|
// Name
|
||||||
|
|
||||||
|
void Visit(NameExpression* node) override;
|
||||||
|
void Visit(TupleName* node) override;
|
||||||
|
void Visit(VariantName* node) override;
|
||||||
|
void Visit(AnnotatedName* node) override;
|
||||||
|
|
||||||
|
// Type, typeclass, etc. -----------------
|
||||||
|
|
||||||
|
// Type
|
||||||
|
|
||||||
|
void Visit(FunctionType* node) override;
|
||||||
|
void Visit(TupleType* node) override;
|
||||||
|
void Visit(VariantType* node) override;
|
||||||
|
void Visit(TypeExpression* node) override;
|
||||||
|
|
||||||
|
void Visit(ExtendedScopedAnyType* node) override;
|
||||||
|
|
||||||
|
// Typeclass
|
||||||
|
|
||||||
|
void Visit(ParametrizedTypeclass* node) override;
|
||||||
|
|
||||||
|
// Typeclass & Type
|
||||||
|
|
||||||
|
void Visit(ParametrizedType* node) override;
|
||||||
|
|
||||||
|
// Identifiers, constants, etc. -----------------
|
||||||
|
|
||||||
|
void Visit(ExtendedName* node) override;
|
||||||
|
|
||||||
|
void Visit(std::string* node) override; // std::string
|
||||||
|
|
||||||
|
void Visit(FloatNumberLiteral* node) override;
|
||||||
|
void Visit(NumberLiteral* node) override;
|
||||||
|
void Visit(StringLiteral* node) override;
|
||||||
|
void Visit(CharLiteral* node) override;
|
||||||
|
void Visit(UnitLiteral* node) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::ostream& out_;
|
||||||
|
info::TypeInfoContextManager& context_manager_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace interpreter
|
||||||
|
|
@ -234,6 +234,8 @@ public:
|
||||||
|
|
||||||
std::optional<utils::IdType> GetFieldType(const std::string& name) const;
|
std::optional<utils::IdType> GetFieldType(const std::string& name) const;
|
||||||
|
|
||||||
|
std::string GetTypeName() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::variant<AbstractType,
|
std::variant<AbstractType,
|
||||||
DefinedType,
|
DefinedType,
|
||||||
|
|
|
||||||
|
|
@ -107,4 +107,32 @@ private:
|
||||||
std::vector<size_t> ranks_;
|
std::vector<size_t> ranks_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void BackVisitDfs(size_t id,
|
||||||
|
std::vector<size_t>& verticles,
|
||||||
|
std::vector<size_t>& marks,
|
||||||
|
const std::vector<std::vector<size_t>>& edges,
|
||||||
|
size_t mark) {
|
||||||
|
if (marks[id] != 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
marks[id] = mark;
|
||||||
|
verticles.push_back(id);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < edges[id].size(); ++i) {
|
||||||
|
BackVisitDfs(id, verticles, marks, edges, mark);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::vector<size_t> BackTopSort(const std::vector<std::vector<size_t>>& edges_) {
|
||||||
|
std::vector<size_t> sorted_verticles;
|
||||||
|
std::vector<size_t> marks(edges_.size(), 0);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < marks.size(); ++i) {
|
||||||
|
BackVisitDfs(i, sorted_verticles, marks, edges_, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return sorted_verticles;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace utils
|
} // namespace utils
|
||||||
|
|
|
||||||
|
|
@ -1 +1 @@
|
||||||
Subproject commit d7c58c48cc7afc031e5dc350eb2a497c1bac5511
|
Subproject commit 60a270ed79f866baae441b823f241cbc032999da
|
||||||
Binary file not shown.
|
|
@ -50,9 +50,9 @@ void LinkSymbolsVisitor::Visit(TypeExpression* node) { // TODO: check
|
||||||
node->type_id_ = namespace_visitor_.FindType(path, node->type.type);
|
node->type_id_ = namespace_visitor_.FindType(path, node->type.type);
|
||||||
node->constructor_id_ = namespace_visitor_.FindConstructor(path, node->type.type);
|
node->constructor_id_ = namespace_visitor_.FindConstructor(path, node->type.type);
|
||||||
|
|
||||||
if (!node->type_id_.has_value() && !node->constructor_id_.has_value()) {
|
// if (!node->type_id_.has_value() && !node->constructor_id_.has_value()) { // TODO: check, that not bastract types
|
||||||
error_handling::HandleTypecheckError("Type or constructor not found", node->base);
|
// error_handling::HandleTypecheckError("Type or constructor not found", node->base);
|
||||||
}
|
// }
|
||||||
|
|
||||||
if (node->constructor_id_.has_value()) {
|
if (node->constructor_id_.has_value()) {
|
||||||
utils::IdType constructor_type_id = namespace_visitor_.GetGlobalInfo()->GetConstructorInfo(node->constructor_id_.value()).type_id;
|
utils::IdType constructor_type_id = namespace_visitor_.GetGlobalInfo()->GetConstructorInfo(node->constructor_id_.value()).type_id;
|
||||||
|
|
|
||||||
20
src/main.cpp
20
src/main.cpp
|
|
@ -4,9 +4,15 @@
|
||||||
|
|
||||||
// for clangd
|
// for clangd
|
||||||
#include "../include/parse_tree.hpp"
|
#include "../include/parse_tree.hpp"
|
||||||
|
#include "../include/global_info.hpp"
|
||||||
|
#include "../include/type_info_contexts.hpp"
|
||||||
#include "../include/interpreter_tree.hpp"
|
#include "../include/interpreter_tree.hpp"
|
||||||
#include "../include/build_visitor.hpp"
|
#include "../include/build_visitor.hpp"
|
||||||
#include "../include/print_visitor.hpp"
|
#include "../include/print_visitor.hpp"
|
||||||
|
#include "../include/find_symbols_visitor.hpp"
|
||||||
|
#include "../include/link_symbols_visitor.hpp"
|
||||||
|
#include "../include/type_check_visitor.hpp"
|
||||||
|
#include "../include/typed_print_visitor.hpp"
|
||||||
#include "../include/error_handling.hpp"
|
#include "../include/error_handling.hpp"
|
||||||
|
|
||||||
int main(int argc, char** argv) { // TODO, only test version
|
int main(int argc, char** argv) { // TODO, only test version
|
||||||
|
|
@ -37,9 +43,19 @@ int main(int argc, char** argv) { // TODO, only test version
|
||||||
std::unique_ptr<interpreter::tokens::SourceFile> source_file =
|
std::unique_ptr<interpreter::tokens::SourceFile> source_file =
|
||||||
std::make_unique<interpreter::tokens::SourceFile>();
|
std::make_unique<interpreter::tokens::SourceFile>();
|
||||||
|
|
||||||
|
info::GlobalInfo global_info;
|
||||||
|
info::TypeInfoContextManager context_manager;
|
||||||
|
|
||||||
interpreter::BuildVisitor build_visitor(parse_tree);
|
interpreter::BuildVisitor build_visitor(parse_tree);
|
||||||
interpreter::PrintVisitor print_visitor(std::cout);
|
// interpreter::PrintVisitor print_visitor(std::cout);
|
||||||
|
interpreter::FindSymbolsVisitor find_symbols_visitor(global_info);
|
||||||
|
interpreter::LinkSymbolsVisitor link_symbols_visitor(global_info);
|
||||||
|
interpreter::TypeCheckVisitor type_check_visitor(global_info, context_manager);
|
||||||
|
interpreter::TypedPrintVisitor typed_print_visitor(std::cout, context_manager);
|
||||||
|
|
||||||
build_visitor.VisitSourceFile(source_file.get());
|
build_visitor.VisitSourceFile(source_file.get());
|
||||||
print_visitor.VisitSourceFile(source_file.get());
|
find_symbols_visitor.VisitSourceFile(source_file.get());
|
||||||
|
link_symbols_visitor.VisitSourceFile(source_file.get());
|
||||||
|
type_check_visitor.VisitSourceFile(source_file.get());
|
||||||
|
typed_print_visitor.VisitSourceFile(source_file.get());
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -19,6 +19,8 @@ void TypeCheckVisitor::Visit(SourceFile* node) {
|
||||||
Visitor::Visit(statement);
|
Visitor::Visit(statement);
|
||||||
}
|
}
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Namespaces, partitions -----------------
|
// Namespaces, partitions -----------------
|
||||||
|
|
@ -28,11 +30,15 @@ void TypeCheckVisitor::Visit(PartitionSources* node) {
|
||||||
Visitor::Visit(statement);
|
Visitor::Visit(statement);
|
||||||
}
|
}
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(Partition* node) {
|
void TypeCheckVisitor::Visit(Partition* node) {
|
||||||
Visit(&node->scope);
|
Visit(&node->scope);
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(NamespaceSources* node) {
|
void TypeCheckVisitor::Visit(NamespaceSources* node) {
|
||||||
|
|
@ -40,6 +46,8 @@ void TypeCheckVisitor::Visit(NamespaceSources* node) {
|
||||||
Visitor::Visit(statement);
|
Visitor::Visit(statement);
|
||||||
}
|
}
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(Namespace* node) { // TODO: two var namespces for class: const and var
|
void TypeCheckVisitor::Visit(Namespace* node) { // TODO: two var namespces for class: const and var
|
||||||
|
|
@ -78,6 +86,8 @@ void TypeCheckVisitor::Visit(Namespace* node) { // TODO: two var namespces for c
|
||||||
namespace_visitor_.ExitNamespace();
|
namespace_visitor_.ExitNamespace();
|
||||||
context_manager_.ExitContext();
|
context_manager_.ExitContext();
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Definitions -----------------
|
// Definitions -----------------
|
||||||
|
|
@ -107,6 +117,8 @@ void TypeCheckVisitor::Visit(AliasDefinitionStatement* node) {
|
||||||
// Visit(node->value.get());
|
// Visit(node->value.get());
|
||||||
//
|
//
|
||||||
// context_manager_.ExitContext();
|
// context_manager_.ExitContext();
|
||||||
|
//
|
||||||
|
// node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(VariableDefinitionStatement* node) {
|
void TypeCheckVisitor::Visit(VariableDefinitionStatement* node) {
|
||||||
|
|
@ -127,6 +139,8 @@ void TypeCheckVisitor::Visit(VariableDefinitionStatement* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
is_in_statement_ = false;
|
is_in_statement_ = false;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(FunctionDeclaration* node) {
|
void TypeCheckVisitor::Visit(FunctionDeclaration* node) {
|
||||||
|
|
@ -138,6 +152,8 @@ void TypeCheckVisitor::Visit(FunctionDeclaration* node) {
|
||||||
if (!was_in_statement) {
|
if (!was_in_statement) {
|
||||||
is_in_statement_ = false;
|
is_in_statement_ = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(FunctionDefinitionStatement* node) {
|
void TypeCheckVisitor::Visit(FunctionDefinitionStatement* node) {
|
||||||
|
|
@ -190,6 +206,8 @@ void TypeCheckVisitor::Visit(FunctionDefinitionStatement* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
is_in_statement_ = false;
|
is_in_statement_ = false;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(TypeDefinitionStatement* node) {
|
void TypeCheckVisitor::Visit(TypeDefinitionStatement* node) {
|
||||||
|
|
@ -198,6 +216,8 @@ void TypeCheckVisitor::Visit(TypeDefinitionStatement* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
is_in_statement_ = false;
|
is_in_statement_ = false;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(AbstractTypeDefinitionStatement* node) {
|
void TypeCheckVisitor::Visit(AbstractTypeDefinitionStatement* node) {
|
||||||
|
|
@ -219,6 +239,8 @@ void TypeCheckVisitor::Visit(AbstractTypeDefinitionStatement* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
is_in_statement_ = false;
|
is_in_statement_ = false;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(TypeclassDefinitionStatement* node) {
|
void TypeCheckVisitor::Visit(TypeclassDefinitionStatement* node) {
|
||||||
|
|
@ -227,20 +249,28 @@ void TypeCheckVisitor::Visit(TypeclassDefinitionStatement* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
is_in_statement_ = false;
|
is_in_statement_ = false;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Definition parts
|
// Definition parts
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(FunctionDefinition* node) {
|
void TypeCheckVisitor::Visit(FunctionDefinition* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(TypeDefinition* node) {
|
void TypeCheckVisitor::Visit(TypeDefinition* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(AnyAnnotatedType* node) {
|
void TypeCheckVisitor::Visit(AnyAnnotatedType* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Unit, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Flow control -----------------
|
// Flow control -----------------
|
||||||
|
|
@ -313,6 +343,8 @@ void TypeCheckVisitor::Visit(TypeConstructorPattern* node) { // TODO: match name
|
||||||
Visitor::Visit(*type_info.value);
|
Visitor::Visit(*type_info.value);
|
||||||
current_type_ = TypeInContext(current_type_, context);
|
current_type_ = TypeInContext(current_type_, context);
|
||||||
current_type_ = context_manager_.ToModifiedType(current_type_, utils::ValueType::Tmp);
|
current_type_ = context_manager_.ToModifiedType(current_type_, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(MatchCase* node) {
|
void TypeCheckVisitor::Visit(MatchCase* node) {
|
||||||
|
|
@ -339,6 +371,8 @@ void TypeCheckVisitor::Visit(MatchCase* node) {
|
||||||
Visitor::Visit(node->statement.value());
|
Visitor::Visit(node->statement.value());
|
||||||
}
|
}
|
||||||
// current_type_ from statement is current_type_ for MatchCase
|
// current_type_ from statement is current_type_ for MatchCase
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(Match* node) { // TODO: move value to match
|
void TypeCheckVisitor::Visit(Match* node) { // TODO: move value to match
|
||||||
|
|
@ -376,6 +410,8 @@ void TypeCheckVisitor::Visit(Match* node) { // TODO: move value to match
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = type;
|
current_type_ = type;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(Condition* node) {
|
void TypeCheckVisitor::Visit(Condition* node) {
|
||||||
|
|
@ -413,6 +449,8 @@ void TypeCheckVisitor::Visit(Condition* node) {
|
||||||
info::type::OptionalType(type, context_manager_.GetTypeManager()),
|
info::type::OptionalType(type, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp); // ??
|
utils::ValueType::Tmp); // ??
|
||||||
}
|
}
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(DoWhileLoop* node) {
|
void TypeCheckVisitor::Visit(DoWhileLoop* node) {
|
||||||
|
|
@ -443,6 +481,8 @@ void TypeCheckVisitor::Visit(WhileLoop* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::ArrayType(0, current_type_, context_manager_.GetTypeManager()),
|
info::type::ArrayType(0, current_type_, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(ForLoop* node) {
|
void TypeCheckVisitor::Visit(ForLoop* node) {
|
||||||
|
|
@ -479,6 +519,8 @@ void TypeCheckVisitor::Visit(LoopLoop* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::ArrayType(0, current_type_, context_manager_.GetTypeManager()),
|
info::type::ArrayType(0, current_type_, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Statements, expressions, blocks, etc. -----------------
|
// Statements, expressions, blocks, etc. -----------------
|
||||||
|
|
@ -511,11 +553,15 @@ void TypeCheckVisitor::Visit(Block* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = type;
|
current_type_ = type;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(ScopedStatement* node) {
|
void TypeCheckVisitor::Visit(ScopedStatement* node) {
|
||||||
Visitor::Visit(node->statement);
|
Visitor::Visit(node->statement);
|
||||||
// current_type_ is type of statement
|
// current_type_ is type of statement
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(LoopControlExpression& node) { // enum
|
void TypeCheckVisitor::Visit(LoopControlExpression& node) { // enum
|
||||||
|
|
@ -524,7 +570,7 @@ void TypeCheckVisitor::Visit(LoopControlExpression& node) { // enum
|
||||||
|
|
||||||
// Operators
|
// Operators
|
||||||
|
|
||||||
// TODO
|
// TODO: better structure
|
||||||
void TypeCheckVisitor::Visit(BinaryOperatorExpression* node) {
|
void TypeCheckVisitor::Visit(BinaryOperatorExpression* node) {
|
||||||
// TODO: Check, that type is not abstract ??
|
// TODO: Check, that type is not abstract ??
|
||||||
auto maybe_operator_id = namespace_visitor_.FindFunction(std::nullopt, node->operator_name);
|
auto maybe_operator_id = namespace_visitor_.FindFunction(std::nullopt, node->operator_name);
|
||||||
|
|
@ -632,8 +678,11 @@ void TypeCheckVisitor::Visit(BinaryOperatorExpression* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Visitor::Visit(*operator_info.declaration.value().argument_types.back());
|
Visitor::Visit(*operator_info.declaration.value().argument_types.back());
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: can be method ??
|
||||||
void TypeCheckVisitor::Visit(UnaryOperatorExpression* node) {
|
void TypeCheckVisitor::Visit(UnaryOperatorExpression* node) {
|
||||||
// TODO: Check, that type is not abstract ??
|
// TODO: Check, that type is not abstract ??
|
||||||
auto maybe_operator_id = namespace_visitor_.FindFunction({}, node->operator_name);
|
auto maybe_operator_id = namespace_visitor_.FindFunction({}, node->operator_name);
|
||||||
|
|
@ -652,7 +701,7 @@ void TypeCheckVisitor::Visit(UnaryOperatorExpression* node) {
|
||||||
error_handling::HandleTypecheckError("Operator definition not found", node->base);
|
error_handling::HandleTypecheckError("Operator definition not found", node->base);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (operator_info.argument_count != 1) {
|
if (operator_info.argument_count != 2) { // 1 + return type
|
||||||
error_handling::HandleTypecheckError("Operator wrong argument count", node->base);
|
error_handling::HandleTypecheckError("Operator wrong argument count", node->base);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -667,6 +716,10 @@ void TypeCheckVisitor::Visit(UnaryOperatorExpression* node) {
|
||||||
if (!context_manager_.AddTypeRequirement(current_type_, expression_type)) {
|
if (!context_manager_.AddTypeRequirement(current_type_, expression_type)) {
|
||||||
error_handling::HandleTypecheckError("Operator expression has wrong type", node->base);
|
error_handling::HandleTypecheckError("Operator expression has wrong type", node->base);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Visitor::Visit(*operator_info.declaration.value().argument_types.back());
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(ReferenceExpression* node) {
|
void TypeCheckVisitor::Visit(ReferenceExpression* node) {
|
||||||
|
|
@ -693,6 +746,8 @@ void TypeCheckVisitor::Visit(AccessExpression* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = maybe_type_value.value()->GetElementsType();
|
current_type_ = maybe_type_value.value()->GetElementsType();
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Other Expressions
|
// Other Expressions
|
||||||
|
|
@ -701,6 +756,7 @@ void TypeCheckVisitor::Visit(AccessExpression* node) {
|
||||||
// TODO: builtin functions/methods
|
// TODO: builtin functions/methods
|
||||||
// TODO: alias types, abstract types, etc.
|
// TODO: alias types, abstract types, etc.
|
||||||
// TODO: deduce function parameter types
|
// TODO: deduce function parameter types
|
||||||
|
// TODO: find functions in typeclasses
|
||||||
void TypeCheckVisitor::Visit(FunctionCallExpression* node) {
|
void TypeCheckVisitor::Visit(FunctionCallExpression* node) {
|
||||||
std::optional<utils::IdType> maybe_function_id;
|
std::optional<utils::IdType> maybe_function_id;
|
||||||
std::unordered_map<std::string, utils::IdType> context;
|
std::unordered_map<std::string, utils::IdType> context;
|
||||||
|
|
@ -830,6 +886,8 @@ void TypeCheckVisitor::Visit(TupleExpression* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::TupleType(std::nullopt, fields, context_manager_.GetTypeManager()),
|
info::type::TupleType(std::nullopt, fields, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ??
|
// ??
|
||||||
|
|
@ -847,11 +905,15 @@ void TypeCheckVisitor::Visit(VariantExpression* node) {
|
||||||
|
|
||||||
current_type_ = context_manager_.AddType(info::type::VariantType(std::nullopt, constructors, -1),
|
current_type_ = context_manager_.AddType(info::type::VariantType(std::nullopt, constructors, -1),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(ReturnExpression* node) {
|
void TypeCheckVisitor::Visit(ReturnExpression* node) {
|
||||||
Visitor::Visit(node->expression);
|
Visitor::Visit(node->expression);
|
||||||
returned_type_ = current_type_;
|
returned_type_ = current_type_;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(TypeConstructorParameter* node) {} // Handeled in TypeConstructor visit
|
void TypeCheckVisitor::Visit(TypeConstructorParameter* node) {} // Handeled in TypeConstructor visit
|
||||||
|
|
@ -931,11 +993,15 @@ void TypeCheckVisitor::Visit(TypeConstructor* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = TypeInContext(current_type_, context);
|
current_type_ = TypeInContext(current_type_, context);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(LambdaFunction* node) { // TODO
|
void TypeCheckVisitor::Visit(LambdaFunction* node) { // TODO
|
||||||
error_handling::HandleInternalError("Unimplemented (unsolved type deduction problems)",
|
error_handling::HandleInternalError("Unimplemented (unsolved type deduction problems)",
|
||||||
"TypeCheckVisitor.LambdaFunction");
|
"TypeCheckVisitor.LambdaFunction");
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(ArrayExpression* node) {
|
void TypeCheckVisitor::Visit(ArrayExpression* node) {
|
||||||
|
|
@ -959,6 +1025,8 @@ void TypeCheckVisitor::Visit(ArrayExpression* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::ArrayType(node->elements.size(), elements_type, context_manager_.GetTypeManager()),
|
info::type::ArrayType(node->elements.size(), elements_type, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Name
|
// Name
|
||||||
|
|
@ -993,6 +1061,8 @@ void TypeCheckVisitor::Visit(NameExpression* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = context_manager_.ToModifiedType(current_type_, variable_value_type);
|
current_type_ = context_manager_.ToModifiedType(current_type_, variable_value_type);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(TupleName* node) {
|
void TypeCheckVisitor::Visit(TupleName* node) {
|
||||||
|
|
@ -1034,6 +1104,8 @@ void TypeCheckVisitor::Visit(TupleName* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = type;
|
current_type_ = type;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(VariantName* node) {
|
void TypeCheckVisitor::Visit(VariantName* node) {
|
||||||
|
|
@ -1084,6 +1156,8 @@ void TypeCheckVisitor::Visit(VariantName* node) {
|
||||||
}
|
}
|
||||||
|
|
||||||
current_type_ = type;
|
current_type_ = type;
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(AnnotatedName* node) {
|
void TypeCheckVisitor::Visit(AnnotatedName* node) {
|
||||||
|
|
@ -1109,6 +1183,8 @@ void TypeCheckVisitor::Visit(AnnotatedName* node) {
|
||||||
error_handling::HandleTypecheckError("Wrong type annotation in annotated name", node->base);
|
error_handling::HandleTypecheckError("Wrong type annotation in annotated name", node->base);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Type, typeclass, etc. -----------------
|
// Type, typeclass, etc. -----------------
|
||||||
|
|
@ -1129,6 +1205,8 @@ void TypeCheckVisitor::Visit(FunctionType* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::FunctionType(argument_types, return_type, context_manager_.GetTypeManager()),
|
info::type::FunctionType(argument_types, return_type, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(TupleType* node) {
|
void TypeCheckVisitor::Visit(TupleType* node) {
|
||||||
|
|
@ -1143,6 +1221,8 @@ void TypeCheckVisitor::Visit(TupleType* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::TupleType(node->type, fields, context_manager_.GetTypeManager()),
|
info::type::TupleType(node->type, fields, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(VariantType* node) {
|
void TypeCheckVisitor::Visit(VariantType* node) {
|
||||||
|
|
@ -1167,6 +1247,8 @@ void TypeCheckVisitor::Visit(VariantType* node) {
|
||||||
|
|
||||||
current_type_ = context_manager_.AddType(info::type::VariantType(node->type, constructors, -1),
|
current_type_ = context_manager_.AddType(info::type::VariantType(node->type, constructors, -1),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO handle local abstract types, abstract types, aliases, etc.
|
// TODO handle local abstract types, abstract types, aliases, etc.
|
||||||
|
|
@ -1194,6 +1276,8 @@ void TypeCheckVisitor::Visit(TypeExpression* node) {
|
||||||
info::type::ArrayType(node->array_size.value(), current_type_, context_manager_.GetTypeManager()),
|
info::type::ArrayType(node->array_size.value(), current_type_, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(ExtendedScopedAnyType* node) {
|
void TypeCheckVisitor::Visit(ExtendedScopedAnyType* node) {
|
||||||
|
|
@ -1202,6 +1286,8 @@ void TypeCheckVisitor::Visit(ExtendedScopedAnyType* node) {
|
||||||
current_type_ = context_manager_.AddType(
|
current_type_ = context_manager_.AddType(
|
||||||
info::type::ReferenceToType(node->references, current_type_, context_manager_.GetTypeManager()),
|
info::type::ReferenceToType(node->references, current_type_, context_manager_.GetTypeManager()),
|
||||||
utils::ValueType::Tmp);
|
utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Typeclass
|
// Typeclass
|
||||||
|
|
@ -1210,6 +1296,8 @@ void TypeCheckVisitor::Visit(ParametrizedTypeclass* node) { // TODO ??
|
||||||
for (auto& parameter : node->parameters) {
|
for (auto& parameter : node->parameters) {
|
||||||
Visit(parameter.get());
|
Visit(parameter.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Typeclass & Type -----------------
|
// Typeclass & Type -----------------
|
||||||
|
|
@ -1220,18 +1308,26 @@ void TypeCheckVisitor::Visit(ParametrizedType* node) {} // Handled in TypeExpres
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(FloatNumberLiteral* node) {
|
void TypeCheckVisitor::Visit(FloatNumberLiteral* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Float, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Float, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(NumberLiteral* node) {
|
void TypeCheckVisitor::Visit(NumberLiteral* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Int, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Int, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(StringLiteral* node) {
|
void TypeCheckVisitor::Visit(StringLiteral* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::String, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::String, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCheckVisitor::Visit(CharLiteral* node) {
|
void TypeCheckVisitor::Visit(CharLiteral* node) {
|
||||||
current_type_ = context_manager_.AddType(info::type::InternalType::Char, utils::ValueType::Tmp);
|
current_type_ = context_manager_.AddType(info::type::InternalType::Char, utils::ValueType::Tmp);
|
||||||
|
|
||||||
|
node->base.type_ = current_type_;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
|
|
||||||
1061
src/typed_print_visitor.cpp
Normal file
1061
src/typed_print_visitor.cpp
Normal file
File diff suppressed because it is too large
Load diff
|
|
@ -436,6 +436,36 @@ std::optional<utils::IdType> Type::GetFieldType(const std::string& name) const {
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string Type::GetTypeName() const {
|
||||||
|
size_t index = type_.index();
|
||||||
|
|
||||||
|
switch (index) {
|
||||||
|
case 0:
|
||||||
|
return "AbstractType";
|
||||||
|
case 1:
|
||||||
|
return "DefinedType";
|
||||||
|
case 2:
|
||||||
|
return "Builtin";
|
||||||
|
case 3:
|
||||||
|
return "TupleType";
|
||||||
|
case 4:
|
||||||
|
return "VariantType";
|
||||||
|
case 5:
|
||||||
|
return "ReferenceToType";
|
||||||
|
case 6:
|
||||||
|
return "FunctionType";
|
||||||
|
case 7:
|
||||||
|
return "ArrayType";
|
||||||
|
case 8:
|
||||||
|
return "OptionalType";
|
||||||
|
default:
|
||||||
|
// error
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ""; // ??
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue