driver.cpp revision 13124bb43061cdaae15b1d0dbfdd0b83565570e2
13ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//===-- driver.cpp - Clang GCC-Compatible Driver --------------------------===//
23ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//
33ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//                     The LLVM Compiler Infrastructure
43ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//
53ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar// This file is distributed under the University of Illinois Open Source
63ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar// License. See LICENSE.TXT for details.
73ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//
83ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//===----------------------------------------------------------------------===//
93ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//
101eb4e64eed08837b85a375d6a953503daa844f07Daniel Dunbar// This is the entry point to the clang driver; it is a thin wrapper
111eb4e64eed08837b85a375d6a953503daa844f07Daniel Dunbar// for functionality in the Driver clang library.
123ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//
133ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar//===----------------------------------------------------------------------===//
143ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar
153f6f51e28231f65de9c2dd150a2d757b2162cfa3Jordan Rose#include "clang/Basic/CharInfo.h"
1602c23ebf41ae2f70da0ba7337e05c51fbfe35f7fDouglas Gregor#include "clang/Basic/DiagnosticOptions.h"
173ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar#include "clang/Driver/Compilation.h"
183ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar#include "clang/Driver/Driver.h"
19a57b3b4fea1c962678972af8fc74e121ff8b296dRichard Smith#include "clang/Driver/DriverDiagnostic.h"
20f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "clang/Driver/Options.h"
212dec85b21822f950bf6035640c496ad835e11728Chad Rosier#include "clang/Frontend/CompilerInvocation.h"
22af20afb761a2426cd715fb8db36b90092e0bb6efDaniel Dunbar#include "clang/Frontend/TextDiagnosticPrinter.h"
232dec85b21822f950bf6035640c496ad835e11728Chad Rosier#include "clang/Frontend/Utils.h"
247f9fc3f7ce076645cb6aefc99c64d9446caf13d6Chris Lattner#include "llvm/ADT/ArrayRef.h"
25f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "llvm/ADT/OwningPtr.h"
26510d73200ec48496a2b10703385fe99b51e31fa5Daniel Dunbar#include "llvm/ADT/SmallString.h"
278a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola#include "llvm/ADT/SmallVector.h"
28429d3be1ae25df8d6451f8864188ab34451a6a10Craig Topper#include "llvm/ADT/STLExtras.h"
29b1e25a1bc03292dc538d336573e0be1490223171Reid Kleckner#include "llvm/Option/ArgList.h"
30b1e25a1bc03292dc538d336573e0be1490223171Reid Kleckner#include "llvm/Option/OptTable.h"
31b1e25a1bc03292dc538d336573e0be1490223171Reid Kleckner#include "llvm/Option/Option.h"
3287f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner#include "llvm/Support/CommandLine.h"
338a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola#include "llvm/Support/ErrorHandling.h"
34fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer#include "llvm/Support/FileSystem.h"
35f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "llvm/Support/Host.h"
368f25c79e59d88df41c5a3cabba1c58035d384a6eDaniel Dunbar#include "llvm/Support/ManagedStatic.h"
378a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola#include "llvm/Support/MemoryBuffer.h"
3803013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Path.h"
39f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "llvm/Support/PrettyStackTrace.h"
4003013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Program.h"
41f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "llvm/Support/Regex.h"
4203013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Signals.h"
43a6b4045dc462c03cd1e7cb9c3ec9dbfdb9c6ad62Evan Cheng#include "llvm/Support/TargetRegistry.h"
44a6b4045dc462c03cd1e7cb9c3ec9dbfdb9c6ad62Evan Cheng#include "llvm/Support/TargetSelect.h"
45f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "llvm/Support/Timer.h"
46f59edb96b2d0bfe612b732f19519ab84bb995bd4Chandler Carruth#include "llvm/Support/raw_ostream.h"
473a321e23f66128dbb986343927456ff6702af617Michael J. Spencer#include "llvm/Support/system_error.h"
484ad4b3ebbe5769143389dccfcfadb666a4ba5940Daniel Dunbarusing namespace clang;
491b3bb6efc59a21f794b534078f9ae7e95393f510Daniel Dunbarusing namespace clang::driver;
50b1e25a1bc03292dc538d336573e0be1490223171Reid Klecknerusing namespace llvm::opt;
513ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar
52ac1db6b2bde50e9a0f3846243f654f6c13682725Rafael Espindolastd::string GetExecutablePath(const char *Argv0, bool CanonicalPrefixes) {
530f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola  if (!CanonicalPrefixes)
54ac1db6b2bde50e9a0f3846243f654f6c13682725Rafael Espindola    return Argv0;
550f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola
56734932c7dc77a16b15d466588e10d76bcd6d13c7Daniel Dunbar  // This just needs to be some symbol in the binary; C++ doesn't
57734932c7dc77a16b15d466588e10d76bcd6d13c7Daniel Dunbar  // allow taking the address of ::main however.
58734932c7dc77a16b15d466588e10d76bcd6d13c7Daniel Dunbar  void *P = (void*) (intptr_t) GetExecutablePath;
59ac1db6b2bde50e9a0f3846243f654f6c13682725Rafael Espindola  return llvm::sys::fs::getMainExecutable(Argv0, P);
60734932c7dc77a16b15d466588e10d76bcd6d13c7Daniel Dunbar}
61734932c7dc77a16b15d466588e10d76bcd6d13c7Daniel Dunbar
62237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbarstatic const char *SaveStringInSet(std::set<std::string> &SavedStrings,
635f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner                                   StringRef S) {
64ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar  return SavedStrings.insert(S).first->c_str();
65ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar}
66ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar
67ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar/// ApplyQAOverride - Apply a list of edits to the input argument lists.
68ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
69ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar/// The input string is a space separate list of edits to perform,
70ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar/// they are applied in order to the input argument lists. Edits
71ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar/// should be one of the following forms:
72ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
73e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar///  '#': Silence information about the changes to the command line arguments.
74e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar///
75ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  '^': Add FOO as a new argument at the beginning of the command line.
76ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
77ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  '+': Add FOO as a new argument at the end of the command line.
78ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
790de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar///  's/XXX/YYY/': Substitute the regular expression XXX with YYY in the command
800de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar///  line.
81ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
82ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  'xOPTION': Removes all instances of the literal argument OPTION.
83ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
84ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  'XOPTION': Removes all instances of the literal argument OPTION,
85ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  and the following argument.
86ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///
87ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  'Ox': Removes all flags matching 'O' or 'O[sz0-9]' and adds 'Ox'
88ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar///  at the end of the command line.
89e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar///
90e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar/// \param OS - The stream to write edit information to.
91e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar/// \param Args - The vector of command line arguments.
92e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar/// \param Edit - The override command to perform.
93e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar/// \param SavedStrings - Set to use for storing string representations.
945f9e272e632e951b1efe824cd16acb4d96077930Chris Lattnerstatic void ApplyOneQAOverride(raw_ostream &OS,
955f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner                               SmallVectorImpl<const char*> &Args,
965f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner                               StringRef Edit,
9730bc7e8e560941f730fd94109e81ac600767b0abChris Lattner                               std::set<std::string> &SavedStrings) {
98ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar  // This does not need to be efficient.
99ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar
100237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar  if (Edit[0] == '^') {
101237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    const char *Str =
1020de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar      SaveStringInSet(SavedStrings, Edit.substr(1));
103237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    OS << "### Adding argument " << Str << " at beginning\n";
104237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    Args.insert(Args.begin() + 1, Str);
105237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar  } else if (Edit[0] == '+') {
106237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    const char *Str =
1070de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar      SaveStringInSet(SavedStrings, Edit.substr(1));
108237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    OS << "### Adding argument " << Str << " at end\n";
109237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    Args.push_back(Str);
1100de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar  } else if (Edit[0] == 's' && Edit[1] == '/' && Edit.endswith("/") &&
1115f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner             Edit.slice(2, Edit.size()-1).find('/') != StringRef::npos) {
1125f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef MatchPattern = Edit.substr(2).split('/').first;
1135f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef ReplPattern = Edit.substr(2).split('/').second;
1140de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar    ReplPattern = ReplPattern.slice(0, ReplPattern.size()-1);
1150de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar
1160de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar    for (unsigned i = 1, e = Args.size(); i != e; ++i) {
1170de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar      std::string Repl = llvm::Regex(MatchPattern).sub(ReplPattern, Args[i]);
1180de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar
1190de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar      if (Repl != Args[i]) {
1200de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar        OS << "### Replacing '" << Args[i] << "' with '" << Repl << "'\n";
1210de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar        Args[i] = SaveStringInSet(SavedStrings, Repl);
1220de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar      }
1230de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar    }
124237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar  } else if (Edit[0] == 'x' || Edit[0] == 'X') {
125237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    std::string Option = Edit.substr(1, std::string::npos);
126237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    for (unsigned i = 1; i < Args.size();) {
127237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar      if (Option == Args[i]) {
128237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        OS << "### Deleting argument " << Args[i] << '\n';
129237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        Args.erase(Args.begin() + i);
130237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        if (Edit[0] == 'X') {
131237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar          if (i < Args.size()) {
132237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar            OS << "### Deleting argument " << Args[i] << '\n';
133237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar            Args.erase(Args.begin() + i);
134237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar          } else
135237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar            OS << "### Invalid X edit, end of command line!\n";
136237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        }
137237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar      } else
138237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        ++i;
139237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    }
140237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar  } else if (Edit[0] == 'O') {
141237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    for (unsigned i = 1; i < Args.size();) {
142237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar      const char *A = Args[i];
143237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar      if (A[0] == '-' && A[1] == 'O' &&
144237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar          (A[2] == '\0' ||
145237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar           (A[3] == '\0' && (A[2] == 's' || A[2] == 'z' ||
146237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar                             ('0' <= A[2] && A[2] <= '9'))))) {
147237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        OS << "### Deleting argument " << Args[i] << '\n';
148237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        Args.erase(Args.begin() + i);
149237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar      } else
150237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar        ++i;
151237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    }
152237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    OS << "### Adding argument " << Edit << " at end\n";
1530de9a7beb107a4306d05cf3589f10c3e01cfcee2Daniel Dunbar    Args.push_back(SaveStringInSet(SavedStrings, '-' + Edit.str()));
154237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar  } else {
155237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar    OS << "### Unrecognized edit: " << Edit << "\n";
156237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar  }
157ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar}
158ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar
159ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar/// ApplyQAOverride - Apply a comma separate list of edits to the
160ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar/// input argument lists. See ApplyOneQAOverride.
1615f9e272e632e951b1efe824cd16acb4d96077930Chris Lattnerstatic void ApplyQAOverride(SmallVectorImpl<const char*> &Args,
16230bc7e8e560941f730fd94109e81ac600767b0abChris Lattner                            const char *OverrideStr,
16330bc7e8e560941f730fd94109e81ac600767b0abChris Lattner                            std::set<std::string> &SavedStrings) {
1645f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  raw_ostream *OS = &llvm::errs();
165237a31bf663c8c3783bd71dc7cfbdd0425a8ba69Daniel Dunbar
166e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar  if (OverrideStr[0] == '#') {
167e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar    ++OverrideStr;
168e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar    OS = &llvm::nulls();
169e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar  }
170e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar
171e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar  *OS << "### QA_OVERRIDE_GCC3_OPTIONS: " << OverrideStr << "\n";
172ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar
173ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar  // This does not need to be efficient.
174ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar
175ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar  const char *S = OverrideStr;
176ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar  while (*S) {
177ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar    const char *End = ::strchr(S, ' ');
178ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar    if (!End)
179ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar      End = S + strlen(S);
180ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar    if (End != S)
181e3d6023cbf99a24609f52adb0af1fa1c6dafefd7Daniel Dunbar      ApplyOneQAOverride(*OS, Args, std::string(S, End), SavedStrings);
182ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar    S = End;
183ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar    if (*S != '\0')
184ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar      ++S;
185ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar  }
186ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar}
187ec9587d5bed6149f6df8b57192bb787c62aedb1bDaniel Dunbar
188c88aa79a6bdfd445df800006f723f83f39934e98Daniel Dunbarextern int cc1_main(const char **ArgBegin, const char **ArgEnd,
189545c28117c64e78521b8b402caa060ba0b58a530Daniel Dunbar                    const char *Argv0, void *MainAddr);
19041b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbarextern int cc1as_main(const char **ArgBegin, const char **ArgEnd,
19141b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar                      const char *Argv0, void *MainAddr);
192217acbfa3524d5805fda7900b26c1e779443588dDaniel Dunbar
1935f9e272e632e951b1efe824cd16acb4d96077930Chris Lattnerstatic void ParseProgName(SmallVectorImpl<const char *> &ArgVector,
1940ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger                          std::set<std::string> &SavedStrings,
1950ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger                          Driver &TheDriver)
1960ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger{
1970ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // Try to infer frontend type and default target from the program name.
1980ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
1990ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // suffixes[] contains the list of known driver suffixes.
2000ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // Suffixes are compared against the program name in order.
2010ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // If there is a match, the frontend type is updated as necessary (CPP/C++).
2020ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // If there is no match, a second round is done after stripping the last
2030ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // hyphen and everything following it. This allows using something like
2040ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // "clang++-2.9".
2050ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
2060ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // If there is a match in either the first or second round,
2070ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // the function tries to identify a target as prefix. E.g.
2080ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // "x86_64-linux-clang" as interpreted as suffix "clang" with
2090ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  // target prefix "x86_64-linux". If such a target prefix is found,
2109ec60dfe771ff28a84889dced6f8fd3748d3d55eSebastian Pop  // is gets added via -target as implicit first argument.
2110ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  static const struct {
2120ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    const char *Suffix;
21376b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    const char *ModeFlag;
2140ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  } suffixes [] = {
21576b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang",     0 },
21676b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang++",   "--driver-mode=g++" },
21776b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang-c++", "--driver-mode=g++" },
21876b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang-cc",  0 },
21976b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang-cpp", "--driver-mode=cpp" },
22076b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang-g++", "--driver-mode=g++" },
22176b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "clang-gcc", 0 },
222c2f531a1300cc7a79cb8dde12cb993da82beba1bHans Wennborg    { "clang-cl",  "--driver-mode=cl"  },
22376b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "cc",        0 },
22476b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "cpp",       "--driver-mode=cpp" },
225c2f531a1300cc7a79cb8dde12cb993da82beba1bHans Wennborg    { "cl" ,       "--driver-mode=cl"  },
22676b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg    { "++",        "--driver-mode=g++" },
2270ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  };
2280ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  std::string ProgName(llvm::sys::path::stem(ArgVector[0]));
22913124bb43061cdaae15b1d0dbfdd0b83565570e2Hans Wennborg  std::transform(ProgName.begin(), ProgName.end(), ProgName.begin(),
23013124bb43061cdaae15b1d0dbfdd0b83565570e2Hans Wennborg                 toLowercase);
2315f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef ProgNameRef(ProgName);
2325f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Prefix;
2330ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
2340ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  for (int Components = 2; Components; --Components) {
2350ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    bool FoundMatch = false;
2360ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    size_t i;
2370ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
23826a4001a4bbc33ebdf6102d604d6b793bdbffa99Craig Topper    for (i = 0; i < sizeof(suffixes) / sizeof(suffixes[0]); ++i) {
2390ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger      if (ProgNameRef.endswith(suffixes[i].Suffix)) {
2400ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger        FoundMatch = true;
24176b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg        SmallVectorImpl<const char *>::iterator it = ArgVector.begin();
24276b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg        if (it != ArgVector.end())
24376b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg          ++it;
24476b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg        if (suffixes[i].ModeFlag)
24576b86c2e79a58910a641d449f229889d671d8a38Hans Wennborg          ArgVector.insert(it, suffixes[i].ModeFlag);
2460ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger        break;
2470ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger      }
2480ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    }
2490ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
2500ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    if (FoundMatch) {
2515f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner      StringRef::size_type LastComponent = ProgNameRef.rfind('-',
2520ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger        ProgNameRef.size() - strlen(suffixes[i].Suffix));
2535f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner      if (LastComponent != StringRef::npos)
2540ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger        Prefix = ProgNameRef.slice(0, LastComponent);
2550ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger      break;
2560ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    }
2570ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
2585f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef::size_type LastComponent = ProgNameRef.rfind('-');
2595f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (LastComponent == StringRef::npos)
2600ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger      break;
2610ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    ProgNameRef = ProgNameRef.slice(0, LastComponent);
2620ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  }
2630ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
2640ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  if (Prefix.empty())
2650ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger    return;
2660ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
2670ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  std::string IgnoredError;
2680ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  if (llvm::TargetRegistry::lookupTarget(Prefix, IgnoredError)) {
2695f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    SmallVectorImpl<const char *>::iterator it = ArgVector.begin();
270ca0264320a02c636beaedf834e8725b7d1d767e7Joerg Sonnenberger    if (it != ArgVector.end())
271ca0264320a02c636beaedf834e8725b7d1d767e7Joerg Sonnenberger      ++it;
272ca0264320a02c636beaedf834e8725b7d1d767e7Joerg Sonnenberger    ArgVector.insert(it, SaveStringInSet(SavedStrings, Prefix));
273ca0264320a02c636beaedf834e8725b7d1d767e7Joerg Sonnenberger    ArgVector.insert(it,
2749ec60dfe771ff28a84889dced6f8fd3748d3d55eSebastian Pop      SaveStringInSet(SavedStrings, std::string("-target")));
2750ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  }
2760ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger}
2770ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger
27887f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Klecknernamespace {
27987f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  class StringSetSaver : public llvm::cl::StringSaver {
28087f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  public:
28187f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner    StringSetSaver(std::set<std::string> &Storage) : Storage(Storage) {}
28287f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner    const char *SaveString(const char *Str) LLVM_OVERRIDE {
28387f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner      return SaveStringInSet(Storage, Str);
28487f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner    }
28587f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  private:
28687f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner    std::set<std::string> &Storage;
28787f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  };
28887f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner}
28987f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner
2908a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindolaint main(int argc_, const char **argv_) {
2913ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar  llvm::sys::PrintStackTraceOnErrorSignal();
2928a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola  llvm::PrettyStackTraceProgram X(argc_, argv_);
2938a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola
2948a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola  std::set<std::string> SavedStrings;
29587f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  SmallVector<const char*, 256> argv(argv_, argv_ + argc_);
29687f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  StringSetSaver Saver(SavedStrings);
29787f2cb5aec7b81ea6f6f9fbf6e14ef4f2de6dea6Reid Kleckner  llvm::cl::ExpandResponseFiles(Saver, llvm::cl::TokenizeGNUCommandLine, argv);
2983ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar
29941b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar  // Handle -cc1 integrated tools.
3005f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  if (argv.size() > 1 && StringRef(argv[1]).startswith("-cc1")) {
3015f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Tool = argv[1] + 4;
30241b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar
30341b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar    if (Tool == "")
304e5d69670b6e49d11287bc8f031e0e2081755cfacDaniel Dunbar      return cc1_main(argv.data()+2, argv.data()+argv.size(), argv[0],
30541b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar                      (void*) (intptr_t) GetExecutablePath);
30641b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar    if (Tool == "as")
307e5d69670b6e49d11287bc8f031e0e2081755cfacDaniel Dunbar      return cc1as_main(argv.data()+2, argv.data()+argv.size(), argv[0],
30841b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar                      (void*) (intptr_t) GetExecutablePath);
30941b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar
31041b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar    // Reject unknown tools.
31141b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar    llvm::errs() << "error: unknown integrated tool '" << Tool << "'\n";
31241b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar    return 1;
31341b5b17445ab3bdf957ebd4be6c8670f09a212a8Daniel Dunbar  }
314c88aa79a6bdfd445df800006f723f83f39934e98Daniel Dunbar
3150f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola  bool CanonicalPrefixes = true;
3168a1af325b424c3cf62a6164a43466b473ec5a666Rafael Espindola  for (int i = 1, size = argv.size(); i < size; ++i) {
3175f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (StringRef(argv[i]) == "-no-canonical-prefixes") {
3180f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola      CanonicalPrefixes = false;
3190f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola      break;
3200f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola    }
3210f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola  }
3220f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola
32379165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier  // Handle QA_OVERRIDE_GCC3_OPTIONS and CCC_ADD_ARGS, used for editing a
32479165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier  // command line behind the scenes.
32579165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier  if (const char *OverrideStr = ::getenv("QA_OVERRIDE_GCC3_OPTIONS")) {
32679165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier    // FIXME: Driver shouldn't take extra initial argument.
32779165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier    ApplyQAOverride(argv, OverrideStr, SavedStrings);
32879165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier  }
32979165b82ddc881c705275fe9eb5a745f717a1edaChad Rosier
330ac1db6b2bde50e9a0f3846243f654f6c13682725Rafael Espindola  std::string Path = GetExecutablePath(argv[0], CanonicalPrefixes);
3310f4c59c3beae302b5a3ecea13f2e257bdaad3188Rafael Espindola
33202c23ebf41ae2f70da0ba7337e05c51fbfe35f7fDouglas Gregor  IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions;
3332dec85b21822f950bf6035640c496ad835e11728Chad Rosier  {
334f3db6835d15c9d0f6345d78c1569e730dfdbb93eHans Wennborg    OwningPtr<OptTable> Opts(createDriverOptTable());
3352dec85b21822f950bf6035640c496ad835e11728Chad Rosier    unsigned MissingArgIndex, MissingArgCount;
336f3db6835d15c9d0f6345d78c1569e730dfdbb93eHans Wennborg    OwningPtr<InputArgList> Args(Opts->ParseArgs(argv.begin()+1, argv.end(),
337f3db6835d15c9d0f6345d78c1569e730dfdbb93eHans Wennborg                                                 MissingArgIndex,
338f3db6835d15c9d0f6345d78c1569e730dfdbb93eHans Wennborg                                                 MissingArgCount));
3392dec85b21822f950bf6035640c496ad835e11728Chad Rosier    // We ignore MissingArgCount and the return value of ParseDiagnosticArgs.
3402dec85b21822f950bf6035640c496ad835e11728Chad Rosier    // Any errors that would be diagnosed here will also be diagnosed later,
3412dec85b21822f950bf6035640c496ad835e11728Chad Rosier    // when the DiagnosticsEngine actually exists.
34202c23ebf41ae2f70da0ba7337e05c51fbfe35f7fDouglas Gregor    (void) ParseDiagnosticArgs(*DiagOpts, *Args);
3432dec85b21822f950bf6035640c496ad835e11728Chad Rosier  }
3442dec85b21822f950bf6035640c496ad835e11728Chad Rosier  // Now we can create the DiagnosticsEngine with a properly-filled-out
3452dec85b21822f950bf6035640c496ad835e11728Chad Rosier  // DiagnosticOptions instance.
346bdbb004f38978da0c4a75af3294d1c7b5ff84af1Douglas Gregor  TextDiagnosticPrinter *DiagClient
34702c23ebf41ae2f70da0ba7337e05c51fbfe35f7fDouglas Gregor    = new TextDiagnosticPrinter(llvm::errs(), &*DiagOpts);
348ac1db6b2bde50e9a0f3846243f654f6c13682725Rafael Espindola  DiagClient->setPrefix(llvm::sys::path::filename(Path));
349c93dc7889644293e318e19d82830ea2acc45b678Dylan Noblesmith  IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs());
3502dec85b21822f950bf6035640c496ad835e11728Chad Rosier
35102c23ebf41ae2f70da0ba7337e05c51fbfe35f7fDouglas Gregor  DiagnosticsEngine Diags(DiagID, &*DiagOpts, DiagClient);
35208e79d2f075d1326ff7a849fd1bc88dadf99d574Chad Rosier  ProcessWarningOptions(Diags, *DiagOpts, /*ReportDiags=*/false);
353510d73200ec48496a2b10703385fe99b51e31fa5Daniel Dunbar
354ac1db6b2bde50e9a0f3846243f654f6c13682725Rafael Espindola  Driver TheDriver(Path, llvm::sys::getDefaultTargetTriple(), "a.out", Diags);
355e5be6dac24946e9521af78fce8e531e605f563d4Daniel Dunbar
356edf29b00bc1641034b38be92146e15f0625a45d2Daniel Dunbar  // Attempt to find the original path used to invoke the driver, to determine
357edf29b00bc1641034b38be92146e15f0625a45d2Daniel Dunbar  // the installed path. We do this manually, because we want to support that
358edf29b00bc1641034b38be92146e15f0625a45d2Daniel Dunbar  // path being a symlink.
359fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer  {
36036d592718ff342f762e32cbde73d1113f88cb275Dylan Noblesmith    SmallString<128> InstalledPath(argv[0]);
361fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer
362fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    // Do a PATH lookup, if there are no directory components.
363fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    if (llvm::sys::path::filename(InstalledPath) == InstalledPath) {
3642289933d3dfad854c006c02738abb1c57d1d587eRafael Espindola      std::string Tmp = llvm::sys::FindProgramByName(
365fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer        llvm::sys::path::filename(InstalledPath.str()));
366fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer      if (!Tmp.empty())
3672289933d3dfad854c006c02738abb1c57d1d587eRafael Espindola        InstalledPath = Tmp;
368fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    }
369fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    llvm::sys::fs::make_absolute(InstalledPath);
370fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    InstalledPath = llvm::sys::path::parent_path(InstalledPath);
371fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    bool exists;
372fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer    if (!llvm::sys::fs::exists(InstalledPath.str(), exists) && exists)
373fbfd180495e7800975c6d9bdc6d24e706ef70e34Michael J. Spencer      TheDriver.setInstalledDir(InstalledPath);
374edf29b00bc1641034b38be92146e15f0625a45d2Daniel Dunbar  }
375edf29b00bc1641034b38be92146e15f0625a45d2Daniel Dunbar
3760ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  llvm::InitializeAllTargets();
3770ce89c6b36004b2def9e7606c7d4678885314d5fJoerg Sonnenberger  ParseProgName(argv, SavedStrings, TheDriver);
3789ade4ae4fb7ed1fcbd63835d9f8f53052f0657a2Joerg Sonnenberger
3794c00fcdf98d3d7c4cb47b64f8b770f8f4bff1357Daniel Dunbar  // Handle CC_PRINT_OPTIONS and CC_PRINT_OPTIONS_FILE.
3804c00fcdf98d3d7c4cb47b64f8b770f8f4bff1357Daniel Dunbar  TheDriver.CCPrintOptions = !!::getenv("CC_PRINT_OPTIONS");
3814c00fcdf98d3d7c4cb47b64f8b770f8f4bff1357Daniel Dunbar  if (TheDriver.CCPrintOptions)
3824c00fcdf98d3d7c4cb47b64f8b770f8f4bff1357Daniel Dunbar    TheDriver.CCPrintOptionsFilename = ::getenv("CC_PRINT_OPTIONS_FILE");
3834c00fcdf98d3d7c4cb47b64f8b770f8f4bff1357Daniel Dunbar
384322c29fefe7fa33f03273136eb5f8b7f5b4df7c0Daniel Dunbar  // Handle CC_PRINT_HEADERS and CC_PRINT_HEADERS_FILE.
385322c29fefe7fa33f03273136eb5f8b7f5b4df7c0Daniel Dunbar  TheDriver.CCPrintHeaders = !!::getenv("CC_PRINT_HEADERS");
386322c29fefe7fa33f03273136eb5f8b7f5b4df7c0Daniel Dunbar  if (TheDriver.CCPrintHeaders)
387322c29fefe7fa33f03273136eb5f8b7f5b4df7c0Daniel Dunbar    TheDriver.CCPrintHeadersFilename = ::getenv("CC_PRINT_HEADERS_FILE");
388322c29fefe7fa33f03273136eb5f8b7f5b4df7c0Daniel Dunbar
389c8a22b091677c02003880166e5d5767ca8d74b70Daniel Dunbar  // Handle CC_LOG_DIAGNOSTICS and CC_LOG_DIAGNOSTICS_FILE.
390c8a22b091677c02003880166e5d5767ca8d74b70Daniel Dunbar  TheDriver.CCLogDiagnostics = !!::getenv("CC_LOG_DIAGNOSTICS");
391c8a22b091677c02003880166e5d5767ca8d74b70Daniel Dunbar  if (TheDriver.CCLogDiagnostics)
392c8a22b091677c02003880166e5d5767ca8d74b70Daniel Dunbar    TheDriver.CCLogDiagnosticsFilename = ::getenv("CC_LOG_DIAGNOSTICS_FILE");
393c8a22b091677c02003880166e5d5767ca8d74b70Daniel Dunbar
3941e4c01b79273b9cd4e9e9ecfd3422df3900b8356Dylan Noblesmith  OwningPtr<Compilation> C(TheDriver.BuildCompilation(argv));
395af96def468042cfbed55a4cc12b1bb917ead4f33Daniel Dunbar  int Res = 0;
396a16355c31878403443f99077cc8df8318457faf5Chad Rosier  SmallVector<std::pair<int, const Command *>, 4> FailingCommands;
397af96def468042cfbed55a4cc12b1bb917ead4f33Daniel Dunbar  if (C.get())
398a16355c31878403443f99077cc8df8318457faf5Chad Rosier    Res = TheDriver.ExecuteCompilation(*C, FailingCommands);
3992b81910618f63e4ce2373c926a26e76b4b91373fChad Rosier
40080ec127fc9af8b83eea326fbd65265a1adc307d8Chad Rosier  // Force a crash to test the diagnostics.
401a57b3b4fea1c962678972af8fc74e121ff8b296dRichard Smith  if (::getenv("FORCE_CLANG_DIAGNOSTICS_CRASH")) {
402a57b3b4fea1c962678972af8fc74e121ff8b296dRichard Smith    Diags.Report(diag::err_drv_force_crash) << "FORCE_CLANG_DIAGNOSTICS_CRASH";
403a16355c31878403443f99077cc8df8318457faf5Chad Rosier    const Command *FailingCommand = 0;
404a16355c31878403443f99077cc8df8318457faf5Chad Rosier    FailingCommands.push_back(std::make_pair(-1, FailingCommand));
405a57b3b4fea1c962678972af8fc74e121ff8b296dRichard Smith  }
40680ec127fc9af8b83eea326fbd65265a1adc307d8Chad Rosier
407a16355c31878403443f99077cc8df8318457faf5Chad Rosier  for (SmallVectorImpl< std::pair<int, const Command *> >::iterator it =
408a16355c31878403443f99077cc8df8318457faf5Chad Rosier         FailingCommands.begin(), ie = FailingCommands.end(); it != ie; ++it) {
409a16355c31878403443f99077cc8df8318457faf5Chad Rosier    int CommandRes = it->first;
410a16355c31878403443f99077cc8df8318457faf5Chad Rosier    const Command *FailingCommand = it->second;
411a16355c31878403443f99077cc8df8318457faf5Chad Rosier    if (!Res)
412a16355c31878403443f99077cc8df8318457faf5Chad Rosier      Res = CommandRes;
413a16355c31878403443f99077cc8df8318457faf5Chad Rosier
414a16355c31878403443f99077cc8df8318457faf5Chad Rosier    // If result status is < 0, then the driver command signalled an error.
415a16355c31878403443f99077cc8df8318457faf5Chad Rosier    // If result status is 70, then the driver command reported a fatal error.
416a16355c31878403443f99077cc8df8318457faf5Chad Rosier    // In these cases, generate additional diagnostic information if possible.
417a16355c31878403443f99077cc8df8318457faf5Chad Rosier    if (CommandRes < 0 || CommandRes == 70) {
418a16355c31878403443f99077cc8df8318457faf5Chad Rosier      TheDriver.generateCompilationDiagnostics(*C, FailingCommand);
419a16355c31878403443f99077cc8df8318457faf5Chad Rosier      break;
420a16355c31878403443f99077cc8df8318457faf5Chad Rosier    }
421a16355c31878403443f99077cc8df8318457faf5Chad Rosier  }
4222b81910618f63e4ce2373c926a26e76b4b91373fChad Rosier
42330bc7e8e560941f730fd94109e81ac600767b0abChris Lattner  // If any timers were active but haven't been destroyed yet, print their
42430bc7e8e560941f730fd94109e81ac600767b0abChris Lattner  // results now.  This happens in -disable-free mode.
42530bc7e8e560941f730fd94109e81ac600767b0abChris Lattner  llvm::TimerGroup::printAll(llvm::errs());
42630bc7e8e560941f730fd94109e81ac600767b0abChris Lattner
4278f25c79e59d88df41c5a3cabba1c58035d384a6eDaniel Dunbar  llvm::llvm_shutdown();
4288f25c79e59d88df41c5a3cabba1c58035d384a6eDaniel Dunbar
429df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi#ifdef _WIN32
430df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi  // Exit status should not be negative on Win32, unless abnormal termination.
431df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi  // Once abnormal termiation was caught, negative status should not be
432df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi  // propagated.
433df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi  if (Res < 0)
434df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi    Res = 1;
435df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi#endif
436df9b933aa354b8108cee028cc4726cfa9f2df62cNAKAMURA Takumi
437a16355c31878403443f99077cc8df8318457faf5Chad Rosier  // If we have multiple failing commands, we return the result of the first
438a16355c31878403443f99077cc8df8318457faf5Chad Rosier  // failing command.
439af96def468042cfbed55a4cc12b1bb917ead4f33Daniel Dunbar  return Res;
4403ede8d0a7d1813f678ccc6011a99a0834b1b6116Daniel Dunbar}
441