opts.c 59.3 KB
Newer Older
1
/* Command line option handling.
2
   Copyright (C) 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011
3
   Free Software Foundation, Inc.
4 5 6 7 8 9
   Contributed by Neil Booth.

This file is part of GCC.

GCC is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free
10
Software Foundation; either version 3, or (at your option) any later
11 12 13 14 15 16 17 18
version.

GCC is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
for more details.

You should have received a copy of the GNU General Public License
19 20
along with GCC; see the file COPYING3.  If not see
<http://www.gnu.org/licenses/>.  */
21 22 23

#include "config.h"
#include "system.h"
Neil Booth committed
24
#include "intl.h"
25
#include "coretypes.h"
26 27 28
#include "tm.h" /* Needed by rtl.h and used for STACK_CHECK_BUILTIN,
		   STACK_CHECK_STATIC_BUILTIN, DEFAULT_GDB_EXTENSIONS,
		   DWARF2_DEBUGGING_INFO and DBX_DEBUGGING_INFO.  */
29
#include "rtl.h" /* Needed by insn-attr.h.  */
30
#include "opts.h"
31 32
#include "options.h"
#include "flags.h"
Neil Booth committed
33
#include "params.h"
Neil Booth committed
34
#include "diagnostic.h"
35
#include "opts-diagnostic.h"
36
#include "insn-attr.h"		/* For INSN_SCHEDULING and DELAY_SLOTS.  */
37
#include "common/common-target.h"
38

39 40 41 42 43 44 45 46
/* Parse the -femit-struct-debug-detailed option value
   and set the flag variables. */

#define MATCH( prefix, string ) \
  ((strncmp (prefix, string, sizeof prefix - 1) == 0) \
   ? ((string += sizeof prefix - 1), 1) : 0)

void
47 48
set_struct_debug_option (struct gcc_options *opts, location_t loc,
			 const char *spec)
49 50
{
  /* various labels for comparison */
51 52 53 54
  static const char dfn_lbl[] = "dfn:", dir_lbl[] = "dir:", ind_lbl[] = "ind:";
  static const char ord_lbl[] = "ord:", gen_lbl[] = "gen:";
  static const char none_lbl[] = "none", any_lbl[] = "any";
  static const char base_lbl[] = "base", sys_lbl[] = "sys";
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84

  enum debug_struct_file files = DINFO_STRUCT_FILE_ANY;
  /* Default is to apply to as much as possible. */
  enum debug_info_usage usage = DINFO_USAGE_NUM_ENUMS;
  int ord = 1, gen = 1;

  /* What usage? */
  if (MATCH (dfn_lbl, spec))
    usage = DINFO_USAGE_DFN;
  else if (MATCH (dir_lbl, spec))
    usage = DINFO_USAGE_DIR_USE;
  else if (MATCH (ind_lbl, spec))
    usage = DINFO_USAGE_IND_USE;

  /* Generics or not? */
  if (MATCH (ord_lbl, spec))
    gen = 0;
  else if (MATCH (gen_lbl, spec))
    ord = 0;

  /* What allowable environment? */
  if (MATCH (none_lbl, spec))
    files = DINFO_STRUCT_FILE_NONE;
  else if (MATCH (any_lbl, spec))
    files = DINFO_STRUCT_FILE_ANY;
  else if (MATCH (sys_lbl, spec))
    files = DINFO_STRUCT_FILE_SYS;
  else if (MATCH (base_lbl, spec))
    files = DINFO_STRUCT_FILE_BASE;
  else
85 86 87 88
    error_at (loc,
	      "argument %qs to %<-femit-struct-debug-detailed%> "
	      "not recognized",
	      spec);
89 90 91 92 93 94

  /* Effect the specification. */
  if (usage == DINFO_USAGE_NUM_ENUMS)
    {
      if (ord)
        {
95 96 97
          opts->x_debug_struct_ordinary[DINFO_USAGE_DFN] = files;
          opts->x_debug_struct_ordinary[DINFO_USAGE_DIR_USE] = files;
          opts->x_debug_struct_ordinary[DINFO_USAGE_IND_USE] = files;
98 99 100
        }
      if (gen)
        {
101 102 103
          opts->x_debug_struct_generic[DINFO_USAGE_DFN] = files;
          opts->x_debug_struct_generic[DINFO_USAGE_DIR_USE] = files;
          opts->x_debug_struct_generic[DINFO_USAGE_IND_USE] = files;
104 105 106 107 108
        }
    }
  else
    {
      if (ord)
109
        opts->x_debug_struct_ordinary[usage] = files;
110
      if (gen)
111
        opts->x_debug_struct_generic[usage] = files;
112 113 114
    }

  if (*spec == ',')
115
    set_struct_debug_option (opts, loc, spec+1);
116 117 118 119 120
  else
    {
      /* No more -femit-struct-debug-detailed specifications.
         Do final checks. */
      if (*spec != '\0')
121 122 123
	error_at (loc,
		  "argument %qs to %<-femit-struct-debug-detailed%> unknown",
		  spec);
124 125 126 127
      if (opts->x_debug_struct_ordinary[DINFO_USAGE_DIR_USE]
		< opts->x_debug_struct_ordinary[DINFO_USAGE_IND_USE]
	  || opts->x_debug_struct_generic[DINFO_USAGE_DIR_USE]
		< opts->x_debug_struct_generic[DINFO_USAGE_IND_USE])
128 129 130 131
	error_at (loc,
		  "%<-femit-struct-debug-detailed=dir:...%> must allow "
		  "at least as much as "
		  "%<-femit-struct-debug-detailed=ind:...%>");
132 133 134
    }
}

135 136 137 138 139 140 141 142 143 144 145 146 147
/* Handle -ftree-vectorizer-verbose=VAL for options OPTS.  */

static void
vect_set_verbosity_level (struct gcc_options *opts, int val)
{
   if (val < MAX_VERBOSITY_LEVEL)
     opts->x_user_vect_verbosity_level = (enum vect_verbosity_levels) val;
   else
     opts->x_user_vect_verbosity_level
      = (enum vect_verbosity_levels) (MAX_VERBOSITY_LEVEL - 1);
}


148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
/* Strip off a legitimate source ending from the input string NAME of
   length LEN.  Rather than having to know the names used by all of
   our front ends, we strip off an ending of a period followed by
   up to five characters.  (Java uses ".class".)  */

void
strip_off_ending (char *name, int len)
{
  int i;
  for (i = 2; i < 6 && len > i; i++)
    {
      if (name[len - i] == '.')
	{
	  name[len - i] = '\0';
	  break;
	}
    }
}

167 168
/* Find the base name of a path, stripping off both directories and
   a single final extension. */
169
int
170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
base_of_path (const char *path, const char **base_out)
{
  const char *base = path;
  const char *dot = 0;
  const char *p = path;
  char c = *p;
  while (c)
    {
      if (IS_DIR_SEPARATOR(c))
        {
          base = p + 1;
          dot = 0;
        }
      else if (c == '.')
        dot = p;
      c = *++p;
    }
  if (!dot)
    dot = p;
  *base_out = base;
  return dot - base;
}

193 194 195
/* What to print when a switch has no documentation.  */
static const char undocumented_msg[] = N_("This switch lacks documentation");

196 197 198 199
typedef char *char_p; /* For DEF_VEC_P.  */
DEF_VEC_P(char_p);
DEF_VEC_ALLOC_P(char_p,heap);

200
static void handle_param (struct gcc_options *opts,
201 202
			  struct gcc_options *opts_set, location_t loc,
			  const char *carg);
203
static void set_debug_level (enum debug_info_type type, int extended,
204
			     const char *arg, struct gcc_options *opts,
205 206
			     struct gcc_options *opts_set,
			     location_t loc);
207
static void set_fast_math_flags (struct gcc_options *opts, int set);
208 209
static void decode_d_option (const char *arg, struct gcc_options *opts,
			     location_t loc, diagnostic_context *dc);
210 211
static void set_unsafe_math_optimizations_flags (struct gcc_options *opts,
						 int set);
212 213 214
static void enable_warning_as_error (const char *arg, int value,
				     unsigned int lang_mask,
				     const struct cl_option_handlers *handlers,
215 216
				     struct gcc_options *opts,
				     struct gcc_options *opts_set,
217 218
				     location_t loc,
				     diagnostic_context *dc);
219

220 221
/* Handle a back-end option; arguments and return value as for
   handle_option.  */
222

223
bool
224
target_handle_option (struct gcc_options *opts,
225
		      struct gcc_options *opts_set,
226
		      const struct cl_decoded_option *decoded,
227
		      unsigned int lang_mask ATTRIBUTE_UNUSED, int kind,
228
		      location_t loc,
229 230
		      const struct cl_option_handlers *handlers ATTRIBUTE_UNUSED,
		      diagnostic_context *dc)
231
{
232
  gcc_assert (dc == global_dc);
233
  gcc_assert (kind == DK_UNSPECIFIED);
234
  return targetm_common.handle_option (opts, opts_set, decoded, loc);
235
}
236

237
/* Add comma-separated strings to a char_p vector.  */
238 239

static void
240
add_comma_separated_to_vector (void **pvec, const char *arg)
241 242 243 244 245
{
  char *tmp;
  char *r;
  char *w;
  char *token_start;
246
  VEC(char_p,heap) *vec = (VEC(char_p,heap) *) *pvec;
247 248 249 250 251 252 253 254 255 256 257 258 259 260

  /* We never free this string.  */
  tmp = xstrdup (arg);

  r = tmp;
  w = tmp;
  token_start = tmp;

  while (*r != '\0')
    {
      if (*r == ',')
	{
	  *w++ = '\0';
	  ++r;
261
	  VEC_safe_push (char_p, heap, vec, token_start);
262 263 264 265 266 267 268 269 270 271 272
	  token_start = w;
	}
      if (*r == '\\' && r[1] == ',')
	{
	  *w++ = ',';
	  r += 2;
	}
      else
	*w++ = *r++;
    }
  if (*token_start != '\0')
273
    VEC_safe_push (char_p, heap, vec, token_start);
274

275
  *pvec = vec;
276 277
}

278 279 280 281 282
/* Initialize OPTS and OPTS_SET before using them in parsing options.  */

void
init_options_struct (struct gcc_options *opts, struct gcc_options *opts_set)
{
283 284
  size_t num_params = get_num_compiler_params ();

285 286 287
  *opts = global_options_init;
  memset (opts_set, 0, sizeof (*opts_set));

288 289 290 291
  opts->x_param_values = XNEWVEC (int, num_params);
  opts_set->x_param_values = XCNEWVEC (int, num_params);
  init_param_values (opts->x_param_values);

292 293 294 295 296 297
  /* Initialize whether `char' is signed.  */
  opts->x_flag_signed_char = DEFAULT_SIGNED_CHAR;
  /* Set this to a special "uninitialized" value.  The actual default
     is set after target options have been processed.  */
  opts->x_flag_short_enums = 2;

298
  /* Initialize target_flags before default_options_optimization
299
     so the latter can modify it.  */
300
  opts->x_target_flags = targetm_common.default_target_flags;
301 302

  /* Some targets have ABI-specified unwind tables.  */
303
  opts->x_flag_unwind_tables = targetm_common.unwind_tables_default;
304 305

  /* Some targets have other target-specific initialization.  */
306
  targetm_common.option_init_struct (opts);
307 308
}

309 310
/* If indicated by the optimization level LEVEL (-Os if SIZE is set,
   -Ofast if FAST is set), apply the option DEFAULT_OPT to OPTS and
311 312
   OPTS_SET, diagnostic context DC, location LOC, with language mask
   LANG_MASK and option handlers HANDLERS.  */
313 314 315 316 317 318 319 320

static void
maybe_default_option (struct gcc_options *opts,
		      struct gcc_options *opts_set,
		      const struct default_options *default_opt,
		      int level, bool size, bool fast,
		      unsigned int lang_mask,
		      const struct cl_option_handlers *handlers,
321
		      location_t loc,
322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381
		      diagnostic_context *dc)
{
  const struct cl_option *option = &cl_options[default_opt->opt_index];
  bool enabled;

  if (size)
    gcc_assert (level == 2);
  if (fast)
    gcc_assert (level == 3);

  switch (default_opt->levels)
    {
    case OPT_LEVELS_ALL:
      enabled = true;
      break;

    case OPT_LEVELS_0_ONLY:
      enabled = (level == 0);
      break;

    case OPT_LEVELS_1_PLUS:
      enabled = (level >= 1);
      break;

    case OPT_LEVELS_1_PLUS_SPEED_ONLY:
      enabled = (level >= 1 && !size);
      break;

    case OPT_LEVELS_2_PLUS:
      enabled = (level >= 2);
      break;

    case OPT_LEVELS_2_PLUS_SPEED_ONLY:
      enabled = (level >= 2 && !size);
      break;

    case OPT_LEVELS_3_PLUS:
      enabled = (level >= 3);
      break;

    case OPT_LEVELS_3_PLUS_AND_SIZE:
      enabled = (level >= 3 || size);
      break;

    case OPT_LEVELS_SIZE:
      enabled = size;
      break;

    case OPT_LEVELS_FAST:
      enabled = fast;
      break;

    case OPT_LEVELS_NONE:
    default:
      gcc_unreachable ();
    }

  if (enabled)
    handle_generated_option (opts, opts_set, default_opt->opt_index,
			     default_opt->arg, default_opt->value,
382 383
			     lang_mask, DK_UNSPECIFIED, loc,
			     handlers, dc);
384
  else if (default_opt->arg == NULL
385
	   && !option->cl_reject_negative)
386 387
    handle_generated_option (opts, opts_set, default_opt->opt_index,
			     default_opt->arg, !default_opt->value,
388 389
			     lang_mask, DK_UNSPECIFIED, loc,
			     handlers, dc);
390 391 392 393
}

/* As indicated by the optimization level LEVEL (-Os if SIZE is set,
   -Ofast if FAST is set), apply the options in array DEFAULT_OPTS to
394 395
   OPTS and OPTS_SET, diagnostic context DC, location LOC, with
   language mask LANG_MASK and option handlers HANDLERS.  */
396 397 398 399 400 401 402 403

static void
maybe_default_options (struct gcc_options *opts,
		       struct gcc_options *opts_set,
		       const struct default_options *default_opts,
		       int level, bool size, bool fast,
		       unsigned int lang_mask,
		       const struct cl_option_handlers *handlers,
404
		       location_t loc,
405 406 407 408 409 410
		       diagnostic_context *dc)
{
  size_t i;

  for (i = 0; default_opts[i].levels != OPT_LEVELS_NONE; i++)
    maybe_default_option (opts, opts_set, &default_opts[i],
411
			  level, size, fast, lang_mask, handlers, loc, dc);
412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445
}

/* Table of options enabled by default at different levels.  */

static const struct default_options default_options_table[] =
  {
    /* -O1 optimizations.  */
    { OPT_LEVELS_1_PLUS, OPT_fdefer_pop, NULL, 1 },
#ifdef DELAY_SLOTS
    { OPT_LEVELS_1_PLUS, OPT_fdelayed_branch, NULL, 1 },
#endif
    { OPT_LEVELS_1_PLUS, OPT_fguess_branch_probability, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fcprop_registers, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fforward_propagate, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fif_conversion, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fif_conversion2, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fipa_pure_const, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fipa_reference, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fipa_profile, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fmerge_constants, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fsplit_wide_types, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_ccp, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_bit_ccp, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_dce, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_dominator_opts, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_dse, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_ter, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_sra, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_copyrename, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_fre, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_copy_prop, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_sink, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_ftree_ch, NULL, 1 },
    { OPT_LEVELS_1_PLUS, OPT_fcombine_stack_adjustments, NULL, 1 },
446
    { OPT_LEVELS_1_PLUS, OPT_fcompare_elim, NULL, 1 },
447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475

    /* -O2 optimizations.  */
    { OPT_LEVELS_2_PLUS, OPT_finline_small_functions, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_findirect_inlining, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fpartial_inlining, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fthread_jumps, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fcrossjumping, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_foptimize_sibling_calls, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fcse_follow_jumps, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fgcse, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fexpensive_optimizations, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_frerun_cse_after_loop, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fcaller_saves, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fpeephole2, NULL, 1 },
#ifdef INSN_SCHEDULING
  /* Only run the pre-regalloc scheduling pass if optimizing for speed.  */
    { OPT_LEVELS_2_PLUS_SPEED_ONLY, OPT_fschedule_insns, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fschedule_insns2, NULL, 1 },
#endif
    { OPT_LEVELS_2_PLUS, OPT_fregmove, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fstrict_aliasing, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fstrict_overflow, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_freorder_blocks, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_freorder_functions, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_ftree_vrp, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_ftree_builtin_call_dce, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_ftree_pre, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_ftree_switch_conversion, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_fipa_cp, NULL, 1 },
476
    { OPT_LEVELS_2_PLUS, OPT_fdevirtualize, NULL, 1 },
477 478 479 480 481 482 483 484 485 486 487 488
    { OPT_LEVELS_2_PLUS, OPT_fipa_sra, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_falign_loops, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_falign_jumps, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_falign_labels, NULL, 1 },
    { OPT_LEVELS_2_PLUS, OPT_falign_functions, NULL, 1 },

    /* -O3 optimizations.  */
    { OPT_LEVELS_3_PLUS, OPT_ftree_loop_distribute_patterns, NULL, 1 },
    { OPT_LEVELS_3_PLUS, OPT_fpredictive_commoning, NULL, 1 },
    /* Inlining of functions reducing size is a good idea with -Os
       regardless of them being declared inline.  */
    { OPT_LEVELS_3_PLUS_AND_SIZE, OPT_finline_functions, NULL, 1 },
489
    { OPT_LEVELS_1_PLUS, OPT_finline_functions_called_once, NULL, 1 },
490 491 492 493 494 495 496 497 498 499 500
    { OPT_LEVELS_3_PLUS, OPT_funswitch_loops, NULL, 1 },
    { OPT_LEVELS_3_PLUS, OPT_fgcse_after_reload, NULL, 1 },
    { OPT_LEVELS_3_PLUS, OPT_ftree_vectorize, NULL, 1 },
    { OPT_LEVELS_3_PLUS, OPT_fipa_cp_clone, NULL, 1 },

    /* -Ofast adds optimizations to -O3.  */
    { OPT_LEVELS_FAST, OPT_ffast_math, NULL, 1 },

    { OPT_LEVELS_NONE, 0, NULL, 0 }
  };

501 502
/* Default the options in OPTS and OPTS_SET based on the optimization
   settings in DECODED_OPTIONS and DECODED_OPTIONS_COUNT.  */
503
void
504 505 506
default_options_optimization (struct gcc_options *opts,
			      struct gcc_options *opts_set,
			      struct cl_decoded_option *decoded_options,
507
			      unsigned int decoded_options_count,
508
			      location_t loc,
509 510 511
			      unsigned int lang_mask,
			      const struct cl_option_handlers *handlers,
			      diagnostic_context *dc)
512 513
{
  unsigned int i;
514
  int opt2;
Neil Booth committed
515 516 517

  /* Scan to see what optimization level has been specified.  That will
     determine the default value of many flags.  */
518
  for (i = 1; i < decoded_options_count; i++)
Neil Booth committed
519
    {
520
      struct cl_decoded_option *opt = &decoded_options[i];
521
      switch (opt->opt_index)
Neil Booth committed
522
	{
523 524
	case OPT_O:
	  if (*opt->arg == '\0')
525
	    {
526 527
	      opts->x_optimize = 1;
	      opts->x_optimize_size = 0;
528
	      opts->x_optimize_fast = 0;
Neil Booth committed
529 530 531
	    }
	  else
	    {
532 533
	      const int optimize_val = integral_argument (opt->arg);
	      if (optimize_val == -1)
534 535 536
		error_at (loc,
			  "argument to %qs should be a non-negative integer",
			  "-O");
537
	      else
Neil Booth committed
538
		{
539 540 541 542
		  opts->x_optimize = optimize_val;
		  if ((unsigned int) opts->x_optimize > 255)
		    opts->x_optimize = 255;
		  opts->x_optimize_size = 0;
543
		  opts->x_optimize_fast = 0;
Neil Booth committed
544 545
		}
	    }
546 547 548
	  break;

	case OPT_Os:
549
	  opts->x_optimize_size = 1;
550 551

	  /* Optimizing for size forces optimize to be 2.  */
552
	  opts->x_optimize = 2;
553
	  opts->x_optimize_fast = 0;
554 555 556 557
	  break;

	case OPT_Ofast:
	  /* -Ofast only adds flags to -O3.  */
558 559
	  opts->x_optimize_size = 0;
	  opts->x_optimize = 3;
560
	  opts->x_optimize_fast = 1;
561 562 563 564 565
	  break;

	default:
	  /* Ignore other options in this prescan.  */
	  break;
Neil Booth committed
566 567
	}
    }
H.J. Lu committed
568

569 570
  maybe_default_options (opts, opts_set, default_options_table,
			 opts->x_optimize, opts->x_optimize_size,
571
			 opts->x_optimize_fast, lang_mask, handlers, loc, dc);
572 573 574

  /* -O2 param settings.  */
  opt2 = (opts->x_optimize >= 2);
575

576
  /* Track fields in field-sensitive alias analysis.  */
577 578 579 580
  maybe_set_param_value
    (PARAM_MAX_FIELDS_FOR_FIELD_SENSITIVE,
     opt2 ? 100 : default_param_value (PARAM_MAX_FIELDS_FOR_FIELD_SENSITIVE),
     opts->x_param_values, opts_set->x_param_values);
581

582
  /* For -O1 only do loop invariant motion for very small loops.  */
583 584 585 586
  maybe_set_param_value
    (PARAM_LOOP_INVARIANT_MAX_BBS_IN_LOOP,
     opt2 ? default_param_value (PARAM_LOOP_INVARIANT_MAX_BBS_IN_LOOP) : 1000,
     opts->x_param_values, opts_set->x_param_values);
587

588 589 590 591
  if (opts->x_optimize_size)
    /* We want to crossjump as much as possible.  */
    maybe_set_param_value (PARAM_MIN_CROSSJUMP_INSNS, 1,
			   opts->x_param_values, opts_set->x_param_values);
592
  else
593
    maybe_set_param_value (PARAM_MIN_CROSSJUMP_INSNS,
594 595
			   default_param_value (PARAM_MIN_CROSSJUMP_INSNS),
			   opts->x_param_values, opts_set->x_param_values);
596

597
  /* Allow default optimizations to be specified on a per-machine basis.  */
598
  maybe_default_options (opts, opts_set,
599
			 targetm_common.option_optimization_table,
600
			 opts->x_optimize, opts->x_optimize_size,
601
			 opts->x_optimize_fast, lang_mask, handlers, loc, dc);
602 603
}

604 605
/* After all options at LOC have been read into OPTS and OPTS_SET,
   finalize settings of those options and diagnose incompatible
606
   combinations.  */
607
void
608 609
finish_options (struct gcc_options *opts, struct gcc_options *opts_set,
		location_t loc)
610 611 612
{
  enum unwind_info_type ui_except;

613
  if (opts->x_dump_base_name && ! IS_ABSOLUTE_PATH (opts->x_dump_base_name))
614
    {
615 616 617 618 619 620 621 622
      /* First try to make OPTS->X_DUMP_BASE_NAME relative to the
	 OPTS->X_DUMP_DIR_NAME directory.  Then try to make
	 OPTS->X_DUMP_BASE_NAME relative to the OPTS->X_AUX_BASE_NAME
	 directory, typically the directory to contain the object
	 file.  */
      if (opts->x_dump_dir_name)
	opts->x_dump_base_name = concat (opts->x_dump_dir_name,
					 opts->x_dump_base_name, NULL);
623 624
      else if (opts->x_aux_base_name
	       && strcmp (opts->x_aux_base_name, HOST_BIT_BUCKET) != 0)
625
	{
626 627
	  const char *aux_base;

628 629
	  base_of_path (opts->x_aux_base_name, &aux_base);
	  if (opts->x_aux_base_name != aux_base)
630
	    {
631
	      int dir_len = aux_base - opts->x_aux_base_name;
632
	      char *new_dump_base_name =
633
		XNEWVEC (char, strlen (opts->x_dump_base_name) + dir_len + 1);
634

635 636 637 638 639
	      /* Copy directory component from OPTS->X_AUX_BASE_NAME.  */
	      memcpy (new_dump_base_name, opts->x_aux_base_name, dir_len);
	      /* Append existing OPTS->X_DUMP_BASE_NAME.  */
	      strcpy (new_dump_base_name + dir_len, opts->x_dump_base_name);
	      opts->x_dump_base_name = new_dump_base_name;
640
	    }
641 642 643
	}
    }

644 645
  /* Handle related options for unit-at-a-time, toplevel-reorder, and
     section-anchors.  */
646
  if (!opts->x_flag_unit_at_a_time)
647
    {
648
      if (opts->x_flag_section_anchors && opts_set->x_flag_section_anchors)
649 650
	error_at (loc, "section anchors must be disabled when unit-at-a-time "
		  "is disabled");
651 652
      opts->x_flag_section_anchors = 0;
      if (opts->x_flag_toplevel_reorder == 1)
653 654
	error_at (loc, "toplevel reorder must be disabled when unit-at-a-time "
		  "is disabled");
655
      opts->x_flag_toplevel_reorder = 0;
656
    }
657 658

  /* -Wmissing-noreturn is alias for -Wsuggest-attribute=noreturn.  */
659 660
  if (opts->x_warn_missing_noreturn)
    opts->x_warn_suggest_attribute_noreturn = true;
661
    
662 663 664
  /* Unless the user has asked for section anchors, we disable toplevel
     reordering at -O0 to disable transformations that might be surprising
     to end users and to get -fno-toplevel-reorder tested.  */
665
  if (!opts->x_optimize
666 667
      && opts->x_flag_toplevel_reorder == 2
      && !(opts->x_flag_section_anchors && opts_set->x_flag_section_anchors))
668
    {
669 670
      opts->x_flag_toplevel_reorder = 0;
      opts->x_flag_section_anchors = 0;
671
    }
672
  if (!opts->x_flag_toplevel_reorder)
673
    {
674
      if (opts->x_flag_section_anchors && opts_set->x_flag_section_anchors)
675 676
	error_at (loc, "section anchors must be disabled when toplevel reorder"
		  " is disabled");
677
      opts->x_flag_section_anchors = 0;
678 679
    }

680
  if (!opts->x_flag_opts_finished)
681
    {
682 683 684 685
      if (opts->x_flag_pie)
	opts->x_flag_pic = opts->x_flag_pie;
      if (opts->x_flag_pic && !opts->x_flag_pie)
	opts->x_flag_shlib = 1;
686
      opts->x_flag_opts_finished = true;
687
    }
Neil Booth committed
688

689
  if (opts->x_optimize == 0)
Neil Booth committed
690 691 692
    {
      /* Inlining does not work if not optimizing,
	 so force it not to be done.  */
693 694
      opts->x_warn_inline = 0;
      opts->x_flag_no_inline = 1;
Neil Booth committed
695 696
    }

697 698
  /* The optimization to partition hot and cold basic blocks into separate
     sections of the .o and executable files does not work (currently)
699
     with exception handling.  This is because there is no support for
700 701
     generating unwind info.  If opts->x_flag_exceptions is turned on
     we need to turn off the partitioning optimization.  */
702

703
  ui_except = targetm_common.except_unwind_info (opts);
704

705 706
  if (opts->x_flag_exceptions
      && opts->x_flag_reorder_blocks_and_partition
707
      && (ui_except == UI_SJLJ || ui_except == UI_TARGET))
708
    {
709
      inform (loc,
710 711
	      "-freorder-blocks-and-partition does not work "
	      "with exceptions on this architecture");
712 713
      opts->x_flag_reorder_blocks_and_partition = 0;
      opts->x_flag_reorder_blocks = 1;
714
    }
715

716 717 718
  /* If user requested unwind info, then turn off the partitioning
     optimization.  */

719
  if (opts->x_flag_unwind_tables
720
      && !targetm_common.unwind_tables_default
721
      && opts->x_flag_reorder_blocks_and_partition
722
      && (ui_except == UI_SJLJ || ui_except == UI_TARGET))
723
    {
724
      inform (loc,
725 726
	      "-freorder-blocks-and-partition does not support "
	      "unwind info on this architecture");
727 728
      opts->x_flag_reorder_blocks_and_partition = 0;
      opts->x_flag_reorder_blocks = 1;
729 730 731 732 733 734
    }

  /* If the target requested unwind info, then turn off the partitioning
     optimization with a different message.  Likewise, if the target does not
     support named sections.  */

735
  if (opts->x_flag_reorder_blocks_and_partition
736
      && (!targetm_common.have_named_sections
737
	  || (opts->x_flag_unwind_tables
738
	      && targetm_common.unwind_tables_default
739
	      && (ui_except == UI_SJLJ || ui_except == UI_TARGET))))
740
    {
741
      inform (loc,
742 743
	      "-freorder-blocks-and-partition does not work "
	      "on this architecture");
744 745
      opts->x_flag_reorder_blocks_and_partition = 0;
      opts->x_flag_reorder_blocks = 1;
746
    }
747

748 749 750 751
  if (opts->x_flag_reorder_blocks_and_partition
      && !opts_set->x_flag_reorder_functions)
    opts->x_flag_reorder_functions = 1;

752 753
  /* Pipelining of outer loops is only possible when general pipelining
     capabilities are requested.  */
754 755
  if (!opts->x_flag_sel_sched_pipelining)
    opts->x_flag_sel_sched_pipelining_outer_loops = 0;
756

757
  if (opts->x_flag_conserve_stack)
758
    {
759 760 761 762
      maybe_set_param_value (PARAM_LARGE_STACK_FRAME, 100,
			     opts->x_param_values, opts_set->x_param_values);
      maybe_set_param_value (PARAM_STACK_FRAME_GROWTH, 40,
			     opts->x_param_values, opts_set->x_param_values);
763
    }
764
  if (opts->x_flag_wpa || opts->x_flag_ltrans)
765 766
    {
      /* These passes are not WHOPR compatible yet.  */
767
      opts->x_flag_ipa_pta = 0;
768
    }
769

770
  if (opts->x_flag_lto)
771 772
    {
#ifdef ENABLE_LTO
773
      opts->x_flag_generate_lto = 1;
774 775 776 777

      /* When generating IL, do not operate in whole-program mode.
	 Otherwise, symbols will be privatized too early, causing link
	 errors later.  */
778
      opts->x_flag_whole_program = 0;
779
#else
780
      error_at (loc, "LTO support has not been enabled in this configuration");
781 782
#endif
    }
783 784
  if ((opts->x_flag_lto_partition_balanced != 0) + (opts->x_flag_lto_partition_1to1 != 0)
       + (opts->x_flag_lto_partition_none != 0) >= 1)
785
    {
786 787 788
      if ((opts->x_flag_lto_partition_balanced != 0)
	   + (opts->x_flag_lto_partition_1to1 != 0)
	   + (opts->x_flag_lto_partition_none != 0) > 1)
789
	error_at (loc, "only one -flto-partition value can be specified");
790
    }
791

792
  /* We initialize opts->x_flag_split_stack to -1 so that targets can set a
793
     default value if they choose based on other options.  */
794 795 796
  if (opts->x_flag_split_stack == -1)
    opts->x_flag_split_stack = 0;
  else if (opts->x_flag_split_stack)
797
    {
798
      if (!targetm_common.supports_split_stack (true, opts))
799
	{
800 801
	  error_at (loc, "%<-fsplit-stack%> is not supported by "
		    "this compiler configuration");
802
	  opts->x_flag_split_stack = 0;
803 804
	}
    }
805 806 807 808 809 810

  /* Set PARAM_MAX_STORES_TO_SINK to 0 if either vectorization or if-conversion
     is disabled.  */
  if (!opts->x_flag_tree_vectorize || !opts->x_flag_tree_loop_if_convert)
    maybe_set_param_value (PARAM_MAX_STORES_TO_SINK, 0,
                           opts->x_param_values, opts_set->x_param_values);
811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835

  /* This replaces set_Wunused.  */
  if (opts->x_warn_unused_function == -1)
    opts->x_warn_unused_function = opts->x_warn_unused;
  if (opts->x_warn_unused_label == -1)
    opts->x_warn_unused_label = opts->x_warn_unused;
  /* Wunused-parameter is enabled if both -Wunused -Wextra are enabled.  */
  if (opts->x_warn_unused_parameter == -1)
    opts->x_warn_unused_parameter = (opts->x_warn_unused
				     && opts->x_extra_warnings);
  if (opts->x_warn_unused_variable == -1)
    opts->x_warn_unused_variable = opts->x_warn_unused;
  /* Wunused-but-set-parameter is enabled if both -Wunused -Wextra are
     enabled.  */
  if (opts->x_warn_unused_but_set_parameter == -1)
    opts->x_warn_unused_but_set_parameter = (opts->x_warn_unused
					     && opts->x_extra_warnings);
  if (opts->x_warn_unused_but_set_variable == -1)
    opts->x_warn_unused_but_set_variable = opts->x_warn_unused;
  if (opts->x_warn_unused_value == -1)
    opts->x_warn_unused_value = opts->x_warn_unused;

  /* This replaces set_Wextra.  */
  if (opts->x_warn_uninitialized == -1)
    opts->x_warn_uninitialized = opts->x_extra_warnings;
Neil Booth committed
836 837
}

838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891
#define LEFT_COLUMN	27

/* Output ITEM, of length ITEM_WIDTH, in the left column,
   followed by word-wrapped HELP in a second column.  */
static void
wrap_help (const char *help,
	   const char *item,
	   unsigned int item_width,
	   unsigned int columns)
{
  unsigned int col_width = LEFT_COLUMN;
  unsigned int remaining, room, len;

  remaining = strlen (help);

  do
    {
      room = columns - 3 - MAX (col_width, item_width);
      if (room > columns)
	room = 0;
      len = remaining;

      if (room < len)
	{
	  unsigned int i;

	  for (i = 0; help[i]; i++)
	    {
	      if (i >= room && len != remaining)
		break;
	      if (help[i] == ' ')
		len = i;
	      else if ((help[i] == '-' || help[i] == '/')
		       && help[i + 1] != ' '
		       && i > 0 && ISALPHA (help[i - 1]))
		len = i + 1;
	    }
	}

      printf( "  %-*.*s %.*s\n", col_width, item_width, item, len, help);
      item_width = 0;
      while (help[len] == ' ')
	len++;
      help += len;
      remaining -= len;
    }
  while (remaining);
}

/* Print help for a specific front-end, etc.  */
static void
print_filtered_help (unsigned int include_flags,
		     unsigned int exclude_flags,
		     unsigned int any_flags,
892
		     unsigned int columns,
893 894
		     struct gcc_options *opts,
		     unsigned int lang_mask)
895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923
{
  unsigned int i;
  const char *help;
  bool found = false;
  bool displayed = false;

  if (include_flags == CL_PARAMS)
    {
      for (i = 0; i < LAST_PARAM; i++)
	{
	  const char *param = compiler_params[i].option;

	  help = compiler_params[i].help;
	  if (help == NULL || *help == '\0')
	    {
	      if (exclude_flags & CL_UNDOCUMENTED)
		continue;
	      help = undocumented_msg;
	    }

	  /* Get the translation.  */
	  help = _(help);

	  wrap_help (help, param, strlen (param), columns);
	}
      putchar ('\n');
      return;
    }

924 925
  if (!opts->x_help_printed)
    opts->x_help_printed = XCNEWVAR (char, cl_options_count);
926

927 928 929
  if (!opts->x_help_enum_printed)
    opts->x_help_enum_printed = XCNEWVAR (char, cl_enums_count);

930 931
  for (i = 0; i < cl_options_count; i++)
    {
932
      char new_help[128];
933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948
      const struct cl_option *option = cl_options + i;
      unsigned int len;
      const char *opt;
      const char *tab;

      if (include_flags == 0
	  || ((option->flags & include_flags) != include_flags))
	{
	  if ((option->flags & any_flags) == 0)
	    continue;
	}

      /* Skip unwanted switches.  */
      if ((option->flags & exclude_flags) != 0)
	continue;

949 950 951 952 953 954
      /* The driver currently prints its own help text.  */
      if ((option->flags & CL_DRIVER) != 0
	  && (option->flags & (((1U << cl_lang_count) - 1)
			       | CL_COMMON | CL_TARGET)) == 0)
	continue;

955 956
      found = true;
      /* Skip switches that have already been printed.  */
957
      if (opts->x_help_printed[i])
958 959
	continue;

960
      opts->x_help_printed[i] = true;
961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991

      help = option->help;
      if (help == NULL)
	{
	  if (exclude_flags & CL_UNDOCUMENTED)
	    continue;
	  help = undocumented_msg;
	}

      /* Get the translation.  */
      help = _(help);

      /* Find the gap between the name of the
	 option and its descriptive text.  */
      tab = strchr (help, '\t');
      if (tab)
	{
	  len = tab - help;
	  opt = help;
	  help = tab + 1;
	}
      else
	{
	  opt = option->opt_text;
	  len = strlen (opt);
	}

      /* With the -Q option enabled we change the descriptive text associated
	 with an option to be an indication of its current setting.  */
      if (!quiet_flag)
	{
992
	  void *flag_var = option_flag_var (i, opts);
993

994 995 996 997 998
	  if (len < (LEFT_COLUMN + 2))
	    strcpy (new_help, "\t\t");
	  else
	    strcpy (new_help, "\t");

999 1000
	  if (flag_var != NULL
	      && option->var_type != CLVC_DEFER)
1001 1002 1003 1004 1005
	    {
	      if (option->flags & CL_JOINED)
		{
		  if (option->var_type == CLVC_STRING)
		    {
1006
		      if (* (const char **) flag_var != NULL)
1007 1008
			snprintf (new_help + strlen (new_help),
				  sizeof (new_help) - strlen (new_help),
1009
				  * (const char **) flag_var);
1010
		    }
1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024
		  else if (option->var_type == CLVC_ENUM)
		    {
		      const struct cl_enum *e = &cl_enums[option->var_enum];
		      int value;
		      const char *arg = NULL;

		      value = e->get (flag_var);
		      enum_value_to_arg (e->values, &arg, value, lang_mask);
		      if (arg == NULL)
			arg = _("[default]");
		      snprintf (new_help + strlen (new_help),
				sizeof (new_help) - strlen (new_help),
				arg);
		    }
1025 1026
		  else
		    sprintf (new_help + strlen (new_help),
1027
			     "%#x", * (int *) flag_var);
1028 1029
		}
	      else
1030
		strcat (new_help, option_enabled (i, opts)
1031 1032 1033 1034 1035 1036 1037 1038
			? _("[enabled]") : _("[disabled]"));
	    }

	  help = new_help;
	}

      wrap_help (help, opt, len, columns);
      displayed = true;
1039 1040 1041 1042

      if (option->var_type == CLVC_ENUM
	  && opts->x_help_enum_printed[option->var_enum] != 2)
	opts->x_help_enum_printed[option->var_enum] = 1;
1043 1044 1045
    }

  if (! found)
1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061
    {
      unsigned int langs = include_flags & CL_LANG_ALL;

      if (langs == 0)
	printf (_(" No options with the desired characteristics were found\n"));
      else
	{
	  unsigned int i;

	  /* PR 31349: Tell the user how to see all of the
	     options supported by a specific front end.  */
	  for (i = 0; (1U << i) < CL_LANG_ALL; i ++)
	    if ((1U << i) & langs)
	      printf (_(" None found.  Use --help=%s to show *all* the options supported by the %s front-end\n"),
		      lang_names[i], lang_names[i]);
	}
H.J. Lu committed
1062

1063
    }
1064 1065 1066 1067
  else if (! displayed)
    printf (_(" All options with the desired characteristics have already been displayed\n"));

  putchar ('\n');
1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105

  /* Print details of enumerated option arguments, if those
     enumerations have help text headings provided.  If no help text
     is provided, presume that the possible values are listed in the
     help text for the relevant options.  */
  for (i = 0; i < cl_enums_count; i++)
    {
      unsigned int j, pos;

      if (opts->x_help_enum_printed[i] != 1)
	continue;
      if (cl_enums[i].help == NULL)
	continue;
      printf ("  %s\n    ", _(cl_enums[i].help));
      pos = 4;
      for (j = 0; cl_enums[i].values[j].arg != NULL; j++)
	{
	  unsigned int len = strlen (cl_enums[i].values[j].arg);

	  if (pos > 4 && pos + 1 + len <= columns)
	    {
	      printf (" %s", cl_enums[i].values[j].arg);
	      pos += 1 + len;
	    }
	  else
	    {
	      if (pos > 4)
		{
		  printf ("\n    ");
		  pos = 4;
		}
	      printf ("%s", cl_enums[i].values[j].arg);
	      pos += len;
	    }
	}
      printf ("\n\n");
      opts->x_help_enum_printed[i] = 2;
    }
1106 1107 1108 1109 1110
}

/* Display help for a specified type of option.
   The options must have ALL of the INCLUDE_FLAGS set
   ANY of the flags in the ANY_FLAGS set
1111
   and NONE of the EXCLUDE_FLAGS set.  The current option state is in
1112
   OPTS; LANG_MASK is used for interpreting enumerated option state.  */
1113 1114 1115
static void
print_specific_help (unsigned int include_flags,
		     unsigned int exclude_flags,
1116
		     unsigned int any_flags,
1117 1118
		     struct gcc_options *opts,
		     unsigned int lang_mask)
1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131
{
  unsigned int all_langs_mask = (1U << cl_lang_count) - 1;
  const char * description = NULL;
  const char * descrip_extra = "";
  size_t i;
  unsigned int flag;

  /* Sanity check: Make sure that we do not have more
     languages than we have bits available to enumerate them.  */
  gcc_assert ((1U << cl_lang_count) < CL_MIN_OPTION_CLASS);

  /* If we have not done so already, obtain
     the desired maximum width of the output.  */
1132
  if (opts->x_help_columns == 0)
1133 1134 1135
    {
      const char *p;

1136
      p = getenv ("COLUMNS");
1137 1138 1139 1140 1141
      if (p != NULL)
	{
	  int value = atoi (p);

	  if (value > 0)
1142
	    opts->x_help_columns = value;
1143 1144
	}

1145
      if (opts->x_help_columns == 0)
1146
	/* Use a reasonable default.  */
1147
	opts->x_help_columns = 80;
1148 1149 1150 1151 1152 1153 1154 1155
    }

  /* Decide upon the title for the options that we are going to display.  */
  for (i = 0, flag = 1; flag <= CL_MAX_OPTION_CLASS; flag <<= 1, i ++)
    {
      switch (flag & include_flags)
	{
	case 0:
1156
	case CL_DRIVER:
1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176
	  break;

	case CL_TARGET:
	  description = _("The following options are target specific");
	  break;
	case CL_WARNING:
	  description = _("The following options control compiler warning messages");
	  break;
	case CL_OPTIMIZATION:
	  description = _("The following options control optimizations");
	  break;
	case CL_COMMON:
	  description = _("The following options are language-independent");
	  break;
	case CL_PARAMS:
	  description = _("The --param option recognizes the following as parameters");
	  break;
	default:
	  if (i >= cl_lang_count)
	    break;
1177
	  if (exclude_flags & all_langs_mask)
1178
	    description = _("The following options are specific to just the language ");
1179
	  else
1180
	    description = _("The following options are supported by the language ");
1181
	  descrip_extra = lang_names [i];
1182 1183 1184 1185 1186 1187 1188 1189
	  break;
	}
    }

  if (description == NULL)
    {
      if (any_flags == 0)
	{
1190
	  if (include_flags & CL_UNDOCUMENTED)
1191
	    description = _("The following options are not documented");
1192 1193 1194 1195
	  else if (include_flags & CL_SEPARATE)
	    description = _("The following options take separate arguments");
	  else if (include_flags & CL_JOINED)
	    description = _("The following options take joined arguments");
1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212
	  else
	    {
	      internal_error ("unrecognized include_flags 0x%x passed to print_specific_help",
			      include_flags);
	      return;
	    }
	}
      else
	{
	  if (any_flags & all_langs_mask)
	    description = _("The following options are language-related");
	  else
	    description = _("The following options are language-independent");
	}
    }

  printf ("%s%s:\n", description, descrip_extra);
1213
  print_filtered_help (include_flags, exclude_flags, any_flags,
1214
		       opts->x_help_columns, opts, lang_mask);
1215 1216
}

1217
/* Handle target- and language-independent options.  Return zero to
1218 1219
   generate an "unknown option" message.  Only options that need
   extra handling need to be listed here; if you simply want
1220
   DECODED->value assigned to a variable, it happens automatically.  */
1221

1222
bool
1223
common_handle_option (struct gcc_options *opts,
1224
		      struct gcc_options *opts_set,
1225
		      const struct cl_decoded_option *decoded,
1226
		      unsigned int lang_mask, int kind ATTRIBUTE_UNUSED,
1227
		      location_t loc,
1228 1229
		      const struct cl_option_handlers *handlers,
		      diagnostic_context *dc)
1230
{
1231 1232 1233
  size_t scode = decoded->opt_index;
  const char *arg = decoded->arg;
  int value = decoded->value;
1234 1235
  enum opt_code code = (enum opt_code) scode;

1236 1237
  gcc_assert (decoded->canonical_option_num_elements <= 2);

1238 1239
  switch (code)
    {
Neil Booth committed
1240
    case OPT__param:
1241
      handle_param (opts, opts_set, loc, arg);
Neil Booth committed
1242 1243
      break;

1244 1245 1246 1247 1248 1249
    case OPT__help:
      {
	unsigned int all_langs_mask = (1U << cl_lang_count) - 1;
	unsigned int undoc_mask;
	unsigned int i;

1250 1251 1252
	undoc_mask = ((opts->x_verbose_flag | opts->x_extra_warnings)
		      ? 0
		      : CL_UNDOCUMENTED);
1253 1254 1255
	/* First display any single language specific options.  */
	for (i = 0; i < cl_lang_count; i++)
	  print_specific_help
1256 1257
	    (1U << i, (all_langs_mask & (~ (1U << i))) | undoc_mask, 0, opts,
	     lang_mask);
1258
	/* Next display any multi language specific options.  */
1259
	print_specific_help (0, undoc_mask, all_langs_mask, opts, lang_mask);
1260 1261
	/* Then display any remaining, non-language options.  */
	for (i = CL_MIN_OPTION_CLASS; i <= CL_MAX_OPTION_CLASS; i <<= 1)
1262
	  if (i != CL_DRIVER)
1263
	    print_specific_help (i, undoc_mask, 0, opts, lang_mask);
1264
	opts->x_exit_after_options = true;
1265 1266 1267
	break;
      }

1268
    case OPT__target_help:
1269
      print_specific_help (CL_TARGET, CL_UNDOCUMENTED, 0, opts, lang_mask);
1270
      opts->x_exit_after_options = true;
1271 1272
      break;

1273 1274 1275 1276 1277 1278
    case OPT__help_:
      {
	const char * a = arg;
	unsigned int include_flags = 0;
	/* Note - by default we include undocumented options when listing
	   specific classes.  If you only want to see documented options
1279
	   then add ",^undocumented" to the --help= option.  E.g.:
1280 1281 1282 1283 1284 1285 1286

	   --help=target,^undocumented  */
	unsigned int exclude_flags = 0;

	/* Walk along the argument string, parsing each word in turn.
	   The format is:
	   arg = [^]{word}[,{arg}]
1287 1288
	   word = {optimizers|target|warnings|undocumented|
		   params|common|<language>}  */
1289 1290
	while (* a != 0)
	  {
1291
	    static const struct
1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302
	    {
	      const char * string;
	      unsigned int flag;
	    }
	    specifics[] =
	    {
	      { "optimizers", CL_OPTIMIZATION },
	      { "target", CL_TARGET },
	      { "warnings", CL_WARNING },
	      { "undocumented", CL_UNDOCUMENTED },
	      { "params", CL_PARAMS },
1303 1304
	      { "joined", CL_JOINED },
	      { "separate", CL_SEPARATE },
1305
	      { "common", CL_COMMON },
1306 1307 1308
	      { NULL, 0 }
	    };
	    unsigned int * pflags;
1309
	    const char * comma;
1310
	    unsigned int lang_flag, specific_flag;
1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326
	    unsigned int len;
	    unsigned int i;

	    if (* a == '^')
	      {
		++ a;
		pflags = & exclude_flags;
	      }
	    else
	      pflags = & include_flags;

	    comma = strchr (a, ',');
	    if (comma == NULL)
	      len = strlen (a);
	    else
	      len = comma - a;
1327 1328 1329 1330 1331
	    if (len == 0)
	      {
		a = comma + 1;
		continue;
	      }
1332

1333 1334
	    /* Check to see if the string matches an option class name.  */
	    for (i = 0, specific_flag = 0; specifics[i].string != NULL; i++)
1335 1336
	      if (strncasecmp (a, specifics[i].string, len) == 0)
		{
1337 1338 1339
		  specific_flag = specifics[i].flag;
		  break;
		}
1340

1341 1342 1343
	    /* Check to see if the string matches a language name.
	       Note - we rely upon the alpha-sorted nature of the entries in
	       the lang_names array, specifically that shorter names appear
1344
	       before their longer variants.  (i.e. C before C++).  That way
1345 1346 1347 1348 1349 1350
	       when we are attempting to match --help=c for example we will
	       match with C first and not C++.  */
	    for (i = 0, lang_flag = 0; i < cl_lang_count; i++)
	      if (strncasecmp (a, lang_names[i], len) == 0)
		{
		  lang_flag = 1U << i;
1351 1352 1353
		  break;
		}

1354
	    if (specific_flag != 0)
1355
	      {
1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367
		if (lang_flag == 0)
		  * pflags |= specific_flag;
		else
		  {
		    /* The option's argument matches both the start of a
		       language name and the start of an option class name.
		       We have a special case for when the user has
		       specified "--help=c", but otherwise we have to issue
		       a warning.  */
		    if (strncasecmp (a, "c", len) == 0)
		      * pflags |= lang_flag;
		    else
1368 1369 1370 1371
		      warning_at (loc, 0,
				  "--help argument %q.*s is ambiguous, "
				  "please be more specific",
				  len, a);
1372
		  }
1373
	      }
1374 1375 1376
	    else if (lang_flag != 0)
	      * pflags |= lang_flag;
	    else
1377 1378 1379
	      warning_at (loc, 0,
			  "unrecognized argument to --help= option: %q.*s",
			  len, a);
1380 1381 1382 1383 1384 1385 1386

	    if (comma == NULL)
	      break;
	    a = comma + 1;
	  }

	if (include_flags)
1387 1388
	  print_specific_help (include_flags, exclude_flags, 0, opts,
			       lang_mask);
1389
	opts->x_exit_after_options = true;
1390 1391 1392
	break;
      }

1393
    case OPT__version:
1394
      opts->x_exit_after_options = true;
1395 1396
      break;

Neil Booth committed
1397 1398
    case OPT_O:
    case OPT_Os:
1399
    case OPT_Ofast:
Neil Booth committed
1400 1401 1402
      /* Currently handled in a prescan.  */
      break;

1403
    case OPT_Werror_:
1404 1405
      enable_warning_as_error (arg, value, lang_mask, handlers,
			       opts, opts_set, loc, dc);
1406 1407
      break;

1408
    case OPT_Wlarger_than_:
1409 1410
      opts->x_larger_than_size = value;
      opts->x_warn_larger_than = value != -1;
1411 1412
      break;

1413
    case OPT_Wfatal_errors:
1414
      dc->fatal_errors = value;
1415 1416
      break;

1417
    case OPT_Wframe_larger_than_:
1418 1419
      opts->x_frame_larger_than_size = value;
      opts->x_warn_frame_larger_than = value != -1;
1420 1421
      break;

1422 1423 1424 1425 1426
    case OPT_Wstack_usage_:
      opts->x_warn_stack_usage = value;
      opts->x_flag_stack_usage_info = value != -1;
      break;

1427
    case OPT_Wstrict_aliasing:
1428
      set_Wstrict_aliasing (opts, value);
1429 1430
      break;

1431
    case OPT_Wstrict_overflow:
1432 1433 1434
      opts->x_warn_strict_overflow = (value
				      ? (int) WARN_STRICT_OVERFLOW_CONDITIONAL
				      : 0);
1435 1436
      break;

1437
    case OPT_Wsystem_headers:
1438
      dc->dc_warn_system_headers = value;
Neil Booth committed
1439 1440
      break;

1441
    case OPT_aux_info:
1442
      opts->x_flag_gen_aux_info = 1;
1443 1444 1445 1446 1447 1448 1449
      break;

    case OPT_auxbase_strip:
      {
	char *tmp = xstrdup (arg);
	strip_off_ending (tmp, strlen (tmp));
	if (tmp[0])
1450
	  opts->x_aux_base_name = tmp;
1451 1452 1453 1454
      }
      break;

    case OPT_d:
1455
      decode_d_option (arg, opts, loc, dc);
1456 1457
      break;

Neil Booth committed
1458 1459
    case OPT_fcall_used_:
    case OPT_fcall_saved_:
1460
      /* Deferred.  */
Neil Booth committed
1461
      break;
1462 1463

    case OPT_fdbg_cnt_:
1464
    case OPT_fdbg_cnt_list:
1465
      /* Deferred.  */
1466
      break;
Neil Booth committed
1467

1468
    case OPT_fdebug_prefix_map_:
1469
      /* Deferred.  */
1470 1471
      break;

Neil Booth committed
1472
    case OPT_fdiagnostics_show_location_:
1473
      diagnostic_prefixing_rule (dc) = (diagnostic_prefixing_rule_t) value;
Neil Booth committed
1474 1475
      break;

1476
    case OPT_fdiagnostics_show_option:
1477
      dc->show_option_requested = value;
1478 1479
      break;

1480
    case OPT_fdump_:
1481
      /* Deferred.  */
1482 1483
      break;

Neil Booth committed
1484
    case OPT_ffast_math:
1485
      set_fast_math_flags (opts, value);
Neil Booth committed
1486 1487
      break;

1488
    case OPT_funsafe_math_optimizations:
1489
      set_unsafe_math_optimizations_flags (opts, value);
1490 1491
      break;

Neil Booth committed
1492
    case OPT_ffixed_:
1493
      /* Deferred.  */
Neil Booth committed
1494 1495
      break;

1496
    case OPT_finline_limit_:
1497 1498 1499 1500
      set_param_value ("max-inline-insns-single", value / 2,
		       opts->x_param_values, opts_set->x_param_values);
      set_param_value ("max-inline-insns-auto", value / 2,
		       opts->x_param_values, opts_set->x_param_values);
1501 1502
      break;

1503
    case OPT_finstrument_functions_exclude_function_list_:
1504
      add_comma_separated_to_vector
1505
	(&opts->x_flag_instrument_functions_exclude_functions, arg);
1506 1507 1508
      break;

    case OPT_finstrument_functions_exclude_file_list_:
1509
      add_comma_separated_to_vector
1510
	(&opts->x_flag_instrument_functions_exclude_files, arg);
1511 1512
      break;

Neil Booth committed
1513
    case OPT_fmessage_length_:
1514
      pp_set_line_maximum_length (dc->printer, value);
Neil Booth committed
1515 1516
      break;

1517 1518
    case OPT_fpack_struct_:
      if (value <= 0 || (value & (value - 1)) || value > 16)
1519 1520 1521
	error_at (loc,
		  "structure alignment must be a small power of two, not %d",
		  value);
1522
      else
1523
	opts->x_initial_max_fld_align = value;
1524 1525
      break;

Diego Novillo committed
1526 1527
    case OPT_fplugin_:
    case OPT_fplugin_arg_:
1528
      /* Deferred.  */
Diego Novillo committed
1529 1530
      break;

1531
    case OPT_fprofile_use_:
1532
      opts->x_profile_data_prefix = xstrdup (arg);
1533
      opts->x_flag_profile_use = true;
1534 1535
      value = true;
      /* No break here - do -fprofile-use processing. */
1536
    case OPT_fprofile_use:
1537
      if (!opts_set->x_flag_branch_probabilities)
1538
	opts->x_flag_branch_probabilities = value;
1539
      if (!opts_set->x_flag_profile_values)
1540
	opts->x_flag_profile_values = value;
1541
      if (!opts_set->x_flag_unroll_loops)
1542
	opts->x_flag_unroll_loops = value;
1543
      if (!opts_set->x_flag_peel_loops)
1544
	opts->x_flag_peel_loops = value;
1545
      if (!opts_set->x_flag_tracer)
1546
	opts->x_flag_tracer = value;
1547
      if (!opts_set->x_flag_value_profile_transformations)
1548
	opts->x_flag_value_profile_transformations = value;
1549
      if (!opts_set->x_flag_inline_functions)
1550
	opts->x_flag_inline_functions = value;
1551
      if (!opts_set->x_flag_ipa_cp)
1552
	opts->x_flag_ipa_cp = value;
1553
      if (!opts_set->x_flag_ipa_cp_clone
1554 1555
	  && value && opts->x_flag_ipa_cp)
	opts->x_flag_ipa_cp_clone = value;
1556
      if (!opts_set->x_flag_predictive_commoning)
1557
	opts->x_flag_predictive_commoning = value;
1558
      if (!opts_set->x_flag_unswitch_loops)
1559
	opts->x_flag_unswitch_loops = value;
1560
      if (!opts_set->x_flag_gcse_after_reload)
1561
	opts->x_flag_gcse_after_reload = value;
1562 1563
      break;

1564
    case OPT_fprofile_generate_:
1565
      opts->x_profile_data_prefix = xstrdup (arg);
1566 1567
      value = true;
      /* No break here - do -fprofile-generate processing. */
1568
    case OPT_fprofile_generate:
1569
      if (!opts_set->x_profile_arc_flag)
1570
	opts->x_profile_arc_flag = value;
1571
      if (!opts_set->x_flag_profile_values)
1572
	opts->x_flag_profile_values = value;
1573
      if (!opts_set->x_flag_value_profile_transformations)
1574
	opts->x_flag_value_profile_transformations = value;
1575
      if (!opts_set->x_flag_inline_functions)
1576
	opts->x_flag_inline_functions = value;
1577 1578 1579 1580 1581
      /* FIXME: Instrumentation we insert makes ipa-reference bitmaps
	 quadratic.  Disable the pass until better memory representation
	 is done.  */
      if (!opts_set->x_flag_ipa_reference && in_lto_p)
        opts->x_flag_ipa_reference = false;
1582 1583
      break;

1584
    case OPT_fshow_column:
1585
      dc->show_column = value;
1586 1587
      break;

Neil Booth committed
1588 1589 1590
    case OPT_frandom_seed:
      /* The real switch is -fno-random-seed.  */
      if (value)
1591
	return false;
1592
      /* Deferred.  */
Neil Booth committed
1593 1594 1595
      break;

    case OPT_frandom_seed_:
1596
      /* Deferred.  */
Neil Booth committed
1597 1598 1599 1600
      break;

    case OPT_fsched_verbose_:
#ifdef INSN_SCHEDULING
1601
      /* Handled with Var in common.opt.  */
Neil Booth committed
1602 1603
      break;
#else
1604
      return false;
Neil Booth committed
1605 1606
#endif

1607
    case OPT_fsched_stalled_insns_:
1608 1609 1610
      opts->x_flag_sched_stalled_insns = value;
      if (opts->x_flag_sched_stalled_insns == 0)
	opts->x_flag_sched_stalled_insns = -1;
1611 1612 1613
      break;

    case OPT_fsched_stalled_insns_dep_:
1614
      opts->x_flag_sched_stalled_insns_dep = value;
1615
      break;
1616

1617 1618
    case OPT_fstack_check_:
      if (!strcmp (arg, "no"))
1619
	opts->x_flag_stack_check = NO_STACK_CHECK;
1620 1621
      else if (!strcmp (arg, "generic"))
	/* This is the old stack checking method.  */
1622
	opts->x_flag_stack_check = STACK_CHECK_BUILTIN
1623 1624 1625 1626
			   ? FULL_BUILTIN_STACK_CHECK
			   : GENERIC_STACK_CHECK;
      else if (!strcmp (arg, "specific"))
	/* This is the new stack checking method.  */
1627
	opts->x_flag_stack_check = STACK_CHECK_BUILTIN
1628 1629 1630 1631 1632
			   ? FULL_BUILTIN_STACK_CHECK
			   : STACK_CHECK_STATIC_BUILTIN
			     ? STATIC_BUILTIN_STACK_CHECK
			     : GENERIC_STACK_CHECK;
      else
1633
	warning_at (loc, 0, "unknown stack check parameter \"%s\"", arg);
1634 1635
      break;

Neil Booth committed
1636 1637 1638
    case OPT_fstack_limit:
      /* The real switch is -fno-stack-limit.  */
      if (value)
1639
	return false;
1640
      /* Deferred.  */
Neil Booth committed
1641 1642
      break;

Neil Booth committed
1643 1644
    case OPT_fstack_limit_register_:
    case OPT_fstack_limit_symbol_:
1645
      /* Deferred.  */
Neil Booth committed
1646 1647
      break;

1648 1649 1650 1651 1652
    case OPT_fstack_usage:
      opts->x_flag_stack_usage = value;
      opts->x_flag_stack_usage_info = value != 0;
      break;

1653
    case OPT_ftree_vectorizer_verbose_:
1654
      vect_set_verbosity_level (opts, value);
1655 1656
      break;

1657
    case OPT_g:
1658 1659
      set_debug_level (NO_DEBUG, DEFAULT_GDB_EXTENSIONS, arg, opts, opts_set,
		       loc);
1660 1661 1662
      break;

    case OPT_gcoff:
1663
      set_debug_level (SDB_DEBUG, false, arg, opts, opts_set, loc);
1664 1665
      break;

1666
    case OPT_gdwarf_:
1667
      if (value < 2 || value > 4)
1668
	error_at (loc, "dwarf version %d is not supported", value);
1669 1670
      else
	dwarf_version = value;
1671
      set_debug_level (DWARF2_DEBUG, false, "", opts, opts_set, loc);
1672 1673 1674
      break;

    case OPT_ggdb:
1675
      set_debug_level (NO_DEBUG, 2, arg, opts, opts_set, loc);
1676 1677 1678 1679
      break;

    case OPT_gstabs:
    case OPT_gstabs_:
1680 1681
      set_debug_level (DBX_DEBUG, code == OPT_gstabs_, arg, opts, opts_set,
		       loc);
1682 1683 1684
      break;

    case OPT_gvms:
1685
      set_debug_level (VMS_DEBUG, false, arg, opts, opts_set, loc);
1686 1687 1688 1689
      break;

    case OPT_gxcoff:
    case OPT_gxcoff_:
1690 1691
      set_debug_level (XCOFF_DEBUG, code == OPT_gxcoff_, arg, opts, opts_set,
		       loc);
1692 1693
      break;

1694
    case OPT_pedantic_errors:
1695 1696
      opts->x_pedantic = 1;
      dc->pedantic_errors = 1;
1697 1698
      break;

1699
    case OPT_flto:
Andi Kleen committed
1700
      opts->x_flag_lto = value ? "" : NULL;
1701 1702
      break;

1703
    case OPT_w:
1704
      dc->dc_inhibit_warnings = true;
1705 1706
      break;

1707 1708 1709 1710
    case OPT_fmax_errors_:
      dc->max_errors = value;
      break;

1711 1712 1713 1714
    case OPT_fuse_linker_plugin:
      /* No-op. Used by the driver and passed to us because it starts with f.*/
      break;

1715 1716 1717 1718 1719
    case OPT_Wuninitialized:
      /* Also turn on maybe uninitialized warning.  */
      warn_maybe_uninitialized = value;
      break;

1720 1721 1722
    default:
      /* If the flag was handled in a standard way, assume the lack of
	 processing here is intentional.  */
1723
      gcc_assert (option_flag_var (scode, opts));
1724
      break;
1725 1726
    }

1727
  return true;
1728
}
Neil Booth committed
1729 1730 1731

/* Handle --param NAME=VALUE.  */
static void
1732
handle_param (struct gcc_options *opts, struct gcc_options *opts_set,
1733
	      location_t loc, const char *carg)
Neil Booth committed
1734 1735 1736 1737 1738 1739 1740
{
  char *equal, *arg;
  int value;

  arg = xstrdup (carg);
  equal = strchr (arg, '=');
  if (!equal)
1741 1742
    error_at (loc, "%s: --param arguments should be of the form NAME=VALUE",
	      arg);
Neil Booth committed
1743 1744 1745 1746
  else
    {
      value = integral_argument (equal + 1);
      if (value == -1)
1747
	error_at (loc, "invalid --param value %qs", equal + 1);
Neil Booth committed
1748 1749 1750
      else
	{
	  *equal = '\0';
1751 1752
	  set_param_value (arg, value,
			   opts->x_param_values, opts_set->x_param_values);
Neil Booth committed
1753 1754 1755 1756 1757 1758
	}
    }

  free (arg);
}

1759
/* Used to set the level of strict aliasing warnings in OPTS,
1760 1761 1762 1763 1764 1765
   when no level is specified (i.e., when -Wstrict-aliasing, and not
   -Wstrict-aliasing=level was given).
   ONOFF is assumed to take value 1 when -Wstrict-aliasing is specified,
   and 0 otherwise.  After calling this function, wstrict_aliasing will be
   set to the default value of -Wstrict_aliasing=level, currently 3.  */
void
1766
set_Wstrict_aliasing (struct gcc_options *opts, int onoff)
1767 1768 1769
{
  gcc_assert (onoff == 0 || onoff == 1);
  if (onoff != 0)
1770
    opts->x_warn_strict_aliasing = 3;
1771
  else
1772
    opts->x_warn_strict_aliasing = 0;
1773 1774
}

Neil Booth committed
1775 1776
/* The following routines are useful in setting all the flags that
   -ffast-math and -fno-fast-math imply.  */
1777
static void
1778
set_fast_math_flags (struct gcc_options *opts, int set)
Neil Booth committed
1779
{
1780 1781 1782 1783 1784 1785 1786 1787 1788
  if (!opts->frontend_set_flag_unsafe_math_optimizations)
    {
      opts->x_flag_unsafe_math_optimizations = set;
      set_unsafe_math_optimizations_flags (opts, set);
    }
  if (!opts->frontend_set_flag_finite_math_only)
    opts->x_flag_finite_math_only = set;
  if (!opts->frontend_set_flag_errno_math)
    opts->x_flag_errno_math = !set;
Neil Booth committed
1789
  if (set)
1790
    {
1791 1792 1793 1794 1795 1796
      if (!opts->frontend_set_flag_signaling_nans)
	opts->x_flag_signaling_nans = 0;
      if (!opts->frontend_set_flag_rounding_math)
	opts->x_flag_rounding_math = 0;
      if (!opts->frontend_set_flag_cx_limited_range)
	opts->x_flag_cx_limited_range = 1;
1797
    }
Neil Booth committed
1798 1799
}

H.J. Lu committed
1800 1801
/* When -funsafe-math-optimizations is set the following
   flags are set as well.  */
1802
static void
1803
set_unsafe_math_optimizations_flags (struct gcc_options *opts, int set)
1804
{
1805 1806 1807 1808 1809 1810 1811 1812
  if (!opts->frontend_set_flag_trapping_math)
    opts->x_flag_trapping_math = !set;
  if (!opts->frontend_set_flag_signed_zeros)
    opts->x_flag_signed_zeros = !set;
  if (!opts->frontend_set_flag_associative_math)
    opts->x_flag_associative_math = set;
  if (!opts->frontend_set_flag_reciprocal_math)
    opts->x_flag_reciprocal_math = set;
1813 1814
}

1815
/* Return true iff flags in OPTS are set as if -ffast-math.  */
Neil Booth committed
1816
bool
1817
fast_math_flags_set_p (const struct gcc_options *opts)
Neil Booth committed
1818
{
1819 1820 1821 1822 1823
  return (!opts->x_flag_trapping_math
	  && opts->x_flag_unsafe_math_optimizations
	  && opts->x_flag_finite_math_only
	  && !opts->x_flag_signed_zeros
	  && !opts->x_flag_errno_math);
Neil Booth committed
1824
}
Neil Booth committed
1825

1826 1827 1828 1829 1830
/* Return true iff flags are set as if -ffast-math but using the flags stored
   in the struct cl_optimization structure.  */
bool
fast_math_flags_struct_set_p (struct cl_optimization *opt)
{
1831 1832 1833 1834 1835
  return (!opt->x_flag_trapping_math
	  && opt->x_flag_unsafe_math_optimizations
	  && opt->x_flag_finite_math_only
	  && !opt->x_flag_signed_zeros
	  && !opt->x_flag_errno_math);
1836 1837
}

1838 1839
/* Handle a debug output -g switch for options OPTS
   (OPTS_SET->x_write_symbols storing whether a debug type was passed
1840 1841
   explicitly), location LOC.  EXTENDED is true or false to support
   extended output (2 is special and means "-ggdb" was given).  */
1842
static void
1843
set_debug_level (enum debug_info_type type, int extended, const char *arg,
1844 1845
		 struct gcc_options *opts, struct gcc_options *opts_set,
		 location_t loc)
1846
{
1847
  opts->x_use_gnu_debug_info_extensions = extended;
1848 1849 1850

  if (type == NO_DEBUG)
    {
1851
      if (opts->x_write_symbols == NO_DEBUG)
1852
	{
1853
	  opts->x_write_symbols = PREFERRED_DEBUGGING_TYPE;
1854 1855 1856 1857

	  if (extended == 2)
	    {
#ifdef DWARF2_DEBUGGING_INFO
1858
	      opts->x_write_symbols = DWARF2_DEBUG;
1859
#elif defined DBX_DEBUGGING_INFO
1860
	      opts->x_write_symbols = DBX_DEBUG;
1861 1862 1863
#endif
	    }

1864
	  if (opts->x_write_symbols == NO_DEBUG)
1865
	    warning_at (loc, 0, "target system does not support debug output");
1866 1867 1868 1869 1870
	}
    }
  else
    {
      /* Does it conflict with an already selected type?  */
1871 1872 1873
      if (opts_set->x_write_symbols != NO_DEBUG
	  && opts->x_write_symbols != NO_DEBUG
	  && type != opts->x_write_symbols)
1874 1875
	error_at (loc, "debug format \"%s\" conflicts with prior selection",
		  debug_type_names[type]);
1876 1877
      opts->x_write_symbols = type;
      opts_set->x_write_symbols = type;
1878 1879 1880 1881 1882
    }

  /* A debug flag without a level defaults to level 2.  */
  if (*arg == '\0')
    {
1883 1884
      if (!opts->x_debug_info_level)
	opts->x_debug_info_level = DINFO_LEVEL_NORMAL;
1885 1886 1887
    }
  else
    {
1888 1889
      int argval = integral_argument (arg);
      if (argval == -1)
1890
	error_at (loc, "unrecognised debug output level \"%s\"", arg);
1891
      else if (argval > 3)
1892
	error_at (loc, "debug output level %s is too high", arg);
1893
      else
1894
	opts->x_debug_info_level = (enum debug_info_levels) argval;
1895 1896 1897
    }
}

1898 1899 1900
/* Arrange to dump core on error for diagnostic context DC.  (The
   regular error message is still printed first, except in the case of
   abort ().)  */
1901

1902
static void
1903
setup_core_dumping (diagnostic_context *dc)
1904
{
1905 1906 1907 1908 1909 1910 1911 1912 1913 1914 1915 1916 1917
#ifdef SIGABRT
  signal (SIGABRT, SIG_DFL);
#endif
#if defined(HAVE_SETRLIMIT)
  {
    struct rlimit rlim;
    if (getrlimit (RLIMIT_CORE, &rlim) != 0)
      fatal_error ("getting core file size maximum limit: %m");
    rlim.rlim_cur = rlim.rlim_max;
    if (setrlimit (RLIMIT_CORE, &rlim) != 0)
      fatal_error ("setting core file size limit to maximum: %m");
  }
#endif
1918
  diagnostic_abort_on_error (dc);
1919
}
1920

1921 1922
/* Parse a -d<ARG> command line switch for OPTS, location LOC,
   diagnostic context DC.  */
1923

1924
static void
1925 1926
decode_d_option (const char *arg, struct gcc_options *opts,
		 location_t loc, diagnostic_context *dc)
1927
{
1928
  int c;
1929

1930 1931 1932 1933
  while (*arg)
    switch (c = *arg++)
      {
      case 'A':
1934
	opts->x_flag_debug_asm = 1;
1935 1936
	break;
      case 'p':
1937
	opts->x_flag_print_asm_name = 1;
1938 1939
	break;
      case 'P':
1940 1941
	opts->x_flag_dump_rtl_in_asm = 1;
	opts->x_flag_print_asm_name = 1;
1942 1943
	break;
      case 'v':
1944
	opts->x_graph_dump_format = vcg;
1945 1946
	break;
      case 'x':
1947
	opts->x_rtl_dump_and_exit = 1;
1948 1949 1950 1951 1952 1953 1954 1955
	break;
      case 'D':	/* These are handled by the preprocessor.  */
      case 'I':
      case 'M':
      case 'N':
      case 'U':
	break;
      case 'H':
1956
	setup_core_dumping (dc);
1957 1958
	break;
      case 'a':
1959
	opts->x_flag_dump_all_passed = true;
1960
	break;
1961

1962
      default:
1963
	  warning_at (loc, 0, "unrecognized gcc debugging option: %c", c);
1964 1965
	break;
      }
1966
}
1967

1968
/* Enable (or disable if VALUE is 0) a warning option ARG (language
1969 1970 1971
   mask LANG_MASK, option handlers HANDLERS) as an error for option
   structures OPTS and OPTS_SET, diagnostic context DC (possibly
   NULL), location LOC.  This is used by -Werror=.  */
1972

1973
static void
1974
enable_warning_as_error (const char *arg, int value, unsigned int lang_mask,
1975
			 const struct cl_option_handlers *handlers,
1976 1977
			 struct gcc_options *opts,
			 struct gcc_options *opts_set,
1978
			 location_t loc, diagnostic_context *dc)
1979 1980 1981 1982 1983 1984 1985 1986
{
  char *new_option;
  int option_index;

  new_option = XNEWVEC (char, strlen (arg) + 2);
  new_option[0] = 'W';
  strcpy (new_option + 1, arg);
  option_index = find_opt (new_option, lang_mask);
1987
  if (option_index == OPT_SPECIAL_unknown)
1988
    {
1989
      error_at (loc, "-Werror=%s: no option -%s", arg, new_option);
1990 1991 1992
    }
  else
    {
1993 1994
      const diagnostic_t kind = value ? DK_ERROR : DK_WARNING;

1995 1996 1997
      control_warning_option (option_index, (int) kind, value,
			      loc, lang_mask,
			      handlers, opts, opts_set, dc);
1998 1999 2000
      if (option_index == OPT_Wuninitialized)
        enable_warning_as_error ("maybe-uninitialized", value, lang_mask,
	                         handlers, opts, opts_set, loc, dc);
2001 2002 2003
    }
  free (new_option);
}
2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031 2032 2033 2034 2035 2036 2037 2038

/* Return malloced memory for the name of the option OPTION_INDEX
   which enabled a diagnostic (context CONTEXT), originally of type
   ORIG_DIAG_KIND but possibly converted to DIAG_KIND by options such
   as -Werror.  */

char *
option_name (diagnostic_context *context, int option_index,
	     diagnostic_t orig_diag_kind, diagnostic_t diag_kind)
{
  if (option_index)
    {
      /* A warning classified as an error.  */
      if ((orig_diag_kind == DK_WARNING || orig_diag_kind == DK_PEDWARN)
	  && diag_kind == DK_ERROR)
	return concat (cl_options[OPT_Werror_].opt_text,
		       /* Skip over "-W".  */
		       cl_options[option_index].opt_text + 2,
		       NULL);
      /* A warning with option.  */
      else
	return xstrdup (cl_options[option_index].opt_text);
    }
  /* A warning without option classified as an error.  */
  else if (orig_diag_kind == DK_WARNING || orig_diag_kind == DK_PEDWARN
	   || diag_kind == DK_WARNING)
    {
      if (context->warning_as_error_requested)
	return xstrdup (cl_options[OPT_Werror].opt_text);
      else
	return xstrdup (_("enabled by default"));
    }
  else
    return NULL;
}