Daniel Dunbar | 2fcaa54 | 2010-05-20 17:49:16 +0000 | [diff] [blame] | 1 | //===-- cc1_main.cpp - Clang CC1 Compiler Frontend ------------------------===// |
Daniel Dunbar | 51cd8f0 | 2009-11-19 07:37:51 +0000 | [diff] [blame] | 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. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 10 | // This is the entry point to the clang -cc1 functionality, which implements the |
| 11 | // core compiler functionality along with a number of additional tools for |
| 12 | // demonstration and testing purposes. |
Daniel Dunbar | 51cd8f0 | 2009-11-19 07:37:51 +0000 | [diff] [blame] | 13 | // |
| 14 | //===----------------------------------------------------------------------===// |
| 15 | |
Reid Kleckner | 898229a | 2013-06-14 17:17:23 +0000 | [diff] [blame] | 16 | #include "llvm/Option/Arg.h" |
Adrian Prantl | bc06858 | 2015-07-08 01:00:30 +0000 | [diff] [blame] | 17 | #include "clang/CodeGen/ObjectFilePCHContainerOperations.h" |
Chris Bieneman | a6b39ab | 2016-08-23 20:07:07 +0000 | [diff] [blame] | 18 | #include "clang/Config/config.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 19 | #include "clang/Driver/DriverDiagnostic.h" |
Chandler Carruth | cc0694c | 2012-12-04 09:25:21 +0000 | [diff] [blame] | 20 | #include "clang/Driver/Options.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 21 | #include "clang/Frontend/CompilerInstance.h" |
| 22 | #include "clang/Frontend/CompilerInvocation.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 23 | #include "clang/Frontend/FrontendDiagnostic.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 24 | #include "clang/Frontend/TextDiagnosticBuffer.h" |
| 25 | #include "clang/Frontend/TextDiagnosticPrinter.h" |
Kostya Serebryany | ce2c726 | 2013-12-27 08:11:08 +0000 | [diff] [blame] | 26 | #include "clang/Frontend/Utils.h" |
Peter Collingbourne | 85dd0bd | 2010-08-24 00:31:22 +0000 | [diff] [blame] | 27 | #include "clang/FrontendTool/Utils.h" |
Douglas Gregor | 171b780 | 2010-03-30 17:33:59 +0000 | [diff] [blame] | 28 | #include "llvm/ADT/Statistic.h" |
Chandler Carruth | cc0694c | 2012-12-04 09:25:21 +0000 | [diff] [blame] | 29 | #include "llvm/LinkAllPasses.h" |
Reid Kleckner | 898229a | 2013-06-14 17:17:23 +0000 | [diff] [blame] | 30 | #include "llvm/Option/ArgList.h" |
| 31 | #include "llvm/Option/OptTable.h" |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 32 | #include "llvm/Support/Compiler.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 33 | #include "llvm/Support/ErrorHandling.h" |
| 34 | #include "llvm/Support/ManagedStatic.h" |
Chad Rosier | 7ea7397 | 2012-11-12 19:39:37 +0000 | [diff] [blame] | 35 | #include "llvm/Support/Signals.h" |
Evan Cheng | 494eb06 | 2011-08-24 18:09:14 +0000 | [diff] [blame] | 36 | #include "llvm/Support/TargetSelect.h" |
Chris Lattner | 09f8cc8 | 2010-03-30 05:39:52 +0000 | [diff] [blame] | 37 | #include "llvm/Support/Timer.h" |
Daniel Dunbar | 51cd8f0 | 2009-11-19 07:37:51 +0000 | [diff] [blame] | 38 | #include "llvm/Support/raw_ostream.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 39 | #include <cstdio> |
Richard Smith | c33b837 | 2016-08-18 18:22:22 +0000 | [diff] [blame] | 40 | |
Chris Bieneman | a6b39ab | 2016-08-23 20:07:07 +0000 | [diff] [blame] | 41 | #ifdef CLANG_HAVE_RLIMITS |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 42 | #include <sys/resource.h> |
| 43 | #endif |
Richard Smith | c33b837 | 2016-08-18 18:22:22 +0000 | [diff] [blame] | 44 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 45 | using namespace clang; |
Reid Kleckner | 898229a | 2013-06-14 17:17:23 +0000 | [diff] [blame] | 46 | using namespace llvm::opt; |
Daniel Dunbar | 51cd8f0 | 2009-11-19 07:37:51 +0000 | [diff] [blame] | 47 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 48 | //===----------------------------------------------------------------------===// |
| 49 | // Main driver |
| 50 | //===----------------------------------------------------------------------===// |
| 51 | |
Chad Rosier | 05c71aa | 2013-03-27 18:28:23 +0000 | [diff] [blame] | 52 | static void LLVMErrorHandler(void *UserData, const std::string &Message, |
| 53 | bool GenCrashDiag) { |
David Blaikie | 9c902b5 | 2011-09-25 23:23:43 +0000 | [diff] [blame] | 54 | DiagnosticsEngine &Diags = *static_cast<DiagnosticsEngine*>(UserData); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 55 | |
| 56 | Diags.Report(diag::err_fe_error_backend) << Message; |
| 57 | |
Chad Rosier | 7ea7397 | 2012-11-12 19:39:37 +0000 | [diff] [blame] | 58 | // Run the interrupt handlers to make sure any special cleanups get done, in |
| 59 | // particular that we remove files registered with RemoveFileOnSignal. |
| 60 | llvm::sys::RunInterruptHandlers(); |
| 61 | |
Chad Rosier | ad6e96d | 2012-11-12 21:32:24 +0000 | [diff] [blame] | 62 | // We cannot recover from llvm errors. When reporting a fatal error, exit |
Chad Rosier | 05c71aa | 2013-03-27 18:28:23 +0000 | [diff] [blame] | 63 | // with status 70 to generate crash diagnostics. For BSD systems this is |
| 64 | // defined as an internal software error. Otherwise, exit with status 1. |
| 65 | exit(GenCrashDiag ? 70 : 1); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 66 | } |
| 67 | |
Sebastian Pop | 17fac04 | 2014-03-14 04:04:27 +0000 | [diff] [blame] | 68 | #ifdef LINK_POLLY_INTO_TOOLS |
| 69 | namespace polly { |
| 70 | void initializePollyPasses(llvm::PassRegistry &Registry); |
| 71 | } |
| 72 | #endif |
| 73 | |
Chris Bieneman | a6b39ab | 2016-08-23 20:07:07 +0000 | [diff] [blame] | 74 | #ifdef CLANG_HAVE_RLIMITS |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 75 | // The amount of stack we think is "sufficient". If less than this much is |
| 76 | // available, we may be unable to reach our template instantiation depth |
| 77 | // limit and other similar limits. |
| 78 | // FIXME: Unify this with the stack we request when spawning a thread to build |
| 79 | // a module. |
| 80 | static const int kSufficientStack = 8 << 20; |
| 81 | |
| 82 | #if defined(__linux__) && defined(__PIE__) |
Richard Smith | f80a27e | 2016-10-14 19:51:36 +0000 | [diff] [blame] | 83 | static size_t getCurrentStackAllocation() { |
| 84 | // If we can't compute the current stack usage, allow for 512K of command |
| 85 | // line arguments and environment. |
| 86 | size_t Usage = 512 * 1024; |
| 87 | if (FILE *StatFile = fopen("/proc/self/stat", "r")) { |
| 88 | // We assume that the stack extends from its current address to the end of |
| 89 | // the environment space. In reality, there is another string literal (the |
| 90 | // program name) after the environment, but this is close enough (we only |
| 91 | // need to be within 100K or so). |
| 92 | unsigned long StackPtr, EnvEnd; |
Richard Smith | 49db68d | 2016-10-15 01:59:52 +0000 | [diff] [blame] | 93 | // Disable silly GCC -Wformat warning that complains about length |
| 94 | // modifiers on ignored format specifiers. We want to retain these |
| 95 | // for documentation purposes even though they have no effect. |
| 96 | #if defined(__GNUC__) && !defined(__clang__) |
| 97 | #pragma GCC diagnostic push |
| 98 | #pragma GCC diagnostic ignored "-Wformat" |
| 99 | #endif |
Richard Smith | f80a27e | 2016-10-14 19:51:36 +0000 | [diff] [blame] | 100 | if (fscanf(StatFile, |
| 101 | "%*d %*s %*c %*d %*d %*d %*d %*d %*u %*lu %*lu %*lu %*lu %*lu " |
| 102 | "%*lu %*ld %*ld %*ld %*ld %*ld %*ld %*llu %*lu %*ld %*lu %*lu " |
| 103 | "%*lu %*lu %lu %*lu %*lu %*lu %*lu %*lu %*llu %*lu %*lu %*d %*d " |
| 104 | "%*u %*u %*llu %*lu %*ld %*lu %*lu %*lu %*lu %*lu %*lu %lu %*d", |
| 105 | &StackPtr, &EnvEnd) == 2) { |
Richard Smith | 49db68d | 2016-10-15 01:59:52 +0000 | [diff] [blame] | 106 | #if defined(__GNUC__) && !defined(__clang__) |
| 107 | #pragma GCC diagnostic pop |
| 108 | #endif |
Richard Smith | f80a27e | 2016-10-14 19:51:36 +0000 | [diff] [blame] | 109 | Usage = StackPtr < EnvEnd ? EnvEnd - StackPtr : StackPtr - EnvEnd; |
| 110 | } |
| 111 | fclose(StatFile); |
| 112 | } |
| 113 | return Usage; |
| 114 | } |
| 115 | |
| 116 | #include <alloca.h> |
| 117 | |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 118 | LLVM_ATTRIBUTE_NOINLINE |
Richard Smith | f80a27e | 2016-10-14 19:51:36 +0000 | [diff] [blame] | 119 | static void ensureStackAddressSpace(int ExtraChunks = 0) { |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 120 | // Linux kernels prior to 4.1 will sometimes locate the heap of a PIE binary |
| 121 | // relatively close to the stack (they are only guaranteed to be 128MiB |
| 122 | // apart). This results in crashes if we happen to heap-allocate more than |
| 123 | // 128MiB before we reach our stack high-water mark. |
| 124 | // |
| 125 | // To avoid these crashes, ensure that we have sufficient virtual memory |
Richard Smith | f80a27e | 2016-10-14 19:51:36 +0000 | [diff] [blame] | 126 | // pages allocated before we start running. |
| 127 | size_t Curr = getCurrentStackAllocation(); |
| 128 | const int kTargetStack = kSufficientStack - 256 * 1024; |
| 129 | if (Curr < kTargetStack) { |
| 130 | volatile char *volatile Alloc = |
| 131 | static_cast<volatile char *>(alloca(kTargetStack - Curr)); |
| 132 | Alloc[0] = 0; |
| 133 | Alloc[kTargetStack - Curr - 1] = 0; |
| 134 | } |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 135 | } |
| 136 | #else |
| 137 | static void ensureStackAddressSpace() {} |
| 138 | #endif |
| 139 | |
| 140 | /// Attempt to ensure that we have at least 8MiB of usable stack space. |
| 141 | static void ensureSufficientStack() { |
| 142 | struct rlimit rlim; |
| 143 | if (getrlimit(RLIMIT_STACK, &rlim) != 0) |
| 144 | return; |
| 145 | |
| 146 | // Increase the soft stack limit to our desired level, if necessary and |
| 147 | // possible. |
| 148 | if (rlim.rlim_cur != RLIM_INFINITY && rlim.rlim_cur < kSufficientStack) { |
| 149 | // Try to allocate sufficient stack. |
| 150 | if (rlim.rlim_max == RLIM_INFINITY || rlim.rlim_max >= kSufficientStack) |
| 151 | rlim.rlim_cur = kSufficientStack; |
| 152 | else if (rlim.rlim_cur == rlim.rlim_max) |
| 153 | return; |
| 154 | else |
| 155 | rlim.rlim_cur = rlim.rlim_max; |
| 156 | |
| 157 | if (setrlimit(RLIMIT_STACK, &rlim) != 0 || |
| 158 | rlim.rlim_cur != kSufficientStack) |
| 159 | return; |
| 160 | } |
| 161 | |
| 162 | // We should now have a stack of size at least kSufficientStack. Ensure |
| 163 | // that we can actually use that much, if necessary. |
| 164 | ensureStackAddressSpace(); |
| 165 | } |
| 166 | #else |
Richard Smith | 525ce25 | 2016-08-17 02:22:39 +0000 | [diff] [blame] | 167 | static void ensureSufficientStack() {} |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 168 | #endif |
| 169 | |
Sean Silva | 070cd2d | 2014-08-15 21:38:36 +0000 | [diff] [blame] | 170 | int cc1_main(ArrayRef<const char *> Argv, const char *Argv0, void *MainAddr) { |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 171 | ensureSufficientStack(); |
| 172 | |
Adrian Prantl | fb2398d | 2015-07-17 01:19:54 +0000 | [diff] [blame] | 173 | std::unique_ptr<CompilerInstance> Clang(new CompilerInstance()); |
Dylan Noblesmith | c95d819 | 2012-02-20 14:00:23 +0000 | [diff] [blame] | 174 | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
Daniel Dunbar | e922d9b | 2010-02-16 01:54:47 +0000 | [diff] [blame] | 175 | |
Adrian Prantl | fb2398d | 2015-07-17 01:19:54 +0000 | [diff] [blame] | 176 | // Register the support for object-file-wrapped Clang modules. |
| 177 | auto PCHOps = Clang->getPCHContainerOperations(); |
| 178 | PCHOps->registerWriter(llvm::make_unique<ObjectFilePCHContainerWriter>()); |
| 179 | PCHOps->registerReader(llvm::make_unique<ObjectFilePCHContainerReader>()); |
| 180 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 181 | // Initialize targets first, so that --version shows registered targets. |
| 182 | llvm::InitializeAllTargets(); |
Evan Cheng | c391a58 | 2011-07-22 21:59:11 +0000 | [diff] [blame] | 183 | llvm::InitializeAllTargetMCs(); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 184 | llvm::InitializeAllAsmPrinters(); |
Chris Lattner | 61955ab | 2010-04-05 23:33:20 +0000 | [diff] [blame] | 185 | llvm::InitializeAllAsmParsers(); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 186 | |
Sebastian Pop | 17fac04 | 2014-03-14 04:04:27 +0000 | [diff] [blame] | 187 | #ifdef LINK_POLLY_INTO_TOOLS |
| 188 | llvm::PassRegistry &Registry = *llvm::PassRegistry::getPassRegistry(); |
| 189 | polly::initializePollyPasses(Registry); |
| 190 | #endif |
| 191 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 192 | // Buffer diagnostics from argument parsing so that we can output them using a |
| 193 | // well formed diagnostic object. |
Douglas Gregor | 811db4e | 2012-10-23 22:26:28 +0000 | [diff] [blame] | 194 | IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
Douglas Gregor | 2dd19f1 | 2010-08-18 22:29:43 +0000 | [diff] [blame] | 195 | TextDiagnosticBuffer *DiagsBuffer = new TextDiagnosticBuffer; |
Douglas Gregor | 811db4e | 2012-10-23 22:26:28 +0000 | [diff] [blame] | 196 | DiagnosticsEngine Diags(DiagID, &*DiagOpts, DiagsBuffer); |
Sean Silva | 070cd2d | 2014-08-15 21:38:36 +0000 | [diff] [blame] | 197 | bool Success = CompilerInvocation::CreateFromArgs( |
| 198 | Clang->getInvocation(), Argv.begin(), Argv.end(), Diags); |
Daniel Dunbar | d613677 | 2009-12-13 03:45:58 +0000 | [diff] [blame] | 199 | |
| 200 | // Infer the builtin include path if unspecified. |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 201 | if (Clang->getHeaderSearchOpts().UseBuiltinIncludes && |
| 202 | Clang->getHeaderSearchOpts().ResourceDir.empty()) |
| 203 | Clang->getHeaderSearchOpts().ResourceDir = |
Daniel Dunbar | a5a166d | 2009-12-15 00:06:45 +0000 | [diff] [blame] | 204 | CompilerInvocation::GetResourcesPath(Argv0, MainAddr); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 205 | |
Daniel Dunbar | 1cdf04c | 2010-08-12 02:53:07 +0000 | [diff] [blame] | 206 | // Create the actual diagnostics engine. |
Sean Silva | f1b49e2 | 2013-01-20 01:58:28 +0000 | [diff] [blame] | 207 | Clang->createDiagnostics(); |
Daniel Dunbar | 1cdf04c | 2010-08-12 02:53:07 +0000 | [diff] [blame] | 208 | if (!Clang->hasDiagnostics()) |
| 209 | return 1; |
| 210 | |
| 211 | // Set an error handler, so that any LLVM backend diagnostics go through our |
| 212 | // error handler. |
| 213 | llvm::install_fatal_error_handler(LLVMErrorHandler, |
| 214 | static_cast<void*>(&Clang->getDiagnostics())); |
| 215 | |
Douglas Gregor | 2dd19f1 | 2010-08-18 22:29:43 +0000 | [diff] [blame] | 216 | DiagsBuffer->FlushDiagnostics(Clang->getDiagnostics()); |
Argyrios Kyrtzidis | be6d89d | 2012-01-25 20:00:43 +0000 | [diff] [blame] | 217 | if (!Success) |
| 218 | return 1; |
Daniel Dunbar | 1cdf04c | 2010-08-12 02:53:07 +0000 | [diff] [blame] | 219 | |
Daniel Dunbar | e9e91b0 | 2010-08-12 02:53:12 +0000 | [diff] [blame] | 220 | // Execute the frontend actions. |
Dylan Noblesmith | e99b27f | 2011-12-23 03:05:38 +0000 | [diff] [blame] | 221 | Success = ExecuteCompilerInvocation(Clang.get()); |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 222 | |
Chris Lattner | 09f8cc8 | 2010-03-30 05:39:52 +0000 | [diff] [blame] | 223 | // If any timers were active but haven't been destroyed yet, print their |
| 224 | // results now. This happens in -disable-free mode. |
| 225 | llvm::TimerGroup::printAll(llvm::errs()); |
Daniel Dunbar | 2be9674 | 2010-08-02 15:31:28 +0000 | [diff] [blame] | 226 | |
Dan Gohman | b37af7d | 2010-08-18 21:23:17 +0000 | [diff] [blame] | 227 | // Our error handler depends on the Diagnostics object, which we're |
| 228 | // potentially about to delete. Uninstall the handler now so that any |
| 229 | // later errors use the default handling behavior instead. |
| 230 | llvm::remove_fatal_error_handler(); |
| 231 | |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 232 | // When running with -disable-free, don't do any destruction or shutdown. |
| 233 | if (Clang->getFrontendOpts().DisableFree) { |
David Blaikie | 3c13a7f | 2014-08-29 17:02:26 +0000 | [diff] [blame] | 234 | BuryPointer(std::move(Clang)); |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 235 | return !Success; |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 236 | } |
| 237 | |
Daniel Dunbar | 4f2bc55 | 2010-01-13 00:48:06 +0000 | [diff] [blame] | 238 | return !Success; |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 239 | } |