diff options
| author | Patrick Walton <[email protected]> | 2011-05-05 10:48:02 -0700 |
|---|---|---|
| committer | Patrick Walton <[email protected]> | 2011-05-05 10:48:02 -0700 |
| commit | fea623211fafb2ce719eeb9ed47cdb9624e1fbe8 (patch) | |
| tree | 6384aa2efacd0c81f293677e84d24516f568b536 /src/comp/back | |
| parent | rustllvm: Add bindings to the LLVM linker (diff) | |
| download | rust-fea623211fafb2ce719eeb9ed47cdb9624e1fbe8.tar.xz rust-fea623211fafb2ce719eeb9ed47cdb9624e1fbe8.zip | |
rustc: Add a Link module; move crate writing to it to slim down trans slightly
Diffstat (limited to 'src/comp/back')
| -rw-r--r-- | src/comp/back/Link.rs | 176 |
1 files changed, 176 insertions, 0 deletions
diff --git a/src/comp/back/Link.rs b/src/comp/back/Link.rs new file mode 100644 index 00000000..5d9eb9d6 --- /dev/null +++ b/src/comp/back/Link.rs @@ -0,0 +1,176 @@ +import driver.session; +import lib.llvm.llvm; +import middle.trans; +import std._str; + +import lib.llvm.llvm.ModuleRef; +import lib.llvm.llvm.ValueRef; +import lib.llvm.mk_pass_manager; +import lib.llvm.mk_target_data; +import lib.llvm.mk_type_names; + +tag output_type { + output_type_none; + output_type_bitcode; + output_type_assembly; + output_type_object; +} + +mod Write { + fn is_object_or_assembly(output_type ot) -> bool { + if (ot == output_type_assembly) { + ret true; + } + if (ot == output_type_object) { + ret true; + } + ret false; + } + + // Decides what to call an intermediate file, given the name of the output + // and the extension to use. + fn mk_intermediate_name(str output_path, str extension) -> str { + auto dot_pos = _str.index(output_path, '.' as u8); + auto stem; + if (dot_pos < 0) { + stem = output_path; + } else { + stem = _str.substr(output_path, 0u, dot_pos as uint); + } + ret stem + "." + extension; + } + + fn run_passes(session.session sess, ModuleRef llmod, str output) { + auto pm = mk_pass_manager(); + auto opts = sess.get_opts(); + + // TODO: run the linter here also, once there are llvm-c bindings for + // it. + + // Generate a pre-optimization intermediate file if -save-temps was + // specified. + if (opts.save_temps) { + alt (opts.output_type) { + case (output_type_bitcode) { + if (opts.optimize) { + auto filename = mk_intermediate_name(output, + "no-opt.bc"); + llvm.LLVMWriteBitcodeToFile(llmod, + _str.buf(filename)); + } + } + case (_) { + auto filename = mk_intermediate_name(output, "bc"); + llvm.LLVMWriteBitcodeToFile(llmod, _str.buf(filename)); + } + } + } + + // FIXME: This is mostly a copy of the bits of opt's -O2 that are + // available in the C api. + // FIXME2: We might want to add optimization levels like -O1, -O2, + // -Os, etc + // FIXME3: Should we expose and use the pass lists used by the opt + // tool? + if (opts.optimize) { + auto fpm = mk_pass_manager(); + + // createStandardFunctionPasses + llvm.LLVMAddTypeBasedAliasAnalysisPass(fpm.llpm); + llvm.LLVMAddBasicAliasAnalysisPass(fpm.llpm); + llvm.LLVMAddCFGSimplificationPass(fpm.llpm); + llvm.LLVMAddScalarReplAggregatesPass(fpm.llpm); + llvm.LLVMAddEarlyCSEPass(fpm.llpm); + + llvm.LLVMRunPassManager(fpm.llpm, llmod); + + // createStandardModulePasses + llvm.LLVMAddTypeBasedAliasAnalysisPass(pm.llpm); + llvm.LLVMAddBasicAliasAnalysisPass(pm.llpm); + llvm.LLVMAddGlobalOptimizerPass(pm.llpm); + llvm.LLVMAddIPSCCPPass(pm.llpm); + llvm.LLVMAddDeadArgEliminationPass(pm.llpm); + llvm.LLVMAddInstructionCombiningPass(pm.llpm); + llvm.LLVMAddCFGSimplificationPass(pm.llpm); + llvm.LLVMAddPruneEHPass(pm.llpm); + llvm.LLVMAddFunctionInliningPass(pm.llpm); + llvm.LLVMAddFunctionAttrsPass(pm.llpm); + llvm.LLVMAddScalarReplAggregatesPassSSA(pm.llpm); + llvm.LLVMAddEarlyCSEPass(pm.llpm); + llvm.LLVMAddSimplifyLibCallsPass(pm.llpm); + llvm.LLVMAddJumpThreadingPass(pm.llpm); + llvm.LLVMAddCorrelatedValuePropagationPass(pm.llpm); + llvm.LLVMAddCFGSimplificationPass(pm.llpm); + llvm.LLVMAddInstructionCombiningPass(pm.llpm); + llvm.LLVMAddTailCallEliminationPass(pm.llpm); + llvm.LLVMAddCFGSimplificationPass(pm.llpm); + llvm.LLVMAddReassociatePass(pm.llpm); + llvm.LLVMAddLoopRotatePass(pm.llpm); + llvm.LLVMAddLICMPass(pm.llpm); + llvm.LLVMAddLoopUnswitchPass(pm.llpm); + llvm.LLVMAddInstructionCombiningPass(pm.llpm); + llvm.LLVMAddIndVarSimplifyPass(pm.llpm); + llvm.LLVMAddLoopIdiomPass(pm.llpm); + llvm.LLVMAddLoopDeletionPass(pm.llpm); + llvm.LLVMAddLoopUnrollPass(pm.llpm); + llvm.LLVMAddInstructionCombiningPass(pm.llpm); + llvm.LLVMAddGVNPass(pm.llpm); + llvm.LLVMAddMemCpyOptPass(pm.llpm); + llvm.LLVMAddSCCPPass(pm.llpm); + llvm.LLVMAddInstructionCombiningPass(pm.llpm); + llvm.LLVMAddJumpThreadingPass(pm.llpm); + llvm.LLVMAddCorrelatedValuePropagationPass(pm.llpm); + llvm.LLVMAddDeadStoreEliminationPass(pm.llpm); + llvm.LLVMAddAggressiveDCEPass(pm.llpm); + llvm.LLVMAddCFGSimplificationPass(pm.llpm); + llvm.LLVMAddStripDeadPrototypesPass(pm.llpm); + llvm.LLVMAddDeadTypeEliminationPass(pm.llpm); + llvm.LLVMAddConstantMergePass(pm.llpm); + } + + if (opts.verify) { + llvm.LLVMAddVerifierPass(pm.llpm); + } + + // TODO: Write .s if -c was specified and -save-temps was on. + if (is_object_or_assembly(opts.output_type)) { + let int LLVMAssemblyFile = 0; + let int LLVMObjectFile = 1; + let int LLVMNullFile = 2; + auto FileType; + if (opts.output_type == output_type_object) { + FileType = LLVMObjectFile; + } else { + FileType = LLVMAssemblyFile; + } + + // Write optimized bitcode if --save-temps was on. + if (opts.save_temps) { + alt (opts.output_type) { + case (output_type_bitcode) { /* nothing to do */ } + case (_) { + auto filename = mk_intermediate_name(output, + "opt.bc"); + llvm.LLVMRunPassManager(pm.llpm, llmod); + llvm.LLVMWriteBitcodeToFile(llmod, + _str.buf(filename)); + pm = mk_pass_manager(); + } + } + } + + llvm.LLVMRustWriteOutputFile(pm.llpm, llmod, + _str.buf(x86.get_target_triple()), + _str.buf(output), + FileType); + llvm.LLVMDisposeModule(llmod); + ret; + } + + llvm.LLVMRunPassManager(pm.llpm, llmod); + + llvm.LLVMWriteBitcodeToFile(llmod, _str.buf(output)); + llvm.LLVMDisposeModule(llmod); + } +} + |