xref: /netbsd-src/external/gpl3/gcc.old/dist/fixincludes/fixlib.c (revision 1debfc3d3fad8af6f31804271c18e67f77b4d718)
1*1debfc3dSmrg 
2*1debfc3dSmrg /* Install modified versions of certain ANSI-incompatible system header
3*1debfc3dSmrg    files which are fixed to work correctly with ANSI C and placed in a
4*1debfc3dSmrg    directory that GCC will search.
5*1debfc3dSmrg 
6*1debfc3dSmrg    Copyright (C) 1999, 2000, 2001, 2004, 2009 Free Software Foundation, Inc.
7*1debfc3dSmrg 
8*1debfc3dSmrg This file is part of GCC.
9*1debfc3dSmrg 
10*1debfc3dSmrg GCC is free software; you can redistribute it and/or modify
11*1debfc3dSmrg it under the terms of the GNU General Public License as published by
12*1debfc3dSmrg the Free Software Foundation; either version 3, or (at your option)
13*1debfc3dSmrg any later version.
14*1debfc3dSmrg 
15*1debfc3dSmrg GCC is distributed in the hope that it will be useful,
16*1debfc3dSmrg but WITHOUT ANY WARRANTY; without even the implied warranty of
17*1debfc3dSmrg MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*1debfc3dSmrg GNU General Public License for more details.
19*1debfc3dSmrg 
20*1debfc3dSmrg You should have received a copy of the GNU General Public License
21*1debfc3dSmrg along with GCC; see the file COPYING3.  If not see
22*1debfc3dSmrg <http://www.gnu.org/licenses/>.  */
23*1debfc3dSmrg 
24*1debfc3dSmrg #include "fixlib.h"
25*1debfc3dSmrg 
26*1debfc3dSmrg /* * * * * * * * * * * * *
27*1debfc3dSmrg 
28*1debfc3dSmrg    load_file_data loads all the contents of a file into malloc-ed memory.
29*1debfc3dSmrg    Its argument is the file pointer of the file to read in; the returned
30*1debfc3dSmrg    result is the NUL terminated contents of the file.  The file
31*1debfc3dSmrg    is presumed to be an ASCII text file containing no NULs.  */
32*1debfc3dSmrg 
33*1debfc3dSmrg char *
load_file_data(FILE * fp)34*1debfc3dSmrg load_file_data (FILE* fp)
35*1debfc3dSmrg {
36*1debfc3dSmrg   char *pz_data = (char*)NULL;
37*1debfc3dSmrg   int    space_left = -1;  /* allow for terminating NUL */
38*1debfc3dSmrg   size_t space_used = 0;
39*1debfc3dSmrg 
40*1debfc3dSmrg   if (fp == (FILE*)NULL)
41*1debfc3dSmrg     return pz_data;
42*1debfc3dSmrg 
43*1debfc3dSmrg   do
44*1debfc3dSmrg     {
45*1debfc3dSmrg       size_t  size_read;
46*1debfc3dSmrg 
47*1debfc3dSmrg       if (space_left < 1024)
48*1debfc3dSmrg         {
49*1debfc3dSmrg           space_left += 4096;
50*1debfc3dSmrg 	  pz_data = XRESIZEVEC (char, pz_data, space_left + space_used + 1 );
51*1debfc3dSmrg         }
52*1debfc3dSmrg       size_read = fread (pz_data + space_used, 1, space_left, fp);
53*1debfc3dSmrg 
54*1debfc3dSmrg       if (size_read == 0)
55*1debfc3dSmrg         {
56*1debfc3dSmrg           if (feof (fp))
57*1debfc3dSmrg             break;
58*1debfc3dSmrg 
59*1debfc3dSmrg           if (ferror (fp))
60*1debfc3dSmrg             {
61*1debfc3dSmrg               int err = errno;
62*1debfc3dSmrg               if (err != EISDIR)
63*1debfc3dSmrg                 fprintf (stderr, "error %d (%s) reading input\n", err,
64*1debfc3dSmrg                          xstrerror (err));
65*1debfc3dSmrg               free ((void *) pz_data);
66*1debfc3dSmrg               return (char *) NULL;
67*1debfc3dSmrg             }
68*1debfc3dSmrg         }
69*1debfc3dSmrg 
70*1debfc3dSmrg       space_left -= size_read;
71*1debfc3dSmrg       space_used += size_read;
72*1debfc3dSmrg     } while (! feof (fp));
73*1debfc3dSmrg 
74*1debfc3dSmrg   pz_data = XRESIZEVEC (char, pz_data, space_used+1 );
75*1debfc3dSmrg   pz_data[ space_used ] = NUL;
76*1debfc3dSmrg 
77*1debfc3dSmrg   return pz_data;
78*1debfc3dSmrg }
79*1debfc3dSmrg 
80*1debfc3dSmrg #ifdef IS_CXX_HEADER_NEEDED
81*1debfc3dSmrg t_bool
is_cxx_header(tCC * fname,tCC * text)82*1debfc3dSmrg is_cxx_header (tCC* fname, tCC* text)
83*1debfc3dSmrg {
84*1debfc3dSmrg   /*  First, check to see if the file is in a C++ directory */
85*1debfc3dSmrg   for (;;)
86*1debfc3dSmrg     {
87*1debfc3dSmrg       switch (*(fname++))
88*1debfc3dSmrg         {
89*1debfc3dSmrg         case 'C': /* check for "CC/" */
90*1debfc3dSmrg           if ((fname[0] == 'C') && (fname[1] == '/'))
91*1debfc3dSmrg             return BOOL_TRUE;
92*1debfc3dSmrg           break;
93*1debfc3dSmrg 
94*1debfc3dSmrg         case 'x': /* check for "xx/" */
95*1debfc3dSmrg           if ((fname[0] == 'x') && (fname[1] == '/'))
96*1debfc3dSmrg             return BOOL_TRUE;
97*1debfc3dSmrg           break;
98*1debfc3dSmrg 
99*1debfc3dSmrg         case '+': /* check for "++" */
100*1debfc3dSmrg           if (fname[0] == '+')
101*1debfc3dSmrg             return BOOL_TRUE;
102*1debfc3dSmrg           break;
103*1debfc3dSmrg 
104*1debfc3dSmrg         case NUL:
105*1debfc3dSmrg           goto not_cxx_name;
106*1debfc3dSmrg         }
107*1debfc3dSmrg     } not_cxx_name:;
108*1debfc3dSmrg 
109*1debfc3dSmrg   /* Or it might contain one of several phrases which indicate C++ code.
110*1debfc3dSmrg      Currently recognized are:
111*1debfc3dSmrg      extern "C++"
112*1debfc3dSmrg      -*- (Mode: )? C++ -*-   (emacs mode marker)
113*1debfc3dSmrg      template <
114*1debfc3dSmrg    */
115*1debfc3dSmrg     {
116*1debfc3dSmrg       tSCC cxxpat[] = "\
117*1debfc3dSmrg extern[ \t]*\"C\\+\\+\"|\
118*1debfc3dSmrg -\\*-[ \t]*([mM]ode:[ \t]*)?[cC]\\+\\+[; \t]*-\\*-|\
119*1debfc3dSmrg template[ \t]*<|\
120*1debfc3dSmrg ^[ \t]*class[ \t]|\
121*1debfc3dSmrg (public|private|protected):|\
122*1debfc3dSmrg ^[ \t]*#[ \t]*pragma[ \t]+(interface|implementation)\
123*1debfc3dSmrg ";
124*1debfc3dSmrg       static regex_t cxxre;
125*1debfc3dSmrg       static int compiled;
126*1debfc3dSmrg 
127*1debfc3dSmrg       if (!compiled)
128*1debfc3dSmrg 	compile_re (cxxpat, &cxxre, 0, "contents check", "is_cxx_header");
129*1debfc3dSmrg 
130*1debfc3dSmrg       if (xregexec (&cxxre, text, 0, 0, 0) == 0)
131*1debfc3dSmrg 	return BOOL_TRUE;
132*1debfc3dSmrg     }
133*1debfc3dSmrg 
134*1debfc3dSmrg   return BOOL_FALSE;
135*1debfc3dSmrg }
136*1debfc3dSmrg #endif /* CXX_TYPE_NEEDED */
137*1debfc3dSmrg 
138*1debfc3dSmrg #ifdef SKIP_QUOTE_NEEDED
139*1debfc3dSmrg /*
140*1debfc3dSmrg  *  Skip over a quoted string.  Single quote strings may
141*1debfc3dSmrg  *  contain multiple characters if the first character is
142*1debfc3dSmrg  *  a backslash.  Especially a backslash followed by octal digits.
143*1debfc3dSmrg  *  We are not doing a correctness syntax check here.
144*1debfc3dSmrg  */
145*1debfc3dSmrg tCC*
skip_quote(char q,char * text)146*1debfc3dSmrg skip_quote(char q, char* text )
147*1debfc3dSmrg {
148*1debfc3dSmrg   for (;;)
149*1debfc3dSmrg     {
150*1debfc3dSmrg       char ch = *(text++);
151*1debfc3dSmrg       switch (ch)
152*1debfc3dSmrg         {
153*1debfc3dSmrg         case '\\':
154*1debfc3dSmrg           text++; /* skip over whatever character follows */
155*1debfc3dSmrg           break;
156*1debfc3dSmrg 
157*1debfc3dSmrg         case '"':
158*1debfc3dSmrg         case '\'':
159*1debfc3dSmrg           if (ch != q)
160*1debfc3dSmrg             break;
161*1debfc3dSmrg           /*FALLTHROUGH*/
162*1debfc3dSmrg 
163*1debfc3dSmrg         case '\n':
164*1debfc3dSmrg         case NUL:
165*1debfc3dSmrg           goto skip_done;
166*1debfc3dSmrg         }
167*1debfc3dSmrg     } skip_done:;
168*1debfc3dSmrg 
169*1debfc3dSmrg   return text;
170*1debfc3dSmrg }
171*1debfc3dSmrg #endif /* SKIP_QUOTE_NEEDED */
172*1debfc3dSmrg 
173*1debfc3dSmrg /* * * * * * * * * * * * *
174*1debfc3dSmrg 
175*1debfc3dSmrg    Compile one regular expression pattern for later use.  PAT contains
176*1debfc3dSmrg    the pattern, RE points to a regex_t structure (which should have
177*1debfc3dSmrg    been bzeroed).  MATCH is 1 if we need to know where the regex
178*1debfc3dSmrg    matched, 0 if not. If xregcomp fails, prints an error message and
179*1debfc3dSmrg    aborts; E1 and E2 are strings to shove into the error message.
180*1debfc3dSmrg 
181*1debfc3dSmrg    The patterns we search for are all egrep patterns.
182*1debfc3dSmrg    REG_EXTENDED|REG_NEWLINE produces identical regex syntax/semantics
183*1debfc3dSmrg    to egrep (verified from 4.4BSD Programmer's Reference Manual).  */
184*1debfc3dSmrg void
compile_re(tCC * pat,regex_t * re,int match,tCC * e1,tCC * e2)185*1debfc3dSmrg compile_re( tCC* pat, regex_t* re, int match, tCC* e1, tCC* e2 )
186*1debfc3dSmrg {
187*1debfc3dSmrg   tSCC z_bad_comp[] = "fixincl ERROR:  cannot compile %s regex for %s\n\
188*1debfc3dSmrg \texpr = `%s'\n\terror %s\n";
189*1debfc3dSmrg   int flags, err;
190*1debfc3dSmrg 
191*1debfc3dSmrg   flags = (match ? REG_EXTENDED|REG_NEWLINE
192*1debfc3dSmrg 	   : REG_EXTENDED|REG_NEWLINE|REG_NOSUB);
193*1debfc3dSmrg   err = xregcomp (re, pat, flags);
194*1debfc3dSmrg 
195*1debfc3dSmrg   if (err)
196*1debfc3dSmrg     {
197*1debfc3dSmrg       char rerrbuf[1024];
198*1debfc3dSmrg       regerror (err, re, rerrbuf, 1024);
199*1debfc3dSmrg       fprintf (stderr, z_bad_comp, e1, e2, pat, rerrbuf);
200*1debfc3dSmrg       exit (EXIT_FAILURE);
201*1debfc3dSmrg     }
202*1debfc3dSmrg }
203*1debfc3dSmrg 
204*1debfc3dSmrg /* * * * * * * * * * * * *
205*1debfc3dSmrg 
206*1debfc3dSmrg    Helper routine and data for the machine_name test and fix.  */
207*1debfc3dSmrg 
208*1debfc3dSmrg tSCC mn_label_pat[] = "^[ \t]*#[ \t]*(if|ifdef|ifndef)[ \t]+";
209*1debfc3dSmrg static regex_t mn_label_re;
210*1debfc3dSmrg static regex_t mn_name_re;
211*1debfc3dSmrg 
212*1debfc3dSmrg static int mn_compiled = 0;
213*1debfc3dSmrg 
214*1debfc3dSmrg t_bool
mn_get_regexps(regex_t ** label_re,regex_t ** name_re,tCC * who)215*1debfc3dSmrg mn_get_regexps(regex_t** label_re, regex_t** name_re, tCC* who )
216*1debfc3dSmrg {
217*1debfc3dSmrg   if (! pz_mn_name_pat)
218*1debfc3dSmrg     return BOOL_FALSE;
219*1debfc3dSmrg 
220*1debfc3dSmrg   if (! mn_compiled)
221*1debfc3dSmrg     {
222*1debfc3dSmrg       compile_re (mn_label_pat, &mn_label_re, 1, "label pattern", who);
223*1debfc3dSmrg       compile_re (pz_mn_name_pat, &mn_name_re, 1, "name pattern", who);
224*1debfc3dSmrg       mn_compiled++;
225*1debfc3dSmrg     }
226*1debfc3dSmrg   *label_re = &mn_label_re;
227*1debfc3dSmrg   *name_re = &mn_name_re;
228*1debfc3dSmrg   return BOOL_TRUE;
229*1debfc3dSmrg }
230*1debfc3dSmrg 
231*1debfc3dSmrg 
232*1debfc3dSmrg #ifdef SEPARATE_FIX_PROC
233*1debfc3dSmrg 
234*1debfc3dSmrg char*
make_raw_shell_str(char * pz_d,tCC * pz_s,size_t smax)235*1debfc3dSmrg make_raw_shell_str( char* pz_d, tCC* pz_s, size_t smax )
236*1debfc3dSmrg {
237*1debfc3dSmrg   tSCC zQ[] = "'\\''";
238*1debfc3dSmrg   size_t     dtaSize;
239*1debfc3dSmrg   char*      pz_d_start = pz_d;
240*1debfc3dSmrg 
241*1debfc3dSmrg   smax--; /* adjust for trailing NUL */
242*1debfc3dSmrg 
243*1debfc3dSmrg   dtaSize = strlen( pz_s ) + 3;
244*1debfc3dSmrg 
245*1debfc3dSmrg   {
246*1debfc3dSmrg     const char* pz = pz_s - 1;
247*1debfc3dSmrg 
248*1debfc3dSmrg     for (;;) {
249*1debfc3dSmrg       pz = strchr( pz+1, '\'' );
250*1debfc3dSmrg       if (pz == (char*)NULL)
251*1debfc3dSmrg         break;
252*1debfc3dSmrg       dtaSize += sizeof( zQ )-1;
253*1debfc3dSmrg     }
254*1debfc3dSmrg   }
255*1debfc3dSmrg   if (dtaSize > smax)
256*1debfc3dSmrg     return (char*)NULL;
257*1debfc3dSmrg 
258*1debfc3dSmrg   *(pz_d++) = '\'';
259*1debfc3dSmrg 
260*1debfc3dSmrg   for (;;) {
261*1debfc3dSmrg     if ((size_t) (pz_d - pz_d_start) >= smax)
262*1debfc3dSmrg       return (char*)NULL;
263*1debfc3dSmrg     switch (*(pz_d++) = *(pz_s++)) {
264*1debfc3dSmrg     case NUL:
265*1debfc3dSmrg       goto loopDone;
266*1debfc3dSmrg 
267*1debfc3dSmrg     case '\'':
268*1debfc3dSmrg       if ((size_t) (pz_d - pz_d_start) >= smax - sizeof( zQ )-1)
269*1debfc3dSmrg 	return (char*)NULL;
270*1debfc3dSmrg       strcpy( pz_d-1, zQ );
271*1debfc3dSmrg       pz_d += sizeof( zQ )-2;
272*1debfc3dSmrg     }
273*1debfc3dSmrg   } loopDone:;
274*1debfc3dSmrg   pz_d[-1] = '\'';
275*1debfc3dSmrg   *pz_d    = NUL;
276*1debfc3dSmrg 
277*1debfc3dSmrg   return pz_d;
278*1debfc3dSmrg }
279*1debfc3dSmrg 
280*1debfc3dSmrg #endif
281*1debfc3dSmrg 
282*1debfc3dSmrg #if defined(__MINGW32__)
283*1debfc3dSmrg void
fix_path_separators(char * p)284*1debfc3dSmrg fix_path_separators (char* p)
285*1debfc3dSmrg {
286*1debfc3dSmrg     while (p != NULL)
287*1debfc3dSmrg       {
288*1debfc3dSmrg         p = strchr (p, '\\');
289*1debfc3dSmrg         if (p != NULL)
290*1debfc3dSmrg           {
291*1debfc3dSmrg             *p = '/';
292*1debfc3dSmrg             ++p;
293*1debfc3dSmrg           }
294*1debfc3dSmrg       }
295*1debfc3dSmrg }
296*1debfc3dSmrg 
297*1debfc3dSmrg /* Count number of needle character ocurrences in str */
298*1debfc3dSmrg static int
count_occurrences_of_char(char * str,char needle)299*1debfc3dSmrg count_occurrences_of_char (char* str, char needle)
300*1debfc3dSmrg {
301*1debfc3dSmrg   int cnt = 0;
302*1debfc3dSmrg 
303*1debfc3dSmrg   while (str)
304*1debfc3dSmrg     {
305*1debfc3dSmrg        str = strchr (str, needle);
306*1debfc3dSmrg        if (str)
307*1debfc3dSmrg          {
308*1debfc3dSmrg            ++str;
309*1debfc3dSmrg            ++cnt;
310*1debfc3dSmrg          }
311*1debfc3dSmrg     }
312*1debfc3dSmrg 
313*1debfc3dSmrg   return cnt;
314*1debfc3dSmrg }
315*1debfc3dSmrg 
316*1debfc3dSmrg /* On Mingw32, system function will just start cmd by default.
317*1debfc3dSmrg    Call system function, but prepend ${CONFIG_SHELL} or ${SHELL} -c to the command,
318*1debfc3dSmrg    replace newlines with '$'\n'', enclose command with double quotes
319*1debfc3dSmrg    and escape special characters which were originally enclosed in single quotes.
320*1debfc3dSmrg  */
321*1debfc3dSmrg int
system_with_shell(char * s)322*1debfc3dSmrg system_with_shell (char* s)
323*1debfc3dSmrg {
324*1debfc3dSmrg   static const char z_shell_start_args[] = " -c \"";
325*1debfc3dSmrg   static const char z_shell_end_args[] = "\"";
326*1debfc3dSmrg   static const char z_shell_newline[] = "'$'\\n''";
327*1debfc3dSmrg 
328*1debfc3dSmrg   /* Use configured shell if present */
329*1debfc3dSmrg   char *env_shell = getenv ("CONFIG_SHELL");
330*1debfc3dSmrg   int newline_cnt = count_occurrences_of_char (s, '\n');
331*1debfc3dSmrg   int escapes_cnt  = count_occurrences_of_char( s, '\\')
332*1debfc3dSmrg                       + count_occurrences_of_char (s, '"')
333*1debfc3dSmrg                       + count_occurrences_of_char (s, '`');
334*1debfc3dSmrg   char *long_cmd;
335*1debfc3dSmrg   char *cmd_endp;
336*1debfc3dSmrg   int sys_result;
337*1debfc3dSmrg   char *s_scan;
338*1debfc3dSmrg   int in_quotes;
339*1debfc3dSmrg 
340*1debfc3dSmrg   if (env_shell == NULL)
341*1debfc3dSmrg     env_shell = getenv ("SHELL");
342*1debfc3dSmrg 
343*1debfc3dSmrg   /* If neither CONFIGURED_SHELL nor SHELL is set, just call standard system function */
344*1debfc3dSmrg   if (env_shell == NULL)
345*1debfc3dSmrg     return system (s);
346*1debfc3dSmrg 
347*1debfc3dSmrg   /* Allocate enough memory to fit newly created command string */
348*1debfc3dSmrg   long_cmd = XNEWVEC (char, strlen (env_shell)
349*1debfc3dSmrg                       + strlen (z_shell_start_args)
350*1debfc3dSmrg                       + strlen (s)
351*1debfc3dSmrg                       + newline_cnt * (strlen (z_shell_newline) - 1)
352*1debfc3dSmrg                       + escapes_cnt
353*1debfc3dSmrg                       + strlen (z_shell_end_args)
354*1debfc3dSmrg                       + 1);
355*1debfc3dSmrg 
356*1debfc3dSmrg   /* Start with ${SHELL} */
357*1debfc3dSmrg   strcpy (long_cmd, env_shell);
358*1debfc3dSmrg   cmd_endp = long_cmd + strlen (long_cmd);
359*1debfc3dSmrg 
360*1debfc3dSmrg   /* Opening quote */
361*1debfc3dSmrg   strcpy (cmd_endp, z_shell_start_args);
362*1debfc3dSmrg   cmd_endp += strlen (z_shell_start_args);
363*1debfc3dSmrg 
364*1debfc3dSmrg   /* Replace newlines and escape special chars */
365*1debfc3dSmrg   in_quotes = 0;
366*1debfc3dSmrg   for (s_scan = s; *s_scan; ++s_scan)
367*1debfc3dSmrg     {
368*1debfc3dSmrg       switch (*s_scan)
369*1debfc3dSmrg         {
370*1debfc3dSmrg           case '\n':
371*1debfc3dSmrg             if (in_quotes)
372*1debfc3dSmrg               {
373*1debfc3dSmrg                 /* Replace newline inside quotes with '$'\n'' */
374*1debfc3dSmrg                 strcpy (cmd_endp, z_shell_newline);
375*1debfc3dSmrg                 cmd_endp += strlen (z_shell_newline);
376*1debfc3dSmrg               }
377*1debfc3dSmrg             else
378*1debfc3dSmrg               {
379*1debfc3dSmrg                 /* Replace newlines outside quotes with ; and merge subsequent newlines */
380*1debfc3dSmrg                 *(cmd_endp++) = ';';
381*1debfc3dSmrg                 *(cmd_endp++) = ' ';
382*1debfc3dSmrg                 while (*(s_scan + 1) == '\n' || *(s_scan + 1) == ' ' || *(s_scan + 1) == '\t')
383*1debfc3dSmrg                   ++s_scan;
384*1debfc3dSmrg               }
385*1debfc3dSmrg             break;
386*1debfc3dSmrg           case '\'':
387*1debfc3dSmrg             /* Escape single quote and toggle in_quotes flag */
388*1debfc3dSmrg             in_quotes = !in_quotes;
389*1debfc3dSmrg             *(cmd_endp++) = *s_scan;
390*1debfc3dSmrg             break;
391*1debfc3dSmrg           case '\\':
392*1debfc3dSmrg           case '`':
393*1debfc3dSmrg             /* Escape backslash and backtick inside quotes */
394*1debfc3dSmrg             if (in_quotes)
395*1debfc3dSmrg                *(cmd_endp++) = '\\';
396*1debfc3dSmrg             *(cmd_endp++) = *s_scan;
397*1debfc3dSmrg             break;
398*1debfc3dSmrg           case '"':
399*1debfc3dSmrg             /* Escape double quotes always */
400*1debfc3dSmrg             *(cmd_endp++) = '\\';
401*1debfc3dSmrg             *(cmd_endp++) = *s_scan;
402*1debfc3dSmrg             break;
403*1debfc3dSmrg           default:
404*1debfc3dSmrg             *(cmd_endp++) = *s_scan;
405*1debfc3dSmrg         }
406*1debfc3dSmrg     }
407*1debfc3dSmrg 
408*1debfc3dSmrg   /* Closing quote */
409*1debfc3dSmrg   strcpy (cmd_endp, z_shell_end_args);
410*1debfc3dSmrg 
411*1debfc3dSmrg   sys_result = system (long_cmd);
412*1debfc3dSmrg 
413*1debfc3dSmrg   free (long_cmd);
414*1debfc3dSmrg 
415*1debfc3dSmrg   return sys_result;
416*1debfc3dSmrg }
417*1debfc3dSmrg 
418*1debfc3dSmrg #endif /* defined(__MINGW32__) */
419