Get basic G-machine compilation working.
This commit is contained in:
parent
d729611486
commit
da515437e6
|
@ -2,5 +2,5 @@ cmake_minimum_required(VERSION 3.0)
|
|||
project(lily)
|
||||
|
||||
set(CMAKE_CXX_STANDARD 14)
|
||||
add_executable(lily src/main.cpp src/parser.cpp src/parser.c src/type.cpp src/type_manager.cpp src/ast.cpp src/type_checker.cpp src/pattern.cpp)
|
||||
add_executable(lily src/main.cpp src/parser.cpp src/parser.c src/type.cpp src/type_manager.cpp src/ast.cpp src/type_checker.cpp src/pattern.cpp src/gmachine.cpp src/compiler.cpp)
|
||||
target_include_directories(lily PUBLIC src)
|
||||
|
|
52
src/ast.cpp
52
src/ast.cpp
|
@ -93,4 +93,56 @@ namespace lily {
|
|||
|
||||
return branch_type;
|
||||
}
|
||||
|
||||
void ast_num::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
into.push_back(mgr.add_instruction<instruction_push_int>(3));
|
||||
}
|
||||
|
||||
void ast_var::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
if(env->is_variable(name)) {
|
||||
into.push_back(mgr.add_instruction<instruction_push>(env->get_offset(name)));
|
||||
} else {
|
||||
into.push_back(mgr.add_instruction<instruction_push_global>(name));
|
||||
}
|
||||
}
|
||||
|
||||
void ast_app::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
auto new_env = std::make_shared<compile_env_offset>(1);
|
||||
new_env->set_parent(env);
|
||||
right->compile(mgr, into, env);
|
||||
left->compile(mgr, into, new_env);
|
||||
into.push_back(mgr.add_instruction<instruction_mkapp>());
|
||||
}
|
||||
|
||||
void ast_op::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
auto new_env = std::make_shared<compile_env_offset>(1);
|
||||
new_env->set_parent(env);
|
||||
right->compile(mgr, into, env);
|
||||
left->compile(mgr, into, new_env);
|
||||
into.push_back(mgr.add_instruction<instruction_push_global>(op_supercombinator(op)));
|
||||
into.push_back(mgr.add_instruction<instruction_mkapp>());
|
||||
into.push_back(mgr.add_instruction<instruction_mkapp>());
|
||||
}
|
||||
|
||||
void ast_let::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
expr->compile(mgr, into, env);
|
||||
auto new_env = std::make_shared<compile_env_var>(name);
|
||||
new_env->set_parent(env);
|
||||
in->compile(mgr, into, new_env);
|
||||
into.push_back(mgr.add_instruction<instruction_slide>(1));
|
||||
}
|
||||
|
||||
void ast_letrec::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
into.push_back(mgr.add_instruction<instruction_alloc>(1));
|
||||
auto new_env = std::make_shared<compile_env_var>(name);
|
||||
new_env->set_parent(env);
|
||||
expr->compile(mgr, into, new_env);
|
||||
into.push_back(mgr.add_instruction<instruction_update>(0));
|
||||
in->compile(mgr, into, new_env);
|
||||
into.push_back(mgr.add_instruction<instruction_slide>(1));
|
||||
}
|
||||
|
||||
void ast_case::compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) {
|
||||
throw error("case expressions unimplemented");
|
||||
}
|
||||
}
|
||||
|
|
19
src/ast.hpp
19
src/ast.hpp
|
@ -4,6 +4,8 @@
|
|||
#include "pattern.hpp"
|
||||
#include "type.hpp"
|
||||
#include "type_manager.hpp"
|
||||
#include "compiler.hpp"
|
||||
#include "binop.hpp"
|
||||
|
||||
namespace lily {
|
||||
class type_env;
|
||||
|
@ -11,6 +13,7 @@ namespace lily {
|
|||
struct ast {
|
||||
virtual ~ast() = default;
|
||||
virtual type* check(type_manager& mgr, std::shared_ptr<type_env> env) = 0;
|
||||
virtual void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env) = 0;
|
||||
};
|
||||
|
||||
typedef std::unique_ptr<ast> ast_ptr;
|
||||
|
@ -21,6 +24,7 @@ namespace lily {
|
|||
ast_num(int i) : num(i) {}
|
||||
~ast_num() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
|
||||
struct ast_var : ast {
|
||||
|
@ -30,6 +34,7 @@ namespace lily {
|
|||
name(std::move(n)) {}
|
||||
~ast_var() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
|
||||
struct ast_app : ast {
|
||||
|
@ -40,22 +45,19 @@ namespace lily {
|
|||
left(std::move(l)), right(std::move(r)) {}
|
||||
~ast_app() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
|
||||
struct ast_op : ast {
|
||||
enum class op {
|
||||
add,
|
||||
subtract,
|
||||
times,
|
||||
divide
|
||||
} op;
|
||||
binop op;
|
||||
std::unique_ptr<ast> left;
|
||||
std::unique_ptr<ast> right;
|
||||
|
||||
ast_op(enum op o, ast_ptr l, ast_ptr r) :
|
||||
ast_op(binop o, ast_ptr l, ast_ptr r) :
|
||||
op(o), left(std::move(l)), right(std::move(r)) {}
|
||||
~ast_op() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
|
||||
struct ast_let : ast {
|
||||
|
@ -67,6 +69,7 @@ namespace lily {
|
|||
name(std::move(n)), expr(std::move(e)), in(std::move(i)) {}
|
||||
~ast_let() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
|
||||
struct ast_letrec : ast_let {
|
||||
|
@ -74,6 +77,7 @@ namespace lily {
|
|||
ast_let(std::move(n), std::move(e), std::move(i)) {}
|
||||
~ast_letrec() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
|
||||
struct ast_case : ast {
|
||||
|
@ -87,5 +91,6 @@ namespace lily {
|
|||
|
||||
~ast_case() = default;
|
||||
type* check(type_manager& mgr, std::shared_ptr<type_env> env);
|
||||
void compile(instruction_manager& mgr, std::vector<instruction*>& into, std::shared_ptr<compile_env> env);
|
||||
};
|
||||
}
|
||||
|
|
20
src/binop.hpp
Normal file
20
src/binop.hpp
Normal file
|
@ -0,0 +1,20 @@
|
|||
#pragma once
|
||||
#include <string>
|
||||
|
||||
namespace lily {
|
||||
enum class binop {
|
||||
add,
|
||||
subtract,
|
||||
times,
|
||||
divide
|
||||
};
|
||||
|
||||
static std::string op_supercombinator(binop op) {
|
||||
switch(op) {
|
||||
case binop::add: return "+";
|
||||
case binop::subtract: return "-";
|
||||
case binop::times: return "*";
|
||||
case binop::divide: return "/";
|
||||
}
|
||||
}
|
||||
};
|
30
src/compiler.cpp
Normal file
30
src/compiler.cpp
Normal file
|
@ -0,0 +1,30 @@
|
|||
#include "compiler.hpp"
|
||||
#include "error.hpp"
|
||||
|
||||
namespace lily {
|
||||
void compile_env::set_parent(std::shared_ptr<compile_env> p) {
|
||||
parent = p;
|
||||
}
|
||||
|
||||
int compile_env_var::get_offset(const std::string& name) {
|
||||
if(this->var == name) return 0;
|
||||
if(parent) return parent->get_offset(name) + 1;
|
||||
throw error("unknown variable name");
|
||||
}
|
||||
|
||||
bool compile_env_var::is_variable(const std::string& name) {
|
||||
if(this->var == name) return true;
|
||||
if(parent) return parent->is_variable(name);
|
||||
return false;
|
||||
}
|
||||
|
||||
int compile_env_offset::get_offset(const std::string& name) {
|
||||
if(parent) return parent->get_offset(name) + offset;
|
||||
throw error("unknown variable name");
|
||||
}
|
||||
|
||||
bool compile_env_offset::is_variable(const std::string& name) {
|
||||
if(parent) return parent->is_variable(name);
|
||||
return false;
|
||||
}
|
||||
}
|
32
src/compiler.hpp
Normal file
32
src/compiler.hpp
Normal file
|
@ -0,0 +1,32 @@
|
|||
#pragma once
|
||||
#include <map>
|
||||
#include "gmachine.hpp"
|
||||
|
||||
namespace lily {
|
||||
class compile_env {
|
||||
protected:
|
||||
std::shared_ptr<compile_env> parent = nullptr;
|
||||
public:
|
||||
virtual int get_offset(const std::string& name) = 0;
|
||||
virtual bool is_variable(const std::string& name) = 0;
|
||||
void set_parent(std::shared_ptr<compile_env> p);
|
||||
};
|
||||
|
||||
class compile_env_var : public compile_env {
|
||||
private:
|
||||
std::string var;
|
||||
public:
|
||||
compile_env_var(std::string v) : var(std::move(v)), compile_env() {}
|
||||
int get_offset(const std::string& name);
|
||||
bool is_variable(const std::string& name);
|
||||
};
|
||||
|
||||
class compile_env_offset : public compile_env {
|
||||
private:
|
||||
int offset;
|
||||
public:
|
||||
compile_env_offset(int o) : offset(o), compile_env() {}
|
||||
int get_offset(const std::string& name);
|
||||
bool is_variable(const std::string& name);
|
||||
};
|
||||
};
|
89
src/gmachine.cpp
Normal file
89
src/gmachine.cpp
Normal file
|
@ -0,0 +1,89 @@
|
|||
#include "gmachine.hpp"
|
||||
#include <iostream>
|
||||
|
||||
namespace lily {
|
||||
std::ostream& operator<<(std::ostream& os, instruction& inst) {
|
||||
inst.to_stream(os);
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_slide::to_stream(std::ostream& os) {
|
||||
os << "slide(" << amount << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_alloc::to_stream(std::ostream& os) {
|
||||
os << "alloc(" << amount << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_pop::to_stream(std::ostream& os) {
|
||||
os << "pop(" << amount << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_unwind::to_stream(std::ostream& os) {
|
||||
os << "unwind";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_push_global::to_stream(std::ostream& os) {
|
||||
os << "pushglobal(" << name << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_push_int::to_stream(std::ostream& os) {
|
||||
os << "pushint(" << value << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_push_str::to_stream(std::ostream& os) {
|
||||
os << "pushstr(" << str << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_push::to_stream(std::ostream& os) {
|
||||
os << "push(" << offset << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_mkapp::to_stream(std::ostream& os) {
|
||||
os << "mkapp";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_eval::to_stream(std::ostream& os) {
|
||||
os << "eval";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_op::to_stream(std::ostream& os) {
|
||||
os << "op(" << op_supercombinator(op) << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_cond::to_stream(std::ostream& os) {
|
||||
os << "cond";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_update::to_stream(std::ostream& os) {
|
||||
os << "update(" << offset << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_pack::to_stream(std::ostream& os) {
|
||||
os << "pack(" << constructor << ", " << arity << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_split::to_stream(std::ostream& os) {
|
||||
os << "split(" << arity << ")";
|
||||
return os;
|
||||
}
|
||||
|
||||
std::ostream& instruction_jump::to_stream(std::ostream& os) {
|
||||
os << "jump";
|
||||
return os;
|
||||
}
|
||||
}
|
131
src/gmachine.hpp
Normal file
131
src/gmachine.hpp
Normal file
|
@ -0,0 +1,131 @@
|
|||
#pragma once
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include "binop.hpp"
|
||||
|
||||
namespace lily {
|
||||
struct instruction {
|
||||
virtual std::ostream& to_stream(std::ostream& os) = 0;
|
||||
virtual ~instruction() = default;
|
||||
};
|
||||
|
||||
std::ostream& operator<<(std::ostream& os, instruction& inst);
|
||||
|
||||
struct instruction_slide : instruction {
|
||||
int amount;
|
||||
|
||||
instruction_slide(int a) : amount(a) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_alloc : instruction {
|
||||
int amount;
|
||||
|
||||
instruction_alloc(int a) : amount(a) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_pop : instruction {
|
||||
int amount;
|
||||
|
||||
instruction_pop(int a) : amount(a) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_unwind : instruction {
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_push_global : instruction {
|
||||
std::string name;
|
||||
|
||||
instruction_push_global(std::string n) : name(std::move(n)) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_push_int : instruction {
|
||||
int value;
|
||||
|
||||
instruction_push_int(int v) : value(v) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_push_str : instruction {
|
||||
std::string str;
|
||||
|
||||
instruction_push_str(std::string s) : str(std::move(s)) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_push : instruction {
|
||||
int offset;
|
||||
|
||||
instruction_push(int o) : offset(o) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_mkapp : instruction {
|
||||
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_eval : instruction {
|
||||
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_op : instruction {
|
||||
binop op;
|
||||
|
||||
instruction_op(binop o) : op(o) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_cond : instruction {
|
||||
std::vector<instruction*> true_branch;
|
||||
std::vector<instruction*> false_branch;
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_update : instruction {
|
||||
int offset;
|
||||
|
||||
instruction_update(int o) : offset(o) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_pack : instruction {
|
||||
int constructor;
|
||||
int arity;
|
||||
|
||||
instruction_pack(int c, int a) :
|
||||
constructor(c), arity(a) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_split : instruction {
|
||||
int arity;
|
||||
|
||||
instruction_split(int a) : arity(a) {}
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
struct instruction_jump : instruction {
|
||||
std::vector<std::vector<instruction*>> instructions;
|
||||
std::ostream& to_stream(std::ostream& os);
|
||||
};
|
||||
|
||||
class instruction_manager {
|
||||
private:
|
||||
std::vector<std::unique_ptr<instruction>> instructions;
|
||||
public:
|
||||
template <typename T, typename ... Ts>
|
||||
T* add_instruction(Ts ... ts) {
|
||||
auto new_inst = std::make_unique<T>(ts...);
|
||||
T* raw = new_inst.get();
|
||||
instructions.push_back(std::move(new_inst));
|
||||
return raw;
|
||||
}
|
||||
};
|
||||
|
||||
}
|
20
src/main.cpp
20
src/main.cpp
|
@ -1,17 +1,23 @@
|
|||
#include "ast.hpp"
|
||||
#include "pattern.hpp"
|
||||
#include "parser.hpp"
|
||||
#include "gmachine.hpp"
|
||||
|
||||
int main() {
|
||||
try {
|
||||
lily::parse(
|
||||
"data Bool = { True, False }\n"
|
||||
"data Color = { Red, Black }\n"
|
||||
"data IntList = { Nil, Cons(Int, IntList) }\n"
|
||||
lily::program_ptr prog = lily::parse(
|
||||
"defn other x y = { 3 }\n"
|
||||
"defn add x y = { x + y }\n"
|
||||
"defn ones = { Cons 1 ones }\n"
|
||||
"defn len l = { case l of { Nil -> { 0 } Cons(x, xs) -> { 1 + len xs } } }");
|
||||
"defn otherr x y = { let sum = { x + y } in { sum + sum } }\n"
|
||||
);
|
||||
std::map<std::string, std::vector<lily::instruction*>> into;
|
||||
lily::instruction_manager mgr;
|
||||
prog->compile(mgr, into);
|
||||
for(auto& pair : into) {
|
||||
std::cout << pair.first << std::endl;
|
||||
for(auto& op : pair.second) {
|
||||
std::cout << " " << *op << std::endl;
|
||||
}
|
||||
}
|
||||
} catch(lily::error& e) {
|
||||
std::cout << e.message << std::endl;
|
||||
}
|
||||
|
|
|
@ -51,8 +51,8 @@ namespace lily {
|
|||
pgs_tree* right = PGS_TREE_NT_CHILD(*mul, 2);
|
||||
pgs_tree* op = PGS_TREE_NT_CHILD(*mul, 1);
|
||||
|
||||
enum ast_op::op o =
|
||||
source[PGS_TREE_T_FROM(*op)] == '*' ? ast_op::op::times : ast_op::op::divide;
|
||||
enum binop o =
|
||||
source[PGS_TREE_T_FROM(*op)] == '*' ? binop::times : binop::divide;
|
||||
ast_ptr aleft = expr_mul(left, source);
|
||||
ast_ptr aright = expr_app(right, source);
|
||||
return ast_ptr(new ast_op(o, std::move(aleft), std::move(aright)));
|
||||
|
@ -67,8 +67,8 @@ namespace lily {
|
|||
pgs_tree* right = PGS_TREE_NT_CHILD(*add, 2);
|
||||
pgs_tree* op = PGS_TREE_NT_CHILD(*add, 1);
|
||||
|
||||
enum ast_op::op o =
|
||||
source[PGS_TREE_T_FROM(*op)] == '+' ? ast_op::op::add : ast_op::op::subtract;
|
||||
enum binop o =
|
||||
source[PGS_TREE_T_FROM(*op)] == '+' ? binop::add : binop::subtract;
|
||||
ast_ptr aleft = expr_add(left, source);
|
||||
ast_ptr aright = expr_mul(right, source);
|
||||
return ast_ptr(new ast_op(o, std::move(aleft), std::move(aright)));
|
||||
|
@ -290,4 +290,48 @@ namespace lily {
|
|||
throw error("unable to unify function type");
|
||||
}
|
||||
}
|
||||
|
||||
template <binop o>
|
||||
static void generate_binop(instruction_manager& mgr, std::map<std::string, std::vector<instruction*>>& into) {
|
||||
std::vector<instruction*> dest;
|
||||
dest.push_back(mgr.add_instruction<instruction_push>(1));
|
||||
dest.push_back(mgr.add_instruction<instruction_eval>());
|
||||
dest.push_back(mgr.add_instruction<instruction_push>(1));
|
||||
dest.push_back(mgr.add_instruction<instruction_eval>());
|
||||
dest.push_back(mgr.add_instruction<instruction_op>(o));
|
||||
dest.push_back(mgr.add_instruction<instruction_update>(2));
|
||||
dest.push_back(mgr.add_instruction<instruction_pop>(2));
|
||||
dest.push_back(mgr.add_instruction<instruction_unwind>());
|
||||
into[op_supercombinator(o)] = std::move(dest);
|
||||
}
|
||||
|
||||
static void register_internal(instruction_manager& mgr, std::map<std::string, std::vector<instruction*>>& into) {
|
||||
generate_binop<binop::add>(mgr, into);
|
||||
generate_binop<binop::subtract>(mgr, into);
|
||||
generate_binop<binop::times>(mgr, into);
|
||||
generate_binop<binop::divide>(mgr, into);
|
||||
}
|
||||
|
||||
void program::compile(instruction_manager& mgr, std::map<std::string, std::vector<instruction*>>& into) {
|
||||
register_internal(mgr, into);
|
||||
for(auto& pair : functions) {
|
||||
std::shared_ptr<compile_env> fresh_env = std::make_shared<compile_env_offset>(0);
|
||||
size_t count = pair.second.params.size();
|
||||
for(size_t i = 0; i < count; i++) {
|
||||
auto new_env = std::make_shared<compile_env_var>(pair.second.params[count - i - 1]);
|
||||
new_env->set_parent(fresh_env);
|
||||
fresh_env = new_env;
|
||||
}
|
||||
auto new_env = std::make_shared<compile_env_offset>(1);
|
||||
new_env->set_parent(fresh_env);
|
||||
fresh_env = new_env;
|
||||
|
||||
std::vector<instruction*> destination;
|
||||
pair.second.body->compile(mgr, destination, fresh_env);
|
||||
destination.push_back(mgr.add_instruction<instruction_update>(count));
|
||||
destination.push_back(mgr.add_instruction<instruction_pop>(count));
|
||||
destination.push_back(mgr.add_instruction<instruction_unwind>());
|
||||
into[pair.first] = std::move(destination);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
#include "function.hpp"
|
||||
#include "type.hpp"
|
||||
#include "type_manager.hpp"
|
||||
#include "gmachine.hpp"
|
||||
|
||||
namespace lily {
|
||||
struct program {
|
||||
|
@ -13,6 +14,7 @@ namespace lily {
|
|||
std::map<std::string, function> functions;
|
||||
|
||||
void check();
|
||||
void compile(instruction_manager& mgr, std::map<std::string, std::vector<instruction*>>& into);
|
||||
};
|
||||
|
||||
typedef std::unique_ptr<program> program_ptr;
|
||||
|
|
Loading…
Reference in New Issue
Block a user