blob: ef5a191bda0a638dce6edf520ab7504bf959bae4 [file] [log] [blame]
Daniel Dunbar2fcaa542010-05-20 17:49:16 +00001//===-- cc1_main.cpp - Clang CC1 Compiler Frontend ------------------------===//
Daniel Dunbar51cd8f02009-11-19 07:37:51 +00002//
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 Dunbarf72bdf72009-12-11 22:20:12 +000010// 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 Dunbar51cd8f02009-11-19 07:37:51 +000013//
14//===----------------------------------------------------------------------===//
15
Reid Kleckner898229a2013-06-14 17:17:23 +000016#include "llvm/Option/Arg.h"
Adrian Prantlbc068582015-07-08 01:00:30 +000017#include "clang/CodeGen/ObjectFilePCHContainerOperations.h"
Chris Bienemana6b39ab2016-08-23 20:07:07 +000018#include "clang/Config/config.h"
Richard Smith0a7b2972018-07-03 21:34:13 +000019#include "clang/Basic/Stack.h"
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000020#include "clang/Driver/DriverDiagnostic.h"
Chandler Carruthcc0694c2012-12-04 09:25:21 +000021#include "clang/Driver/Options.h"
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000022#include "clang/Frontend/CompilerInstance.h"
23#include "clang/Frontend/CompilerInvocation.h"
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000024#include "clang/Frontend/FrontendDiagnostic.h"
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000025#include "clang/Frontend/TextDiagnosticBuffer.h"
26#include "clang/Frontend/TextDiagnosticPrinter.h"
Kostya Serebryanyce2c7262013-12-27 08:11:08 +000027#include "clang/Frontend/Utils.h"
Peter Collingbourne85dd0bd2010-08-24 00:31:22 +000028#include "clang/FrontendTool/Utils.h"
Douglas Gregor171b7802010-03-30 17:33:59 +000029#include "llvm/ADT/Statistic.h"
Nico Weberd637c052018-04-30 13:52:15 +000030#include "llvm/Config/llvm-config.h"
Chandler Carruthcc0694c2012-12-04 09:25:21 +000031#include "llvm/LinkAllPasses.h"
Reid Kleckner898229a2013-06-14 17:17:23 +000032#include "llvm/Option/ArgList.h"
33#include "llvm/Option/OptTable.h"
Richard Smith194b6a32016-08-17 01:05:07 +000034#include "llvm/Support/Compiler.h"
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000035#include "llvm/Support/ErrorHandling.h"
36#include "llvm/Support/ManagedStatic.h"
Chad Rosier7ea73972012-11-12 19:39:37 +000037#include "llvm/Support/Signals.h"
Evan Cheng494eb062011-08-24 18:09:14 +000038#include "llvm/Support/TargetSelect.h"
Chris Lattner09f8cc82010-03-30 05:39:52 +000039#include "llvm/Support/Timer.h"
Daniel Dunbar51cd8f02009-11-19 07:37:51 +000040#include "llvm/Support/raw_ostream.h"
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000041#include <cstdio>
Richard Smithc33b8372016-08-18 18:22:22 +000042
Chris Bienemana6b39ab2016-08-23 20:07:07 +000043#ifdef CLANG_HAVE_RLIMITS
Richard Smith194b6a32016-08-17 01:05:07 +000044#include <sys/resource.h>
45#endif
Richard Smithc33b8372016-08-18 18:22:22 +000046
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000047using namespace clang;
Reid Kleckner898229a2013-06-14 17:17:23 +000048using namespace llvm::opt;
Daniel Dunbar51cd8f02009-11-19 07:37:51 +000049
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000050//===----------------------------------------------------------------------===//
51// Main driver
52//===----------------------------------------------------------------------===//
53
Chad Rosier05c71aa2013-03-27 18:28:23 +000054static void LLVMErrorHandler(void *UserData, const std::string &Message,
55 bool GenCrashDiag) {
David Blaikie9c902b52011-09-25 23:23:43 +000056 DiagnosticsEngine &Diags = *static_cast<DiagnosticsEngine*>(UserData);
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000057
58 Diags.Report(diag::err_fe_error_backend) << Message;
59
Chad Rosier7ea73972012-11-12 19:39:37 +000060 // Run the interrupt handlers to make sure any special cleanups get done, in
61 // particular that we remove files registered with RemoveFileOnSignal.
62 llvm::sys::RunInterruptHandlers();
63
Chad Rosierad6e96d2012-11-12 21:32:24 +000064 // We cannot recover from llvm errors. When reporting a fatal error, exit
Chad Rosier05c71aa2013-03-27 18:28:23 +000065 // with status 70 to generate crash diagnostics. For BSD systems this is
66 // defined as an internal software error. Otherwise, exit with status 1.
67 exit(GenCrashDiag ? 70 : 1);
Daniel Dunbarf72bdf72009-12-11 22:20:12 +000068}
69
Sebastian Pop17fac042014-03-14 04:04:27 +000070#ifdef LINK_POLLY_INTO_TOOLS
71namespace polly {
72void initializePollyPasses(llvm::PassRegistry &Registry);
73}
74#endif
75
Chris Bienemana6b39ab2016-08-23 20:07:07 +000076#ifdef CLANG_HAVE_RLIMITS
Richard Smith194b6a32016-08-17 01:05:07 +000077#if defined(__linux__) && defined(__PIE__)
Richard Smithf80a27e2016-10-14 19:51:36 +000078static size_t getCurrentStackAllocation() {
79 // If we can't compute the current stack usage, allow for 512K of command
80 // line arguments and environment.
81 size_t Usage = 512 * 1024;
82 if (FILE *StatFile = fopen("/proc/self/stat", "r")) {
83 // We assume that the stack extends from its current address to the end of
84 // the environment space. In reality, there is another string literal (the
85 // program name) after the environment, but this is close enough (we only
86 // need to be within 100K or so).
87 unsigned long StackPtr, EnvEnd;
Richard Smith49db68d2016-10-15 01:59:52 +000088 // Disable silly GCC -Wformat warning that complains about length
89 // modifiers on ignored format specifiers. We want to retain these
90 // for documentation purposes even though they have no effect.
91#if defined(__GNUC__) && !defined(__clang__)
92#pragma GCC diagnostic push
93#pragma GCC diagnostic ignored "-Wformat"
94#endif
Richard Smithf80a27e2016-10-14 19:51:36 +000095 if (fscanf(StatFile,
96 "%*d %*s %*c %*d %*d %*d %*d %*d %*u %*lu %*lu %*lu %*lu %*lu "
97 "%*lu %*ld %*ld %*ld %*ld %*ld %*ld %*llu %*lu %*ld %*lu %*lu "
98 "%*lu %*lu %lu %*lu %*lu %*lu %*lu %*lu %*llu %*lu %*lu %*d %*d "
99 "%*u %*u %*llu %*lu %*ld %*lu %*lu %*lu %*lu %*lu %*lu %lu %*d",
100 &StackPtr, &EnvEnd) == 2) {
Richard Smith49db68d2016-10-15 01:59:52 +0000101#if defined(__GNUC__) && !defined(__clang__)
102#pragma GCC diagnostic pop
103#endif
Richard Smithf80a27e2016-10-14 19:51:36 +0000104 Usage = StackPtr < EnvEnd ? EnvEnd - StackPtr : StackPtr - EnvEnd;
105 }
106 fclose(StatFile);
107 }
108 return Usage;
109}
110
111#include <alloca.h>
112
Richard Smith194b6a32016-08-17 01:05:07 +0000113LLVM_ATTRIBUTE_NOINLINE
Richard Smith0a7b2972018-07-03 21:34:13 +0000114static void ensureStackAddressSpace() {
Richard Smith194b6a32016-08-17 01:05:07 +0000115 // Linux kernels prior to 4.1 will sometimes locate the heap of a PIE binary
116 // relatively close to the stack (they are only guaranteed to be 128MiB
117 // apart). This results in crashes if we happen to heap-allocate more than
118 // 128MiB before we reach our stack high-water mark.
119 //
120 // To avoid these crashes, ensure that we have sufficient virtual memory
Richard Smithf80a27e2016-10-14 19:51:36 +0000121 // pages allocated before we start running.
122 size_t Curr = getCurrentStackAllocation();
Richard Smith0a7b2972018-07-03 21:34:13 +0000123 const int kTargetStack = DesiredStackSize - 256 * 1024;
Richard Smithf80a27e2016-10-14 19:51:36 +0000124 if (Curr < kTargetStack) {
125 volatile char *volatile Alloc =
126 static_cast<volatile char *>(alloca(kTargetStack - Curr));
127 Alloc[0] = 0;
128 Alloc[kTargetStack - Curr - 1] = 0;
129 }
Richard Smith194b6a32016-08-17 01:05:07 +0000130}
131#else
132static void ensureStackAddressSpace() {}
133#endif
134
135/// Attempt to ensure that we have at least 8MiB of usable stack space.
136static void ensureSufficientStack() {
137 struct rlimit rlim;
138 if (getrlimit(RLIMIT_STACK, &rlim) != 0)
139 return;
140
141 // Increase the soft stack limit to our desired level, if necessary and
142 // possible.
Fangrui Song4eab1ba2018-07-25 06:57:31 +0000143 if (rlim.rlim_cur != RLIM_INFINITY &&
144 rlim.rlim_cur < rlim_t(DesiredStackSize)) {
Richard Smith194b6a32016-08-17 01:05:07 +0000145 // Try to allocate sufficient stack.
Fangrui Song4eab1ba2018-07-25 06:57:31 +0000146 if (rlim.rlim_max == RLIM_INFINITY ||
147 rlim.rlim_max >= rlim_t(DesiredStackSize))
Richard Smith0a7b2972018-07-03 21:34:13 +0000148 rlim.rlim_cur = DesiredStackSize;
Richard Smith194b6a32016-08-17 01:05:07 +0000149 else if (rlim.rlim_cur == rlim.rlim_max)
150 return;
151 else
152 rlim.rlim_cur = rlim.rlim_max;
153
154 if (setrlimit(RLIMIT_STACK, &rlim) != 0 ||
Richard Smith0a7b2972018-07-03 21:34:13 +0000155 rlim.rlim_cur != DesiredStackSize)
Richard Smith194b6a32016-08-17 01:05:07 +0000156 return;
157 }
158
Richard Smith0a7b2972018-07-03 21:34:13 +0000159 // We should now have a stack of size at least DesiredStackSize. Ensure
Richard Smith194b6a32016-08-17 01:05:07 +0000160 // that we can actually use that much, if necessary.
161 ensureStackAddressSpace();
162}
163#else
Richard Smith525ce252016-08-17 02:22:39 +0000164static void ensureSufficientStack() {}
Richard Smith194b6a32016-08-17 01:05:07 +0000165#endif
166
Sean Silva070cd2d2014-08-15 21:38:36 +0000167int cc1_main(ArrayRef<const char *> Argv, const char *Argv0, void *MainAddr) {
Richard Smith194b6a32016-08-17 01:05:07 +0000168 ensureSufficientStack();
169
Adrian Prantlfb2398d2015-07-17 01:19:54 +0000170 std::unique_ptr<CompilerInstance> Clang(new CompilerInstance());
Dylan Noblesmithc95d8192012-02-20 14:00:23 +0000171 IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs());
Daniel Dunbare922d9b2010-02-16 01:54:47 +0000172
Adrian Prantlfb2398d2015-07-17 01:19:54 +0000173 // Register the support for object-file-wrapped Clang modules.
174 auto PCHOps = Clang->getPCHContainerOperations();
175 PCHOps->registerWriter(llvm::make_unique<ObjectFilePCHContainerWriter>());
176 PCHOps->registerReader(llvm::make_unique<ObjectFilePCHContainerReader>());
177
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000178 // Initialize targets first, so that --version shows registered targets.
179 llvm::InitializeAllTargets();
Evan Chengc391a582011-07-22 21:59:11 +0000180 llvm::InitializeAllTargetMCs();
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000181 llvm::InitializeAllAsmPrinters();
Chris Lattner61955ab2010-04-05 23:33:20 +0000182 llvm::InitializeAllAsmParsers();
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000183
Sebastian Pop17fac042014-03-14 04:04:27 +0000184#ifdef LINK_POLLY_INTO_TOOLS
185 llvm::PassRegistry &Registry = *llvm::PassRegistry::getPassRegistry();
186 polly::initializePollyPasses(Registry);
187#endif
188
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000189 // Buffer diagnostics from argument parsing so that we can output them using a
190 // well formed diagnostic object.
Douglas Gregor811db4e2012-10-23 22:26:28 +0000191 IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions();
Douglas Gregor2dd19f12010-08-18 22:29:43 +0000192 TextDiagnosticBuffer *DiagsBuffer = new TextDiagnosticBuffer;
Douglas Gregor811db4e2012-10-23 22:26:28 +0000193 DiagnosticsEngine Diags(DiagID, &*DiagOpts, DiagsBuffer);
Sean Silva070cd2d2014-08-15 21:38:36 +0000194 bool Success = CompilerInvocation::CreateFromArgs(
195 Clang->getInvocation(), Argv.begin(), Argv.end(), Diags);
Daniel Dunbard6136772009-12-13 03:45:58 +0000196
197 // Infer the builtin include path if unspecified.
Daniel Dunbar30b24e92010-03-23 05:09:16 +0000198 if (Clang->getHeaderSearchOpts().UseBuiltinIncludes &&
199 Clang->getHeaderSearchOpts().ResourceDir.empty())
200 Clang->getHeaderSearchOpts().ResourceDir =
Daniel Dunbara5a166d2009-12-15 00:06:45 +0000201 CompilerInvocation::GetResourcesPath(Argv0, MainAddr);
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000202
Daniel Dunbar1cdf04c2010-08-12 02:53:07 +0000203 // Create the actual diagnostics engine.
Sean Silvaf1b49e22013-01-20 01:58:28 +0000204 Clang->createDiagnostics();
Daniel Dunbar1cdf04c2010-08-12 02:53:07 +0000205 if (!Clang->hasDiagnostics())
206 return 1;
207
208 // Set an error handler, so that any LLVM backend diagnostics go through our
209 // error handler.
210 llvm::install_fatal_error_handler(LLVMErrorHandler,
211 static_cast<void*>(&Clang->getDiagnostics()));
212
Douglas Gregor2dd19f12010-08-18 22:29:43 +0000213 DiagsBuffer->FlushDiagnostics(Clang->getDiagnostics());
Argyrios Kyrtzidisbe6d89d2012-01-25 20:00:43 +0000214 if (!Success)
215 return 1;
Daniel Dunbar1cdf04c2010-08-12 02:53:07 +0000216
Daniel Dunbare9e91b02010-08-12 02:53:12 +0000217 // Execute the frontend actions.
Dylan Noblesmithe99b27f2011-12-23 03:05:38 +0000218 Success = ExecuteCompilerInvocation(Clang.get());
Daniel Dunbar30b24e92010-03-23 05:09:16 +0000219
Chris Lattner09f8cc82010-03-30 05:39:52 +0000220 // If any timers were active but haven't been destroyed yet, print their
221 // results now. This happens in -disable-free mode.
222 llvm::TimerGroup::printAll(llvm::errs());
Daniel Dunbar2be96742010-08-02 15:31:28 +0000223
Dan Gohmanb37af7d2010-08-18 21:23:17 +0000224 // Our error handler depends on the Diagnostics object, which we're
225 // potentially about to delete. Uninstall the handler now so that any
226 // later errors use the default handling behavior instead.
227 llvm::remove_fatal_error_handler();
228
Daniel Dunbar30b24e92010-03-23 05:09:16 +0000229 // When running with -disable-free, don't do any destruction or shutdown.
230 if (Clang->getFrontendOpts().DisableFree) {
David Blaikie3c13a7f2014-08-29 17:02:26 +0000231 BuryPointer(std::move(Clang));
Daniel Dunbar30b24e92010-03-23 05:09:16 +0000232 return !Success;
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000233 }
234
Daniel Dunbar4f2bc552010-01-13 00:48:06 +0000235 return !Success;
Daniel Dunbarf72bdf72009-12-11 22:20:12 +0000236}