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