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