blob: 7b0559e120ee8fd456204f7459f6a95b22009113 [file] [log] [blame]
Daniel Dunbar47ac7d22009-03-18 06:00:36 +00001//===--- Tools.cpp - Tools Implementations ------------------------------*-===//
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
10#include "Tools.h"
11
Daniel Dunbar1d460332009-03-18 10:01:51 +000012#include "clang/Driver/Action.h"
Daniel Dunbar871adcf2009-03-18 07:06:02 +000013#include "clang/Driver/Arg.h"
Daniel Dunbarb488c1d2009-03-18 08:07:30 +000014#include "clang/Driver/ArgList.h"
Daniel Dunbar1d460332009-03-18 10:01:51 +000015#include "clang/Driver/Driver.h" // FIXME: Remove?
16#include "clang/Driver/DriverDiagnostic.h" // FIXME: Remove?
Daniel Dunbar871adcf2009-03-18 07:06:02 +000017#include "clang/Driver/Compilation.h"
18#include "clang/Driver/Job.h"
Daniel Dunbarb488c1d2009-03-18 08:07:30 +000019#include "clang/Driver/HostInfo.h"
20#include "clang/Driver/Option.h"
21#include "clang/Driver/ToolChain.h"
Daniel Dunbar871adcf2009-03-18 07:06:02 +000022#include "clang/Driver/Util.h"
23
Daniel Dunbar88137642009-09-09 22:32:48 +000024#include "llvm/ADT/SmallString.h"
Daniel Dunbar5b750fe2009-09-09 22:32:34 +000025#include "llvm/ADT/Twine.h"
Daniel Dunbar02633b52009-03-26 16:23:12 +000026#include "llvm/Support/Format.h"
27#include "llvm/Support/raw_ostream.h"
Daniel Dunbar871adcf2009-03-18 07:06:02 +000028
29#include "InputInfo.h"
Daniel Dunbar02633b52009-03-26 16:23:12 +000030#include "ToolChains.h"
Daniel Dunbar871adcf2009-03-18 07:06:02 +000031
Daniel Dunbar47ac7d22009-03-18 06:00:36 +000032using namespace clang::driver;
33using namespace clang::driver::tools;
34
Daniel Dunbar868bd0a2009-05-06 03:16:41 +000035static const char *MakeFormattedString(const ArgList &Args,
36 const llvm::format_object_base &Fmt) {
Daniel Dunbar88137642009-09-09 22:32:48 +000037 llvm::SmallString<256> Str;
38 llvm::raw_svector_ostream(Str) << Fmt;
39 return Args.MakeArgString(Str.str());
Daniel Dunbar868bd0a2009-05-06 03:16:41 +000040}
41
Daniel Dunbar88a3d6c2009-09-10 01:21:05 +000042/// CheckPreprocessingOptions - Perform some validation of preprocessing
43/// arguments that is shared with gcc.
44static void CheckPreprocessingOptions(const Driver &D, const ArgList &Args) {
45 if (Arg *A = Args.getLastArg(options::OPT_C, options::OPT_CC))
46 if (!Args.hasArg(options::OPT_E))
47 D.Diag(clang::diag::err_drv_argument_only_allowed_with)
48 << A->getAsString(Args) << "-E";
49}
50
Daniel Dunbare2fd6642009-09-10 01:21:12 +000051/// CheckCodeGenerationOptions - Perform some validation of code generation
52/// arguments that is shared with gcc.
53static void CheckCodeGenerationOptions(const Driver &D, const ArgList &Args) {
54 // In gcc, only ARM checks this, but it seems reasonable to check universally.
55 if (Args.hasArg(options::OPT_static))
56 if (const Arg *A = Args.getLastArg(options::OPT_dynamic,
57 options::OPT_mdynamic_no_pic))
58 D.Diag(clang::diag::err_drv_argument_not_allowed_with)
59 << A->getAsString(Args) << "-static";
60}
61
Mike Stump1eb44332009-09-09 15:08:12 +000062void Clang::AddPreprocessingOptions(const Driver &D,
Douglas Gregordf91ef32009-04-18 00:34:01 +000063 const ArgList &Args,
Daniel Dunbarc21c4852009-04-08 23:54:23 +000064 ArgStringList &CmdArgs,
65 const InputInfo &Output,
66 const InputInfoList &Inputs) const {
Daniel Dunbarc21c4852009-04-08 23:54:23 +000067 Arg *A;
Daniel Dunbar3a183d32009-06-08 21:48:20 +000068
Daniel Dunbar88a3d6c2009-09-10 01:21:05 +000069 CheckPreprocessingOptions(D, Args);
70
71 Args.AddLastArg(CmdArgs, options::OPT_C);
72 Args.AddLastArg(CmdArgs, options::OPT_CC);
Daniel Dunbar3a183d32009-06-08 21:48:20 +000073
74 // Handle dependency file generation.
Daniel Dunbarc21c4852009-04-08 23:54:23 +000075 if ((A = Args.getLastArg(options::OPT_M)) ||
76 (A = Args.getLastArg(options::OPT_MM)) ||
77 (A = Args.getLastArg(options::OPT_MD)) ||
78 (A = Args.getLastArg(options::OPT_MMD))) {
79 // Determine the output location.
80 const char *DepFile;
81 if (Output.getType() == types::TY_Dependencies) {
82 if (Output.isPipe())
83 DepFile = "-";
84 else
85 DepFile = Output.getFilename();
86 } else if (Arg *MF = Args.getLastArg(options::OPT_MF)) {
87 DepFile = MF->getValue(Args);
88 } else if (A->getOption().getId() == options::OPT_M ||
89 A->getOption().getId() == options::OPT_MM) {
90 DepFile = "-";
91 } else {
92 DepFile = darwin::CC1::getDependencyFileName(Args, Inputs);
93 }
94 CmdArgs.push_back("-dependency-file");
95 CmdArgs.push_back(DepFile);
96
97 // Add an -MT option if the user didn't specify their own.
Daniel Dunbare0be8b12009-09-08 16:39:16 +000098 //
Daniel Dunbarc21c4852009-04-08 23:54:23 +000099 // FIXME: This should use -MQ, when we support it.
100 if (!Args.hasArg(options::OPT_MT) && !Args.hasArg(options::OPT_MQ)) {
101 const char *DepTarget;
102
103 // If user provided -o, that is the dependency target, except
104 // when we are only generating a dependency file.
105 Arg *OutputOpt = Args.getLastArg(options::OPT_o);
106 if (OutputOpt && Output.getType() != types::TY_Dependencies) {
107 DepTarget = OutputOpt->getValue(Args);
108 } else {
109 // Otherwise derive from the base input.
110 //
111 // FIXME: This should use the computed output file location.
112 llvm::sys::Path P(Inputs[0].getBaseInput());
113
114 P.eraseSuffix();
115 P.appendSuffix("o");
Daniel Dunbar88137642009-09-09 22:32:48 +0000116 DepTarget = Args.MakeArgString(P.getLast());
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000117 }
118
119 CmdArgs.push_back("-MT");
120 CmdArgs.push_back(DepTarget);
121 }
122
123 if (A->getOption().getId() == options::OPT_M ||
124 A->getOption().getId() == options::OPT_MD)
125 CmdArgs.push_back("-sys-header-deps");
126 }
127
128 Args.AddLastArg(CmdArgs, options::OPT_MP);
129 Args.AddAllArgs(CmdArgs, options::OPT_MT);
130
Douglas Gregordf91ef32009-04-18 00:34:01 +0000131 // Add -i* options, and automatically translate to
132 // -include-pch/-include-pth for transparent PCH support. It's
133 // wonky, but we include looking for .gch so we can support seamless
134 // replacement into a build system already set up to be generating
135 // .gch files.
Daniel Dunbare0be8b12009-09-08 16:39:16 +0000136 //
137 // FIXME: Use iterator.
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000138 for (ArgList::const_iterator
139 it = Args.begin(), ie = Args.end(); it != ie; ++it) {
140 const Arg *A = *it;
141 if (!A->getOption().matches(options::OPT_clang_i_Group))
142 continue;
143
144 if (A->getOption().matches(options::OPT_include)) {
145 bool FoundPTH = false;
Douglas Gregordf91ef32009-04-18 00:34:01 +0000146 bool FoundPCH = false;
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000147 llvm::sys::Path P(A->getValue(Args));
Douglas Gregordf91ef32009-04-18 00:34:01 +0000148 if (D.CCCUsePCH) {
149 P.appendSuffix("pch");
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000150 if (P.exists())
Douglas Gregordf91ef32009-04-18 00:34:01 +0000151 FoundPCH = true;
Mike Stump1eb44332009-09-09 15:08:12 +0000152 else
Douglas Gregordf91ef32009-04-18 00:34:01 +0000153 P.eraseSuffix();
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000154 }
155
Douglas Gregordf91ef32009-04-18 00:34:01 +0000156 if (!FoundPCH) {
157 P.appendSuffix("pth");
Mike Stump1eb44332009-09-09 15:08:12 +0000158 if (P.exists())
Douglas Gregordf91ef32009-04-18 00:34:01 +0000159 FoundPTH = true;
160 else
161 P.eraseSuffix();
Mike Stump1eb44332009-09-09 15:08:12 +0000162 }
163
Douglas Gregordf91ef32009-04-18 00:34:01 +0000164 if (!FoundPCH && !FoundPTH) {
165 P.appendSuffix("gch");
166 if (P.exists()) {
167 FoundPCH = D.CCCUsePCH;
168 FoundPTH = !D.CCCUsePCH;
169 }
Mike Stump1eb44332009-09-09 15:08:12 +0000170 else
Douglas Gregordf91ef32009-04-18 00:34:01 +0000171 P.eraseSuffix();
172 }
173
174 if (FoundPCH || FoundPTH) {
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000175 A->claim();
Daniel Dunbarea71a5b2009-04-28 19:38:45 +0000176 if (D.CCCUsePCH)
Douglas Gregordf91ef32009-04-18 00:34:01 +0000177 CmdArgs.push_back("-include-pch");
178 else
179 CmdArgs.push_back("-include-pth");
Daniel Dunbar88137642009-09-09 22:32:48 +0000180 CmdArgs.push_back(Args.MakeArgString(P.str()));
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000181 continue;
182 }
183 }
184
185 // Not translated, render as usual.
186 A->claim();
187 A->render(Args, CmdArgs);
188 }
189
190 Args.AddAllArgs(CmdArgs, options::OPT_D, options::OPT_U);
191 Args.AddAllArgs(CmdArgs, options::OPT_I_Group, options::OPT_F);
192
193 // Add -Wp, and -Xassembler if using the preprocessor.
194
195 // FIXME: There is a very unfortunate problem here, some troubled
196 // souls abuse -Wp, to pass preprocessor options in gcc syntax. To
197 // really support that we would have to parse and then translate
198 // those options. :(
199 Args.AddAllArgValues(CmdArgs, options::OPT_Wp_COMMA,
200 options::OPT_Xpreprocessor);
201}
202
Daniel Dunbar6acda162009-09-09 22:33:08 +0000203void Clang::AddX86TargetArgs(const ArgList &Args,
204 ArgStringList &CmdArgs) const {
205 if (const Arg *A = Args.getLastArg(options::OPT_march_EQ)) {
206 // FIXME: We may need some translation here from the options gcc takes to
207 // names the LLVM backend understand?
208 CmdArgs.push_back("-mcpu");
209 CmdArgs.push_back(A->getValue(Args));
210 } else {
211 // Select default CPU.
212
213 // FIXME: Need target hooks.
214 if (memcmp(getToolChain().getOS().c_str(), "darwin", 6) == 0) {
215 if (getToolChain().getArchName() == "x86_64")
216 CmdArgs.push_back("--mcpu=core2");
217 else if (getToolChain().getArchName() == "i386")
218 CmdArgs.push_back("--mcpu=yonah");
219 } else {
220 if (getToolChain().getArchName() == "x86_64")
221 CmdArgs.push_back("--mcpu=x86-64");
222 else if (getToolChain().getArchName() == "i386")
223 CmdArgs.push_back("--mcpu=pentium4");
224 }
225 }
226
227 // FIXME: Use iterator.
228 for (ArgList::const_iterator
229 it = Args.begin(), ie = Args.end(); it != ie; ++it) {
230 const Arg *A = *it;
231 if (A->getOption().matches(options::OPT_m_x86_Features_Group)) {
232 llvm::StringRef Name = A->getOption().getName();
233
234 // Skip over "-m".
235 assert(Name.startswith("-m") && "Invalid feature name.");
236 Name = Name.substr(2);
237
238 bool IsNegative = Name.startswith("no-");
239 if (IsNegative)
240 Name = Name.substr(3);
241
242 A->claim();
243 CmdArgs.push_back("-target-feature");
244 CmdArgs.push_back(Args.MakeArgString((IsNegative ? "-" : "+") + Name));
245 }
246 }
247}
248
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000249void Clang::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbar871adcf2009-03-18 07:06:02 +0000250 Job &Dest,
251 const InputInfo &Output,
Daniel Dunbar62cf6012009-03-18 06:07:59 +0000252 const InputInfoList &Inputs,
Daniel Dunbar1d460332009-03-18 10:01:51 +0000253 const ArgList &Args,
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000254 const char *LinkingOutput) const {
Daniel Dunbar5c1aaaf2009-04-07 19:18:24 +0000255 const Driver &D = getToolChain().getHost().getDriver();
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000256 ArgStringList CmdArgs;
257
Daniel Dunbar077ba6a2009-03-31 20:53:55 +0000258 assert(Inputs.size() == 1 && "Unable to handle multiple inputs.");
259
Daniel Dunbaraf07f932009-03-31 17:35:15 +0000260 CmdArgs.push_back("-triple");
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000261 const char *TripleStr =
Daniel Dunbar88137642009-09-09 22:32:48 +0000262 Args.MakeArgString(getToolChain().getTripleString());
Daniel Dunbaraf07f932009-03-31 17:35:15 +0000263 CmdArgs.push_back(TripleStr);
264
Daniel Dunbar1d460332009-03-18 10:01:51 +0000265 if (isa<AnalyzeJobAction>(JA)) {
266 assert(JA.getType() == types::TY_Plist && "Invalid output type.");
267 CmdArgs.push_back("-analyze");
268 } else if (isa<PreprocessJobAction>(JA)) {
Daniel Dunbarcd8e4c42009-03-30 06:36:42 +0000269 if (Output.getType() == types::TY_Dependencies)
270 CmdArgs.push_back("-Eonly");
271 else
272 CmdArgs.push_back("-E");
Daniel Dunbar1d460332009-03-18 10:01:51 +0000273 } else if (isa<PrecompileJobAction>(JA)) {
Douglas Gregordf91ef32009-04-18 00:34:01 +0000274 if (D.CCCUsePCH)
275 CmdArgs.push_back("-emit-pch");
276 else
277 CmdArgs.push_back("-emit-pth");
Daniel Dunbar1d460332009-03-18 10:01:51 +0000278 } else {
279 assert(isa<CompileJobAction>(JA) && "Invalid action for clang tool.");
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000280
Daniel Dunbar1d460332009-03-18 10:01:51 +0000281 if (JA.getType() == types::TY_Nothing) {
282 CmdArgs.push_back("-fsyntax-only");
283 } else if (JA.getType() == types::TY_LLVMAsm) {
284 CmdArgs.push_back("-emit-llvm");
285 } else if (JA.getType() == types::TY_LLVMBC) {
286 CmdArgs.push_back("-emit-llvm-bc");
287 } else if (JA.getType() == types::TY_PP_Asm) {
Daniel Dunbar5915fbf2009-09-01 16:57:46 +0000288 if (Inputs[0].getType() == types::TY_AST)
289 CmdArgs.push_back("-compile-ast");
290 else
291 CmdArgs.push_back("-S");
292 } else if (JA.getType() == types::TY_AST) {
293 CmdArgs.push_back("-emit-pch");
Daniel Dunbar1d460332009-03-18 10:01:51 +0000294 }
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000295 }
296
Daniel Dunbar1d460332009-03-18 10:01:51 +0000297 // The make clang go fast button.
298 CmdArgs.push_back("-disable-free");
299
Daniel Dunbarc9abc042009-04-08 05:11:16 +0000300 // Set the main file name, so that debug info works even with
301 // -save-temps.
302 CmdArgs.push_back("-main-file-name");
303 CmdArgs.push_back(darwin::CC1::getBaseInputName(Args, Inputs));
304
Daniel Dunbar3bbc7532009-04-08 18:03:55 +0000305 // Some flags which affect the language (via preprocessor
306 // defines). See darwin::CC1::AddCPPArgs.
307 if (Args.hasArg(options::OPT_static))
308 CmdArgs.push_back("-static-define");
309
Daniel Dunbar1d460332009-03-18 10:01:51 +0000310 if (isa<AnalyzeJobAction>(JA)) {
311 // Add default argument set.
Daniel Dunbard8fc0f22009-05-22 00:38:15 +0000312 if (!Args.hasArg(options::OPT__analyzer_no_default_checks)) {
313 CmdArgs.push_back("-warn-dead-stores");
Ted Kremenek11727512009-07-24 20:03:11 +0000314 CmdArgs.push_back("-warn-security-syntactic");
Daniel Dunbard8fc0f22009-05-22 00:38:15 +0000315 CmdArgs.push_back("-checker-cfref");
316 CmdArgs.push_back("-analyzer-eagerly-assume");
317 CmdArgs.push_back("-warn-objc-methodsigs");
318 // Do not enable the missing -dealloc check.
319 // '-warn-objc-missing-dealloc',
320 CmdArgs.push_back("-warn-objc-unused-ivars");
321 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000322
Daniel Dunbard8fc0f22009-05-22 00:38:15 +0000323 // Set the output format. The default is plist, for (lame) historical
324 // reasons.
325 CmdArgs.push_back("-analyzer-output");
326 if (Arg *A = Args.getLastArg(options::OPT__analyzer_output))
327 CmdArgs.push_back(A->getValue(Args));
328 else
329 CmdArgs.push_back("plist");
Daniel Dunbar1d460332009-03-18 10:01:51 +0000330
331 // Add -Xanalyzer arguments when running as analyzer.
332 Args.AddAllArgValues(CmdArgs, options::OPT_Xanalyzer);
Mike Stump1eb44332009-09-09 15:08:12 +0000333 }
334
Daniel Dunbare2fd6642009-09-10 01:21:12 +0000335 CheckCodeGenerationOptions(D, Args);
336
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000337 // Perform argument translation for LLVM backend. This
338 // takes some care in reconciling with llvm-gcc. The
339 // issue is that llvm-gcc translates these options based on
340 // the values in cc1, whereas we are processing based on
341 // the driver arguments.
342 //
343 // FIXME: This is currently broken for -f flags when -fno
344 // variants are present.
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000345
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000346 // This comes from the default translation the driver + cc1
347 // would do to enable flag_pic.
348 //
349 // FIXME: Centralize this code.
350 bool PICEnabled = (Args.hasArg(options::OPT_fPIC) ||
351 Args.hasArg(options::OPT_fpic) ||
352 Args.hasArg(options::OPT_fPIE) ||
353 Args.hasArg(options::OPT_fpie));
354 bool PICDisabled = (Args.hasArg(options::OPT_mkernel) ||
355 Args.hasArg(options::OPT_static));
356 const char *Model = getToolChain().GetForcedPicModel();
357 if (!Model) {
358 if (Args.hasArg(options::OPT_mdynamic_no_pic))
359 Model = "dynamic-no-pic";
360 else if (PICDisabled)
361 Model = "static";
362 else if (PICEnabled)
363 Model = "pic";
Daniel Dunbar1d460332009-03-18 10:01:51 +0000364 else
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000365 Model = getToolChain().GetDefaultRelocationModel();
Daniel Dunbar1d460332009-03-18 10:01:51 +0000366 }
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000367 CmdArgs.push_back("--relocation-model");
368 CmdArgs.push_back(Model);
369
370 // Infer the __PIC__ value.
371 //
372 // FIXME: This isn't quite right on Darwin, which always sets
373 // __PIC__=2.
374 if (strcmp(Model, "pic") == 0 || strcmp(Model, "dynamic-no-pic") == 0) {
375 if (Args.hasArg(options::OPT_fPIC))
376 CmdArgs.push_back("-pic-level=2");
377 else
378 CmdArgs.push_back("-pic-level=1");
379 }
380
381 if (Args.hasArg(options::OPT_ftime_report))
382 CmdArgs.push_back("--time-passes");
383 // FIXME: Set --enable-unsafe-fp-math.
Benjamin Kramer21656dd2009-08-05 19:47:38 +0000384 if (Args.hasFlag(options::OPT_fno_omit_frame_pointer,
385 options::OPT_fomit_frame_pointer))
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000386 CmdArgs.push_back("--disable-fp-elim");
387 if (!Args.hasFlag(options::OPT_fzero_initialized_in_bss,
388 options::OPT_fno_zero_initialized_in_bss,
389 true))
390 CmdArgs.push_back("--nozero-initialized-in-bss");
391 if (Args.hasArg(options::OPT_dA) || Args.hasArg(options::OPT_fverbose_asm))
392 CmdArgs.push_back("--asm-verbose");
393 if (Args.hasArg(options::OPT_fdebug_pass_structure))
394 CmdArgs.push_back("--debug-pass=Structure");
395 if (Args.hasArg(options::OPT_fdebug_pass_arguments))
396 CmdArgs.push_back("--debug-pass=Arguments");
397 // FIXME: set --inline-threshhold=50 if (optimize_size || optimize
398 // < 3)
399 if (Args.hasFlag(options::OPT_funwind_tables,
400 options::OPT_fno_unwind_tables,
401 (getToolChain().IsUnwindTablesDefault() &&
402 !Args.hasArg(options::OPT_mkernel))))
403 CmdArgs.push_back("--unwind-tables=1");
404 else
405 CmdArgs.push_back("--unwind-tables=0");
406 if (!Args.hasFlag(options::OPT_mred_zone,
407 options::OPT_mno_red_zone,
408 true) ||
409 Args.hasArg(options::OPT_mkernel) ||
410 Args.hasArg(options::OPT_fapple_kext))
411 CmdArgs.push_back("--disable-red-zone");
412 if (Args.hasFlag(options::OPT_msoft_float,
413 options::OPT_mno_soft_float,
414 false))
Devang Patelacebb392009-06-05 22:05:48 +0000415 CmdArgs.push_back("--no-implicit-float");
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000416
Daniel Dunbar868bd0a2009-05-06 03:16:41 +0000417 // FIXME: Handle -mtune=.
418 (void) Args.hasArg(options::OPT_mtune_EQ);
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000419
Benjamin Kramer8e9ef0d2009-08-05 14:30:52 +0000420 if (Arg *A = Args.getLastArg(options::OPT_mcmodel_EQ)) {
421 CmdArgs.push_back("-code-model");
422 CmdArgs.push_back(A->getValue(Args));
423 }
424
Daniel Dunbar6acda162009-09-09 22:33:08 +0000425 // Add target specific cpu and features flags.
426 switch(getToolChain().getTriple().getArch()) {
427 default:
428 break;
Daniel Dunbar868bd0a2009-05-06 03:16:41 +0000429
Daniel Dunbar6acda162009-09-09 22:33:08 +0000430 case llvm::Triple::x86:
431 case llvm::Triple::x86_64:
432 AddX86TargetArgs(Args, CmdArgs);
433 break;
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000434 }
435
436 if (Args.hasFlag(options::OPT_fmath_errno,
437 options::OPT_fno_math_errno,
438 getToolChain().IsMathErrnoDefault()))
439 CmdArgs.push_back("--fmath-errno=1");
440 else
441 CmdArgs.push_back("--fmath-errno=0");
442
443 if (Arg *A = Args.getLastArg(options::OPT_flimited_precision_EQ)) {
444 CmdArgs.push_back("--limit-float-precision");
445 CmdArgs.push_back(A->getValue(Args));
446 }
447
448 // FIXME: Add --stack-protector-buffer-size=<xxx> on
449 // -fstack-protect.
450
451 Arg *Unsupported;
452 if ((Unsupported = Args.getLastArg(options::OPT_MG)) ||
Daniel Dunbar95a0da72009-05-13 19:05:04 +0000453 (Unsupported = Args.getLastArg(options::OPT_MQ)) ||
454 (Unsupported = Args.getLastArg(options::OPT_iframework)))
Daniel Dunbare027a4b2009-05-22 19:02:20 +0000455 D.Diag(clang::diag::err_drv_clang_unsupported)
Daniel Dunbarbc85be82009-04-29 18:32:25 +0000456 << Unsupported->getOption().getName();
Daniel Dunbar1d460332009-03-18 10:01:51 +0000457
458 Args.AddAllArgs(CmdArgs, options::OPT_v);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000459 Args.AddLastArg(CmdArgs, options::OPT_P);
Daniel Dunbar2ac9fc22009-04-07 21:42:00 +0000460 Args.AddLastArg(CmdArgs, options::OPT_mmacosx_version_min_EQ);
Daniel Dunbarff8857a2009-04-10 20:11:50 +0000461 Args.AddLastArg(CmdArgs, options::OPT_miphoneos_version_min_EQ);
Mike Stump1eb44332009-09-09 15:08:12 +0000462 Args.AddLastArg(CmdArgs, options::OPT_print_ivar_layout);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000463
464 // Special case debug options to only pass -g to clang. This is
465 // wrong.
466 if (Args.hasArg(options::OPT_g_Group))
467 CmdArgs.push_back("-g");
468
469 Args.AddLastArg(CmdArgs, options::OPT_nostdinc);
470
Daniel Dunbar2ac9fc22009-04-07 21:42:00 +0000471 Args.AddLastArg(CmdArgs, options::OPT_isysroot);
472
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000473 // Add preprocessing options like -I, -D, etc. if we are using the
474 // preprocessor.
475 //
476 // FIXME: Support -fpreprocessed
477 types::ID InputType = Inputs[0].getType();
478 if (types::getPreprocessedType(InputType) != types::TY_INVALID)
Douglas Gregordf91ef32009-04-18 00:34:01 +0000479 AddPreprocessingOptions(D, Args, CmdArgs, Output, Inputs);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000480
Daniel Dunbar337a6272009-03-24 20:17:30 +0000481 // Manually translate -O to -O1 and -O4 to -O3; let clang reject
482 // others.
483 if (Arg *A = Args.getLastArg(options::OPT_O_Group)) {
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000484 if (A->getOption().getId() == options::OPT_O4)
Daniel Dunbar337a6272009-03-24 20:17:30 +0000485 CmdArgs.push_back("-O3");
486 else if (A->getValue(Args)[0] == '\0')
Daniel Dunbar1d460332009-03-18 10:01:51 +0000487 CmdArgs.push_back("-O1");
488 else
Daniel Dunbar5697aa02009-03-18 23:39:35 +0000489 A->render(Args, CmdArgs);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000490 }
491
Daniel Dunbar06ef3c32009-04-16 03:44:10 +0000492 Args.AddAllArgs(CmdArgs, options::OPT_W_Group, options::OPT_pedantic_Group);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000493 Args.AddLastArg(CmdArgs, options::OPT_w);
Daniel Dunbard573d262009-04-07 22:13:21 +0000494
495 // Handle -{std, ansi, trigraphs} -- take the last of -{std, ansi}
496 // (-ansi is equivalent to -std=c89).
497 //
498 // If a std is supplied, only add -trigraphs if it follows the
499 // option.
500 if (Arg *Std = Args.getLastArg(options::OPT_std_EQ, options::OPT_ansi)) {
501 if (Std->getOption().matches(options::OPT_ansi))
502 CmdArgs.push_back("-std=c89");
503 else
504 Std->render(Args, CmdArgs);
505
506 if (Arg *A = Args.getLastArg(options::OPT_trigraphs))
507 if (A->getIndex() > Std->getIndex())
508 A->render(Args, CmdArgs);
Daniel Dunbara3ff2022009-04-26 01:10:38 +0000509 } else {
510 // Honor -std-default.
511 Args.AddAllArgsTranslated(CmdArgs, options::OPT_std_default_EQ,
512 "-std=", /*Joined=*/true);
Daniel Dunbard573d262009-04-07 22:13:21 +0000513 Args.AddLastArg(CmdArgs, options::OPT_trigraphs);
Daniel Dunbara3ff2022009-04-26 01:10:38 +0000514 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000515
Daniel Dunbar1d460332009-03-18 10:01:51 +0000516 if (Arg *A = Args.getLastArg(options::OPT_ftemplate_depth_)) {
517 CmdArgs.push_back("-ftemplate-depth");
518 CmdArgs.push_back(A->getValue(Args));
519 }
520
Douglas Gregore650c8c2009-07-07 00:12:59 +0000521 if (Args.hasArg(options::OPT__relocatable_pch, true))
522 CmdArgs.push_back("--relocatable-pch");
Mike Stump1eb44332009-09-09 15:08:12 +0000523
David Chisnall8a5a9aa2009-08-31 16:41:57 +0000524 if (Arg *A = Args.getLastArg(options::OPT_fconstant_string_class_EQ)) {
525 CmdArgs.push_back("-fconstant-string-class");
526 CmdArgs.push_back(A->getValue(Args));
527 }
528
Daniel Dunbar48d1ef72009-04-07 21:16:11 +0000529 // Forward -f options which we can pass directly.
Daniel Dunbar3aaf0822009-04-07 21:51:40 +0000530 Args.AddLastArg(CmdArgs, options::OPT_femit_all_decls);
Daniel Dunbar3aaf0822009-04-07 21:51:40 +0000531 Args.AddLastArg(CmdArgs, options::OPT_ffreestanding);
532 Args.AddLastArg(CmdArgs, options::OPT_fheinous_gnu_extensions);
533 Args.AddLastArg(CmdArgs, options::OPT_fgnu_runtime);
534 Args.AddLastArg(CmdArgs, options::OPT_flax_vector_conversions);
Douglas Gregorfffd93f2009-05-01 21:53:04 +0000535 Args.AddLastArg(CmdArgs, options::OPT_fmessage_length_EQ);
Daniel Dunbar3aaf0822009-04-07 21:51:40 +0000536 Args.AddLastArg(CmdArgs, options::OPT_fms_extensions);
537 Args.AddLastArg(CmdArgs, options::OPT_fnext_runtime);
538 Args.AddLastArg(CmdArgs, options::OPT_fno_caret_diagnostics);
539 Args.AddLastArg(CmdArgs, options::OPT_fno_show_column);
540 Args.AddLastArg(CmdArgs, options::OPT_fobjc_gc_only);
541 Args.AddLastArg(CmdArgs, options::OPT_fobjc_gc);
Fariborz Jahanian34e65772009-05-22 20:17:16 +0000542 Args.AddLastArg(CmdArgs, options::OPT_fobjc_sender_dependent_dispatch);
Daniel Dunbar3aaf0822009-04-07 21:51:40 +0000543 // FIXME: Should we remove this?
544 Args.AddLastArg(CmdArgs, options::OPT_fobjc_nonfragile_abi);
Daniel Dunbard6884a02009-05-04 05:16:21 +0000545 Args.AddLastArg(CmdArgs, options::OPT_fobjc_tight_layout);
Chris Lattner182e0922009-04-21 05:34:31 +0000546 Args.AddLastArg(CmdArgs, options::OPT_fdiagnostics_print_source_range_info);
Daniel Dunbar3aaf0822009-04-07 21:51:40 +0000547 Args.AddLastArg(CmdArgs, options::OPT_ftime_report);
548 Args.AddLastArg(CmdArgs, options::OPT_ftrapv);
549 Args.AddLastArg(CmdArgs, options::OPT_fvisibility_EQ);
550 Args.AddLastArg(CmdArgs, options::OPT_fwritable_strings);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000551
Daniel Dunbar5345c392009-09-03 04:54:28 +0000552 Args.AddLastArg(CmdArgs, options::OPT_pthread);
553
Bill Wendling45483f72009-06-28 07:36:13 +0000554 // Forward stack protector flags.
555 if (Arg *A = Args.getLastArg(options::OPT_fno_stack_protector,
556 options::OPT_fstack_protector_all,
557 options::OPT_fstack_protector)) {
558 if (A->getOption().matches(options::OPT_fno_stack_protector))
559 CmdArgs.push_back("--stack-protector=0");
560 else if (A->getOption().matches(options::OPT_fstack_protector))
561 CmdArgs.push_back("--stack-protector=1");
562 else
563 CmdArgs.push_back("--stack-protector=2");
564 }
565
Daniel Dunbar48d1ef72009-04-07 21:16:11 +0000566 // Forward -f options with positive and negative forms; we translate
567 // these by hand.
568
569 // -fbuiltin is default, only pass non-default.
570 if (!Args.hasFlag(options::OPT_fbuiltin, options::OPT_fno_builtin))
571 CmdArgs.push_back("-fbuiltin=0");
572
Daniel Dunbar7695fba2009-04-19 21:20:32 +0000573 // -fblocks default varies depending on platform and language; only
574 // pass if specified.
Daniel Dunbar48d1ef72009-04-07 21:16:11 +0000575 if (Arg *A = Args.getLastArg(options::OPT_fblocks, options::OPT_fno_blocks)) {
576 if (A->getOption().matches(options::OPT_fblocks))
577 CmdArgs.push_back("-fblocks");
578 else
579 CmdArgs.push_back("-fblocks=0");
580 }
581
Mike Stump738f8c22009-07-31 23:15:31 +0000582 // -fexceptions default varies depending on platform and language; only
583 // pass if specified.
584 if (Arg *A = Args.getLastArg(options::OPT_fexceptions,
585 options::OPT_fno_exceptions)) {
586 if (A->getOption().matches(options::OPT_fexceptions))
587 CmdArgs.push_back("-fexceptions");
588 else
589 CmdArgs.push_back("-fexceptions=0");
590 }
591
592 // -frtti is default, only pass non-default.
593 if (!Args.hasFlag(options::OPT_frtti, options::OPT_fno_rtti))
594 CmdArgs.push_back("-frtti=0");
595
Eli Friedman5a779732009-06-05 07:21:14 +0000596 // -fsigned-char/-funsigned-char default varies depending on platform; only
597 // pass if specified.
598 if (Arg *A = Args.getLastArg(options::OPT_fsigned_char,
599 options::OPT_funsigned_char)) {
600 if (A->getOption().matches(options::OPT_fsigned_char))
601 CmdArgs.push_back("-fsigned-char");
602 else
603 CmdArgs.push_back("-fsigned-char=0");
604 }
605
Daniel Dunbar82d00682009-04-07 23:51:44 +0000606 // -fno-pascal-strings is default, only pass non-default. If the
607 // -tool chain happened to translate to -mpascal-strings, we want to
608 // -back translate here.
609 //
610 // FIXME: This is gross; that translation should be pulled from the
611 // tool chain.
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000612 if (Args.hasFlag(options::OPT_fpascal_strings,
Daniel Dunbar82d00682009-04-07 23:51:44 +0000613 options::OPT_fno_pascal_strings,
614 false) ||
615 Args.hasFlag(options::OPT_mpascal_strings,
616 options::OPT_mno_pascal_strings,
617 false))
Daniel Dunbar48d1ef72009-04-07 21:16:11 +0000618 CmdArgs.push_back("-fpascal-strings");
619
620 // -fcommon is default, only pass non-default.
621 if (!Args.hasFlag(options::OPT_fcommon, options::OPT_fno_common))
622 CmdArgs.push_back("-fno-common");
623
Daniel Dunbar70d3c922009-04-15 02:37:43 +0000624 // -fsigned-bitfields is default, and clang doesn't yet support
625 // --funsigned-bitfields.
Mike Stump1eb44332009-09-09 15:08:12 +0000626 if (!Args.hasFlag(options::OPT_fsigned_bitfields,
Daniel Dunbar70d3c922009-04-15 02:37:43 +0000627 options::OPT_funsigned_bitfields))
628 D.Diag(clang::diag::warn_drv_clang_unsupported)
629 << Args.getLastArg(options::OPT_funsigned_bitfields)->getAsString(Args);
630
Daniel Dunbar49138fc2009-04-19 21:09:34 +0000631 // -fdiagnostics-fixit-info is default, only pass non-default.
Mike Stump1eb44332009-09-09 15:08:12 +0000632 if (!Args.hasFlag(options::OPT_fdiagnostics_fixit_info,
Daniel Dunbar49138fc2009-04-19 21:09:34 +0000633 options::OPT_fno_diagnostics_fixit_info))
634 CmdArgs.push_back("-fno-diagnostics-fixit-info");
635
Daniel Dunbar9e820ee2009-04-16 06:32:38 +0000636 // Enable -fdiagnostics-show-option by default.
Mike Stump1eb44332009-09-09 15:08:12 +0000637 if (Args.hasFlag(options::OPT_fdiagnostics_show_option,
Daniel Dunbar9e820ee2009-04-16 06:32:38 +0000638 options::OPT_fno_diagnostics_show_option))
639 CmdArgs.push_back("-fdiagnostics-show-option");
Torok Edwina46c71a2009-06-04 07:27:53 +0000640 if (!Args.hasFlag(options::OPT_fcolor_diagnostics,
Daniel Dunbar75eb1d62009-06-08 21:13:54 +0000641 options::OPT_fno_color_diagnostics))
Torok Edwina46c71a2009-06-04 07:27:53 +0000642 CmdArgs.push_back("-fno-color-diagnostics");
Daniel Dunbar75eb1d62009-06-08 21:13:54 +0000643 if (!Args.hasFlag(options::OPT_fshow_source_location,
644 options::OPT_fno_show_source_location))
645 CmdArgs.push_back("-fno-show-source-location");
Daniel Dunbar9e820ee2009-04-16 06:32:38 +0000646
Daniel Dunbar7695fba2009-04-19 21:20:32 +0000647 // -fdollars-in-identifiers default varies depending on platform and
648 // language; only pass if specified.
Mike Stump1eb44332009-09-09 15:08:12 +0000649 if (Arg *A = Args.getLastArg(options::OPT_fdollars_in_identifiers,
Daniel Dunbar7695fba2009-04-19 21:20:32 +0000650 options::OPT_fno_dollars_in_identifiers)) {
651 if (A->getOption().matches(options::OPT_fdollars_in_identifiers))
652 CmdArgs.push_back("-fdollars-in-identifiers=1");
653 else
654 CmdArgs.push_back("-fdollars-in-identifiers=0");
655 }
656
Daniel Dunbare027a4b2009-05-22 19:02:20 +0000657 // -funit-at-a-time is default, and we don't support -fno-unit-at-a-time for
658 // practical purposes.
Mike Stump1eb44332009-09-09 15:08:12 +0000659 if (Arg *A = Args.getLastArg(options::OPT_funit_at_a_time,
Daniel Dunbare027a4b2009-05-22 19:02:20 +0000660 options::OPT_fno_unit_at_a_time)) {
661 if (A->getOption().matches(options::OPT_fno_unit_at_a_time))
662 D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args);
663 }
Eli Friedmanceb5c5b2009-07-14 21:58:17 +0000664
Mike Stump1eb44332009-09-09 15:08:12 +0000665 if (Arg *A = Args.getLastArg(options::OPT_traditional,
Eli Friedmanceb5c5b2009-07-14 21:58:17 +0000666 options::OPT_traditional_cpp))
667 D.Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args);
668
Daniel Dunbar1d460332009-03-18 10:01:51 +0000669 Args.AddLastArg(CmdArgs, options::OPT_dM);
Chris Lattnerd82df3a2009-04-12 01:56:53 +0000670 Args.AddLastArg(CmdArgs, options::OPT_dD);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000671
672 Args.AddAllArgValues(CmdArgs, options::OPT_Xclang);
Daniel Dunbare5280282009-06-03 16:16:27 +0000673 Args.AddAllArgValues(CmdArgs, options::OPT_mllvm);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000674
Daniel Dunbarcd8e4c42009-03-30 06:36:42 +0000675 if (Output.getType() == types::TY_Dependencies) {
676 // Handled with other dependency code.
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000677 } else if (Output.isPipe()) {
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000678 CmdArgs.push_back("-o");
679 CmdArgs.push_back("-");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000680 } else if (Output.isFilename()) {
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000681 CmdArgs.push_back("-o");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000682 CmdArgs.push_back(Output.getFilename());
683 } else {
684 assert(Output.isNothing() && "Invalid output.");
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000685 }
686
Daniel Dunbar1d460332009-03-18 10:01:51 +0000687 for (InputInfoList::const_iterator
688 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
689 const InputInfo &II = *it;
690 CmdArgs.push_back("-x");
691 CmdArgs.push_back(types::getTypeName(II.getType()));
692 if (II.isPipe())
693 CmdArgs.push_back("-");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000694 else if (II.isFilename())
695 CmdArgs.push_back(II.getFilename());
Daniel Dunbar1d460332009-03-18 10:01:51 +0000696 else
Daniel Dunbar115a7922009-03-19 07:29:38 +0000697 II.getInputArg().renderAsInput(Args, CmdArgs);
Daniel Dunbar1d460332009-03-18 10:01:51 +0000698 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000699
700 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +0000701 Args.MakeArgString(getToolChain().GetProgramPath(C, "clang-cc"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +0000702 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbara880db02009-03-23 19:03:36 +0000703
Daniel Dunbar5c1aaaf2009-04-07 19:18:24 +0000704 // Explicitly warn that these options are unsupported, even though
705 // we are allowing compilation to continue.
706 // FIXME: Use iterator.
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000707 for (ArgList::const_iterator
Daniel Dunbar5c1aaaf2009-04-07 19:18:24 +0000708 it = Args.begin(), ie = Args.end(); it != ie; ++it) {
709 const Arg *A = *it;
710 if (A->getOption().matches(options::OPT_pg)) {
711 A->claim();
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000712 D.Diag(clang::diag::warn_drv_clang_unsupported)
Daniel Dunbar5c1aaaf2009-04-07 19:18:24 +0000713 << A->getAsString(Args);
714 }
715 }
716
Daniel Dunbar68fb4692009-04-03 20:51:31 +0000717 // Claim some arguments which clang supports automatically.
718
719 // -fpch-preprocess is used with gcc to add a special marker in the
720 // -output to include the PCH file. Clang's PTH solution is
721 // -completely transparent, so we do not need to deal with it at
722 // -all.
723 Args.ClaimAllArgs(options::OPT_fpch_preprocess);
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000724
Daniel Dunbara880db02009-03-23 19:03:36 +0000725 // Claim some arguments which clang doesn't support, but we don't
726 // care to warn the user about.
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000727
Daniel Dunbara880db02009-03-23 19:03:36 +0000728 // FIXME: Use iterator.
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000729 for (ArgList::const_iterator
Daniel Dunbara880db02009-03-23 19:03:36 +0000730 it = Args.begin(), ie = Args.end(); it != ie; ++it) {
731 const Arg *A = *it;
Daniel Dunbar06ef3c32009-04-16 03:44:10 +0000732 if (A->getOption().matches(options::OPT_clang_ignored_f_Group) ||
Daniel Dunbar16fd3a92009-04-07 02:59:27 +0000733 A->getOption().matches(options::OPT_clang_ignored_m_Group))
Daniel Dunbara880db02009-03-23 19:03:36 +0000734 A->claim();
735 }
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000736}
737
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000738void gcc::Common::ConstructJob(Compilation &C, const JobAction &JA,
739 Job &Dest,
740 const InputInfo &Output,
741 const InputInfoList &Inputs,
Daniel Dunbar1d460332009-03-18 10:01:51 +0000742 const ArgList &Args,
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000743 const char *LinkingOutput) const {
Daniel Dunbara8304f62009-05-02 20:14:53 +0000744 const Driver &D = getToolChain().getHost().getDriver();
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000745 ArgStringList CmdArgs;
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000746
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000747 for (ArgList::const_iterator
Daniel Dunbar1d460332009-03-18 10:01:51 +0000748 it = Args.begin(), ie = Args.end(); it != ie; ++it) {
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000749 Arg *A = *it;
Daniel Dunbar75877192009-03-19 07:55:12 +0000750 if (A->getOption().hasForwardToGCC()) {
751 // It is unfortunate that we have to claim here, as this means
752 // we will basically never report anything interesting for
Daniel Dunbar6ecc7a92009-05-02 21:41:52 +0000753 // platforms using a generic gcc, even if we are just using gcc
754 // to get to the assembler.
Daniel Dunbar75877192009-03-19 07:55:12 +0000755 A->claim();
Daniel Dunbar1d460332009-03-18 10:01:51 +0000756 A->render(Args, CmdArgs);
Daniel Dunbar75877192009-03-19 07:55:12 +0000757 }
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000758 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000759
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000760 RenderExtraToolArgs(CmdArgs);
761
762 // If using a driver driver, force the arch.
Daniel Dunbar7cfe31a2009-05-22 02:21:04 +0000763 const std::string &Arch = getToolChain().getArchName();
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000764 if (getToolChain().getHost().useDriverDriver()) {
765 CmdArgs.push_back("-arch");
Daniel Dunbarbf54a062009-04-01 20:33:11 +0000766
767 // FIXME: Remove these special cases.
Daniel Dunbar7cfe31a2009-05-22 02:21:04 +0000768 if (Arch == "powerpc")
769 CmdArgs.push_back("ppc");
770 else if (Arch == "powerpc64")
771 CmdArgs.push_back("ppc64");
772 else
Daniel Dunbar88137642009-09-09 22:32:48 +0000773 CmdArgs.push_back(Args.MakeArgString(Arch));
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000774 }
775
Daniel Dunbar6ecc7a92009-05-02 21:41:52 +0000776 // Try to force gcc to match the tool chain we want, if we recognize
777 // the arch.
Daniel Dunbar7cfe31a2009-05-22 02:21:04 +0000778 //
779 // FIXME: The triple class should directly provide the information we want
780 // here.
781 if (Arch == "i386" || Arch == "powerpc")
Daniel Dunbar6ecc7a92009-05-02 21:41:52 +0000782 CmdArgs.push_back("-m32");
Daniel Dunbar7cfe31a2009-05-22 02:21:04 +0000783 else if (Arch == "x86_64" || Arch == "powerpc64")
Daniel Dunbar6ecc7a92009-05-02 21:41:52 +0000784 CmdArgs.push_back("-m64");
785
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000786 if (Output.isPipe()) {
787 CmdArgs.push_back("-o");
788 CmdArgs.push_back("-");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000789 } else if (Output.isFilename()) {
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000790 CmdArgs.push_back("-o");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000791 CmdArgs.push_back(Output.getFilename());
792 } else {
793 assert(Output.isNothing() && "Unexpected output");
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000794 CmdArgs.push_back("-fsyntax-only");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000795 }
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000796
797
798 // Only pass -x if gcc will understand it; otherwise hope gcc
799 // understands the suffix correctly. The main use case this would go
800 // wrong in is for linker inputs if they happened to have an odd
801 // suffix; really the only way to get this to happen is a command
802 // like '-x foobar a.c' which will treat a.c like a linker input.
803 //
804 // FIXME: For the linker case specifically, can we safely convert
805 // inputs into '-Wl,' options?
806 for (InputInfoList::const_iterator
807 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
808 const InputInfo &II = *it;
Daniel Dunbara8304f62009-05-02 20:14:53 +0000809
Daniel Dunbar5915fbf2009-09-01 16:57:46 +0000810 // Don't try to pass LLVM or AST inputs to a generic gcc.
Daniel Dunbara8304f62009-05-02 20:14:53 +0000811 if (II.getType() == types::TY_LLVMBC)
812 D.Diag(clang::diag::err_drv_no_linker_llvm_support)
Daniel Dunbar88137642009-09-09 22:32:48 +0000813 << getToolChain().getTripleString();
Daniel Dunbar5915fbf2009-09-01 16:57:46 +0000814 else if (II.getType() == types::TY_AST)
815 D.Diag(clang::diag::err_drv_no_ast_support)
Daniel Dunbar88137642009-09-09 22:32:48 +0000816 << getToolChain().getTripleString();
Daniel Dunbara8304f62009-05-02 20:14:53 +0000817
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000818 if (types::canTypeBeUserSpecified(II.getType())) {
819 CmdArgs.push_back("-x");
820 CmdArgs.push_back(types::getTypeName(II.getType()));
821 }
822
823 if (II.isPipe())
824 CmdArgs.push_back("-");
Daniel Dunbar115a7922009-03-19 07:29:38 +0000825 else if (II.isFilename())
826 CmdArgs.push_back(II.getFilename());
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000827 else
Daniel Dunbar115a7922009-03-19 07:29:38 +0000828 // Don't render as input, we need gcc to do the translations.
829 II.getInputArg().render(Args, CmdArgs);
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000830 }
831
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000832 const char *GCCName =
Daniel Dunbar78d8a082009-04-01 23:34:41 +0000833 getToolChain().getHost().getDriver().CCCGenericGCCName.c_str();
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000834 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +0000835 Args.MakeArgString(getToolChain().GetProgramPath(C, GCCName));
Daniel Dunbarcae087e2009-07-01 19:02:28 +0000836 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000837}
838
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000839void gcc::Preprocess::RenderExtraToolArgs(ArgStringList &CmdArgs) const {
840 CmdArgs.push_back("-E");
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000841}
842
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000843void gcc::Precompile::RenderExtraToolArgs(ArgStringList &CmdArgs) const {
844 // The type is good enough.
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000845}
846
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000847void gcc::Compile::RenderExtraToolArgs(ArgStringList &CmdArgs) const {
848 CmdArgs.push_back("-S");
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000849}
850
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000851void gcc::Assemble::RenderExtraToolArgs(ArgStringList &CmdArgs) const {
852 CmdArgs.push_back("-c");
Daniel Dunbar47ac7d22009-03-18 06:00:36 +0000853}
Daniel Dunbarb488c1d2009-03-18 08:07:30 +0000854
855void gcc::Link::RenderExtraToolArgs(ArgStringList &CmdArgs) const {
856 // The types are (hopefully) good enough.
857}
858
Daniel Dunbar40f12652009-03-29 17:08:39 +0000859const char *darwin::CC1::getCC1Name(types::ID Type) const {
860 switch (Type) {
861 default:
862 assert(0 && "Unexpected type for Darwin CC1 tool.");
863 case types::TY_Asm:
864 case types::TY_C: case types::TY_CHeader:
865 case types::TY_PP_C: case types::TY_PP_CHeader:
866 return "cc1";
867 case types::TY_ObjC: case types::TY_ObjCHeader:
868 case types::TY_PP_ObjC: case types::TY_PP_ObjCHeader:
869 return "cc1obj";
870 case types::TY_CXX: case types::TY_CXXHeader:
871 case types::TY_PP_CXX: case types::TY_PP_CXXHeader:
872 return "cc1plus";
873 case types::TY_ObjCXX: case types::TY_ObjCXXHeader:
874 case types::TY_PP_ObjCXX: case types::TY_PP_ObjCXXHeader:
875 return "cc1objplus";
876 }
877}
878
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000879const char *darwin::CC1::getBaseInputName(const ArgList &Args,
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000880 const InputInfoList &Inputs) {
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000881 llvm::sys::Path P(Inputs[0].getBaseInput());
Daniel Dunbar88137642009-09-09 22:32:48 +0000882 return Args.MakeArgString(P.getLast());
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000883}
884
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000885const char *darwin::CC1::getBaseInputStem(const ArgList &Args,
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000886 const InputInfoList &Inputs) {
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000887 const char *Str = getBaseInputName(Args, Inputs);
888
889 if (const char *End = strchr(Str, '.'))
Daniel Dunbar88137642009-09-09 22:32:48 +0000890 return Args.MakeArgString(std::string(Str, End));
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000891
892 return Str;
893}
894
895const char *
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000896darwin::CC1::getDependencyFileName(const ArgList &Args,
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000897 const InputInfoList &Inputs) {
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000898 // FIXME: Think about this more.
899 std::string Res;
900
901 if (Arg *OutputOpt = Args.getLastArg(options::OPT_o)) {
902 std::string Str(OutputOpt->getValue(Args));
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000903
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000904 Res = Str.substr(0, Str.rfind('.'));
905 } else
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000906 Res = darwin::CC1::getBaseInputStem(Args, Inputs);
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000907
Daniel Dunbar88137642009-09-09 22:32:48 +0000908 return Args.MakeArgString(Res + ".d");
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000909}
910
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000911void darwin::CC1::AddCC1Args(const ArgList &Args,
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000912 ArgStringList &CmdArgs) const {
Daniel Dunbare2fd6642009-09-10 01:21:12 +0000913 const Driver &D = getToolChain().getHost().getDriver();
914
915 CheckCodeGenerationOptions(D, Args);
916
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000917 // Derived from cc1 spec.
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000918 if (!Args.hasArg(options::OPT_mkernel) && !Args.hasArg(options::OPT_static) &&
919 !Args.hasArg(options::OPT_mdynamic_no_pic))
920 CmdArgs.push_back("-fPIC");
921
922 // gcc has some code here to deal with when no -mmacosx-version-min
923 // and no -miphoneos-version-min is present, but this never happens
924 // due to tool chain specific argument translation.
925
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000926 if (Args.hasArg(options::OPT_g_Flag) &&
927 !Args.hasArg(options::OPT_fno_eliminate_unused_debug_symbols))
928 CmdArgs.push_back("-feliminate-unused-debug-symbols");
929}
930
931void darwin::CC1::AddCC1OptionsArgs(const ArgList &Args, ArgStringList &CmdArgs,
932 const InputInfoList &Inputs,
933 const ArgStringList &OutputArgs) const {
934 const Driver &D = getToolChain().getHost().getDriver();
935
936 // Derived from cc1_options spec.
937 if (Args.hasArg(options::OPT_fast) ||
938 Args.hasArg(options::OPT_fastf) ||
939 Args.hasArg(options::OPT_fastcp))
940 CmdArgs.push_back("-O3");
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000941
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000942 if (Arg *A = Args.getLastArg(options::OPT_pg))
943 if (Args.hasArg(options::OPT_fomit_frame_pointer))
944 D.Diag(clang::diag::err_drv_argument_not_allowed_with)
945 << A->getAsString(Args) << "-fomit-frame-pointer";
946
947 AddCC1Args(Args, CmdArgs);
948
949 if (!Args.hasArg(options::OPT_Q))
950 CmdArgs.push_back("-quiet");
951
952 CmdArgs.push_back("-dumpbase");
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000953 CmdArgs.push_back(darwin::CC1::getBaseInputName(Args, Inputs));
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000954
955 Args.AddAllArgs(CmdArgs, options::OPT_d_Group);
956
957 Args.AddAllArgs(CmdArgs, options::OPT_m_Group);
958 Args.AddAllArgs(CmdArgs, options::OPT_a_Group);
959
960 // FIXME: The goal is to use the user provided -o if that is our
961 // final output, otherwise to drive from the original input
962 // name. Find a clean way to go about this.
963 if ((Args.hasArg(options::OPT_c) || Args.hasArg(options::OPT_S)) &&
964 Args.hasArg(options::OPT_o)) {
965 Arg *OutputOpt = Args.getLastArg(options::OPT_o);
966 CmdArgs.push_back("-auxbase-strip");
967 CmdArgs.push_back(OutputOpt->getValue(Args));
968 } else {
969 CmdArgs.push_back("-auxbase");
Daniel Dunbara5a7bd02009-03-30 00:34:04 +0000970 CmdArgs.push_back(darwin::CC1::getBaseInputStem(Args, Inputs));
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000971 }
972
973 Args.AddAllArgs(CmdArgs, options::OPT_g_Group);
974
975 Args.AddAllArgs(CmdArgs, options::OPT_O);
976 // FIXME: -Wall is getting some special treatment. Investigate.
977 Args.AddAllArgs(CmdArgs, options::OPT_W_Group, options::OPT_pedantic_Group);
978 Args.AddLastArg(CmdArgs, options::OPT_w);
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000979 Args.AddAllArgs(CmdArgs, options::OPT_std_EQ, options::OPT_ansi,
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000980 options::OPT_trigraphs);
Daniel Dunbara3ff2022009-04-26 01:10:38 +0000981 if (!Args.getLastArg(options::OPT_std_EQ, options::OPT_ansi)) {
982 // Honor -std-default.
983 Args.AddAllArgsTranslated(CmdArgs, options::OPT_std_default_EQ,
984 "-std=", /*Joined=*/true);
985 }
986
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000987 if (Args.hasArg(options::OPT_v))
988 CmdArgs.push_back("-version");
989 if (Args.hasArg(options::OPT_pg))
990 CmdArgs.push_back("-p");
991 Args.AddLastArg(CmdArgs, options::OPT_p);
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000992
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000993 // The driver treats -fsyntax-only specially.
994 Args.AddAllArgs(CmdArgs, options::OPT_f_Group, options::OPT_fsyntax_only);
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000995
Daniel Dunbara3ec60e2009-03-29 18:40:18 +0000996 Args.AddAllArgs(CmdArgs, options::OPT_undef);
997 if (Args.hasArg(options::OPT_Qn))
998 CmdArgs.push_back("-fno-ident");
Daniel Dunbarc21c4852009-04-08 23:54:23 +0000999
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001000 // FIXME: This isn't correct.
1001 //Args.AddLastArg(CmdArgs, options::OPT__help)
1002 //Args.AddLastArg(CmdArgs, options::OPT__targetHelp)
1003
1004 CmdArgs.append(OutputArgs.begin(), OutputArgs.end());
1005
1006 // FIXME: Still don't get what is happening here. Investigate.
1007 Args.AddAllArgs(CmdArgs, options::OPT__param);
1008
1009 if (Args.hasArg(options::OPT_fmudflap) ||
1010 Args.hasArg(options::OPT_fmudflapth)) {
1011 CmdArgs.push_back("-fno-builtin");
1012 CmdArgs.push_back("-fno-merge-constants");
1013 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001014
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001015 if (Args.hasArg(options::OPT_coverage)) {
1016 CmdArgs.push_back("-fprofile-arcs");
1017 CmdArgs.push_back("-ftest-coverage");
1018 }
1019
1020 if (types::isCXX(Inputs[0].getType()))
1021 CmdArgs.push_back("-D__private_extern__=extern");
1022}
1023
1024void darwin::CC1::AddCPPOptionsArgs(const ArgList &Args, ArgStringList &CmdArgs,
1025 const InputInfoList &Inputs,
1026 const ArgStringList &OutputArgs) const {
1027 // Derived from cpp_options
1028 AddCPPUniqueOptionsArgs(Args, CmdArgs, Inputs);
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001029
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001030 CmdArgs.append(OutputArgs.begin(), OutputArgs.end());
1031
1032 AddCC1Args(Args, CmdArgs);
1033
1034 // NOTE: The code below has some commonality with cpp_options, but
1035 // in classic gcc style ends up sending things in different
1036 // orders. This may be a good merge candidate once we drop pedantic
1037 // compatibility.
1038
1039 Args.AddAllArgs(CmdArgs, options::OPT_m_Group);
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001040 Args.AddAllArgs(CmdArgs, options::OPT_std_EQ, options::OPT_ansi,
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001041 options::OPT_trigraphs);
Daniel Dunbara3ff2022009-04-26 01:10:38 +00001042 if (!Args.getLastArg(options::OPT_std_EQ, options::OPT_ansi)) {
1043 // Honor -std-default.
1044 Args.AddAllArgsTranslated(CmdArgs, options::OPT_std_default_EQ,
1045 "-std=", /*Joined=*/true);
1046 }
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001047 Args.AddAllArgs(CmdArgs, options::OPT_W_Group, options::OPT_pedantic_Group);
1048 Args.AddLastArg(CmdArgs, options::OPT_w);
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001049
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001050 // The driver treats -fsyntax-only specially.
1051 Args.AddAllArgs(CmdArgs, options::OPT_f_Group, options::OPT_fsyntax_only);
1052
1053 if (Args.hasArg(options::OPT_g_Group) && !Args.hasArg(options::OPT_g0) &&
1054 !Args.hasArg(options::OPT_fno_working_directory))
1055 CmdArgs.push_back("-fworking-directory");
1056
1057 Args.AddAllArgs(CmdArgs, options::OPT_O);
1058 Args.AddAllArgs(CmdArgs, options::OPT_undef);
1059 if (Args.hasArg(options::OPT_save_temps))
1060 CmdArgs.push_back("-fpch-preprocess");
1061}
1062
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001063void darwin::CC1::AddCPPUniqueOptionsArgs(const ArgList &Args,
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001064 ArgStringList &CmdArgs,
Mike Stump1eb44332009-09-09 15:08:12 +00001065 const InputInfoList &Inputs) const {
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001066 const Driver &D = getToolChain().getHost().getDriver();
1067
Daniel Dunbar88a3d6c2009-09-10 01:21:05 +00001068 CheckPreprocessingOptions(D, Args);
1069
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001070 // Derived from cpp_unique_options.
Daniel Dunbar88a3d6c2009-09-10 01:21:05 +00001071 // -{C,CC} only with -E is checked in CheckPreprocessingOptions().
1072 Args.AddLastArg(CmdArgs, options::OPT_C);
1073 Args.AddLastArg(CmdArgs, options::OPT_CC);
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001074 if (!Args.hasArg(options::OPT_Q))
1075 CmdArgs.push_back("-quiet");
1076 Args.AddAllArgs(CmdArgs, options::OPT_nostdinc);
1077 Args.AddLastArg(CmdArgs, options::OPT_v);
1078 Args.AddAllArgs(CmdArgs, options::OPT_I_Group, options::OPT_F);
1079 Args.AddLastArg(CmdArgs, options::OPT_P);
1080
1081 // FIXME: Handle %I properly.
1082 if (getToolChain().getArchName() == "x86_64") {
1083 CmdArgs.push_back("-imultilib");
1084 CmdArgs.push_back("x86_64");
1085 }
1086
1087 if (Args.hasArg(options::OPT_MD)) {
1088 CmdArgs.push_back("-MD");
Daniel Dunbara5a7bd02009-03-30 00:34:04 +00001089 CmdArgs.push_back(darwin::CC1::getDependencyFileName(Args, Inputs));
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001090 }
1091
1092 if (Args.hasArg(options::OPT_MMD)) {
1093 CmdArgs.push_back("-MMD");
Daniel Dunbara5a7bd02009-03-30 00:34:04 +00001094 CmdArgs.push_back(darwin::CC1::getDependencyFileName(Args, Inputs));
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001095 }
1096
1097 Args.AddLastArg(CmdArgs, options::OPT_M);
1098 Args.AddLastArg(CmdArgs, options::OPT_MM);
1099 Args.AddAllArgs(CmdArgs, options::OPT_MF);
1100 Args.AddLastArg(CmdArgs, options::OPT_MG);
1101 Args.AddLastArg(CmdArgs, options::OPT_MP);
1102 Args.AddAllArgs(CmdArgs, options::OPT_MQ);
1103 Args.AddAllArgs(CmdArgs, options::OPT_MT);
1104 if (!Args.hasArg(options::OPT_M) && !Args.hasArg(options::OPT_MM) &&
1105 (Args.hasArg(options::OPT_MD) || Args.hasArg(options::OPT_MMD))) {
1106 if (Arg *OutputOpt = Args.getLastArg(options::OPT_o)) {
1107 CmdArgs.push_back("-MQ");
1108 CmdArgs.push_back(OutputOpt->getValue(Args));
1109 }
1110 }
1111
1112 Args.AddLastArg(CmdArgs, options::OPT_remap);
1113 if (Args.hasArg(options::OPT_g3))
1114 CmdArgs.push_back("-dD");
1115 Args.AddLastArg(CmdArgs, options::OPT_H);
1116
1117 AddCPPArgs(Args, CmdArgs);
1118
1119 Args.AddAllArgs(CmdArgs, options::OPT_D, options::OPT_U, options::OPT_A);
1120 Args.AddAllArgs(CmdArgs, options::OPT_i_Group);
1121
1122 for (InputInfoList::const_iterator
1123 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
1124 const InputInfo &II = *it;
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001125
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001126 if (II.isPipe())
1127 CmdArgs.push_back("-");
1128 else
1129 CmdArgs.push_back(II.getFilename());
1130 }
1131
1132 Args.AddAllArgValues(CmdArgs, options::OPT_Wp_COMMA,
1133 options::OPT_Xpreprocessor);
1134
1135 if (Args.hasArg(options::OPT_fmudflap)) {
1136 CmdArgs.push_back("-D_MUDFLAP");
1137 CmdArgs.push_back("-include");
1138 CmdArgs.push_back("mf-runtime.h");
1139 }
1140
1141 if (Args.hasArg(options::OPT_fmudflapth)) {
1142 CmdArgs.push_back("-D_MUDFLAP");
1143 CmdArgs.push_back("-D_MUDFLAPTH");
1144 CmdArgs.push_back("-include");
1145 CmdArgs.push_back("mf-runtime.h");
1146 }
1147}
1148
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001149void darwin::CC1::AddCPPArgs(const ArgList &Args,
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001150 ArgStringList &CmdArgs) const {
1151 // Derived from cpp spec.
1152
1153 if (Args.hasArg(options::OPT_static)) {
1154 // The gcc spec is broken here, it refers to dynamic but
1155 // that has been translated. Start by being bug compatible.
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001156
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001157 // if (!Args.hasArg(arglist.parser.dynamicOption))
1158 CmdArgs.push_back("-D__STATIC__");
1159 } else
1160 CmdArgs.push_back("-D__DYNAMIC__");
1161
1162 if (Args.hasArg(options::OPT_pthread))
1163 CmdArgs.push_back("-D_REENTRANT");
1164}
1165
Daniel Dunbar40f12652009-03-29 17:08:39 +00001166void darwin::Preprocess::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001167 Job &Dest, const InputInfo &Output,
1168 const InputInfoList &Inputs,
1169 const ArgList &Args,
Daniel Dunbar40f12652009-03-29 17:08:39 +00001170 const char *LinkingOutput) const {
1171 ArgStringList CmdArgs;
1172
1173 assert(Inputs.size() == 1 && "Unexpected number of inputs!");
1174
1175 CmdArgs.push_back("-E");
1176
1177 if (Args.hasArg(options::OPT_traditional) ||
Daniel Dunbar40f12652009-03-29 17:08:39 +00001178 Args.hasArg(options::OPT_traditional_cpp))
1179 CmdArgs.push_back("-traditional-cpp");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001180
Daniel Dunbar40f12652009-03-29 17:08:39 +00001181 ArgStringList OutputArgs;
1182 if (Output.isFilename()) {
1183 OutputArgs.push_back("-o");
1184 OutputArgs.push_back(Output.getFilename());
1185 } else {
1186 assert(Output.isPipe() && "Unexpected CC1 output.");
1187 }
1188
Daniel Dunbar9120f172009-03-29 22:27:40 +00001189 if (Args.hasArg(options::OPT_E)) {
1190 AddCPPOptionsArgs(Args, CmdArgs, Inputs, OutputArgs);
1191 } else {
1192 AddCPPOptionsArgs(Args, CmdArgs, Inputs, ArgStringList());
1193 CmdArgs.append(OutputArgs.begin(), OutputArgs.end());
1194 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001195
Daniel Dunbar8a2073a2009-04-03 01:27:06 +00001196 Args.AddAllArgs(CmdArgs, options::OPT_d_Group);
1197
Daniel Dunbar40f12652009-03-29 17:08:39 +00001198 const char *CC1Name = getCC1Name(Inputs[0].getType());
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001199 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001200 Args.MakeArgString(getToolChain().GetProgramPath(C, CC1Name));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00001201 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar40f12652009-03-29 17:08:39 +00001202}
1203
1204void darwin::Compile::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001205 Job &Dest, const InputInfo &Output,
1206 const InputInfoList &Inputs,
1207 const ArgList &Args,
Daniel Dunbar40f12652009-03-29 17:08:39 +00001208 const char *LinkingOutput) const {
1209 const Driver &D = getToolChain().getHost().getDriver();
1210 ArgStringList CmdArgs;
1211
1212 assert(Inputs.size() == 1 && "Unexpected number of inputs!");
1213
1214 types::ID InputType = Inputs[0].getType();
1215 const Arg *A;
Eli Friedmanceb5c5b2009-07-14 21:58:17 +00001216 if ((A = Args.getLastArg(options::OPT_traditional)))
Daniel Dunbar40f12652009-03-29 17:08:39 +00001217 D.Diag(clang::diag::err_drv_argument_only_allowed_with)
1218 << A->getAsString(Args) << "-E";
1219
1220 if (Output.getType() == types::TY_LLVMAsm)
1221 CmdArgs.push_back("-emit-llvm");
1222 else if (Output.getType() == types::TY_LLVMBC)
1223 CmdArgs.push_back("-emit-llvm-bc");
Daniel Dunbar5915fbf2009-09-01 16:57:46 +00001224 else if (Output.getType() == types::TY_AST)
1225 D.Diag(clang::diag::err_drv_no_ast_support)
Daniel Dunbar88137642009-09-09 22:32:48 +00001226 << getToolChain().getTripleString();
Daniel Dunbar40f12652009-03-29 17:08:39 +00001227
1228 ArgStringList OutputArgs;
1229 if (Output.getType() != types::TY_PCH) {
1230 OutputArgs.push_back("-o");
1231 if (Output.isPipe())
1232 OutputArgs.push_back("-");
1233 else if (Output.isNothing())
1234 OutputArgs.push_back("/dev/null");
1235 else
1236 OutputArgs.push_back(Output.getFilename());
1237 }
1238
1239 // There is no need for this level of compatibility, but it makes
1240 // diffing easier.
1241 bool OutputArgsEarly = (Args.hasArg(options::OPT_fsyntax_only) ||
1242 Args.hasArg(options::OPT_S));
1243
1244 if (types::getPreprocessedType(InputType) != types::TY_INVALID) {
Daniel Dunbara3ec60e2009-03-29 18:40:18 +00001245 AddCPPUniqueOptionsArgs(Args, CmdArgs, Inputs);
Daniel Dunbar40f12652009-03-29 17:08:39 +00001246 if (OutputArgsEarly) {
1247 AddCC1OptionsArgs(Args, CmdArgs, Inputs, OutputArgs);
1248 } else {
1249 AddCC1OptionsArgs(Args, CmdArgs, Inputs, ArgStringList());
1250 CmdArgs.append(OutputArgs.begin(), OutputArgs.end());
1251 }
1252 } else {
1253 CmdArgs.push_back("-fpreprocessed");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001254
Daniel Dunbar40f12652009-03-29 17:08:39 +00001255 for (InputInfoList::const_iterator
1256 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
1257 const InputInfo &II = *it;
1258
Daniel Dunbar5915fbf2009-09-01 16:57:46 +00001259 // Reject AST inputs.
1260 if (II.getType() == types::TY_AST) {
1261 D.Diag(clang::diag::err_drv_no_ast_support)
Daniel Dunbar88137642009-09-09 22:32:48 +00001262 << getToolChain().getTripleString();
Daniel Dunbar5915fbf2009-09-01 16:57:46 +00001263 return;
1264 }
1265
Daniel Dunbar40f12652009-03-29 17:08:39 +00001266 if (II.isPipe())
1267 CmdArgs.push_back("-");
1268 else
1269 CmdArgs.push_back(II.getFilename());
1270 }
1271
1272 if (OutputArgsEarly) {
1273 AddCC1OptionsArgs(Args, CmdArgs, Inputs, OutputArgs);
1274 } else {
1275 AddCC1OptionsArgs(Args, CmdArgs, Inputs, ArgStringList());
1276 CmdArgs.append(OutputArgs.begin(), OutputArgs.end());
1277 }
1278 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001279
Daniel Dunbar40f12652009-03-29 17:08:39 +00001280 if (Output.getType() == types::TY_PCH) {
1281 assert(Output.isFilename() && "Invalid PCH output.");
1282
1283 CmdArgs.push_back("-o");
1284 // NOTE: gcc uses a temp .s file for this, but there doesn't seem
1285 // to be a good reason.
1286 CmdArgs.push_back("/dev/null");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001287
Daniel Dunbar40f12652009-03-29 17:08:39 +00001288 CmdArgs.push_back("--output-pch=");
1289 CmdArgs.push_back(Output.getFilename());
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001290 }
Daniel Dunbar40f12652009-03-29 17:08:39 +00001291
1292 const char *CC1Name = getCC1Name(Inputs[0].getType());
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001293 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001294 Args.MakeArgString(getToolChain().GetProgramPath(C, CC1Name));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00001295 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar40f12652009-03-29 17:08:39 +00001296}
1297
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001298void darwin::Assemble::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001299 Job &Dest, const InputInfo &Output,
1300 const InputInfoList &Inputs,
1301 const ArgList &Args,
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001302 const char *LinkingOutput) const {
1303 ArgStringList CmdArgs;
1304
1305 assert(Inputs.size() == 1 && "Unexpected number of inputs.");
1306 const InputInfo &Input = Inputs[0];
1307
1308 // Bit of a hack, this is only used for original inputs.
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001309 //
Daniel Dunbar8e4fea62009-04-01 00:27:44 +00001310 // FIXME: This is broken for preprocessed .s inputs.
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001311 if (Input.isFilename() &&
Daniel Dunbar8e4fea62009-04-01 00:27:44 +00001312 strcmp(Input.getFilename(), Input.getBaseInput()) == 0) {
1313 if (Args.hasArg(options::OPT_gstabs))
1314 CmdArgs.push_back("--gstabs");
1315 else if (Args.hasArg(options::OPT_g_Group))
1316 CmdArgs.push_back("--gdwarf2");
1317 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001318
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001319 // Derived from asm spec.
Daniel Dunbarcc6f8032009-09-09 18:36:27 +00001320 AddDarwinArch(Args, CmdArgs);
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001321
Daniel Dunbarcc6f8032009-09-09 18:36:27 +00001322 if (!getDarwinToolChain().isIPhone() ||
1323 Args.hasArg(options::OPT_force__cpusubtype__ALL))
1324 CmdArgs.push_back("-force_cpusubtype_ALL");
1325
Daniel Dunbar0e2679d2009-08-24 22:26:16 +00001326 if (getToolChain().getTriple().getArch() != llvm::Triple::x86_64 &&
1327 (Args.hasArg(options::OPT_mkernel) ||
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001328 Args.hasArg(options::OPT_static) ||
Daniel Dunbar0e2679d2009-08-24 22:26:16 +00001329 Args.hasArg(options::OPT_fapple_kext)))
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001330 CmdArgs.push_back("-static");
1331
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001332 Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA,
1333 options::OPT_Xassembler);
1334
1335 assert(Output.isFilename() && "Unexpected lipo output.");
1336 CmdArgs.push_back("-o");
1337 CmdArgs.push_back(Output.getFilename());
1338
1339 if (Input.isPipe()) {
1340 CmdArgs.push_back("-");
1341 } else {
1342 assert(Input.isFilename() && "Invalid input.");
1343 CmdArgs.push_back(Input.getFilename());
1344 }
1345
1346 // asm_final spec is empty.
1347
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001348 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001349 Args.MakeArgString(getToolChain().GetProgramPath(C, "as"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00001350 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar8cac5f72009-03-20 16:06:39 +00001351}
Daniel Dunbarff7488d2009-03-20 00:52:38 +00001352
Daniel Dunbar02633b52009-03-26 16:23:12 +00001353/// Helper routine for seeing if we should use dsymutil; this is a
1354/// gcc compatible hack, we should remove it and use the input
1355/// type information.
1356static bool isSourceSuffix(const char *Str) {
1357 // match: 'C', 'CPP', 'c', 'cc', 'cp', 'c++', 'cpp', 'cxx', 'm',
1358 // 'mm'.
1359 switch (strlen(Str)) {
1360 default:
1361 return false;
1362 case 1:
1363 return (memcmp(Str, "C", 1) == 0 ||
1364 memcmp(Str, "c", 1) == 0 ||
1365 memcmp(Str, "m", 1) == 0);
1366 case 2:
1367 return (memcmp(Str, "cc", 2) == 0 ||
1368 memcmp(Str, "cp", 2) == 0 ||
1369 memcmp(Str, "mm", 2) == 0);
1370 case 3:
1371 return (memcmp(Str, "CPP", 3) == 0 ||
1372 memcmp(Str, "c++", 3) == 0 ||
1373 memcmp(Str, "cpp", 3) == 0 ||
1374 memcmp(Str, "cxx", 3) == 0);
1375 }
1376}
1377
1378static bool isMacosxVersionLT(unsigned (&A)[3], unsigned (&B)[3]) {
1379 for (unsigned i=0; i < 3; ++i) {
1380 if (A[i] > B[i]) return false;
1381 if (A[i] < B[i]) return true;
1382 }
1383 return false;
1384}
1385
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001386static bool isMacosxVersionLT(unsigned (&A)[3],
Daniel Dunbar02633b52009-03-26 16:23:12 +00001387 unsigned V0, unsigned V1=0, unsigned V2=0) {
1388 unsigned B[3] = { V0, V1, V2 };
1389 return isMacosxVersionLT(A, B);
1390}
1391
Daniel Dunbar78dbd582009-09-04 18:35:31 +00001392// FIXME: Can we tablegen this?
1393static const char *GetArmArchForMArch(llvm::StringRef Value) {
1394 if (Value == "armv6k")
1395 return "armv6";
1396
1397 if (Value == "armv5tej")
1398 return "armv5";
1399
1400 if (Value == "xscale")
1401 return "xscale";
1402
1403 if (Value == "armv4t")
1404 return "armv4t";
1405
1406 if (Value == "armv7" || Value == "armv7-a" || Value == "armv7-r" ||
1407 Value == "armv7-m" || Value == "armv7a" || Value == "armv7r" ||
1408 Value == "armv7m")
1409 return "armv7";
1410
1411 return 0;
1412}
1413
1414// FIXME: Can we tablegen this?
1415static const char *GetArmArchForMCpu(llvm::StringRef Value) {
1416 if (Value == "arm10tdmi" || Value == "arm1020t" || Value == "arm9e" ||
1417 Value == "arm946e-s" || Value == "arm966e-s" ||
1418 Value == "arm968e-s" || Value == "arm10e" ||
1419 Value == "arm1020e" || Value == "arm1022e" || Value == "arm926ej-s" ||
1420 Value == "arm1026ej-s")
1421 return "armv5";
1422
1423 if (Value == "xscale")
1424 return "xscale";
1425
1426 if (Value == "arm1136j-s" || Value == "arm1136jf-s" ||
1427 Value == "arm1176jz-s" || Value == "arm1176jzf-s")
1428 return "armv6";
1429
1430 if (Value == "cortex-a8" || Value == "cortex-r4" || Value == "cortex-m3")
1431 return "armv7";
1432
1433 return 0;
1434}
1435
Daniel Dunbarfbefe6b2009-09-09 18:36:20 +00001436void darwin::DarwinTool::AddDarwinArch(const ArgList &Args,
1437 ArgStringList &CmdArgs) const {
Daniel Dunbar02633b52009-03-26 16:23:12 +00001438 // Derived from darwin_arch spec.
1439 CmdArgs.push_back("-arch");
Daniel Dunbar78dbd582009-09-04 18:35:31 +00001440
1441 switch (getToolChain().getTriple().getArch()) {
1442 default:
Daniel Dunbar88137642009-09-09 22:32:48 +00001443 CmdArgs.push_back(Args.MakeArgString(getToolChain().getArchName()));
Daniel Dunbar78dbd582009-09-04 18:35:31 +00001444 break;
1445
1446 case llvm::Triple::arm: {
Daniel Dunbar78dbd582009-09-04 18:35:31 +00001447 if (const Arg *A = Args.getLastArg(options::OPT_march_EQ)) {
1448 if (const char *Arch = GetArmArchForMArch(A->getValue(Args))) {
1449 CmdArgs.push_back(Arch);
1450 return;
1451 }
1452 }
1453
1454 if (const Arg *A = Args.getLastArg(options::OPT_mcpu_EQ)) {
1455 if (const char *Arch = GetArmArchForMCpu(A->getValue(Args))) {
1456 CmdArgs.push_back(Arch);
1457 return;
1458 }
1459 }
1460
1461 CmdArgs.push_back("arm");
1462 CmdArgs.push_back("-force_cpusubtype_ALL");
1463 return;
1464 }
1465 }
Daniel Dunbar02633b52009-03-26 16:23:12 +00001466}
1467
Daniel Dunbarfbefe6b2009-09-09 18:36:20 +00001468void darwin::DarwinTool::AddDarwinSubArch(const ArgList &Args,
1469 ArgStringList &CmdArgs) const {
Daniel Dunbar02633b52009-03-26 16:23:12 +00001470 // Derived from darwin_subarch spec, not sure what the distinction
1471 // exists for but at least for this chain it is the same.
1472 AddDarwinArch(Args, CmdArgs);
1473}
1474
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001475void darwin::Link::AddLinkArgs(const ArgList &Args,
Daniel Dunbar02633b52009-03-26 16:23:12 +00001476 ArgStringList &CmdArgs) const {
1477 const Driver &D = getToolChain().getHost().getDriver();
1478
1479 // Derived from the "link" spec.
1480 Args.AddAllArgs(CmdArgs, options::OPT_static);
1481 if (!Args.hasArg(options::OPT_static))
1482 CmdArgs.push_back("-dynamic");
1483 if (Args.hasArg(options::OPT_fgnu_runtime)) {
1484 // FIXME: gcc replaces -lobjc in forward args with -lobjc-gnu
1485 // here. How do we wish to handle such things?
1486 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001487
Daniel Dunbar02633b52009-03-26 16:23:12 +00001488 if (!Args.hasArg(options::OPT_dynamiclib)) {
1489 if (Args.hasArg(options::OPT_force__cpusubtype__ALL)) {
1490 AddDarwinArch(Args, CmdArgs);
1491 CmdArgs.push_back("-force_cpusubtype_ALL");
1492 } else
1493 AddDarwinSubArch(Args, CmdArgs);
1494
1495 Args.AddLastArg(CmdArgs, options::OPT_bundle);
1496 Args.AddAllArgs(CmdArgs, options::OPT_bundle__loader);
1497 Args.AddAllArgs(CmdArgs, options::OPT_client__name);
1498
1499 Arg *A;
1500 if ((A = Args.getLastArg(options::OPT_compatibility__version)) ||
1501 (A = Args.getLastArg(options::OPT_current__version)) ||
1502 (A = Args.getLastArg(options::OPT_install__name)))
1503 D.Diag(clang::diag::err_drv_argument_only_allowed_with)
1504 << A->getAsString(Args) << "-dynamiclib";
1505
1506 Args.AddLastArg(CmdArgs, options::OPT_force__flat__namespace);
1507 Args.AddLastArg(CmdArgs, options::OPT_keep__private__externs);
1508 Args.AddLastArg(CmdArgs, options::OPT_private__bundle);
1509 } else {
1510 CmdArgs.push_back("-dylib");
1511
1512 Arg *A;
1513 if ((A = Args.getLastArg(options::OPT_bundle)) ||
1514 (A = Args.getLastArg(options::OPT_bundle__loader)) ||
1515 (A = Args.getLastArg(options::OPT_client__name)) ||
1516 (A = Args.getLastArg(options::OPT_force__flat__namespace)) ||
1517 (A = Args.getLastArg(options::OPT_keep__private__externs)) ||
1518 (A = Args.getLastArg(options::OPT_private__bundle)))
1519 D.Diag(clang::diag::err_drv_argument_not_allowed_with)
1520 << A->getAsString(Args) << "-dynamiclib";
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001521
Daniel Dunbar02633b52009-03-26 16:23:12 +00001522 Args.AddAllArgsTranslated(CmdArgs, options::OPT_compatibility__version,
1523 "-dylib_compatibility_version");
1524 Args.AddAllArgsTranslated(CmdArgs, options::OPT_current__version,
1525 "-dylib_current_version");
1526
1527 if (Args.hasArg(options::OPT_force__cpusubtype__ALL)) {
1528 AddDarwinArch(Args, CmdArgs);
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001529 // NOTE: We don't add -force_cpusubtype_ALL on this path. Ok.
Daniel Dunbar02633b52009-03-26 16:23:12 +00001530 } else
1531 AddDarwinSubArch(Args, CmdArgs);
1532
1533 Args.AddAllArgsTranslated(CmdArgs, options::OPT_install__name,
1534 "-dylib_install_name");
1535 }
1536
1537 Args.AddLastArg(CmdArgs, options::OPT_all__load);
1538 Args.AddAllArgs(CmdArgs, options::OPT_allowable__client);
1539 Args.AddLastArg(CmdArgs, options::OPT_bind__at__load);
Daniel Dunbard82f8fa2009-09-04 18:35:41 +00001540 if (getDarwinToolChain().isIPhone())
1541 Args.AddLastArg(CmdArgs, options::OPT_arch__errors__fatal);
Daniel Dunbar02633b52009-03-26 16:23:12 +00001542 Args.AddLastArg(CmdArgs, options::OPT_dead__strip);
1543 Args.AddLastArg(CmdArgs, options::OPT_no__dead__strip__inits__and__terms);
1544 Args.AddAllArgs(CmdArgs, options::OPT_dylib__file);
1545 Args.AddLastArg(CmdArgs, options::OPT_dynamic);
1546 Args.AddAllArgs(CmdArgs, options::OPT_exported__symbols__list);
1547 Args.AddLastArg(CmdArgs, options::OPT_flat__namespace);
1548 Args.AddAllArgs(CmdArgs, options::OPT_headerpad__max__install__names);
1549 Args.AddAllArgs(CmdArgs, options::OPT_image__base);
1550 Args.AddAllArgs(CmdArgs, options::OPT_init);
1551
Daniel Dunbar30392de2009-09-04 18:35:21 +00001552 if (!Args.hasArg(options::OPT_mmacosx_version_min_EQ) &&
1553 !Args.hasArg(options::OPT_miphoneos_version_min_EQ)) {
1554 // Add default version min.
1555 if (!getDarwinToolChain().isIPhone()) {
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001556 CmdArgs.push_back("-macosx_version_min");
1557 CmdArgs.push_back(getDarwinToolChain().getMacosxVersionStr());
Daniel Dunbar30392de2009-09-04 18:35:21 +00001558 } else {
1559 CmdArgs.push_back("-iphoneos_version_min");
1560 CmdArgs.push_back(getDarwinToolChain().getIPhoneOSVersionStr());
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001561 }
Daniel Dunbar02633b52009-03-26 16:23:12 +00001562 }
1563
Daniel Dunbar30392de2009-09-04 18:35:21 +00001564 // Adding all arguments doesn't make sense here but this is what
1565 // gcc does.
1566 Args.AddAllArgsTranslated(CmdArgs, options::OPT_mmacosx_version_min_EQ,
1567 "-macosx_version_min");
Daniel Dunbar02633b52009-03-26 16:23:12 +00001568 Args.AddAllArgsTranslated(CmdArgs, options::OPT_miphoneos_version_min_EQ,
1569 "-iphoneos_version_min");
1570 Args.AddLastArg(CmdArgs, options::OPT_nomultidefs);
1571 Args.AddLastArg(CmdArgs, options::OPT_multi__module);
1572 Args.AddLastArg(CmdArgs, options::OPT_single__module);
1573 Args.AddAllArgs(CmdArgs, options::OPT_multiply__defined);
1574 Args.AddAllArgs(CmdArgs, options::OPT_multiply__defined__unused);
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001575
Daniel Dunbar02633b52009-03-26 16:23:12 +00001576 if (Args.hasArg(options::OPT_fpie))
1577 CmdArgs.push_back("-pie");
1578
1579 Args.AddLastArg(CmdArgs, options::OPT_prebind);
1580 Args.AddLastArg(CmdArgs, options::OPT_noprebind);
1581 Args.AddLastArg(CmdArgs, options::OPT_nofixprebinding);
1582 Args.AddLastArg(CmdArgs, options::OPT_prebind__all__twolevel__modules);
1583 Args.AddLastArg(CmdArgs, options::OPT_read__only__relocs);
1584 Args.AddAllArgs(CmdArgs, options::OPT_sectcreate);
1585 Args.AddAllArgs(CmdArgs, options::OPT_sectorder);
1586 Args.AddAllArgs(CmdArgs, options::OPT_seg1addr);
1587 Args.AddAllArgs(CmdArgs, options::OPT_segprot);
1588 Args.AddAllArgs(CmdArgs, options::OPT_segaddr);
1589 Args.AddAllArgs(CmdArgs, options::OPT_segs__read__only__addr);
1590 Args.AddAllArgs(CmdArgs, options::OPT_segs__read__write__addr);
1591 Args.AddAllArgs(CmdArgs, options::OPT_seg__addr__table);
1592 Args.AddAllArgs(CmdArgs, options::OPT_seg__addr__table__filename);
1593 Args.AddAllArgs(CmdArgs, options::OPT_sub__library);
1594 Args.AddAllArgs(CmdArgs, options::OPT_sub__umbrella);
Daniel Dunbard82f8fa2009-09-04 18:35:41 +00001595
Daniel Dunbar02633b52009-03-26 16:23:12 +00001596 Args.AddAllArgsTranslated(CmdArgs, options::OPT_isysroot, "-syslibroot");
Daniel Dunbard82f8fa2009-09-04 18:35:41 +00001597 if (getDarwinToolChain().isIPhone()) {
1598 if (!Args.hasArg(options::OPT_isysroot)) {
1599 CmdArgs.push_back("-syslibroot");
1600 CmdArgs.push_back("/Developer/SDKs/Extra");
1601 }
1602 }
1603
Daniel Dunbar02633b52009-03-26 16:23:12 +00001604 Args.AddLastArg(CmdArgs, options::OPT_twolevel__namespace);
1605 Args.AddLastArg(CmdArgs, options::OPT_twolevel__namespace__hints);
1606 Args.AddAllArgs(CmdArgs, options::OPT_umbrella);
1607 Args.AddAllArgs(CmdArgs, options::OPT_undefined);
1608 Args.AddAllArgs(CmdArgs, options::OPT_unexported__symbols__list);
Daniel Dunbar02633b52009-03-26 16:23:12 +00001609
Daniel Dunbard82f8fa2009-09-04 18:35:41 +00001610 Args.AddAllArgs(CmdArgs, options::OPT_weak__reference__mismatches);
Daniel Dunbar02633b52009-03-26 16:23:12 +00001611 if (!Args.hasArg(options::OPT_weak__reference__mismatches)) {
1612 CmdArgs.push_back("-weak_reference_mismatches");
1613 CmdArgs.push_back("non-weak");
1614 }
1615
1616 Args.AddLastArg(CmdArgs, options::OPT_X_Flag);
1617 Args.AddAllArgs(CmdArgs, options::OPT_y);
1618 Args.AddLastArg(CmdArgs, options::OPT_w);
1619 Args.AddAllArgs(CmdArgs, options::OPT_pagezero__size);
1620 Args.AddAllArgs(CmdArgs, options::OPT_segs__read__);
1621 Args.AddLastArg(CmdArgs, options::OPT_seglinkedit);
1622 Args.AddLastArg(CmdArgs, options::OPT_noseglinkedit);
1623 Args.AddAllArgs(CmdArgs, options::OPT_sectalign);
1624 Args.AddAllArgs(CmdArgs, options::OPT_sectobjectsymbols);
1625 Args.AddAllArgs(CmdArgs, options::OPT_segcreate);
1626 Args.AddLastArg(CmdArgs, options::OPT_whyload);
1627 Args.AddLastArg(CmdArgs, options::OPT_whatsloaded);
1628 Args.AddAllArgs(CmdArgs, options::OPT_dylinker__install__name);
1629 Args.AddLastArg(CmdArgs, options::OPT_dylinker);
1630 Args.AddLastArg(CmdArgs, options::OPT_Mach);
1631}
1632
1633void darwin::Link::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001634 Job &Dest, const InputInfo &Output,
1635 const InputInfoList &Inputs,
1636 const ArgList &Args,
Daniel Dunbar02633b52009-03-26 16:23:12 +00001637 const char *LinkingOutput) const {
1638 assert(Output.getType() == types::TY_Image && "Invalid linker output type.");
Daniel Dunbare0be8b12009-09-08 16:39:16 +00001639
Daniel Dunbar02633b52009-03-26 16:23:12 +00001640 // The logic here is derived from gcc's behavior; most of which
1641 // comes from specs (starting with link_command). Consult gcc for
1642 // more information.
Daniel Dunbar02633b52009-03-26 16:23:12 +00001643 ArgStringList CmdArgs;
1644
1645 // I'm not sure why this particular decomposition exists in gcc, but
1646 // we follow suite for ease of comparison.
1647 AddLinkArgs(Args, CmdArgs);
1648
Daniel Dunbar02633b52009-03-26 16:23:12 +00001649 Args.AddAllArgs(CmdArgs, options::OPT_d_Flag);
1650 Args.AddAllArgs(CmdArgs, options::OPT_s);
1651 Args.AddAllArgs(CmdArgs, options::OPT_t);
1652 Args.AddAllArgs(CmdArgs, options::OPT_Z_Flag);
1653 Args.AddAllArgs(CmdArgs, options::OPT_u_Group);
1654 Args.AddAllArgs(CmdArgs, options::OPT_A);
1655 Args.AddLastArg(CmdArgs, options::OPT_e);
1656 Args.AddAllArgs(CmdArgs, options::OPT_m_Separate);
1657 Args.AddAllArgs(CmdArgs, options::OPT_r);
1658
Daniel Dunbar02633b52009-03-26 16:23:12 +00001659 CmdArgs.push_back("-o");
1660 CmdArgs.push_back(Output.getFilename());
1661
1662 unsigned MacosxVersion[3];
1663 if (Arg *A = Args.getLastArg(options::OPT_mmacosx_version_min_EQ)) {
1664 bool HadExtra;
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001665 if (!Driver::GetReleaseVersion(A->getValue(Args), MacosxVersion[0],
Daniel Dunbar02633b52009-03-26 16:23:12 +00001666 MacosxVersion[1], MacosxVersion[2],
1667 HadExtra) ||
1668 HadExtra) {
1669 const Driver &D = getToolChain().getHost().getDriver();
1670 D.Diag(clang::diag::err_drv_invalid_version_number)
1671 << A->getAsString(Args);
1672 }
1673 } else {
1674 getDarwinToolChain().getMacosxVersion(MacosxVersion);
1675 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001676
Daniel Dunbar02633b52009-03-26 16:23:12 +00001677 if (!Args.hasArg(options::OPT_A) &&
1678 !Args.hasArg(options::OPT_nostdlib) &&
1679 !Args.hasArg(options::OPT_nostartfiles)) {
1680 // Derived from startfile spec.
1681 if (Args.hasArg(options::OPT_dynamiclib)) {
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001682 // Derived from darwin_dylib1 spec.
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001683 if (isMacosxVersionLT(MacosxVersion, 10, 5))
Daniel Dunbar02633b52009-03-26 16:23:12 +00001684 CmdArgs.push_back("-ldylib1.o");
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001685 else if (isMacosxVersionLT(MacosxVersion, 10, 6))
Daniel Dunbar02633b52009-03-26 16:23:12 +00001686 CmdArgs.push_back("-ldylib1.10.5.o");
1687 } else {
1688 if (Args.hasArg(options::OPT_bundle)) {
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001689 if (!Args.hasArg(options::OPT_static)) {
1690 // Derived from darwin_bundle1 spec.
1691 if (isMacosxVersionLT(MacosxVersion, 10, 6))
1692 CmdArgs.push_back("-lbundle1.o");
1693 }
Daniel Dunbar02633b52009-03-26 16:23:12 +00001694 } else {
1695 if (Args.hasArg(options::OPT_pg)) {
1696 if (Args.hasArg(options::OPT_static) ||
1697 Args.hasArg(options::OPT_object) ||
1698 Args.hasArg(options::OPT_preload)) {
1699 CmdArgs.push_back("-lgcrt0.o");
1700 } else {
1701 CmdArgs.push_back("-lgcrt1.o");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001702
Daniel Dunbar02633b52009-03-26 16:23:12 +00001703 // darwin_crt2 spec is empty.
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001704 }
Daniel Dunbar02633b52009-03-26 16:23:12 +00001705 } else {
1706 if (Args.hasArg(options::OPT_static) ||
1707 Args.hasArg(options::OPT_object) ||
1708 Args.hasArg(options::OPT_preload)) {
1709 CmdArgs.push_back("-lcrt0.o");
1710 } else {
1711 // Derived from darwin_crt1 spec.
Daniel Dunbarae33f8f2009-09-04 18:35:47 +00001712 if (getDarwinToolChain().isIPhone()) {
1713 CmdArgs.push_back("-lcrt1.o");
1714 } else if (isMacosxVersionLT(MacosxVersion, 10, 5))
Daniel Dunbar02633b52009-03-26 16:23:12 +00001715 CmdArgs.push_back("-lcrt1.o");
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001716 else if (isMacosxVersionLT(MacosxVersion, 10, 6))
Daniel Dunbar02633b52009-03-26 16:23:12 +00001717 CmdArgs.push_back("-lcrt1.10.5.o");
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001718 else
1719 CmdArgs.push_back("-lcrt1.10.6.o");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001720
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001721 // darwin_crt2 spec is empty.
Daniel Dunbar02633b52009-03-26 16:23:12 +00001722 }
1723 }
1724 }
1725 }
1726
1727 if (Args.hasArg(options::OPT_shared_libgcc) &&
1728 !Args.hasArg(options::OPT_miphoneos_version_min_EQ) &&
1729 isMacosxVersionLT(MacosxVersion, 10, 5)) {
Daniel Dunbar88137642009-09-09 22:32:48 +00001730 const char *Str =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001731 Args.MakeArgString(getToolChain().GetFilePath(C, "crt3.o"));
Daniel Dunbar88137642009-09-09 22:32:48 +00001732 CmdArgs.push_back(Str);
Daniel Dunbar02633b52009-03-26 16:23:12 +00001733 }
1734 }
1735
1736 Args.AddAllArgs(CmdArgs, options::OPT_L);
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001737
Daniel Dunbar02633b52009-03-26 16:23:12 +00001738 if (Args.hasArg(options::OPT_fopenmp))
1739 // This is more complicated in gcc...
1740 CmdArgs.push_back("-lgomp");
1741
1742 // FIXME: Derive these correctly.
Daniel Dunbar302e2ea2009-09-09 22:32:41 +00001743 llvm::StringRef TCDir = getDarwinToolChain().getToolChainDir();
Daniel Dunbar02633b52009-03-26 16:23:12 +00001744 if (getToolChain().getArchName() == "x86_64") {
Daniel Dunbar302e2ea2009-09-09 22:32:41 +00001745 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/gcc/" + TCDir +
1746 "/x86_64"));
Daniel Dunbar02633b52009-03-26 16:23:12 +00001747 // Intentionally duplicated for (temporary) gcc bug compatibility.
Daniel Dunbar302e2ea2009-09-09 22:32:41 +00001748 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/gcc/" + TCDir +
1749 "/x86_64"));
Daniel Dunbar02633b52009-03-26 16:23:12 +00001750 }
Daniel Dunbar302e2ea2009-09-09 22:32:41 +00001751 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/" + TCDir));
1752 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/gcc/" + TCDir));
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001753 // Intentionally duplicated for (temporary) gcc bug compatibility.
Daniel Dunbar302e2ea2009-09-09 22:32:41 +00001754 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/gcc/" + TCDir));
1755 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/gcc/" + TCDir +
1756 "/../../../" + TCDir));
1757 CmdArgs.push_back(Args.MakeArgString("-L/usr/lib/gcc/" + TCDir +
1758 "/../../.."));
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001759
Daniel Dunbar02633b52009-03-26 16:23:12 +00001760 for (InputInfoList::const_iterator
1761 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
1762 const InputInfo &II = *it;
1763 if (II.isFilename())
1764 CmdArgs.push_back(II.getFilename());
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001765 else
Daniel Dunbar02633b52009-03-26 16:23:12 +00001766 II.getInputArg().renderAsInput(Args, CmdArgs);
1767 }
1768
1769 if (LinkingOutput) {
1770 CmdArgs.push_back("-arch_multiple");
1771 CmdArgs.push_back("-final_output");
1772 CmdArgs.push_back(LinkingOutput);
1773 }
1774
1775 if (Args.hasArg(options::OPT_fprofile_arcs) ||
1776 Args.hasArg(options::OPT_fprofile_generate) ||
1777 Args.hasArg(options::OPT_fcreate_profile) ||
1778 Args.hasArg(options::OPT_coverage))
1779 CmdArgs.push_back("-lgcov");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001780
Daniel Dunbar02633b52009-03-26 16:23:12 +00001781 if (Args.hasArg(options::OPT_fnested_functions))
1782 CmdArgs.push_back("-allow_stack_execute");
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001783
Daniel Dunbar02633b52009-03-26 16:23:12 +00001784 if (!Args.hasArg(options::OPT_nostdlib) &&
1785 !Args.hasArg(options::OPT_nodefaultlibs)) {
Daniel Dunbaredfa02b2009-04-08 06:06:21 +00001786 // FIXME: g++ is more complicated here, it tries to put -lstdc++
1787 // before -lm, for example.
1788 if (getToolChain().getHost().getDriver().CCCIsCXX)
1789 CmdArgs.push_back("-lstdc++");
1790
Daniel Dunbar02633b52009-03-26 16:23:12 +00001791 // link_ssp spec is empty.
1792
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001793 // Derived from libgcc and lib specs but refactored.
Daniel Dunbar02633b52009-03-26 16:23:12 +00001794 if (Args.hasArg(options::OPT_static)) {
1795 CmdArgs.push_back("-lgcc_static");
Daniel Dunbar02633b52009-03-26 16:23:12 +00001796 } else {
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001797 if (Args.hasArg(options::OPT_static_libgcc)) {
1798 CmdArgs.push_back("-lgcc_eh");
1799 } else if (Args.hasArg(options::OPT_miphoneos_version_min_EQ)) {
1800 // Derived from darwin_iphoneos_libgcc spec.
Daniel Dunbarae33f8f2009-09-04 18:35:47 +00001801 if (getDarwinToolChain().isIPhone()) {
1802 CmdArgs.push_back("-lgcc_s.1");
1803 } else {
1804 CmdArgs.push_back("-lgcc_s.10.5");
1805 }
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001806 } else if (Args.hasArg(options::OPT_shared_libgcc) ||
Mike Stump738f8c22009-07-31 23:15:31 +00001807 // FIXME: -fexceptions -fno-exceptions means no exceptions
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001808 Args.hasArg(options::OPT_fexceptions) ||
1809 Args.hasArg(options::OPT_fgnu_runtime)) {
1810 // FIXME: This is probably broken on 10.3?
1811 if (isMacosxVersionLT(MacosxVersion, 10, 5))
1812 CmdArgs.push_back("-lgcc_s.10.4");
1813 else if (isMacosxVersionLT(MacosxVersion, 10, 6))
1814 CmdArgs.push_back("-lgcc_s.10.5");
1815 } else {
1816 if (isMacosxVersionLT(MacosxVersion, 10, 3, 9))
1817 ; // Do nothing.
1818 else if (isMacosxVersionLT(MacosxVersion, 10, 5))
1819 CmdArgs.push_back("-lgcc_s.10.4");
1820 else if (isMacosxVersionLT(MacosxVersion, 10, 6))
1821 CmdArgs.push_back("-lgcc_s.10.5");
1822 }
Daniel Dunbar02633b52009-03-26 16:23:12 +00001823
Daniel Dunbarae33f8f2009-09-04 18:35:47 +00001824 if (getDarwinToolChain().isIPhone() ||
1825 isMacosxVersionLT(MacosxVersion, 10, 6)) {
Daniel Dunbar8a8d8af2009-04-01 03:17:40 +00001826 CmdArgs.push_back("-lgcc");
1827 CmdArgs.push_back("-lSystem");
1828 } else {
1829 CmdArgs.push_back("-lSystem");
1830 CmdArgs.push_back("-lgcc");
1831 }
1832 }
Daniel Dunbar02633b52009-03-26 16:23:12 +00001833 }
1834
1835 if (!Args.hasArg(options::OPT_A) &&
1836 !Args.hasArg(options::OPT_nostdlib) &&
1837 !Args.hasArg(options::OPT_nostartfiles)) {
1838 // endfile_spec is empty.
1839 }
1840
1841 Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
1842 Args.AddAllArgs(CmdArgs, options::OPT_F);
1843
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001844 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001845 Args.MakeArgString(getToolChain().GetProgramPath(C, "ld"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00001846 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar02633b52009-03-26 16:23:12 +00001847
Daniel Dunbar0b46e1b2009-04-04 00:55:30 +00001848 // Find the first non-empty base input (we want to ignore linker
1849 // inputs).
1850 const char *BaseInput = "";
1851 for (unsigned i = 0, e = Inputs.size(); i != e; ++i) {
1852 if (Inputs[i].getBaseInput()[0] != '\0') {
1853 BaseInput = Inputs[i].getBaseInput();
1854 break;
1855 }
1856 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001857
Daniel Dunbar3ed29452009-04-24 03:03:52 +00001858 // Run dsymutil if we are making an executable in a single step.
1859 //
1860 // FIXME: Currently we don't want to do this when we are part of a
1861 // universal build step, as this would end up creating stray temp
1862 // files.
1863 if (!LinkingOutput &&
1864 Args.getLastArg(options::OPT_g_Group) &&
Daniel Dunbar02633b52009-03-26 16:23:12 +00001865 !Args.getLastArg(options::OPT_gstabs) &&
1866 !Args.getLastArg(options::OPT_g0)) {
1867 // FIXME: This is gross, but matches gcc. The test only considers
1868 // the suffix (not the -x type), and then only of the first
Daniel Dunbar0b46e1b2009-04-04 00:55:30 +00001869 // source input. Awesome.
1870 const char *Suffix = strrchr(BaseInput, '.');
Daniel Dunbar02633b52009-03-26 16:23:12 +00001871 if (Suffix && isSourceSuffix(Suffix + 1)) {
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001872 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001873 Args.MakeArgString(getToolChain().GetProgramPath(C, "dsymutil"));
Daniel Dunbar02633b52009-03-26 16:23:12 +00001874 ArgStringList CmdArgs;
1875 CmdArgs.push_back(Output.getFilename());
Daniel Dunbarcae087e2009-07-01 19:02:28 +00001876 C.getJobs().addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar02633b52009-03-26 16:23:12 +00001877 }
1878 }
1879}
1880
Daniel Dunbarff7488d2009-03-20 00:52:38 +00001881void darwin::Lipo::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001882 Job &Dest, const InputInfo &Output,
1883 const InputInfoList &Inputs,
1884 const ArgList &Args,
Daniel Dunbarff7488d2009-03-20 00:52:38 +00001885 const char *LinkingOutput) const {
1886 ArgStringList CmdArgs;
1887
1888 CmdArgs.push_back("-create");
1889 assert(Output.isFilename() && "Unexpected lipo output.");
Daniel Dunbara428df82009-03-24 00:24:37 +00001890
1891 CmdArgs.push_back("-output");
Daniel Dunbarff7488d2009-03-20 00:52:38 +00001892 CmdArgs.push_back(Output.getFilename());
Daniel Dunbara428df82009-03-24 00:24:37 +00001893
Daniel Dunbarff7488d2009-03-20 00:52:38 +00001894 for (InputInfoList::const_iterator
1895 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
1896 const InputInfo &II = *it;
1897 assert(II.isFilename() && "Unexpected lipo input.");
1898 CmdArgs.push_back(II.getFilename());
1899 }
Daniel Dunbarc21c4852009-04-08 23:54:23 +00001900 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001901 Args.MakeArgString(getToolChain().GetProgramPath(C, "lipo"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00001902 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbarff7488d2009-03-20 00:52:38 +00001903}
Daniel Dunbar68a31d42009-03-31 17:45:15 +00001904
Edward O'Callaghane7925a02009-08-22 01:06:46 +00001905void auroraux::Assemble::ConstructJob(Compilation &C, const JobAction &JA,
1906 Job &Dest, const InputInfo &Output,
1907 const InputInfoList &Inputs,
1908 const ArgList &Args,
Mike Stump1eb44332009-09-09 15:08:12 +00001909 const char *LinkingOutput) const {
Edward O'Callaghane7925a02009-08-22 01:06:46 +00001910 ArgStringList CmdArgs;
1911
1912 Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA,
1913 options::OPT_Xassembler);
1914
1915 CmdArgs.push_back("-o");
1916 if (Output.isPipe())
1917 CmdArgs.push_back("-");
1918 else
1919 CmdArgs.push_back(Output.getFilename());
1920
1921 for (InputInfoList::const_iterator
1922 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
1923 const InputInfo &II = *it;
1924 if (II.isPipe())
1925 CmdArgs.push_back("-");
1926 else
1927 CmdArgs.push_back(II.getFilename());
1928 }
1929
1930 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001931 Args.MakeArgString(getToolChain().GetProgramPath(C, "as"));
Edward O'Callaghane7925a02009-08-22 01:06:46 +00001932 Dest.addCommand(new Command(JA, Exec, CmdArgs));
1933}
1934
1935void auroraux::Link::ConstructJob(Compilation &C, const JobAction &JA,
1936 Job &Dest, const InputInfo &Output,
1937 const InputInfoList &Inputs,
1938 const ArgList &Args,
1939 const char *LinkingOutput) const {
1940 const Driver &D = getToolChain().getHost().getDriver();
1941 ArgStringList CmdArgs;
1942
1943 if ((!Args.hasArg(options::OPT_nostdlib)) &&
1944 (!Args.hasArg(options::OPT_shared))) {
1945 CmdArgs.push_back("-e");
1946 CmdArgs.push_back("__start");
1947 }
1948
1949 if (Args.hasArg(options::OPT_static)) {
1950 CmdArgs.push_back("-Bstatic");
1951 } else {
1952 CmdArgs.push_back("--eh-frame-hdr");
1953 CmdArgs.push_back("-Bdynamic");
1954 if (Args.hasArg(options::OPT_shared)) {
1955 CmdArgs.push_back("-shared");
1956 } else {
1957 CmdArgs.push_back("-dynamic-linker");
1958 CmdArgs.push_back("/lib/ld.so.1"); // 64Bit Path /lib/amd64/ld.so.1
1959 }
1960 }
1961
1962 if (Output.isPipe()) {
1963 CmdArgs.push_back("-o");
1964 CmdArgs.push_back("-");
1965 } else if (Output.isFilename()) {
1966 CmdArgs.push_back("-o");
1967 CmdArgs.push_back(Output.getFilename());
1968 } else {
1969 assert(Output.isNothing() && "Invalid output.");
1970 }
1971
1972 if (!Args.hasArg(options::OPT_nostdlib) &&
1973 !Args.hasArg(options::OPT_nostartfiles)) {
1974 if (!Args.hasArg(options::OPT_shared)) {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001975 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crt0.o")));
1976 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbegin.o")));
Edward O'Callaghane7925a02009-08-22 01:06:46 +00001977 } else {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00001978 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbeginS.o")));
Edward O'Callaghane7925a02009-08-22 01:06:46 +00001979 }
1980 }
1981
1982 CmdArgs.push_back(MakeFormattedString(Args,
1983 llvm::format("-L/opt/gcc4/lib/gcc/%s/4.2.4",
1984 getToolChain().getTripleString().c_str())));
1985
1986 Args.AddAllArgs(CmdArgs, options::OPT_L);
1987 Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
1988 Args.AddAllArgs(CmdArgs, options::OPT_e);
1989
1990 for (InputInfoList::const_iterator
1991 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
1992 const InputInfo &II = *it;
1993
1994 // Don't try to pass LLVM inputs to a generic gcc.
1995 if (II.getType() == types::TY_LLVMBC)
1996 D.Diag(clang::diag::err_drv_no_linker_llvm_support)
Daniel Dunbar88137642009-09-09 22:32:48 +00001997 << getToolChain().getTripleString();
Edward O'Callaghane7925a02009-08-22 01:06:46 +00001998
1999 if (II.isPipe())
2000 CmdArgs.push_back("-");
2001 else if (II.isFilename())
2002 CmdArgs.push_back(II.getFilename());
2003 else
2004 II.getInputArg().renderAsInput(Args, CmdArgs);
2005 }
2006
2007 if (!Args.hasArg(options::OPT_nostdlib) &&
2008 !Args.hasArg(options::OPT_nodefaultlibs)) {
2009 // FIXME: For some reason GCC passes -lgcc before adding
2010 // the default system libraries. Just mimic this for now.
2011 CmdArgs.push_back("-lgcc");
2012
2013 if (Args.hasArg(options::OPT_pthread))
2014 CmdArgs.push_back("-pthread");
2015 if (!Args.hasArg(options::OPT_shared))
2016 CmdArgs.push_back("-lc");
2017 CmdArgs.push_back("-lgcc");
2018 }
2019
2020 if (!Args.hasArg(options::OPT_nostdlib) &&
2021 !Args.hasArg(options::OPT_nostartfiles)) {
2022 if (!Args.hasArg(options::OPT_shared))
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002023 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtend.o")));
Edward O'Callaghane7925a02009-08-22 01:06:46 +00002024 else
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002025 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtendS.o")));
Edward O'Callaghane7925a02009-08-22 01:06:46 +00002026 }
2027
2028 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002029 Args.MakeArgString(getToolChain().GetProgramPath(C, "ld"));
Edward O'Callaghane7925a02009-08-22 01:06:46 +00002030 Dest.addCommand(new Command(JA, Exec, CmdArgs));
2031}
2032
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002033void openbsd::Assemble::ConstructJob(Compilation &C, const JobAction &JA,
2034 Job &Dest, const InputInfo &Output,
2035 const InputInfoList &Inputs,
2036 const ArgList &Args,
Mike Stump1eb44332009-09-09 15:08:12 +00002037 const char *LinkingOutput) const {
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002038 ArgStringList CmdArgs;
2039
2040 Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA,
2041 options::OPT_Xassembler);
2042
2043 CmdArgs.push_back("-o");
2044 if (Output.isPipe())
2045 CmdArgs.push_back("-");
2046 else
2047 CmdArgs.push_back(Output.getFilename());
2048
2049 for (InputInfoList::const_iterator
2050 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
2051 const InputInfo &II = *it;
2052 if (II.isPipe())
2053 CmdArgs.push_back("-");
2054 else
2055 CmdArgs.push_back(II.getFilename());
2056 }
2057
2058 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002059 Args.MakeArgString(getToolChain().GetProgramPath(C, "as"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00002060 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002061}
2062
2063void openbsd::Link::ConstructJob(Compilation &C, const JobAction &JA,
2064 Job &Dest, const InputInfo &Output,
2065 const InputInfoList &Inputs,
2066 const ArgList &Args,
2067 const char *LinkingOutput) const {
2068 const Driver &D = getToolChain().getHost().getDriver();
2069 ArgStringList CmdArgs;
2070
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002071 if ((!Args.hasArg(options::OPT_nostdlib)) &&
2072 (!Args.hasArg(options::OPT_shared))) {
2073 CmdArgs.push_back("-e");
2074 CmdArgs.push_back("__start");
2075 }
2076
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002077 if (Args.hasArg(options::OPT_static)) {
2078 CmdArgs.push_back("-Bstatic");
2079 } else {
2080 CmdArgs.push_back("--eh-frame-hdr");
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002081 CmdArgs.push_back("-Bdynamic");
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002082 if (Args.hasArg(options::OPT_shared)) {
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002083 CmdArgs.push_back("-shared");
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002084 } else {
2085 CmdArgs.push_back("-dynamic-linker");
2086 CmdArgs.push_back("/usr/libexec/ld.so");
2087 }
2088 }
2089
2090 if (Output.isPipe()) {
2091 CmdArgs.push_back("-o");
2092 CmdArgs.push_back("-");
2093 } else if (Output.isFilename()) {
2094 CmdArgs.push_back("-o");
2095 CmdArgs.push_back(Output.getFilename());
2096 } else {
2097 assert(Output.isNothing() && "Invalid output.");
2098 }
2099
2100 if (!Args.hasArg(options::OPT_nostdlib) &&
2101 !Args.hasArg(options::OPT_nostartfiles)) {
2102 if (!Args.hasArg(options::OPT_shared)) {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002103 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crt0.o")));
2104 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbegin.o")));
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002105 } else {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002106 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbeginS.o")));
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002107 }
2108 }
2109
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002110 CmdArgs.push_back(MakeFormattedString(Args,
2111 llvm::format("-L/usr/lib/gcc-lib/%s/3.3.5",
Daniel Dunbar88137642009-09-09 22:32:48 +00002112 getToolChain().getTripleString().c_str())));
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002113
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002114 Args.AddAllArgs(CmdArgs, options::OPT_L);
2115 Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
2116 Args.AddAllArgs(CmdArgs, options::OPT_e);
2117
2118 for (InputInfoList::const_iterator
2119 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
2120 const InputInfo &II = *it;
2121
2122 // Don't try to pass LLVM inputs to a generic gcc.
2123 if (II.getType() == types::TY_LLVMBC)
2124 D.Diag(clang::diag::err_drv_no_linker_llvm_support)
Daniel Dunbar88137642009-09-09 22:32:48 +00002125 << getToolChain().getTripleString();
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002126
2127 if (II.isPipe())
2128 CmdArgs.push_back("-");
2129 else if (II.isFilename())
2130 CmdArgs.push_back(II.getFilename());
2131 else
2132 II.getInputArg().renderAsInput(Args, CmdArgs);
2133 }
2134
2135 if (!Args.hasArg(options::OPT_nostdlib) &&
2136 !Args.hasArg(options::OPT_nodefaultlibs)) {
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002137 // FIXME: For some reason GCC passes -lgcc before adding
2138 // the default system libraries. Just mimic this for now.
2139 CmdArgs.push_back("-lgcc");
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002140
2141 if (Args.hasArg(options::OPT_pthread))
2142 CmdArgs.push_back("-pthread");
Daniel Dunbar2bbcf662009-08-03 01:28:59 +00002143 if (!Args.hasArg(options::OPT_shared))
2144 CmdArgs.push_back("-lc");
2145 CmdArgs.push_back("-lgcc");
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002146 }
2147
2148 if (!Args.hasArg(options::OPT_nostdlib) &&
2149 !Args.hasArg(options::OPT_nostartfiles)) {
2150 if (!Args.hasArg(options::OPT_shared))
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002151 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtend.o")));
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002152 else
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002153 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtendS.o")));
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002154 }
2155
2156 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002157 Args.MakeArgString(getToolChain().GetProgramPath(C, "ld"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00002158 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbarf7b8eec2009-06-29 20:52:51 +00002159}
Ed Schoutenc66a5a32009-04-02 19:13:12 +00002160
Daniel Dunbar68a31d42009-03-31 17:45:15 +00002161void freebsd::Assemble::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00002162 Job &Dest, const InputInfo &Output,
2163 const InputInfoList &Inputs,
2164 const ArgList &Args,
Mike Stump1eb44332009-09-09 15:08:12 +00002165 const char *LinkingOutput) const {
Daniel Dunbar68a31d42009-03-31 17:45:15 +00002166 ArgStringList CmdArgs;
2167
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002168 // When building 32-bit code on FreeBSD/amd64, we have to explicitly
2169 // instruct as in the base system to assemble 32-bit code.
2170 if (getToolChain().getArchName() == "i386")
Daniel Dunbar68a31d42009-03-31 17:45:15 +00002171 CmdArgs.push_back("--32");
2172
2173 Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA,
2174 options::OPT_Xassembler);
2175
2176 CmdArgs.push_back("-o");
2177 if (Output.isPipe())
2178 CmdArgs.push_back("-");
2179 else
2180 CmdArgs.push_back(Output.getFilename());
2181
2182 for (InputInfoList::const_iterator
2183 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
2184 const InputInfo &II = *it;
2185 if (II.isPipe())
2186 CmdArgs.push_back("-");
2187 else
2188 CmdArgs.push_back(II.getFilename());
2189 }
2190
Daniel Dunbarc21c4852009-04-08 23:54:23 +00002191 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002192 Args.MakeArgString(getToolChain().GetProgramPath(C, "as"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00002193 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar68a31d42009-03-31 17:45:15 +00002194}
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002195
2196void freebsd::Link::ConstructJob(Compilation &C, const JobAction &JA,
Daniel Dunbarc21c4852009-04-08 23:54:23 +00002197 Job &Dest, const InputInfo &Output,
2198 const InputInfoList &Inputs,
2199 const ArgList &Args,
Daniel Dunbara8304f62009-05-02 20:14:53 +00002200 const char *LinkingOutput) const {
2201 const Driver &D = getToolChain().getHost().getDriver();
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002202 ArgStringList CmdArgs;
2203
2204 if (Args.hasArg(options::OPT_static)) {
2205 CmdArgs.push_back("-Bstatic");
2206 } else {
2207 CmdArgs.push_back("--eh-frame-hdr");
2208 if (Args.hasArg(options::OPT_shared)) {
2209 CmdArgs.push_back("-Bshareable");
2210 } else {
2211 CmdArgs.push_back("-dynamic-linker");
2212 CmdArgs.push_back("/libexec/ld-elf.so.1");
2213 }
2214 }
2215
2216 // When building 32-bit code on FreeBSD/amd64, we have to explicitly
2217 // instruct ld in the base system to link 32-bit code.
2218 if (getToolChain().getArchName() == "i386") {
2219 CmdArgs.push_back("-m");
2220 CmdArgs.push_back("elf_i386_fbsd");
2221 }
2222
2223 if (Output.isPipe()) {
2224 CmdArgs.push_back("-o");
2225 CmdArgs.push_back("-");
2226 } else if (Output.isFilename()) {
2227 CmdArgs.push_back("-o");
2228 CmdArgs.push_back(Output.getFilename());
2229 } else {
2230 assert(Output.isNothing() && "Invalid output.");
2231 }
2232
2233 if (!Args.hasArg(options::OPT_nostdlib) &&
2234 !Args.hasArg(options::OPT_nostartfiles)) {
2235 if (!Args.hasArg(options::OPT_shared)) {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002236 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crt1.o")));
2237 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crti.o")));
2238 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbegin.o")));
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002239 } else {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002240 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crti.o")));
2241 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbeginS.o")));
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002242 }
2243 }
2244
2245 Args.AddAllArgs(CmdArgs, options::OPT_L);
2246 Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
2247 Args.AddAllArgs(CmdArgs, options::OPT_e);
2248
2249 for (InputInfoList::const_iterator
Daniel Dunbarc21c4852009-04-08 23:54:23 +00002250 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002251 const InputInfo &II = *it;
Daniel Dunbara8304f62009-05-02 20:14:53 +00002252
2253 // Don't try to pass LLVM inputs to a generic gcc.
2254 if (II.getType() == types::TY_LLVMBC)
2255 D.Diag(clang::diag::err_drv_no_linker_llvm_support)
Daniel Dunbar88137642009-09-09 22:32:48 +00002256 << getToolChain().getTripleString();
Daniel Dunbara8304f62009-05-02 20:14:53 +00002257
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002258 if (II.isPipe())
2259 CmdArgs.push_back("-");
2260 else if (II.isFilename())
2261 CmdArgs.push_back(II.getFilename());
2262 else
2263 II.getInputArg().renderAsInput(Args, CmdArgs);
2264 }
2265
2266 if (!Args.hasArg(options::OPT_nostdlib) &&
2267 !Args.hasArg(options::OPT_nodefaultlibs)) {
2268 // FIXME: For some reason GCC passes -lgcc and -lgcc_s before adding
2269 // the default system libraries. Just mimic this for now.
2270 CmdArgs.push_back("-lgcc");
2271 if (Args.hasArg(options::OPT_static)) {
2272 CmdArgs.push_back("-lgcc_eh");
2273 } else {
2274 CmdArgs.push_back("--as-needed");
2275 CmdArgs.push_back("-lgcc_s");
2276 CmdArgs.push_back("--no-as-needed");
2277 }
2278
2279 if (Args.hasArg(options::OPT_pthread))
2280 CmdArgs.push_back("-lpthread");
2281 CmdArgs.push_back("-lc");
2282
2283 CmdArgs.push_back("-lgcc");
2284 if (Args.hasArg(options::OPT_static)) {
2285 CmdArgs.push_back("-lgcc_eh");
2286 } else {
2287 CmdArgs.push_back("--as-needed");
2288 CmdArgs.push_back("-lgcc_s");
2289 CmdArgs.push_back("--no-as-needed");
2290 }
2291 }
2292
2293 if (!Args.hasArg(options::OPT_nostdlib) &&
2294 !Args.hasArg(options::OPT_nostartfiles)) {
2295 if (!Args.hasArg(options::OPT_shared))
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002296 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtend.o")));
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002297 else
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002298 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtendS.o")));
2299 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtn.o")));
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002300 }
2301
Daniel Dunbarc21c4852009-04-08 23:54:23 +00002302 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002303 Args.MakeArgString(getToolChain().GetProgramPath(C, "ld"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00002304 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar008f54a2009-04-01 19:36:32 +00002305}
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002306
2307/// DragonFly Tools
2308
2309// For now, DragonFly Assemble does just about the same as for
2310// FreeBSD, but this may change soon.
2311void dragonfly::Assemble::ConstructJob(Compilation &C, const JobAction &JA,
2312 Job &Dest, const InputInfo &Output,
2313 const InputInfoList &Inputs,
2314 const ArgList &Args,
2315 const char *LinkingOutput) const {
2316 ArgStringList CmdArgs;
2317
2318 // When building 32-bit code on DragonFly/pc64, we have to explicitly
2319 // instruct as in the base system to assemble 32-bit code.
2320 if (getToolChain().getArchName() == "i386")
2321 CmdArgs.push_back("--32");
2322
2323 Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA,
2324 options::OPT_Xassembler);
2325
2326 CmdArgs.push_back("-o");
2327 if (Output.isPipe())
2328 CmdArgs.push_back("-");
2329 else
2330 CmdArgs.push_back(Output.getFilename());
2331
2332 for (InputInfoList::const_iterator
2333 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
2334 const InputInfo &II = *it;
2335 if (II.isPipe())
2336 CmdArgs.push_back("-");
2337 else
2338 CmdArgs.push_back(II.getFilename());
2339 }
2340
2341 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002342 Args.MakeArgString(getToolChain().GetProgramPath(C, "as"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00002343 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002344}
2345
2346void dragonfly::Link::ConstructJob(Compilation &C, const JobAction &JA,
2347 Job &Dest, const InputInfo &Output,
2348 const InputInfoList &Inputs,
2349 const ArgList &Args,
2350 const char *LinkingOutput) const {
Daniel Dunbara8304f62009-05-02 20:14:53 +00002351 const Driver &D = getToolChain().getHost().getDriver();
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002352 ArgStringList CmdArgs;
2353
2354 if (Args.hasArg(options::OPT_static)) {
2355 CmdArgs.push_back("-Bstatic");
2356 } else {
2357 if (Args.hasArg(options::OPT_shared))
2358 CmdArgs.push_back("-Bshareable");
2359 else {
2360 CmdArgs.push_back("-dynamic-linker");
2361 CmdArgs.push_back("/usr/libexec/ld-elf.so.2");
2362 }
2363 }
2364
2365 // When building 32-bit code on DragonFly/pc64, we have to explicitly
2366 // instruct ld in the base system to link 32-bit code.
2367 if (getToolChain().getArchName() == "i386") {
2368 CmdArgs.push_back("-m");
2369 CmdArgs.push_back("elf_i386");
2370 }
2371
2372 if (Output.isPipe()) {
2373 CmdArgs.push_back("-o");
2374 CmdArgs.push_back("-");
2375 } else if (Output.isFilename()) {
2376 CmdArgs.push_back("-o");
2377 CmdArgs.push_back(Output.getFilename());
2378 } else {
2379 assert(Output.isNothing() && "Invalid output.");
2380 }
2381
2382 if (!Args.hasArg(options::OPT_nostdlib) &&
2383 !Args.hasArg(options::OPT_nostartfiles)) {
2384 if (!Args.hasArg(options::OPT_shared)) {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002385 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crt1.o")));
2386 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crti.o")));
2387 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbegin.o")));
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002388 } else {
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002389 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crti.o")));
2390 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtbeginS.o")));
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002391 }
2392 }
2393
2394 Args.AddAllArgs(CmdArgs, options::OPT_L);
2395 Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
2396 Args.AddAllArgs(CmdArgs, options::OPT_e);
2397
2398 for (InputInfoList::const_iterator
2399 it = Inputs.begin(), ie = Inputs.end(); it != ie; ++it) {
2400 const InputInfo &II = *it;
Daniel Dunbara8304f62009-05-02 20:14:53 +00002401
2402 // Don't try to pass LLVM inputs to a generic gcc.
2403 if (II.getType() == types::TY_LLVMBC)
2404 D.Diag(clang::diag::err_drv_no_linker_llvm_support)
Daniel Dunbar88137642009-09-09 22:32:48 +00002405 << getToolChain().getTripleString();
Daniel Dunbara8304f62009-05-02 20:14:53 +00002406
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002407 if (II.isPipe())
2408 CmdArgs.push_back("-");
2409 else if (II.isFilename())
2410 CmdArgs.push_back(II.getFilename());
2411 else
2412 II.getInputArg().renderAsInput(Args, CmdArgs);
2413 }
2414
2415 if (!Args.hasArg(options::OPT_nostdlib) &&
2416 !Args.hasArg(options::OPT_nodefaultlibs)) {
2417 // FIXME: GCC passes on -lgcc, -lgcc_pic and a whole lot of
2418 // rpaths
2419 CmdArgs.push_back("-L/usr/lib/gcc41");
2420
2421 if (!Args.hasArg(options::OPT_static)) {
2422 CmdArgs.push_back("-rpath");
2423 CmdArgs.push_back("/usr/lib/gcc41");
2424
2425 CmdArgs.push_back("-rpath-link");
2426 CmdArgs.push_back("/usr/lib/gcc41");
2427
2428 CmdArgs.push_back("-rpath");
2429 CmdArgs.push_back("/usr/lib");
2430
2431 CmdArgs.push_back("-rpath-link");
2432 CmdArgs.push_back("/usr/lib");
2433 }
2434
2435 if (Args.hasArg(options::OPT_shared)) {
2436 CmdArgs.push_back("-lgcc_pic");
2437 } else {
2438 CmdArgs.push_back("-lgcc");
2439 }
2440
2441
2442 if (Args.hasArg(options::OPT_pthread))
2443 CmdArgs.push_back("-lthread_xu");
2444
2445 if (!Args.hasArg(options::OPT_nolibc)) {
2446 CmdArgs.push_back("-lc");
2447 }
2448
2449 if (Args.hasArg(options::OPT_shared)) {
2450 CmdArgs.push_back("-lgcc_pic");
2451 } else {
2452 CmdArgs.push_back("-lgcc");
2453 }
2454 }
2455
2456 if (!Args.hasArg(options::OPT_nostdlib) &&
2457 !Args.hasArg(options::OPT_nostartfiles)) {
2458 if (!Args.hasArg(options::OPT_shared))
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002459 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtend.o")));
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002460 else
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002461 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtendS.o")));
2462 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath(C, "crtn.o")));
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002463 }
2464
2465 const char *Exec =
Daniel Dunbar5ed34f42009-09-09 22:33:00 +00002466 Args.MakeArgString(getToolChain().GetProgramPath(C, "ld"));
Daniel Dunbarcae087e2009-07-01 19:02:28 +00002467 Dest.addCommand(new Command(JA, Exec, CmdArgs));
Daniel Dunbar11e1b402009-05-02 18:28:39 +00002468}