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