OSDN Git Service

3e0d185ae16f492a2b02ca843f3fb5f7cef7c2ab
[pf3gnuchains/gcc-fork.git] / gcc / fixinc / fixfixes.c
1
2 /*
3
4    Test to see if a particular fix should be applied to a header file.
5
6    Copyright (C) 1997, 1998, 1999, 2000 Free Software Foundation, Inc.
7
8 = = = = = = = = = = = = = = = = = = = = = = = = =
9
10 NOTE TO DEVELOPERS
11
12 The routines you write here must work closely with fixincl.c.
13
14 Here are the rules:
15
16 1.  Every test procedure name must be suffixed with "_fix".
17     These routines will be referenced from inclhack.def, sans the suffix.
18
19 2.  Use the "FIX_PROC_HEAD()" macro _with_ the "_fix" suffix
20     (I cannot use the ## magic from ANSI C) for defining your entry point.
21
22 3.  Put your test name into the FIXUP_TABLE.
23
24 4.  Do not read anything from stdin.  It is closed.
25
26 5.  Write to stderr only in the event of a reportable error
27     In such an event, call "exit (EXIT_FAILURE)".
28
29 6.  You have access to the fixDescList entry for the fix in question.
30     This may be useful, for example, if there are interesting strings
31     or pre-compiled regular expressions stored there.
32
33     It is also possible to access fix descriptions by using the
34     index of a known fix, "my_fix_name" for example:
35
36         tFixDesc*  p_desc  = fixDescList + MY_FIX_NAME_FIXIDX;
37         tTestDesc* p_tlist = p_desc->p_test_desc;
38
39         regexec (p_tlist->p_test_regex, ...)
40
41 = = = = = = = = = = = = = = = = = = = = = = = = =
42
43 This file is part of GNU CC.
44
45 GNU CC is free software; you can redistribute it and/or modify
46 it under the terms of the GNU General Public License as published by
47 the Free Software Foundation; either version 2, or (at your option)
48 any later version.
49
50 GNU CC is distributed in the hope that it will be useful,
51 but WITHOUT ANY WARRANTY; without even the implied warranty of
52 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
53 GNU General Public License for more details.
54
55 You should have received a copy of the GNU General Public License
56 along with GNU CC; see the file COPYING.  If not, write to
57 the Free Software Foundation, 59 Temple Place - Suite 330,
58 Boston, MA 02111-1307, USA.  */
59
60 #include "fixlib.h"
61 #define    GTYPE_SE_CT 1
62
63 #ifdef SEPARATE_FIX_PROC
64 #include "fixincl.x"
65 #endif
66
67 tSCC zNeedsArg[] = "fixincl error:  `%s' needs %s argument (c_fix_arg[%d])\n";
68
69 typedef void t_fix_proc PARAMS ((const char *, const char *, tFixDesc *));
70 typedef struct {
71     const char*  fix_name;
72     t_fix_proc*  fix_proc;
73 } fix_entry_t;
74
75 #define FIXUP_TABLE \
76   _FT_( "char_macro_def",   char_macro_def_fix ) \
77   _FT_( "char_macro_use",   char_macro_use_fix ) \
78   _FT_( "format",           format_fix )         \
79   _FT_( "machine_name",     machine_name_fix )   \
80   _FT_( "wrap",             wrap_fix )           \
81   _FT_( "gnu_type",         gnu_type_fix )
82
83
84 #define FIX_PROC_HEAD( fix )                    \
85 static void fix PARAMS ((const char *, const char *, tFixDesc *)); /* avoid warning */      \
86 static void fix ( filname, text, p_fixd )       \
87     const char* filname;                        \
88     const char* text;                           \
89     tFixDesc* p_fixd;
90
91 #ifdef NEED_PRINT_QUOTE
92 /*
93  *  Skip over a quoted string.  Single quote strings may
94  *  contain multiple characters if the first character is
95  *  a backslash.  Especially a backslash followed by octal digits.
96  *  We are not doing a correctness syntax check here.
97  */
98 static char*
99 print_quote( q, text )
100   char  q;
101   char* text;
102 {
103   fputc( q, stdout );
104
105   for (;;)
106     {
107       char ch = *(text++);
108       fputc( ch, stdout );
109
110       switch (ch)
111         {
112         case '\\':
113           if (*text == NUL)
114             goto quote_done;
115
116           fputc( *(text++), stdout );
117           break;
118
119         case '"':
120         case '\'':
121           if (ch != q)
122             break;
123           /*FALLTHROUGH*/
124
125         case '\n':
126         case NUL:
127           goto quote_done;
128         }
129     } quote_done:;
130
131   return text;
132 }
133 #endif /* NEED_PRINT_QUOTE */
134
135
136 /*
137  *  Emit the GNU standard type wrapped up in such a way that
138  *  this thing can be encountered countless times during a compile
139  *  and not cause even a warning.
140  */
141 static const char *emit_gnu_type PARAMS ((const char *, regmatch_t *));
142 static const char*
143 emit_gnu_type ( text, rm )
144   const char* text;
145   regmatch_t* rm;
146 {
147   char z_TYPE[ 64 ];
148   char z_type[ 64 ];
149
150   fwrite (text, rm[0].rm_so, 1, stdout);
151
152   {
153     const char* ps = text   + rm[1].rm_so;
154     const char* pe = text   + rm[1].rm_eo;
155     char* pd = z_type;
156     char* pD = z_TYPE;
157
158     while (ps < pe)
159       *(pD++) = toupper( *(pd++) = *(ps++) );
160
161     *pD = *pd = NUL;
162   }
163
164   /*
165    *  Now print out the reformed typedef,
166    *  with a C++ guard for WCHAR
167    */
168   {
169     tSCC z_fmt[] = "\
170 #if !defined(_GCC_%s_T)%s\n\
171 #define _GCC_%s_T\n\
172 typedef __%s_TYPE__ %s_t;\n\
173 #endif\n";
174
175     const char* pz_guard = (strcmp (z_type, "wchar") == 0)
176                            ? " && ! defined(__cplusplus)" : "";
177
178     printf (z_fmt, z_TYPE, pz_guard, z_TYPE, z_TYPE, z_type);
179   }
180
181   return text += rm[0].rm_eo;
182 }
183
184
185 /*
186  *  Copy the `format' string to std out, replacing `%n' expressions
187  *  with the matched text from a regular expression evaluation.
188  *  Doubled '%' characters will be replaced with a single copy.
189  *  '%' characters in other contexts and all other characters are
190  *  copied out verbatim.
191  */
192 static void format_write PARAMS ((tCC *, tCC *, regmatch_t[]));
193 static void
194 format_write (format, text, av)
195      tCC* format;
196      tCC* text;
197      regmatch_t av[];
198 {
199   int c;
200
201   while ((c = (unsigned)*(format++)) != NUL) {
202
203     if (c != '%')
204       {
205         putchar(c);
206         continue;
207       }
208
209     c = (unsigned)*(format++);
210
211     /*
212      *  IF the character following a '%' is not a digit,
213      *  THEN we will always emit a '%' and we may or may
214      *  not emit the following character.  We will end on
215      *  a NUL and we will emit only one of a pair of '%'.
216      */
217     if (! ISDIGIT ( c ))
218       {
219         putchar( '%' );
220         switch (c) {
221         case NUL:
222           return;
223         case '%':
224           break;
225         default:
226           putchar(c);
227         }
228       }
229
230     /*
231      *  Emit the matched subexpression numbered 'c'.
232      *  IF, of course, there was such a match...
233      */
234     else {
235       regmatch_t*  pRM = av + (c - (unsigned)'0');
236       size_t len;
237
238       if (pRM->rm_so < 0)
239         continue;
240
241       len = pRM->rm_eo - pRM->rm_so;
242       if (len > 0)
243         fwrite(text + pRM->rm_so, len, 1, stdout);
244     }
245   }
246 }
247
248
249 /*
250  *  Search for multiple copies of a regular expression.  Each block
251  *  of matched text is replaced with the format string, as described
252  *  above in `format_write'.
253  */
254 FIX_PROC_HEAD( format_fix )
255 {
256   tCC*  pz_pat = p_fixd->patch_args[2];
257   tCC*  pz_fmt = p_fixd->patch_args[1];
258   regex_t re;
259   regmatch_t rm[10];
260
261   /*
262    *  We must have a format
263    */
264   if (pz_fmt == (tCC*)NULL)
265     {
266       fprintf( stderr, zNeedsArg, p_fixd->fix_name, "replacement format", 0 );
267       exit (EXIT_BROKEN);
268     }
269
270   /*
271    *  IF we don't have a search text, then go find the first
272    *  regular expression among the tests.
273    */
274   if (pz_pat == (tCC*)NULL)
275     {
276       tTestDesc* pTD = p_fixd->p_test_desc;
277       int        ct  = p_fixd->test_ct;
278       for (;;)
279         {
280           if (ct-- <= 0)
281             {
282               fprintf( stderr, zNeedsArg, p_fixd->fix_name, "search text", 1 );
283               exit (EXIT_BROKEN);
284             }
285
286           if (pTD->type == TT_EGREP)
287             {
288               pz_pat = pTD->pz_test_text;
289               break;
290             }
291
292           pTD++;
293         }
294     }
295
296   /*
297    *  Replace every copy of the text we find
298    */
299   compile_re (pz_pat, &re, 1, "format search-text", "format_fix" );
300   while (regexec (&re, text, 10, rm, 0) == 0)
301     {
302       fwrite( text, rm[0].rm_so, 1, stdout );
303       format_write( pz_fmt, text, rm );
304       text += rm[0].rm_eo;
305     }
306
307   /*
308    *  Dump out the rest of the file
309    */
310   fputs (text, stdout);
311 }
312
313
314 /* Scan the input file for all occurrences of text like this:
315
316    #define TIOCCONS _IO(T, 12)
317
318    and change them to read like this:
319
320    #define TIOCCONS _IO('T', 12)
321
322    which is the required syntax per the C standard.  (The definition of
323    _IO also has to be tweaked - see below.)  'IO' is actually whatever you
324    provide as the `c_fix_arg' argument.  */
325
326 FIX_PROC_HEAD( char_macro_use_fix )
327 {
328   /* This regexp looks for a traditional-syntax #define (# in column 1)
329      of an object-like macro.  */
330   static const char pat[] =
331     "^#[ \t]*define[ \t]+[_A-Za-z][_A-Za-z0-9]*[ \t]+";
332   static regex_t re;
333
334   const char* str = p_fixd->patch_args[1];
335   regmatch_t rm[1];
336   const char *p, *limit;
337   size_t len;
338
339   if (str == NULL)
340     {
341       fprintf (stderr, zNeedsArg, p_fixd->fix_name, "ioctl type", 0);
342       exit (EXIT_BROKEN);
343     }
344
345   len = strlen (str);
346   compile_re (pat, &re, 1, "macro pattern", "char_macro_use_fix");
347
348   for (p = text;
349        regexec (&re, p, 1, rm, 0) == 0;
350        p = limit + 1)
351     {
352       /* p + rm[0].rm_eo is the first character of the macro replacement.
353          Find the end of the macro replacement, and the STR we were
354          sent to look for within the replacement.  */
355       p += rm[0].rm_eo;
356       limit = p - 1;
357       do
358         {
359           limit = strchr (limit + 1, '\n');
360           if (!limit)
361             goto done;
362         }
363       while (limit[-1] == '\\');
364
365       do
366         {
367           if (*p == str[0] && !strncmp (p+1, str+1, len-1))
368             goto found;
369         }
370       while (++p < limit - len);
371       /* Hit end of line.  */
372       continue;
373
374     found:
375       /* Found STR on this line.  If the macro needs fixing,
376          the next few chars will be whitespace or uppercase,
377          then an open paren, then a single letter.  */
378       while ((ISSPACE (*p) || ISUPPER (*p)) && p < limit) p++;
379       if (*p++ != '(')
380         continue;
381       if (!ISALPHA (*p))
382         continue;
383       if (ISALNUM (p[1]) || p[1] == '_')
384         continue;
385
386       /* Splat all preceding text into the output buffer,
387          quote the character at p, then proceed.  */
388       fwrite (text, 1, p - text, stdout);
389       putchar ('\'');
390       putchar (*p);
391       putchar ('\'');
392       text = p + 1;
393     }
394  done:
395   fputs (text, stdout);
396 }
397
398
399 /* Scan the input file for all occurrences of text like this:
400
401    #define xxxIOxx(x, y) (....'x'<<16....)
402
403    and change them to read like this:
404
405    #define xxxIOxx(x, y) (....x<<16....)
406
407    which is the required syntax per the C standard.  (The uses of _IO
408    also has to be tweaked - see above.)  'IO' is actually whatever
409    you provide as the `c_fix_arg' argument.  */
410 FIX_PROC_HEAD( char_macro_def_fix )
411 {
412   /* This regexp looks for any traditional-syntax #define (# in column 1).  */
413   static const char pat[] =
414     "^#[ \t]*define[ \t]+";
415   static regex_t re;
416
417   const char* str = p_fixd->patch_args[1];
418   regmatch_t rm[1];
419   const char *p, *limit;
420   char arg;
421   size_t len;
422
423   if (str == NULL)
424     {
425       fprintf (stderr, zNeedsArg, p_fixd->fix_name, "ioctl type", 0);
426       exit (EXIT_BROKEN);
427     }
428
429   len = strlen (str);
430   compile_re (pat, &re, 1, "macro pattern", "fix_char_macro_defines");
431
432   for (p = text;
433        regexec (&re, p, 1, rm, 0) == 0;
434        p = limit + 1)
435     {
436       /* p + rm[0].rm_eo is the first character of the macro name.
437          Find the end of the macro replacement, and the STR we were
438          sent to look for within the name.  */
439       p += rm[0].rm_eo;
440       limit = p - 1;
441       do
442         {
443           limit = strchr (limit + 1, '\n');
444           if (!limit)
445             goto done;
446         }
447       while (limit[-1] == '\\');
448
449       do
450         {
451           if (*p == str[0] && !strncmp (p+1, str+1, len-1))
452             goto found;
453           p++;
454         }
455       while (ISALPHA (*p) || ISALNUM (*p) || *p == '_');
456       /* Hit end of macro name without finding the string.  */
457       continue;
458
459     found:
460       /* Found STR in this macro name.  If the macro needs fixing,
461          there may be a few uppercase letters, then there will be an
462          open paren with _no_ intervening whitespace, and then a
463          single letter.  */
464       while (ISUPPER (*p) && p < limit) p++;
465       if (*p++ != '(')
466         continue;
467       if (!ISALPHA (*p))
468         continue;
469       if (ISALNUM (p[1]) || p[1] == '_')
470         continue;
471
472       /* The character at P is the one to look for in the following
473          text.  */
474       arg = *p;
475       p += 2;
476
477       while (p < limit)
478         {
479           if (p[-1] == '\'' && p[0] == arg && p[1] == '\'')
480             {
481               /* Remove the quotes from this use of ARG.  */
482               p--;
483               fwrite (text, 1, p - text, stdout);
484               putchar (arg);
485               p += 3;
486               text = p;
487             }
488           else
489             p++;
490         }
491     }
492  done:
493   fputs (text, stdout);
494 }
495
496 /* Fix for machine name #ifdefs that are not in the namespace reserved
497    by the C standard.  They won't be defined if compiling with -ansi,
498    and the headers will break.  We go to some trouble to only change
499    #ifdefs where the macro is defined by GCC in non-ansi mode; this
500    minimizes the number of headers touched.  */
501
502 #define SCRATCHSZ 64   /* hopefully long enough */
503
504 FIX_PROC_HEAD( machine_name_fix )
505 {
506 #ifndef MN_NAME_PAT
507   fputs( "The target machine has no needed machine name fixes\n", stderr );
508 #else
509   regmatch_t match[2];
510   const char *line, *base, *limit, *p, *q;
511   regex_t *label_re, *name_re;
512   char scratch[SCRATCHSZ];
513   size_t len;
514
515   mn_get_regexps (&label_re, &name_re, "machine_name_fix");
516
517   scratch[0] = '_';
518   scratch[1] = '_';
519
520   for (base = text;
521        regexec (label_re, base, 2, match, 0) == 0;
522        base = limit)
523     {
524       base += match[0].rm_eo;
525       /* We're looking at an #if or #ifdef.  Scan forward for the
526          next non-escaped newline.  */
527       line = limit = base;
528       do
529         {
530           limit++;
531           limit = strchr (limit, '\n');
532           if (!limit)
533             goto done;
534         }
535       while (limit[-1] == '\\');
536
537       /* If the 'name_pat' matches in between base and limit, we have
538          a bogon.  It is not worth the hassle of excluding comments
539          because comments on #if/#ifdef lines are rare, and strings on
540          such lines are illegal.
541
542          REG_NOTBOL means 'base' is not at the beginning of a line, which
543          shouldn't matter since the name_re has no ^ anchor, but let's
544          be accurate anyway.  */
545
546       for (;;)
547         {
548         again:
549           if (base == limit)
550             break;
551
552           if (regexec (name_re, base, 1, match, REG_NOTBOL))
553             goto done;  /* No remaining match in this file */
554
555           /* Match; is it on the line?  */
556           if (match[0].rm_eo > limit - base)
557             break;
558
559           p = base + match[0].rm_so;
560           base += match[0].rm_eo;
561
562           /* One more test: if on the same line we have the same string
563              with the appropriate underscores, then leave it alone.
564              We want exactly two leading and trailing underscores.  */
565           if (*p == '_')
566             {
567               len = base - p - ((*base == '_') ? 2 : 1);
568               q = p + 1;
569             }
570           else
571             {
572               len = base - p - ((*base == '_') ? 1 : 0);
573               q = p;
574             }
575           if (len + 4 > SCRATCHSZ)
576             abort ();
577           memcpy (&scratch[2], q, len);
578           len += 2;
579           scratch[len++] = '_';
580           scratch[len++] = '_';
581
582           for (q = line; q <= limit - len; q++)
583             if (*q == '_' && !strncmp (q, scratch, len))
584               goto again;
585           
586           fwrite (text, 1, p - text, stdout);
587           fwrite (scratch, 1, len, stdout);
588
589           text = base;
590         }
591     }
592  done:
593 #endif
594   fputs (text, stdout);
595 }
596
597
598 FIX_PROC_HEAD( wrap_fix )
599 {
600   tSCC   z_no_wrap_pat[] = "^#if.*__need_";
601   static regex_t no_wrapping_re = { NULL, 0, 0 };
602
603   char   z_fixname[ 64 ];
604   tCC*   pz_src  = p_fixd->fix_name;
605   tCC*   pz_name = z_fixname;
606   char*  pz_dst  = z_fixname;
607   int    do_end  = 0;
608   size_t len     = 0;
609
610   if (no_wrapping_re.allocated == 0)
611     compile_re( z_no_wrap_pat, &no_wrapping_re, 0, "no-wrap pattern",
612                 "wrap-fix" );
613
614   for (;;) {
615     char ch = *(pz_src++);
616
617     if (ISLOWER (ch))
618       *(pz_dst++) = TOUPPER ( ch );
619
620     else if (ISALNUM ( ch ))
621       *(pz_dst++) = ch;
622
623     else if (ch == NUL) {
624       *(pz_dst++) = ch;
625       break;
626     }
627     else
628       *(pz_dst++) = '_';
629
630     if (++len >= sizeof( z_fixname )) {
631       void* p = xmalloc( len + strlen( pz_src ) + 1 );
632       memcpy( p, (void*)z_fixname, len );
633       pz_name = (tCC*)p;
634       pz_dst  = (char*)pz_name + len;
635     }
636   }
637
638   /*
639    *  IF we do *not* match the no-wrap re, then we have a double negative.
640    *  A double negative means YES.
641    */
642   if (regexec (&no_wrapping_re, text, 0, NULL, 0) != 0)
643     {
644       printf( "#ifndef FIXINC_%s_CHECK\n", pz_name );
645       printf( "#define FIXINC_%s_CHECK 1\n\n", pz_name );
646       do_end = 1;
647     }
648
649   if (p_fixd->patch_args[1] == (tCC*)NULL)
650     fputs( text, stdout );
651
652   else {
653     fputs( p_fixd->patch_args[1], stdout );
654     fputs( text, stdout );
655     if (p_fixd->patch_args[2] != (tCC*)NULL)
656       fputs( p_fixd->patch_args[2], stdout );
657   }
658
659   if (do_end != 0)
660     printf( "\n#endif  /* FIXINC_%s_CHECK */\n", pz_name );
661
662   if (pz_name != z_fixname)
663     free( (void*)pz_name );
664 }
665
666
667 /*
668  *  Search for multiple copies of a regular expression.  Each block
669  *  of matched text is replaced with the format string, as described
670  *  above in `format_write'.
671  */
672 FIX_PROC_HEAD( gnu_type_fix )
673 {
674   const char* pz_pat;
675   regex_t    re;
676   regmatch_t rm[GTYPE_SE_CT+1];
677
678   {
679     tTestDesc* pTD = p_fixd->p_test_desc;
680     int        ct  = p_fixd->test_ct;
681     for (;;)
682       {
683         if (ct-- <= 0)
684           {
685             fprintf (stderr, zNeedsArg, p_fixd->fix_name, "search text", 1);
686             exit (EXIT_BROKEN);
687           }
688
689         if (pTD->type == TT_EGREP)
690           {
691             pz_pat = pTD->pz_test_text;
692             break;
693           }
694
695         pTD++;
696       }
697   }
698
699   compile_re (pz_pat, &re, 1, "gnu type typedef", "gnu_type_fix");
700
701   while (regexec (&re, text, GTYPE_SE_CT+1, rm, 0) == 0)
702     {
703       text = emit_gnu_type (text, rm);
704     }
705
706   /*
707    *  Dump out the rest of the file
708    */
709   fputs (text, stdout);
710 }
711
712
713 /* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
714
715      test for fix selector
716
717      THIS IS THE ONLY EXPORTED ROUTINE
718
719 */
720 void
721 apply_fix( p_fixd, filname )
722   tFixDesc* p_fixd;
723   tCC* filname;
724 {
725 #define _FT_(n,p) { n, p },
726   static fix_entry_t fix_table[] = { FIXUP_TABLE { NULL, NULL }};
727 #undef _FT_
728 #define FIX_TABLE_CT (ARRAY_SIZE (fix_table)-1)
729
730   tCC* fixname = p_fixd->patch_args[0];
731   char* buf;
732   int ct = FIX_TABLE_CT;
733   fix_entry_t* pfe = fix_table;
734
735   for (;;)
736     {
737       if (strcmp (pfe->fix_name, fixname) == 0)
738         break;
739       if (--ct <= 0)
740         {
741           fprintf (stderr, "fixincl error:  the `%s' fix is unknown\n",
742                    fixname );
743           exit (EXIT_BROKEN);
744         }
745       pfe++;
746     }
747
748   buf = load_file_data (stdin);
749   (*pfe->fix_proc)( filname, buf, p_fixd );
750 }
751
752 #ifdef SEPARATE_FIX_PROC
753 tSCC z_usage[] =
754 "USAGE: applyfix <fix-name> <file-to-fix> <file-source> <file-destination>\n";
755 tSCC z_reopen[] =
756 "FS error %d (%s) reopening %s as std%s\n";
757
758 int
759 main( argc, argv )
760   int     argc;
761   char**  argv;
762 {
763   tFixDesc* pFix;
764   char* pz_tmptmp;
765   char* pz_tmp_base;
766   char* pz_tmp_dot;
767
768   if (argc != 5)
769     {
770     usage_failure:
771       fputs (z_usage, stderr);
772       return EXIT_FAILURE;
773     }
774
775   {
776     char* pz = argv[1];
777     long  idx;
778
779     if (! ISDIGIT ( *pz ))
780       goto usage_failure;
781
782     idx = strtol (pz, &pz, 10);
783     if ((*pz != NUL) || ((unsigned)idx >= FIX_COUNT))
784       goto usage_failure;
785     pFix = fixDescList + idx;
786   }
787
788   if (freopen (argv[3], "r", stdin) != stdin)
789     {
790       fprintf (stderr, z_reopen, errno, strerror( errno ), argv[3], "in");
791       return EXIT_FAILURE;
792     }
793
794   pz_tmptmp = (char*)xmalloc( strlen( argv[4] ) + 5 );
795   strcpy( pz_tmptmp, argv[4] );
796
797   /* Don't lose because "12345678" and "12345678X" map to the same
798      file under DOS restricted 8+3 file namespace.  Note that DOS
799      doesn't allow more than one dot in the trunk of a file name.  */
800   pz_tmp_base = basename( pz_tmptmp );
801   pz_tmp_dot = strchr( pz_tmp_base, '.' );
802   if (pathconf( pz_tmptmp, _PC_NAME_MAX ) <= 12 /* is this DOS or Windows9X? */
803       && pz_tmp_dot != (char*)NULL)
804     strcpy (pz_tmp_dot+1, "X"); /* nuke the original extension */
805   else
806     strcat (pz_tmptmp, ".X");
807   if (freopen (pz_tmptmp, "w", stdout) != stdout)
808     {
809       fprintf (stderr, z_reopen, errno, strerror( errno ), pz_tmptmp, "out");
810       return EXIT_FAILURE;
811     }
812
813   apply_fix (pFix, argv[1]);
814   fclose (stdout);
815   fclose (stdin);
816   unlink (argv[4]);
817   if (rename (pz_tmptmp, argv[4]) != 0)
818     {
819       fprintf (stderr, "error %d (%s) renaming %s to %s\n", errno,
820                strerror( errno ), pz_tmptmp, argv[4]);
821       return EXIT_FAILURE;
822     }
823
824   return EXIT_SUCCESS;
825 }
826 #endif