2 * Copyright (c) 1993 Paul Kranenburg
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by Paul Kranenburg.
16 * 4. The name of the author may not be used to endorse or promote products
17 * derived from this software without specific prior written permission
19 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
23 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
24 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 * $FreeBSD: src/usr.bin/ldd/ldd.c,v 1.18.2.7 2002/02/27 18:35:53 sobomax Exp $
31 * $DragonFly: src/usr.bin/ldd/ldd.c,v 1.6 2006/01/12 13:43:11 corecode Exp $
34 #include <sys/param.h>
36 #include <machine/elf.h>
45 extern void dump_file(const char *);
46 extern int error_count
;
51 fprintf(stderr
, "usage: ldd [-v] [-f format] program ...\n");
56 main(int argc
, char **argv
)
58 char *fmt1
= NULL
, *fmt2
= NULL
;
63 while ((c
= getopt(argc
, argv
, "vf:")) != -1) {
71 errx(1, "too many formats");
85 errx(1, "-v may not be used with -f");
94 for (c
= 0; c
< argc
; c
++)
96 exit(error_count
== 0 ? EXIT_SUCCESS
: EXIT_FAILURE
);
101 if (setenv("LD_TRACE_LOADED_OBJECTS", "1", 1) == -1)
102 err(1, "setenv: cannot set LD_TRACE_LOADED_OBJECTS=1");
104 if (setenv("LD_TRACE_LOADED_OBJECTS_FMT1", fmt1
, 1) == -1)
105 err(1, "setenv: cannot set LD_TRACE_LOADED_OBJECTS_FMT1=%s", fmt1
);
108 if (setenv("LD_TRACE_LOADED_OBJECTS_FMT2", fmt2
, 1) == -1)
109 err(1, "setenv: cannot set LD_TRACE_LOADED_OBJECTS_FMT2=%s", fmt2
);
113 for ( ; argc
> 0; argc
--, argv
++) {
124 if ((fd
= open(*argv
, O_RDONLY
, 0)) < 0) {
129 if ((n
= read(fd
, &hdr
, sizeof hdr
)) == -1) {
130 warn("%s: can't read program header", *argv
);
138 if (n
>= sizeof hdr
.aout
&& !N_BADMAG(hdr
.aout
)) {
140 if ((N_GETFLAG(hdr
.aout
) & EX_DPMASK
) != EX_DYNAMIC
141 #if 1 /* Compatibility */
142 || hdr
.aout
.a_entry
< __LDPGSZ
145 warnx("%s: not a dynamic executable", *argv
);
148 } else if (n
>= sizeof hdr
.elf
&& IS_ELF(hdr
.elf
)) {
153 if (lseek(fd
, 0, SEEK_SET
) == -1 ||
154 read(fd
, &ehdr
, sizeof ehdr
) != sizeof ehdr
||
155 lseek(fd
, ehdr
.e_phoff
, SEEK_SET
) == -1
157 warnx("%s: can't read program header", *argv
);
160 for (i
= 0; i
< ehdr
.e_phnum
; i
++) {
161 if (read(fd
, &phdr
, ehdr
.e_phentsize
)
163 warnx("%s: can't read program header",
168 if (phdr
.p_type
== PT_DYNAMIC
)
173 warnx("%s: not a dynamic executable", *argv
);
175 } else if (hdr
.elf
.e_type
== ET_DYN
) {
179 warnx("%s: not a dynamic executable", *argv
);
188 if (setenv("LD_TRACE_LOADED_OBJECTS_PROGNAME", *argv
, 1) == -1)
189 err(1, "setenv: cannot set LD_TRACE_LOADED_OBJECTS_PROGNAME=%s", *argv
);
190 if (fmt1
== NULL
&& fmt2
== NULL
)
191 /* Default formats */
192 printf("%s:\n", *argv
);
201 if (wait(&status
) <= 0) {
204 } else if (WIFSIGNALED(status
)) {
205 fprintf(stderr
, "%s: signal %d\n",
206 *argv
, WTERMSIG(status
));
208 } else if (WIFEXITED(status
) && WEXITSTATUS(status
)) {
209 fprintf(stderr
, "%s: exit status %d\n",
210 *argv
, WEXITSTATUS(status
));
216 execl(*argv
, *argv
, (char *)NULL
);
219 dlopen(*argv
, RTLD_TRACE
);
220 warnx("%s: %s", *argv
, dlerror());