1
2 /*
3 * CDDL HEADER START
4 *
5 * The contents of this file are subject to the terms of the
6 * Common Development and Distribution License (the "License").
7 * You may not use this file except in compliance with the License.
8 *
9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10 * or http://www.opensolaris.org/os/licensing.
11 * See the License for the specific language governing permissions
12 * and limitations under the License.
13 *
14 * When distributing Covered Code, include this CDDL HEADER in each
15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16 * If applicable, add the following below this CDDL HEADER, with the
17 * fields enclosed by brackets "[]" replaced with your own identifying
18 * information: Portions Copyright [yyyy] [name of copyright owner]
19 *
20 * CDDL HEADER END
21 */
22
23 /*
24 * Copyright 2018, Richard Lowe.
25 */
26 /*
27 * Copyright 2010 Sun Microsystems, Inc. All rights reserved.
28 * Use is subject to license terms.
29 *
30 * Copyright 2018 Joyent, Inc.
31 */
32
33 /*
34 * Wrapper for the GNU C compiler to make it accept the Sun C compiler
35 * arguments where possible.
36 *
37 * Since the translation is inexact, this is something of a work-in-progress.
38 *
39 */
40
41 /* If you modify this file, you must increment CW_VERSION */
42 #define CW_VERSION "3.0"
43
44 /*
45 * -# Verbose mode
46 * -### Show compiler commands built by driver, no compilation
47 * -A<name[(tokens)]> Preprocessor predicate assertion
48 * -B<[static|dynamic]> Specify dynamic or static binding
49 * -C Prevent preprocessor from removing comments
50 * -c Compile only - produce .o files, suppress linking
51 * -cg92 Alias for -xtarget=ss1000
52 * -D<name[=token]> Associate name with token as if by #define
53 * -d[y|n] dynamic [-dy] or static [-dn] option to linker
54 * -E Compile source through preprocessor only, output to stdout
55 * -erroff=<t> Suppress warnings specified by tags t(%none, %all, <tag list>)
56 * -errtags=<a> Display messages with tags a(no, yes)
57 * -errwarn=<t> Treats warnings specified by tags t(%none, %all, <tag list>)
58 * as errors
59 * -fast Optimize using a selection of options
60 * -fd Report old-style function definitions and declarations
61 * -fnonstd Initialize floating-point hardware to non-standard preferences
62 * -fns[=<yes|no>] Select non-standard floating point mode
63 * -fprecision=<p> Set FP rounding precision mode p(single, double, extended)
64 * -fround=<r> Select the IEEE rounding mode in effect at startup
65 * -fsimple[=<n>] Select floating-point optimization preferences <n>
66 * -fsingle Use single-precision arithmetic (-Xt and -Xs modes only)
67 * -ftrap=<t> Select floating-point trapping mode in effect at startup
68 * -fstore force floating pt. values to target precision on assignment
69 * -G Build a dynamic shared library
70 * -g Compile for debugging
71 * -H Print path name of each file included during compilation
72 * -h <name> Assign <name> to generated dynamic shared library
73 * -I<dir> Add <dir> to preprocessor #include file search path
74 * -i Passed to linker to ignore any LD_LIBRARY_PATH setting
75 * -keeptmp Keep temporary files created during compilation
76 * -L<dir> Pass to linker to add <dir> to the library search path
77 * -l<name> Link with library lib<name>.a or lib<name>.so
78 * -mc Remove duplicate strings from .comment section of output files
79 * -mr Remove all strings from .comment section of output files
80 * -mr,"string" Remove all strings and append "string" to .comment section
81 * -mt Specify options needed when compiling multi-threaded code
82 * -native Find available processor, generate code accordingly
83 * -nofstore Do not force floating pt. values to target precision
84 * on assignment
85 * -norunpath Do not build in a runtime path for shared libraries
86 * -O Use default optimization level (-xO2 or -xO3. Check man page.)
87 * -o <outputfile> Set name of output file to <outputfile>
88 * -P Compile source through preprocessor only, output to .i file
89 * -p Compile for profiling with prof
90 * -Q[y|n] Emit/don't emit identification info to output file
91 * -R<dir[:dir]> Build runtime search path list into executable
92 * -S Compile and only generate assembly code (.s)
93 * -s Strip symbol table from the executable file
94 * -t Turn off duplicate symbol warnings when linking
95 * -U<name> Delete initial definition of preprocessor symbol <name>
96 * -V Report version number of each compilation phase
97 * -v Do stricter semantic checking
98 * -W<c>,<arg> Pass <arg> to specified component <c> (a,l,m,p,0,2,h,i,u)
99 * -w Suppress compiler warning messages
100 * -Xa Compile assuming ANSI C conformance, allow K & R extensions
101 * (default mode)
102 * -Xs Compile assuming (pre-ANSI) K & R C style code
103 * -Xt Compile assuming K & R conformance, allow ANSI C
104 * -xarch=<a> Specify target architecture instruction set
105 * -xbuiltin[=<b>] When profitable inline, or substitute intrinisic functions
106 * for system functions, b={%all,%none}
107 * -xCC Accept C++ style comments
108 * -xchip=<c> Specify the target processor for use by the optimizer
109 * -xcode=<c> Generate different code for forming addresses
110 * -xcrossfile[=<n>] Enable optimization and inlining across source files,
111 * n={0|1}
112 * -xe Perform only syntax/semantic checking, no code generation
113 * -xF Compile for later mapfile reordering or unused section
114 * elimination
115 * -xhelp=<f> Display on-line help information f(flags, readme, errors)
116 * -xildoff Cancel -xildon
117 * -xildon Enable use of the incremental linker, ild
118 * -xinline=[<a>,...,<a>] Attempt inlining of specified user routines,
119 * <a>={%auto,func,no%func}
120 * -xlibmieee Force IEEE 754 return values for math routines in
121 * exceptional cases
122 * -xlibmil Inline selected libm math routines for optimization
123 * -xlic_lib=sunperf Link in the Sun supplied performance libraries
124 * -xlicinfo Show license server information
125 * -xmaxopt=[off,1,2,3,4,5] maximum optimization level allowed on #pragma opt
126 * -xO<n> Generate optimized code (n={1|2|3|4|5})
127 * -xP Print prototypes for function definitions
128 * -xprofile=<p> Collect data for a profile or use a profile to optimize
129 * <p>={{collect,use}[:<path>],tcov}
130 * -xregs=<r> Control register allocation
131 * -xs Allow debugging without object (.o) files
132 * -xsb Compile for use with the WorkShop source browser
133 * -xsbfast Generate only WorkShop source browser info, no compilation
134 * -xsfpconst Represent unsuffixed floating point constants as single
135 * precision
136 * -xspace Do not do optimizations that increase code size
137 * -xstrconst Place string literals into read-only data segment
138 * -xtarget=<t> Specify target system for optimization
139 * -xtemp=<dir> Set directory for temporary files to <dir>
140 * -xtime Report the execution time for each compilation phase
141 * -xunroll=n Enable unrolling loops n times where possible
142 * -Y<c>,<dir> Specify <dir> for location of component <c> (a,l,m,p,0,h,i,u)
143 * -YA,<dir> Change default directory searched for components
144 * -YI,<dir> Change default directory searched for include files
145 * -YP,<dir> Change default directory for finding libraries files
146 * -YS,<dir> Change default directory for startup object files
147 */
148
149 /*
150 * Translation table:
151 */
152 /*
153 * -# -v
154 * -### error
155 * -A<name[(tokens)]> pass-thru
156 * -B<[static|dynamic]> pass-thru (syntax error for anything else)
157 * -C pass-thru
158 * -c pass-thru
159 * -cg92 -m32 -mcpu=v8 -mtune=supersparc (SPARC only)
160 * -D<name[=token]> pass-thru
161 * -dy or -dn -Wl,-dy or -Wl,-dn
162 * -E pass-thru
163 * -erroff=E_EMPTY_TRANSLATION_UNIT ignore
164 * -errtags=%all -Wall
165 * -errwarn=%all -Werror else -Wno-error
166 * -fast error
167 * -fd error
168 * -fnonstd error
169 * -fns[=<yes|no>] error
170 * -fprecision=<p> error
171 * -fround=<r> error
172 * -fsimple[=<n>] error
173 * -fsingle[=<n>] error
174 * -ftrap=<t> error
175 * -fstore error
176 * -G pass-thru
177 * -g pass-thru
178 * -H pass-thru
179 * -h <name> pass-thru
180 * -I<dir> pass-thru
181 * -i pass-thru
182 * -keeptmp -save-temps
183 * -L<dir> pass-thru
184 * -l<name> pass-thru
185 * -mc error
186 * -mr error
187 * -mr,"string" error
188 * -mt -D_REENTRANT
189 * -native error
190 * -nofstore error
191 * -nolib -nodefaultlibs
192 * -norunpath ignore
193 * -O -O1 (Check the man page to be certain)
194 * -o <outputfile> pass-thru
195 * -P -E -o filename.i (or error)
196 * -p pass-thru
197 * -Q[y|n] error
198 * -R<dir[:dir]> pass-thru
199 * -S pass-thru
200 * -s -Wl,-s
201 * -t -Wl,-t
202 * -U<name> pass-thru
203 * -V --version
204 * -v -Wall
205 * -Wa,<arg> pass-thru
206 * -Wp,<arg> pass-thru except -xc99=<a>
207 * -Wl,<arg> pass-thru
208 * -W{m,0,2,h,i,u> error/ignore
209 * -xmodel=kernel -ffreestanding -mcmodel=kernel -mno-red-zone
210 * -Wu,-save_args -msave-args
211 * -w pass-thru
212 * -Xa -std=iso9899:199409 or -ansi
213 * -Xt error
214 * -Xs -traditional -std=c89
215 * -xarch=<a> table
216 * -xbuiltin[=<b>] -fbuiltin (-fno-builtin otherwise)
217 * -xCC ignore
218 * -xchip=<c> table
219 * -xcode=<c> table
220 * -xdebugformat=<format> ignore (always use dwarf-2 for gcc)
221 * -xcrossfile[=<n>] ignore
222 * -xe error
223 * -xF error
224 * -xhelp=<f> error
225 * -xildoff ignore
226 * -xildon ignore
227 * -xinline ignore
228 * -xlibmieee error
229 * -xlibmil error
230 * -xlic_lib=sunperf error
231 * -xmaxopt=[...] error
232 * -xO<n> -O<n>
233 * -xP error
234 * -xprofile=<p> error
235 * -xregs=<r> table
236 * -xs error
237 * -xsb error
238 * -xsbfast error
239 * -xsfpconst error
240 * -xspace ignore (-not -Os)
241 * -xstrconst ignore
242 * -xtarget=<t> table
243 * -xtemp=<dir> error
244 * -xtime error
245 * -xtransition -Wtransition
246 * -xunroll=n error
247 * -W0,-xdbggen=no%usedonly -fno-eliminate-unused-debug-symbols
248 * -fno-eliminate-unused-debug-types
249 * -Y<c>,<dir> error
250 * -YA,<dir> error
251 * -YI,<dir> -nostdinc -I<dir>
252 * -YP,<dir> error
253 * -YS,<dir> error
254 */
255
256 #include <ctype.h>
257 #include <err.h>
258 #include <errno.h>
259 #include <fcntl.h>
260 #include <getopt.h>
261 #include <stdio.h>
262 #include <stdlib.h>
263 #include <string.h>
264 #include <unistd.h>
265
266 #include <sys/param.h>
267 #include <sys/stat.h>
268 #include <sys/types.h>
269 #include <sys/utsname.h>
270 #include <sys/wait.h>
271
272 #define CW_F_CXX 0x01
273 #define CW_F_SHADOW 0x02
274 #define CW_F_EXEC 0x04
275 #define CW_F_ECHO 0x08
276 #define CW_F_XLATE 0x10
277 #define CW_F_PROG 0x20
278
279 typedef enum cw_op {
280 CW_O_NONE = 0,
281 CW_O_PREPROCESS,
282 CW_O_COMPILE,
283 CW_O_LINK
284 } cw_op_t;
285
286 struct aelist {
287 struct ae {
288 struct ae *ae_next;
289 char *ae_arg;
290 } *ael_head, *ael_tail;
291 int ael_argc;
292 };
293
294 typedef enum {
295 GNU,
296 SUN,
297 SMATCH
298 } compiler_style_t;
299
300 typedef struct {
301 char *c_name;
302 char *c_path;
303 compiler_style_t c_style;
304 } cw_compiler_t;
305
306 typedef struct cw_ictx {
307 struct cw_ictx *i_next;
308 cw_compiler_t *i_compiler;
309 struct aelist *i_ae;
310 uint32_t i_flags;
311 int i_oldargc;
312 char **i_oldargv;
313 pid_t i_pid;
314 char i_discard[MAXPATHLEN];
315 char *i_stderr;
316 } cw_ictx_t;
317
318 /*
319 * Status values to indicate which Studio compiler and associated
320 * flags are being used.
321 */
322 #define M32 0x01 /* -m32 - only on Studio 12 */
323 #define M64 0x02 /* -m64 - only on Studio 12 */
324 #define SS11 0x100 /* Studio 11 */
325 #define SS12 0x200 /* Studio 12 */
326
327 #define TRANS_ENTRY 5
328 /*
329 * Translation table definition for the -xarch= flag. The "x_arg"
330 * value is translated into the appropriate gcc flags according
331 * to the values in x_trans[n]. The x_flags indicates what compiler
332 * is being used and what flags have been set via the use of
333 * "x_arg".
334 */
335 typedef struct xarch_table {
336 char *x_arg;
337 int x_flags;
338 char *x_trans[TRANS_ENTRY];
339 } xarch_table_t;
340
341 /*
342 * The translation table for the -xarch= flag used in the Studio compilers.
343 */
344 static const xarch_table_t xtbl[] = {
345 #if defined(__x86)
346 { "generic", SS11, {NULL} },
347 { "generic64", (SS11|M64), { "-m64", "-mtune=opteron" } },
348 { "amd64", (SS11|M64), { "-m64", "-mtune=opteron" } },
349 { "386", SS11, { "-march=i386" } },
350 { "pentium_pro", SS11, { "-march=pentiumpro" } },
351 { "sse", SS11, { "-msse", "-mfpmath=sse" } },
352 { "sse2", SS11, { "-msse2", "-mfpmath=sse" } },
353 #elif defined(__sparc)
354 { "generic", (SS11|M32), { "-m32", "-mcpu=v8" } },
355 { "generic64", (SS11|M64), { "-m64", "-mcpu=v9" } },
356 { "v8", (SS11|M32), { "-m32", "-mcpu=v8", "-mno-v8plus" } },
357 { "v8plus", (SS11|M32), { "-m32", "-mcpu=v9", "-mv8plus" } },
358 { "v8plusa", (SS11|M32), { "-m32", "-mcpu=ultrasparc", "-mv8plus",
359 "-mvis" } },
360 { "v8plusb", (SS11|M32), { "-m32", "-mcpu=ultrasparc3", "-mv8plus",
361 "-mvis" } },
362 { "v9", (SS11|M64), { "-m64", "-mcpu=v9" } },
363 { "v9a", (SS11|M64), { "-m64", "-mcpu=ultrasparc", "-mvis" } },
364 { "v9b", (SS11|M64), { "-m64", "-mcpu=ultrasparc3", "-mvis" } },
365 { "sparc", SS12, { "-mcpu=v9", "-mv8plus" } },
366 { "sparcvis", SS12, { "-mcpu=ultrasparc", "-mvis" } },
367 { "sparcvis2", SS12, { "-mcpu=ultrasparc3", "-mvis" } }
368 #endif
369 };
370
371 static int xtbl_size = sizeof (xtbl) / sizeof (xarch_table_t);
372
373 static const char *xchip_tbl[] = {
374 #if defined(__x86)
375 "386", "-mtune=i386", NULL,
376 "486", "-mtune=i486", NULL,
377 "pentium", "-mtune=pentium", NULL,
378 "pentium_pro", "-mtune=pentiumpro", NULL,
379 #elif defined(__sparc)
380 "super", "-mtune=supersparc", NULL,
381 "ultra", "-mtune=ultrasparc", NULL,
382 "ultra3", "-mtune=ultrasparc3", NULL,
383 #endif
384 NULL, NULL
385 };
386
387 static const char *xcode_tbl[] = {
388 #if defined(__sparc)
389 "abs32", "-fno-pic", "-mcmodel=medlow", NULL,
390 "abs44", "-fno-pic", "-mcmodel=medmid", NULL,
391 "abs64", "-fno-pic", "-mcmodel=medany", NULL,
392 "pic13", "-fpic", NULL,
393 "pic32", "-fPIC", NULL,
394 #endif
395 NULL, NULL
396 };
397
398 static const char *xtarget_tbl[] = {
399 #if defined(__x86)
400 "pentium_pro", "-march=pentiumpro", NULL,
401 #endif /* __x86 */
402 NULL, NULL
403 };
404
405 static const char *xregs_tbl[] = {
406 #if defined(__sparc)
407 "appl", "-mapp-regs", NULL,
408 "no%appl", "-mno-app-regs", NULL,
409 "float", "-mfpu", NULL,
410 "no%float", "-mno-fpu", NULL,
411 #endif /* __sparc */
412 NULL, NULL
413 };
414
415 static void
416 nomem(void)
417 {
418 errx(1, "out of memory");
419 }
420
421 static void
422 newae(struct aelist *ael, const char *arg)
423 {
424 struct ae *ae;
425
426 if ((ae = calloc(sizeof (*ae), 1)) == NULL)
427 nomem();
428 ae->ae_arg = strdup(arg);
429 if (ael->ael_tail == NULL)
430 ael->ael_head = ae;
431 else
432 ael->ael_tail->ae_next = ae;
433 ael->ael_tail = ae;
434 ael->ael_argc++;
435 }
436
437 static cw_ictx_t *
438 newictx(void)
439 {
440 cw_ictx_t *ctx = calloc(sizeof (cw_ictx_t), 1);
441 if (ctx)
442 if ((ctx->i_ae = calloc(sizeof (struct aelist), 1)) == NULL) {
443 free(ctx);
444 return (NULL);
445 }
446
447 return (ctx);
448 }
449
450 static void
451 error(const char *arg)
452 {
453 errx(2, "error: mapping failed at or near arg '%s'", arg);
454 }
455
456 /*
457 * Add the current favourite set of warnings to the gcc invocation.
458 */
459 static void
460 warnings(struct aelist *h)
461 {
462 static int warningsonce;
463
464 if (warningsonce++)
465 return;
466
467 /*
468 * Enable as many warnings as exist, then disable those that we never
469 * ever want.
470 */
471 newae(h, "-Wall");
472 newae(h, "-Wextra");
473 }
474
475 static void
476 optim_disable(struct aelist *h, int level)
477 {
478 if (level >= 2) {
479 newae(h, "-fno-strict-aliasing");
480 newae(h, "-fno-unit-at-a-time");
481 newae(h, "-fno-optimize-sibling-calls");
482 }
483 }
484
485 static void
486 Xsmode(struct aelist *h)
487 {
488 static int xsonce;
489
490 if (xsonce++)
491 return;
492
493 newae(h, "-traditional");
494 newae(h, "-traditional-cpp");
495 }
496
497 static void
498 usage()
499 {
500 extern char *__progname;
501 (void) fprintf(stderr,
502 "usage: %s [-C] [--versions] --primary <compiler> "
503 "[--shadow <compiler>]... -- cflags...\n",
504 __progname);
505 (void) fprintf(stderr, "compilers take the form: name,path,style\n"
506 " - name: a unique name usable in flag specifiers\n"
507 " - path: path to the compiler binary\n"
508 " - style: the style of flags expected: either sun or gnu\n");
509 exit(2);
510 }
511
512 static int
513 xlate_xtb(struct aelist *h, const char *xarg)
514 {
515 int i, j;
516
517 for (i = 0; i < xtbl_size; i++) {
518 if (strcmp(xtbl[i].x_arg, xarg) == 0)
519 break;
520 }
521
522 /*
523 * At the end of the table and so no matching "arg" entry
524 * found and so this must be a bad -xarch= flag.
525 */
526 if (i == xtbl_size)
527 error(xarg);
528
529 for (j = 0; j < TRANS_ENTRY; j++) {
530 if (xtbl[i].x_trans[j] != NULL)
531 newae(h, xtbl[i].x_trans[j]);
532 else
533 break;
534 }
535 return (xtbl[i].x_flags);
536
537 }
538
539 static void
540 xlate(struct aelist *h, const char *xarg, const char **table)
541 {
542 while (*table != NULL && strcmp(xarg, *table) != 0) {
543 while (*table != NULL)
544 table++;
545 table++;
546 }
547
548 if (*table == NULL)
549 error(xarg);
550
551 table++;
552
553 while (*table != NULL) {
554 newae(h, *table);
555 table++;
556 }
557 }
558
559 static void
560 do_gcc(cw_ictx_t *ctx)
561 {
562 int c;
563 int nolibc = 0;
564 int in_output = 0, seen_o = 0, c_files = 0;
565 cw_op_t op = CW_O_LINK;
566 char *model = NULL;
567 char *nameflag;
568 int mflag = 0;
569
570 if (ctx->i_flags & CW_F_PROG) {
571 newae(ctx->i_ae, "--version");
572 return;
573 }
574
575 newae(ctx->i_ae, "-fident");
576 newae(ctx->i_ae, "-finline");
577 newae(ctx->i_ae, "-fno-inline-functions");
578 newae(ctx->i_ae, "-fno-builtin");
579 newae(ctx->i_ae, "-fno-asm");
580 newae(ctx->i_ae, "-fdiagnostics-show-option");
581 newae(ctx->i_ae, "-nodefaultlibs");
582
583 #if defined(__sparc)
584 /*
585 * The SPARC ldd and std instructions require 8-byte alignment of
586 * their address operand. gcc correctly uses them only when the
587 * ABI requires 8-byte alignment; unfortunately we have a number of
588 * pieces of buggy code that doesn't conform to the ABI. This
589 * flag makes gcc work more like Studio with -xmemalign=4.
590 */
591 newae(ctx->i_ae, "-mno-integer-ldd-std");
592 #endif
593
594 /*
595 * This is needed because 'u' is defined
596 * under a conditional on 'sun'. Should
597 * probably just remove the conditional,
598 * or make it be dependent on '__sun'.
599 *
600 * -Dunix is also missing in enhanced ANSI mode
601 */
602 newae(ctx->i_ae, "-D__sun");
603
604 if (asprintf(&nameflag, "-_%s=", ctx->i_compiler->c_name) == -1)
605 nomem();
606
607 /*
608 * Walk the argument list, translating as we go ..
609 */
610 while (--ctx->i_oldargc > 0) {
611 char *arg = *++ctx->i_oldargv;
612 size_t arglen = strlen(arg);
613
614 if (*arg == '-') {
615 arglen--;
616 } else {
617 /*
618 * Discard inline files that gcc doesn't grok
619 */
620 if (!in_output && arglen > 3 &&
621 strcmp(arg + arglen - 3, ".il") == 0)
622 continue;
623
624 if (!in_output && arglen > 2 &&
625 arg[arglen - 2] == '.' &&
626 (arg[arglen - 1] == 'S' || arg[arglen - 1] == 's' ||
627 arg[arglen - 1] == 'c' || arg[arglen - 1] == 'i'))
628 c_files++;
629
630 /*
631 * Otherwise, filenames and partial arguments
632 * are passed through for gcc to chew on. However,
633 * output is always discarded for the secondary
634 * compiler.
635 */
636 if ((ctx->i_flags & CW_F_SHADOW) && in_output)
637 newae(ctx->i_ae, ctx->i_discard);
638 else
639 newae(ctx->i_ae, arg);
640 in_output = 0;
641 continue;
642 }
643
644 if (ctx->i_flags & CW_F_CXX) {
645 if (strncmp(arg, "-_g++=", 6) == 0) {
646 newae(ctx->i_ae, strchr(arg, '=') + 1);
647 continue;
648 }
649 if (strncmp(arg, "-compat=", 8) == 0) {
650 /* discard -compat=4 and -compat=5 */
651 continue;
652 }
653 if (strcmp(arg, "-Qoption") == 0) {
654 /* discard -Qoption and its two arguments */
655 if (ctx->i_oldargc < 3)
656 error(arg);
657 ctx->i_oldargc -= 2;
658 ctx->i_oldargv += 2;
659 continue;
660 }
661 if (strcmp(arg, "-xwe") == 0) {
662 /* turn warnings into errors */
663 newae(ctx->i_ae, "-Werror");
664 continue;
665 }
666 if (strcmp(arg, "-norunpath") == 0) {
667 /* gcc has no corresponding option */
668 continue;
669 }
670 if (strcmp(arg, "-nolib") == 0) {
671 /* -nodefaultlibs is on by default */
672 nolibc = 1;
673 continue;
674 }
675 #if defined(__sparc)
676 if (strcmp(arg, "-cg92") == 0) {
677 mflag |= xlate_xtb(ctx->i_ae, "v8");
678 xlate(ctx->i_ae, "super", xchip_tbl);
679 continue;
680 }
681 #endif /* __sparc */
682 }
683
684 switch ((c = arg[1])) {
685 case '_':
686 if ((strncmp(arg, nameflag, strlen(nameflag)) == 0) ||
687 (strncmp(arg, "-_gcc=", 6) == 0) ||
688 (strncmp(arg, "-_gnu=", 6) == 0)) {
689 newae(ctx->i_ae, strchr(arg, '=') + 1);
690 }
691 break;
692 case '#':
693 if (arglen == 1) {
694 newae(ctx->i_ae, "-v");
695 break;
696 }
697 error(arg);
698 break;
699 case 'f':
700 if ((strcmp(arg, "-fpic") == 0) ||
701 (strcmp(arg, "-fPIC") == 0)) {
702 newae(ctx->i_ae, arg);
703 break;
704 }
705 error(arg);
706 break;
707 case 'g':
708 newae(ctx->i_ae, "-gdwarf-2");
709 break;
710 case 'E':
711 if (arglen == 1) {
712 newae(ctx->i_ae, "-xc");
713 newae(ctx->i_ae, arg);
714 op = CW_O_PREPROCESS;
715 nolibc = 1;
716 break;
717 }
718 error(arg);
719 break;
720 case 'c':
721 case 'S':
722 if (arglen == 1) {
723 op = CW_O_COMPILE;
724 nolibc = 1;
725 }
726 /* FALLTHROUGH */
727 case 'C':
728 case 'H':
729 case 'p':
730 if (arglen == 1) {
731 newae(ctx->i_ae, arg);
732 break;
733 }
734 error(arg);
735 break;
736 case 'A':
737 case 'h':
738 case 'I':
739 case 'i':
740 case 'L':
741 case 'l':
742 case 'R':
743 case 'U':
744 case 'u':
745 case 'w':
746 newae(ctx->i_ae, arg);
747 break;
748 case 'o':
749 seen_o = 1;
750 if (arglen == 1) {
751 in_output = 1;
752 newae(ctx->i_ae, arg);
753 } else if (ctx->i_flags & CW_F_SHADOW) {
754 newae(ctx->i_ae, "-o");
755 newae(ctx->i_ae, ctx->i_discard);
756 } else {
757 newae(ctx->i_ae, arg);
758 }
759 break;
760 case 'D':
761 newae(ctx->i_ae, arg);
762 /*
763 * XXX Clearly a hack ... do we need _KADB too?
764 */
765 if (strcmp(arg, "-D_KERNEL") == 0 ||
766 strcmp(arg, "-D_BOOT") == 0)
767 newae(ctx->i_ae, "-ffreestanding");
768 break;
769 case 'd':
770 if (arglen == 2) {
771 if (strcmp(arg, "-dy") == 0) {
772 newae(ctx->i_ae, "-Wl,-dy");
773 break;
774 }
775 if (strcmp(arg, "-dn") == 0) {
776 newae(ctx->i_ae, "-Wl,-dn");
777 break;
778 }
779 }
780 if (strcmp(arg, "-dalign") == 0) {
781 /*
782 * -dalign forces alignment in some cases;
783 * gcc does not need any flag to do this.
784 */
785 break;
786 }
787 error(arg);
788 break;
789 case 'e':
790 if (strcmp(arg,
791 "-erroff=E_EMPTY_TRANSLATION_UNIT") == 0) {
792 /*
793 * Accept but ignore this -- gcc doesn't
794 * seem to complain about empty translation
795 * units
796 */
797 break;
798 }
799 /* XX64 -- ignore all -erroff= options, for now */
800 if (strncmp(arg, "-erroff=", 8) == 0)
801 break;
802 if (strcmp(arg, "-errtags=yes") == 0) {
803 warnings(ctx->i_ae);
804 break;
805 }
806 if (strcmp(arg, "-errwarn=%all") == 0) {
807 newae(ctx->i_ae, "-Werror");
808 break;
809 }
810 error(arg);
811 break;
812 case 'G':
813 newae(ctx->i_ae, "-shared");
814 nolibc = 1;
815 break;
816 case 'k':
817 if (strcmp(arg, "-keeptmp") == 0) {
818 newae(ctx->i_ae, "-save-temps");
819 break;
820 }
821 error(arg);
822 break;
823 case 'm':
824 if (strcmp(arg, "-mt") == 0) {
825 newae(ctx->i_ae, "-D_REENTRANT");
826 break;
827 }
828 if (strcmp(arg, "-m64") == 0) {
829 newae(ctx->i_ae, "-m64");
830 #if defined(__x86)
831 newae(ctx->i_ae, "-mtune=opteron");
832 #endif
833 mflag |= M64;
834 break;
835 }
836 if (strcmp(arg, "-m32") == 0) {
837 newae(ctx->i_ae, "-m32");
838 mflag |= M32;
839 break;
840 }
841 error(arg);
842 break;
843 case 'B': /* linker options */
844 case 'M':
845 case 'z':
846 {
847 char *opt;
848 size_t len;
849 char *s;
850
851 if (arglen == 1) {
852 opt = *++ctx->i_oldargv;
853 if (opt == NULL || *opt == '\0')
854 error(arg);
855 ctx->i_oldargc--;
856 } else {
857 opt = arg + 2;
858 }
859 len = strlen(opt) + 7;
860 if ((s = malloc(len)) == NULL)
861 nomem();
862 (void) snprintf(s, len, "-Wl,-%c%s", c, opt);
863 newae(ctx->i_ae, s);
864 free(s);
865 }
866 break;
867 case 'O':
868 if (arglen == 1) {
869 newae(ctx->i_ae, "-O");
870 break;
871 }
872 error(arg);
873 break;
874 case 'P':
875 /*
876 * We could do '-E -o filename.i', but that's hard,
877 * and we don't need it for the case that's triggering
878 * this addition. We'll require the user to specify
879 * -o in the Makefile. If they don't they'll find out
880 * in a hurry.
881 */
882 newae(ctx->i_ae, "-E");
883 op = CW_O_PREPROCESS;
884 nolibc = 1;
885 break;
886 case 's':
887 if (arglen == 1) {
888 newae(ctx->i_ae, "-Wl,-s");
889 break;
890 }
891 error(arg);
892 break;
893 case 't':
894 if (arglen == 1) {
895 newae(ctx->i_ae, "-Wl,-t");
896 break;
897 }
898 error(arg);
899 break;
900 case 'V':
901 if (arglen == 1) {
902 ctx->i_flags &= ~CW_F_ECHO;
903 newae(ctx->i_ae, "--version");
904 break;
905 }
906 error(arg);
907 break;
908 case 'v':
909 if (arglen == 1) {
910 warnings(ctx->i_ae);
911 break;
912 }
913 error(arg);
914 break;
915 case 'W':
916 if (strncmp(arg, "-Wp,-xc99", 9) == 0) {
917 /*
918 * gcc's preprocessor will accept c99
919 * regardless, so accept and ignore.
920 */
921 break;
922 }
923 if (strncmp(arg, "-Wa,", 4) == 0 ||
924 strncmp(arg, "-Wp,", 4) == 0 ||
925 strncmp(arg, "-Wl,", 4) == 0) {
926 newae(ctx->i_ae, arg);
927 break;
928 }
929 if (strcmp(arg, "-W0,-noglobal") == 0 ||
930 strcmp(arg, "-W0,-xglobalstatic") == 0) {
931 /*
932 * gcc doesn't prefix local symbols
933 * in debug mode, so this is not needed.
934 */
935 break;
936 }
937 if (strcmp(arg, "-W0,-Lt") == 0) {
938 /*
939 * Generate tests at the top of loops.
940 * There is no direct gcc equivalent, ignore.
941 */
942 break;
943 }
944 if (strcmp(arg, "-W0,-xdbggen=no%usedonly") == 0) {
945 newae(ctx->i_ae,
946 "-fno-eliminate-unused-debug-symbols");
947 newae(ctx->i_ae,
948 "-fno-eliminate-unused-debug-types");
949 break;
950 }
951 if (strcmp(arg, "-W2,-xwrap_int") == 0) {
952 /*
953 * Use the legacy behaviour (pre-SS11)
954 * for integer wrapping.
955 * gcc does not need this.
956 */
957 break;
958 }
959 if (strcmp(arg, "-Wd,-xsafe=unboundsym") == 0) {
960 /*
961 * Prevents optimizing away checks for
962 * unbound weak symbol addresses. gcc does
963 * not do this, so it's not needed.
964 */
965 break;
966 }
967 if (strncmp(arg, "-Wc,-xcode=", 11) == 0) {
968 xlate(ctx->i_ae, arg + 11, xcode_tbl);
969 break;
970 }
971 if (strncmp(arg, "-Wc,-Qiselect", 13) == 0) {
972 /*
973 * Prevents insertion of register symbols.
974 * gcc doesn't do this, so ignore it.
975 */
976 break;
977 }
978 if (strcmp(arg, "-Wc,-Qassembler-ounrefsym=0") == 0) {
979 /*
980 * Prevents optimizing away of static variables.
981 * gcc does not do this, so it's not needed.
982 */
983 break;
984 }
985 #if defined(__x86)
986 if (strcmp(arg, "-Wu,-save_args") == 0) {
987 newae(ctx->i_ae, "-msave-args");
988 break;
989 }
990 #endif /* __x86 */
991 error(arg);
992 break;
993 case 'X':
994 if (strcmp(arg, "-Xa") == 0 ||
995 strcmp(arg, "-Xt") == 0) {
996 break;
997 }
998 if (strcmp(arg, "-Xs") == 0) {
999 Xsmode(ctx->i_ae);
1000 break;
1001 }
1002 error(arg);
1003 break;
1004 case 'x':
1005 if (arglen == 1)
1006 error(arg);
1007 switch (arg[2]) {
1008 case 'a':
1009 if (strncmp(arg, "-xarch=", 7) == 0) {
1010 mflag |= xlate_xtb(ctx->i_ae, arg + 7);
1011 break;
1012 }
1013 error(arg);
1014 break;
1015 case 'b':
1016 if (strncmp(arg, "-xbuiltin=", 10) == 0) {
1017 if (strcmp(arg + 10, "%all"))
1018 newae(ctx->i_ae, "-fbuiltin");
1019 break;
1020 }
1021 error(arg);
1022 break;
1023 case 'C':
1024 /* Accept C++ style comments -- ignore */
1025 if (strcmp(arg, "-xCC") == 0)
1026 break;
1027 error(arg);
1028 break;
1029 case 'c':
1030 if (strncmp(arg, "-xc99=%all", 10) == 0) {
1031 newae(ctx->i_ae, "-std=gnu99");
1032 break;
1033 }
1034 if (strncmp(arg, "-xc99=%none", 11) == 0) {
1035 newae(ctx->i_ae, "-std=gnu89");
1036 break;
1037 }
1038 if (strncmp(arg, "-xchip=", 7) == 0) {
1039 xlate(ctx->i_ae, arg + 7, xchip_tbl);
1040 break;
1041 }
1042 if (strncmp(arg, "-xcode=", 7) == 0) {
1043 xlate(ctx->i_ae, arg + 7, xcode_tbl);
1044 break;
1045 }
1046 if (strncmp(arg, "-xcrossfile", 11) == 0)
1047 break;
1048 error(arg);
1049 break;
1050 case 'd':
1051 if (strncmp(arg, "-xdebugformat=", 14) == 0)
1052 break;
1053 error(arg);
1054 break;
1055 case 'F':
1056 /*
1057 * Compile for mapfile reordering, or unused
1058 * section elimination, syntax can be -xF or
1059 * more complex, like -xF=%all -- ignore.
1060 */
1061 if (strncmp(arg, "-xF", 3) == 0)
1062 break;
1063 error(arg);
1064 break;
1065 case 'i':
1066 if (strncmp(arg, "-xinline", 8) == 0)
1067 /* No inlining; ignore */
1068 break;
1069 if (strcmp(arg, "-xildon") == 0 ||
1070 strcmp(arg, "-xildoff") == 0)
1071 /* No incremental linking; ignore */
1072 break;
1073 error(arg);
1074 break;
1075 #if defined(__x86)
1076 case 'm':
1077 if (strcmp(arg, "-xmodel=kernel") == 0) {
1078 newae(ctx->i_ae, "-ffreestanding");
1079 newae(ctx->i_ae, "-mno-red-zone");
1080 model = "-mcmodel=kernel";
1081 nolibc = 1;
1082 break;
1083 }
1084 error(arg);
1085 break;
1086 #endif /* __x86 */
1087 case 'O':
1088 if (strncmp(arg, "-xO", 3) == 0) {
1089 size_t len = strlen(arg);
1090 char *s = NULL;
1091 int c = *(arg + 3);
1092 int level;
1093
1094 if (len != 4 || !isdigit(c))
1095 error(arg);
1096
1097 level = atoi(arg + 3);
1098 if (level > 5)
1099 error(arg);
1100 if (level >= 2) {
1101 /*
1102 * For gcc-3.4.x at -O2 we
1103 * need to disable optimizations
1104 * that break ON.
1105 */
1106 optim_disable(ctx->i_ae, level);
1107 /*
1108 * limit -xO3 to -O2 as well.
1109 */
1110 level = 2;
1111 }
1112 if (asprintf(&s, "-O%d", level) == -1)
1113 nomem();
1114 newae(ctx->i_ae, s);
1115 free(s);
1116 break;
1117 }
1118 error(arg);
1119 break;
1120 case 'r':
1121 if (strncmp(arg, "-xregs=", 7) == 0) {
1122 xlate(ctx->i_ae, arg + 7, xregs_tbl);
1123 break;
1124 }
1125 error(arg);
1126 break;
1127 case 's':
1128 if (strcmp(arg, "-xs") == 0 ||
1129 strcmp(arg, "-xspace") == 0 ||
1130 strcmp(arg, "-xstrconst") == 0)
1131 break;
1132 error(arg);
1133 break;
1134 case 't':
1135 if (strncmp(arg, "-xtarget=", 9) == 0) {
1136 xlate(ctx->i_ae, arg + 9, xtarget_tbl);
1137 break;
1138 }
1139 error(arg);
1140 break;
1141 case 'e':
1142 case 'h':
1143 case 'l':
1144 default:
1145 error(arg);
1146 break;
1147 }
1148 break;
1149 case 'Y':
1150 if (arglen == 1) {
1151 if ((arg = *++ctx->i_oldargv) == NULL ||
1152 *arg == '\0')
1153 error("-Y");
1154 ctx->i_oldargc--;
1155 arglen = strlen(arg + 1);
1156 } else {
1157 arg += 2;
1158 }
1159 /* Just ignore -YS,... for now */
1160 if (strncmp(arg, "S,", 2) == 0)
1161 break;
1162 if (strncmp(arg, "l,", 2) == 0) {
1163 char *s = strdup(arg);
1164 s[0] = '-';
1165 s[1] = 'B';
1166 newae(ctx->i_ae, s);
1167 free(s);
1168 break;
1169 }
1170 if (strncmp(arg, "I,", 2) == 0) {
1171 char *s = strdup(arg);
1172 s[0] = '-';
1173 s[1] = 'I';
1174 newae(ctx->i_ae, "-nostdinc");
1175 newae(ctx->i_ae, s);
1176 free(s);
1177 break;
1178 }
1179 error(arg);
1180 break;
1181 case 'Q':
1182 /*
1183 * We could map -Qy into -Wl,-Qy etc.
1184 */
1185 default:
1186 error(arg);
1187 break;
1188 }
1189 }
1190
1191 free(nameflag);
1192
1193 if (c_files > 1 && (ctx->i_flags & CW_F_SHADOW) &&
1194 op != CW_O_PREPROCESS) {
1195 errx(2, "multiple source files are "
1196 "allowed only with -E or -P");
1197 }
1198
1199 /*
1200 * Make sure that we do not have any unintended interactions between
1201 * the xarch options passed in and the version of the Studio compiler
1202 * used.
1203 */
1204 if ((mflag & (SS11|SS12)) == (SS11|SS12)) {
1205 errx(2,
1206 "Conflicting \"-xarch=\" flags (both Studio 11 and 12)\n");
1207 }
1208
1209 switch (mflag) {
1210 case 0:
1211 /* FALLTHROUGH */
1212 case M32:
1213 #if defined(__sparc)
1214 /*
1215 * Only -m32 is defined and so put in the missing xarch
1216 * translation.
1217 */
1218 newae(ctx->i_ae, "-mcpu=v8");
1219 newae(ctx->i_ae, "-mno-v8plus");
1220 #endif
1221 break;
1222 case M64:
1223 #if defined(__sparc)
1224 /*
1225 * Only -m64 is defined and so put in the missing xarch
1226 * translation.
1227 */
1228 newae(ctx->i_ae, "-mcpu=v9");
1229 #endif
1230 break;
1231 case SS12:
1232 #if defined(__sparc)
1233 /* no -m32/-m64 flag used - this is an error for sparc builds */
1234 (void) fprintf(stderr, "No -m32/-m64 flag defined\n");
1235 exit(2);
1236 #endif
1237 break;
1238 case SS11:
1239 /* FALLTHROUGH */
1240 case (SS11|M32):
1241 case (SS11|M64):
1242 break;
1243 case (SS12|M32):
1244 #if defined(__sparc)
1245 /*
1246 * Need to add in further 32 bit options because with SS12
1247 * the xarch=sparcvis option can be applied to 32 or 64
1248 * bit, and so the translatation table (xtbl) cannot handle
1249 * that.
1250 */
1251 newae(ctx->i_ae, "-mv8plus");
1252 #endif
1253 break;
1254 case (SS12|M64):
1255 break;
1256 default:
1257 (void) fprintf(stderr,
1258 "Incompatible -xarch= and/or -m32/-m64 options used.\n");
1259 exit(2);
1260 }
1261
1262 if ((op == CW_O_LINK || op == CW_O_PREPROCESS) &&
1263 (ctx->i_flags & CW_F_SHADOW))
1264 exit(0);
1265
1266 if (model != NULL)
1267 newae(ctx->i_ae, model);
1268 if (!nolibc)
1269 newae(ctx->i_ae, "-lc");
1270 if (!seen_o && (ctx->i_flags & CW_F_SHADOW)) {
1271 newae(ctx->i_ae, "-o");
1272 newae(ctx->i_ae, ctx->i_discard);
1273 }
1274 }
1275
1276 static void
1277 do_smatch(cw_ictx_t *ctx)
1278 {
1279 if (ctx->i_flags & CW_F_PROG) {
1280 newae(ctx->i_ae, "--version");
1281 return;
1282 }
1283
1284 /*
1285 * Some sources shouldn't run smatch at all.
1286 */
1287 for (int i = 0; i < ctx->i_oldargc; i++) {
1288 char *arg = ctx->i_oldargv[i];
1289
1290 if (strcmp(arg, "-_smatch=off") == 0) {
1291 ctx->i_flags &= ~ (CW_F_EXEC | CW_F_ECHO);
1292 return;
1293 }
1294 }
1295
1296 /*
1297 * smatch can handle gcc's options.
1298 */
1299 do_gcc(ctx);
1300 }
1301
1302 static void
1303 do_cc(cw_ictx_t *ctx)
1304 {
1305 int in_output = 0, seen_o = 0;
1306 cw_op_t op = CW_O_LINK;
1307 char *nameflag;
1308
1309 if (ctx->i_flags & CW_F_PROG) {
1310 newae(ctx->i_ae, "-V");
1311 return;
1312 }
1313
1314 if (asprintf(&nameflag, "-_%s=", ctx->i_compiler->c_name) == -1)
1315 nomem();
1316
1317 while (--ctx->i_oldargc > 0) {
1318 char *arg = *++ctx->i_oldargv;
1319
1320 if (strncmp(arg, "-_CC=", 5) == 0) {
1321 newae(ctx->i_ae, strchr(arg, '=') + 1);
1322 continue;
1323 }
1324
1325 if (*arg != '-') {
1326 if (in_output == 0 || !(ctx->i_flags & CW_F_SHADOW)) {
1327 newae(ctx->i_ae, arg);
1328 } else {
1329 in_output = 0;
1330 newae(ctx->i_ae, ctx->i_discard);
1331 }
1332 continue;
1333 }
1334 switch (*(arg + 1)) {
1335 case '_':
1336 if ((strncmp(arg, nameflag, strlen(nameflag)) == 0) ||
1337 (strncmp(arg, "-_cc=", 5) == 0) ||
1338 (strncmp(arg, "-_sun=", 6) == 0)) {
1339 newae(ctx->i_ae, strchr(arg, '=') + 1);
1340 }
1341 break;
1342
1343 case 'V':
1344 ctx->i_flags &= ~CW_F_ECHO;
1345 newae(ctx->i_ae, arg);
1346 break;
1347 case 'o':
1348 seen_o = 1;
1349 if (strlen(arg) == 2) {
1350 in_output = 1;
1351 newae(ctx->i_ae, arg);
1352 } else if (ctx->i_flags & CW_F_SHADOW) {
1353 newae(ctx->i_ae, "-o");
1354 newae(ctx->i_ae, ctx->i_discard);
1355 } else {
1356 newae(ctx->i_ae, arg);
1357 }
1358 break;
1359 case 'c':
1360 case 'S':
1361 if (strlen(arg) == 2)
1362 op = CW_O_COMPILE;
1363 newae(ctx->i_ae, arg);
1364 break;
1365 case 'E':
1366 case 'P':
1367 if (strlen(arg) == 2)
1368 op = CW_O_PREPROCESS;
1369 /*FALLTHROUGH*/
1370 default:
1371 newae(ctx->i_ae, arg);
1372 }
1373 }
1374
1375 free(nameflag);
1376
1377 if ((op == CW_O_LINK || op == CW_O_PREPROCESS) &&
1378 (ctx->i_flags & CW_F_SHADOW))
1379 exit(0);
1380
1381 if (!seen_o && (ctx->i_flags & CW_F_SHADOW)) {
1382 newae(ctx->i_ae, "-o");
1383 newae(ctx->i_ae, ctx->i_discard);
1384 }
1385 }
1386
1387 static void
1388 prepctx(cw_ictx_t *ctx)
1389 {
1390 newae(ctx->i_ae, ctx->i_compiler->c_path);
1391
1392 if (ctx->i_flags & CW_F_PROG) {
1393 (void) printf("%s: %s\n", (ctx->i_flags & CW_F_SHADOW) ?
1394 "shadow" : "primary", ctx->i_compiler->c_path);
1395 (void) fflush(stdout);
1396 }
1397
1398 if (!(ctx->i_flags & CW_F_XLATE))
1399 return;
1400
1401 switch (ctx->i_compiler->c_style) {
1402 case SUN:
1403 do_cc(ctx);
1404 break;
1405 case GNU:
1406 do_gcc(ctx);
1407 break;
1408 case SMATCH:
1409 do_smatch(ctx);
1410 break;
1411 }
1412 }
1413
1414 static int
1415 invoke(cw_ictx_t *ctx)
1416 {
1417 char **newargv;
1418 int ac;
1419 struct ae *a;
1420
1421 if ((newargv = calloc(sizeof (*newargv), ctx->i_ae->ael_argc + 1)) ==
1422 NULL)
1423 nomem();
1424
1425 if (ctx->i_flags & CW_F_ECHO)
1426 (void) fprintf(stderr, "+ ");
1427
1428 for (ac = 0, a = ctx->i_ae->ael_head; a; a = a->ae_next, ac++) {
1429 newargv[ac] = a->ae_arg;
1430 if (ctx->i_flags & CW_F_ECHO)
1431 (void) fprintf(stderr, "%s ", a->ae_arg);
1432 if (a == ctx->i_ae->ael_tail)
1433 break;
1434 }
1435
1436 if (ctx->i_flags & CW_F_ECHO) {
1437 (void) fprintf(stderr, "\n");
1438 (void) fflush(stderr);
1439 }
1440
1441 if (!(ctx->i_flags & CW_F_EXEC))
1442 return (0);
1443
1444 /*
1445 * We must fix up the environment here so that the dependency files are
1446 * not trampled by the shadow compiler. Also take care of GCC
1447 * environment variables that will throw off gcc. This assumes a primary
1448 * gcc.
1449 */
1450 if ((ctx->i_flags & CW_F_SHADOW) &&
1451 (unsetenv("SUNPRO_DEPENDENCIES") != 0 ||
1452 unsetenv("DEPENDENCIES_OUTPUT") != 0 ||
1453 unsetenv("GCC_ROOT") != 0)) {
1454 (void) fprintf(stderr, "error: environment setup failed: %s\n",
1455 strerror(errno));
1456 return (-1);
1457 }
1458
1459 (void) execv(newargv[0], newargv);
1460 warn("couldn't run %s", newargv[0]);
1461
1462 return (-1);
1463 }
1464
1465 static int
1466 reap(cw_ictx_t *ctx)
1467 {
1468 int status, ret = 0;
1469 char buf[1024];
1470 struct stat s;
1471
1472 /*
1473 * Only wait for one specific child.
1474 */
1475 if (ctx->i_pid <= 0)
1476 return (-1);
1477
1478 do {
1479 if (waitpid(ctx->i_pid, &status, 0) < 0) {
1480 warn("cannot reap child");
1481 return (-1);
1482 }
1483 if (status != 0) {
1484 if (WIFSIGNALED(status)) {
1485 ret = -WTERMSIG(status);
1486 break;
1487 } else if (WIFEXITED(status)) {
1488 ret = WEXITSTATUS(status);
1489 break;
1490 }
1491 }
1492 } while (!WIFEXITED(status) && !WIFSIGNALED(status));
1493
1494 (void) unlink(ctx->i_discard);
1495
1496 if (stat(ctx->i_stderr, &s) < 0) {
1497 warn("stat failed on child cleanup");
1498 return (-1);
1499 }
1500 if (s.st_size != 0) {
1501 FILE *f;
1502
1503 if ((f = fopen(ctx->i_stderr, "r")) != NULL) {
1504 while (fgets(buf, sizeof (buf), f))
1505 (void) fprintf(stderr, "%s", buf);
1506 (void) fflush(stderr);
1507 (void) fclose(f);
1508 }
1509 }
1510 (void) unlink(ctx->i_stderr);
1511 free(ctx->i_stderr);
1512
1513 /*
1514 * cc returns an error code when given -V; we want that to succeed.
1515 */
1516 if (ctx->i_flags & CW_F_PROG)
1517 return (0);
1518
1519 return (ret);
1520 }
1521
1522 static int
1523 exec_ctx(cw_ictx_t *ctx, int block)
1524 {
1525 char *file;
1526
1527 /*
1528 * To avoid offending cc's sensibilities, the name of its output
1529 * file must end in '.o'.
1530 */
1531 if ((file = tempnam(NULL, ".cw")) == NULL) {
1532 nomem();
1533 return (-1);
1534 }
1535 (void) strlcpy(ctx->i_discard, file, MAXPATHLEN);
1536 (void) strlcat(ctx->i_discard, ".o", MAXPATHLEN);
1537 free(file);
1538
1539 if ((ctx->i_stderr = tempnam(NULL, ".cw")) == NULL) {
1540 nomem();
1541 return (-1);
1542 }
1543
1544 if ((ctx->i_pid = fork()) == 0) {
1545 int fd;
1546
1547 (void) fclose(stderr);
1548 if ((fd = open(ctx->i_stderr, O_WRONLY | O_CREAT | O_EXCL,
1549 0666)) < 0) {
1550 err(1, "open failed for standard error");
1551 }
1552 if (dup2(fd, 2) < 0) {
1553 err(1, "dup2 failed for standard error");
1554 }
1555 if (fd != 2)
1556 (void) close(fd);
1557 if (freopen("/dev/fd/2", "w", stderr) == NULL) {
1558 err(1, "freopen failed for /dev/fd/2");
1559 }
1560
1561 prepctx(ctx);
1562 exit(invoke(ctx));
1563 }
1564
1565 if (ctx->i_pid < 0) {
1566 err(1, "fork failed");
1567 }
1568
1569 if (block)
1570 return (reap(ctx));
1571
1572 return (0);
1573 }
1574
1575 static void
1576 parse_compiler(const char *spec, cw_compiler_t *compiler)
1577 {
1578 char *tspec, *token;
1579
1580 if ((tspec = strdup(spec)) == NULL)
1581 nomem();
1582
1583 if ((token = strsep(&tspec, ",")) == NULL)
1584 errx(1, "Compiler is missing a name: %s", spec);
1585 compiler->c_name = token;
1586
1587 if ((token = strsep(&tspec, ",")) == NULL)
1588 errx(1, "Compiler is missing a path: %s", spec);
1589 compiler->c_path = token;
1590
1591 if ((token = strsep(&tspec, ",")) == NULL)
1592 errx(1, "Compiler is missing a style: %s", spec);
1593
1594 if ((strcasecmp(token, "gnu") == 0) ||
1595 (strcasecmp(token, "gcc") == 0)) {
1596 compiler->c_style = GNU;
1597 } else if ((strcasecmp(token, "sun") == 0) ||
1598 (strcasecmp(token, "cc") == 0)) {
1599 compiler->c_style = SUN;
1600 } else if ((strcasecmp(token, "smatch") == 0)) {
1601 compiler->c_style = SMATCH;
1602 } else {
1603 errx(1, "unknown compiler style: %s", token);
1604 }
1605
1606 if (tspec != NULL)
1607 errx(1, "Excess tokens in compiler: %s", spec);
1608 }
1609
1610 int
1611 main(int argc, char **argv)
1612 {
1613 int ch;
1614 cw_compiler_t primary = { NULL, NULL, 0 };
1615 cw_compiler_t shadows[10];
1616 int nshadows = 0;
1617 int ret = 0;
1618 boolean_t do_serial = B_FALSE;
1619 boolean_t do_exec = B_FALSE;
1620 boolean_t vflg = B_FALSE;
1621 boolean_t Cflg = B_FALSE;
1622 boolean_t cflg = B_FALSE;
1623 boolean_t nflg = B_FALSE;
1624
1625 cw_ictx_t *main_ctx;
1626
1627 static struct option longopts[] = {
1628 { "compiler", no_argument, NULL, 'c' },
1629 { "noecho", no_argument, NULL, 'n' },
1630 { "primary", required_argument, NULL, 'p' },
1631 { "shadow", required_argument, NULL, 's' },
1632 { "versions", no_argument, NULL, 'v' },
1633 { NULL, 0, NULL, 0 },
1634 };
1635
1636
1637 if ((main_ctx = newictx()) == NULL)
1638 nomem();
1639
1640 while ((ch = getopt_long(argc, argv, "C", longopts, NULL)) != -1) {
1641 switch (ch) {
1642 case 'c':
1643 cflg = B_TRUE;
1644 break;
1645 case 'C':
1646 Cflg = B_TRUE;
1647 break;
1648 case 'n':
1649 nflg = B_TRUE;
1650 break;
1651 case 'p':
1652 if (primary.c_path != NULL) {
1653 warnx("Only one primary compiler may "
1654 "be specified");
1655 usage();
1656 }
1657
1658 parse_compiler(optarg, &primary);
1659 break;
1660 case 's':
1661 if (nshadows >= 10)
1662 errx(1, "May only use 10 shadows at "
1663 "the moment");
1664 parse_compiler(optarg, &shadows[nshadows]);
1665 nshadows++;
1666 break;
1667 case 'v':
1668 vflg = B_TRUE;
1669 break;
1670 default:
1671 (void) fprintf(stderr, "Did you forget '--'?\n");
1672 usage();
1673 }
1674 }
1675
1676 if (primary.c_path == NULL) {
1677 warnx("A primary compiler must be specified");
1678 usage();
1679 }
1680
1681 do_serial = (getenv("CW_SHADOW_SERIAL") == NULL) ? B_FALSE : B_TRUE;
1682 do_exec = (getenv("CW_NO_EXEC") == NULL) ? B_TRUE : B_FALSE;
1683
1684 /* Leave room for argv[0] */
1685 argc -= (optind - 1);
1686 argv += (optind - 1);
1687
1688 main_ctx->i_oldargc = argc;
1689 main_ctx->i_oldargv = argv;
1690 main_ctx->i_flags = CW_F_XLATE;
1691 if (nflg == 0)
1692 main_ctx->i_flags |= CW_F_ECHO;
1693 if (do_exec)
1694 main_ctx->i_flags |= CW_F_EXEC;
1695 if (Cflg)
1696 main_ctx->i_flags |= CW_F_CXX;
1697 main_ctx->i_compiler = &primary;
1698
1699 if (cflg) {
1700 (void) fputs(primary.c_path, stdout);
1701 }
1702
1703 if (vflg) {
1704 (void) printf("cw version %s\n", CW_VERSION);
1705 (void) fflush(stdout);
1706 main_ctx->i_flags &= ~CW_F_ECHO;
1707 main_ctx->i_flags |= CW_F_PROG | CW_F_EXEC;
1708 do_serial = 1;
1709 }
1710
1711 ret |= exec_ctx(main_ctx, do_serial);
1712
1713 for (int i = 0; i < nshadows; i++) {
1714 int r;
1715 cw_ictx_t *shadow_ctx;
1716
1717 if ((shadow_ctx = newictx()) == NULL)
1718 nomem();
1719
1720 memcpy(shadow_ctx, main_ctx, sizeof (cw_ictx_t));
1721
1722 shadow_ctx->i_flags |= CW_F_SHADOW;
1723 shadow_ctx->i_compiler = &shadows[i];
1724
1725 r = exec_ctx(shadow_ctx, do_serial);
1726 if (r == 0) {
1727 shadow_ctx->i_next = main_ctx->i_next;
1728 main_ctx->i_next = shadow_ctx;
1729 }
1730 ret |= r;
1731 }
1732
1733 if (!do_serial) {
1734 cw_ictx_t *next = main_ctx;
1735 while (next != NULL) {
1736 cw_ictx_t *toreap = next;
1737 next = next->i_next;
1738 ret |= reap(toreap);
1739 }
1740 }
1741
1742 return (ret);
1743 }