kernel - Order ipfw3 module before other ipfw3_* modules
[dragonfly.git] / usr.bin / objformat / objformat.c
blobb658bca0af5f5717161e3b46f8da20ec0d96df22
1 /*-
2 * Copyright (c) 2004, The DragonFly Project. All rights reserved.
3 * Copyright (c) 1998, Peter Wemm <peter@netplex.com.au>
4 * All rights reserved.
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25 * SUCH DAMAGE.
27 * $FreeBSD: src/usr.bin/objformat/objformat.c,v 1.6 1998/10/24 02:01:30 jdp Exp $
30 #include <sys/param.h>
32 #include <err.h>
33 #include <objformat.h>
34 #include <stdio.h>
35 #include <stdlib.h>
36 #include <string.h>
37 #include <unistd.h>
39 #ifndef CCVER_DEFAULT
40 #define CCVER_DEFAULT "gcc50"
41 #endif
43 #ifndef BINUTILSVER_DEFAULT
44 #define BINUTILSVER_DEFAULT "binutils227"
45 #endif
47 #define LINKER_BFD "ld.bfd"
48 #define LINKER_GOLD "ld.gold"
49 #define LINKER_DEFAULT LINKER_GOLD
50 #define LINKER_ALT LINKER_BFD
52 #ifndef OBJFORMAT_PATH_DEFAULT
53 #define OBJFORMAT_PATH_DEFAULT ""
54 #endif
56 /* Macro for array size */
57 #ifndef NELEM
58 #define NELEM(ary) (sizeof(ary) / sizeof((ary)[0]))
59 #endif
61 enum cmd_type { OBJFORMAT, COMPILER, BINUTILS, LINKER };
63 struct command {
64 const char *cmd;
65 enum cmd_type type;
68 static struct command commands[] = {
69 {"CC", COMPILER},
70 {"c++", COMPILER},
71 {"cc", COMPILER},
72 {"cpp", COMPILER},
73 {"g++", COMPILER},
74 {"gcc", COMPILER},
75 {"gcov", COMPILER},
76 {"ld", LINKER},
77 {"addr2line", BINUTILS},
78 {"ar", BINUTILS},
79 {"as", BINUTILS},
80 {"c++filt", BINUTILS},
81 {"elfedit", BINUTILS},
82 {"gprof", BINUTILS},
83 {"ld.bfd", BINUTILS},
84 {"ld.gold", BINUTILS},
85 {"nm", BINUTILS},
86 {"objcopy", BINUTILS},
87 {"objdump", BINUTILS},
88 {"ranlib", BINUTILS},
89 {"readelf", BINUTILS},
90 {"size", BINUTILS},
91 {"strings", BINUTILS},
92 {"strip", BINUTILS},
93 {"incremental-dump", BINUTILS},
94 {"objformat", OBJFORMAT},
95 {"", -1}
98 int
99 main(int argc, char **argv)
101 char ld_def[] = LINKER_DEFAULT;
102 char ld_alt[] = LINKER_ALT;
103 struct command *cmds;
104 char objformat[32];
105 char *path, *chunk;
106 char *cmd, *newcmd = NULL;
107 char *ldcmd = ld_def;
108 const char *objformat_path;
109 const char *ccver;
110 const char *buver;
111 const char *ldver;
112 const char *env_value = NULL;
113 const char *base_path = NULL;
114 int use_objformat = 0;
116 if (getobjformat(objformat, sizeof objformat, &argc, argv) == -1)
117 errx(1, "Invalid object format");
120 * Get the last path element of the program name being executed
122 cmd = strrchr(argv[0], '/');
123 if (cmd != NULL)
124 cmd++;
125 else
126 cmd = argv[0];
128 for (cmds = commands; cmds < &commands[NELEM(commands) - 1]; ++cmds) {
129 if (strcmp(cmd, cmds->cmd) == 0)
130 break;
133 if (cmds) {
134 switch (cmds->type) {
135 case COMPILER:
136 ccver = getenv("CCVER");
137 if ((ccver == NULL) || ccver[0] == 0)
138 ccver = CCVER_DEFAULT;
139 base_path = "/usr/libexec";
140 use_objformat = 0;
141 env_value = ccver;
142 break;
143 case BINUTILS:
144 buver = getenv("BINUTILSVER");
145 if (buver == NULL)
146 buver = BINUTILSVER_DEFAULT;
147 base_path = "/usr/libexec";
148 use_objformat = 1;
149 env_value = buver;
150 break;
151 case LINKER:
152 buver = getenv("BINUTILSVER");
153 if (buver == NULL)
154 buver = BINUTILSVER_DEFAULT;
155 ldver = getenv("LDVER");
156 if ((ldver != NULL) && (strcmp(ldver, ld_alt) == 0))
157 ldcmd = ld_alt;
158 base_path = "/usr/libexec";
159 use_objformat = 1;
160 env_value = buver;
161 cmd = ldcmd;
162 break;
163 case OBJFORMAT:
164 break;
165 default:
166 errx(1, "unknown command type");
167 break;
172 * The objformat command itself doesn't need another exec
174 if (cmds->type == OBJFORMAT) {
175 if (argc != 1) {
176 fprintf(stderr, "Usage: objformat\n");
177 exit(1);
180 printf("%s\n", objformat);
181 exit(0);
185 * make buildworld glue and CCVER overrides.
188 * CCVER=clang check temporary; only for base clang import work
190 if (cmds && cmds->type == COMPILER &&
191 strcmp (env_value, "clang") == 0) {
192 objformat_path = "";
193 } else {
194 objformat_path = getenv("OBJFORMAT_PATH");
195 if (objformat_path == NULL)
196 objformat_path = OBJFORMAT_PATH_DEFAULT;
199 again:
200 path = strdup(objformat_path);
202 if (setenv("OBJFORMAT", objformat, 1) == -1)
203 err(1, "setenv: cannot set OBJFORMAT=%s", objformat);
206 * objformat_path could be sequence of colon-separated paths.
208 while ((chunk = strsep(&path, ":")) != NULL) {
209 if (newcmd != NULL) {
210 free(newcmd);
211 newcmd = NULL;
213 if (use_objformat) {
214 asprintf(&newcmd, "%s%s/%s/%s/%s",
215 chunk, base_path, env_value, objformat, cmd);
216 } else {
217 asprintf(&newcmd, "%s%s/%s/%s",
218 chunk, base_path, env_value, cmd);
220 if (newcmd == NULL)
221 err(1, "cannot allocate memory");
223 argv[0] = newcmd;
224 execv(newcmd, argv);
228 * Fallback: if we're searching for a compiler, but didn't
229 * find any, try again using the custom compiler driver.
231 if (cmds && cmds->type == COMPILER &&
232 strcmp(env_value, "custom") != 0) {
233 env_value = "custom";
234 goto again;
237 if (use_objformat) {
238 err(1, "in path [%s]%s/%s/%s/%s",
239 objformat_path, base_path, env_value, objformat, cmd);
240 } else {
241 err(1, "in path [%s]%s/%s/%s",
242 objformat_path, base_path, env_value, cmd);