1 //===--- MinGWToolChain.cpp - MinGWToolChain Implementation ---------------===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 #include "ToolChains.h"
11 #include "clang/Driver/Driver.h"
12 #include "clang/Driver/Options.h"
13 #include "llvm/Option/ArgList.h"
14 #include "llvm/Support/FileSystem.h"
15 #include "llvm/Support/Path.h"
17 using namespace clang::diag;
18 using namespace clang::driver;
19 using namespace clang::driver::toolchains;
20 using namespace clang;
21 using namespace llvm::opt;
23 // Simplified from Generic_GCC::GCCInstallationDetector::ScanLibDirForGCCTriple.
24 static bool findGccVersion(StringRef LibDir, std::string &GccLibDir,
26 Generic_GCC::GCCVersion Version = Generic_GCC::GCCVersion::Parse("0.0.0");
28 for (llvm::sys::fs::directory_iterator LI(LibDir, EC), LE; !EC && LI != LE;
29 LI = LI.increment(EC)) {
30 StringRef VersionText = llvm::sys::path::filename(LI->path());
31 Generic_GCC::GCCVersion CandidateVersion =
32 Generic_GCC::GCCVersion::Parse(VersionText);
33 if (CandidateVersion.Major == -1)
35 if (CandidateVersion <= Version)
38 GccLibDir = LI->path();
43 void MinGW::findGccLibDir() {
44 llvm::SmallVector<llvm::SmallString<32>, 2> Archs;
45 Archs.emplace_back(getTriple().getArchName());
46 Archs[0] += "-w64-mingw32";
47 Archs.emplace_back("mingw32");
48 Arch = Archs[0].str();
49 // lib: Arch Linux, Ubuntu, Windows
50 // lib64: openSUSE Linux
51 for (StringRef CandidateLib : {"lib", "lib64"}) {
52 for (StringRef CandidateArch : Archs) {
53 llvm::SmallString<1024> LibDir(Base);
54 llvm::sys::path::append(LibDir, CandidateLib, "gcc", CandidateArch);
55 if (findGccVersion(LibDir, GccLibDir, Ver)) {
63 MinGW::MinGW(const Driver &D, const llvm::Triple &Triple, const ArgList &Args)
64 : ToolChain(D, Triple, Args), CudaInstallation(D, Triple, Args) {
65 getProgramPaths().push_back(getDriver().getInstalledDir());
67 // In Windows there aren't any standard install locations, we search
68 // for gcc on the PATH. In Linux the base is always /usr.
70 if (getDriver().SysRoot.size())
71 Base = getDriver().SysRoot;
72 else if (llvm::ErrorOr<std::string> GPPName =
73 llvm::sys::findProgramByName("gcc"))
74 Base = llvm::sys::path::parent_path(
75 llvm::sys::path::parent_path(GPPName.get()));
77 Base = llvm::sys::path::parent_path(getDriver().getInstalledDir());
79 if (getDriver().SysRoot.size())
80 Base = getDriver().SysRoot;
85 Base += llvm::sys::path::get_separator();
87 // GccLibDir must precede Base/lib so that the
88 // correct crtbegin.o ,cetend.o would be found.
89 getFilePaths().push_back(GccLibDir);
90 getFilePaths().push_back(
91 (Base + Arch + llvm::sys::path::get_separator() + "lib").str());
92 getFilePaths().push_back(Base + "lib");
94 getFilePaths().push_back(Base + Arch + "/sys-root/mingw/lib");
97 bool MinGW::IsIntegratedAssemblerDefault() const { return true; }
99 Tool *MinGW::getTool(Action::ActionClass AC) const {
101 case Action::PreprocessJobClass:
103 Preprocessor.reset(new tools::gcc::Preprocessor(*this));
104 return Preprocessor.get();
105 case Action::CompileJobClass:
107 Compiler.reset(new tools::gcc::Compiler(*this));
108 return Compiler.get();
110 return ToolChain::getTool(AC);
114 Tool *MinGW::buildAssembler() const {
115 return new tools::MinGW::Assembler(*this);
118 Tool *MinGW::buildLinker() const { return new tools::MinGW::Linker(*this); }
120 bool MinGW::IsUnwindTablesDefault() const {
121 return getArch() == llvm::Triple::x86_64;
124 bool MinGW::isPICDefault() const { return getArch() == llvm::Triple::x86_64; }
126 bool MinGW::isPIEDefault() const { return false; }
128 bool MinGW::isPICDefaultForced() const {
129 return getArch() == llvm::Triple::x86_64;
132 bool MinGW::UseSEHExceptions() const {
133 return getArch() == llvm::Triple::x86_64;
136 void MinGW::AddCudaIncludeArgs(const ArgList &DriverArgs,
137 ArgStringList &CC1Args) const {
138 CudaInstallation.AddCudaIncludeArgs(DriverArgs, CC1Args);
141 void MinGW::printVerboseInfo(raw_ostream &OS) const {
142 CudaInstallation.print(OS);
145 // Include directories for various hosts:
147 // Windows, mingw.org
148 // c:\mingw\lib\gcc\mingw32\4.8.1\include\c++
149 // c:\mingw\lib\gcc\mingw32\4.8.1\include\c++\mingw32
150 // c:\mingw\lib\gcc\mingw32\4.8.1\include\c++\backward
151 // c:\mingw\lib\gcc\mingw32\4.8.1\include
153 // c:\mingw\lib\gcc\mingw32\4.8.1\include-fixed
154 // c:\mingw\mingw32\include
156 // Windows, mingw-w64 mingw-builds
157 // c:\mingw32\lib\gcc\i686-w64-mingw32\4.9.1\include
158 // c:\mingw32\lib\gcc\i686-w64-mingw32\4.9.1\include-fixed
159 // c:\mingw32\i686-w64-mingw32\include
160 // c:\mingw32\i686-w64-mingw32\include\c++
161 // c:\mingw32\i686-w64-mingw32\include\c++\i686-w64-mingw32
162 // c:\mingw32\i686-w64-mingw32\include\c++\backward
164 // Windows, mingw-w64 msys2
165 // c:\msys64\mingw32\lib\gcc\i686-w64-mingw32\4.9.2\include
166 // c:\msys64\mingw32\include
167 // c:\msys64\mingw32\lib\gcc\i686-w64-mingw32\4.9.2\include-fixed
168 // c:\msys64\mingw32\i686-w64-mingw32\include
169 // c:\msys64\mingw32\include\c++\4.9.2
170 // c:\msys64\mingw32\include\c++\4.9.2\i686-w64-mingw32
171 // c:\msys64\mingw32\include\c++\4.9.2\backward
174 // /usr/lib64/gcc/x86_64-w64-mingw32/5.1.0/include/c++
175 // /usr/lib64/gcc/x86_64-w64-mingw32/5.1.0/include/c++/x86_64-w64-mingw32
176 // /usr/lib64/gcc/x86_64-w64-mingw32/5.1.0/include/c++/backward
177 // /usr/lib64/gcc/x86_64-w64-mingw32/5.1.0/include
178 // /usr/lib64/gcc/x86_64-w64-mingw32/5.1.0/include-fixed
179 // /usr/x86_64-w64-mingw32/sys-root/mingw/include
182 // /usr/i686-w64-mingw32/include/c++/5.1.0
183 // /usr/i686-w64-mingw32/include/c++/5.1.0/i686-w64-mingw32
184 // /usr/i686-w64-mingw32/include/c++/5.1.0/backward
185 // /usr/lib/gcc/i686-w64-mingw32/5.1.0/include
186 // /usr/lib/gcc/i686-w64-mingw32/5.1.0/include-fixed
187 // /usr/i686-w64-mingw32/include
190 // /usr/include/c++/4.8
191 // /usr/include/c++/4.8/x86_64-w64-mingw32
192 // /usr/include/c++/4.8/backward
193 // /usr/lib/gcc/x86_64-w64-mingw32/4.8/include
194 // /usr/lib/gcc/x86_64-w64-mingw32/4.8/include-fixed
195 // /usr/x86_64-w64-mingw32/include
197 void MinGW::AddClangSystemIncludeArgs(const ArgList &DriverArgs,
198 ArgStringList &CC1Args) const {
199 if (DriverArgs.hasArg(options::OPT_nostdinc))
202 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
203 SmallString<1024> P(getDriver().ResourceDir);
204 llvm::sys::path::append(P, "include");
205 addSystemInclude(DriverArgs, CC1Args, P.str());
208 if (DriverArgs.hasArg(options::OPT_nostdlibinc))
211 if (GetRuntimeLibType(DriverArgs) == ToolChain::RLT_Libgcc) {
212 llvm::SmallString<1024> IncludeDir(GccLibDir);
213 llvm::sys::path::append(IncludeDir, "include");
214 addSystemInclude(DriverArgs, CC1Args, IncludeDir.c_str());
215 IncludeDir += "-fixed";
217 addSystemInclude(DriverArgs, CC1Args,
218 Base + Arch + "/sys-root/mingw/include");
219 addSystemInclude(DriverArgs, CC1Args, IncludeDir.c_str());
221 addSystemInclude(DriverArgs, CC1Args,
222 Base + Arch + llvm::sys::path::get_separator() + "include");
223 addSystemInclude(DriverArgs, CC1Args, Base + "include");
226 void MinGW::AddClangCXXStdlibIncludeArgs(const ArgList &DriverArgs,
227 ArgStringList &CC1Args) const {
228 if (DriverArgs.hasArg(options::OPT_nostdlibinc) ||
229 DriverArgs.hasArg(options::OPT_nostdincxx))
232 switch (GetCXXStdlibType(DriverArgs)) {
233 case ToolChain::CST_Libcxx:
234 addSystemInclude(DriverArgs, CC1Args,
235 Base + "include" + llvm::sys::path::get_separator() +
236 "c++" + llvm::sys::path::get_separator() + "v1");
239 case ToolChain::CST_Libstdcxx:
240 llvm::SmallVector<llvm::SmallString<1024>, 4> CppIncludeBases;
241 CppIncludeBases.emplace_back(Base);
242 llvm::sys::path::append(CppIncludeBases[0], Arch, "include", "c++");
243 CppIncludeBases.emplace_back(Base);
244 llvm::sys::path::append(CppIncludeBases[1], Arch, "include", "c++", Ver);
245 CppIncludeBases.emplace_back(Base);
246 llvm::sys::path::append(CppIncludeBases[2], "include", "c++", Ver);
247 CppIncludeBases.emplace_back(GccLibDir);
248 llvm::sys::path::append(CppIncludeBases[3], "include", "c++");
249 for (auto &CppIncludeBase : CppIncludeBases) {
250 addSystemInclude(DriverArgs, CC1Args, CppIncludeBase);
251 CppIncludeBase += llvm::sys::path::get_separator();
252 addSystemInclude(DriverArgs, CC1Args, CppIncludeBase + Arch);
253 addSystemInclude(DriverArgs, CC1Args, CppIncludeBase + "backward");