New files
[official-gcc.git] / gcc / d / d-spec.cc
blobf4744763ab617e3adf762ca0a791a512597b0fa8
1 /* d-spec.c -- Specific flags and argument handling of the D front end.
2 Copyright (C) 2006-2020 Free Software Foundation, Inc.
4 GCC is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 3, or (at your option)
7 any later version.
9 GCC is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with GCC; see the file COPYING3. If not see
16 <http://www.gnu.org/licenses/>. */
18 #include "config.h"
19 #include "system.h"
20 #include "coretypes.h"
21 #include "opt-suggestions.h"
22 #include "gcc.h"
23 #include "tm.h"
24 #include "opts.h"
26 /* This bit is set if the arguments is a D source file. */
27 #define DSOURCE (1<<1)
28 /* This bit is set if they did `-lstdc++'. */
29 #define WITHLIBCXX (1<<2)
30 /* Skip this option. */
31 #define SKIPOPT (1<<3)
33 #ifndef LIBSTDCXX
34 #define LIBSTDCXX "stdc++"
35 #endif
36 #ifndef LIBSTDCXX_PROFILE
37 #define LIBSTDCXX_PROFILE LIBSTDCXX
38 #endif
40 #ifndef LIBPHOBOS
41 #define LIBPHOBOS "gphobos"
42 #endif
43 #ifndef LIBPHOBOS_PROFILE
44 #define LIBPHOBOS_PROFILE LIBPHOBOS
45 #endif
47 /* What do with libgphobos. */
48 enum phobos_action
50 /* libgphobos should not be linked in. */
51 PHOBOS_NOLINK = -1,
52 /* libgphobos should be linked in if it is needed. */
53 PHOBOS_DEFAULT = 0,
54 /* libgphobos is needed and should be linked in. */
55 PHOBOS_LINK,
56 /* libgphobos is needed and should be linked statically. */
57 PHOBOS_STATIC,
58 /* libgphobos is needed and should be linked dynamically. */
59 PHOBOS_DYNAMIC,
62 static phobos_action phobos_library = PHOBOS_DEFAULT;
64 /* If true, do load libgphobos.spec even if not needed otherwise. */
65 static bool need_spec = false;
67 void
68 lang_specific_driver (cl_decoded_option **in_decoded_options,
69 unsigned int *in_decoded_options_count,
70 int *in_added_libraries)
72 unsigned int i, j;
74 /* If nonzero, the user gave us the `-p' or `-pg' flag. */
75 int saw_profile_flag = 0;
77 /* If true, the user gave `-g'. Used by -debuglib. */
78 bool saw_debug_flag = false;
80 /* The new argument list will be contained in this. */
81 cl_decoded_option *new_decoded_options;
83 /* "-lstdc++" if it appears on the command line. */
84 const cl_decoded_option *saw_libcxx = 0;
86 /* Whether we need the C++ STD library. */
87 bool need_stdcxx = false;
89 /* True if we saw -static. */
90 bool static_link = false;
92 /* True if we should add -shared-libgcc to the command-line. */
93 bool shared_libgcc = true;
95 /* What default library to use instead of phobos. */
96 const char *defaultlib = NULL;
98 /* What debug library to use instead of phobos. */
99 const char *debuglib = NULL;
101 /* The total number of arguments with the new stuff. */
102 unsigned int num_args = 1;
104 /* "-fonly" if it appears on the command line. */
105 const char *only_source_option = 0;
107 /* Whether the -o option was used. */
108 bool saw_opt_o = false;
110 /* Whether the -c option was used. Also used for -E, -fsyntax-only,
111 in general anything which implies only compilation and not linking. */
112 bool saw_opt_c = false;
114 /* Whether the -S option was used. */
115 bool saw_opt_S = false;
117 /* The first input file with an extension of .d. */
118 const char *first_d_file = NULL;
120 /* The total number of arguments with the new stuff. */
121 unsigned int argc = *in_decoded_options_count;
123 /* The argument list. */
124 cl_decoded_option *decoded_options = *in_decoded_options;
126 /* The number of libraries added in. */
127 int added_libraries = *in_added_libraries;
129 /* An array used to flag each argument that needs a bit set for
130 DSOURCE, MATHLIB, WITHTHREAD, WITHLIBC or WITHLIBCXX. */
131 int *args = XCNEWVEC (int, argc);
133 for (i = 1; i < argc; i++)
135 const char *arg = decoded_options[i].arg;
136 const int value = decoded_options[i].value;
138 switch (decoded_options[i].opt_index)
140 case OPT_dstartfiles:
141 need_spec = true;
142 break;
144 case OPT_nostdlib:
145 case OPT_nodefaultlibs:
146 phobos_library = PHOBOS_NOLINK;
147 break;
149 case OPT_nophoboslib:
150 phobos_library = PHOBOS_NOLINK;
151 args[i] |= SKIPOPT;
152 break;
154 case OPT_fdruntime:
155 if (!value)
156 phobos_library = PHOBOS_NOLINK;
157 else
158 phobos_library = PHOBOS_LINK;
159 break;
161 case OPT_defaultlib_:
162 if (defaultlib != NULL)
163 free (CONST_CAST (char *, defaultlib));
164 if (arg != NULL)
166 args[i] |= SKIPOPT;
167 defaultlib = XNEWVEC (char, strlen (arg));
168 strcpy (CONST_CAST (char *, defaultlib), arg);
170 break;
172 case OPT_debuglib_:
173 if (debuglib != NULL)
174 free (CONST_CAST (char *, debuglib));
175 if (arg != NULL)
177 args[i] |= SKIPOPT;
178 debuglib = XNEWVEC (char, strlen (arg));
179 strcpy (CONST_CAST (char *, debuglib), arg);
181 break;
183 case OPT_l:
184 if ((strcmp (arg, LIBSTDCXX) == 0)
185 || (strcmp (arg, LIBSTDCXX_PROFILE) == 0))
187 args[i] |= WITHLIBCXX;
188 need_stdcxx = false;
190 /* Unrecognized libraries (e.g. -ltango) may require libphobos. */
191 else if (phobos_library == PHOBOS_DEFAULT)
192 phobos_library = PHOBOS_LINK;
193 break;
195 case OPT_pg:
196 case OPT_p:
197 saw_profile_flag++;
198 break;
200 case OPT_g:
201 saw_debug_flag = true;
202 break;
204 case OPT_v:
205 /* If they only gave us `-v', don't try to link in libphobos. */
206 if (argc == 2)
207 phobos_library = PHOBOS_NOLINK;
208 break;
210 case OPT_x:
211 if (phobos_library == PHOBOS_DEFAULT && (strcmp (arg, "d") == 0))
212 phobos_library = PHOBOS_LINK;
213 break;
215 case OPT_Xlinker:
216 case OPT_Wl_:
217 /* Arguments that go directly to the linker might be .o files
218 or something, and so might cause libphobos to be needed. */
219 if (phobos_library == PHOBOS_DEFAULT)
220 phobos_library = PHOBOS_LINK;
221 break;
223 case OPT_c:
224 case OPT_E:
225 case OPT_M:
226 case OPT_MM:
227 case OPT_fsyntax_only:
228 /* Don't specify libaries if we won't link, since that would
229 cause a warning. */
230 saw_opt_c = true;
231 phobos_library = PHOBOS_NOLINK;
232 break;
234 case OPT_S:
235 saw_opt_S = true;
236 phobos_library = PHOBOS_NOLINK;
237 break;
239 case OPT_o:
240 saw_opt_o = true;
241 break;
243 case OPT_static:
244 static_link = true;
245 break;
247 case OPT_static_libgcc:
248 shared_libgcc = false;
249 break;
251 case OPT_static_libphobos:
252 if (phobos_library != PHOBOS_NOLINK)
253 phobos_library = PHOBOS_STATIC;
254 args[i] |= SKIPOPT;
255 break;
257 case OPT_shared_libphobos:
258 if (phobos_library != PHOBOS_NOLINK)
259 phobos_library = PHOBOS_DYNAMIC;
260 args[i] |= SKIPOPT;
261 break;
263 case OPT_fonly_:
264 args[i] |= SKIPOPT;
265 only_source_option = decoded_options[i].orig_option_with_args_text;
267 if (arg != NULL)
269 const char *suffix = strrchr (only_source_option, '.');
270 if (suffix == NULL || strcmp (suffix, ".d") != 0)
271 only_source_option = concat (only_source_option, ".d", NULL);
273 break;
275 case OPT_SPECIAL_input_file:
277 if (arg[0] == '\0' || arg[1] == '\0')
278 continue;
280 if (phobos_library == PHOBOS_DEFAULT)
281 phobos_library = PHOBOS_LINK;
283 /* Record that this is a D source file. */
284 const char *suffix = strrchr (arg, '.');
285 if (suffix != NULL && strcmp (suffix, ".d") == 0)
287 if (first_d_file == NULL)
288 first_d_file = arg;
290 args[i] |= DSOURCE;
293 /* If this is a C++ source file, we'll need to link
294 against libstdc++ library. */
295 if (suffix != NULL
296 && (strcmp (suffix, ".cc") == 0
297 || (strcmp (suffix, ".cpp") == 0)
298 || (strcmp (suffix, ".c++") == 0)))
299 need_stdcxx = true;
301 break;
306 /* There's no point adding -shared-libgcc if we don't have a shared
307 libgcc. */
308 #ifndef ENABLE_SHARED_LIBGCC
309 shared_libgcc = false;
310 #endif
312 /* Make sure to have room for the trailing NULL argument.
313 - need_stdcxx might add `-lstdcxx'
314 - libphobos adds `-Bstatic -lphobos -Bdynamic'
315 - only_source adds 1 more arg, also maybe add `-o'. */
316 num_args = argc + need_stdcxx + shared_libgcc
317 + (phobos_library != PHOBOS_NOLINK) * 4 + 2;
318 new_decoded_options = XNEWVEC (cl_decoded_option, num_args);
320 i = 0;
321 j = 0;
323 /* Copy the 0th argument, i.e., the name of the program itself. */
324 new_decoded_options[j++] = decoded_options[i++];
326 /* NOTE: We start at 1 now, not 0. */
327 while (i < argc)
329 if (args[i] & SKIPOPT)
331 ++i;
332 continue;
335 new_decoded_options[j] = decoded_options[i];
337 if (!saw_libcxx && (args[i] & WITHLIBCXX))
339 --j;
340 saw_libcxx = &decoded_options[i];
343 if (args[i] & DSOURCE)
345 if (only_source_option)
346 --j;
349 i++;
350 j++;
353 if (only_source_option)
355 const char *only_source_arg = only_source_option + 7;
356 generate_option (OPT_fonly_, only_source_arg, 1, CL_DRIVER,
357 &new_decoded_options[j]);
358 j++;
360 generate_option_input_file (only_source_arg,
361 &new_decoded_options[j++]);
364 /* If no reason to link against libphobos library, then don't add it. */
365 if (phobos_library == PHOBOS_DEFAULT)
366 phobos_library = PHOBOS_NOLINK;
368 /* If we didn't see a -o option, add one. This is because we need the
369 driver to pass all .d files to the D compiler. Without a -o option
370 the driver will invoke the compiler separately for each input file. */
371 if (first_d_file != NULL && !saw_opt_o)
373 if (saw_opt_c || saw_opt_S)
375 const char *base = lbasename (first_d_file);
376 int baselen = strlen (base) - 2;
377 char *out = XNEWVEC (char, baselen + 3);
379 memcpy (out, base, baselen);
380 /* The driver will convert .o to some other suffix if appropriate. */
381 out[baselen] = '.';
382 if (saw_opt_S)
383 out[baselen + 1] = 's';
384 else
385 out[baselen + 1] = 'o';
386 out[baselen + 2] = '\0';
387 generate_option (OPT_o, out, 1, CL_DRIVER,
388 &new_decoded_options[j]);
390 else
392 /* Wouldn't be necessary if the driver converted .out also. */
393 const char *out = NULL;
395 #ifdef TARGET_EXECUTABLE_SUFFIX
396 if (TARGET_EXECUTABLE_SUFFIX[0] != 0)
397 out = "a" TARGET_EXECUTABLE_SUFFIX;
398 #endif
399 if (out == NULL)
400 out = "a.out";
402 generate_option (OPT_o, out, 1, CL_DRIVER,
403 &new_decoded_options[j]);
405 j++;
408 /* Add `-lgphobos' if we haven't already done so. */
409 if (phobos_library != PHOBOS_NOLINK)
411 /* Default to static linking. */
412 if (phobos_library != PHOBOS_DYNAMIC)
413 phobos_library = PHOBOS_STATIC;
415 #ifdef HAVE_LD_STATIC_DYNAMIC
416 if (phobos_library == PHOBOS_STATIC && !static_link)
418 generate_option (OPT_Wl_, LD_STATIC_OPTION, 1, CL_DRIVER,
419 &new_decoded_options[j++]);
421 #endif
422 /* Order of precedence in determining what library to link against is:
423 - `-l<lib>' from `-debuglib=<lib>' if `-g' was also seen.
424 - `-l<lib>' from `-defaultlib=<lib>'.
425 - `-lgphobos' unless `-nophoboslib' or `-fno-druntime' was seen. */
426 if (debuglib && saw_debug_flag)
428 generate_option (OPT_l, debuglib, 1, CL_DRIVER,
429 &new_decoded_options[j++]);
430 added_libraries++;
432 else if (defaultlib)
434 generate_option (OPT_l, defaultlib, 1, CL_DRIVER,
435 &new_decoded_options[j++]);
436 added_libraries++;
438 else
440 generate_option (OPT_l,
441 saw_profile_flag ? LIBPHOBOS_PROFILE : LIBPHOBOS, 1,
442 CL_DRIVER, &new_decoded_options[j++]);
443 added_libraries++;
446 #ifdef HAVE_LD_STATIC_DYNAMIC
447 if (phobos_library == PHOBOS_STATIC && !static_link)
449 generate_option (OPT_Wl_, LD_DYNAMIC_OPTION, 1, CL_DRIVER,
450 &new_decoded_options[j++]);
452 #endif
455 if (saw_libcxx)
456 new_decoded_options[j++] = *saw_libcxx;
457 else if (need_stdcxx)
459 generate_option (OPT_l,
460 (saw_profile_flag
461 ? LIBSTDCXX_PROFILE
462 : LIBSTDCXX),
463 1, CL_DRIVER, &new_decoded_options[j++]);
464 added_libraries++;
467 if (shared_libgcc && !static_link)
469 generate_option (OPT_shared_libgcc, NULL, 1, CL_DRIVER,
470 &new_decoded_options[j++]);
473 *in_decoded_options_count = j;
474 *in_decoded_options = new_decoded_options;
475 *in_added_libraries = added_libraries;
478 /* Called before linking. Returns 0 on success and -1 on failure. */
481 lang_specific_pre_link (void)
483 if ((phobos_library != PHOBOS_NOLINK) || need_spec)
484 do_spec ("%:include(libgphobos.spec)");
486 return 0;
489 /* Number of extra output files that lang_specific_pre_link may generate. */
491 int lang_specific_extra_outfiles = 0; /* Not used for D. */