xref: /aosp_15_r20/external/elfutils/src/size.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Print size information from ELF file.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2000-2007,2009,2012,2014,2015 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker    This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker    Written by Ulrich Drepper <[email protected]>, 2000.
5*7304104dSAndroid Build Coastguard Worker 
6*7304104dSAndroid Build Coastguard Worker    This file is free software; you can redistribute it and/or modify
7*7304104dSAndroid Build Coastguard Worker    it under the terms of the GNU General Public License as published by
8*7304104dSAndroid Build Coastguard Worker    the Free Software Foundation; either version 3 of the License, or
9*7304104dSAndroid Build Coastguard Worker    (at your option) any later version.
10*7304104dSAndroid Build Coastguard Worker 
11*7304104dSAndroid Build Coastguard Worker    elfutils is distributed in the hope that it will be useful, but
12*7304104dSAndroid Build Coastguard Worker    WITHOUT ANY WARRANTY; without even the implied warranty of
13*7304104dSAndroid Build Coastguard Worker    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14*7304104dSAndroid Build Coastguard Worker    GNU General Public License for more details.
15*7304104dSAndroid Build Coastguard Worker 
16*7304104dSAndroid Build Coastguard Worker    You should have received a copy of the GNU General Public License
17*7304104dSAndroid Build Coastguard Worker    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
18*7304104dSAndroid Build Coastguard Worker 
19*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
20*7304104dSAndroid Build Coastguard Worker # include <config.h>
21*7304104dSAndroid Build Coastguard Worker #endif
22*7304104dSAndroid Build Coastguard Worker 
23*7304104dSAndroid Build Coastguard Worker #include <argp.h>
24*7304104dSAndroid Build Coastguard Worker #include <fcntl.h>
25*7304104dSAndroid Build Coastguard Worker #include <gelf.h>
26*7304104dSAndroid Build Coastguard Worker #include <inttypes.h>
27*7304104dSAndroid Build Coastguard Worker #include <libelf.h>
28*7304104dSAndroid Build Coastguard Worker #include <locale.h>
29*7304104dSAndroid Build Coastguard Worker #include <stdbool.h>
30*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
31*7304104dSAndroid Build Coastguard Worker #include <stdio_ext.h>
32*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
33*7304104dSAndroid Build Coastguard Worker #include <string.h>
34*7304104dSAndroid Build Coastguard Worker #include <unistd.h>
35*7304104dSAndroid Build Coastguard Worker 
36*7304104dSAndroid Build Coastguard Worker #include <system.h>
37*7304104dSAndroid Build Coastguard Worker #include <printversion.h>
38*7304104dSAndroid Build Coastguard Worker 
39*7304104dSAndroid Build Coastguard Worker /* Name and version of program.  */
40*7304104dSAndroid Build Coastguard Worker ARGP_PROGRAM_VERSION_HOOK_DEF = print_version;
41*7304104dSAndroid Build Coastguard Worker 
42*7304104dSAndroid Build Coastguard Worker /* Bug report address.  */
43*7304104dSAndroid Build Coastguard Worker ARGP_PROGRAM_BUG_ADDRESS_DEF = PACKAGE_BUGREPORT;
44*7304104dSAndroid Build Coastguard Worker 
45*7304104dSAndroid Build Coastguard Worker 
46*7304104dSAndroid Build Coastguard Worker /* Values for the parameters which have no short form.  */
47*7304104dSAndroid Build Coastguard Worker #define OPT_FORMAT	0x100
48*7304104dSAndroid Build Coastguard Worker #define OPT_RADIX	0x101
49*7304104dSAndroid Build Coastguard Worker 
50*7304104dSAndroid Build Coastguard Worker /* Definitions of arguments for argp functions.  */
51*7304104dSAndroid Build Coastguard Worker static const struct argp_option options[] =
52*7304104dSAndroid Build Coastguard Worker {
53*7304104dSAndroid Build Coastguard Worker   { NULL, 0, NULL, 0, N_("Output format:"), 0 },
54*7304104dSAndroid Build Coastguard Worker   { "format", OPT_FORMAT, "FORMAT", 0,
55*7304104dSAndroid Build Coastguard Worker     N_("Use the output format FORMAT.  FORMAT can be `bsd' or `sysv'.  "
56*7304104dSAndroid Build Coastguard Worker        "The default is `bsd'"), 0 },
57*7304104dSAndroid Build Coastguard Worker   { NULL, 'A', NULL, 0, N_("Same as `--format=sysv'"), 0 },
58*7304104dSAndroid Build Coastguard Worker   { NULL, 'B', NULL, 0, N_("Same as `--format=bsd'"), 0 },
59*7304104dSAndroid Build Coastguard Worker   { "radix", OPT_RADIX, "RADIX", 0, N_("Use RADIX for printing symbol values"),
60*7304104dSAndroid Build Coastguard Worker     0},
61*7304104dSAndroid Build Coastguard Worker   { NULL, 'd', NULL, 0, N_("Same as `--radix=10'"), 0 },
62*7304104dSAndroid Build Coastguard Worker   { NULL, 'o', NULL, 0, N_("Same as `--radix=8'"), 0 },
63*7304104dSAndroid Build Coastguard Worker   { NULL, 'x', NULL, 0, N_("Same as `--radix=16'"), 0 },
64*7304104dSAndroid Build Coastguard Worker   { NULL, 'f', NULL, 0,
65*7304104dSAndroid Build Coastguard Worker     N_("Similar to `--format=sysv' output but in one line"), 0 },
66*7304104dSAndroid Build Coastguard Worker 
67*7304104dSAndroid Build Coastguard Worker   { NULL, 0, NULL, 0, N_("Output options:"), 0 },
68*7304104dSAndroid Build Coastguard Worker   { NULL, 'F', NULL, 0,
69*7304104dSAndroid Build Coastguard Worker     N_("Print size and permission flags for loadable segments"), 0 },
70*7304104dSAndroid Build Coastguard Worker   { "totals", 't', NULL, 0, N_("Display the total sizes (bsd only)"), 0 },
71*7304104dSAndroid Build Coastguard Worker   { NULL, 0, NULL, 0, NULL, 0 }
72*7304104dSAndroid Build Coastguard Worker };
73*7304104dSAndroid Build Coastguard Worker 
74*7304104dSAndroid Build Coastguard Worker /* Short description of program.  */
75*7304104dSAndroid Build Coastguard Worker static const char doc[] = N_("\
76*7304104dSAndroid Build Coastguard Worker List section sizes of FILEs (a.out by default).");
77*7304104dSAndroid Build Coastguard Worker 
78*7304104dSAndroid Build Coastguard Worker /* Strings for arguments in help texts.  */
79*7304104dSAndroid Build Coastguard Worker static const char args_doc[] = N_("[FILE...]");
80*7304104dSAndroid Build Coastguard Worker 
81*7304104dSAndroid Build Coastguard Worker /* Prototype for option handler.  */
82*7304104dSAndroid Build Coastguard Worker static error_t parse_opt (int key, char *arg, struct argp_state *state);
83*7304104dSAndroid Build Coastguard Worker 
84*7304104dSAndroid Build Coastguard Worker /* Data structure to communicate with argp functions.  */
85*7304104dSAndroid Build Coastguard Worker static struct argp argp =
86*7304104dSAndroid Build Coastguard Worker {
87*7304104dSAndroid Build Coastguard Worker   options, parse_opt, args_doc, doc, NULL, NULL, NULL
88*7304104dSAndroid Build Coastguard Worker };
89*7304104dSAndroid Build Coastguard Worker 
90*7304104dSAndroid Build Coastguard Worker 
91*7304104dSAndroid Build Coastguard Worker /* Print symbols in file named FNAME.  */
92*7304104dSAndroid Build Coastguard Worker static int process_file (const char *fname);
93*7304104dSAndroid Build Coastguard Worker 
94*7304104dSAndroid Build Coastguard Worker /* Handle content of archive.  */
95*7304104dSAndroid Build Coastguard Worker static int handle_ar (int fd, Elf *elf, const char *prefix, const char *fname);
96*7304104dSAndroid Build Coastguard Worker 
97*7304104dSAndroid Build Coastguard Worker /* Handle ELF file.  */
98*7304104dSAndroid Build Coastguard Worker static void handle_elf (Elf *elf, const char *fullname, const char *fname);
99*7304104dSAndroid Build Coastguard Worker 
100*7304104dSAndroid Build Coastguard Worker /* Show total size.  */
101*7304104dSAndroid Build Coastguard Worker static void show_bsd_totals (void);
102*7304104dSAndroid Build Coastguard Worker 
103*7304104dSAndroid Build Coastguard Worker #define INTERNAL_ERROR(fname) \
104*7304104dSAndroid Build Coastguard Worker   error_exit (0, _("%s: INTERNAL ERROR %d (%s): %s"),      \
105*7304104dSAndroid Build Coastguard Worker 	      fname, __LINE__, PACKAGE_VERSION, elf_errmsg (-1))
106*7304104dSAndroid Build Coastguard Worker 
107*7304104dSAndroid Build Coastguard Worker 
108*7304104dSAndroid Build Coastguard Worker /* User-selectable options.  */
109*7304104dSAndroid Build Coastguard Worker 
110*7304104dSAndroid Build Coastguard Worker /* The selected output format.  */
111*7304104dSAndroid Build Coastguard Worker static enum
112*7304104dSAndroid Build Coastguard Worker {
113*7304104dSAndroid Build Coastguard Worker   format_bsd = 0,
114*7304104dSAndroid Build Coastguard Worker   format_sysv,
115*7304104dSAndroid Build Coastguard Worker   format_sysv_one_line,
116*7304104dSAndroid Build Coastguard Worker   format_segments
117*7304104dSAndroid Build Coastguard Worker } format;
118*7304104dSAndroid Build Coastguard Worker 
119*7304104dSAndroid Build Coastguard Worker /* Radix for printed numbers.  */
120*7304104dSAndroid Build Coastguard Worker static enum
121*7304104dSAndroid Build Coastguard Worker {
122*7304104dSAndroid Build Coastguard Worker   radix_decimal = 0,
123*7304104dSAndroid Build Coastguard Worker   radix_hex,
124*7304104dSAndroid Build Coastguard Worker   radix_octal
125*7304104dSAndroid Build Coastguard Worker } radix;
126*7304104dSAndroid Build Coastguard Worker 
127*7304104dSAndroid Build Coastguard Worker 
128*7304104dSAndroid Build Coastguard Worker /* Mapping of radix and binary class to length.  */
129*7304104dSAndroid Build Coastguard Worker static const int length_map[2][3] =
130*7304104dSAndroid Build Coastguard Worker {
131*7304104dSAndroid Build Coastguard Worker   [ELFCLASS32 - 1] =
132*7304104dSAndroid Build Coastguard Worker   {
133*7304104dSAndroid Build Coastguard Worker     [radix_hex] = 8,
134*7304104dSAndroid Build Coastguard Worker     [radix_decimal] = 10,
135*7304104dSAndroid Build Coastguard Worker     [radix_octal] = 11
136*7304104dSAndroid Build Coastguard Worker   },
137*7304104dSAndroid Build Coastguard Worker   [ELFCLASS64 - 1] =
138*7304104dSAndroid Build Coastguard Worker   {
139*7304104dSAndroid Build Coastguard Worker     [radix_hex] = 16,
140*7304104dSAndroid Build Coastguard Worker     [radix_decimal] = 20,
141*7304104dSAndroid Build Coastguard Worker     [radix_octal] = 22
142*7304104dSAndroid Build Coastguard Worker   }
143*7304104dSAndroid Build Coastguard Worker };
144*7304104dSAndroid Build Coastguard Worker 
145*7304104dSAndroid Build Coastguard Worker /* True if total sizes should be printed.  */
146*7304104dSAndroid Build Coastguard Worker static bool totals;
147*7304104dSAndroid Build Coastguard Worker /* To print the total sizes in a reasonable format remember the highest
148*7304104dSAndroid Build Coastguard Worker    "class" of ELF binaries processed.  */
149*7304104dSAndroid Build Coastguard Worker static int totals_class;
150*7304104dSAndroid Build Coastguard Worker 
151*7304104dSAndroid Build Coastguard Worker 
152*7304104dSAndroid Build Coastguard Worker int
main(int argc,char * argv[])153*7304104dSAndroid Build Coastguard Worker main (int argc, char *argv[])
154*7304104dSAndroid Build Coastguard Worker {
155*7304104dSAndroid Build Coastguard Worker   int remaining;
156*7304104dSAndroid Build Coastguard Worker   int result = 0;
157*7304104dSAndroid Build Coastguard Worker 
158*7304104dSAndroid Build Coastguard Worker   /* We use no threads here which can interfere with handling a stream.  */
159*7304104dSAndroid Build Coastguard Worker   __fsetlocking (stdin, FSETLOCKING_BYCALLER);
160*7304104dSAndroid Build Coastguard Worker   __fsetlocking (stdout, FSETLOCKING_BYCALLER);
161*7304104dSAndroid Build Coastguard Worker   __fsetlocking (stderr, FSETLOCKING_BYCALLER);
162*7304104dSAndroid Build Coastguard Worker 
163*7304104dSAndroid Build Coastguard Worker   /* Set locale.  */
164*7304104dSAndroid Build Coastguard Worker   setlocale (LC_ALL, "");
165*7304104dSAndroid Build Coastguard Worker 
166*7304104dSAndroid Build Coastguard Worker   /* Make sure the message catalog can be found.  */
167*7304104dSAndroid Build Coastguard Worker   bindtextdomain (PACKAGE_TARNAME, LOCALEDIR);
168*7304104dSAndroid Build Coastguard Worker 
169*7304104dSAndroid Build Coastguard Worker   /* Initialize the message catalog.  */
170*7304104dSAndroid Build Coastguard Worker   textdomain (PACKAGE_TARNAME);
171*7304104dSAndroid Build Coastguard Worker 
172*7304104dSAndroid Build Coastguard Worker   /* Parse and process arguments.  */
173*7304104dSAndroid Build Coastguard Worker   argp_parse (&argp, argc, argv, 0, &remaining, NULL);
174*7304104dSAndroid Build Coastguard Worker 
175*7304104dSAndroid Build Coastguard Worker 
176*7304104dSAndroid Build Coastguard Worker   /* Tell the library which version we are expecting.  */
177*7304104dSAndroid Build Coastguard Worker   elf_version (EV_CURRENT);
178*7304104dSAndroid Build Coastguard Worker 
179*7304104dSAndroid Build Coastguard Worker   if (remaining == argc)
180*7304104dSAndroid Build Coastguard Worker     /* The user didn't specify a name so we use a.out.  */
181*7304104dSAndroid Build Coastguard Worker     result = process_file ("a.out");
182*7304104dSAndroid Build Coastguard Worker   else
183*7304104dSAndroid Build Coastguard Worker     /* Process all the remaining files.  */
184*7304104dSAndroid Build Coastguard Worker     do
185*7304104dSAndroid Build Coastguard Worker       result |= process_file (argv[remaining]);
186*7304104dSAndroid Build Coastguard Worker     while (++remaining < argc);
187*7304104dSAndroid Build Coastguard Worker 
188*7304104dSAndroid Build Coastguard Worker   /* Print the total sizes but only if the output format is BSD and at
189*7304104dSAndroid Build Coastguard Worker      least one file has been correctly read (i.e., we recognized the
190*7304104dSAndroid Build Coastguard Worker      class).  */
191*7304104dSAndroid Build Coastguard Worker   if (totals && format == format_bsd && totals_class != 0)
192*7304104dSAndroid Build Coastguard Worker     show_bsd_totals ();
193*7304104dSAndroid Build Coastguard Worker 
194*7304104dSAndroid Build Coastguard Worker   return result;
195*7304104dSAndroid Build Coastguard Worker }
196*7304104dSAndroid Build Coastguard Worker 
197*7304104dSAndroid Build Coastguard Worker 
198*7304104dSAndroid Build Coastguard Worker /* Handle program arguments.  */
199*7304104dSAndroid Build Coastguard Worker static error_t
parse_opt(int key,char * arg,struct argp_state * state)200*7304104dSAndroid Build Coastguard Worker parse_opt (int key, char *arg,
201*7304104dSAndroid Build Coastguard Worker 	   struct argp_state *state __attribute__ ((unused)))
202*7304104dSAndroid Build Coastguard Worker {
203*7304104dSAndroid Build Coastguard Worker   switch (key)
204*7304104dSAndroid Build Coastguard Worker     {
205*7304104dSAndroid Build Coastguard Worker     case 'd':
206*7304104dSAndroid Build Coastguard Worker       radix = radix_decimal;
207*7304104dSAndroid Build Coastguard Worker       break;
208*7304104dSAndroid Build Coastguard Worker 
209*7304104dSAndroid Build Coastguard Worker     case 'f':
210*7304104dSAndroid Build Coastguard Worker       format = format_sysv_one_line;
211*7304104dSAndroid Build Coastguard Worker       break;
212*7304104dSAndroid Build Coastguard Worker 
213*7304104dSAndroid Build Coastguard Worker     case 'o':
214*7304104dSAndroid Build Coastguard Worker       radix = radix_octal;
215*7304104dSAndroid Build Coastguard Worker       break;
216*7304104dSAndroid Build Coastguard Worker 
217*7304104dSAndroid Build Coastguard Worker     case 'x':
218*7304104dSAndroid Build Coastguard Worker       radix = radix_hex;
219*7304104dSAndroid Build Coastguard Worker       break;
220*7304104dSAndroid Build Coastguard Worker 
221*7304104dSAndroid Build Coastguard Worker     case 'A':
222*7304104dSAndroid Build Coastguard Worker       format = format_sysv;
223*7304104dSAndroid Build Coastguard Worker       break;
224*7304104dSAndroid Build Coastguard Worker 
225*7304104dSAndroid Build Coastguard Worker     case 'B':
226*7304104dSAndroid Build Coastguard Worker       format = format_bsd;
227*7304104dSAndroid Build Coastguard Worker       break;
228*7304104dSAndroid Build Coastguard Worker 
229*7304104dSAndroid Build Coastguard Worker     case 'F':
230*7304104dSAndroid Build Coastguard Worker       format = format_segments;
231*7304104dSAndroid Build Coastguard Worker       break;
232*7304104dSAndroid Build Coastguard Worker 
233*7304104dSAndroid Build Coastguard Worker     case OPT_FORMAT:
234*7304104dSAndroid Build Coastguard Worker       if (strcmp (arg, "bsd") == 0 || strcmp (arg, "berkeley") == 0)
235*7304104dSAndroid Build Coastguard Worker 	format = format_bsd;
236*7304104dSAndroid Build Coastguard Worker       else if (likely (strcmp (arg, "sysv") == 0))
237*7304104dSAndroid Build Coastguard Worker 	format = format_sysv;
238*7304104dSAndroid Build Coastguard Worker       else
239*7304104dSAndroid Build Coastguard Worker 	error_exit (0, _("Invalid format: %s"), arg);
240*7304104dSAndroid Build Coastguard Worker       break;
241*7304104dSAndroid Build Coastguard Worker 
242*7304104dSAndroid Build Coastguard Worker     case OPT_RADIX:
243*7304104dSAndroid Build Coastguard Worker       if (strcmp (arg, "x") == 0 || strcmp (arg, "16") == 0)
244*7304104dSAndroid Build Coastguard Worker 	radix = radix_hex;
245*7304104dSAndroid Build Coastguard Worker       else if (strcmp (arg, "d") == 0 || strcmp (arg, "10") == 0)
246*7304104dSAndroid Build Coastguard Worker 	radix = radix_decimal;
247*7304104dSAndroid Build Coastguard Worker       else if (strcmp (arg, "o") == 0 || strcmp (arg, "8") == 0)
248*7304104dSAndroid Build Coastguard Worker 	radix = radix_octal;
249*7304104dSAndroid Build Coastguard Worker       else
250*7304104dSAndroid Build Coastguard Worker 	error_exit (0, _("Invalid radix: %s"), arg);
251*7304104dSAndroid Build Coastguard Worker       break;
252*7304104dSAndroid Build Coastguard Worker 
253*7304104dSAndroid Build Coastguard Worker     case 't':
254*7304104dSAndroid Build Coastguard Worker       totals = true;
255*7304104dSAndroid Build Coastguard Worker       break;
256*7304104dSAndroid Build Coastguard Worker 
257*7304104dSAndroid Build Coastguard Worker     default:
258*7304104dSAndroid Build Coastguard Worker       return ARGP_ERR_UNKNOWN;
259*7304104dSAndroid Build Coastguard Worker     }
260*7304104dSAndroid Build Coastguard Worker   return 0;
261*7304104dSAndroid Build Coastguard Worker }
262*7304104dSAndroid Build Coastguard Worker 
263*7304104dSAndroid Build Coastguard Worker 
264*7304104dSAndroid Build Coastguard Worker /* Open the file and determine the type.  */
265*7304104dSAndroid Build Coastguard Worker static int
process_file(const char * fname)266*7304104dSAndroid Build Coastguard Worker process_file (const char *fname)
267*7304104dSAndroid Build Coastguard Worker {
268*7304104dSAndroid Build Coastguard Worker   int fd = open (fname, O_RDONLY);
269*7304104dSAndroid Build Coastguard Worker   if (unlikely (fd == -1))
270*7304104dSAndroid Build Coastguard Worker     {
271*7304104dSAndroid Build Coastguard Worker       error (0, errno, _("cannot open '%s'"), fname);
272*7304104dSAndroid Build Coastguard Worker       return 1;
273*7304104dSAndroid Build Coastguard Worker     }
274*7304104dSAndroid Build Coastguard Worker 
275*7304104dSAndroid Build Coastguard Worker   /* Now get the ELF descriptor.  */
276*7304104dSAndroid Build Coastguard Worker   Elf *elf = elf_begin (fd, ELF_C_READ_MMAP, NULL);
277*7304104dSAndroid Build Coastguard Worker   if (likely (elf != NULL))
278*7304104dSAndroid Build Coastguard Worker     {
279*7304104dSAndroid Build Coastguard Worker       if (elf_kind (elf) == ELF_K_ELF)
280*7304104dSAndroid Build Coastguard Worker 	{
281*7304104dSAndroid Build Coastguard Worker 	  handle_elf (elf, NULL, fname);
282*7304104dSAndroid Build Coastguard Worker 
283*7304104dSAndroid Build Coastguard Worker 	  if (unlikely (elf_end (elf) != 0))
284*7304104dSAndroid Build Coastguard Worker 	    INTERNAL_ERROR (fname);
285*7304104dSAndroid Build Coastguard Worker 
286*7304104dSAndroid Build Coastguard Worker 	  if (unlikely (close (fd) != 0))
287*7304104dSAndroid Build Coastguard Worker 	    error_exit (errno, _("while closing '%s'"), fname);
288*7304104dSAndroid Build Coastguard Worker 
289*7304104dSAndroid Build Coastguard Worker 	  return 0;
290*7304104dSAndroid Build Coastguard Worker 	}
291*7304104dSAndroid Build Coastguard Worker       else if (likely (elf_kind (elf) == ELF_K_AR))
292*7304104dSAndroid Build Coastguard Worker 	{
293*7304104dSAndroid Build Coastguard Worker 	  int result = handle_ar (fd, elf, NULL, fname);
294*7304104dSAndroid Build Coastguard Worker 
295*7304104dSAndroid Build Coastguard Worker 	  if (unlikely  (close (fd) != 0))
296*7304104dSAndroid Build Coastguard Worker 	    error_exit (errno, _("while closing '%s'"), fname);
297*7304104dSAndroid Build Coastguard Worker 
298*7304104dSAndroid Build Coastguard Worker 	  return result;
299*7304104dSAndroid Build Coastguard Worker 	}
300*7304104dSAndroid Build Coastguard Worker 
301*7304104dSAndroid Build Coastguard Worker       /* We cannot handle this type.  Close the descriptor anyway.  */
302*7304104dSAndroid Build Coastguard Worker       if (unlikely (elf_end (elf) != 0))
303*7304104dSAndroid Build Coastguard Worker 	INTERNAL_ERROR (fname);
304*7304104dSAndroid Build Coastguard Worker     }
305*7304104dSAndroid Build Coastguard Worker 
306*7304104dSAndroid Build Coastguard Worker   if (unlikely (close (fd) != 0))
307*7304104dSAndroid Build Coastguard Worker     error_exit (errno, _("while closing '%s'"), fname);
308*7304104dSAndroid Build Coastguard Worker 
309*7304104dSAndroid Build Coastguard Worker   error (0, 0, _("%s: file format not recognized"), fname);
310*7304104dSAndroid Build Coastguard Worker 
311*7304104dSAndroid Build Coastguard Worker   return 1;
312*7304104dSAndroid Build Coastguard Worker }
313*7304104dSAndroid Build Coastguard Worker 
314*7304104dSAndroid Build Coastguard Worker 
315*7304104dSAndroid Build Coastguard Worker /* Print the BSD-style header.  This is done exactly once.  */
316*7304104dSAndroid Build Coastguard Worker static void
print_header(Elf * elf)317*7304104dSAndroid Build Coastguard Worker print_header (Elf *elf)
318*7304104dSAndroid Build Coastguard Worker {
319*7304104dSAndroid Build Coastguard Worker   static int done;
320*7304104dSAndroid Build Coastguard Worker 
321*7304104dSAndroid Build Coastguard Worker   if (! done)
322*7304104dSAndroid Build Coastguard Worker     {
323*7304104dSAndroid Build Coastguard Worker       int ddigits = length_map[gelf_getclass (elf) - 1][radix_decimal];
324*7304104dSAndroid Build Coastguard Worker       int xdigits = length_map[gelf_getclass (elf) - 1][radix_hex];
325*7304104dSAndroid Build Coastguard Worker 
326*7304104dSAndroid Build Coastguard Worker       printf ("%*s %*s %*s %*s %*s %s\n",
327*7304104dSAndroid Build Coastguard Worker 	      ddigits - 2, sgettext ("bsd|text"),
328*7304104dSAndroid Build Coastguard Worker 	      ddigits - 2, sgettext ("bsd|data"),
329*7304104dSAndroid Build Coastguard Worker 	      ddigits - 2, sgettext ("bsd|bss"),
330*7304104dSAndroid Build Coastguard Worker 	      ddigits - 2, sgettext ("bsd|dec"),
331*7304104dSAndroid Build Coastguard Worker 	      xdigits - 2, sgettext ("bsd|hex"),
332*7304104dSAndroid Build Coastguard Worker 	      sgettext ("bsd|filename"));
333*7304104dSAndroid Build Coastguard Worker 
334*7304104dSAndroid Build Coastguard Worker       done = 1;
335*7304104dSAndroid Build Coastguard Worker     }
336*7304104dSAndroid Build Coastguard Worker }
337*7304104dSAndroid Build Coastguard Worker 
338*7304104dSAndroid Build Coastguard Worker 
339*7304104dSAndroid Build Coastguard Worker static int
handle_ar(int fd,Elf * elf,const char * prefix,const char * fname)340*7304104dSAndroid Build Coastguard Worker handle_ar (int fd, Elf *elf, const char *prefix, const char *fname)
341*7304104dSAndroid Build Coastguard Worker {
342*7304104dSAndroid Build Coastguard Worker   size_t prefix_len = prefix == NULL ? 0 : strlen (prefix);
343*7304104dSAndroid Build Coastguard Worker   size_t fname_len = strlen (fname) + 1;
344*7304104dSAndroid Build Coastguard Worker   char new_prefix[prefix_len + 1 + fname_len];
345*7304104dSAndroid Build Coastguard Worker   char *cp = new_prefix;
346*7304104dSAndroid Build Coastguard Worker 
347*7304104dSAndroid Build Coastguard Worker   /* Create the full name of the file.  */
348*7304104dSAndroid Build Coastguard Worker   if (prefix != NULL)
349*7304104dSAndroid Build Coastguard Worker     {
350*7304104dSAndroid Build Coastguard Worker       cp = mempcpy (cp, prefix, prefix_len);
351*7304104dSAndroid Build Coastguard Worker       *cp++ = ':';
352*7304104dSAndroid Build Coastguard Worker     }
353*7304104dSAndroid Build Coastguard Worker   memcpy (cp, fname, fname_len);
354*7304104dSAndroid Build Coastguard Worker 
355*7304104dSAndroid Build Coastguard Worker   /* Process all the files contained in the archive.  */
356*7304104dSAndroid Build Coastguard Worker   int result = 0;
357*7304104dSAndroid Build Coastguard Worker   Elf *subelf;
358*7304104dSAndroid Build Coastguard Worker   Elf_Cmd cmd = ELF_C_READ_MMAP;
359*7304104dSAndroid Build Coastguard Worker   while ((subelf = elf_begin (fd, cmd, elf)) != NULL)
360*7304104dSAndroid Build Coastguard Worker     {
361*7304104dSAndroid Build Coastguard Worker       /* The the header for this element.  */
362*7304104dSAndroid Build Coastguard Worker       Elf_Arhdr *arhdr = elf_getarhdr (subelf);
363*7304104dSAndroid Build Coastguard Worker 
364*7304104dSAndroid Build Coastguard Worker       if (elf_kind (subelf) == ELF_K_ELF)
365*7304104dSAndroid Build Coastguard Worker 	handle_elf (subelf, new_prefix, arhdr->ar_name);
366*7304104dSAndroid Build Coastguard Worker       else if (likely (elf_kind (subelf) == ELF_K_AR))
367*7304104dSAndroid Build Coastguard Worker 	result |= handle_ar (fd, subelf, new_prefix, arhdr->ar_name);
368*7304104dSAndroid Build Coastguard Worker       /* else signal error??? */
369*7304104dSAndroid Build Coastguard Worker 
370*7304104dSAndroid Build Coastguard Worker       /* Get next archive element.  */
371*7304104dSAndroid Build Coastguard Worker       cmd = elf_next (subelf);
372*7304104dSAndroid Build Coastguard Worker       if (unlikely (elf_end (subelf) != 0))
373*7304104dSAndroid Build Coastguard Worker 	INTERNAL_ERROR (fname);
374*7304104dSAndroid Build Coastguard Worker     }
375*7304104dSAndroid Build Coastguard Worker 
376*7304104dSAndroid Build Coastguard Worker   /* Only close ELF handle if this was a "top level" ar file.  */
377*7304104dSAndroid Build Coastguard Worker   if (prefix == NULL)
378*7304104dSAndroid Build Coastguard Worker     if (unlikely (elf_end (elf) != 0))
379*7304104dSAndroid Build Coastguard Worker       INTERNAL_ERROR (fname);
380*7304104dSAndroid Build Coastguard Worker 
381*7304104dSAndroid Build Coastguard Worker   return result;
382*7304104dSAndroid Build Coastguard Worker }
383*7304104dSAndroid Build Coastguard Worker 
384*7304104dSAndroid Build Coastguard Worker 
385*7304104dSAndroid Build Coastguard Worker /* Show sizes in SysV format.  */
386*7304104dSAndroid Build Coastguard Worker static void
show_sysv(Elf * elf,const char * prefix,const char * fname,const char * fullname)387*7304104dSAndroid Build Coastguard Worker show_sysv (Elf *elf, const char *prefix, const char *fname,
388*7304104dSAndroid Build Coastguard Worker 	   const char *fullname)
389*7304104dSAndroid Build Coastguard Worker {
390*7304104dSAndroid Build Coastguard Worker   int maxlen = 10;
391*7304104dSAndroid Build Coastguard Worker   const int digits = length_map[gelf_getclass (elf) - 1][radix];
392*7304104dSAndroid Build Coastguard Worker 
393*7304104dSAndroid Build Coastguard Worker   /* Get the section header string table index.  */
394*7304104dSAndroid Build Coastguard Worker   size_t shstrndx;
395*7304104dSAndroid Build Coastguard Worker   if (unlikely (elf_getshdrstrndx (elf, &shstrndx) < 0))
396*7304104dSAndroid Build Coastguard Worker     error_exit (0, _("cannot get section header string table index"));
397*7304104dSAndroid Build Coastguard Worker 
398*7304104dSAndroid Build Coastguard Worker   /* First round over the sections: determine the longest section name.  */
399*7304104dSAndroid Build Coastguard Worker   Elf_Scn *scn = NULL;
400*7304104dSAndroid Build Coastguard Worker   while ((scn = elf_nextscn (elf, scn)) != NULL)
401*7304104dSAndroid Build Coastguard Worker     {
402*7304104dSAndroid Build Coastguard Worker       GElf_Shdr shdr_mem;
403*7304104dSAndroid Build Coastguard Worker       GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
404*7304104dSAndroid Build Coastguard Worker 
405*7304104dSAndroid Build Coastguard Worker       if (shdr == NULL)
406*7304104dSAndroid Build Coastguard Worker 	INTERNAL_ERROR (fullname);
407*7304104dSAndroid Build Coastguard Worker 
408*7304104dSAndroid Build Coastguard Worker       /* Ignore all sections which are not used at runtime.  */
409*7304104dSAndroid Build Coastguard Worker       const char *name = elf_strptr (elf, shstrndx, shdr->sh_name);
410*7304104dSAndroid Build Coastguard Worker       if (name != NULL && (shdr->sh_flags & SHF_ALLOC) != 0)
411*7304104dSAndroid Build Coastguard Worker 	maxlen = MAX (maxlen, (int) strlen (name));
412*7304104dSAndroid Build Coastguard Worker     }
413*7304104dSAndroid Build Coastguard Worker 
414*7304104dSAndroid Build Coastguard Worker   fputs_unlocked (fname, stdout);
415*7304104dSAndroid Build Coastguard Worker   if (prefix != NULL)
416*7304104dSAndroid Build Coastguard Worker     printf (_(" (ex %s)"), prefix);
417*7304104dSAndroid Build Coastguard Worker   printf (":\n%-*s %*s %*s\n",
418*7304104dSAndroid Build Coastguard Worker 	  maxlen, sgettext ("sysv|section"),
419*7304104dSAndroid Build Coastguard Worker 	  digits - 2, sgettext ("sysv|size"),
420*7304104dSAndroid Build Coastguard Worker 	  digits, sgettext ("sysv|addr"));
421*7304104dSAndroid Build Coastguard Worker 
422*7304104dSAndroid Build Coastguard Worker   /* Iterate over all sections.  */
423*7304104dSAndroid Build Coastguard Worker   GElf_Off total = 0;
424*7304104dSAndroid Build Coastguard Worker   while ((scn = elf_nextscn (elf, scn)) != NULL)
425*7304104dSAndroid Build Coastguard Worker     {
426*7304104dSAndroid Build Coastguard Worker       GElf_Shdr shdr_mem;
427*7304104dSAndroid Build Coastguard Worker       GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
428*7304104dSAndroid Build Coastguard Worker 
429*7304104dSAndroid Build Coastguard Worker       if (shdr == NULL)
430*7304104dSAndroid Build Coastguard Worker 	INTERNAL_ERROR (fullname);
431*7304104dSAndroid Build Coastguard Worker 
432*7304104dSAndroid Build Coastguard Worker       /* Ignore all sections which are not used at runtime.  */
433*7304104dSAndroid Build Coastguard Worker       if ((shdr->sh_flags & SHF_ALLOC) != 0)
434*7304104dSAndroid Build Coastguard Worker 	{
435*7304104dSAndroid Build Coastguard Worker 	  printf ((radix == radix_hex
436*7304104dSAndroid Build Coastguard Worker 		   ? "%-*s %*" PRIx64 " %*" PRIx64 "\n"
437*7304104dSAndroid Build Coastguard Worker 		   : (radix == radix_decimal
438*7304104dSAndroid Build Coastguard Worker 		      ? "%-*s %*" PRId64 " %*" PRId64 "\n"
439*7304104dSAndroid Build Coastguard Worker 		      : "%-*s %*" PRIo64 " %*" PRIo64 "\n")),
440*7304104dSAndroid Build Coastguard Worker 		  maxlen, elf_strptr (elf, shstrndx, shdr->sh_name),
441*7304104dSAndroid Build Coastguard Worker 		  digits - 2, shdr->sh_size,
442*7304104dSAndroid Build Coastguard Worker 		  digits, shdr->sh_addr);
443*7304104dSAndroid Build Coastguard Worker 
444*7304104dSAndroid Build Coastguard Worker 	  total += shdr->sh_size;
445*7304104dSAndroid Build Coastguard Worker 	}
446*7304104dSAndroid Build Coastguard Worker     }
447*7304104dSAndroid Build Coastguard Worker 
448*7304104dSAndroid Build Coastguard Worker   if (radix == radix_hex)
449*7304104dSAndroid Build Coastguard Worker     printf ("%-*s %*" PRIx64 "\n\n\n", maxlen, sgettext ("sysv|Total"),
450*7304104dSAndroid Build Coastguard Worker 	    digits - 2, total);
451*7304104dSAndroid Build Coastguard Worker   else if (radix == radix_decimal)
452*7304104dSAndroid Build Coastguard Worker     printf ("%-*s %*" PRId64 "\n\n\n", maxlen, sgettext ("sysv|Total"),
453*7304104dSAndroid Build Coastguard Worker 	    digits - 2, total);
454*7304104dSAndroid Build Coastguard Worker   else
455*7304104dSAndroid Build Coastguard Worker     printf ("%-*s %*" PRIo64 "\n\n\n", maxlen, sgettext ("sysv|Total"),
456*7304104dSAndroid Build Coastguard Worker 	    digits - 2, total);
457*7304104dSAndroid Build Coastguard Worker }
458*7304104dSAndroid Build Coastguard Worker 
459*7304104dSAndroid Build Coastguard Worker 
460*7304104dSAndroid Build Coastguard Worker /* Show sizes in SysV format in one line.  */
461*7304104dSAndroid Build Coastguard Worker static void
show_sysv_one_line(Elf * elf)462*7304104dSAndroid Build Coastguard Worker show_sysv_one_line (Elf *elf)
463*7304104dSAndroid Build Coastguard Worker {
464*7304104dSAndroid Build Coastguard Worker   /* Get the section header string table index.  */
465*7304104dSAndroid Build Coastguard Worker   size_t shstrndx;
466*7304104dSAndroid Build Coastguard Worker   if (unlikely (elf_getshdrstrndx (elf, &shstrndx) < 0))
467*7304104dSAndroid Build Coastguard Worker     error_exit (0, _("cannot get section header string table index"));
468*7304104dSAndroid Build Coastguard Worker 
469*7304104dSAndroid Build Coastguard Worker   /* Iterate over all sections.  */
470*7304104dSAndroid Build Coastguard Worker   GElf_Off total = 0;
471*7304104dSAndroid Build Coastguard Worker   bool first = true;
472*7304104dSAndroid Build Coastguard Worker   Elf_Scn *scn = NULL;
473*7304104dSAndroid Build Coastguard Worker   while ((scn = elf_nextscn (elf, scn)) != NULL)
474*7304104dSAndroid Build Coastguard Worker     {
475*7304104dSAndroid Build Coastguard Worker       GElf_Shdr shdr_mem;
476*7304104dSAndroid Build Coastguard Worker       GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
477*7304104dSAndroid Build Coastguard Worker 
478*7304104dSAndroid Build Coastguard Worker       if (unlikely (shdr == NULL))
479*7304104dSAndroid Build Coastguard Worker 	error_exit (0, _("cannot get section header"));
480*7304104dSAndroid Build Coastguard Worker 
481*7304104dSAndroid Build Coastguard Worker       /* Ignore all sections which are not used at runtime.  */
482*7304104dSAndroid Build Coastguard Worker       if ((shdr->sh_flags & SHF_ALLOC) == 0)
483*7304104dSAndroid Build Coastguard Worker 	continue;
484*7304104dSAndroid Build Coastguard Worker 
485*7304104dSAndroid Build Coastguard Worker       if (! first)
486*7304104dSAndroid Build Coastguard Worker 	fputs_unlocked (" + ", stdout);
487*7304104dSAndroid Build Coastguard Worker       first = false;
488*7304104dSAndroid Build Coastguard Worker 
489*7304104dSAndroid Build Coastguard Worker       printf ((radix == radix_hex ? "%" PRIx64 "(%s)"
490*7304104dSAndroid Build Coastguard Worker 	       : (radix == radix_decimal ? "%" PRId64 "(%s)"
491*7304104dSAndroid Build Coastguard Worker 		  : "%" PRIo64 "(%s)")),
492*7304104dSAndroid Build Coastguard Worker 	      shdr->sh_size, elf_strptr (elf, shstrndx, shdr->sh_name));
493*7304104dSAndroid Build Coastguard Worker 
494*7304104dSAndroid Build Coastguard Worker       total += shdr->sh_size;
495*7304104dSAndroid Build Coastguard Worker     }
496*7304104dSAndroid Build Coastguard Worker 
497*7304104dSAndroid Build Coastguard Worker   if (radix == radix_hex)
498*7304104dSAndroid Build Coastguard Worker     printf (" = %#" PRIx64 "\n", total);
499*7304104dSAndroid Build Coastguard Worker   else if (radix == radix_decimal)
500*7304104dSAndroid Build Coastguard Worker     printf (" = %" PRId64 "\n", total);
501*7304104dSAndroid Build Coastguard Worker   else
502*7304104dSAndroid Build Coastguard Worker     printf (" = %" PRIo64 "\n", total);
503*7304104dSAndroid Build Coastguard Worker }
504*7304104dSAndroid Build Coastguard Worker 
505*7304104dSAndroid Build Coastguard Worker 
506*7304104dSAndroid Build Coastguard Worker /* Variables to add up the sizes of all files.  */
507*7304104dSAndroid Build Coastguard Worker static uintmax_t total_textsize;
508*7304104dSAndroid Build Coastguard Worker static uintmax_t total_datasize;
509*7304104dSAndroid Build Coastguard Worker static uintmax_t total_bsssize;
510*7304104dSAndroid Build Coastguard Worker 
511*7304104dSAndroid Build Coastguard Worker 
512*7304104dSAndroid Build Coastguard Worker /* Show sizes in BSD format.  */
513*7304104dSAndroid Build Coastguard Worker static void
show_bsd(Elf * elf,const char * prefix,const char * fname,const char * fullname)514*7304104dSAndroid Build Coastguard Worker show_bsd (Elf *elf, const char *prefix, const char *fname,
515*7304104dSAndroid Build Coastguard Worker 	  const char *fullname)
516*7304104dSAndroid Build Coastguard Worker {
517*7304104dSAndroid Build Coastguard Worker   GElf_Off textsize = 0;
518*7304104dSAndroid Build Coastguard Worker   GElf_Off datasize = 0;
519*7304104dSAndroid Build Coastguard Worker   GElf_Off bsssize = 0;
520*7304104dSAndroid Build Coastguard Worker   const int ddigits = length_map[gelf_getclass (elf) - 1][radix_decimal];
521*7304104dSAndroid Build Coastguard Worker   const int xdigits = length_map[gelf_getclass (elf) - 1][radix_hex];
522*7304104dSAndroid Build Coastguard Worker 
523*7304104dSAndroid Build Coastguard Worker   /* Iterate over all sections.  */
524*7304104dSAndroid Build Coastguard Worker   Elf_Scn *scn = NULL;
525*7304104dSAndroid Build Coastguard Worker   while ((scn = elf_nextscn (elf, scn)) != NULL)
526*7304104dSAndroid Build Coastguard Worker     {
527*7304104dSAndroid Build Coastguard Worker       GElf_Shdr shdr_mem;
528*7304104dSAndroid Build Coastguard Worker       GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
529*7304104dSAndroid Build Coastguard Worker 
530*7304104dSAndroid Build Coastguard Worker       if (shdr == NULL)
531*7304104dSAndroid Build Coastguard Worker 	INTERNAL_ERROR (fullname);
532*7304104dSAndroid Build Coastguard Worker 
533*7304104dSAndroid Build Coastguard Worker       /* Ignore all sections which are not marked as loaded.  */
534*7304104dSAndroid Build Coastguard Worker       if ((shdr->sh_flags & SHF_ALLOC) == 0)
535*7304104dSAndroid Build Coastguard Worker 	continue;
536*7304104dSAndroid Build Coastguard Worker 
537*7304104dSAndroid Build Coastguard Worker       if ((shdr->sh_flags & SHF_WRITE) == 0)
538*7304104dSAndroid Build Coastguard Worker 	textsize += shdr->sh_size;
539*7304104dSAndroid Build Coastguard Worker       else if (shdr->sh_type == SHT_NOBITS)
540*7304104dSAndroid Build Coastguard Worker 	bsssize += shdr->sh_size;
541*7304104dSAndroid Build Coastguard Worker       else
542*7304104dSAndroid Build Coastguard Worker 	datasize += shdr->sh_size;
543*7304104dSAndroid Build Coastguard Worker     }
544*7304104dSAndroid Build Coastguard Worker 
545*7304104dSAndroid Build Coastguard Worker   printf (radix == radix_decimal
546*7304104dSAndroid Build Coastguard Worker           ? "%*" PRId64 " %*" PRId64 " %*" PRId64 " %*" PRId64 " %*" PRIx64 " %s"
547*7304104dSAndroid Build Coastguard Worker 	  : radix == radix_hex
548*7304104dSAndroid Build Coastguard Worker 	  ? "%#*" PRIx64 " %#*" PRIx64 " %#*" PRIx64 " %*" PRId64 " %*" PRIx64 " %s"
549*7304104dSAndroid Build Coastguard Worker 	  : "%#*" PRIo64 " %#*" PRIo64 " %#*" PRIo64 " %*" PRId64 " %*" PRIx64 " %s",
550*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, textsize,
551*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, datasize,
552*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, bsssize,
553*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, textsize + datasize + bsssize,
554*7304104dSAndroid Build Coastguard Worker 	  xdigits - 2, textsize + datasize + bsssize,
555*7304104dSAndroid Build Coastguard Worker 	  fname);
556*7304104dSAndroid Build Coastguard Worker   if (prefix != NULL)
557*7304104dSAndroid Build Coastguard Worker     printf (_(" (ex %s)"), prefix);
558*7304104dSAndroid Build Coastguard Worker   fputs_unlocked ("\n", stdout);
559*7304104dSAndroid Build Coastguard Worker 
560*7304104dSAndroid Build Coastguard Worker   total_textsize += textsize;
561*7304104dSAndroid Build Coastguard Worker   total_datasize += datasize;
562*7304104dSAndroid Build Coastguard Worker   total_bsssize += bsssize;
563*7304104dSAndroid Build Coastguard Worker 
564*7304104dSAndroid Build Coastguard Worker   totals_class = MAX (totals_class, gelf_getclass (elf));
565*7304104dSAndroid Build Coastguard Worker }
566*7304104dSAndroid Build Coastguard Worker 
567*7304104dSAndroid Build Coastguard Worker 
568*7304104dSAndroid Build Coastguard Worker /* Show total size.  */
569*7304104dSAndroid Build Coastguard Worker static void
show_bsd_totals(void)570*7304104dSAndroid Build Coastguard Worker show_bsd_totals (void)
571*7304104dSAndroid Build Coastguard Worker {
572*7304104dSAndroid Build Coastguard Worker   int ddigits = length_map[totals_class - 1][radix_decimal];
573*7304104dSAndroid Build Coastguard Worker   int xdigits = length_map[totals_class - 1][radix_hex];
574*7304104dSAndroid Build Coastguard Worker 
575*7304104dSAndroid Build Coastguard Worker   printf ("%*" PRIuMAX " %*" PRIuMAX " %*" PRIuMAX " %*" PRIuMAX " %*"
576*7304104dSAndroid Build Coastguard Worker 	  PRIxMAX " %s",
577*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, total_textsize,
578*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, total_datasize,
579*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, total_bsssize,
580*7304104dSAndroid Build Coastguard Worker 	  ddigits - 2, total_textsize + total_datasize + total_bsssize,
581*7304104dSAndroid Build Coastguard Worker 	  xdigits - 2, total_textsize + total_datasize + total_bsssize,
582*7304104dSAndroid Build Coastguard Worker 	  _("(TOTALS)\n"));
583*7304104dSAndroid Build Coastguard Worker }
584*7304104dSAndroid Build Coastguard Worker 
585*7304104dSAndroid Build Coastguard Worker 
586*7304104dSAndroid Build Coastguard Worker /* Show size and permission of loadable segments.  */
587*7304104dSAndroid Build Coastguard Worker static void
show_segments(Elf * elf,const char * fullname)588*7304104dSAndroid Build Coastguard Worker show_segments (Elf *elf, const char *fullname)
589*7304104dSAndroid Build Coastguard Worker {
590*7304104dSAndroid Build Coastguard Worker   size_t phnum;
591*7304104dSAndroid Build Coastguard Worker   if (elf_getphdrnum (elf, &phnum) != 0)
592*7304104dSAndroid Build Coastguard Worker     INTERNAL_ERROR (fullname);
593*7304104dSAndroid Build Coastguard Worker 
594*7304104dSAndroid Build Coastguard Worker   GElf_Off total = 0;
595*7304104dSAndroid Build Coastguard Worker   bool first = true;
596*7304104dSAndroid Build Coastguard Worker   for (size_t cnt = 0; cnt < phnum; ++cnt)
597*7304104dSAndroid Build Coastguard Worker     {
598*7304104dSAndroid Build Coastguard Worker       GElf_Phdr phdr_mem;
599*7304104dSAndroid Build Coastguard Worker       GElf_Phdr *phdr;
600*7304104dSAndroid Build Coastguard Worker 
601*7304104dSAndroid Build Coastguard Worker       phdr = gelf_getphdr (elf, cnt, &phdr_mem);
602*7304104dSAndroid Build Coastguard Worker       if (phdr == NULL)
603*7304104dSAndroid Build Coastguard Worker 	INTERNAL_ERROR (fullname);
604*7304104dSAndroid Build Coastguard Worker 
605*7304104dSAndroid Build Coastguard Worker       if (phdr->p_type != PT_LOAD)
606*7304104dSAndroid Build Coastguard Worker 	/* Only load segments.  */
607*7304104dSAndroid Build Coastguard Worker 	continue;
608*7304104dSAndroid Build Coastguard Worker 
609*7304104dSAndroid Build Coastguard Worker       if (! first)
610*7304104dSAndroid Build Coastguard Worker 	fputs_unlocked (" + ", stdout);
611*7304104dSAndroid Build Coastguard Worker       first = false;
612*7304104dSAndroid Build Coastguard Worker 
613*7304104dSAndroid Build Coastguard Worker       printf (radix == radix_hex ? "%" PRIx64 "(%c%c%c)"
614*7304104dSAndroid Build Coastguard Worker 	      : (radix == radix_decimal ? "%" PRId64 "(%c%c%c)"
615*7304104dSAndroid Build Coastguard Worker 		 : "%" PRIo64 "(%c%c%c)"),
616*7304104dSAndroid Build Coastguard Worker 	      phdr->p_memsz,
617*7304104dSAndroid Build Coastguard Worker 	      (phdr->p_flags & PF_R) == 0 ? '-' : 'r',
618*7304104dSAndroid Build Coastguard Worker 	      (phdr->p_flags & PF_W) == 0 ? '-' : 'w',
619*7304104dSAndroid Build Coastguard Worker 	      (phdr->p_flags & PF_X) == 0 ? '-' : 'x');
620*7304104dSAndroid Build Coastguard Worker 
621*7304104dSAndroid Build Coastguard Worker       total += phdr->p_memsz;
622*7304104dSAndroid Build Coastguard Worker     }
623*7304104dSAndroid Build Coastguard Worker 
624*7304104dSAndroid Build Coastguard Worker   if (radix == radix_hex)
625*7304104dSAndroid Build Coastguard Worker     printf (" = %#" PRIx64 "\n", total);
626*7304104dSAndroid Build Coastguard Worker   else if (radix == radix_decimal)
627*7304104dSAndroid Build Coastguard Worker     printf (" = %" PRId64 "\n", total);
628*7304104dSAndroid Build Coastguard Worker   else
629*7304104dSAndroid Build Coastguard Worker     printf (" = %" PRIo64 "\n", total);
630*7304104dSAndroid Build Coastguard Worker }
631*7304104dSAndroid Build Coastguard Worker 
632*7304104dSAndroid Build Coastguard Worker 
633*7304104dSAndroid Build Coastguard Worker static void
handle_elf(Elf * elf,const char * prefix,const char * fname)634*7304104dSAndroid Build Coastguard Worker handle_elf (Elf *elf, const char *prefix, const char *fname)
635*7304104dSAndroid Build Coastguard Worker {
636*7304104dSAndroid Build Coastguard Worker   size_t prefix_len = prefix == NULL ? 0 : strlen (prefix);
637*7304104dSAndroid Build Coastguard Worker   size_t fname_len = strlen (fname) + 1;
638*7304104dSAndroid Build Coastguard Worker   char fullname[prefix_len + 1 + fname_len];
639*7304104dSAndroid Build Coastguard Worker   char *cp = fullname;
640*7304104dSAndroid Build Coastguard Worker 
641*7304104dSAndroid Build Coastguard Worker   /* Create the full name of the file.  */
642*7304104dSAndroid Build Coastguard Worker   if (prefix != NULL)
643*7304104dSAndroid Build Coastguard Worker     {
644*7304104dSAndroid Build Coastguard Worker       cp = mempcpy (cp, prefix, prefix_len);
645*7304104dSAndroid Build Coastguard Worker       *cp++ = ':';
646*7304104dSAndroid Build Coastguard Worker     }
647*7304104dSAndroid Build Coastguard Worker   memcpy (cp, fname, fname_len);
648*7304104dSAndroid Build Coastguard Worker 
649*7304104dSAndroid Build Coastguard Worker   if (format == format_sysv)
650*7304104dSAndroid Build Coastguard Worker     show_sysv (elf, prefix, fname, fullname);
651*7304104dSAndroid Build Coastguard Worker   else if (format == format_sysv_one_line)
652*7304104dSAndroid Build Coastguard Worker     show_sysv_one_line (elf);
653*7304104dSAndroid Build Coastguard Worker   else if (format == format_segments)
654*7304104dSAndroid Build Coastguard Worker     show_segments (elf, fullname);
655*7304104dSAndroid Build Coastguard Worker   else
656*7304104dSAndroid Build Coastguard Worker     {
657*7304104dSAndroid Build Coastguard Worker       print_header (elf);
658*7304104dSAndroid Build Coastguard Worker 
659*7304104dSAndroid Build Coastguard Worker       show_bsd (elf, prefix, fname, fullname);
660*7304104dSAndroid Build Coastguard Worker     }
661*7304104dSAndroid Build Coastguard Worker }
662*7304104dSAndroid Build Coastguard Worker 
663*7304104dSAndroid Build Coastguard Worker 
664*7304104dSAndroid Build Coastguard Worker #include "debugpred.h"
665