2 * Copyright (c) 2004, The DragonFly Project. All rights reserved.
3 * Copyright (c) 1998, Peter Wemm <peter@netplex.com.au>
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
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
27 * $FreeBSD: src/usr.bin/objformat/objformat.c,v 1.6 1998/10/24 02:01:30 jdp Exp $
30 #include <sys/param.h>
33 #include <objformat.h>
40 #define CCVER_DEFAULT "gcc50"
43 #ifndef BINUTILSVER_DEFAULT
44 #define BINUTILSVER_DEFAULT "binutils225"
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 ""
56 /* Macro for array size */
58 #define NELEM(ary) (sizeof(ary) / sizeof((ary)[0]))
61 enum cmd_type
{ OBJFORMAT
, COMPILER
, BINUTILS
, LINKER
};
68 static struct command commands
[] = {
77 {"addr2line", BINUTILS
},
80 {"c++filt", BINUTILS
},
81 {"elfedit", BINUTILS
},
84 {"objcopy", BINUTILS
},
85 {"objdump", BINUTILS
},
87 {"readelf", BINUTILS
},
89 {"strings", BINUTILS
},
91 {"incremental-dump", BINUTILS
},
92 {"objformat", OBJFORMAT
},
97 main(int argc
, char **argv
)
99 char ld_def
[] = LINKER_DEFAULT
;
100 char ld_alt
[] = LINKER_ALT
;
101 struct command
*cmds
;
104 char *cmd
, *newcmd
= NULL
;
105 char *ldcmd
= ld_def
;
106 const char *objformat_path
;
110 const char *env_value
= NULL
;
111 const char *base_path
= NULL
;
112 int use_objformat
= 0;
114 if (getobjformat(objformat
, sizeof objformat
, &argc
, argv
) == -1)
115 errx(1, "Invalid object format");
118 * Get the last path element of the program name being executed
120 cmd
= strrchr(argv
[0], '/');
126 for (cmds
= commands
; cmds
< &commands
[NELEM(commands
) - 1]; ++cmds
) {
127 if (strcmp(cmd
, cmds
->cmd
) == 0)
132 switch (cmds
->type
) {
134 ccver
= getenv("CCVER");
135 if ((ccver
== NULL
) || ccver
[0] == 0)
136 ccver
= CCVER_DEFAULT
;
137 base_path
= "/usr/libexec";
142 buver
= getenv("BINUTILSVER");
144 buver
= BINUTILSVER_DEFAULT
;
145 base_path
= "/usr/libexec";
150 buver
= getenv("BINUTILSVER");
152 buver
= BINUTILSVER_DEFAULT
;
153 ldver
= getenv("LDVER");
154 if ((ldver
!= NULL
) && (strcmp(ldver
, ld_alt
) == 0))
156 base_path
= "/usr/libexec";
164 errx(1, "unknown command type");
170 * The objformat command itself doesn't need another exec
172 if (cmds
->type
== OBJFORMAT
) {
174 fprintf(stderr
, "Usage: objformat\n");
178 printf("%s\n", objformat
);
183 * make buildworld glue and CCVER overrides.
186 * CCVER=clang check temporary; only for base clang import work
188 if (cmds
&& cmds
->type
== COMPILER
&&
189 strcmp (env_value
, "clang") == 0) {
192 objformat_path
= getenv("OBJFORMAT_PATH");
193 if (objformat_path
== NULL
)
194 objformat_path
= OBJFORMAT_PATH_DEFAULT
;
198 path
= strdup(objformat_path
);
200 if (setenv("OBJFORMAT", objformat
, 1) == -1)
201 err(1, "setenv: cannot set OBJFORMAT=%s", objformat
);
204 * objformat_path could be sequence of colon-separated paths.
206 while ((chunk
= strsep(&path
, ":")) != NULL
) {
207 if (newcmd
!= NULL
) {
212 asprintf(&newcmd
, "%s%s/%s/%s/%s",
213 chunk
, base_path
, env_value
, objformat
, cmd
);
215 asprintf(&newcmd
, "%s%s/%s/%s",
216 chunk
, base_path
, env_value
, cmd
);
219 err(1, "cannot allocate memory");
226 * Fallback: if we're searching for a compiler, but didn't
227 * find any, try again using the custom compiler driver.
229 if (cmds
&& cmds
->type
== COMPILER
&&
230 strcmp(env_value
, "custom") != 0) {
231 env_value
= "custom";
236 err(1, "in path [%s]%s/%s/%s/%s",
237 objformat_path
, base_path
, env_value
, objformat
, cmd
);
239 err(1, "in path [%s]%s/%s/%s",
240 objformat_path
, base_path
, env_value
, cmd
);