]> git.ipfire.org Git - thirdparty/gcc.git/blob - gcc/opts.c
driver: fix a problem with implementation of -falign-foo=0 [PR96247]
[thirdparty/gcc.git] / gcc / opts.c
1 /* Command line option handling.
2 Copyright (C) 2002-2020 Free Software Foundation, Inc.
3 Contributed by Neil Booth.
4
5 This file is part of GCC.
6
7 GCC is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 3, or (at your option) any later
10 version.
11
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
15 for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with GCC; see the file COPYING3. If not see
19 <http://www.gnu.org/licenses/>. */
20
21 #include "config.h"
22 #include "system.h"
23 #include "intl.h"
24 #include "coretypes.h"
25 #include "opts.h"
26 #include "tm.h"
27 #include "flags.h"
28 #include "diagnostic.h"
29 #include "opts-diagnostic.h"
30 #include "insn-attr-common.h"
31 #include "common/common-target.h"
32 #include "spellcheck.h"
33 #include "opt-suggestions.h"
34 #include "diagnostic-color.h"
35 #include "selftest.h"
36
37 static void set_Wstrict_aliasing (struct gcc_options *opts, int onoff);
38
39 /* Indexed by enum debug_info_type. */
40 const char *const debug_type_names[] =
41 {
42 "none", "stabs", "dwarf-2", "xcoff", "vms"
43 };
44
45 /* Parse the -femit-struct-debug-detailed option value
46 and set the flag variables. */
47
48 #define MATCH( prefix, string ) \
49 ((strncmp (prefix, string, sizeof prefix - 1) == 0) \
50 ? ((string += sizeof prefix - 1), 1) : 0)
51
52 void
53 set_struct_debug_option (struct gcc_options *opts, location_t loc,
54 const char *spec)
55 {
56 /* various labels for comparison */
57 static const char dfn_lbl[] = "dfn:", dir_lbl[] = "dir:", ind_lbl[] = "ind:";
58 static const char ord_lbl[] = "ord:", gen_lbl[] = "gen:";
59 static const char none_lbl[] = "none", any_lbl[] = "any";
60 static const char base_lbl[] = "base", sys_lbl[] = "sys";
61
62 enum debug_struct_file files = DINFO_STRUCT_FILE_ANY;
63 /* Default is to apply to as much as possible. */
64 enum debug_info_usage usage = DINFO_USAGE_NUM_ENUMS;
65 int ord = 1, gen = 1;
66
67 /* What usage? */
68 if (MATCH (dfn_lbl, spec))
69 usage = DINFO_USAGE_DFN;
70 else if (MATCH (dir_lbl, spec))
71 usage = DINFO_USAGE_DIR_USE;
72 else if (MATCH (ind_lbl, spec))
73 usage = DINFO_USAGE_IND_USE;
74
75 /* Generics or not? */
76 if (MATCH (ord_lbl, spec))
77 gen = 0;
78 else if (MATCH (gen_lbl, spec))
79 ord = 0;
80
81 /* What allowable environment? */
82 if (MATCH (none_lbl, spec))
83 files = DINFO_STRUCT_FILE_NONE;
84 else if (MATCH (any_lbl, spec))
85 files = DINFO_STRUCT_FILE_ANY;
86 else if (MATCH (sys_lbl, spec))
87 files = DINFO_STRUCT_FILE_SYS;
88 else if (MATCH (base_lbl, spec))
89 files = DINFO_STRUCT_FILE_BASE;
90 else
91 error_at (loc,
92 "argument %qs to %<-femit-struct-debug-detailed%> "
93 "not recognized",
94 spec);
95
96 /* Effect the specification. */
97 if (usage == DINFO_USAGE_NUM_ENUMS)
98 {
99 if (ord)
100 {
101 opts->x_debug_struct_ordinary[DINFO_USAGE_DFN] = files;
102 opts->x_debug_struct_ordinary[DINFO_USAGE_DIR_USE] = files;
103 opts->x_debug_struct_ordinary[DINFO_USAGE_IND_USE] = files;
104 }
105 if (gen)
106 {
107 opts->x_debug_struct_generic[DINFO_USAGE_DFN] = files;
108 opts->x_debug_struct_generic[DINFO_USAGE_DIR_USE] = files;
109 opts->x_debug_struct_generic[DINFO_USAGE_IND_USE] = files;
110 }
111 }
112 else
113 {
114 if (ord)
115 opts->x_debug_struct_ordinary[usage] = files;
116 if (gen)
117 opts->x_debug_struct_generic[usage] = files;
118 }
119
120 if (*spec == ',')
121 set_struct_debug_option (opts, loc, spec+1);
122 else
123 {
124 /* No more -femit-struct-debug-detailed specifications.
125 Do final checks. */
126 if (*spec != '\0')
127 error_at (loc,
128 "argument %qs to %<-femit-struct-debug-detailed%> unknown",
129 spec);
130 if (opts->x_debug_struct_ordinary[DINFO_USAGE_DIR_USE]
131 < opts->x_debug_struct_ordinary[DINFO_USAGE_IND_USE]
132 || opts->x_debug_struct_generic[DINFO_USAGE_DIR_USE]
133 < opts->x_debug_struct_generic[DINFO_USAGE_IND_USE])
134 error_at (loc,
135 "%<-femit-struct-debug-detailed=dir:...%> must allow "
136 "at least as much as "
137 "%<-femit-struct-debug-detailed=ind:...%>");
138 }
139 }
140
141 /* Strip off a legitimate source ending from the input string NAME of
142 length LEN. Rather than having to know the names used by all of
143 our front ends, we strip off an ending of a period followed by
144 up to fource characters. (C++ uses ".cpp".) */
145
146 void
147 strip_off_ending (char *name, int len)
148 {
149 int i;
150 for (i = 2; i < 5 && len > i; i++)
151 {
152 if (name[len - i] == '.')
153 {
154 name[len - i] = '\0';
155 break;
156 }
157 }
158 }
159
160 /* Find the base name of a path, stripping off both directories and
161 a single final extension. */
162 int
163 base_of_path (const char *path, const char **base_out)
164 {
165 const char *base = path;
166 const char *dot = 0;
167 const char *p = path;
168 char c = *p;
169 while (c)
170 {
171 if (IS_DIR_SEPARATOR (c))
172 {
173 base = p + 1;
174 dot = 0;
175 }
176 else if (c == '.')
177 dot = p;
178 c = *++p;
179 }
180 if (!dot)
181 dot = p;
182 *base_out = base;
183 return dot - base;
184 }
185
186 /* What to print when a switch has no documentation. */
187 static const char undocumented_msg[] = N_("This option lacks documentation.");
188 static const char use_diagnosed_msg[] = N_("Uses of this option are diagnosed.");
189
190 typedef char *char_p; /* For DEF_VEC_P. */
191
192 static void set_debug_level (enum debug_info_type type, int extended,
193 const char *arg, struct gcc_options *opts,
194 struct gcc_options *opts_set,
195 location_t loc);
196 static void set_fast_math_flags (struct gcc_options *opts, int set);
197 static void decode_d_option (const char *arg, struct gcc_options *opts,
198 location_t loc, diagnostic_context *dc);
199 static void set_unsafe_math_optimizations_flags (struct gcc_options *opts,
200 int set);
201 static void enable_warning_as_error (const char *arg, int value,
202 unsigned int lang_mask,
203 const struct cl_option_handlers *handlers,
204 struct gcc_options *opts,
205 struct gcc_options *opts_set,
206 location_t loc,
207 diagnostic_context *dc);
208
209 /* Handle a back-end option; arguments and return value as for
210 handle_option. */
211
212 bool
213 target_handle_option (struct gcc_options *opts,
214 struct gcc_options *opts_set,
215 const struct cl_decoded_option *decoded,
216 unsigned int lang_mask ATTRIBUTE_UNUSED, int kind,
217 location_t loc,
218 const struct cl_option_handlers *handlers ATTRIBUTE_UNUSED,
219 diagnostic_context *dc, void (*) (void))
220 {
221 gcc_assert (dc == global_dc);
222 gcc_assert (kind == DK_UNSPECIFIED);
223 return targetm_common.handle_option (opts, opts_set, decoded, loc);
224 }
225
226 /* Add comma-separated strings to a char_p vector. */
227
228 static void
229 add_comma_separated_to_vector (void **pvec, const char *arg)
230 {
231 char *tmp;
232 char *r;
233 char *w;
234 char *token_start;
235 vec<char_p> *v = (vec<char_p> *) *pvec;
236
237 vec_check_alloc (v, 1);
238
239 /* We never free this string. */
240 tmp = xstrdup (arg);
241
242 r = tmp;
243 w = tmp;
244 token_start = tmp;
245
246 while (*r != '\0')
247 {
248 if (*r == ',')
249 {
250 *w++ = '\0';
251 ++r;
252 v->safe_push (token_start);
253 token_start = w;
254 }
255 if (*r == '\\' && r[1] == ',')
256 {
257 *w++ = ',';
258 r += 2;
259 }
260 else
261 *w++ = *r++;
262 }
263
264 *w = '\0';
265 if (*token_start != '\0')
266 v->safe_push (token_start);
267
268 *pvec = v;
269 }
270
271 /* Initialize opts_obstack. */
272
273 void
274 init_opts_obstack (void)
275 {
276 gcc_obstack_init (&opts_obstack);
277 }
278
279 /* Initialize OPTS and OPTS_SET before using them in parsing options. */
280
281 void
282 init_options_struct (struct gcc_options *opts, struct gcc_options *opts_set)
283 {
284 /* Ensure that opts_obstack has already been initialized by the time
285 that we initialize any gcc_options instances (PR jit/68446). */
286 gcc_assert (opts_obstack.chunk_size > 0);
287
288 *opts = global_options_init;
289
290 if (opts_set)
291 memset (opts_set, 0, sizeof (*opts_set));
292
293 /* Initialize whether `char' is signed. */
294 opts->x_flag_signed_char = DEFAULT_SIGNED_CHAR;
295 /* Set this to a special "uninitialized" value. The actual default
296 is set after target options have been processed. */
297 opts->x_flag_short_enums = 2;
298
299 /* Initialize target_flags before default_options_optimization
300 so the latter can modify it. */
301 opts->x_target_flags = targetm_common.default_target_flags;
302
303 /* Some targets have ABI-specified unwind tables. */
304 opts->x_flag_unwind_tables = targetm_common.unwind_tables_default;
305
306 /* Some targets have other target-specific initialization. */
307 targetm_common.option_init_struct (opts);
308 }
309
310 /* If indicated by the optimization level LEVEL (-Os if SIZE is set,
311 -Ofast if FAST is set, -Og if DEBUG is set), apply the option DEFAULT_OPT
312 to OPTS and OPTS_SET, diagnostic context DC, location LOC, with language
313 mask LANG_MASK and option handlers HANDLERS. */
314
315 static void
316 maybe_default_option (struct gcc_options *opts,
317 struct gcc_options *opts_set,
318 const struct default_options *default_opt,
319 int level, bool size, bool fast, bool debug,
320 unsigned int lang_mask,
321 const struct cl_option_handlers *handlers,
322 location_t loc,
323 diagnostic_context *dc)
324 {
325 const struct cl_option *option = &cl_options[default_opt->opt_index];
326 bool enabled;
327
328 if (size)
329 gcc_assert (level == 2);
330 if (fast)
331 gcc_assert (level == 3);
332 if (debug)
333 gcc_assert (level == 1);
334
335 switch (default_opt->levels)
336 {
337 case OPT_LEVELS_ALL:
338 enabled = true;
339 break;
340
341 case OPT_LEVELS_0_ONLY:
342 enabled = (level == 0);
343 break;
344
345 case OPT_LEVELS_1_PLUS:
346 enabled = (level >= 1);
347 break;
348
349 case OPT_LEVELS_1_PLUS_SPEED_ONLY:
350 enabled = (level >= 1 && !size && !debug);
351 break;
352
353 case OPT_LEVELS_1_PLUS_NOT_DEBUG:
354 enabled = (level >= 1 && !debug);
355 break;
356
357 case OPT_LEVELS_2_PLUS:
358 enabled = (level >= 2);
359 break;
360
361 case OPT_LEVELS_2_PLUS_SPEED_ONLY:
362 enabled = (level >= 2 && !size && !debug);
363 break;
364
365 case OPT_LEVELS_3_PLUS:
366 enabled = (level >= 3);
367 break;
368
369 case OPT_LEVELS_3_PLUS_AND_SIZE:
370 enabled = (level >= 3 || size);
371 break;
372
373 case OPT_LEVELS_SIZE:
374 enabled = size;
375 break;
376
377 case OPT_LEVELS_FAST:
378 enabled = fast;
379 break;
380
381 case OPT_LEVELS_NONE:
382 default:
383 gcc_unreachable ();
384 }
385
386 if (enabled)
387 handle_generated_option (opts, opts_set, default_opt->opt_index,
388 default_opt->arg, default_opt->value,
389 lang_mask, DK_UNSPECIFIED, loc,
390 handlers, true, dc);
391 else if (default_opt->arg == NULL
392 && !option->cl_reject_negative
393 && !(option->flags & CL_PARAMS))
394 handle_generated_option (opts, opts_set, default_opt->opt_index,
395 default_opt->arg, !default_opt->value,
396 lang_mask, DK_UNSPECIFIED, loc,
397 handlers, true, dc);
398 }
399
400 /* As indicated by the optimization level LEVEL (-Os if SIZE is set,
401 -Ofast if FAST is set), apply the options in array DEFAULT_OPTS to
402 OPTS and OPTS_SET, diagnostic context DC, location LOC, with
403 language mask LANG_MASK and option handlers HANDLERS. */
404
405 static void
406 maybe_default_options (struct gcc_options *opts,
407 struct gcc_options *opts_set,
408 const struct default_options *default_opts,
409 int level, bool size, bool fast, bool debug,
410 unsigned int lang_mask,
411 const struct cl_option_handlers *handlers,
412 location_t loc,
413 diagnostic_context *dc)
414 {
415 size_t i;
416
417 for (i = 0; default_opts[i].levels != OPT_LEVELS_NONE; i++)
418 maybe_default_option (opts, opts_set, &default_opts[i],
419 level, size, fast, debug,
420 lang_mask, handlers, loc, dc);
421 }
422
423 /* Table of options enabled by default at different levels.
424 Please keep this list sorted by level and alphabetized within
425 each level; this makes it easier to keep the documentation
426 in sync. */
427
428 static const struct default_options default_options_table[] =
429 {
430 /* -O1 and -Og optimizations. */
431 { OPT_LEVELS_1_PLUS, OPT_fcombine_stack_adjustments, NULL, 1 },
432 { OPT_LEVELS_1_PLUS, OPT_fcompare_elim, NULL, 1 },
433 { OPT_LEVELS_1_PLUS, OPT_fcprop_registers, NULL, 1 },
434 { OPT_LEVELS_1_PLUS, OPT_fdefer_pop, NULL, 1 },
435 { OPT_LEVELS_1_PLUS, OPT_fforward_propagate, NULL, 1 },
436 { OPT_LEVELS_1_PLUS, OPT_fguess_branch_probability, NULL, 1 },
437 { OPT_LEVELS_1_PLUS, OPT_fipa_profile, NULL, 1 },
438 { OPT_LEVELS_1_PLUS, OPT_fipa_pure_const, NULL, 1 },
439 { OPT_LEVELS_1_PLUS, OPT_fipa_reference, NULL, 1 },
440 { OPT_LEVELS_1_PLUS, OPT_fipa_reference_addressable, NULL, 1 },
441 { OPT_LEVELS_1_PLUS, OPT_fmerge_constants, NULL, 1 },
442 { OPT_LEVELS_1_PLUS, OPT_fomit_frame_pointer, NULL, 1 },
443 { OPT_LEVELS_1_PLUS, OPT_freorder_blocks, NULL, 1 },
444 { OPT_LEVELS_1_PLUS, OPT_fshrink_wrap, NULL, 1 },
445 { OPT_LEVELS_1_PLUS, OPT_fsplit_wide_types, NULL, 1 },
446 { OPT_LEVELS_1_PLUS, OPT_ftree_builtin_call_dce, NULL, 1 },
447 { OPT_LEVELS_1_PLUS, OPT_ftree_ccp, NULL, 1 },
448 { OPT_LEVELS_1_PLUS, OPT_ftree_ch, NULL, 1 },
449 { OPT_LEVELS_1_PLUS, OPT_ftree_coalesce_vars, NULL, 1 },
450 { OPT_LEVELS_1_PLUS, OPT_ftree_copy_prop, NULL, 1 },
451 { OPT_LEVELS_1_PLUS, OPT_ftree_dce, NULL, 1 },
452 { OPT_LEVELS_1_PLUS, OPT_ftree_dominator_opts, NULL, 1 },
453 { OPT_LEVELS_1_PLUS, OPT_ftree_fre, NULL, 1 },
454 { OPT_LEVELS_1_PLUS, OPT_ftree_sink, NULL, 1 },
455 { OPT_LEVELS_1_PLUS, OPT_ftree_slsr, NULL, 1 },
456 { OPT_LEVELS_1_PLUS, OPT_ftree_ter, NULL, 1 },
457
458 /* -O1 (and not -Og) optimizations. */
459 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fbranch_count_reg, NULL, 1 },
460 #if DELAY_SLOTS
461 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fdelayed_branch, NULL, 1 },
462 #endif
463 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fdse, NULL, 1 },
464 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fif_conversion, NULL, 1 },
465 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fif_conversion2, NULL, 1 },
466 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_finline_functions_called_once, NULL, 1 },
467 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fmove_loop_invariants, NULL, 1 },
468 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_fssa_phiopt, NULL, 1 },
469 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_ftree_bit_ccp, NULL, 1 },
470 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_ftree_dse, NULL, 1 },
471 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_ftree_pta, NULL, 1 },
472 { OPT_LEVELS_1_PLUS_NOT_DEBUG, OPT_ftree_sra, NULL, 1 },
473
474 /* -O2 and -Os optimizations. */
475 { OPT_LEVELS_2_PLUS, OPT_fcaller_saves, NULL, 1 },
476 { OPT_LEVELS_2_PLUS, OPT_fcode_hoisting, NULL, 1 },
477 { OPT_LEVELS_2_PLUS, OPT_fcrossjumping, NULL, 1 },
478 { OPT_LEVELS_2_PLUS, OPT_fcse_follow_jumps, NULL, 1 },
479 { OPT_LEVELS_2_PLUS, OPT_fdevirtualize, NULL, 1 },
480 { OPT_LEVELS_2_PLUS, OPT_fdevirtualize_speculatively, NULL, 1 },
481 { OPT_LEVELS_2_PLUS, OPT_fexpensive_optimizations, NULL, 1 },
482 { OPT_LEVELS_2_PLUS, OPT_fgcse, NULL, 1 },
483 { OPT_LEVELS_2_PLUS, OPT_fhoist_adjacent_loads, NULL, 1 },
484 { OPT_LEVELS_2_PLUS, OPT_findirect_inlining, NULL, 1 },
485 { OPT_LEVELS_2_PLUS, OPT_finline_small_functions, NULL, 1 },
486 { OPT_LEVELS_2_PLUS, OPT_fipa_bit_cp, NULL, 1 },
487 { OPT_LEVELS_2_PLUS, OPT_fipa_cp, NULL, 1 },
488 { OPT_LEVELS_2_PLUS, OPT_fipa_icf, NULL, 1 },
489 { OPT_LEVELS_2_PLUS, OPT_fipa_ra, NULL, 1 },
490 { OPT_LEVELS_2_PLUS, OPT_fipa_sra, NULL, 1 },
491 { OPT_LEVELS_2_PLUS, OPT_fipa_vrp, NULL, 1 },
492 { OPT_LEVELS_2_PLUS, OPT_fisolate_erroneous_paths_dereference, NULL, 1 },
493 { OPT_LEVELS_2_PLUS, OPT_flra_remat, NULL, 1 },
494 { OPT_LEVELS_2_PLUS, OPT_foptimize_sibling_calls, NULL, 1 },
495 { OPT_LEVELS_2_PLUS, OPT_fpartial_inlining, NULL, 1 },
496 { OPT_LEVELS_2_PLUS, OPT_fpeephole2, NULL, 1 },
497 { OPT_LEVELS_2_PLUS, OPT_freorder_functions, NULL, 1 },
498 { OPT_LEVELS_2_PLUS, OPT_frerun_cse_after_loop, NULL, 1 },
499 #ifdef INSN_SCHEDULING
500 { OPT_LEVELS_2_PLUS, OPT_fschedule_insns2, NULL, 1 },
501 #endif
502 { OPT_LEVELS_2_PLUS, OPT_fstrict_aliasing, NULL, 1 },
503 { OPT_LEVELS_2_PLUS, OPT_fstore_merging, NULL, 1 },
504 { OPT_LEVELS_2_PLUS, OPT_fthread_jumps, NULL, 1 },
505 { OPT_LEVELS_2_PLUS, OPT_ftree_pre, NULL, 1 },
506 { OPT_LEVELS_2_PLUS, OPT_ftree_switch_conversion, NULL, 1 },
507 { OPT_LEVELS_2_PLUS, OPT_ftree_tail_merge, NULL, 1 },
508 { OPT_LEVELS_2_PLUS, OPT_ftree_vrp, NULL, 1 },
509 { OPT_LEVELS_2_PLUS, OPT_fvect_cost_model_, NULL, VECT_COST_MODEL_CHEAP },
510 { OPT_LEVELS_2_PLUS, OPT_finline_functions, NULL, 1 },
511 { OPT_LEVELS_2_PLUS, OPT_ftree_loop_distribute_patterns, NULL, 1 },
512
513 /* -O2 and above optimizations, but not -Os or -Og. */
514 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_falign_functions, NULL, 1 },
515 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_falign_jumps, NULL, 1 },
516 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_falign_labels, NULL, 1 },
517 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_falign_loops, NULL, 1 },
518 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_foptimize_strlen, NULL, 1 },
519 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_freorder_blocks_algorithm_, NULL,
520 REORDER_BLOCKS_ALGORITHM_STC },
521 #ifdef INSN_SCHEDULING
522 /* Only run the pre-regalloc scheduling pass if optimizing for speed. */
523 { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_fschedule_insns, NULL, 1 },
524 #endif
525
526 /* -O3 and -Os optimizations. */
527
528 /* -O3 optimizations. */
529 { OPT_LEVELS_3_PLUS, OPT_fgcse_after_reload, NULL, 1 },
530 { OPT_LEVELS_3_PLUS, OPT_fipa_cp_clone, NULL, 1 },
531 { OPT_LEVELS_3_PLUS, OPT_floop_interchange, NULL, 1 },
532 { OPT_LEVELS_3_PLUS, OPT_floop_unroll_and_jam, NULL, 1 },
533 { OPT_LEVELS_3_PLUS, OPT_fpeel_loops, NULL, 1 },
534 { OPT_LEVELS_3_PLUS, OPT_fpredictive_commoning, NULL, 1 },
535 { OPT_LEVELS_3_PLUS, OPT_fsplit_loops, NULL, 1 },
536 { OPT_LEVELS_3_PLUS, OPT_fsplit_paths, NULL, 1 },
537 { OPT_LEVELS_3_PLUS, OPT_ftree_loop_distribution, NULL, 1 },
538 { OPT_LEVELS_3_PLUS, OPT_ftree_loop_vectorize, NULL, 1 },
539 { OPT_LEVELS_3_PLUS, OPT_ftree_partial_pre, NULL, 1 },
540 { OPT_LEVELS_3_PLUS, OPT_ftree_slp_vectorize, NULL, 1 },
541 { OPT_LEVELS_3_PLUS, OPT_funswitch_loops, NULL, 1 },
542 { OPT_LEVELS_3_PLUS, OPT_fvect_cost_model_, NULL, VECT_COST_MODEL_DYNAMIC },
543 { OPT_LEVELS_3_PLUS, OPT_fversion_loops_for_strides, NULL, 1 },
544
545 /* -O3 parameters. */
546 { OPT_LEVELS_3_PLUS, OPT__param_max_inline_insns_auto_, NULL, 30 },
547 { OPT_LEVELS_3_PLUS, OPT__param_early_inlining_insns_, NULL, 14 },
548 { OPT_LEVELS_3_PLUS, OPT__param_inline_heuristics_hint_percent_, NULL, 600 },
549 { OPT_LEVELS_3_PLUS, OPT__param_inline_min_speedup_, NULL, 15 },
550 { OPT_LEVELS_3_PLUS, OPT__param_max_inline_insns_single_, NULL, 200 },
551
552 /* -Ofast adds optimizations to -O3. */
553 { OPT_LEVELS_FAST, OPT_ffast_math, NULL, 1 },
554 { OPT_LEVELS_FAST, OPT_fallow_store_data_races, NULL, 1 },
555
556 { OPT_LEVELS_NONE, 0, NULL, 0 }
557 };
558
559 /* Default the options in OPTS and OPTS_SET based on the optimization
560 settings in DECODED_OPTIONS and DECODED_OPTIONS_COUNT. */
561 void
562 default_options_optimization (struct gcc_options *opts,
563 struct gcc_options *opts_set,
564 struct cl_decoded_option *decoded_options,
565 unsigned int decoded_options_count,
566 location_t loc,
567 unsigned int lang_mask,
568 const struct cl_option_handlers *handlers,
569 diagnostic_context *dc)
570 {
571 unsigned int i;
572 int opt2;
573 bool openacc_mode = false;
574
575 /* Scan to see what optimization level has been specified. That will
576 determine the default value of many flags. */
577 for (i = 1; i < decoded_options_count; i++)
578 {
579 struct cl_decoded_option *opt = &decoded_options[i];
580 switch (opt->opt_index)
581 {
582 case OPT_O:
583 if (*opt->arg == '\0')
584 {
585 opts->x_optimize = 1;
586 opts->x_optimize_size = 0;
587 opts->x_optimize_fast = 0;
588 opts->x_optimize_debug = 0;
589 }
590 else
591 {
592 const int optimize_val = integral_argument (opt->arg);
593 if (optimize_val == -1)
594 error_at (loc, "argument to %<-O%> should be a non-negative "
595 "integer, %<g%>, %<s%> or %<fast%>");
596 else
597 {
598 opts->x_optimize = optimize_val;
599 if ((unsigned int) opts->x_optimize > 255)
600 opts->x_optimize = 255;
601 opts->x_optimize_size = 0;
602 opts->x_optimize_fast = 0;
603 opts->x_optimize_debug = 0;
604 }
605 }
606 break;
607
608 case OPT_Os:
609 opts->x_optimize_size = 1;
610
611 /* Optimizing for size forces optimize to be 2. */
612 opts->x_optimize = 2;
613 opts->x_optimize_fast = 0;
614 opts->x_optimize_debug = 0;
615 break;
616
617 case OPT_Ofast:
618 /* -Ofast only adds flags to -O3. */
619 opts->x_optimize_size = 0;
620 opts->x_optimize = 3;
621 opts->x_optimize_fast = 1;
622 opts->x_optimize_debug = 0;
623 break;
624
625 case OPT_Og:
626 /* -Og selects optimization level 1. */
627 opts->x_optimize_size = 0;
628 opts->x_optimize = 1;
629 opts->x_optimize_fast = 0;
630 opts->x_optimize_debug = 1;
631 break;
632
633 case OPT_fopenacc:
634 if (opt->value)
635 openacc_mode = true;
636 break;
637
638 default:
639 /* Ignore other options in this prescan. */
640 break;
641 }
642 }
643
644 maybe_default_options (opts, opts_set, default_options_table,
645 opts->x_optimize, opts->x_optimize_size,
646 opts->x_optimize_fast, opts->x_optimize_debug,
647 lang_mask, handlers, loc, dc);
648
649 /* -O2 param settings. */
650 opt2 = (opts->x_optimize >= 2);
651
652 if (openacc_mode)
653 SET_OPTION_IF_UNSET (opts, opts_set, flag_ipa_pta, true);
654
655 /* Track fields in field-sensitive alias analysis. */
656 if (opt2)
657 SET_OPTION_IF_UNSET (opts, opts_set, param_max_fields_for_field_sensitive,
658 100);
659
660 if (opts->x_optimize_size)
661 /* We want to crossjump as much as possible. */
662 SET_OPTION_IF_UNSET (opts, opts_set, param_min_crossjump_insns, 1);
663
664 /* Restrict the amount of work combine does at -Og while retaining
665 most of its useful transforms. */
666 if (opts->x_optimize_debug)
667 SET_OPTION_IF_UNSET (opts, opts_set, param_max_combine_insns, 2);
668
669 /* Allow default optimizations to be specified on a per-machine basis. */
670 maybe_default_options (opts, opts_set,
671 targetm_common.option_optimization_table,
672 opts->x_optimize, opts->x_optimize_size,
673 opts->x_optimize_fast, opts->x_optimize_debug,
674 lang_mask, handlers, loc, dc);
675 }
676
677 /* Control IPA optimizations based on different live patching LEVEL. */
678 static void
679 control_options_for_live_patching (struct gcc_options *opts,
680 struct gcc_options *opts_set,
681 enum live_patching_level level,
682 location_t loc)
683 {
684 gcc_assert (level > LIVE_PATCHING_NONE);
685
686 switch (level)
687 {
688 case LIVE_PATCHING_INLINE_ONLY_STATIC:
689 if (opts_set->x_flag_ipa_cp_clone && opts->x_flag_ipa_cp_clone)
690 error_at (loc,
691 "%<-fipa-cp-clone%> is incompatible with "
692 "%<-flive-patching=inline-only-static%>");
693 else
694 opts->x_flag_ipa_cp_clone = 0;
695
696 if (opts_set->x_flag_ipa_sra && opts->x_flag_ipa_sra)
697 error_at (loc,
698 "%<-fipa-sra%> is incompatible with "
699 "%<-flive-patching=inline-only-static%>");
700 else
701 opts->x_flag_ipa_sra = 0;
702
703 if (opts_set->x_flag_partial_inlining && opts->x_flag_partial_inlining)
704 error_at (loc,
705 "%<-fpartial-inlining%> is incompatible with "
706 "%<-flive-patching=inline-only-static%>");
707 else
708 opts->x_flag_partial_inlining = 0;
709
710 if (opts_set->x_flag_ipa_cp && opts->x_flag_ipa_cp)
711 error_at (loc,
712 "%<-fipa-cp%> is incompatible with "
713 "%<-flive-patching=inline-only-static%>");
714 else
715 opts->x_flag_ipa_cp = 0;
716
717 /* FALLTHROUGH. */
718 case LIVE_PATCHING_INLINE_CLONE:
719 /* live patching should disable whole-program optimization. */
720 if (opts_set->x_flag_whole_program && opts->x_flag_whole_program)
721 error_at (loc,
722 "%<-fwhole-program%> is incompatible with "
723 "%<-flive-patching=inline-only-static|inline-clone%>");
724 else
725 opts->x_flag_whole_program = 0;
726
727 /* visibility change should be excluded by !flag_whole_program
728 && !in_lto_p && !flag_ipa_cp_clone && !flag_ipa_sra
729 && !flag_partial_inlining. */
730
731 if (opts_set->x_flag_ipa_pta && opts->x_flag_ipa_pta)
732 error_at (loc,
733 "%<-fipa-pta%> is incompatible with "
734 "%<-flive-patching=inline-only-static|inline-clone%>");
735 else
736 opts->x_flag_ipa_pta = 0;
737
738 if (opts_set->x_flag_ipa_reference && opts->x_flag_ipa_reference)
739 error_at (loc,
740 "%<-fipa-reference%> is incompatible with "
741 "%<-flive-patching=inline-only-static|inline-clone%>");
742 else
743 opts->x_flag_ipa_reference = 0;
744
745 if (opts_set->x_flag_ipa_ra && opts->x_flag_ipa_ra)
746 error_at (loc,
747 "%<-fipa-ra%> is incompatible with "
748 "%<-flive-patching=inline-only-static|inline-clone%>");
749 else
750 opts->x_flag_ipa_ra = 0;
751
752 if (opts_set->x_flag_ipa_icf && opts->x_flag_ipa_icf)
753 error_at (loc,
754 "%<-fipa-icf%> is incompatible with "
755 "%<-flive-patching=inline-only-static|inline-clone%>");
756 else
757 opts->x_flag_ipa_icf = 0;
758
759 if (opts_set->x_flag_ipa_icf_functions && opts->x_flag_ipa_icf_functions)
760 error_at (loc,
761 "%<-fipa-icf-functions%> is incompatible with "
762 "%<-flive-patching=inline-only-static|inline-clone%>");
763 else
764 opts->x_flag_ipa_icf_functions = 0;
765
766 if (opts_set->x_flag_ipa_icf_variables && opts->x_flag_ipa_icf_variables)
767 error_at (loc,
768 "%<-fipa-icf-variables%> is incompatible with "
769 "%<-flive-patching=inline-only-static|inline-clone%>");
770 else
771 opts->x_flag_ipa_icf_variables = 0;
772
773 if (opts_set->x_flag_ipa_bit_cp && opts->x_flag_ipa_bit_cp)
774 error_at (loc,
775 "%<-fipa-bit-cp%> is incompatible with "
776 "%<-flive-patching=inline-only-static|inline-clone%>");
777 else
778 opts->x_flag_ipa_bit_cp = 0;
779
780 if (opts_set->x_flag_ipa_vrp && opts->x_flag_ipa_vrp)
781 error_at (loc,
782 "%<-fipa-vrp%> is incompatible with "
783 "%<-flive-patching=inline-only-static|inline-clone%>");
784 else
785 opts->x_flag_ipa_vrp = 0;
786
787 if (opts_set->x_flag_ipa_pure_const && opts->x_flag_ipa_pure_const)
788 error_at (loc,
789 "%<-fipa-pure-const%> is incompatible with "
790 "%<-flive-patching=inline-only-static|inline-clone%>");
791 else
792 opts->x_flag_ipa_pure_const = 0;
793
794 /* FIXME: disable unreachable code removal. */
795
796 /* discovery of functions/variables with no address taken. */
797 if (opts_set->x_flag_ipa_reference_addressable
798 && opts->x_flag_ipa_reference_addressable)
799 error_at (loc,
800 "%<-fipa-reference-addressable%> is incompatible with "
801 "%<-flive-patching=inline-only-static|inline-clone%>");
802 else
803 opts->x_flag_ipa_reference_addressable = 0;
804
805 /* ipa stack alignment propagation. */
806 if (opts_set->x_flag_ipa_stack_alignment
807 && opts->x_flag_ipa_stack_alignment)
808 error_at (loc,
809 "%<-fipa-stack-alignment%> is incompatible with "
810 "%<-flive-patching=inline-only-static|inline-clone%>");
811 else
812 opts->x_flag_ipa_stack_alignment = 0;
813 break;
814 default:
815 gcc_unreachable ();
816 }
817 }
818
819 /* --help option argument if set. */
820 vec<const char *> help_option_arguments;
821
822
823 /* After all options at LOC have been read into OPTS and OPTS_SET,
824 finalize settings of those options and diagnose incompatible
825 combinations. */
826 void
827 finish_options (struct gcc_options *opts, struct gcc_options *opts_set,
828 location_t loc)
829 {
830 enum unwind_info_type ui_except;
831
832 if (opts->x_dump_base_name
833 && ! opts->x_dump_base_name_prefixed)
834 {
835 const char *sep = opts->x_dump_base_name;
836
837 for (; *sep; sep++)
838 if (IS_DIR_SEPARATOR (*sep))
839 break;
840
841 if (*sep)
842 /* If dump_base_path contains subdirectories, don't prepend
843 anything. */;
844 else if (opts->x_dump_dir_name)
845 /* We have a DUMP_DIR_NAME, prepend that. */
846 opts->x_dump_base_name = opts_concat (opts->x_dump_dir_name,
847 opts->x_dump_base_name, NULL);
848
849 /* It is definitely prefixed now. */
850 opts->x_dump_base_name_prefixed = true;
851 }
852
853 /* Handle related options for unit-at-a-time, toplevel-reorder, and
854 section-anchors. */
855 if (!opts->x_flag_unit_at_a_time)
856 {
857 if (opts->x_flag_section_anchors && opts_set->x_flag_section_anchors)
858 error_at (loc, "section anchors must be disabled when unit-at-a-time "
859 "is disabled");
860 opts->x_flag_section_anchors = 0;
861 if (opts->x_flag_toplevel_reorder == 1)
862 error_at (loc, "toplevel reorder must be disabled when unit-at-a-time "
863 "is disabled");
864 opts->x_flag_toplevel_reorder = 0;
865 }
866
867 /* -fself-test depends on the state of the compiler prior to
868 compiling anything. Ideally it should be run on an empty source
869 file. However, in case we get run with actual source, assume
870 -fsyntax-only which will inhibit any compiler initialization
871 which may confuse the self tests. */
872 if (opts->x_flag_self_test)
873 opts->x_flag_syntax_only = 1;
874
875 if (opts->x_flag_tm && opts->x_flag_non_call_exceptions)
876 sorry ("transactional memory is not supported with non-call exceptions");
877
878 /* Unless the user has asked for section anchors, we disable toplevel
879 reordering at -O0 to disable transformations that might be surprising
880 to end users and to get -fno-toplevel-reorder tested. */
881 if (!opts->x_optimize
882 && opts->x_flag_toplevel_reorder == 2
883 && !(opts->x_flag_section_anchors && opts_set->x_flag_section_anchors))
884 {
885 opts->x_flag_toplevel_reorder = 0;
886 opts->x_flag_section_anchors = 0;
887 }
888 if (!opts->x_flag_toplevel_reorder)
889 {
890 if (opts->x_flag_section_anchors && opts_set->x_flag_section_anchors)
891 error_at (loc, "section anchors must be disabled when toplevel reorder"
892 " is disabled");
893 opts->x_flag_section_anchors = 0;
894 }
895
896 if (!opts->x_flag_opts_finished)
897 {
898 /* We initialize opts->x_flag_pie to -1 so that targets can set a
899 default value. */
900 if (opts->x_flag_pie == -1)
901 {
902 /* We initialize opts->x_flag_pic to -1 so that we can tell if
903 -fpic, -fPIC, -fno-pic or -fno-PIC is used. */
904 if (opts->x_flag_pic == -1)
905 opts->x_flag_pie = DEFAULT_FLAG_PIE;
906 else
907 opts->x_flag_pie = 0;
908 }
909 /* If -fPIE or -fpie is used, turn on PIC. */
910 if (opts->x_flag_pie)
911 opts->x_flag_pic = opts->x_flag_pie;
912 else if (opts->x_flag_pic == -1)
913 opts->x_flag_pic = 0;
914 if (opts->x_flag_pic && !opts->x_flag_pie)
915 opts->x_flag_shlib = 1;
916 opts->x_flag_opts_finished = true;
917 }
918
919 /* We initialize opts->x_flag_stack_protect to -1 so that targets
920 can set a default value. */
921 if (opts->x_flag_stack_protect == -1)
922 opts->x_flag_stack_protect = DEFAULT_FLAG_SSP;
923
924 if (opts->x_optimize == 0)
925 {
926 /* Inlining does not work if not optimizing,
927 so force it not to be done. */
928 opts->x_warn_inline = 0;
929 opts->x_flag_no_inline = 1;
930 }
931
932 /* The optimization to partition hot and cold basic blocks into separate
933 sections of the .o and executable files does not work (currently)
934 with exception handling. This is because there is no support for
935 generating unwind info. If opts->x_flag_exceptions is turned on
936 we need to turn off the partitioning optimization. */
937
938 ui_except = targetm_common.except_unwind_info (opts);
939
940 if (opts->x_flag_exceptions
941 && opts->x_flag_reorder_blocks_and_partition
942 && (ui_except == UI_SJLJ || ui_except >= UI_TARGET))
943 {
944 if (opts_set->x_flag_reorder_blocks_and_partition)
945 inform (loc,
946 "%<-freorder-blocks-and-partition%> does not work "
947 "with exceptions on this architecture");
948 opts->x_flag_reorder_blocks_and_partition = 0;
949 opts->x_flag_reorder_blocks = 1;
950 }
951
952 /* If user requested unwind info, then turn off the partitioning
953 optimization. */
954
955 if (opts->x_flag_unwind_tables
956 && !targetm_common.unwind_tables_default
957 && opts->x_flag_reorder_blocks_and_partition
958 && (ui_except == UI_SJLJ || ui_except >= UI_TARGET))
959 {
960 if (opts_set->x_flag_reorder_blocks_and_partition)
961 inform (loc,
962 "%<-freorder-blocks-and-partition%> does not support "
963 "unwind info on this architecture");
964 opts->x_flag_reorder_blocks_and_partition = 0;
965 opts->x_flag_reorder_blocks = 1;
966 }
967
968 /* If the target requested unwind info, then turn off the partitioning
969 optimization with a different message. Likewise, if the target does not
970 support named sections. */
971
972 if (opts->x_flag_reorder_blocks_and_partition
973 && (!targetm_common.have_named_sections
974 || (opts->x_flag_unwind_tables
975 && targetm_common.unwind_tables_default
976 && (ui_except == UI_SJLJ || ui_except >= UI_TARGET))))
977 {
978 if (opts_set->x_flag_reorder_blocks_and_partition)
979 inform (loc,
980 "%<-freorder-blocks-and-partition%> does not work "
981 "on this architecture");
982 opts->x_flag_reorder_blocks_and_partition = 0;
983 opts->x_flag_reorder_blocks = 1;
984 }
985
986
987 /* Pipelining of outer loops is only possible when general pipelining
988 capabilities are requested. */
989 if (!opts->x_flag_sel_sched_pipelining)
990 opts->x_flag_sel_sched_pipelining_outer_loops = 0;
991
992 if (opts->x_flag_conserve_stack)
993 {
994 SET_OPTION_IF_UNSET (opts, opts_set, param_large_stack_frame, 100);
995 SET_OPTION_IF_UNSET (opts, opts_set, param_stack_frame_growth, 40);
996 }
997
998 if (opts->x_flag_lto)
999 {
1000 #ifdef ENABLE_LTO
1001 opts->x_flag_generate_lto = 1;
1002
1003 /* When generating IL, do not operate in whole-program mode.
1004 Otherwise, symbols will be privatized too early, causing link
1005 errors later. */
1006 opts->x_flag_whole_program = 0;
1007 #else
1008 error_at (loc, "LTO support has not been enabled in this configuration");
1009 #endif
1010 if (!opts->x_flag_fat_lto_objects
1011 && (!HAVE_LTO_PLUGIN
1012 || (opts_set->x_flag_use_linker_plugin
1013 && !opts->x_flag_use_linker_plugin)))
1014 {
1015 if (opts_set->x_flag_fat_lto_objects)
1016 error_at (loc, "%<-fno-fat-lto-objects%> are supported only with "
1017 "linker plugin");
1018 opts->x_flag_fat_lto_objects = 1;
1019 }
1020
1021 /* -gsplit-dwarf isn't compatible with LTO, see PR88389. */
1022 if (opts->x_dwarf_split_debug_info)
1023 {
1024 inform (loc, "%<-gsplit-dwarf%> is not supported with LTO,"
1025 " disabling");
1026 opts->x_dwarf_split_debug_info = 0;
1027 }
1028 }
1029
1030 /* We initialize opts->x_flag_split_stack to -1 so that targets can set a
1031 default value if they choose based on other options. */
1032 if (opts->x_flag_split_stack == -1)
1033 opts->x_flag_split_stack = 0;
1034 else if (opts->x_flag_split_stack)
1035 {
1036 if (!targetm_common.supports_split_stack (true, opts))
1037 {
1038 error_at (loc, "%<-fsplit-stack%> is not supported by "
1039 "this compiler configuration");
1040 opts->x_flag_split_stack = 0;
1041 }
1042 }
1043
1044 /* If stack splitting is turned on, and the user did not explicitly
1045 request function partitioning, turn off partitioning, as it
1046 confuses the linker when trying to handle partitioned split-stack
1047 code that calls a non-split-stack functions. But if partitioning
1048 was turned on explicitly just hope for the best. */
1049 if (opts->x_flag_split_stack
1050 && opts->x_flag_reorder_blocks_and_partition)
1051 SET_OPTION_IF_UNSET (opts, opts_set, flag_reorder_blocks_and_partition, 0);
1052
1053 if (opts->x_flag_reorder_blocks_and_partition)
1054 SET_OPTION_IF_UNSET (opts, opts_set, flag_reorder_functions, 1);
1055
1056 /* The -gsplit-dwarf option requires -ggnu-pubnames. */
1057 if (opts->x_dwarf_split_debug_info)
1058 opts->x_debug_generate_pub_sections = 2;
1059
1060 if ((opts->x_flag_sanitize
1061 & (SANITIZE_USER_ADDRESS | SANITIZE_KERNEL_ADDRESS)) == 0)
1062 {
1063 if (opts->x_flag_sanitize & SANITIZE_POINTER_COMPARE)
1064 error_at (loc,
1065 "%<-fsanitize=pointer-compare%> must be combined with "
1066 "%<-fsanitize=address%> or %<-fsanitize=kernel-address%>");
1067 if (opts->x_flag_sanitize & SANITIZE_POINTER_SUBTRACT)
1068 error_at (loc,
1069 "%<-fsanitize=pointer-subtract%> must be combined with "
1070 "%<-fsanitize=address%> or %<-fsanitize=kernel-address%>");
1071 }
1072
1073 /* Userspace and kernel ASan conflict with each other. */
1074 if ((opts->x_flag_sanitize & SANITIZE_USER_ADDRESS)
1075 && (opts->x_flag_sanitize & SANITIZE_KERNEL_ADDRESS))
1076 error_at (loc,
1077 "%<-fsanitize=address%> is incompatible with "
1078 "%<-fsanitize=kernel-address%>");
1079
1080 /* And with TSan. */
1081 if ((opts->x_flag_sanitize & SANITIZE_ADDRESS)
1082 && (opts->x_flag_sanitize & SANITIZE_THREAD))
1083 error_at (loc,
1084 "%<-fsanitize=address%> and %<-fsanitize=kernel-address%> "
1085 "are incompatible with %<-fsanitize=thread%>");
1086
1087 if ((opts->x_flag_sanitize & SANITIZE_LEAK)
1088 && (opts->x_flag_sanitize & SANITIZE_THREAD))
1089 error_at (loc,
1090 "%<-fsanitize=leak%> is incompatible with %<-fsanitize=thread%>");
1091
1092 /* Check error recovery for -fsanitize-recover option. */
1093 for (int i = 0; sanitizer_opts[i].name != NULL; ++i)
1094 if ((opts->x_flag_sanitize_recover & sanitizer_opts[i].flag)
1095 && !sanitizer_opts[i].can_recover)
1096 error_at (loc, "%<-fsanitize-recover=%s%> is not supported",
1097 sanitizer_opts[i].name);
1098
1099 /* When instrumenting the pointers, we don't want to remove
1100 the null pointer checks. */
1101 if (opts->x_flag_sanitize & (SANITIZE_NULL | SANITIZE_NONNULL_ATTRIBUTE
1102 | SANITIZE_RETURNS_NONNULL_ATTRIBUTE))
1103 opts->x_flag_delete_null_pointer_checks = 0;
1104
1105 /* Aggressive compiler optimizations may cause false negatives. */
1106 if (opts->x_flag_sanitize & ~(SANITIZE_LEAK | SANITIZE_UNREACHABLE))
1107 opts->x_flag_aggressive_loop_optimizations = 0;
1108
1109 /* Enable -fsanitize-address-use-after-scope if address sanitizer is
1110 enabled. */
1111 if (opts->x_flag_sanitize & SANITIZE_USER_ADDRESS)
1112 SET_OPTION_IF_UNSET (opts, opts_set, flag_sanitize_address_use_after_scope,
1113 true);
1114
1115 /* Force -fstack-reuse=none in case -fsanitize-address-use-after-scope
1116 is enabled. */
1117 if (opts->x_flag_sanitize_address_use_after_scope)
1118 {
1119 if (opts->x_flag_stack_reuse != SR_NONE
1120 && opts_set->x_flag_stack_reuse != SR_NONE)
1121 error_at (loc,
1122 "%<-fsanitize-address-use-after-scope%> requires "
1123 "%<-fstack-reuse=none%> option");
1124
1125 opts->x_flag_stack_reuse = SR_NONE;
1126 }
1127
1128 if ((opts->x_flag_sanitize & SANITIZE_USER_ADDRESS) && opts->x_flag_tm)
1129 sorry ("transactional memory is not supported with %<-fsanitize=address%>");
1130
1131 if ((opts->x_flag_sanitize & SANITIZE_KERNEL_ADDRESS) && opts->x_flag_tm)
1132 sorry ("transactional memory is not supported with "
1133 "%<-fsanitize=kernel-address%>");
1134
1135 /* Currently live patching is not support for LTO. */
1136 if (opts->x_flag_live_patching && opts->x_flag_lto)
1137 sorry ("live patching is not supported with LTO");
1138
1139 /* Currently vtable verification is not supported for LTO */
1140 if (opts->x_flag_vtable_verify && opts->x_flag_lto)
1141 sorry ("vtable verification is not supported with LTO");
1142
1143 /* Control IPA optimizations based on different -flive-patching level. */
1144 if (opts->x_flag_live_patching)
1145 {
1146 control_options_for_live_patching (opts, opts_set,
1147 opts->x_flag_live_patching,
1148 loc);
1149 }
1150 }
1151
1152 #define LEFT_COLUMN 27
1153
1154 /* Output ITEM, of length ITEM_WIDTH, in the left column,
1155 followed by word-wrapped HELP in a second column. */
1156 static void
1157 wrap_help (const char *help,
1158 const char *item,
1159 unsigned int item_width,
1160 unsigned int columns)
1161 {
1162 unsigned int col_width = LEFT_COLUMN;
1163 unsigned int remaining, room, len;
1164
1165 remaining = strlen (help);
1166
1167 do
1168 {
1169 room = columns - 3 - MAX (col_width, item_width);
1170 if (room > columns)
1171 room = 0;
1172 len = remaining;
1173
1174 if (room < len)
1175 {
1176 unsigned int i;
1177
1178 for (i = 0; help[i]; i++)
1179 {
1180 if (i >= room && len != remaining)
1181 break;
1182 if (help[i] == ' ')
1183 len = i;
1184 else if ((help[i] == '-' || help[i] == '/')
1185 && help[i + 1] != ' '
1186 && i > 0 && ISALPHA (help[i - 1]))
1187 len = i + 1;
1188 }
1189 }
1190
1191 printf (" %-*.*s %.*s\n", col_width, item_width, item, len, help);
1192 item_width = 0;
1193 while (help[len] == ' ')
1194 len++;
1195 help += len;
1196 remaining -= len;
1197 }
1198 while (remaining);
1199 }
1200
1201 /* Data structure used to print list of valid option values. */
1202
1203 class option_help_tuple
1204 {
1205 public:
1206 option_help_tuple (int code, vec<const char *> values):
1207 m_code (code), m_values (values)
1208 {}
1209
1210 /* Code of an option. */
1211 int m_code;
1212
1213 /* List of possible values. */
1214 vec<const char *> m_values;
1215 };
1216
1217 /* Print help for a specific front-end, etc. */
1218 static void
1219 print_filtered_help (unsigned int include_flags,
1220 unsigned int exclude_flags,
1221 unsigned int any_flags,
1222 unsigned int columns,
1223 struct gcc_options *opts,
1224 unsigned int lang_mask)
1225 {
1226 unsigned int i;
1227 const char *help;
1228 bool found = false;
1229 bool displayed = false;
1230 char new_help[256];
1231
1232 if (!opts->x_help_printed)
1233 opts->x_help_printed = XCNEWVAR (char, cl_options_count);
1234
1235 if (!opts->x_help_enum_printed)
1236 opts->x_help_enum_printed = XCNEWVAR (char, cl_enums_count);
1237
1238 auto_vec<option_help_tuple> help_tuples;
1239
1240 for (i = 0; i < cl_options_count; i++)
1241 {
1242 const struct cl_option *option = cl_options + i;
1243 unsigned int len;
1244 const char *opt;
1245 const char *tab;
1246
1247 if (include_flags == 0
1248 || ((option->flags & include_flags) != include_flags))
1249 {
1250 if ((option->flags & any_flags) == 0)
1251 continue;
1252 }
1253
1254 /* Skip unwanted switches. */
1255 if ((option->flags & exclude_flags) != 0)
1256 continue;
1257
1258 /* The driver currently prints its own help text. */
1259 if ((option->flags & CL_DRIVER) != 0
1260 && (option->flags & (((1U << cl_lang_count) - 1)
1261 | CL_COMMON | CL_TARGET)) == 0)
1262 continue;
1263
1264 /* If an option contains a language specification,
1265 exclude it from common unless all languages are present. */
1266 if ((include_flags & CL_COMMON)
1267 && !(option->flags & CL_DRIVER)
1268 && (option->flags & CL_LANG_ALL)
1269 && (option->flags & CL_LANG_ALL) != CL_LANG_ALL)
1270 continue;
1271
1272 found = true;
1273 /* Skip switches that have already been printed. */
1274 if (opts->x_help_printed[i])
1275 continue;
1276
1277 opts->x_help_printed[i] = true;
1278
1279 help = option->help;
1280 if (help == NULL)
1281 {
1282 if (exclude_flags & CL_UNDOCUMENTED)
1283 continue;
1284
1285 help = undocumented_msg;
1286 }
1287
1288 /* Get the translation. */
1289 help = _(help);
1290
1291 if (option->alias_target < N_OPTS
1292 && cl_options [option->alias_target].help)
1293 {
1294 const struct cl_option *target = cl_options + option->alias_target;
1295 if (option->help == NULL)
1296 {
1297 /* The option is undocumented but is an alias for an option that
1298 is documented. If the option has alias arguments, then its
1299 purpose is to provide certain arguments to the other option, so
1300 inform the reader of this. Otherwise, point the reader to the
1301 other option in preference to the former. */
1302
1303 if (option->alias_arg)
1304 {
1305 if (option->neg_alias_arg)
1306 snprintf (new_help, sizeof new_help,
1307 _("Same as %s%s (or, in negated form, %s%s)."),
1308 target->opt_text, option->alias_arg,
1309 target->opt_text, option->neg_alias_arg);
1310 else
1311 snprintf (new_help, sizeof new_help,
1312 _("Same as %s%s."),
1313 target->opt_text, option->alias_arg);
1314 }
1315 else
1316 snprintf (new_help, sizeof new_help,
1317 _("Same as %s."),
1318 target->opt_text);
1319 }
1320 else
1321 {
1322 /* For documented options with aliases, mention the aliased
1323 option's name for reference. */
1324 snprintf (new_help, sizeof new_help,
1325 _("%s Same as %s."),
1326 help, cl_options [option->alias_target].opt_text);
1327 }
1328
1329 help = new_help;
1330 }
1331
1332 if (option->warn_message)
1333 {
1334 /* Mention that the use of the option will trigger a warning. */
1335 if (help == new_help)
1336 snprintf (new_help + strlen (new_help),
1337 sizeof new_help - strlen (new_help),
1338 " %s", _(use_diagnosed_msg));
1339 else
1340 snprintf (new_help, sizeof new_help,
1341 "%s %s", help, _(use_diagnosed_msg));
1342
1343 help = new_help;
1344 }
1345
1346 /* Find the gap between the name of the
1347 option and its descriptive text. */
1348 tab = strchr (help, '\t');
1349 if (tab)
1350 {
1351 len = tab - help;
1352 opt = help;
1353 help = tab + 1;
1354 }
1355 else
1356 {
1357 opt = option->opt_text;
1358 len = strlen (opt);
1359 }
1360
1361 /* With the -Q option enabled we change the descriptive text associated
1362 with an option to be an indication of its current setting. */
1363 if (!opts->x_quiet_flag)
1364 {
1365 void *flag_var = option_flag_var (i, opts);
1366
1367 if (len < (LEFT_COLUMN + 2))
1368 strcpy (new_help, "\t\t");
1369 else
1370 strcpy (new_help, "\t");
1371
1372 /* Set to print whether the option is enabled or disabled,
1373 or, if it's an alias for another option, the name of
1374 the aliased option. */
1375 bool print_state = false;
1376
1377 if (flag_var != NULL
1378 && option->var_type != CLVC_DEFER)
1379 {
1380 /* If OPTION is only available for a specific subset
1381 of languages other than this one, mention them. */
1382 bool avail_for_lang = true;
1383 if (unsigned langset = option->flags & CL_LANG_ALL)
1384 {
1385 if (!(langset & lang_mask))
1386 {
1387 avail_for_lang = false;
1388 strcat (new_help, _("[available in "));
1389 for (unsigned i = 0, n = 0; (1U << i) < CL_LANG_ALL; ++i)
1390 if (langset & (1U << i))
1391 {
1392 if (n++)
1393 strcat (new_help, ", ");
1394 strcat (new_help, lang_names[i]);
1395 }
1396 strcat (new_help, "]");
1397 }
1398 }
1399 if (!avail_for_lang)
1400 ; /* Print nothing else if the option is not available
1401 in the current language. */
1402 else if (option->flags & CL_JOINED)
1403 {
1404 if (option->var_type == CLVC_STRING)
1405 {
1406 if (* (const char **) flag_var != NULL)
1407 snprintf (new_help + strlen (new_help),
1408 sizeof (new_help) - strlen (new_help),
1409 "%s", * (const char **) flag_var);
1410 }
1411 else if (option->var_type == CLVC_ENUM)
1412 {
1413 const struct cl_enum *e = &cl_enums[option->var_enum];
1414 int value;
1415 const char *arg = NULL;
1416
1417 value = e->get (flag_var);
1418 enum_value_to_arg (e->values, &arg, value, lang_mask);
1419 if (arg == NULL)
1420 arg = _("[default]");
1421 snprintf (new_help + strlen (new_help),
1422 sizeof (new_help) - strlen (new_help),
1423 "%s", arg);
1424 }
1425 else
1426 {
1427 if (option->cl_host_wide_int)
1428 sprintf (new_help + strlen (new_help),
1429 _("%llu bytes"), (unsigned long long)
1430 *(unsigned HOST_WIDE_INT *) flag_var);
1431 else
1432 sprintf (new_help + strlen (new_help),
1433 "%i", * (int *) flag_var);
1434 }
1435 }
1436 else
1437 print_state = true;
1438 }
1439 else
1440 /* When there is no argument, print the option state only
1441 if the option takes no argument. */
1442 print_state = !(option->flags & CL_JOINED);
1443
1444 if (print_state)
1445 {
1446 if (option->alias_target < N_OPTS
1447 && option->alias_target != OPT_SPECIAL_warn_removed
1448 && option->alias_target != OPT_SPECIAL_ignore
1449 && option->alias_target != OPT_SPECIAL_input_file
1450 && option->alias_target != OPT_SPECIAL_program_name
1451 && option->alias_target != OPT_SPECIAL_unknown)
1452 {
1453 const struct cl_option *target
1454 = &cl_options[option->alias_target];
1455 sprintf (new_help + strlen (new_help), "%s%s",
1456 target->opt_text,
1457 option->alias_arg ? option->alias_arg : "");
1458 }
1459 else if (option->alias_target == OPT_SPECIAL_ignore)
1460 strcat (new_help, ("[ignored]"));
1461 else
1462 {
1463 /* Print the state for an on/off option. */
1464 int ena = option_enabled (i, lang_mask, opts);
1465 if (ena > 0)
1466 strcat (new_help, _("[enabled]"));
1467 else if (ena == 0)
1468 strcat (new_help, _("[disabled]"));
1469 }
1470 }
1471
1472 help = new_help;
1473 }
1474
1475 if (option->range_max != -1)
1476 {
1477 char b[128];
1478 snprintf (b, sizeof (b), "<%d,%d>", option->range_min,
1479 option->range_max);
1480 opt = concat (opt, b, NULL);
1481 len += strlen (b);
1482 }
1483
1484 wrap_help (help, opt, len, columns);
1485 displayed = true;
1486
1487 if (option->var_type == CLVC_ENUM
1488 && opts->x_help_enum_printed[option->var_enum] != 2)
1489 opts->x_help_enum_printed[option->var_enum] = 1;
1490 else
1491 {
1492 vec<const char *> option_values
1493 = targetm_common.get_valid_option_values (i, NULL);
1494 if (!option_values.is_empty ())
1495 help_tuples.safe_push (option_help_tuple (i, option_values));
1496 }
1497 }
1498
1499 if (! found)
1500 {
1501 unsigned int langs = include_flags & CL_LANG_ALL;
1502
1503 if (langs == 0)
1504 printf (_(" No options with the desired characteristics were found\n"));
1505 else
1506 {
1507 unsigned int i;
1508
1509 /* PR 31349: Tell the user how to see all of the
1510 options supported by a specific front end. */
1511 for (i = 0; (1U << i) < CL_LANG_ALL; i ++)
1512 if ((1U << i) & langs)
1513 printf (_(" None found. Use --help=%s to show *all* the options supported by the %s front-end.\n"),
1514 lang_names[i], lang_names[i]);
1515 }
1516
1517 }
1518 else if (! displayed)
1519 printf (_(" All options with the desired characteristics have already been displayed\n"));
1520
1521 putchar ('\n');
1522
1523 /* Print details of enumerated option arguments, if those
1524 enumerations have help text headings provided. If no help text
1525 is provided, presume that the possible values are listed in the
1526 help text for the relevant options. */
1527 for (i = 0; i < cl_enums_count; i++)
1528 {
1529 unsigned int j, pos;
1530
1531 if (opts->x_help_enum_printed[i] != 1)
1532 continue;
1533 if (cl_enums[i].help == NULL)
1534 continue;
1535 printf (" %s\n ", _(cl_enums[i].help));
1536 pos = 4;
1537 for (j = 0; cl_enums[i].values[j].arg != NULL; j++)
1538 {
1539 unsigned int len = strlen (cl_enums[i].values[j].arg);
1540
1541 if (pos > 4 && pos + 1 + len <= columns)
1542 {
1543 printf (" %s", cl_enums[i].values[j].arg);
1544 pos += 1 + len;
1545 }
1546 else
1547 {
1548 if (pos > 4)
1549 {
1550 printf ("\n ");
1551 pos = 4;
1552 }
1553 printf ("%s", cl_enums[i].values[j].arg);
1554 pos += len;
1555 }
1556 }
1557 printf ("\n\n");
1558 opts->x_help_enum_printed[i] = 2;
1559 }
1560
1561 for (unsigned i = 0; i < help_tuples.length (); i++)
1562 {
1563 const struct cl_option *option = cl_options + help_tuples[i].m_code;
1564 printf (_(" Known valid arguments for %s option:\n "),
1565 option->opt_text);
1566 for (unsigned j = 0; j < help_tuples[i].m_values.length (); j++)
1567 printf (" %s", help_tuples[i].m_values[j]);
1568 printf ("\n\n");
1569 }
1570 }
1571
1572 /* Display help for a specified type of option.
1573 The options must have ALL of the INCLUDE_FLAGS set
1574 ANY of the flags in the ANY_FLAGS set
1575 and NONE of the EXCLUDE_FLAGS set. The current option state is in
1576 OPTS; LANG_MASK is used for interpreting enumerated option state. */
1577 static void
1578 print_specific_help (unsigned int include_flags,
1579 unsigned int exclude_flags,
1580 unsigned int any_flags,
1581 struct gcc_options *opts,
1582 unsigned int lang_mask)
1583 {
1584 unsigned int all_langs_mask = (1U << cl_lang_count) - 1;
1585 const char * description = NULL;
1586 const char * descrip_extra = "";
1587 size_t i;
1588 unsigned int flag;
1589
1590 /* Sanity check: Make sure that we do not have more
1591 languages than we have bits available to enumerate them. */
1592 gcc_assert ((1U << cl_lang_count) <= CL_MIN_OPTION_CLASS);
1593
1594 /* If we have not done so already, obtain
1595 the desired maximum width of the output. */
1596 if (opts->x_help_columns == 0)
1597 {
1598 opts->x_help_columns = get_terminal_width ();
1599 if (opts->x_help_columns == INT_MAX)
1600 /* Use a reasonable default. */
1601 opts->x_help_columns = 80;
1602 }
1603
1604 /* Decide upon the title for the options that we are going to display. */
1605 for (i = 0, flag = 1; flag <= CL_MAX_OPTION_CLASS; flag <<= 1, i ++)
1606 {
1607 switch (flag & include_flags)
1608 {
1609 case 0:
1610 case CL_DRIVER:
1611 break;
1612
1613 case CL_TARGET:
1614 description = _("The following options are target specific");
1615 break;
1616 case CL_WARNING:
1617 description = _("The following options control compiler warning messages");
1618 break;
1619 case CL_OPTIMIZATION:
1620 description = _("The following options control optimizations");
1621 break;
1622 case CL_COMMON:
1623 description = _("The following options are language-independent");
1624 break;
1625 case CL_PARAMS:
1626 description = _("The following options control parameters");
1627 break;
1628 default:
1629 if (i >= cl_lang_count)
1630 break;
1631 if (exclude_flags & all_langs_mask)
1632 description = _("The following options are specific to just the language ");
1633 else
1634 description = _("The following options are supported by the language ");
1635 descrip_extra = lang_names [i];
1636 break;
1637 }
1638 }
1639
1640 if (description == NULL)
1641 {
1642 if (any_flags == 0)
1643 {
1644 if (include_flags & CL_UNDOCUMENTED)
1645 description = _("The following options are not documented");
1646 else if (include_flags & CL_SEPARATE)
1647 description = _("The following options take separate arguments");
1648 else if (include_flags & CL_JOINED)
1649 description = _("The following options take joined arguments");
1650 else
1651 {
1652 internal_error ("unrecognized %<include_flags 0x%x%> passed "
1653 "to %<print_specific_help%>",
1654 include_flags);
1655 return;
1656 }
1657 }
1658 else
1659 {
1660 if (any_flags & all_langs_mask)
1661 description = _("The following options are language-related");
1662 else
1663 description = _("The following options are language-independent");
1664 }
1665 }
1666
1667 printf ("%s%s:\n", description, descrip_extra);
1668 print_filtered_help (include_flags, exclude_flags, any_flags,
1669 opts->x_help_columns, opts, lang_mask);
1670 }
1671
1672 /* Enable FDO-related flags. */
1673
1674 static void
1675 enable_fdo_optimizations (struct gcc_options *opts,
1676 struct gcc_options *opts_set,
1677 int value)
1678 {
1679 SET_OPTION_IF_UNSET (opts, opts_set, flag_branch_probabilities, value);
1680 SET_OPTION_IF_UNSET (opts, opts_set, flag_profile_values, value);
1681 SET_OPTION_IF_UNSET (opts, opts_set, flag_unroll_loops, value);
1682 SET_OPTION_IF_UNSET (opts, opts_set, flag_peel_loops, value);
1683 SET_OPTION_IF_UNSET (opts, opts_set, flag_tracer, value);
1684 SET_OPTION_IF_UNSET (opts, opts_set, flag_value_profile_transformations,
1685 value);
1686 SET_OPTION_IF_UNSET (opts, opts_set, flag_inline_functions, value);
1687 SET_OPTION_IF_UNSET (opts, opts_set, flag_ipa_cp, value);
1688 if (value)
1689 {
1690 SET_OPTION_IF_UNSET (opts, opts_set, flag_ipa_cp_clone, 1);
1691 SET_OPTION_IF_UNSET (opts, opts_set, flag_ipa_bit_cp, 1);
1692 }
1693 SET_OPTION_IF_UNSET (opts, opts_set, flag_predictive_commoning, value);
1694 SET_OPTION_IF_UNSET (opts, opts_set, flag_split_loops, value);
1695 SET_OPTION_IF_UNSET (opts, opts_set, flag_unswitch_loops, value);
1696 SET_OPTION_IF_UNSET (opts, opts_set, flag_gcse_after_reload, value);
1697 SET_OPTION_IF_UNSET (opts, opts_set, flag_tree_loop_vectorize, value);
1698 SET_OPTION_IF_UNSET (opts, opts_set, flag_tree_slp_vectorize, value);
1699 SET_OPTION_IF_UNSET (opts, opts_set, flag_version_loops_for_strides, value);
1700 SET_OPTION_IF_UNSET (opts, opts_set, flag_vect_cost_model,
1701 VECT_COST_MODEL_DYNAMIC);
1702 SET_OPTION_IF_UNSET (opts, opts_set, flag_tree_loop_distribute_patterns,
1703 value);
1704 SET_OPTION_IF_UNSET (opts, opts_set, flag_loop_interchange, value);
1705 SET_OPTION_IF_UNSET (opts, opts_set, flag_unroll_jam, value);
1706 SET_OPTION_IF_UNSET (opts, opts_set, flag_tree_loop_distribution, value);
1707 }
1708
1709 /* -f{,no-}sanitize{,-recover}= suboptions. */
1710 const struct sanitizer_opts_s sanitizer_opts[] =
1711 {
1712 #define SANITIZER_OPT(name, flags, recover) \
1713 { #name, flags, sizeof #name - 1, recover }
1714 SANITIZER_OPT (address, (SANITIZE_ADDRESS | SANITIZE_USER_ADDRESS), true),
1715 SANITIZER_OPT (kernel-address, (SANITIZE_ADDRESS | SANITIZE_KERNEL_ADDRESS),
1716 true),
1717 SANITIZER_OPT (pointer-compare, SANITIZE_POINTER_COMPARE, true),
1718 SANITIZER_OPT (pointer-subtract, SANITIZE_POINTER_SUBTRACT, true),
1719 SANITIZER_OPT (thread, SANITIZE_THREAD, false),
1720 SANITIZER_OPT (leak, SANITIZE_LEAK, false),
1721 SANITIZER_OPT (shift, SANITIZE_SHIFT, true),
1722 SANITIZER_OPT (shift-base, SANITIZE_SHIFT_BASE, true),
1723 SANITIZER_OPT (shift-exponent, SANITIZE_SHIFT_EXPONENT, true),
1724 SANITIZER_OPT (integer-divide-by-zero, SANITIZE_DIVIDE, true),
1725 SANITIZER_OPT (undefined, SANITIZE_UNDEFINED, true),
1726 SANITIZER_OPT (unreachable, SANITIZE_UNREACHABLE, false),
1727 SANITIZER_OPT (vla-bound, SANITIZE_VLA, true),
1728 SANITIZER_OPT (return, SANITIZE_RETURN, false),
1729 SANITIZER_OPT (null, SANITIZE_NULL, true),
1730 SANITIZER_OPT (signed-integer-overflow, SANITIZE_SI_OVERFLOW, true),
1731 SANITIZER_OPT (bool, SANITIZE_BOOL, true),
1732 SANITIZER_OPT (enum, SANITIZE_ENUM, true),
1733 SANITIZER_OPT (float-divide-by-zero, SANITIZE_FLOAT_DIVIDE, true),
1734 SANITIZER_OPT (float-cast-overflow, SANITIZE_FLOAT_CAST, true),
1735 SANITIZER_OPT (bounds, SANITIZE_BOUNDS, true),
1736 SANITIZER_OPT (bounds-strict, SANITIZE_BOUNDS | SANITIZE_BOUNDS_STRICT, true),
1737 SANITIZER_OPT (alignment, SANITIZE_ALIGNMENT, true),
1738 SANITIZER_OPT (nonnull-attribute, SANITIZE_NONNULL_ATTRIBUTE, true),
1739 SANITIZER_OPT (returns-nonnull-attribute, SANITIZE_RETURNS_NONNULL_ATTRIBUTE,
1740 true),
1741 SANITIZER_OPT (object-size, SANITIZE_OBJECT_SIZE, true),
1742 SANITIZER_OPT (vptr, SANITIZE_VPTR, true),
1743 SANITIZER_OPT (pointer-overflow, SANITIZE_POINTER_OVERFLOW, true),
1744 SANITIZER_OPT (builtin, SANITIZE_BUILTIN, true),
1745 SANITIZER_OPT (all, ~0U, true),
1746 #undef SANITIZER_OPT
1747 { NULL, 0U, 0UL, false }
1748 };
1749
1750 /* -f{,no-}sanitize-coverage= suboptions. */
1751 const struct sanitizer_opts_s coverage_sanitizer_opts[] =
1752 {
1753 #define COVERAGE_SANITIZER_OPT(name, flags) \
1754 { #name, flags, sizeof #name - 1, true }
1755 COVERAGE_SANITIZER_OPT (trace-pc, SANITIZE_COV_TRACE_PC),
1756 COVERAGE_SANITIZER_OPT (trace-cmp, SANITIZE_COV_TRACE_CMP),
1757 #undef COVERAGE_SANITIZER_OPT
1758 { NULL, 0U, 0UL, false }
1759 };
1760
1761 /* A struct for describing a run of chars within a string. */
1762
1763 class string_fragment
1764 {
1765 public:
1766 string_fragment (const char *start, size_t len)
1767 : m_start (start), m_len (len) {}
1768
1769 const char *m_start;
1770 size_t m_len;
1771 };
1772
1773 /* Specialization of edit_distance_traits for string_fragment,
1774 for use by get_closest_sanitizer_option. */
1775
1776 template <>
1777 struct edit_distance_traits<const string_fragment &>
1778 {
1779 static size_t get_length (const string_fragment &fragment)
1780 {
1781 return fragment.m_len;
1782 }
1783
1784 static const char *get_string (const string_fragment &fragment)
1785 {
1786 return fragment.m_start;
1787 }
1788 };
1789
1790 /* Given ARG, an unrecognized sanitizer option, return the best
1791 matching sanitizer option, or NULL if there isn't one.
1792 OPTS is array of candidate sanitizer options.
1793 CODE is OPT_fsanitize_, OPT_fsanitize_recover_ or
1794 OPT_fsanitize_coverage_.
1795 VALUE is non-zero for the regular form of the option, zero
1796 for the "no-" form (e.g. "-fno-sanitize-recover="). */
1797
1798 static const char *
1799 get_closest_sanitizer_option (const string_fragment &arg,
1800 const struct sanitizer_opts_s *opts,
1801 enum opt_code code, int value)
1802 {
1803 best_match <const string_fragment &, const char*> bm (arg);
1804 for (int i = 0; opts[i].name != NULL; ++i)
1805 {
1806 /* -fsanitize=all is not valid, so don't offer it. */
1807 if (code == OPT_fsanitize_
1808 && opts[i].flag == ~0U
1809 && value)
1810 continue;
1811
1812 /* For -fsanitize-recover= (and not -fno-sanitize-recover=),
1813 don't offer the non-recoverable options. */
1814 if (code == OPT_fsanitize_recover_
1815 && !opts[i].can_recover
1816 && value)
1817 continue;
1818
1819 bm.consider (opts[i].name);
1820 }
1821 return bm.get_best_meaningful_candidate ();
1822 }
1823
1824 /* Parse comma separated sanitizer suboptions from P for option SCODE,
1825 adjust previous FLAGS and return new ones. If COMPLAIN is false,
1826 don't issue diagnostics. */
1827
1828 unsigned int
1829 parse_sanitizer_options (const char *p, location_t loc, int scode,
1830 unsigned int flags, int value, bool complain)
1831 {
1832 enum opt_code code = (enum opt_code) scode;
1833
1834 const struct sanitizer_opts_s *opts;
1835 if (code == OPT_fsanitize_coverage_)
1836 opts = coverage_sanitizer_opts;
1837 else
1838 opts = sanitizer_opts;
1839
1840 while (*p != 0)
1841 {
1842 size_t len, i;
1843 bool found = false;
1844 const char *comma = strchr (p, ',');
1845
1846 if (comma == NULL)
1847 len = strlen (p);
1848 else
1849 len = comma - p;
1850 if (len == 0)
1851 {
1852 p = comma + 1;
1853 continue;
1854 }
1855
1856 /* Check to see if the string matches an option class name. */
1857 for (i = 0; opts[i].name != NULL; ++i)
1858 if (len == opts[i].len && memcmp (p, opts[i].name, len) == 0)
1859 {
1860 /* Handle both -fsanitize and -fno-sanitize cases. */
1861 if (value && opts[i].flag == ~0U)
1862 {
1863 if (code == OPT_fsanitize_)
1864 {
1865 if (complain)
1866 error_at (loc, "%<-fsanitize=all%> option is not valid");
1867 }
1868 else
1869 flags |= ~(SANITIZE_THREAD | SANITIZE_LEAK
1870 | SANITIZE_UNREACHABLE | SANITIZE_RETURN);
1871 }
1872 else if (value)
1873 {
1874 /* Do not enable -fsanitize-recover=unreachable and
1875 -fsanitize-recover=return if -fsanitize-recover=undefined
1876 is selected. */
1877 if (code == OPT_fsanitize_recover_
1878 && opts[i].flag == SANITIZE_UNDEFINED)
1879 flags |= (SANITIZE_UNDEFINED
1880 & ~(SANITIZE_UNREACHABLE | SANITIZE_RETURN));
1881 else
1882 flags |= opts[i].flag;
1883 }
1884 else
1885 flags &= ~opts[i].flag;
1886 found = true;
1887 break;
1888 }
1889
1890 if (! found && complain)
1891 {
1892 const char *hint
1893 = get_closest_sanitizer_option (string_fragment (p, len),
1894 opts, code, value);
1895
1896 const char *suffix;
1897 if (code == OPT_fsanitize_recover_)
1898 suffix = "-recover";
1899 else if (code == OPT_fsanitize_coverage_)
1900 suffix = "-coverage";
1901 else
1902 suffix = "";
1903
1904 if (hint)
1905 error_at (loc,
1906 "unrecognized argument to %<-f%ssanitize%s=%> "
1907 "option: %q.*s; did you mean %qs?",
1908 value ? "" : "no-",
1909 suffix, (int) len, p, hint);
1910 else
1911 error_at (loc,
1912 "unrecognized argument to %<-f%ssanitize%s=%> option: "
1913 "%q.*s", value ? "" : "no-",
1914 suffix, (int) len, p);
1915 }
1916
1917 if (comma == NULL)
1918 break;
1919 p = comma + 1;
1920 }
1921 return flags;
1922 }
1923
1924 /* Parse string values of no_sanitize attribute passed in VALUE.
1925 Values are separated with comma. */
1926
1927 unsigned int
1928 parse_no_sanitize_attribute (char *value)
1929 {
1930 unsigned int flags = 0;
1931 unsigned int i;
1932 char *q = strtok (value, ",");
1933
1934 while (q != NULL)
1935 {
1936 for (i = 0; sanitizer_opts[i].name != NULL; ++i)
1937 if (strcmp (sanitizer_opts[i].name, q) == 0)
1938 {
1939 flags |= sanitizer_opts[i].flag;
1940 if (sanitizer_opts[i].flag == SANITIZE_UNDEFINED)
1941 flags |= SANITIZE_UNDEFINED_NONDEFAULT;
1942 break;
1943 }
1944
1945 if (sanitizer_opts[i].name == NULL)
1946 warning (OPT_Wattributes,
1947 "%qs attribute directive ignored", q);
1948
1949 q = strtok (NULL, ",");
1950 }
1951
1952 return flags;
1953 }
1954
1955 /* Parse -falign-NAME format for a FLAG value. Return individual
1956 parsed integer values into RESULT_VALUES array. If REPORT_ERROR is
1957 set, print error message at LOC location. */
1958
1959 bool
1960 parse_and_check_align_values (const char *flag,
1961 const char *name,
1962 auto_vec<unsigned> &result_values,
1963 bool report_error,
1964 location_t loc)
1965 {
1966 char *str = xstrdup (flag);
1967 for (char *p = strtok (str, ":"); p; p = strtok (NULL, ":"))
1968 {
1969 char *end;
1970 int v = strtol (p, &end, 10);
1971 if (*end != '\0' || v < 0)
1972 {
1973 if (report_error)
1974 error_at (loc, "invalid arguments for %<-falign-%s%> option: %qs",
1975 name, flag);
1976
1977 return false;
1978 }
1979
1980 result_values.safe_push ((unsigned)v);
1981 }
1982
1983 free (str);
1984
1985 /* Check that we have a correct number of values. */
1986 if (result_values.is_empty () || result_values.length () > 4)
1987 {
1988 if (report_error)
1989 error_at (loc, "invalid number of arguments for %<-falign-%s%> "
1990 "option: %qs", name, flag);
1991 return false;
1992 }
1993
1994 for (unsigned i = 0; i < result_values.length (); i++)
1995 if (result_values[i] > MAX_CODE_ALIGN_VALUE)
1996 {
1997 if (report_error)
1998 error_at (loc, "%<-falign-%s%> is not between 0 and %d",
1999 name, MAX_CODE_ALIGN_VALUE);
2000 return false;
2001 }
2002
2003 return true;
2004 }
2005
2006 /* Check that alignment value FLAG for -falign-NAME is valid at a given
2007 location LOC. OPT_STR points to the stored -falign-NAME=argument and
2008 OPT_FLAG points to the associated -falign-NAME on/off flag. */
2009
2010 static void
2011 check_alignment_argument (location_t loc, const char *flag, const char *name,
2012 int *opt_flag, const char **opt_str)
2013 {
2014 auto_vec<unsigned> align_result;
2015 parse_and_check_align_values (flag, name, align_result, true, loc);
2016
2017 if (align_result.length() >= 1 && align_result[0] == 0)
2018 {
2019 *opt_flag = 1;
2020 *opt_str = NULL;
2021 }
2022 }
2023
2024 /* Print help when OPT__help_ is set. */
2025
2026 void
2027 print_help (struct gcc_options *opts, unsigned int lang_mask,
2028 const char *help_option_argument)
2029 {
2030 const char *a = help_option_argument;
2031 unsigned int include_flags = 0;
2032 /* Note - by default we include undocumented options when listing
2033 specific classes. If you only want to see documented options
2034 then add ",^undocumented" to the --help= option. E.g.:
2035
2036 --help=target,^undocumented */
2037 unsigned int exclude_flags = 0;
2038
2039 if (lang_mask == CL_DRIVER)
2040 return;
2041
2042 /* Walk along the argument string, parsing each word in turn.
2043 The format is:
2044 arg = [^]{word}[,{arg}]
2045 word = {optimizers|target|warnings|undocumented|
2046 params|common|<language>} */
2047 while (*a != 0)
2048 {
2049 static const struct
2050 {
2051 const char *string;
2052 unsigned int flag;
2053 }
2054 specifics[] =
2055 {
2056 { "optimizers", CL_OPTIMIZATION },
2057 { "target", CL_TARGET },
2058 { "warnings", CL_WARNING },
2059 { "undocumented", CL_UNDOCUMENTED },
2060 { "params", CL_PARAMS },
2061 { "joined", CL_JOINED },
2062 { "separate", CL_SEPARATE },
2063 { "common", CL_COMMON },
2064 { NULL, 0 }
2065 };
2066 unsigned int *pflags;
2067 const char *comma;
2068 unsigned int lang_flag, specific_flag;
2069 unsigned int len;
2070 unsigned int i;
2071
2072 if (*a == '^')
2073 {
2074 ++a;
2075 if (*a == '\0')
2076 {
2077 error ("missing argument to %qs", "--help=^");
2078 break;
2079 }
2080 pflags = &exclude_flags;
2081 }
2082 else
2083 pflags = &include_flags;
2084
2085 comma = strchr (a, ',');
2086 if (comma == NULL)
2087 len = strlen (a);
2088 else
2089 len = comma - a;
2090 if (len == 0)
2091 {
2092 a = comma + 1;
2093 continue;
2094 }
2095
2096 /* Check to see if the string matches an option class name. */
2097 for (i = 0, specific_flag = 0; specifics[i].string != NULL; i++)
2098 if (strncasecmp (a, specifics[i].string, len) == 0)
2099 {
2100 specific_flag = specifics[i].flag;
2101 break;
2102 }
2103
2104 /* Check to see if the string matches a language name.
2105 Note - we rely upon the alpha-sorted nature of the entries in
2106 the lang_names array, specifically that shorter names appear
2107 before their longer variants. (i.e. C before C++). That way
2108 when we are attempting to match --help=c for example we will
2109 match with C first and not C++. */
2110 for (i = 0, lang_flag = 0; i < cl_lang_count; i++)
2111 if (strncasecmp (a, lang_names[i], len) == 0)
2112 {
2113 lang_flag = 1U << i;
2114 break;
2115 }
2116
2117 if (specific_flag != 0)
2118 {
2119 if (lang_flag == 0)
2120 *pflags |= specific_flag;
2121 else
2122 {
2123 /* The option's argument matches both the start of a
2124 language name and the start of an option class name.
2125 We have a special case for when the user has
2126 specified "--help=c", but otherwise we have to issue
2127 a warning. */
2128 if (strncasecmp (a, "c", len) == 0)
2129 *pflags |= lang_flag;
2130 else
2131 warning (0,
2132 "%<--help%> argument %q.*s is ambiguous, "
2133 "please be more specific",
2134 len, a);
2135 }
2136 }
2137 else if (lang_flag != 0)
2138 *pflags |= lang_flag;
2139 else
2140 warning (0,
2141 "unrecognized argument to %<--help=%> option: %q.*s",
2142 len, a);
2143
2144 if (comma == NULL)
2145 break;
2146 a = comma + 1;
2147 }
2148
2149 /* We started using PerFunction/Optimization for parameters and
2150 a warning. We should exclude these from optimization options. */
2151 if (include_flags & CL_OPTIMIZATION)
2152 exclude_flags |= CL_WARNING;
2153 if (!(include_flags & CL_PARAMS))
2154 exclude_flags |= CL_PARAMS;
2155
2156 if (include_flags)
2157 print_specific_help (include_flags, exclude_flags, 0, opts,
2158 lang_mask);
2159 }
2160
2161 /* Handle target- and language-independent options. Return zero to
2162 generate an "unknown option" message. Only options that need
2163 extra handling need to be listed here; if you simply want
2164 DECODED->value assigned to a variable, it happens automatically. */
2165
2166 bool
2167 common_handle_option (struct gcc_options *opts,
2168 struct gcc_options *opts_set,
2169 const struct cl_decoded_option *decoded,
2170 unsigned int lang_mask, int kind ATTRIBUTE_UNUSED,
2171 location_t loc,
2172 const struct cl_option_handlers *handlers,
2173 diagnostic_context *dc,
2174 void (*target_option_override_hook) (void))
2175 {
2176 size_t scode = decoded->opt_index;
2177 const char *arg = decoded->arg;
2178 HOST_WIDE_INT value = decoded->value;
2179 enum opt_code code = (enum opt_code) scode;
2180
2181 gcc_assert (decoded->canonical_option_num_elements <= 2);
2182
2183 switch (code)
2184 {
2185 case OPT__help:
2186 {
2187 unsigned int all_langs_mask = (1U << cl_lang_count) - 1;
2188 unsigned int undoc_mask;
2189 unsigned int i;
2190
2191 if (lang_mask == CL_DRIVER)
2192 break;
2193
2194 undoc_mask = ((opts->x_verbose_flag | opts->x_extra_warnings)
2195 ? 0
2196 : CL_UNDOCUMENTED);
2197 target_option_override_hook ();
2198 /* First display any single language specific options. */
2199 for (i = 0; i < cl_lang_count; i++)
2200 print_specific_help
2201 (1U << i, (all_langs_mask & (~ (1U << i))) | undoc_mask, 0, opts,
2202 lang_mask);
2203 /* Next display any multi language specific options. */
2204 print_specific_help (0, undoc_mask, all_langs_mask, opts, lang_mask);
2205 /* Then display any remaining, non-language options. */
2206 for (i = CL_MIN_OPTION_CLASS; i <= CL_MAX_OPTION_CLASS; i <<= 1)
2207 if (i != CL_DRIVER)
2208 print_specific_help (i, undoc_mask, 0, opts, lang_mask);
2209 opts->x_exit_after_options = true;
2210 break;
2211 }
2212
2213 case OPT__target_help:
2214 if (lang_mask == CL_DRIVER)
2215 break;
2216
2217 target_option_override_hook ();
2218 print_specific_help (CL_TARGET, CL_UNDOCUMENTED, 0, opts, lang_mask);
2219 opts->x_exit_after_options = true;
2220 break;
2221
2222 case OPT__help_:
2223 {
2224 help_option_arguments.safe_push (arg);
2225 opts->x_exit_after_options = true;
2226 break;
2227 }
2228
2229 case OPT__version:
2230 if (lang_mask == CL_DRIVER)
2231 break;
2232
2233 opts->x_exit_after_options = true;
2234 break;
2235
2236 case OPT__completion_:
2237 break;
2238
2239 case OPT_fsanitize_:
2240 opts->x_flag_sanitize
2241 = parse_sanitizer_options (arg, loc, code,
2242 opts->x_flag_sanitize, value, true);
2243
2244 /* Kernel ASan implies normal ASan but does not yet support
2245 all features. */
2246 if (opts->x_flag_sanitize & SANITIZE_KERNEL_ADDRESS)
2247 {
2248 SET_OPTION_IF_UNSET (opts, opts_set,
2249 param_asan_instrumentation_with_call_threshold,
2250 0);
2251 SET_OPTION_IF_UNSET (opts, opts_set, param_asan_globals, 0);
2252 SET_OPTION_IF_UNSET (opts, opts_set, param_asan_stack, 0);
2253 SET_OPTION_IF_UNSET (opts, opts_set, param_asan_protect_allocas, 0);
2254 SET_OPTION_IF_UNSET (opts, opts_set, param_asan_use_after_return, 0);
2255 }
2256 break;
2257
2258 case OPT_fsanitize_recover_:
2259 opts->x_flag_sanitize_recover
2260 = parse_sanitizer_options (arg, loc, code,
2261 opts->x_flag_sanitize_recover, value, true);
2262 break;
2263
2264 case OPT_fasan_shadow_offset_:
2265 /* Deferred. */
2266 break;
2267
2268 case OPT_fsanitize_address_use_after_scope:
2269 opts->x_flag_sanitize_address_use_after_scope = value;
2270 break;
2271
2272 case OPT_fsanitize_recover:
2273 if (value)
2274 opts->x_flag_sanitize_recover
2275 |= (SANITIZE_UNDEFINED | SANITIZE_UNDEFINED_NONDEFAULT)
2276 & ~(SANITIZE_UNREACHABLE | SANITIZE_RETURN);
2277 else
2278 opts->x_flag_sanitize_recover
2279 &= ~(SANITIZE_UNDEFINED | SANITIZE_UNDEFINED_NONDEFAULT);
2280 break;
2281
2282 case OPT_fsanitize_coverage_:
2283 opts->x_flag_sanitize_coverage
2284 = parse_sanitizer_options (arg, loc, code,
2285 opts->x_flag_sanitize_coverage, value, true);
2286 break;
2287
2288 case OPT_O:
2289 case OPT_Os:
2290 case OPT_Ofast:
2291 case OPT_Og:
2292 /* Currently handled in a prescan. */
2293 break;
2294
2295 case OPT_Werror:
2296 dc->warning_as_error_requested = value;
2297 break;
2298
2299 case OPT_Werror_:
2300 if (lang_mask == CL_DRIVER)
2301 break;
2302
2303 enable_warning_as_error (arg, value, lang_mask, handlers,
2304 opts, opts_set, loc, dc);
2305 break;
2306
2307 case OPT_Wfatal_errors:
2308 dc->fatal_errors = value;
2309 break;
2310
2311 case OPT_Wstack_usage_:
2312 opts->x_flag_stack_usage_info = value != -1;
2313 break;
2314
2315 case OPT_Wstrict_aliasing:
2316 set_Wstrict_aliasing (opts, value);
2317 break;
2318
2319 case OPT_Wstrict_overflow:
2320 opts->x_warn_strict_overflow = (value
2321 ? (int) WARN_STRICT_OVERFLOW_CONDITIONAL
2322 : 0);
2323 break;
2324
2325 case OPT_Wsystem_headers:
2326 dc->dc_warn_system_headers = value;
2327 break;
2328
2329 case OPT_aux_info:
2330 opts->x_flag_gen_aux_info = 1;
2331 break;
2332
2333 case OPT_d:
2334 decode_d_option (arg, opts, loc, dc);
2335 break;
2336
2337 case OPT_fcall_used_:
2338 case OPT_fcall_saved_:
2339 /* Deferred. */
2340 break;
2341
2342 case OPT_fdbg_cnt_:
2343 /* Deferred. */
2344 break;
2345
2346 case OPT_fdbg_cnt_list:
2347 /* Deferred. */
2348 opts->x_exit_after_options = true;
2349 break;
2350
2351 case OPT_fdebug_prefix_map_:
2352 case OPT_ffile_prefix_map_:
2353 /* Deferred. */
2354 break;
2355
2356 case OPT_fcallgraph_info:
2357 opts->x_flag_callgraph_info = CALLGRAPH_INFO_NAKED;
2358 break;
2359
2360 case OPT_fcallgraph_info_:
2361 {
2362 char *my_arg, *p;
2363 my_arg = xstrdup (arg);
2364 p = strtok (my_arg, ",");
2365 while (p)
2366 {
2367 if (strcmp (p, "su") == 0)
2368 {
2369 opts->x_flag_callgraph_info |= CALLGRAPH_INFO_STACK_USAGE;
2370 opts->x_flag_stack_usage_info = true;
2371 }
2372 else if (strcmp (p, "da") == 0)
2373 opts->x_flag_callgraph_info |= CALLGRAPH_INFO_DYNAMIC_ALLOC;
2374 else
2375 return 0;
2376 p = strtok (NULL, ",");
2377 }
2378 free (my_arg);
2379 }
2380 break;
2381
2382 case OPT_fdiagnostics_show_location_:
2383 diagnostic_prefixing_rule (dc) = (diagnostic_prefixing_rule_t) value;
2384 break;
2385
2386 case OPT_fdiagnostics_show_caret:
2387 dc->show_caret = value;
2388 break;
2389
2390 case OPT_fdiagnostics_show_labels:
2391 dc->show_labels_p = value;
2392 break;
2393
2394 case OPT_fdiagnostics_show_line_numbers:
2395 dc->show_line_numbers_p = value;
2396 break;
2397
2398 case OPT_fdiagnostics_color_:
2399 diagnostic_color_init (dc, value);
2400 break;
2401
2402 case OPT_fdiagnostics_urls_:
2403 diagnostic_urls_init (dc, value);
2404 break;
2405
2406 case OPT_fdiagnostics_format_:
2407 diagnostic_output_format_init (dc,
2408 (enum diagnostics_output_format)value);
2409 break;
2410
2411 case OPT_fdiagnostics_parseable_fixits:
2412 dc->parseable_fixits_p = value;
2413 break;
2414
2415 case OPT_fdiagnostics_column_unit_:
2416 dc->column_unit = (enum diagnostics_column_unit)value;
2417 break;
2418
2419 case OPT_fdiagnostics_column_origin_:
2420 dc->column_origin = value;
2421 break;
2422
2423 case OPT_fdiagnostics_show_cwe:
2424 dc->show_cwe = value;
2425 break;
2426
2427 case OPT_fdiagnostics_path_format_:
2428 dc->path_format = (enum diagnostic_path_format)value;
2429 break;
2430
2431 case OPT_fdiagnostics_show_path_depths:
2432 dc->show_path_depths = value;
2433 break;
2434
2435 case OPT_fdiagnostics_show_option:
2436 dc->show_option_requested = value;
2437 break;
2438
2439 case OPT_fdiagnostics_minimum_margin_width_:
2440 dc->min_margin_width = value;
2441 break;
2442
2443 case OPT_fdump_:
2444 /* Deferred. */
2445 break;
2446
2447 case OPT_ffast_math:
2448 set_fast_math_flags (opts, value);
2449 break;
2450
2451 case OPT_funsafe_math_optimizations:
2452 set_unsafe_math_optimizations_flags (opts, value);
2453 break;
2454
2455 case OPT_ffixed_:
2456 /* Deferred. */
2457 break;
2458
2459 case OPT_finline_limit_:
2460 SET_OPTION_IF_UNSET (opts, opts_set, param_max_inline_insns_single,
2461 value / 2);
2462 SET_OPTION_IF_UNSET (opts, opts_set, param_max_inline_insns_auto,
2463 value / 2);
2464 break;
2465
2466 case OPT_finstrument_functions_exclude_function_list_:
2467 add_comma_separated_to_vector
2468 (&opts->x_flag_instrument_functions_exclude_functions, arg);
2469 break;
2470
2471 case OPT_finstrument_functions_exclude_file_list_:
2472 add_comma_separated_to_vector
2473 (&opts->x_flag_instrument_functions_exclude_files, arg);
2474 break;
2475
2476 case OPT_fmessage_length_:
2477 pp_set_line_maximum_length (dc->printer, value);
2478 diagnostic_set_caret_max_width (dc, value);
2479 break;
2480
2481 case OPT_fopt_info:
2482 case OPT_fopt_info_:
2483 /* Deferred. */
2484 break;
2485
2486 case OPT_foffload_:
2487 {
2488 const char *p = arg;
2489 opts->x_flag_disable_hsa = true;
2490 while (*p != 0)
2491 {
2492 const char *comma = strchr (p, ',');
2493
2494 if ((strncmp (p, "disable", 7) == 0)
2495 && (p[7] == ',' || p[7] == '\0'))
2496 {
2497 opts->x_flag_disable_hsa = true;
2498 break;
2499 }
2500
2501 if ((strncmp (p, "hsa", 3) == 0)
2502 && (p[3] == ',' || p[3] == '\0'))
2503 {
2504 #ifdef ENABLE_HSA
2505 opts->x_flag_disable_hsa = false;
2506 #else
2507 sorry ("HSA has not been enabled during configuration");
2508 #endif
2509 }
2510 if (!comma)
2511 break;
2512 p = comma + 1;
2513 }
2514 break;
2515 }
2516
2517 #ifndef ACCEL_COMPILER
2518 case OPT_foffload_abi_:
2519 error_at (loc, "%<-foffload-abi%> option can be specified only for "
2520 "offload compiler");
2521 break;
2522 #endif
2523
2524 case OPT_fpack_struct_:
2525 if (value <= 0 || (value & (value - 1)) || value > 16)
2526 error_at (loc,
2527 "structure alignment must be a small power of two, not %wu",
2528 value);
2529 else
2530 opts->x_initial_max_fld_align = value;
2531 break;
2532
2533 case OPT_fplugin_:
2534 case OPT_fplugin_arg_:
2535 /* Deferred. */
2536 break;
2537
2538 case OPT_fprofile_use_:
2539 opts->x_profile_data_prefix = xstrdup (arg);
2540 opts->x_flag_profile_use = true;
2541 value = true;
2542 /* No break here - do -fprofile-use processing. */
2543 /* FALLTHRU */
2544 case OPT_fprofile_use:
2545 enable_fdo_optimizations (opts, opts_set, value);
2546 SET_OPTION_IF_UNSET (opts, opts_set, flag_profile_reorder_functions,
2547 value);
2548 /* Indirect call profiling should do all useful transformations
2549 speculative devirtualization does. */
2550 if (opts->x_flag_value_profile_transformations)
2551 SET_OPTION_IF_UNSET (opts, opts_set, flag_devirtualize_speculatively,
2552 false);
2553 break;
2554
2555 case OPT_fauto_profile_:
2556 opts->x_auto_profile_file = xstrdup (arg);
2557 opts->x_flag_auto_profile = true;
2558 value = true;
2559 /* No break here - do -fauto-profile processing. */
2560 /* FALLTHRU */
2561 case OPT_fauto_profile:
2562 enable_fdo_optimizations (opts, opts_set, value);
2563 SET_OPTION_IF_UNSET (opts, opts_set, flag_profile_correction, value);
2564 SET_OPTION_IF_UNSET (opts, opts_set,
2565 param_early_inliner_max_iterations, 10);
2566 break;
2567
2568 case OPT_fprofile_generate_:
2569 opts->x_profile_data_prefix = xstrdup (arg);
2570 value = true;
2571 /* No break here - do -fprofile-generate processing. */
2572 /* FALLTHRU */
2573 case OPT_fprofile_generate:
2574 SET_OPTION_IF_UNSET (opts, opts_set, profile_arc_flag, value);
2575 SET_OPTION_IF_UNSET (opts, opts_set, flag_profile_values, value);
2576 SET_OPTION_IF_UNSET (opts, opts_set, flag_inline_functions, value);
2577 SET_OPTION_IF_UNSET (opts, opts_set, flag_ipa_bit_cp, value);
2578 break;
2579
2580 case OPT_fpatchable_function_entry_:
2581 {
2582 char *patch_area_arg = xstrdup (arg);
2583 char *comma = strchr (patch_area_arg, ',');
2584 if (comma)
2585 {
2586 *comma = '\0';
2587 function_entry_patch_area_size =
2588 integral_argument (patch_area_arg);
2589 function_entry_patch_area_start =
2590 integral_argument (comma + 1);
2591 }
2592 else
2593 {
2594 function_entry_patch_area_size =
2595 integral_argument (patch_area_arg);
2596 function_entry_patch_area_start = 0;
2597 }
2598 if (function_entry_patch_area_size < 0
2599 || function_entry_patch_area_size > USHRT_MAX
2600 || function_entry_patch_area_start < 0
2601 || function_entry_patch_area_start > USHRT_MAX
2602 || function_entry_patch_area_size
2603 < function_entry_patch_area_start)
2604 error ("invalid arguments for %<-fpatchable-function-entry%>");
2605 free (patch_area_arg);
2606 }
2607 break;
2608
2609 case OPT_ftree_vectorize:
2610 /* Automatically sets -ftree-loop-vectorize and
2611 -ftree-slp-vectorize. Nothing more to do here. */
2612 break;
2613 case OPT_fshow_column:
2614 dc->show_column = value;
2615 break;
2616
2617 case OPT_frandom_seed:
2618 /* The real switch is -fno-random-seed. */
2619 if (value)
2620 return false;
2621 /* Deferred. */
2622 break;
2623
2624 case OPT_frandom_seed_:
2625 /* Deferred. */
2626 break;
2627
2628 case OPT_fsched_verbose_:
2629 #ifdef INSN_SCHEDULING
2630 /* Handled with Var in common.opt. */
2631 break;
2632 #else
2633 return false;
2634 #endif
2635
2636 case OPT_fsched_stalled_insns_:
2637 opts->x_flag_sched_stalled_insns = value;
2638 if (opts->x_flag_sched_stalled_insns == 0)
2639 opts->x_flag_sched_stalled_insns = -1;
2640 break;
2641
2642 case OPT_fsched_stalled_insns_dep_:
2643 opts->x_flag_sched_stalled_insns_dep = value;
2644 break;
2645
2646 case OPT_fstack_check_:
2647 if (!strcmp (arg, "no"))
2648 opts->x_flag_stack_check = NO_STACK_CHECK;
2649 else if (!strcmp (arg, "generic"))
2650 /* This is the old stack checking method. */
2651 opts->x_flag_stack_check = STACK_CHECK_BUILTIN
2652 ? FULL_BUILTIN_STACK_CHECK
2653 : GENERIC_STACK_CHECK;
2654 else if (!strcmp (arg, "specific"))
2655 /* This is the new stack checking method. */
2656 opts->x_flag_stack_check = STACK_CHECK_BUILTIN
2657 ? FULL_BUILTIN_STACK_CHECK
2658 : STACK_CHECK_STATIC_BUILTIN
2659 ? STATIC_BUILTIN_STACK_CHECK
2660 : GENERIC_STACK_CHECK;
2661 else
2662 warning_at (loc, 0, "unknown stack check parameter %qs", arg);
2663 break;
2664
2665 case OPT_fstack_limit:
2666 /* The real switch is -fno-stack-limit. */
2667 if (value)
2668 return false;
2669 /* Deferred. */
2670 break;
2671
2672 case OPT_fstack_limit_register_:
2673 case OPT_fstack_limit_symbol_:
2674 /* Deferred. */
2675 break;
2676
2677 case OPT_fstack_usage:
2678 opts->x_flag_stack_usage = value;
2679 opts->x_flag_stack_usage_info = value != 0;
2680 break;
2681
2682 case OPT_g:
2683 set_debug_level (NO_DEBUG, DEFAULT_GDB_EXTENSIONS, arg, opts, opts_set,
2684 loc);
2685 break;
2686
2687 case OPT_gdwarf:
2688 if (arg && strlen (arg) != 0)
2689 {
2690 error_at (loc, "%<-gdwarf%s%> is ambiguous; "
2691 "use %<-gdwarf-%s%> for DWARF version "
2692 "or %<-gdwarf%> %<-g%s%> for debug level", arg, arg, arg);
2693 break;
2694 }
2695 else
2696 value = opts->x_dwarf_version;
2697
2698 /* FALLTHRU */
2699 case OPT_gdwarf_:
2700 if (value < 2 || value > 5)
2701 error_at (loc, "dwarf version %wu is not supported", value);
2702 else
2703 opts->x_dwarf_version = value;
2704 set_debug_level (DWARF2_DEBUG, false, "", opts, opts_set, loc);
2705 break;
2706
2707 case OPT_gsplit_dwarf:
2708 set_debug_level (NO_DEBUG, DEFAULT_GDB_EXTENSIONS, "", opts, opts_set,
2709 loc);
2710 break;
2711
2712 case OPT_ggdb:
2713 set_debug_level (NO_DEBUG, 2, arg, opts, opts_set, loc);
2714 break;
2715
2716 case OPT_gstabs:
2717 case OPT_gstabs_:
2718 set_debug_level (DBX_DEBUG, code == OPT_gstabs_, arg, opts, opts_set,
2719 loc);
2720 break;
2721
2722 case OPT_gvms:
2723 set_debug_level (VMS_DEBUG, false, arg, opts, opts_set, loc);
2724 break;
2725
2726 case OPT_gxcoff:
2727 case OPT_gxcoff_:
2728 set_debug_level (XCOFF_DEBUG, code == OPT_gxcoff_, arg, opts, opts_set,
2729 loc);
2730 break;
2731
2732 case OPT_gz:
2733 case OPT_gz_:
2734 /* Handled completely via specs. */
2735 break;
2736
2737 case OPT_pedantic_errors:
2738 dc->pedantic_errors = 1;
2739 control_warning_option (OPT_Wpedantic, DK_ERROR, NULL, value,
2740 loc, lang_mask,
2741 handlers, opts, opts_set,
2742 dc);
2743 break;
2744
2745 case OPT_flto:
2746 opts->x_flag_lto = value ? "" : NULL;
2747 break;
2748
2749 case OPT_flto_:
2750 if (strcmp (arg, "none") != 0
2751 && strcmp (arg, "jobserver") != 0
2752 && strcmp (arg, "auto") != 0
2753 && atoi (arg) == 0)
2754 error_at (loc,
2755 "unrecognized argument to %<-flto=%> option: %qs", arg);
2756 break;
2757
2758 case OPT_w:
2759 dc->dc_inhibit_warnings = true;
2760 break;
2761
2762 case OPT_fmax_errors_:
2763 dc->max_errors = value;
2764 break;
2765
2766 case OPT_fuse_ld_bfd:
2767 case OPT_fuse_ld_gold:
2768 case OPT_fuse_ld_lld:
2769 case OPT_fuse_linker_plugin:
2770 /* No-op. Used by the driver and passed to us because it starts with f.*/
2771 break;
2772
2773 case OPT_fwrapv:
2774 if (value)
2775 opts->x_flag_trapv = 0;
2776 break;
2777
2778 case OPT_ftrapv:
2779 if (value)
2780 opts->x_flag_wrapv = 0;
2781 break;
2782
2783 case OPT_fstrict_overflow:
2784 opts->x_flag_wrapv = !value;
2785 opts->x_flag_wrapv_pointer = !value;
2786 if (!value)
2787 opts->x_flag_trapv = 0;
2788 break;
2789
2790 case OPT_fipa_icf:
2791 opts->x_flag_ipa_icf_functions = value;
2792 opts->x_flag_ipa_icf_variables = value;
2793 break;
2794
2795 case OPT_falign_loops_:
2796 check_alignment_argument (loc, arg, "loops",
2797 &opts->x_flag_align_loops,
2798 &opts->x_str_align_loops);
2799 break;
2800
2801 case OPT_falign_jumps_:
2802 check_alignment_argument (loc, arg, "jumps",
2803 &opts->x_flag_align_jumps,
2804 &opts->x_str_align_jumps);
2805 break;
2806
2807 case OPT_falign_labels_:
2808 check_alignment_argument (loc, arg, "labels",
2809 &opts->x_flag_align_labels,
2810 &opts->x_str_align_labels);
2811 break;
2812
2813 case OPT_falign_functions_:
2814 check_alignment_argument (loc, arg, "functions",
2815 &opts->x_flag_align_functions,
2816 &opts->x_str_align_functions);
2817 break;
2818
2819 case OPT_ftabstop_:
2820 /* It is documented that we silently ignore silly values. */
2821 if (value >= 1 && value <= 100)
2822 dc->tabstop = value;
2823 break;
2824
2825 default:
2826 /* If the flag was handled in a standard way, assume the lack of
2827 processing here is intentional. */
2828 gcc_assert (option_flag_var (scode, opts));
2829 break;
2830 }
2831
2832 common_handle_option_auto (opts, opts_set, decoded, lang_mask, kind,
2833 loc, handlers, dc);
2834 return true;
2835 }
2836
2837 /* Used to set the level of strict aliasing warnings in OPTS,
2838 when no level is specified (i.e., when -Wstrict-aliasing, and not
2839 -Wstrict-aliasing=level was given).
2840 ONOFF is assumed to take value 1 when -Wstrict-aliasing is specified,
2841 and 0 otherwise. After calling this function, wstrict_aliasing will be
2842 set to the default value of -Wstrict_aliasing=level, currently 3. */
2843 static void
2844 set_Wstrict_aliasing (struct gcc_options *opts, int onoff)
2845 {
2846 gcc_assert (onoff == 0 || onoff == 1);
2847 if (onoff != 0)
2848 opts->x_warn_strict_aliasing = 3;
2849 else
2850 opts->x_warn_strict_aliasing = 0;
2851 }
2852
2853 /* The following routines are useful in setting all the flags that
2854 -ffast-math and -fno-fast-math imply. */
2855 static void
2856 set_fast_math_flags (struct gcc_options *opts, int set)
2857 {
2858 if (!opts->frontend_set_flag_unsafe_math_optimizations)
2859 {
2860 opts->x_flag_unsafe_math_optimizations = set;
2861 set_unsafe_math_optimizations_flags (opts, set);
2862 }
2863 if (!opts->frontend_set_flag_finite_math_only)
2864 opts->x_flag_finite_math_only = set;
2865 if (!opts->frontend_set_flag_errno_math)
2866 opts->x_flag_errno_math = !set;
2867 if (set)
2868 {
2869 if (opts->frontend_set_flag_excess_precision == EXCESS_PRECISION_DEFAULT)
2870 opts->x_flag_excess_precision
2871 = set ? EXCESS_PRECISION_FAST : EXCESS_PRECISION_DEFAULT;
2872 if (!opts->frontend_set_flag_signaling_nans)
2873 opts->x_flag_signaling_nans = 0;
2874 if (!opts->frontend_set_flag_rounding_math)
2875 opts->x_flag_rounding_math = 0;
2876 if (!opts->frontend_set_flag_cx_limited_range)
2877 opts->x_flag_cx_limited_range = 1;
2878 }
2879 }
2880
2881 /* When -funsafe-math-optimizations is set the following
2882 flags are set as well. */
2883 static void
2884 set_unsafe_math_optimizations_flags (struct gcc_options *opts, int set)
2885 {
2886 if (!opts->frontend_set_flag_trapping_math)
2887 opts->x_flag_trapping_math = !set;
2888 if (!opts->frontend_set_flag_signed_zeros)
2889 opts->x_flag_signed_zeros = !set;
2890 if (!opts->frontend_set_flag_associative_math)
2891 opts->x_flag_associative_math = set;
2892 if (!opts->frontend_set_flag_reciprocal_math)
2893 opts->x_flag_reciprocal_math = set;
2894 }
2895
2896 /* Return true iff flags in OPTS are set as if -ffast-math. */
2897 bool
2898 fast_math_flags_set_p (const struct gcc_options *opts)
2899 {
2900 return (!opts->x_flag_trapping_math
2901 && opts->x_flag_unsafe_math_optimizations
2902 && opts->x_flag_finite_math_only
2903 && !opts->x_flag_signed_zeros
2904 && !opts->x_flag_errno_math
2905 && opts->x_flag_excess_precision == EXCESS_PRECISION_FAST);
2906 }
2907
2908 /* Return true iff flags are set as if -ffast-math but using the flags stored
2909 in the struct cl_optimization structure. */
2910 bool
2911 fast_math_flags_struct_set_p (struct cl_optimization *opt)
2912 {
2913 return (!opt->x_flag_trapping_math
2914 && opt->x_flag_unsafe_math_optimizations
2915 && opt->x_flag_finite_math_only
2916 && !opt->x_flag_signed_zeros
2917 && !opt->x_flag_errno_math);
2918 }
2919
2920 /* Handle a debug output -g switch for options OPTS
2921 (OPTS_SET->x_write_symbols storing whether a debug type was passed
2922 explicitly), location LOC. EXTENDED is true or false to support
2923 extended output (2 is special and means "-ggdb" was given). */
2924 static void
2925 set_debug_level (enum debug_info_type type, int extended, const char *arg,
2926 struct gcc_options *opts, struct gcc_options *opts_set,
2927 location_t loc)
2928 {
2929 opts->x_use_gnu_debug_info_extensions = extended;
2930
2931 if (type == NO_DEBUG)
2932 {
2933 if (opts->x_write_symbols == NO_DEBUG)
2934 {
2935 opts->x_write_symbols = PREFERRED_DEBUGGING_TYPE;
2936
2937 if (extended == 2)
2938 {
2939 #if defined DWARF2_DEBUGGING_INFO || defined DWARF2_LINENO_DEBUGGING_INFO
2940 opts->x_write_symbols = DWARF2_DEBUG;
2941 #elif defined DBX_DEBUGGING_INFO
2942 opts->x_write_symbols = DBX_DEBUG;
2943 #endif
2944 }
2945
2946 if (opts->x_write_symbols == NO_DEBUG)
2947 warning_at (loc, 0, "target system does not support debug output");
2948 }
2949 }
2950 else
2951 {
2952 /* Does it conflict with an already selected type? */
2953 if (opts_set->x_write_symbols != NO_DEBUG
2954 && opts->x_write_symbols != NO_DEBUG
2955 && type != opts->x_write_symbols)
2956 error_at (loc, "debug format %qs conflicts with prior selection",
2957 debug_type_names[type]);
2958 opts->x_write_symbols = type;
2959 opts_set->x_write_symbols = type;
2960 }
2961
2962 /* A debug flag without a level defaults to level 2.
2963 If off or at level 1, set it to level 2, but if already
2964 at level 3, don't lower it. */
2965 if (*arg == '\0')
2966 {
2967 if (opts->x_debug_info_level < DINFO_LEVEL_NORMAL)
2968 opts->x_debug_info_level = DINFO_LEVEL_NORMAL;
2969 }
2970 else
2971 {
2972 int argval = integral_argument (arg);
2973 if (argval == -1)
2974 error_at (loc, "unrecognized debug output level %qs", arg);
2975 else if (argval > 3)
2976 error_at (loc, "debug output level %qs is too high", arg);
2977 else
2978 opts->x_debug_info_level = (enum debug_info_levels) argval;
2979 }
2980 }
2981
2982 /* Arrange to dump core on error for diagnostic context DC. (The
2983 regular error message is still printed first, except in the case of
2984 abort ().) */
2985
2986 static void
2987 setup_core_dumping (diagnostic_context *dc)
2988 {
2989 #ifdef SIGABRT
2990 signal (SIGABRT, SIG_DFL);
2991 #endif
2992 #if defined(HAVE_SETRLIMIT)
2993 {
2994 struct rlimit rlim;
2995 if (getrlimit (RLIMIT_CORE, &rlim) != 0)
2996 fatal_error (input_location, "getting core file size maximum limit: %m");
2997 rlim.rlim_cur = rlim.rlim_max;
2998 if (setrlimit (RLIMIT_CORE, &rlim) != 0)
2999 fatal_error (input_location,
3000 "setting core file size limit to maximum: %m");
3001 }
3002 #endif
3003 diagnostic_abort_on_error (dc);
3004 }
3005
3006 /* Parse a -d<ARG> command line switch for OPTS, location LOC,
3007 diagnostic context DC. */
3008
3009 static void
3010 decode_d_option (const char *arg, struct gcc_options *opts,
3011 location_t loc, diagnostic_context *dc)
3012 {
3013 int c;
3014
3015 while (*arg)
3016 switch (c = *arg++)
3017 {
3018 case 'A':
3019 opts->x_flag_debug_asm = 1;
3020 break;
3021 case 'p':
3022 opts->x_flag_print_asm_name = 1;
3023 break;
3024 case 'P':
3025 opts->x_flag_dump_rtl_in_asm = 1;
3026 opts->x_flag_print_asm_name = 1;
3027 break;
3028 case 'x':
3029 opts->x_rtl_dump_and_exit = 1;
3030 break;
3031 case 'D': /* These are handled by the preprocessor. */
3032 case 'I':
3033 case 'M':
3034 case 'N':
3035 case 'U':
3036 break;
3037 case 'H':
3038 setup_core_dumping (dc);
3039 break;
3040 case 'a':
3041 opts->x_flag_dump_all_passed = true;
3042 break;
3043
3044 default:
3045 warning_at (loc, 0, "unrecognized gcc debugging option: %c", c);
3046 break;
3047 }
3048 }
3049
3050 /* Enable (or disable if VALUE is 0) a warning option ARG (language
3051 mask LANG_MASK, option handlers HANDLERS) as an error for option
3052 structures OPTS and OPTS_SET, diagnostic context DC (possibly
3053 NULL), location LOC. This is used by -Werror=. */
3054
3055 static void
3056 enable_warning_as_error (const char *arg, int value, unsigned int lang_mask,
3057 const struct cl_option_handlers *handlers,
3058 struct gcc_options *opts,
3059 struct gcc_options *opts_set,
3060 location_t loc, diagnostic_context *dc)
3061 {
3062 char *new_option;
3063 int option_index;
3064
3065 new_option = XNEWVEC (char, strlen (arg) + 2);
3066 new_option[0] = 'W';
3067 strcpy (new_option + 1, arg);
3068 option_index = find_opt (new_option, lang_mask);
3069 if (option_index == OPT_SPECIAL_unknown)
3070 {
3071 option_proposer op;
3072 const char *hint = op.suggest_option (new_option);
3073 if (hint)
3074 error_at (loc, "%<-W%serror=%s%>: no option %<-%s%>;"
3075 " did you mean %<-%s%>?", value ? "" : "no-",
3076 arg, new_option, hint);
3077 else
3078 error_at (loc, "%<-W%serror=%s%>: no option %<-%s%>",
3079 value ? "" : "no-", arg, new_option);
3080 }
3081 else if (!(cl_options[option_index].flags & CL_WARNING))
3082 error_at (loc, "%<-Werror=%s%>: %<-%s%> is not an option that "
3083 "controls warnings", arg, new_option);
3084 else
3085 {
3086 const diagnostic_t kind = value ? DK_ERROR : DK_WARNING;
3087 const char *arg = NULL;
3088
3089 if (cl_options[option_index].flags & CL_JOINED)
3090 arg = new_option + cl_options[option_index].opt_len;
3091 control_warning_option (option_index, (int) kind, arg, value,
3092 loc, lang_mask,
3093 handlers, opts, opts_set, dc);
3094 }
3095 free (new_option);
3096 }
3097
3098 /* Return malloced memory for the name of the option OPTION_INDEX
3099 which enabled a diagnostic (context CONTEXT), originally of type
3100 ORIG_DIAG_KIND but possibly converted to DIAG_KIND by options such
3101 as -Werror. */
3102
3103 char *
3104 option_name (diagnostic_context *context, int option_index,
3105 diagnostic_t orig_diag_kind, diagnostic_t diag_kind)
3106 {
3107 if (option_index)
3108 {
3109 /* A warning classified as an error. */
3110 if ((orig_diag_kind == DK_WARNING || orig_diag_kind == DK_PEDWARN)
3111 && diag_kind == DK_ERROR)
3112 return concat (cl_options[OPT_Werror_].opt_text,
3113 /* Skip over "-W". */
3114 cl_options[option_index].opt_text + 2,
3115 NULL);
3116 /* A warning with option. */
3117 else
3118 return xstrdup (cl_options[option_index].opt_text);
3119 }
3120 /* A warning without option classified as an error. */
3121 else if ((orig_diag_kind == DK_WARNING || orig_diag_kind == DK_PEDWARN
3122 || diag_kind == DK_WARNING)
3123 && context->warning_as_error_requested)
3124 return xstrdup (cl_options[OPT_Werror].opt_text);
3125 else
3126 return NULL;
3127 }
3128
3129 /* Get the page within the documentation for this option. */
3130
3131 static const char *
3132 get_option_html_page (int option_index)
3133 {
3134 const cl_option *cl_opt = &cl_options[option_index];
3135
3136 /* Analyzer options are on their own page. */
3137 if (strstr(cl_opt->opt_text, "analyzer-"))
3138 return "gcc/Static-Analyzer-Options.html";
3139
3140 #ifdef CL_Fortran
3141 if ((cl_opt->flags & CL_Fortran) != 0
3142 /* If it is option common to both C/C++ and Fortran, it is documented
3143 in gcc/ rather than gfortran/ docs. */
3144 && (cl_opt->flags & CL_C) == 0
3145 #ifdef CL_CXX
3146 && (cl_opt->flags & CL_CXX) == 0
3147 #endif
3148 )
3149 return "gfortran/Error-and-Warning-Options.html";
3150 #endif
3151
3152 return "gcc/Warning-Options.html";
3153 }
3154
3155 /* Return malloced memory for a URL describing the option OPTION_INDEX
3156 which enabled a diagnostic (context CONTEXT). */
3157
3158 char *
3159 get_option_url (diagnostic_context *, int option_index)
3160 {
3161 if (option_index)
3162 return concat (/* DOCUMENTATION_ROOT_URL should be supplied via -D by
3163 the Makefile (see --with-documentation-root-url), and
3164 should have a trailing slash. */
3165 DOCUMENTATION_ROOT_URL,
3166
3167 /* get_option_html_page will return something like
3168 "gcc/Warning-Options.html". */
3169 get_option_html_page (option_index),
3170
3171 /* Expect an anchor of the form "index-Wfoo" e.g.
3172 <a name="index-Wformat"></a>, and thus an id within
3173 the URL of "#index-Wformat". */
3174 "#index", cl_options[option_index].opt_text,
3175 NULL);
3176 else
3177 return NULL;
3178 }
3179
3180 #if CHECKING_P
3181
3182 namespace selftest {
3183
3184 /* Verify that get_option_html_page works as expected. */
3185
3186 static void
3187 test_get_option_html_page ()
3188 {
3189 ASSERT_STREQ (get_option_html_page (OPT_Wcpp), "gcc/Warning-Options.html");
3190 ASSERT_STREQ (get_option_html_page (OPT_Wanalyzer_double_free),
3191 "gcc/Static-Analyzer-Options.html");
3192 #ifdef CL_Fortran
3193 ASSERT_STREQ (get_option_html_page (OPT_Wline_truncation),
3194 "gfortran/Error-and-Warning-Options.html");
3195 #endif
3196 }
3197
3198 /* Run all of the selftests within this file. */
3199
3200 void
3201 opts_c_tests ()
3202 {
3203 test_get_option_html_page ();
3204 }
3205
3206 } // namespace selftest
3207
3208 #endif /* #if CHECKING_P */