Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 1 | //===-lto.cpp - LLVM Link Time Optimizer ----------------------------------===// |
| 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 7 | // |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 10 | // This file implements the Link Time Optimization library. This library is |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 11 | // intended to be used by linker to optimize code at link time. |
| 12 | // |
| 13 | //===----------------------------------------------------------------------===// |
| 14 | |
| 15 | #include "llvm-c/lto.h" |
Benjamin Kramer | 0a446fd | 2015-03-01 21:28:53 +0000 | [diff] [blame] | 16 | #include "llvm/ADT/STLExtras.h" |
Rafael Espindola | 0b385c7 | 2013-09-30 16:39:19 +0000 | [diff] [blame] | 17 | #include "llvm/CodeGen/CommandFlags.h" |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 18 | #include "llvm/IR/DiagnosticInfo.h" |
| 19 | #include "llvm/IR/DiagnosticPrinter.h" |
Duncan P. N. Exon Smith | d34b613 | 2014-12-19 07:19:50 +0000 | [diff] [blame] | 20 | #include "llvm/IR/LLVMContext.h" |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 21 | #include "llvm/LTO/LTOCodeGenerator.h" |
| 22 | #include "llvm/LTO/LTOModule.h" |
Alp Toker | ac90380 | 2014-07-04 00:58:41 +0000 | [diff] [blame] | 23 | #include "llvm/Support/MemoryBuffer.h" |
Michael J. Spencer | 50a20c0 | 2015-01-29 17:20:41 +0000 | [diff] [blame] | 24 | #include "llvm/Support/Signals.h" |
Rafael Espindola | 77c50d2 | 2014-06-19 19:11:22 +0000 | [diff] [blame] | 25 | #include "llvm/Support/TargetSelect.h" |
Reid Kleckner | 7c6b5e1 | 2016-01-29 00:03:34 +0000 | [diff] [blame^] | 26 | #include "llvm/Support/raw_ostream.h" |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 27 | |
Rafael Espindola | 0b385c7 | 2013-09-30 16:39:19 +0000 | [diff] [blame] | 28 | // extra command-line flags needed for LTOCodeGenerator |
Peter Collingbourne | 070843d | 2015-03-19 22:01:00 +0000 | [diff] [blame] | 29 | static cl::opt<char> |
| 30 | OptLevel("O", |
| 31 | cl::desc("Optimization level. [-O0, -O1, -O2, or -O3] " |
| 32 | "(default = '-O2')"), |
| 33 | cl::Prefix, |
| 34 | cl::ZeroOrMore, |
| 35 | cl::init('2')); |
Rafael Espindola | 0b385c7 | 2013-09-30 16:39:19 +0000 | [diff] [blame] | 36 | |
| 37 | static cl::opt<bool> |
| 38 | DisableInline("disable-inlining", cl::init(false), |
| 39 | cl::desc("Do not run the inliner pass")); |
| 40 | |
| 41 | static cl::opt<bool> |
| 42 | DisableGVNLoadPRE("disable-gvn-loadpre", cl::init(false), |
| 43 | cl::desc("Do not run the GVN load PRE pass")); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 44 | |
Arnold Schwaighofer | eb1a38f | 2014-10-26 21:50:58 +0000 | [diff] [blame] | 45 | static cl::opt<bool> |
| 46 | DisableLTOVectorization("disable-lto-vectorization", cl::init(false), |
| 47 | cl::desc("Do not run loop or slp vectorization during LTO")); |
| 48 | |
Duncan P. N. Exon Smith | cff5fef | 2015-09-15 23:05:59 +0000 | [diff] [blame] | 49 | #ifdef NDEBUG |
| 50 | static bool VerifyByDefault = false; |
| 51 | #else |
| 52 | static bool VerifyByDefault = true; |
| 53 | #endif |
| 54 | |
| 55 | static cl::opt<bool> DisableVerify( |
| 56 | "disable-llvm-verifier", cl::init(!VerifyByDefault), |
| 57 | cl::desc("Don't run the LLVM verifier during the optimization pipeline")); |
| 58 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 59 | // Holds most recent error string. |
| 60 | // *** Not thread safe *** |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 61 | static std::string sLastErrorString; |
| 62 | |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 63 | // Holds the initialization state of the LTO module. |
| 64 | // *** Not thread safe *** |
| 65 | static bool initialized = false; |
| 66 | |
Rafael Espindola | efa02d5 | 2013-10-02 14:36:23 +0000 | [diff] [blame] | 67 | // Holds the command-line option parsing state of the LTO module. |
| 68 | static bool parsedOptions = false; |
| 69 | |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 70 | static LLVMContext *LTOContext = nullptr; |
| 71 | |
| 72 | static void diagnosticHandler(const DiagnosticInfo &DI, void *Context) { |
| 73 | if (DI.getSeverity() != DS_Error) { |
| 74 | DiagnosticPrinterRawOStream DP(errs()); |
| 75 | DI.print(DP); |
| 76 | errs() << '\n'; |
| 77 | return; |
| 78 | } |
| 79 | sLastErrorString = ""; |
| 80 | { |
| 81 | raw_string_ostream Stream(sLastErrorString); |
| 82 | DiagnosticPrinterRawOStream DP(Stream); |
| 83 | DI.print(DP); |
| 84 | } |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 85 | } |
| 86 | |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 87 | // Initialize the configured targets if they have not been initialized. |
| 88 | static void lto_initialize() { |
| 89 | if (!initialized) { |
Michael J. Spencer | 50a20c0 | 2015-01-29 17:20:41 +0000 | [diff] [blame] | 90 | #ifdef LLVM_ON_WIN32 |
| 91 | // Dialog box on crash disabling doesn't work across DLL boundaries, so do |
| 92 | // it here. |
| 93 | llvm::sys::DisableSystemDialogsOnCrash(); |
| 94 | #endif |
| 95 | |
Rafael Espindola | 77c50d2 | 2014-06-19 19:11:22 +0000 | [diff] [blame] | 96 | InitializeAllTargetInfos(); |
| 97 | InitializeAllTargets(); |
| 98 | InitializeAllTargetMCs(); |
| 99 | InitializeAllAsmParsers(); |
| 100 | InitializeAllAsmPrinters(); |
| 101 | InitializeAllDisassemblers(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 102 | |
| 103 | LTOContext = &getGlobalContext(); |
| 104 | LTOContext->setDiagnosticHandler(diagnosticHandler, nullptr, true); |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 105 | initialized = true; |
| 106 | } |
| 107 | } |
| 108 | |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 109 | namespace { |
| 110 | |
Yunzhong Gao | ea7b3a2 | 2015-11-11 19:59:08 +0000 | [diff] [blame] | 111 | static void handleLibLTODiagnostic(lto_codegen_diagnostic_severity_t Severity, |
| 112 | const char *Msg, void *) { |
| 113 | sLastErrorString = Msg; |
Yunzhong Gao | ea7b3a2 | 2015-11-11 19:59:08 +0000 | [diff] [blame] | 114 | } |
| 115 | |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 116 | // This derived class owns the native object file. This helps implement the |
| 117 | // libLTO API semantics, which require that the code generator owns the object |
| 118 | // file. |
| 119 | struct LibLTOCodeGenerator : LTOCodeGenerator { |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 120 | LibLTOCodeGenerator() : LTOCodeGenerator(*LTOContext) { |
Yunzhong Gao | ea7b3a2 | 2015-11-11 19:59:08 +0000 | [diff] [blame] | 121 | setDiagnosticHandler(handleLibLTODiagnostic, nullptr); } |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 122 | LibLTOCodeGenerator(std::unique_ptr<LLVMContext> Context) |
Rafael Espindola | 7b8a24e | 2015-12-04 02:42:28 +0000 | [diff] [blame] | 123 | : LTOCodeGenerator(*Context), OwnedContext(std::move(Context)) { |
Yunzhong Gao | ea7b3a2 | 2015-11-11 19:59:08 +0000 | [diff] [blame] | 124 | setDiagnosticHandler(handleLibLTODiagnostic, nullptr); } |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 125 | |
Steven Wu | b5104b5 | 2015-12-09 03:37:51 +0000 | [diff] [blame] | 126 | // Reset the module first in case MergedModule is created in OwnedContext. |
| 127 | // Module must be destructed before its context gets destructed. |
| 128 | ~LibLTOCodeGenerator() { resetMergedModule(); } |
| 129 | |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 130 | std::unique_ptr<MemoryBuffer> NativeObjectFile; |
Rafael Espindola | 7b8a24e | 2015-12-04 02:42:28 +0000 | [diff] [blame] | 131 | std::unique_ptr<LLVMContext> OwnedContext; |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 132 | }; |
| 133 | |
| 134 | } |
| 135 | |
| 136 | DEFINE_SIMPLE_CONVERSION_FUNCTIONS(LibLTOCodeGenerator, lto_code_gen_t) |
Patrik Hagglund | 9be9d87 | 2014-05-05 12:24:08 +0000 | [diff] [blame] | 137 | DEFINE_SIMPLE_CONVERSION_FUNCTIONS(LTOModule, lto_module_t) |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 138 | |
Tom Roeder | fd1bc60 | 2014-04-25 21:46:51 +0000 | [diff] [blame] | 139 | // Convert the subtarget features into a string to pass to LTOCodeGenerator. |
| 140 | static void lto_add_attrs(lto_code_gen_t cg) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 141 | LTOCodeGenerator *CG = unwrap(cg); |
Tom Roeder | fd1bc60 | 2014-04-25 21:46:51 +0000 | [diff] [blame] | 142 | if (MAttrs.size()) { |
| 143 | std::string attrs; |
| 144 | for (unsigned i = 0; i < MAttrs.size(); ++i) { |
| 145 | if (i > 0) |
| 146 | attrs.append(","); |
| 147 | attrs.append(MAttrs[i]); |
| 148 | } |
| 149 | |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 150 | CG->setAttr(attrs.c_str()); |
Tom Roeder | fd1bc60 | 2014-04-25 21:46:51 +0000 | [diff] [blame] | 151 | } |
Peter Collingbourne | 070843d | 2015-03-19 22:01:00 +0000 | [diff] [blame] | 152 | |
| 153 | if (OptLevel < '0' || OptLevel > '3') |
| 154 | report_fatal_error("Optimization level must be between 0 and 3"); |
| 155 | CG->setOptLevel(OptLevel - '0'); |
Tom Roeder | fd1bc60 | 2014-04-25 21:46:51 +0000 | [diff] [blame] | 156 | } |
| 157 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 158 | extern const char* lto_get_version() { |
| 159 | return LTOCodeGenerator::getVersionString(); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 160 | } |
| 161 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 162 | const char* lto_get_error_message() { |
| 163 | return sLastErrorString.c_str(); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 164 | } |
| 165 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 166 | bool lto_module_is_object_file(const char* path) { |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 167 | return LTOModule::isBitcodeFile(path); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 168 | } |
| 169 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 170 | bool lto_module_is_object_file_for_target(const char* path, |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 171 | const char* target_triplet_prefix) { |
Rafael Espindola | adf21f2 | 2014-07-06 17:43:13 +0000 | [diff] [blame] | 172 | ErrorOr<std::unique_ptr<MemoryBuffer>> Buffer = MemoryBuffer::getFile(path); |
| 173 | if (!Buffer) |
Alp Toker | ac90380 | 2014-07-04 00:58:41 +0000 | [diff] [blame] | 174 | return false; |
Rafael Espindola | adf21f2 | 2014-07-06 17:43:13 +0000 | [diff] [blame] | 175 | return LTOModule::isBitcodeForTarget(Buffer->get(), target_triplet_prefix); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 176 | } |
| 177 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 178 | bool lto_module_is_object_file_in_memory(const void* mem, size_t length) { |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 179 | return LTOModule::isBitcodeFile(mem, length); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 180 | } |
| 181 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 182 | bool |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 183 | lto_module_is_object_file_in_memory_for_target(const void* mem, |
| 184 | size_t length, |
| 185 | const char* target_triplet_prefix) { |
Alp Toker | ac90380 | 2014-07-04 00:58:41 +0000 | [diff] [blame] | 186 | std::unique_ptr<MemoryBuffer> buffer(LTOModule::makeBuffer(mem, length)); |
| 187 | if (!buffer) |
| 188 | return false; |
| 189 | return LTOModule::isBitcodeForTarget(buffer.get(), target_triplet_prefix); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 190 | } |
| 191 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 192 | lto_module_t lto_module_create(const char* path) { |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 193 | lto_initialize(); |
Eli Bendersky | f0f2100 | 2014-02-19 17:09:35 +0000 | [diff] [blame] | 194 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 195 | ErrorOr<std::unique_ptr<LTOModule>> M = |
| 196 | LTOModule::createFromFile(*LTOContext, path, Options); |
| 197 | if (!M) |
| 198 | return nullptr; |
| 199 | return wrap(M->release()); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 200 | } |
| 201 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 202 | lto_module_t lto_module_create_from_fd(int fd, const char *path, size_t size) { |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 203 | lto_initialize(); |
Eli Bendersky | f0f2100 | 2014-02-19 17:09:35 +0000 | [diff] [blame] | 204 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 205 | ErrorOr<std::unique_ptr<LTOModule>> M = |
| 206 | LTOModule::createFromOpenFile(*LTOContext, fd, path, size, Options); |
| 207 | if (!M) |
| 208 | return nullptr; |
| 209 | return wrap(M->release()); |
Rafael Espindola | 56e41f7 | 2011-02-08 22:40:47 +0000 | [diff] [blame] | 210 | } |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 211 | |
Rafael Espindola | b39c7c7 | 2011-03-17 00:36:11 +0000 | [diff] [blame] | 212 | lto_module_t lto_module_create_from_fd_at_offset(int fd, const char *path, |
| 213 | size_t file_size, |
| 214 | size_t map_size, |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 215 | off_t offset) { |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 216 | lto_initialize(); |
Eli Bendersky | f0f2100 | 2014-02-19 17:09:35 +0000 | [diff] [blame] | 217 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 218 | ErrorOr<std::unique_ptr<LTOModule>> M = LTOModule::createFromOpenFileSlice( |
| 219 | *LTOContext, fd, path, map_size, offset, Options); |
| 220 | if (!M) |
| 221 | return nullptr; |
| 222 | return wrap(M->release()); |
Rafael Espindola | b39c7c7 | 2011-03-17 00:36:11 +0000 | [diff] [blame] | 223 | } |
| 224 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 225 | lto_module_t lto_module_create_from_memory(const void* mem, size_t length) { |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 226 | lto_initialize(); |
Eli Bendersky | f0f2100 | 2014-02-19 17:09:35 +0000 | [diff] [blame] | 227 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 228 | ErrorOr<std::unique_ptr<LTOModule>> M = |
| 229 | LTOModule::createFromBuffer(*LTOContext, mem, length, Options); |
| 230 | if (!M) |
| 231 | return nullptr; |
| 232 | return wrap(M->release()); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 233 | } |
| 234 | |
Manman Ren | 03456a1 | 2014-02-10 23:26:14 +0000 | [diff] [blame] | 235 | lto_module_t lto_module_create_from_memory_with_path(const void* mem, |
| 236 | size_t length, |
| 237 | const char *path) { |
| 238 | lto_initialize(); |
Eli Bendersky | f0f2100 | 2014-02-19 17:09:35 +0000 | [diff] [blame] | 239 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 240 | ErrorOr<std::unique_ptr<LTOModule>> M = |
| 241 | LTOModule::createFromBuffer(*LTOContext, mem, length, Options, path); |
| 242 | if (!M) |
| 243 | return nullptr; |
| 244 | return wrap(M->release()); |
Manman Ren | 03456a1 | 2014-02-10 23:26:14 +0000 | [diff] [blame] | 245 | } |
| 246 | |
Duncan P. N. Exon Smith | c5800f6 | 2014-11-11 23:19:23 +0000 | [diff] [blame] | 247 | lto_module_t lto_module_create_in_local_context(const void *mem, size_t length, |
| 248 | const char *path) { |
| 249 | lto_initialize(); |
| 250 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 251 | ErrorOr<std::unique_ptr<LTOModule>> M = |
| 252 | LTOModule::createInLocalContext(mem, length, Options, path); |
| 253 | if (!M) |
| 254 | return nullptr; |
| 255 | return wrap(M->release()); |
Duncan P. N. Exon Smith | c5800f6 | 2014-11-11 23:19:23 +0000 | [diff] [blame] | 256 | } |
| 257 | |
| 258 | lto_module_t lto_module_create_in_codegen_context(const void *mem, |
| 259 | size_t length, |
| 260 | const char *path, |
| 261 | lto_code_gen_t cg) { |
| 262 | lto_initialize(); |
| 263 | llvm::TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | a7612b4 | 2015-12-04 16:14:31 +0000 | [diff] [blame] | 264 | ErrorOr<std::unique_ptr<LTOModule>> M = LTOModule::createInContext( |
| 265 | mem, length, Options, path, &unwrap(cg)->getContext()); |
| 266 | return wrap(M->release()); |
Duncan P. N. Exon Smith | c5800f6 | 2014-11-11 23:19:23 +0000 | [diff] [blame] | 267 | } |
| 268 | |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 269 | void lto_module_dispose(lto_module_t mod) { delete unwrap(mod); } |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 270 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 271 | const char* lto_module_get_target_triple(lto_module_t mod) { |
Rafael Espindola | d749fb5 | 2014-07-04 14:19:41 +0000 | [diff] [blame] | 272 | return unwrap(mod)->getTargetTriple().c_str(); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 273 | } |
| 274 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 275 | void lto_module_set_target_triple(lto_module_t mod, const char *triple) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 276 | return unwrap(mod)->setTargetTriple(triple); |
Rafael Espindola | 4ef89f5 | 2010-08-09 21:09:46 +0000 | [diff] [blame] | 277 | } |
| 278 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 279 | unsigned int lto_module_get_num_symbols(lto_module_t mod) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 280 | return unwrap(mod)->getSymbolCount(); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 281 | } |
| 282 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 283 | const char* lto_module_get_symbol_name(lto_module_t mod, unsigned int index) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 284 | return unwrap(mod)->getSymbolName(index); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 285 | } |
| 286 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 287 | lto_symbol_attributes lto_module_get_symbol_attribute(lto_module_t mod, |
| 288 | unsigned int index) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 289 | return unwrap(mod)->getSymbolAttributes(index); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 290 | } |
| 291 | |
Peter Collingbourne | 5d7dffb | 2015-06-29 23:09:12 +0000 | [diff] [blame] | 292 | const char* lto_module_get_linkeropts(lto_module_t mod) { |
Peter Collingbourne | aef3659 | 2015-06-29 22:04:09 +0000 | [diff] [blame] | 293 | return unwrap(mod)->getLinkerOpts(); |
Yunzhong Gao | a88d7ab | 2014-01-21 18:31:27 +0000 | [diff] [blame] | 294 | } |
| 295 | |
Quentin Colombet | 5fa1f6f | 2014-01-15 22:04:35 +0000 | [diff] [blame] | 296 | void lto_codegen_set_diagnostic_handler(lto_code_gen_t cg, |
| 297 | lto_diagnostic_handler_t diag_handler, |
| 298 | void *ctxt) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 299 | unwrap(cg)->setDiagnosticHandler(diag_handler, ctxt); |
Quentin Colombet | 5fa1f6f | 2014-01-15 22:04:35 +0000 | [diff] [blame] | 300 | } |
| 301 | |
Duncan P. N. Exon Smith | d34b613 | 2014-12-19 07:19:50 +0000 | [diff] [blame] | 302 | static lto_code_gen_t createCodeGen(bool InLocalContext) { |
Peter Collingbourne | 4ccf0f1 | 2013-09-24 23:52:22 +0000 | [diff] [blame] | 303 | lto_initialize(); |
Rafael Espindola | 0b385c7 | 2013-09-30 16:39:19 +0000 | [diff] [blame] | 304 | |
Eli Bendersky | f0f2100 | 2014-02-19 17:09:35 +0000 | [diff] [blame] | 305 | TargetOptions Options = InitTargetOptionsFromCodeGenFlags(); |
Rafael Espindola | 0b385c7 | 2013-09-30 16:39:19 +0000 | [diff] [blame] | 306 | |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 307 | LibLTOCodeGenerator *CodeGen = |
| 308 | InLocalContext ? new LibLTOCodeGenerator(make_unique<LLVMContext>()) |
| 309 | : new LibLTOCodeGenerator(); |
| 310 | CodeGen->setTargetOptions(Options); |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 311 | return wrap(CodeGen); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 312 | } |
| 313 | |
Duncan P. N. Exon Smith | d34b613 | 2014-12-19 07:19:50 +0000 | [diff] [blame] | 314 | lto_code_gen_t lto_codegen_create(void) { |
| 315 | return createCodeGen(/* InLocalContext */ false); |
| 316 | } |
| 317 | |
| 318 | lto_code_gen_t lto_codegen_create_in_local_context(void) { |
| 319 | return createCodeGen(/* InLocalContext */ true); |
| 320 | } |
| 321 | |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 322 | void lto_codegen_dispose(lto_code_gen_t cg) { delete unwrap(cg); } |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 323 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 324 | bool lto_codegen_add_module(lto_code_gen_t cg, lto_module_t mod) { |
Rafael Espindola | d12b4a3 | 2014-10-25 04:06:10 +0000 | [diff] [blame] | 325 | return !unwrap(cg)->addModule(unwrap(mod)); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 326 | } |
| 327 | |
Manman Ren | 6487ce9 | 2015-02-24 00:45:56 +0000 | [diff] [blame] | 328 | void lto_codegen_set_module(lto_code_gen_t cg, lto_module_t mod) { |
Peter Collingbourne | 9c8909d | 2015-08-24 22:22:53 +0000 | [diff] [blame] | 329 | unwrap(cg)->setModule(std::unique_ptr<LTOModule>(unwrap(mod))); |
Manman Ren | 6487ce9 | 2015-02-24 00:45:56 +0000 | [diff] [blame] | 330 | } |
| 331 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 332 | bool lto_codegen_set_debug_model(lto_code_gen_t cg, lto_debug_model debug) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 333 | unwrap(cg)->setDebugInfo(debug); |
Shuxin Yang | b6696a9 | 2013-08-07 05:19:23 +0000 | [diff] [blame] | 334 | return false; |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 335 | } |
| 336 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 337 | bool lto_codegen_set_pic_model(lto_code_gen_t cg, lto_codegen_model model) { |
Peter Collingbourne | 44ee84e | 2015-08-21 22:57:17 +0000 | [diff] [blame] | 338 | switch (model) { |
| 339 | case LTO_CODEGEN_PIC_MODEL_STATIC: |
| 340 | unwrap(cg)->setCodePICModel(Reloc::Static); |
| 341 | return false; |
| 342 | case LTO_CODEGEN_PIC_MODEL_DYNAMIC: |
| 343 | unwrap(cg)->setCodePICModel(Reloc::PIC_); |
| 344 | return false; |
| 345 | case LTO_CODEGEN_PIC_MODEL_DYNAMIC_NO_PIC: |
| 346 | unwrap(cg)->setCodePICModel(Reloc::DynamicNoPIC); |
| 347 | return false; |
| 348 | case LTO_CODEGEN_PIC_MODEL_DEFAULT: |
| 349 | unwrap(cg)->setCodePICModel(Reloc::Default); |
| 350 | return false; |
| 351 | } |
| 352 | sLastErrorString = "Unknown PIC model"; |
| 353 | return true; |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 354 | } |
| 355 | |
Bill Wendling | 152e473 | 2012-03-31 10:44:20 +0000 | [diff] [blame] | 356 | void lto_codegen_set_cpu(lto_code_gen_t cg, const char *cpu) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 357 | return unwrap(cg)->setCpu(cpu); |
Rafael Espindola | ccab1dd | 2010-08-11 00:15:13 +0000 | [diff] [blame] | 358 | } |
| 359 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 360 | void lto_codegen_set_assembler_path(lto_code_gen_t cg, const char *path) { |
Rafael Espindola | fac373c | 2011-02-24 21:04:06 +0000 | [diff] [blame] | 361 | // In here only for backwards compatibility. We use MC now. |
Nick Kledzik | cac8c8a | 2009-06-04 00:28:45 +0000 | [diff] [blame] | 362 | } |
| 363 | |
Bill Wendling | 152e473 | 2012-03-31 10:44:20 +0000 | [diff] [blame] | 364 | void lto_codegen_set_assembler_args(lto_code_gen_t cg, const char **args, |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 365 | int nargs) { |
Rafael Espindola | fac373c | 2011-02-24 21:04:06 +0000 | [diff] [blame] | 366 | // In here only for backwards compatibility. We use MC now. |
Rafael Espindola | 0045646 | 2010-08-10 18:55:09 +0000 | [diff] [blame] | 367 | } |
| 368 | |
Bill Wendling | 36cbf03 | 2012-03-30 10:29:38 +0000 | [diff] [blame] | 369 | void lto_codegen_add_must_preserve_symbol(lto_code_gen_t cg, |
Bill Wendling | 152e473 | 2012-03-31 10:44:20 +0000 | [diff] [blame] | 370 | const char *symbol) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 371 | unwrap(cg)->addMustPreserveSymbol(symbol); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 372 | } |
| 373 | |
Peter Collingbourne | 78240e0 | 2015-03-19 22:12:08 +0000 | [diff] [blame] | 374 | static void maybeParseOptions(lto_code_gen_t cg) { |
Rafael Espindola | efa02d5 | 2013-10-02 14:36:23 +0000 | [diff] [blame] | 375 | if (!parsedOptions) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 376 | unwrap(cg)->parseCodeGenDebugOptions(); |
Tom Roeder | fd1bc60 | 2014-04-25 21:46:51 +0000 | [diff] [blame] | 377 | lto_add_attrs(cg); |
Rafael Espindola | efa02d5 | 2013-10-02 14:36:23 +0000 | [diff] [blame] | 378 | parsedOptions = true; |
| 379 | } |
Peter Collingbourne | 070843d | 2015-03-19 22:01:00 +0000 | [diff] [blame] | 380 | } |
| 381 | |
| 382 | bool lto_codegen_write_merged_modules(lto_code_gen_t cg, const char *path) { |
Peter Collingbourne | 78240e0 | 2015-03-19 22:12:08 +0000 | [diff] [blame] | 383 | maybeParseOptions(cg); |
Yunzhong Gao | 8e348cc | 2015-11-17 19:48:12 +0000 | [diff] [blame] | 384 | return !unwrap(cg)->writeMergedModules(path); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 385 | } |
| 386 | |
Bill Wendling | 152e473 | 2012-03-31 10:44:20 +0000 | [diff] [blame] | 387 | const void *lto_codegen_compile(lto_code_gen_t cg, size_t *length) { |
Peter Collingbourne | 78240e0 | 2015-03-19 22:12:08 +0000 | [diff] [blame] | 388 | maybeParseOptions(cg); |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 389 | LibLTOCodeGenerator *CG = unwrap(cg); |
Duncan P. N. Exon Smith | cff5fef | 2015-09-15 23:05:59 +0000 | [diff] [blame] | 390 | CG->NativeObjectFile = |
| 391 | CG->compile(DisableVerify, DisableInline, DisableGVNLoadPRE, |
Yunzhong Gao | 8e348cc | 2015-11-17 19:48:12 +0000 | [diff] [blame] | 392 | DisableLTOVectorization); |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 393 | if (!CG->NativeObjectFile) |
| 394 | return nullptr; |
| 395 | *length = CG->NativeObjectFile->getBufferSize(); |
| 396 | return CG->NativeObjectFile->getBufferStart(); |
Nick Kledzik | 07b4a62 | 2008-02-26 20:26:43 +0000 | [diff] [blame] | 397 | } |
| 398 | |
Manman Ren | 8121e1d | 2015-02-03 18:39:15 +0000 | [diff] [blame] | 399 | bool lto_codegen_optimize(lto_code_gen_t cg) { |
Peter Collingbourne | 78240e0 | 2015-03-19 22:12:08 +0000 | [diff] [blame] | 400 | maybeParseOptions(cg); |
Duncan P. N. Exon Smith | cff5fef | 2015-09-15 23:05:59 +0000 | [diff] [blame] | 401 | return !unwrap(cg)->optimize(DisableVerify, DisableInline, DisableGVNLoadPRE, |
Yunzhong Gao | 8e348cc | 2015-11-17 19:48:12 +0000 | [diff] [blame] | 402 | DisableLTOVectorization); |
Manman Ren | 8121e1d | 2015-02-03 18:39:15 +0000 | [diff] [blame] | 403 | } |
| 404 | |
| 405 | const void *lto_codegen_compile_optimized(lto_code_gen_t cg, size_t *length) { |
Peter Collingbourne | 78240e0 | 2015-03-19 22:12:08 +0000 | [diff] [blame] | 406 | maybeParseOptions(cg); |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 407 | LibLTOCodeGenerator *CG = unwrap(cg); |
Yunzhong Gao | 8e348cc | 2015-11-17 19:48:12 +0000 | [diff] [blame] | 408 | CG->NativeObjectFile = CG->compileOptimized(); |
Peter Collingbourne | 3cc69d90 | 2015-06-01 20:08:30 +0000 | [diff] [blame] | 409 | if (!CG->NativeObjectFile) |
| 410 | return nullptr; |
| 411 | *length = CG->NativeObjectFile->getBufferSize(); |
| 412 | return CG->NativeObjectFile->getBufferStart(); |
Manman Ren | 8121e1d | 2015-02-03 18:39:15 +0000 | [diff] [blame] | 413 | } |
| 414 | |
Reid Kleckner | ddac151 | 2013-10-24 22:26:04 +0000 | [diff] [blame] | 415 | bool lto_codegen_compile_to_file(lto_code_gen_t cg, const char **name) { |
Peter Collingbourne | 78240e0 | 2015-03-19 22:12:08 +0000 | [diff] [blame] | 416 | maybeParseOptions(cg); |
Arnold Schwaighofer | eb1a38f | 2014-10-26 21:50:58 +0000 | [diff] [blame] | 417 | return !unwrap(cg)->compile_to_file( |
Duncan P. N. Exon Smith | cff5fef | 2015-09-15 23:05:59 +0000 | [diff] [blame] | 418 | name, DisableVerify, DisableInline, DisableGVNLoadPRE, |
Yunzhong Gao | 8e348cc | 2015-11-17 19:48:12 +0000 | [diff] [blame] | 419 | DisableLTOVectorization); |
Rafael Espindola | 26b57ff | 2011-03-22 20:57:13 +0000 | [diff] [blame] | 420 | } |
| 421 | |
Bill Wendling | 152e473 | 2012-03-31 10:44:20 +0000 | [diff] [blame] | 422 | void lto_codegen_debug_options(lto_code_gen_t cg, const char *opt) { |
Rafael Espindola | 83ceb8e | 2014-05-03 14:59:52 +0000 | [diff] [blame] | 423 | unwrap(cg)->setCodeGenDebugOptions(opt); |
Duncan Sands | 31554ab | 2009-07-03 15:38:01 +0000 | [diff] [blame] | 424 | } |
Rafael Espindola | a5ef490 | 2015-02-03 19:25:53 +0000 | [diff] [blame] | 425 | |
| 426 | unsigned int lto_api_version() { return LTO_API_VERSION; } |
Manman Ren | ce0a066 | 2015-04-17 17:10:09 +0000 | [diff] [blame] | 427 | |
| 428 | void lto_codegen_set_should_internalize(lto_code_gen_t cg, |
| 429 | bool ShouldInternalize) { |
| 430 | unwrap(cg)->setShouldInternalize(ShouldInternalize); |
| 431 | } |
Duncan P. N. Exon Smith | 5a490d0 | 2015-04-27 23:38:54 +0000 | [diff] [blame] | 432 | |
| 433 | void lto_codegen_set_should_embed_uselists(lto_code_gen_t cg, |
| 434 | lto_bool_t ShouldEmbedUselists) { |
| 435 | unwrap(cg)->setShouldEmbedUselists(ShouldEmbedUselists); |
| 436 | } |