#include "ast.hpp" #include <iostream> #include "binop.hpp" #include "definition.hpp" #include "graph.hpp" #include "instruction.hpp" #include "llvm_context.hpp" #include "parser.hpp" #include "error.hpp" #include "type.hpp" #include "llvm/IR/LegacyPassManager.h" #include "llvm/IR/Verifier.h" #include "llvm/Support/TargetSelect.h" #include "llvm/Support/TargetRegistry.h" #include "llvm/Support/raw_ostream.h" #include "llvm/Support/FileSystem.h" #include "llvm/Target/TargetOptions.h" #include "llvm/Target/TargetMachine.h" void yy::parser::error(const std::string& msg) { std::cout << "An error occured: " << msg << std::endl; } extern std::map<std::string, definition_data_ptr> defs_data; extern std::map<std::string, definition_defn_ptr> defs_defn; void typecheck_program( const std::map<std::string, definition_data_ptr>& defs_data, const std::map<std::string, definition_defn_ptr>& defs_defn, type_mgr& mgr, type_env_ptr& env) { type_ptr int_type = type_ptr(new type_base("Int")); env->bind_type("Int", int_type); type_ptr int_type_app = type_ptr(new type_app(int_type)); type_ptr binop_type = type_ptr(new type_arr( int_type_app, type_ptr(new type_arr(int_type_app, int_type_app)))); env->bind("+", binop_type); env->bind("-", binop_type); env->bind("*", binop_type); env->bind("/", binop_type); for(auto& def_data : defs_data) { def_data.second->insert_types(env); } for(auto& def_data : defs_data) { def_data.second->insert_constructors(); } function_graph dependency_graph; for(auto& def_defn : defs_defn) { def_defn.second->find_free(mgr, env); dependency_graph.add_function(def_defn.second->name); for(auto& dependency : def_defn.second->free_variables) { if(defs_defn.find(dependency) == defs_defn.end()) throw 0; dependency_graph.add_edge(def_defn.second->name, dependency); } } std::vector<group_ptr> groups = dependency_graph.compute_order(); for(auto it = groups.rbegin(); it != groups.rend(); it++) { auto& group = *it; for(auto& def_defnn_name : group->members) { auto& def_defn = defs_defn.find(def_defnn_name)->second; def_defn->insert_types(mgr); } for(auto& def_defnn_name : group->members) { auto& def_defn = defs_defn.find(def_defnn_name)->second; def_defn->typecheck(mgr); } for(auto& def_defnn_name : group->members) { env->generalize(def_defnn_name, mgr); } } for(auto& pair : env->names) { std::cout << pair.first << ": "; pair.second->print(mgr, std::cout); std::cout << std::endl; } } void compile_program(const std::map<std::string, definition_defn_ptr>& defs_defn) { for(auto& def_defn : defs_defn) { def_defn.second->compile(); for(auto& instruction : def_defn.second->instructions) { instruction->print(0, std::cout); } std::cout << std::endl; } } void gen_llvm_internal_op(llvm_context& ctx, binop op) { auto new_function = ctx.create_custom_function(op_action(op), 2); std::vector<instruction_ptr> instructions; instructions.push_back(instruction_ptr(new instruction_push(1))); instructions.push_back(instruction_ptr(new instruction_eval())); instructions.push_back(instruction_ptr(new instruction_push(1))); instructions.push_back(instruction_ptr(new instruction_eval())); instructions.push_back(instruction_ptr(new instruction_binop(op))); instructions.push_back(instruction_ptr(new instruction_update(2))); instructions.push_back(instruction_ptr(new instruction_pop(2))); ctx.builder.SetInsertPoint(&new_function->getEntryBlock()); for(auto& instruction : instructions) { instruction->gen_llvm(ctx, new_function); } ctx.builder.CreateRetVoid(); } void output_llvm(llvm_context& ctx, const std::string& filename) { std::string targetTriple = llvm::sys::getDefaultTargetTriple(); llvm::InitializeNativeTarget(); llvm::InitializeNativeTargetAsmParser(); llvm::InitializeNativeTargetAsmPrinter(); std::string error; const llvm::Target* target = llvm::TargetRegistry::lookupTarget(targetTriple, error); if (!target) { std::cerr << error << std::endl; } else { std::string cpu = "generic"; std::string features = ""; llvm::TargetOptions options; llvm::TargetMachine* targetMachine = target->createTargetMachine(targetTriple, cpu, features, options, llvm::Optional<llvm::Reloc::Model>()); ctx.module.setDataLayout(targetMachine->createDataLayout()); ctx.module.setTargetTriple(targetTriple); std::error_code ec; llvm::raw_fd_ostream file(filename, ec, llvm::sys::fs::F_None); if (ec) { throw 0; } else { llvm::CodeGenFileType type = llvm::CGFT_ObjectFile; llvm::legacy::PassManager pm; if (targetMachine->addPassesToEmitFile(pm, file, NULL, type)) { throw 0; } else { pm.run(ctx.module); file.close(); } } } } void gen_llvm( const std::map<std::string, definition_data_ptr>& defs_data, const std::map<std::string, definition_defn_ptr>& defs_defn) { llvm_context ctx; gen_llvm_internal_op(ctx, PLUS); gen_llvm_internal_op(ctx, MINUS); gen_llvm_internal_op(ctx, TIMES); gen_llvm_internal_op(ctx, DIVIDE); for(auto& def_data : defs_data) { def_data.second->generate_llvm(ctx); } for(auto& def_defn : defs_defn) { def_defn.second->declare_llvm(ctx); } for(auto& def_defn : defs_defn) { def_defn.second->generate_llvm(ctx); } ctx.module.print(llvm::outs(), nullptr); output_llvm(ctx, "program.o"); } int main() { yy::parser parser; type_mgr mgr; type_env_ptr env(new type_env); parser.parse(); for(auto& def_defn : defs_defn) { std::cout << def_defn.second->name; for(auto& param : def_defn.second->params) std::cout << " " << param; std::cout << ":" << std::endl; def_defn.second->body->print(1, std::cout); } try { typecheck_program(defs_data, defs_defn, mgr, env); compile_program(defs_defn); gen_llvm(defs_data, defs_defn); } catch(unification_error& err) { std::cout << "failed to unify types: " << std::endl; std::cout << " (1) \033[34m"; err.left->print(mgr, std::cout); std::cout << "\033[0m" << std::endl; std::cout << " (2) \033[32m"; err.right->print(mgr, std::cout); std::cout << "\033[0m" << std::endl; } catch(type_error& err) { std::cout << "failed to type check program: " << err.description << std::endl; } }