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" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 18 | #include "clang/Driver/DriverDiagnostic.h" |
Chandler Carruth | cc0694c | 2012-12-04 09:25:21 +0000 | [diff] [blame] | 19 | #include "clang/Driver/Options.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 20 | #include "clang/Frontend/CompilerInstance.h" |
| 21 | #include "clang/Frontend/CompilerInvocation.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 22 | #include "clang/Frontend/FrontendDiagnostic.h" |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 23 | #include "clang/Frontend/TextDiagnosticBuffer.h" |
| 24 | #include "clang/Frontend/TextDiagnosticPrinter.h" |
Kostya Serebryany | ce2c726 | 2013-12-27 08:11:08 +0000 | [diff] [blame] | 25 | #include "clang/Frontend/Utils.h" |
Peter Collingbourne | 85dd0bd | 2010-08-24 00:31:22 +0000 | [diff] [blame] | 26 | #include "clang/FrontendTool/Utils.h" |
Douglas Gregor | 171b780 | 2010-03-30 17:33:59 +0000 | [diff] [blame] | 27 | #include "llvm/ADT/Statistic.h" |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 28 | #include "llvm/Config/config.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 | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 40 | #if HAVE_SYS_RESOURCE_H |
| 41 | #include <sys/resource.h> |
| 42 | #endif |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 43 | using namespace clang; |
Reid Kleckner | 898229a | 2013-06-14 17:17:23 +0000 | [diff] [blame] | 44 | using namespace llvm::opt; |
Daniel Dunbar | 51cd8f0 | 2009-11-19 07:37:51 +0000 | [diff] [blame] | 45 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 46 | //===----------------------------------------------------------------------===// |
| 47 | // Main driver |
| 48 | //===----------------------------------------------------------------------===// |
| 49 | |
Chad Rosier | 05c71aa | 2013-03-27 18:28:23 +0000 | [diff] [blame] | 50 | static void LLVMErrorHandler(void *UserData, const std::string &Message, |
| 51 | bool GenCrashDiag) { |
David Blaikie | 9c902b5 | 2011-09-25 23:23:43 +0000 | [diff] [blame] | 52 | DiagnosticsEngine &Diags = *static_cast<DiagnosticsEngine*>(UserData); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 53 | |
| 54 | Diags.Report(diag::err_fe_error_backend) << Message; |
| 55 | |
Chad Rosier | 7ea7397 | 2012-11-12 19:39:37 +0000 | [diff] [blame] | 56 | // Run the interrupt handlers to make sure any special cleanups get done, in |
| 57 | // particular that we remove files registered with RemoveFileOnSignal. |
| 58 | llvm::sys::RunInterruptHandlers(); |
| 59 | |
Chad Rosier | ad6e96d | 2012-11-12 21:32:24 +0000 | [diff] [blame] | 60 | // We cannot recover from llvm errors. When reporting a fatal error, exit |
Chad Rosier | 05c71aa | 2013-03-27 18:28:23 +0000 | [diff] [blame] | 61 | // with status 70 to generate crash diagnostics. For BSD systems this is |
| 62 | // defined as an internal software error. Otherwise, exit with status 1. |
| 63 | exit(GenCrashDiag ? 70 : 1); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 64 | } |
| 65 | |
Sebastian Pop | 17fac04 | 2014-03-14 04:04:27 +0000 | [diff] [blame] | 66 | #ifdef LINK_POLLY_INTO_TOOLS |
| 67 | namespace polly { |
| 68 | void initializePollyPasses(llvm::PassRegistry &Registry); |
| 69 | } |
| 70 | #endif |
| 71 | |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 72 | #if HAVE_SYS_RESOURCE_H && HAVE_GETRLIMIT && HAVE_SETRLIMIT |
| 73 | // The amount of stack we think is "sufficient". If less than this much is |
| 74 | // available, we may be unable to reach our template instantiation depth |
| 75 | // limit and other similar limits. |
| 76 | // FIXME: Unify this with the stack we request when spawning a thread to build |
| 77 | // a module. |
| 78 | static const int kSufficientStack = 8 << 20; |
| 79 | |
| 80 | #if defined(__linux__) && defined(__PIE__) |
| 81 | LLVM_ATTRIBUTE_NOINLINE |
| 82 | static void ensureStackAddressSpace() { |
| 83 | // Linux kernels prior to 4.1 will sometimes locate the heap of a PIE binary |
| 84 | // relatively close to the stack (they are only guaranteed to be 128MiB |
| 85 | // apart). This results in crashes if we happen to heap-allocate more than |
| 86 | // 128MiB before we reach our stack high-water mark. |
| 87 | // |
| 88 | // To avoid these crashes, ensure that we have sufficient virtual memory |
| 89 | // pages allocated before we start running by touching an early page. (We |
| 90 | // allow 512KiB for kernel/libc-provided data such as command-line arguments |
| 91 | // and environment variables, and for main and cc1_main) |
| 92 | volatile char ReservedStack[kSufficientStack - 512 * 1024]; |
| 93 | volatile int N = 0; |
| 94 | (void)+ReservedStack[N]; |
| 95 | } |
| 96 | #else |
| 97 | static void ensureStackAddressSpace() {} |
| 98 | #endif |
| 99 | |
| 100 | /// Attempt to ensure that we have at least 8MiB of usable stack space. |
| 101 | static void ensureSufficientStack() { |
| 102 | struct rlimit rlim; |
| 103 | if (getrlimit(RLIMIT_STACK, &rlim) != 0) |
| 104 | return; |
| 105 | |
| 106 | // Increase the soft stack limit to our desired level, if necessary and |
| 107 | // possible. |
| 108 | if (rlim.rlim_cur != RLIM_INFINITY && rlim.rlim_cur < kSufficientStack) { |
| 109 | // Try to allocate sufficient stack. |
| 110 | if (rlim.rlim_max == RLIM_INFINITY || rlim.rlim_max >= kSufficientStack) |
| 111 | rlim.rlim_cur = kSufficientStack; |
| 112 | else if (rlim.rlim_cur == rlim.rlim_max) |
| 113 | return; |
| 114 | else |
| 115 | rlim.rlim_cur = rlim.rlim_max; |
| 116 | |
| 117 | if (setrlimit(RLIMIT_STACK, &rlim) != 0 || |
| 118 | rlim.rlim_cur != kSufficientStack) |
| 119 | return; |
| 120 | } |
| 121 | |
| 122 | // We should now have a stack of size at least kSufficientStack. Ensure |
| 123 | // that we can actually use that much, if necessary. |
| 124 | ensureStackAddressSpace(); |
| 125 | } |
| 126 | #else |
Richard Smith | 525ce25 | 2016-08-17 02:22:39 +0000 | [diff] [blame^] | 127 | static void ensureSufficientStack() {} |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 128 | #endif |
| 129 | |
Sean Silva | 070cd2d | 2014-08-15 21:38:36 +0000 | [diff] [blame] | 130 | int cc1_main(ArrayRef<const char *> Argv, const char *Argv0, void *MainAddr) { |
Richard Smith | 194b6a3 | 2016-08-17 01:05:07 +0000 | [diff] [blame] | 131 | ensureSufficientStack(); |
| 132 | |
Adrian Prantl | fb2398d | 2015-07-17 01:19:54 +0000 | [diff] [blame] | 133 | std::unique_ptr<CompilerInstance> Clang(new CompilerInstance()); |
Dylan Noblesmith | c95d819 | 2012-02-20 14:00:23 +0000 | [diff] [blame] | 134 | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
Daniel Dunbar | e922d9b | 2010-02-16 01:54:47 +0000 | [diff] [blame] | 135 | |
Adrian Prantl | fb2398d | 2015-07-17 01:19:54 +0000 | [diff] [blame] | 136 | // Register the support for object-file-wrapped Clang modules. |
| 137 | auto PCHOps = Clang->getPCHContainerOperations(); |
| 138 | PCHOps->registerWriter(llvm::make_unique<ObjectFilePCHContainerWriter>()); |
| 139 | PCHOps->registerReader(llvm::make_unique<ObjectFilePCHContainerReader>()); |
| 140 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 141 | // Initialize targets first, so that --version shows registered targets. |
| 142 | llvm::InitializeAllTargets(); |
Evan Cheng | c391a58 | 2011-07-22 21:59:11 +0000 | [diff] [blame] | 143 | llvm::InitializeAllTargetMCs(); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 144 | llvm::InitializeAllAsmPrinters(); |
Chris Lattner | 61955ab | 2010-04-05 23:33:20 +0000 | [diff] [blame] | 145 | llvm::InitializeAllAsmParsers(); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 146 | |
Sebastian Pop | 17fac04 | 2014-03-14 04:04:27 +0000 | [diff] [blame] | 147 | #ifdef LINK_POLLY_INTO_TOOLS |
| 148 | llvm::PassRegistry &Registry = *llvm::PassRegistry::getPassRegistry(); |
| 149 | polly::initializePollyPasses(Registry); |
| 150 | #endif |
| 151 | |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 152 | // Buffer diagnostics from argument parsing so that we can output them using a |
| 153 | // well formed diagnostic object. |
Douglas Gregor | 811db4e | 2012-10-23 22:26:28 +0000 | [diff] [blame] | 154 | IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
Douglas Gregor | 2dd19f1 | 2010-08-18 22:29:43 +0000 | [diff] [blame] | 155 | TextDiagnosticBuffer *DiagsBuffer = new TextDiagnosticBuffer; |
Douglas Gregor | 811db4e | 2012-10-23 22:26:28 +0000 | [diff] [blame] | 156 | DiagnosticsEngine Diags(DiagID, &*DiagOpts, DiagsBuffer); |
Sean Silva | 070cd2d | 2014-08-15 21:38:36 +0000 | [diff] [blame] | 157 | bool Success = CompilerInvocation::CreateFromArgs( |
| 158 | Clang->getInvocation(), Argv.begin(), Argv.end(), Diags); |
Daniel Dunbar | d613677 | 2009-12-13 03:45:58 +0000 | [diff] [blame] | 159 | |
| 160 | // Infer the builtin include path if unspecified. |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 161 | if (Clang->getHeaderSearchOpts().UseBuiltinIncludes && |
| 162 | Clang->getHeaderSearchOpts().ResourceDir.empty()) |
| 163 | Clang->getHeaderSearchOpts().ResourceDir = |
Daniel Dunbar | a5a166d | 2009-12-15 00:06:45 +0000 | [diff] [blame] | 164 | CompilerInvocation::GetResourcesPath(Argv0, MainAddr); |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 165 | |
Daniel Dunbar | 1cdf04c | 2010-08-12 02:53:07 +0000 | [diff] [blame] | 166 | // Create the actual diagnostics engine. |
Sean Silva | f1b49e2 | 2013-01-20 01:58:28 +0000 | [diff] [blame] | 167 | Clang->createDiagnostics(); |
Daniel Dunbar | 1cdf04c | 2010-08-12 02:53:07 +0000 | [diff] [blame] | 168 | if (!Clang->hasDiagnostics()) |
| 169 | return 1; |
| 170 | |
| 171 | // Set an error handler, so that any LLVM backend diagnostics go through our |
| 172 | // error handler. |
| 173 | llvm::install_fatal_error_handler(LLVMErrorHandler, |
| 174 | static_cast<void*>(&Clang->getDiagnostics())); |
| 175 | |
Douglas Gregor | 2dd19f1 | 2010-08-18 22:29:43 +0000 | [diff] [blame] | 176 | DiagsBuffer->FlushDiagnostics(Clang->getDiagnostics()); |
Argyrios Kyrtzidis | be6d89d | 2012-01-25 20:00:43 +0000 | [diff] [blame] | 177 | if (!Success) |
| 178 | return 1; |
Daniel Dunbar | 1cdf04c | 2010-08-12 02:53:07 +0000 | [diff] [blame] | 179 | |
Daniel Dunbar | e9e91b0 | 2010-08-12 02:53:12 +0000 | [diff] [blame] | 180 | // Execute the frontend actions. |
Dylan Noblesmith | e99b27f | 2011-12-23 03:05:38 +0000 | [diff] [blame] | 181 | Success = ExecuteCompilerInvocation(Clang.get()); |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 182 | |
Chris Lattner | 09f8cc8 | 2010-03-30 05:39:52 +0000 | [diff] [blame] | 183 | // If any timers were active but haven't been destroyed yet, print their |
| 184 | // results now. This happens in -disable-free mode. |
| 185 | llvm::TimerGroup::printAll(llvm::errs()); |
Daniel Dunbar | 2be9674 | 2010-08-02 15:31:28 +0000 | [diff] [blame] | 186 | |
Dan Gohman | b37af7d | 2010-08-18 21:23:17 +0000 | [diff] [blame] | 187 | // Our error handler depends on the Diagnostics object, which we're |
| 188 | // potentially about to delete. Uninstall the handler now so that any |
| 189 | // later errors use the default handling behavior instead. |
| 190 | llvm::remove_fatal_error_handler(); |
| 191 | |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 192 | // When running with -disable-free, don't do any destruction or shutdown. |
| 193 | if (Clang->getFrontendOpts().DisableFree) { |
David Blaikie | 3c13a7f | 2014-08-29 17:02:26 +0000 | [diff] [blame] | 194 | BuryPointer(std::move(Clang)); |
Daniel Dunbar | 30b24e9 | 2010-03-23 05:09:16 +0000 | [diff] [blame] | 195 | return !Success; |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 196 | } |
| 197 | |
Daniel Dunbar | 4f2bc55 | 2010-01-13 00:48:06 +0000 | [diff] [blame] | 198 | return !Success; |
Daniel Dunbar | f72bdf7 | 2009-12-11 22:20:12 +0000 | [diff] [blame] | 199 | } |