cppspec.c 5.33 KB
Newer Older
1 2 3
/* Specific flags and argument handling of the C preprocessor.
   Copyright (C) 1999 Free Software Foundation, Inc.

4
This file is part of GCC.
5

6 7 8 9
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
Software Foundation; either version 2, or (at your option) any later
version.
10

11 12 13 14
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.
15 16

You should have received a copy of the GNU General Public License
17 18 19
along with GCC; see the file COPYING.  If not, write to the Free
Software Foundation, 59 Temple Place - Suite 330, Boston, MA
02111-1307, USA.  */
20 21 22

#include "config.h"
#include "system.h"
23 24
#include "coretypes.h"
#include "tm.h"
25
#include "gcc.h"
26 27 28 29 30 31

/* The `cpp' executable installed in $(bindir) and $(cpp_install_dir)
   is a customized version of the gcc driver.  It forces -E; -S and -c
   are errors.  It defaults to -x c for files with unrecognized
   extensions, unless -x options appear in argv, in which case we
   assume the user knows what they're doing.  If no explicit input is
32
   mentioned, it will read stdin.  */
33 34 35 36 37 38 39 40 41

#ifndef SWITCH_TAKES_ARG
#define SWITCH_TAKES_ARG(CHAR) DEFAULT_SWITCH_TAKES_ARG(CHAR)
#endif

#ifndef WORD_SWITCH_TAKES_ARG
#define WORD_SWITCH_TAKES_ARG(STR) DEFAULT_WORD_SWITCH_TAKES_ARG (STR)
#endif

42 43 44
/* Suffixes for known sorts of input files.  Note that we do not list
   files which are normally considered to have been preprocessed already,
   since the user's expectation is that `cpp' always preprocesses.  */
45 46
static const char *const known_suffixes[] =
{
47
  ".c",  ".C",   ".S",   ".m",
48 49 50 51
  ".cc", ".cxx", ".cpp", ".cp",  ".c++",
  NULL
};

52
/* Filter argc and argv before processing by the gcc driver proper.  */
53
void
54 55
lang_specific_driver (int *in_argc, const char *const **in_argv,
		      int *in_added_libraries ATTRIBUTE_UNUSED)
56 57
{
  int argc = *in_argc;
58 59
  const char *const *argv = *in_argv;

60
  /* Do we need to read stdin? */
61
  int read_stdin = 1;
62 63

  /* Do we need to insert -E? */
64
  int need_E = 1;
65

66 67
  /* Have we seen an input file? */
  int seen_input = 0;
68

69
  /* Positions to insert -xc, -xassembler-with-cpp, and -o, if necessary.
70
     0 means unnecessary.  */
71 72 73 74 75 76
  int lang_c_here = 0;
  int lang_S_here = 0;
  int o_here = 0;

  /* Do we need to fix up an input file with an unrecognized suffix? */
  int need_fixups = 1;
77

78 79
  int i, j, quote = 0;
  const char **new_argv;
80
  int new_argc;
81 82 83
  extern int is_cpp_driver;

  is_cpp_driver = 1;
84 85

  /* First pass.  If we see an -S or -c, barf.  If we see an input file,
86 87
     turn off read_stdin.  If we see a second input file, it is actually
     the output file.  If we see a third input file, barf.  */
88 89 90 91 92 93 94
  for (i = 1; i < argc; i++)
    {
      if (quote == 1)
	{
	  quote = 0;
	  continue;
	}
95

96 97 98 99 100 101 102 103 104 105
      if (argv[i][0] == '-')
	{
	  if (argv[i][1] == '\0')
	    read_stdin = 0;
	  else if (argv[i][2] == '\0')
	    {
	      if (argv[i][1] == 'E')
		need_E = 0;
	      else if (argv[i][1] == 'S' || argv[i][1] == 'c')
		{
106
		  fatal ("\"%s\" is not a valid option to the preprocessor",
107
			 argv[i]);
108
		  return;
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
		}
	      else if (argv[i][1] == 'x')
		{
		  need_fixups = 0;
		  quote = 1;
		}
	      else if (SWITCH_TAKES_ARG (argv[i][1]))
		quote = 1;
	    }
	  else if (argv[i][1] == 'x')
	    need_fixups = 0;
	  else if (WORD_SWITCH_TAKES_ARG (&argv[i][1]))
	    quote = 1;
	}
      else /* not an option */
	{
125 126
	  seen_input++;
	  if (seen_input == 3)
127
	    {
128
	      fatal ("too many input files");
129 130 131 132 133
	      return;
	    }
	  else if (seen_input == 2)
	    {
	      o_here = i;
134 135 136
	    }
	  else
	    {
137 138
	      read_stdin = 0;
	      if (need_fixups)
139
		{
140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
		  int l = strlen (argv[i]);
		  int known = 0;
		  const char *const *suff;

		  for (suff = known_suffixes; *suff; suff++)
		    if (!strcmp (*suff, &argv[i][l - strlen(*suff)]))
		      {
			known = 1;
			break;
		      }

		  if (! known)
		    {
		      /* .s files are a special case; we have to treat
			 them like .S files so -D__ASSEMBLER__ will be
			 in effect.  */
		      if (!strcmp (".s", &argv[i][l - 2]))
			lang_S_here = i;
		      else
			lang_c_here = i;
		    }
161 162 163 164 165
		}
	    }
	}
    }

166 167
  /* If we don't need to edit the command line, we can bail early.  */

Neil Booth committed
168
  new_argc = argc + need_E + read_stdin
169 170 171 172
    + !!o_here + !!lang_c_here + !!lang_S_here;

  if (new_argc == argc)
    return;
173

174
  /* One more slot for a terminating null.  */
175
  new_argv = xmalloc ((new_argc + 1) * sizeof(char *));
176 177

  new_argv[0] = argv[0];
178 179
  j = 1;

180
  if (need_E)
181 182 183
    new_argv[j++] = "-E";

  for (i = 1; i < argc; i++, j++)
184
    {
185 186 187 188 189 190 191 192
      if (i == lang_c_here)
	new_argv[j++] = "-xc";
      else if (i == lang_S_here)
	new_argv[j++] = "-xassembler-with-cpp";
      else if (i == o_here)
	new_argv[j++] = "-o";

      new_argv[j] = argv[i];
193 194 195
    }

  if (read_stdin)
196
    new_argv[j++] = "-";
197

198
  new_argv[j] = NULL;
199
  *in_argc = new_argc;
200
  *in_argv = new_argv;
201
}
202

203
/* Called before linking.  Returns 0 on success and -1 on failure.  */
204
int lang_specific_pre_link (void)
205
{
206
  return 0;  /* Not used for cpp.  */
207 208
}

209 210
/* Number of extra output files that lang_specific_pre_link may generate.  */
int lang_specific_extra_outfiles = 0;  /* Not used for cpp.  */
211 212 213 214 215 216

/* Table of language-specific spec functions.  */
const struct spec_function lang_specific_spec_functions[] =
{
  { 0, 0 }
};