| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  | #include "ast.hpp"
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include "binop.hpp"
 | 
					
						
							|  |  |  | #include "definition.hpp"
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  | #include "graph.hpp"
 | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  | #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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  | extern std::map<std::string, definition_data_ptr> defs_data; | 
					
						
							|  |  |  | extern std::map<std::string, definition_defn_ptr> defs_defn; | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | void typecheck_program( | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         const std::map<std::string, definition_data_ptr>& defs_data, | 
					
						
							|  |  |  |         const std::map<std::string, definition_defn_ptr>& defs_defn, | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |         type_mgr& mgr, type_env_ptr& env) { | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     type_ptr int_type = type_ptr(new type_base("Int"));  | 
					
						
							|  |  |  |     type_ptr binop_type = type_ptr(new type_arr( | 
					
						
							|  |  |  |                 int_type, | 
					
						
							|  |  |  |                 type_ptr(new type_arr(int_type, int_type)))); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     env->bind("+", binop_type); | 
					
						
							|  |  |  |     env->bind("-", binop_type); | 
					
						
							|  |  |  |     env->bind("*", binop_type); | 
					
						
							|  |  |  |     env->bind("/", binop_type); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_data : defs_data) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_data.second->insert_types(mgr, env); | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     for(auto& def_data : defs_data) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_data.second->insert_constructors(); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |     function_graph dependency_graph; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_defn : defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         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); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& pair : env->names) { | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |         std::cout << pair.first << ": "; | 
					
						
							|  |  |  |         pair.second->print(mgr, std::cout); | 
					
						
							|  |  |  |         std::cout << std::endl; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  | void compile_program(const std::map<std::string, definition_defn_ptr>& defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_defn : defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_defn.second->compile(); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         for(auto& instruction : def_defn.second->instructions) { | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |             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::TargetMachine::CodeGenFileType type = llvm::TargetMachine::CGFT_ObjectFile; | 
					
						
							|  |  |  |             llvm::legacy::PassManager pm; | 
					
						
							|  |  |  |             if (targetMachine->addPassesToEmitFile(pm, file, NULL, type)) { | 
					
						
							|  |  |  |                 throw 0; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 pm.run(ctx.module); | 
					
						
							|  |  |  |                 file.close(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  | void gen_llvm( | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         const std::map<std::string, definition_data_ptr>& defs_data, | 
					
						
							|  |  |  |         const std::map<std::string, definition_defn_ptr>& defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_data : defs_data) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_data.second->generate_llvm(ctx); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_defn : defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_defn.second->declare_llvm(ctx); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_defn : defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_defn.second->generate_llvm(ctx); | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     ctx.module.print(llvm::outs(), nullptr); | 
					
						
							|  |  |  |     output_llvm(ctx, "program.o"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int main() { | 
					
						
							|  |  |  |     yy::parser parser; | 
					
						
							|  |  |  |     type_mgr mgr; | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     type_env_ptr env(new type_env); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     parser.parse(); | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |     for(auto& def_defn : defs_defn) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         std::cout << def_defn.second->name; | 
					
						
							|  |  |  |         for(auto& param : def_defn.second->params) std::cout << " " << param; | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |         std::cout << ":" << std::endl; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:00:09 -07:00
										 |  |  |         def_defn.second->body->print(1, std::cout); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     try { | 
					
						
							| 
									
										
										
										
											2020-03-24 21:08:02 -07:00
										 |  |  |         typecheck_program(defs_data, defs_defn, mgr, env); | 
					
						
							|  |  |  |         compile_program(defs_defn); | 
					
						
							|  |  |  |         gen_llvm(defs_data, defs_defn); | 
					
						
							| 
									
										
										
										
											2020-03-10 20:58:26 -07:00
										 |  |  |     } 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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |