xref: /titanic_52/usr/src/cmd/sgs/elfdump/common/main.c (revision bde3d612a7c090234c60e6e4578821237a5db135)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved.
24  */
25 
26 /*
27  * Dump an elf file.
28  */
29 #include	<sys/param.h>
30 #include	<fcntl.h>
31 #include	<stdio.h>
32 #include	<stdlib.h>
33 #include	<ctype.h>
34 #include	<_libelf.h>
35 #include	<link.h>
36 #include	<stdarg.h>
37 #include	<unistd.h>
38 #include	<libgen.h>
39 #include	<libintl.h>
40 #include	<locale.h>
41 #include	<errno.h>
42 #include	<strings.h>
43 #include	<debug.h>
44 #include	<conv.h>
45 #include	<msg.h>
46 #include	<_elfdump.h>
47 #include	<sys/elf_SPARC.h>
48 #include	<sys/elf_amd64.h>
49 
50 
51 const Cache	cache_init = {NULL, NULL, NULL, NULL, 0};
52 
53 
54 
55 /*
56  * The -I, -N, and -T options are called "match options", because
57  * they allow selecting the items to be displayed based on matching
58  * their index, name, or type.
59  *
60  * The ELF information to which -I, -N, or -T are applied in
61  * the current invocation is called the "match item".
62  */
63 typedef enum {
64 	MATCH_ITEM_PT,		/* Program header (PT_) */
65 	MATCH_ITEM_SHT		/* Section header (SHT_) */
66 } match_item_t;
67 
68 /* match_opt_t is  used to note which match option was used */
69 typedef enum {
70 	MATCH_OPT_NAME,		/* Record contains a name */
71 	MATCH_OPT_NDX,		/* Record contains a single index */
72 	MATCH_OPT_RANGE,	/* Record contains an index range */
73 	MATCH_OPT_TYPE,		/* Record contains a type (shdr or phdr) */
74 } match_opt_t;
75 
76 typedef struct _match {
77 	struct _match	*next;		/* Pointer to next item in list */
78 	match_opt_t	opt_type;
79 	union {
80 		const char	*name;	/* MATCH_OPT_NAME */
81 		struct {		/* MATCH_OPT_NDX and MATCH_OPT_RANGE */
82 			int	start;
83 			int	end;	/* Only for MATCH_OPT_RANGE */
84 		} ndx;
85 		uint32_t	type;	/* MATCH_OPT_TYPE */
86 	} value;
87 } match_rec_t;
88 
89 static struct {
90 	match_item_t	item_type;	/* Type of item being matched */
91 	match_rec_t	*list;		/* Records for (-I, -N, -T) options */
92 } match_state;
93 
94 
95 
96 const char *
97 _elfdump_msg(Msg mid)
98 {
99 	return (gettext(MSG_ORIG(mid)));
100 }
101 
102 /*
103  * Determine whether a symbol name should be demangled.
104  */
105 const char *
106 demangle(const char *name, uint_t flags)
107 {
108 	if (flags & FLG_CTL_DEMANGLE)
109 		return (Elf_demangle_name(name));
110 	else
111 		return ((char *)name);
112 }
113 
114 /*
115  * Define our own standard error routine.
116  */
117 void
118 failure(const char *file, const char *func)
119 {
120 	(void) fprintf(stderr, MSG_INTL(MSG_ERR_FAILURE),
121 	    file, func, elf_errmsg(elf_errno()));
122 }
123 
124 /*
125  * The full usage message
126  */
127 static void
128 detail_usage()
129 {
130 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL1));
131 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL2));
132 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL3));
133 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL4));
134 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL5));
135 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL6));
136 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL7));
137 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL8));
138 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL9));
139 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL10));
140 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL11));
141 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL12));
142 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL13));
143 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL14));
144 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL15));
145 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL16));
146 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL17));
147 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL18));
148 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL19));
149 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL20));
150 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL21));
151 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL22));
152 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL23));
153 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL24));
154 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL25));
155 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL26));
156 }
157 
158 /*
159  * Output a block of raw data as hex bytes. Each row is given
160  * the index of the first byte in the row.
161  *
162  * entry:
163  *	data - Pointer to first byte of data to be displayed
164  *	n - # of bytes of data
165  *	prefix - String to be output before each line. Useful
166  *		for indenting output.
167  *	bytes_per_col - # of space separated bytes to output
168  *		in each column.
169  *	col_per_row - # of columns to output per row
170  *
171  * exit:
172  *	The formatted data has been sent to stdout. Each row of output
173  *	shows (bytes_per_col * col_per_row) bytes of data.
174  */
175 void
176 dump_hex_bytes(const void *data, size_t n, int indent,
177 	int bytes_per_col, int col_per_row)
178 {
179 	const uchar_t *ldata = data;
180 	int	bytes_per_row = bytes_per_col * col_per_row;
181 	int	ndx, byte, word;
182 	char	string[128], *str = string;
183 	char	index[MAXNDXSIZE];
184 	int	index_width;
185 	int	sp_prefix = 0;
186 
187 
188 	/*
189 	 * Determine the width to use for the index string. We follow
190 	 * 8-byte tab rules, but don't use an actual \t character so
191 	 * that the output can be arbitrarily shifted without odd
192 	 * tab effects, and so that all the columns line up no matter
193 	 * how many lines of output are produced.
194 	 */
195 	ndx = n / bytes_per_row;
196 	(void) snprintf(index, sizeof (index),
197 	    MSG_ORIG(MSG_FMT_INDEX2), EC_WORD(ndx));
198 	index_width = strlen(index);
199 	index_width = S_ROUND(index_width, 8);
200 
201 	for (ndx = byte = word = 0; n > 0; n--, ldata++) {
202 		while (sp_prefix-- > 0)
203 			*str++ = ' ';
204 
205 		(void) snprintf(str, sizeof (string),
206 		    MSG_ORIG(MSG_HEXDUMP_TOK), (int)*ldata);
207 		str += 2;
208 		sp_prefix = 1;
209 
210 		if (++byte == bytes_per_col) {
211 			sp_prefix += 2;
212 			word++;
213 			byte = 0;
214 		}
215 		if (word == col_per_row) {
216 			*str = '\0';
217 			(void) snprintf(index, sizeof (index),
218 			    MSG_ORIG(MSG_FMT_INDEX2), EC_WORD(ndx));
219 			dbg_print(0, MSG_ORIG(MSG_HEXDUMP_ROW),
220 			    indent, MSG_ORIG(MSG_STR_EMPTY),
221 			    index_width, index, string);
222 			sp_prefix = 0;
223 			word = 0;
224 			ndx += bytes_per_row;
225 			str = string;
226 		}
227 	}
228 	if (byte || word) {
229 		*str = '\0';	/*  */
230 		(void) snprintf(index, sizeof (index),
231 		    MSG_ORIG(MSG_FMT_INDEX2), EC_WORD(ndx));
232 		dbg_print(0, MSG_ORIG(MSG_HEXDUMP_ROW), indent,
233 		    MSG_ORIG(MSG_STR_EMPTY), index_width, index, string);
234 	}
235 }
236 
237 /*
238  * Convert the ASCII representation of an index, or index range, into
239  * binary form, and store it in rec:
240  *
241  *	index: An positive or 0 valued integer
242  *	range: Two indexes, separated by a ':' character, denoting
243  *		a range of allowed values. If the second value is omitted,
244  *		any values equal to or greater than the first will match.
245  *
246  * exit:
247  *	On success, *rec is filled in with a MATCH_OPT_NDX or MATCH_OPT_RANGE
248  *	value, and this function returns (1). On failure, the contents
249  *	of *rec are undefined, and (0) is returned.
250  */
251 int
252 process_index_opt(const char *str, match_rec_t *rec)
253 {
254 #define	SKIP_BLANK for (; *str && isspace(*str); str++)
255 
256 	char	*endptr;
257 
258 	rec->value.ndx.start = strtol(str, &endptr, 10);
259 	/* Value must use some of the input, and be 0 or positive */
260 	if ((str == endptr) || (rec->value.ndx.start < 0))
261 		return (0);
262 	str = endptr;
263 
264 	SKIP_BLANK;
265 	if (*str != ':') {
266 		rec->opt_type = MATCH_OPT_NDX;
267 	} else {
268 		str++;					/* Skip the ':' */
269 		rec->opt_type = MATCH_OPT_RANGE;
270 		SKIP_BLANK;
271 		if (*str == '\0') {
272 			rec->value.ndx.end = -1;	/* Indicates "to end" */
273 		} else {
274 			rec->value.ndx.end = strtol(str, &endptr, 10);
275 			if ((str == endptr) || (rec->value.ndx.end < 0))
276 				return (0);
277 			str = endptr;
278 			SKIP_BLANK;
279 		}
280 	}
281 
282 	/* Syntax error if anything is left over */
283 	if (*str != '\0')
284 		return (0);
285 
286 	return (1);
287 
288 #undef	SKIP_BLANK
289 }
290 
291 /*
292  * Convert a string containing a specific type of ELF constant, or an ASCII
293  * representation of a number, to an integer. Strings starting with '0'
294  * are taken to be octal, those staring with '0x' are hex, and all
295  * others are decimal.
296  *
297  * entry:
298  *	str - String to be converted
299  *	ctype - Constant type
300  *	v - Address of variable to receive resulting value.
301  *
302  * exit:
303  *	On success, returns True (1) and *v is set to the value.
304  *	On failure, returns False (0) and *v is undefined.
305  */
306 typedef enum {
307 	ATOUI_PT,
308 	ATOUI_SHT,
309 	ATOUI_OSABI
310 } atoui_type_t;
311 
312 static int
313 atoui(const char *str, atoui_type_t type, uint32_t *v)
314 {
315 	conv_strtol_uvalue_t	uvalue;
316 	char			*endptr;
317 
318 	if (conv_iter_strtol_init(str, &uvalue) != 0) {
319 		switch (type) {
320 		case ATOUI_PT:
321 			if (conv_iter_phdr_type(CONV_OSABI_ALL, CONV_FMT_ALT_CF,
322 			    conv_iter_strtol, &uvalue) == CONV_ITER_DONE)
323 				break;
324 			(void) conv_iter_phdr_type(CONV_OSABI_ALL,
325 			    CONV_FMT_ALT_NF, conv_iter_strtol, &uvalue);
326 			break;
327 		case ATOUI_SHT:
328 			if (conv_iter_sec_type(CONV_OSABI_ALL, CONV_MACH_ALL,
329 			    CONV_FMT_ALT_CF, conv_iter_strtol, &uvalue) ==
330 			    CONV_ITER_DONE)
331 				break;
332 			(void) conv_iter_sec_type(CONV_OSABI_ALL, CONV_MACH_ALL,
333 			    CONV_FMT_ALT_NF, conv_iter_strtol, &uvalue);
334 			break;
335 		case ATOUI_OSABI:
336 			if (conv_iter_ehdr_osabi(CONV_FMT_ALT_CF,
337 			    conv_iter_strtol, &uvalue) == CONV_ITER_DONE)
338 				break;
339 			(void) conv_iter_ehdr_osabi(CONV_FMT_ALT_NF,
340 			    conv_iter_strtol, &uvalue);
341 			break;
342 		}
343 		if (uvalue.csl_found) {
344 			*v = uvalue.csl_value;
345 			return (1);
346 		}
347 	}
348 
349 	*v = strtoull(str, &endptr, 0);
350 
351 	/* If the left over part contains anything but whitespace, fail */
352 	for (; *endptr; endptr++)
353 		if (!isspace(*endptr))
354 			return (0);
355 	return (1);
356 }
357 
358 /*
359  * Called after getopt() processing is finished if there is a non-empty
360  * match list. Prepares the matching code for use.
361  *
362  * exit:
363  *	Returns True (1) if no errors are encountered. Writes an
364  *	error string to stderr and returns False (0) otherwise.
365  */
366 static int
367 match_prepare(char *argv0, uint_t flags)
368 {
369 	match_rec_t	*list;
370 	const char	*str;
371 	int		minus_p = (flags & FLG_SHOW_PHDR) != 0;
372 	atoui_type_t	atoui_type;
373 
374 	/*
375 	 * Flag ambiguous attempt to use match option with both -p and
376 	 * and one or more section SHOW options. In this case, we
377 	 * can't tell what type of item we're supposed to match against.
378 	 */
379 	if (minus_p && (flags & FLG_MASK_SHOW_SHDR)) {
380 		(void) fprintf(stderr, MSG_INTL(MSG_ERR_AMBIG_MATCH),
381 		    basename(argv0));
382 		return (0);
383 	}
384 
385 	/* Set the match type, based on the presence of the -p option */
386 	if (minus_p) {
387 		match_state.item_type = MATCH_ITEM_PT;
388 		atoui_type = ATOUI_PT;
389 	} else {
390 		match_state.item_type = MATCH_ITEM_SHT;
391 		atoui_type = ATOUI_SHT;
392 	}
393 
394 	/*
395 	 * Scan match list and perform any necessary fixups:
396 	 *
397 	 * MATCH_OPT_NAME: If -p is specified, convert MATCH_OPT_NAME (-N)
398 	 *	requests into MATCH_OPT_TYPE (-T).
399 	 *
400 	 * MATCH_OPT_TYPE: Now that we know item type we are matching
401 	 *	against, we can convert the string saved in the name
402 	 *	field during getopt() processing into an integer and
403 	 *	write it into the type field.
404 	 */
405 	for (list = match_state.list; list; list = list->next) {
406 		if ((list->opt_type == MATCH_OPT_NAME) && minus_p)
407 			list->opt_type = MATCH_OPT_TYPE;
408 
409 		if (list->opt_type != MATCH_OPT_TYPE)
410 			continue;
411 
412 		str = list->value.name;
413 		if (atoui(str, atoui_type, &list->value.type) == 0) {
414 			const char *fmt = minus_p ?
415 			    MSG_INTL(MSG_ERR_BAD_T_PT) :
416 			    MSG_INTL(MSG_ERR_BAD_T_SHT);
417 
418 			(void) fprintf(stderr, fmt, basename(argv0), str);
419 			return (0);
420 		}
421 	}
422 
423 	return (1);
424 }
425 
426 
427 /*
428  * Returns True (1) if the item with the given name or index should
429  * be displayed, and False (0) if it should not be.
430  *
431  * entry:
432  *	match_flags - Bitmask specifying matching options, as described
433  *		in _elfdump.h.
434  *	name - If MATCH_F_NAME flag is set, name of item under
435  *		consideration. Otherwise ignored.
436  *		should not be considered.
437  *	ndx - If MATCH_F_NDX flag is set, index of item under consideration.
438  *	type - If MATCH_F_TYPE is set, type of item under consideration.
439  *		If MATCH_F_PHDR is set, this would be a program
440  *		header type (PT_). Otherwise, a section header type (SHT_).
441  *
442  * exit:
443  *	True will be returned if the given name/index matches those given
444  *	by one of the (-I, -N -T) command line options, or if no such option
445  *	was used in the command invocation and MATCH_F_STRICT is not
446  *	set.
447  */
448 int
449 match(match_flags_t match_flags, const char *name, uint_t ndx, uint_t type)
450 {
451 	match_item_t item_type = (match_flags & MATCH_F_PHDR) ?
452 	    MATCH_ITEM_PT  : MATCH_ITEM_SHT;
453 	match_rec_t *list;
454 
455 	/*
456 	 * If there is no match list, then we use the MATCH_F_STRICT
457 	 * flag to decide what to return. In the strict case, we return
458 	 * False (0), in the normal case, True (1).
459 	 */
460 	if (match_state.list == NULL)
461 		return ((match_flags & MATCH_F_STRICT) == 0);
462 
463 	/*
464 	 * If item being checked is not the current match type,
465 	 * then allow it.
466 	 */
467 	if (item_type != match_state.item_type)
468 		return (1);
469 
470 	/* Run through the match records and check for a hit */
471 	for (list = match_state.list; list; list = list->next) {
472 		switch (list->opt_type) {
473 		case MATCH_OPT_NAME:
474 			if (((match_flags & MATCH_F_NAME) == 0) ||
475 			    (name == NULL))
476 				break;
477 			if (strcmp(list->value.name, name) == 0)
478 				return (1);
479 			break;
480 		case MATCH_OPT_NDX:
481 			if ((match_flags & MATCH_F_NDX) &&
482 			    (ndx == list->value.ndx.start))
483 				return (1);
484 			break;
485 		case MATCH_OPT_RANGE:
486 			/*
487 			 * A range end value less than 0 means that any value
488 			 * above the start is acceptible.
489 			 */
490 			if ((match_flags & MATCH_F_NDX) &&
491 			    (ndx >= list->value.ndx.start) &&
492 			    ((list->value.ndx.end < 0) ||
493 			    (ndx <= list->value.ndx.end)))
494 				return (1);
495 			break;
496 
497 		case MATCH_OPT_TYPE:
498 			if ((match_flags & MATCH_F_TYPE) &&
499 			    (type == list->value.type))
500 				return (1);
501 			break;
502 		}
503 	}
504 
505 	/* Nothing matched */
506 	return (0);
507 }
508 
509 /*
510  * Add an entry to match_state.list for use by match(). This routine is for
511  * use during getopt() processing. It should not be called once
512  * match_prepare() has been called.
513  *
514  * Return True (1) for success. On failure, an error is written
515  * to stderr, and False (0) is returned.
516  */
517 static int
518 add_match_record(char *argv0, match_rec_t *data)
519 {
520 	match_rec_t	*rec;
521 	match_rec_t	*list;
522 
523 	if ((rec = malloc(sizeof (*rec))) == NULL) {
524 		int err = errno;
525 		(void) fprintf(stderr, MSG_INTL(MSG_ERR_MALLOC),
526 		    basename(argv0), strerror(err));
527 		return (0);
528 	}
529 
530 	*rec = *data;
531 
532 	/* Insert at end of match_state.list */
533 	if (match_state.list == NULL) {
534 		match_state.list = rec;
535 	} else {
536 		for (list = match_state.list; list->next != NULL;
537 		    list = list->next)
538 			;
539 		list->next = rec;
540 	}
541 
542 	rec->next = NULL;
543 	return (1);
544 }
545 
546 static int
547 decide(const char *file, int fd, Elf *elf, uint_t flags,
548     const char *wname, int wfd, uchar_t osabi)
549 {
550 	int r;
551 
552 	if (gelf_getclass(elf) == ELFCLASS64)
553 		r = regular64(file, fd, elf, flags, wname, wfd, osabi);
554 	else
555 		r = regular32(file, fd, elf, flags, wname, wfd, osabi);
556 
557 	return (r);
558 }
559 
560 static int
561 archive(const char *file, int fd, Elf *elf, uint_t flags,
562     const char *wname, int wfd, uchar_t osabi)
563 {
564 	Elf_Cmd		cmd = ELF_C_READ;
565 	Elf_Arhdr	*arhdr;
566 	Elf		*_elf = NULL;
567 	size_t		ptr;
568 	Elf_Arsym	*arsym = NULL;
569 
570 	/*
571 	 * Determine if the archive symbol table itself is required.
572 	 */
573 	if ((flags & FLG_SHOW_SYMBOLS) &&
574 	    match(MATCH_F_NAME, MSG_ORIG(MSG_ELF_ARSYM), 0, 0)) {
575 		/*
576 		 * Get the archive symbol table.
577 		 */
578 		if (((arsym = elf_getarsym(elf, &ptr)) == 0) && elf_errno()) {
579 			/*
580 			 * The arsym could be 0 even though there was no error.
581 			 * Print the error message only when there was
582 			 * real error from elf_getarsym().
583 			 */
584 			failure(file, MSG_ORIG(MSG_ELF_GETARSYM));
585 			return (0);
586 		}
587 	}
588 
589 	/*
590 	 * Print the archive symbol table only when the archive symbol
591 	 * table exists and it was requested to print.
592 	 */
593 	if (arsym) {
594 		size_t		cnt;
595 		char		index[MAXNDXSIZE];
596 		size_t		offset = 0, _offset = 0;
597 		const char	*fmt_arsym1, *fmt_arsym2;
598 
599 		/*
600 		 * Print out all the symbol entries. The format width used
601 		 * corresponds to whether the archive symbol table is 32
602 		 * or 64-bit. We see them via Elf_Arhdr as size_t values
603 		 * in either case with no information loss (see the comments
604 		 * in libelf/getarsym.c) so this is done simply to improve
605 		 * the user presentation.
606 		 */
607 		if (_elf_getarsymwordsize(elf) == 8) {
608 			dbg_print(0, MSG_INTL(MSG_ARCHIVE_SYMTAB_64));
609 			dbg_print(0, MSG_INTL(MSG_ARCHIVE_FIELDS_64));
610 
611 			fmt_arsym1 = MSG_ORIG(MSG_FMT_ARSYM1_64);
612 			fmt_arsym2 = MSG_ORIG(MSG_FMT_ARSYM2_64);
613 		} else {
614 			dbg_print(0, MSG_INTL(MSG_ARCHIVE_SYMTAB_32));
615 			dbg_print(0, MSG_INTL(MSG_ARCHIVE_FIELDS_32));
616 
617 			fmt_arsym1 = MSG_ORIG(MSG_FMT_ARSYM1_32);
618 			fmt_arsym2 = MSG_ORIG(MSG_FMT_ARSYM2_32);
619 		}
620 
621 		for (cnt = 0; cnt < ptr; cnt++, arsym++) {
622 			/*
623 			 * For each object obtain an elf descriptor so that we
624 			 * can establish the members name.  Note, we have had
625 			 * archives where the archive header has not been
626 			 * obtainable so be lenient with errors.
627 			 */
628 			if ((offset == 0) || ((arsym->as_off != 0) &&
629 			    (arsym->as_off != _offset))) {
630 
631 				if (_elf)
632 					(void) elf_end(_elf);
633 
634 				if (elf_rand(elf, arsym->as_off) !=
635 				    arsym->as_off) {
636 					failure(file, MSG_ORIG(MSG_ELF_RAND));
637 					arhdr = NULL;
638 				} else if ((_elf = elf_begin(fd,
639 				    ELF_C_READ, elf)) == 0) {
640 					failure(file, MSG_ORIG(MSG_ELF_BEGIN));
641 					arhdr = NULL;
642 				} else if ((arhdr = elf_getarhdr(_elf)) == 0) {
643 					failure(file,
644 					    MSG_ORIG(MSG_ELF_GETARHDR));
645 					arhdr = NULL;
646 				}
647 
648 				_offset = arsym->as_off;
649 				if (offset == 0)
650 					offset = _offset;
651 			}
652 
653 			(void) snprintf(index, MAXNDXSIZE,
654 			    MSG_ORIG(MSG_FMT_INDEX), EC_XWORD(cnt));
655 			if (arsym->as_off)
656 				dbg_print(0, fmt_arsym1, index,
657 				    EC_XWORD(arsym->as_off),
658 				    arhdr ? arhdr->ar_name :
659 				    MSG_INTL(MSG_STR_UNKNOWN), (arsym->as_name ?
660 				    demangle(arsym->as_name, flags) :
661 				    MSG_INTL(MSG_STR_NULL)));
662 			else
663 				dbg_print(0, fmt_arsym2, index,
664 				    EC_XWORD(arsym->as_off));
665 		}
666 
667 		if (_elf)
668 			(void) elf_end(_elf);
669 
670 		/*
671 		 * If we only need the archive symbol table return.
672 		 */
673 		if ((flags & FLG_SHOW_SYMBOLS) &&
674 		    match(MATCH_F_STRICT | MATCH_F_NAME,
675 		    MSG_ORIG(MSG_ELF_ARSYM), -1, -1))
676 			return (0);
677 
678 		/*
679 		 * Reset elf descriptor in preparation for processing each
680 		 * member.
681 		 */
682 		if (offset)
683 			(void) elf_rand(elf, offset);
684 	}
685 
686 	/*
687 	 * Process each object within the archive.
688 	 */
689 	while ((_elf = elf_begin(fd, cmd, elf)) != NULL) {
690 		char	name[MAXPATHLEN];
691 
692 		if ((arhdr = elf_getarhdr(_elf)) == NULL) {
693 			failure(file, MSG_ORIG(MSG_ELF_GETARHDR));
694 			return (0);
695 		}
696 		if (*arhdr->ar_name != '/') {
697 			(void) snprintf(name, MAXPATHLEN,
698 			    MSG_ORIG(MSG_FMT_ARNAME), file, arhdr->ar_name);
699 			dbg_print(0, MSG_ORIG(MSG_FMT_NLSTR), name);
700 
701 			switch (elf_kind(_elf)) {
702 			case ELF_K_AR:
703 				if (archive(name, fd, _elf, flags,
704 				    wname, wfd, osabi) == 1)
705 					return (1);
706 				break;
707 			case ELF_K_ELF:
708 				if (decide(name, fd, _elf, flags,
709 				    wname, wfd, osabi) == 1)
710 					return (1);
711 				break;
712 			default:
713 				(void) fprintf(stderr,
714 				    MSG_INTL(MSG_ERR_BADFILE), name);
715 				break;
716 			}
717 		}
718 
719 		cmd = elf_next(_elf);
720 		(void) elf_end(_elf);
721 	}
722 
723 	return (0);
724 }
725 
726 int
727 main(int argc, char **argv, char **envp)
728 {
729 	Elf		*elf;
730 	int		var, fd, wfd = 0;
731 	char		*wname = NULL;
732 	uint_t		flags = 0;
733 	match_rec_t	match_data;
734 	int		ret;
735 	uchar_t		osabi;
736 
737 	/*
738 	 * If we're on a 64-bit kernel, try to exec a full 64-bit version of
739 	 * the binary.  If successful, conv_check_native() won't return.
740 	 */
741 	(void) conv_check_native(argv, envp);
742 
743 	/*
744 	 * Establish locale.
745 	 */
746 	(void) setlocale(LC_MESSAGES, MSG_ORIG(MSG_STR_EMPTY));
747 	(void) textdomain(MSG_ORIG(MSG_SUNW_OST_SGS));
748 
749 	(void) setvbuf(stdout, NULL, _IOLBF, 0);
750 	(void) setvbuf(stderr, NULL, _IOLBF, 0);
751 
752 	opterr = 0;
753 	while ((var = getopt(argc, argv, MSG_ORIG(MSG_STR_OPTIONS))) != EOF) {
754 		switch (var) {
755 		case 'C':
756 			flags |= FLG_CTL_DEMANGLE;
757 			break;
758 		case 'c':
759 			flags |= FLG_SHOW_SHDR;
760 			break;
761 		case 'd':
762 			flags |= FLG_SHOW_DYNAMIC;
763 			break;
764 		case 'e':
765 			flags |= FLG_SHOW_EHDR;
766 			break;
767 		case 'G':
768 			flags |= FLG_SHOW_GOT;
769 			break;
770 		case 'g':
771 			flags |= FLG_SHOW_GROUP;
772 			break;
773 		case 'H':
774 			flags |= FLG_SHOW_CAP;
775 			break;
776 		case 'h':
777 			flags |= FLG_SHOW_HASH;
778 			break;
779 		case 'I':
780 			if (!process_index_opt(optarg, &match_data))
781 				goto usage_brief;
782 			if (!add_match_record(argv[0], &match_data))
783 				return (1);
784 			flags |= FLG_CTL_MATCH;
785 			break;
786 		case 'i':
787 			flags |= FLG_SHOW_INTERP;
788 			break;
789 		case 'k':
790 			flags |= FLG_CALC_CHECKSUM;
791 			break;
792 		case 'l':
793 			flags |= FLG_CTL_LONGNAME;
794 			break;
795 		case 'm':
796 			flags |= FLG_SHOW_MOVE;
797 			break;
798 		case 'N':
799 			match_data.opt_type = MATCH_OPT_NAME;
800 			match_data.value.name = optarg;
801 			if (!add_match_record(argv[0], &match_data))
802 				return (1);
803 			flags |= FLG_CTL_MATCH;
804 			break;
805 		case 'n':
806 			flags |= FLG_SHOW_NOTE;
807 			break;
808 		case 'O':
809 			{
810 				uint32_t val;
811 
812 				/*
813 				 * osabi is a uchar_t in the ELF header.
814 				 * Don't accept any value that exceeds
815 				 * that range.
816 				 */
817 				if ((atoui(optarg, ATOUI_OSABI, &val) == 0) ||
818 				    (val > 255)) {
819 					(void) fprintf(stderr,
820 					    MSG_INTL(MSG_ERR_BAD_T_OSABI),
821 					    basename(argv[0]), optarg);
822 					return (1);
823 				}
824 				osabi = val;
825 			}
826 			flags |= FLG_CTL_OSABI;
827 			break;
828 		case 'P':
829 			flags |= FLG_CTL_FAKESHDR;
830 			break;
831 		case 'p':
832 			flags |= FLG_SHOW_PHDR;
833 			break;
834 		case 'r':
835 			flags |= FLG_SHOW_RELOC;
836 			break;
837 		case 'S':
838 			flags |= FLG_SHOW_SORT;
839 			break;
840 		case 's':
841 			flags |= FLG_SHOW_SYMBOLS;
842 			break;
843 		case 'T':
844 			/*
845 			 * We can't evaluate the value yet, because
846 			 * we need to know if -p is used or not in
847 			 * order to tell if we're seeing section header
848 			 * or program header types. So, we save the
849 			 * string in the name field, and then convert
850 			 * it to a type integer in a following pass.
851 			 */
852 			match_data.opt_type = MATCH_OPT_TYPE;
853 			match_data.value.name = optarg;
854 			if (!add_match_record(argv[0], &match_data))
855 				return (1);
856 			flags |= FLG_CTL_MATCH;
857 			break;
858 		case 'u':
859 			flags |= FLG_SHOW_UNWIND;
860 			break;
861 		case 'v':
862 			flags |= FLG_SHOW_VERSIONS;
863 			break;
864 		case 'w':
865 			wname = optarg;
866 			break;
867 		case 'y':
868 			flags |= FLG_SHOW_SYMINFO;
869 			break;
870 		case '?':
871 			(void) fprintf(stderr, MSG_INTL(MSG_USAGE_BRIEF),
872 			    basename(argv[0]));
873 			detail_usage();
874 			return (1);
875 		default:
876 			break;
877 		}
878 	}
879 
880 	/* -p and -w are mutually exclusive. -w only works with sections */
881 	if (((flags & FLG_SHOW_PHDR) != 0) && (wname != NULL))
882 		goto usage_brief;
883 
884 	/* If a match argument is present, prepare the match state */
885 	if ((match_state.list != NULL) && (match_prepare(argv[0], flags) == 0))
886 		return (1);
887 
888 	/*
889 	 * Decide what to do if no options specifying something to
890 	 * show or do are present.
891 	 *
892 	 * If there is no -w and no match options, then we will set all
893 	 * the show flags, causing a full display of everything in the
894 	 * file that we know how to handle.
895 	 *
896 	 * Otherwise, if there is no match list, we generate a usage
897 	 * error and quit.
898 	 *
899 	 * In the case where there is a match list, we go ahead and call
900 	 * regular() anyway, leaving it to decide what to do. If -w is
901 	 * present, regular() will use the match list to handle it.
902 	 * In addition, in the absence of explicit show/calc flags, regular()
903 	 * will compare the section headers to the match list and use
904 	 * that to generate the FLG_ bits that will display the information
905 	 * specified by the match list.
906 	 */
907 	if ((flags & ~FLG_MASK_CTL) == 0) {
908 		if (!wname && (match_state.list == NULL))
909 			flags |= FLG_MASK_SHOW;
910 		else if (match_state.list == NULL)
911 			goto usage_brief;
912 	}
913 
914 	/* There needs to be at least 1 filename left following the options */
915 	if ((var = argc - optind) == 0)
916 		goto usage_brief;
917 
918 	/*
919 	 * If the -l/-C option is specified, set up the liblddbg.so.
920 	 */
921 	if (flags & FLG_CTL_LONGNAME)
922 		dbg_desc->d_extra |= DBG_E_LONG;
923 	if (flags & FLG_CTL_DEMANGLE)
924 		dbg_desc->d_extra |= DBG_E_DEMANGLE;
925 
926 	/*
927 	 * If the -w option has indicated an output file open it.  It's
928 	 * arguable whether this option has much use when multiple files are
929 	 * being processed.
930 	 *
931 	 * If wname is non-NULL, we know that -p was not specified, due
932 	 * to the test above.
933 	 */
934 	if (wname) {
935 		if ((wfd = open(wname, (O_RDWR | O_CREAT | O_TRUNC),
936 		    0666)) < 0) {
937 			int err = errno;
938 			(void) fprintf(stderr, MSG_INTL(MSG_ERR_OPEN),
939 			    wname, strerror(err));
940 			return (1);
941 		}
942 	}
943 
944 	/*
945 	 * Open the input file, initialize the elf interface, and
946 	 * process it.
947 	 */
948 	ret = 0;
949 	for (; (optind < argc) && (ret == 0); optind++) {
950 		const char	*file = argv[optind];
951 
952 		if ((fd = open(argv[optind], O_RDONLY)) == -1) {
953 			int err = errno;
954 			(void) fprintf(stderr, MSG_INTL(MSG_ERR_OPEN),
955 			    file, strerror(err));
956 			continue;
957 		}
958 		(void) elf_version(EV_CURRENT);
959 		if ((elf = elf_begin(fd, ELF_C_READ, NULL)) == NULL) {
960 			failure(file, MSG_ORIG(MSG_ELF_BEGIN));
961 			(void) close(fd);
962 			continue;
963 		}
964 
965 		if (var > 1)
966 			dbg_print(0, MSG_ORIG(MSG_FMT_NLSTRNL), file);
967 
968 		switch (elf_kind(elf)) {
969 		case ELF_K_AR:
970 			ret = archive(file, fd, elf, flags, wname, wfd, osabi);
971 			break;
972 		case ELF_K_ELF:
973 			ret = decide(file, fd, elf, flags, wname, wfd, osabi);
974 			break;
975 		default:
976 			(void) fprintf(stderr, MSG_INTL(MSG_ERR_BADFILE), file);
977 			break;
978 		}
979 
980 		(void) close(fd);
981 		(void) elf_end(elf);
982 	}
983 
984 	if (wfd)
985 		(void) close(wfd);
986 	return (ret);
987 
988 usage_brief:
989 	/* Control comes here for a simple usage message and exit */
990 	(void) fprintf(stderr, MSG_INTL(MSG_USAGE_BRIEF),
991 	    basename(argv[0]));
992 	return (1);
993 
994 }
995