/* Copyright (C) 1998, 1999, 2000, 2002 Red Hat, Inc. Written by Ulrich Drepper <drepper@redhat.com>, 1998. This program is Open Source software; you can redistribute it and/or modify it under the terms of the Open Software License version 1.0 as published by the Open Source Initiative. You should have received a copy of the Open Software License along with this program; if not, you may obtain a copy of the Open Software License version 1.0 from http://www.opensource.org/licenses/osl.php or by writing the Open Source Initiative c/o Lawrence Rosen, Esq., 3001 King Ranch Road, Ukiah, CA 95482. */ #include <config.h> #include <error.h> #include <fcntl.h> #include <gelf.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <time.h> #include <unistd.h> static const char *machines[] = { #define MACHINE(name) [name] = #name MACHINE (EM_NONE), MACHINE (EM_M32), MACHINE (EM_SPARC), MACHINE (EM_386), MACHINE (EM_68K), MACHINE (EM_88K), MACHINE (EM_860), MACHINE (EM_MIPS), MACHINE (EM_MIPS_RS3_LE), MACHINE (EM_PARISC), MACHINE (EM_VPP500), MACHINE (EM_SPARC32PLUS), MACHINE (EM_960), MACHINE (EM_PPC), MACHINE (EM_PPC64), MACHINE (EM_V800), MACHINE (EM_FR20), MACHINE (EM_RH32), MACHINE (EM_RCE), MACHINE (EM_ARM), MACHINE (EM_FAKE_ALPHA), MACHINE (EM_SH), MACHINE (EM_SPARCV9), MACHINE (EM_TRICORE), MACHINE (EM_ARC), MACHINE (EM_H8_300), MACHINE (EM_H8_300H), MACHINE (EM_H8S), MACHINE (EM_H8_500), MACHINE (EM_IA_64), MACHINE (EM_MIPS_X), MACHINE (EM_COLDFIRE), MACHINE (EM_68HC12), MACHINE (EM_MMA), MACHINE (EM_PCP), MACHINE (EM_NCPU), MACHINE (EM_NDR1), MACHINE (EM_STARCORE), MACHINE (EM_ME16), MACHINE (EM_ST100), MACHINE (EM_TINYJ), MACHINE (EM_FX66), MACHINE (EM_ST9PLUS), MACHINE (EM_ST7), MACHINE (EM_68HC16), MACHINE (EM_68HC11), MACHINE (EM_68HC08), MACHINE (EM_68HC05), MACHINE (EM_SVX), MACHINE (EM_ST19), MACHINE (EM_VAX) }; int main (int argc, char *argv[]) { int fd; Elf *elf; Elf_Cmd cmd; size_t n; int arg = 1; int verbose = 0; /* Recognize optional verbosity flag. */ if (arg < argc && strcmp (argv[arg], "-v") == 0) { verbose = 1; ++arg; } /* Any more arguments available. */ if (arg >= argc) error (EXIT_FAILURE, 0, "No input file given"); /* Open the input file. */ fd = open (argv[arg], O_RDONLY); if (fd == -1) { perror ("cannot open input file"); exit (1); } /* Set the ELF version we are using here. */ if (elf_version (EV_CURRENT) == EV_NONE) { puts ("ELF library too old"); exit (1); } /* Start reading the file. */ cmd = ELF_C_READ; elf = elf_begin (fd, cmd, NULL); if (elf == NULL) { printf ("elf_begin: %s\n", elf_errmsg (-1)); exit (1); } /* If it is no archive punt. */ if (elf_kind (elf) != ELF_K_AR) { printf ("%s is not an archive\n", argv[1]); exit (1); } if (verbose) { /* The verbose variant. We print a lot of information. */ Elf *subelf; char buf[100]; time_t t; /* Get the elements of the archive one after the other. */ while ((subelf = elf_begin (fd, cmd, elf)) != NULL) { /* The the header for this element. */ Elf_Arhdr *arhdr = elf_getarhdr (subelf); if (arhdr == NULL) { printf ("cannot get arhdr: %s\n", elf_errmsg (-1)); break; } switch (elf_kind (subelf)) { case ELF_K_ELF: fputs ("ELF file:\n", stdout); break; case ELF_K_AR: fputs ("archive:\n", stdout); break; default: fputs ("unknown file:\n", stdout); break; } /* Print general information. */ t = arhdr->ar_date; strftime (buf, sizeof buf, "%Y-%m-%dT%H:%M:%S%z", gmtime (&t)); printf (" name : \"%s\"\n" " time : %s\n" " uid : %ld\n" " gid : %ld\n" " mode : %o\n" " size : %ld\n" " rawname : \"%s\"\n", arhdr->ar_name, buf, (long int) arhdr->ar_uid, (long int) arhdr->ar_gid, arhdr->ar_mode, (long int) arhdr->ar_size, arhdr->ar_rawname); /* For ELF files we can provide some more information. */ if (elf_kind (subelf) == ELF_K_ELF) { GElf_Ehdr ehdr; /* Get the ELF header. */ if (gelf_getehdr (subelf, &ehdr) == NULL) printf (" *** cannot get ELF header: %s\n", elf_errmsg (-1)); else { printf (" binary class : %s\n", ehdr.e_ident[EI_CLASS] == ELFCLASS32 ? "ELFCLASS32" : "ELFCLASS64"); printf (" data encoding: %s\n", ehdr.e_ident[EI_DATA] == ELFDATA2LSB ? "ELFDATA2LSB" : "ELFDATA2MSB"); printf (" binary type : %s\n", ehdr.e_type == ET_REL ? "relocatable" : (ehdr.e_type == ET_EXEC ? "executable" : (ehdr.e_type == ET_DYN ? "dynamic" : "core file"))); printf (" machine : %s\n", (ehdr.e_machine >= (sizeof (machines) / sizeof (machines[0])) || machines[ehdr.e_machine] == NULL) ? "???" : machines[ehdr.e_machine]); } } /* Get next archive element. */ cmd = elf_next (subelf); if (elf_end (subelf) != 0) printf ("error while freeing sub-ELF descriptor: %s\n", elf_errmsg (-1)); } } else { /* The simple version. Only print a bit of information. */ Elf_Arsym *arsym = elf_getarsym (elf, &n); if (n == 0) printf ("no symbol table in archive: %s\n", elf_errmsg (-1)); else { --n; while (n-- > 0) printf ("name = \"%s\", offset = %ld, hash = %lx\n", arsym[n].as_name, (long int) arsym[n].as_off, arsym[n].as_hash); } } /* Free the ELF handle. */ if (elf_end (elf) != 0) printf ("error while freeing ELF descriptor: %s\n", elf_errmsg (-1)); /* Close the underlying file. */ close (fd); return 0; }