Date: Tue, 4 Mar 2008 08:39:32 +0100 From: Ed Schouten <ed@80386.nl> To: Joseph Koshy <joseph.koshy@gmail.com> Cc: FreeBSD Current <freebsd-current@freebsd.org> Subject: Re: Yet another BSD licensed replacement: size(1) Message-ID: <20080304073932.GJ80576@hoeg.nl> In-Reply-To: <84dead720803032233m6881e888m6ff69c464a4a3d17@mail.gmail.com> References: <20080303200219.GF80576@hoeg.nl> <84dead720803032233m6881e888m6ff69c464a4a3d17@mail.gmail.com>
next in thread | previous in thread | raw e-mail | index | archive | help
--yaap9KN+GmBP785v Content-Type: multipart/mixed; boundary="jtcAeju3WzRmRF+o" Content-Disposition: inline --jtcAeju3WzRmRF+o Content-Type: text/plain; charset=us-ascii Content-Disposition: inline Content-Transfer-Encoding: quoted-printable * Joseph Koshy <joseph.koshy@gmail.com> wrote: > How does yours compare with Sam Arun Raj's implementation of size(1) in > Perforce? IIRC it was fairly complete. > > See: http://wiki.freebsd.org/ElfToolChain > and: //depot/user/samarunraj/elftoolchain/src/usr.bin/size/ Unfortunately, the wiki is down, but I just did a checkout of Sam's work. Too bad I didn't know there was already an implementation already, but it was fun anyway. ;-) As far as I can see, his version is a lot more complex than mine, but has the advantage that it supports core dumps and archives as well. My version is more compatible with the GNU one, in case someone is using this in shellscripts (has the -B switch, output is exactly the same, etc). Even though I couldn't build Sam's version in one go (lot of warnings), I think we'd better stick to Sam's implementation. I've attached an updated version of my work to this email, in case someone finds something interesting in there. What I can't understand, is why the System V format is called like that. size(1) on Solaris is *completely* different. --=20 Ed Schouten <ed@fxq.nl> WWW: http://g-rave.nl/ --jtcAeju3WzRmRF+o Content-Type: text/plain; charset=us-ascii Content-Disposition: attachment; filename="size.txt" Content-Transfer-Encoding: quoted-printable # This is a shell archive. Save it in a file, remove anything before # this line, and then unpack it by entering "sh file". Note, it may # create directories; files and directories will be owned by you and # have default permissions. # # This archive contains: # # Makefile # size.1 # size.c # echo x - Makefile sed 's/^X//' >Makefile << 'END-of-Makefile' XPROG=3D size XLDADD=3D -lelf XWARNS?=3D 6 X XBINDIR=3D /usr/bin X X.include <bsd.prog.mk> END-of-Makefile echo x - size.1 sed 's/^X//' >size.1 << 'END-of-size.1' X.\" Copyright (c) 2008 Ed Schouten <ed@fxq.nl> X.\" All rights reserved. X.\" X.\" Redistribution and use in source and binary forms, with or without X.\" modification, are permitted provided that the following conditions X.\" are met: X.\" 1. Redistributions of source code must retain the above copyright X.\" notice, this list of conditions and the following disclaimer. X.\" 2. Redistributions in binary form must reproduce the above copyright X.\" notice, this list of conditions and the following disclaimer in the X.\" documentation and/or other materials provided with the distribution. X.\" X.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND X.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE X.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PUR= POSE X.\" ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE X.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUEN= TIAL X.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS X.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) X.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, ST= RICT X.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY = WAY X.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF X.\" SUCH DAMAGE. X.\" X.\" $FreeBSD$ X.\" X.Dd March 3, 2008 X.Dt SIZE 1 X.Os X.Sh NAME X.Nm size X.Nd list section sizes of ELF files X.Sh SYNOPSIS X.Nm X.Op Fl A | Fl B X.Op Fl d | Fl o | Fl x X.Op Fl t X.Ar X.Sh DESCRIPTION XFor each operand, X.Nm Xdisplays the sizes of the read-only sections of the ELF file (text), the Xread-write sections (data) and the zero-initialized sections (bss). XThis format is referred to as the Berkeley format, which is Xshown by default. X.Pp XThis version of X.Nm Xalso supports the System V format, which shows the sizes and load Xaddresses of all sections within the ELF file. X.Pp XThe following options are available: X.Bl -tag -width indent X.It Fl A XShow output using the System V format. X.It Fl B XShow output using the Berkeley format. X.It Fl d XDisplay sizes as decimal numbers. X.It Fl o XDisplay sizes as octal numbers. XWhen enabled, the X.Em dec Xcolumn in the Berkeley format is replaced by a column which displays the Xsize in octal. X.It Fl x XDisplay sizes as hexadecimal numbers. X.It Fl t XWhen using the Berkeley format, show the total size for all ELF files at Xthe bottom. X.Sh EXIT STATUS X.Ex -std X.Sh COMPATIBILITY XThis utility implements most standard options that are implemented in Xthe GNU X.Nm Xutility. XUnlike the GNU version, this utility only supports decimal, Xoctal and hexadecimal radices. X.Sh SEE ALSO X.Xr objdump 1 , X.Xr readelf 1 , X.Xr elf 3 X.Sh HISTORY XIn X.Fx 8.0 , X.An Ed Schouten Aq ed@80386.nl Xreimplemented X.Nm Xusing the X.Lb libelf . END-of-size.1 echo x - size.c sed 's/^X//' >size.c << 'END-of-size.c' X/*- X * Copyright (c) 2008 Ed Schouten <ed@80386.nl> X * All rights reserved. X * X * Redistribution and use in source and binary forms, with or without X * modification, are permitted provided that the following conditions X * are met: X * 1. Redistributions of source code must retain the above copyright X * notice, this list of conditions and the following disclaimer. X * 2. Redistributions in binary form must reproduce the above copyright X * notice, this list of conditions and the following disclaimer in the X * documentation and/or other materials provided with the distribution. X * X * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND X * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE X * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURP= OSE X * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE X * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENT= IAL X * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS X * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) X * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STR= ICT X * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY W= AY X * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF X * SUCH DAMAGE. X * X * $FreeBSD$ X */ X X#include <err.h> X#include <fcntl.h> X#include <stdio.h> X#include <stdlib.h> X#include <unistd.h> X X#include <gelf.h> X#include <libelf.h> X Xstatic int sysvflg =3D -1; Xstatic int radixflg =3D -1; Xstatic int totalflg =3D 0; Xstatic int exitcode =3D 0; X Xstatic void Xusage(void) X{ X fprintf(stderr, "usage: size [-A | -B] [-d | -o | -x] [-t] file ...\n"); X exit(1); X} X Xstatic unsigned int total_text =3D 0, total_data =3D 0, total_bss =3D 0; Xstatic int header_printed =3D 0; X Xstatic void Xprint_line_berkeley(unsigned int text, unsigned int data, X unsigned int bss, const char *filename) X{ X const char *fmt; X unsigned int total; X X total =3D text + data + bss; X X if (!header_printed) { X printf(" text\t data\t bss\t %s\t hex\tfilename\n", X radixflg =3D=3D 8 ? "oct" : "dec"); X header_printed =3D 1; X } X X /* Radix dependant fields */ X if (radixflg =3D=3D 8) { X fmt =3D "%#7o\t%#7o\t%#7o\t%7o"; X } else if (radixflg =3D=3D 16) { X fmt =3D "%#7x\t%#7x\t%#7x\t%7u"; X } else { X fmt =3D "%7u\t%7u\t%7u\t%7u"; X } X printf(fmt, text, data, bss, total); X /* Radix independant fields */ X printf("\t%7x\t%s\n", total, filename); X} X Xstatic void Xprint_entry_berkeley(Elf *e, const char *filename) X{ X unsigned int i, nsec; X unsigned int text =3D 0, data =3D 0, bss =3D 0; X Elf_Scn *es; X GElf_Shdr hdr; X X if (elf_getshnum(e, &nsec) =3D=3D 0) { X warnx("%s: Unable to obtain the number of sections: %s", X filename, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X X for (i =3D 0; i < nsec; i++) { X /* Iterate all the section headers */ X es =3D elf_getscn(e, i); X if (es =3D=3D NULL) { X warnx("%s: Unable to obtain section %u: %s", X filename, i, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X X if (gelf_getshdr(es, &hdr) !=3D &hdr) { X warnx("%s: Unable to obtain section header %u: %s", X filename, i, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X X /* Skip fields that don't take space anyway */ X if ((hdr.sh_flags & SHF_ALLOC) =3D=3D 0) X continue; X if (hdr.sh_flags & SHF_EXECINSTR || X ((hdr.sh_flags & SHF_WRITE) =3D=3D 0)) { X /* Read-only data */ X text +=3D hdr.sh_size; X } else if (hdr.sh_type =3D=3D SHT_NOBITS) { X /* Fields that don't consume space in the binary */ X bss +=3D hdr.sh_size; X } else { X /* Read/write data segments */ X data +=3D hdr.sh_size; X } X } X X print_line_berkeley(text, data, bss, filename); X X total_text +=3D text; X total_data +=3D data; X total_bss +=3D bss; X} X Xstatic void Xprint_footer_berkeley(void) X{ X if (!header_printed || !totalflg) X return; X X print_line_berkeley(total_text, total_data, total_bss, "(TOTALS)"); X} X Xstatic void Xprint_entry_sysv(Elf *e, const char *filename) X{ X unsigned int i, nsec, total =3D 0; X size_t strndx; X const char *fmt; X Elf_Scn *es; X GElf_Shdr hdr; X Elf_Data *sdata; X const char *snames; X X if (elf_getshnum(e, &nsec) =3D=3D 0) { X warnx("%s: Unable to obtain the number of sections: %s", X filename, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X X /* Locate the string table for the section names */ X if (elf_getshstrndx(e, &strndx) =3D=3D 0) { X warnx("%s: Unable to locate the string index: %s", X filename, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X es =3D elf_getscn(e, strndx); X if ((es =3D elf_getscn(e, strndx)) =3D=3D NULL || X (sdata =3D elf_rawdata(es, NULL)) =3D=3D NULL || X sdata->d_buf =3D=3D NULL) { X warnx("%s: Unable to obtain string table: %s", X filename, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X snames =3D sdata->d_buf; X X printf("%s :\nsection size addr\n", filename); X X for (i =3D 0; i < nsec; i++) { X /* Iterate all the section headers */ X es =3D elf_getscn(e, i); X if (es =3D=3D NULL) { X warnx("%s: Unable to obtain section %u: %s", X filename, i, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X X if (gelf_getshdr(es, &hdr) !=3D &hdr) { X warnx("%s: Unable to obtain section header %u: %s", X filename, i, elf_errmsg(-1)); X exitcode =3D 1; X return; X } X X /* Skip fields that don't take space anyway */ X if (hdr.sh_type !=3D SHT_PROGBITS && X (hdr.sh_flags & SHF_ALLOC) =3D=3D 0) X continue; X X /* Make sure we don't overrun the buffer */ X if (hdr.sh_name >=3D sdata->d_size) { X warnx("%s: Invalid name offset: %u", X filename, hdr.sh_name); X exitcode =3D 1; X continue; X } X X total +=3D hdr.sh_size; X X printf("%-16s ", snames + hdr.sh_name); X if (radixflg =3D=3D 8) { X fmt =3D "%#7o %#10o\n"; X } else if (radixflg =3D=3D 16) { X fmt =3D "%#7x %#10x\n"; X } else { X fmt =3D "%7u %10u\n"; X } X printf(fmt, (unsigned int)hdr.sh_size, (unsigned int)hdr.sh_addr); X } X X fputs("Total ", stdout); X if (radixflg =3D=3D 8) { X printf("%#18o", total); X } else if (radixflg =3D=3D 16) { X printf("%#18x", total); X } else { X printf("%18d", total); X } X fputs("\n\n\n", stdout); X} X Xint Xmain(int argc, char *argv[]) X{ X int ch, fd; X Elf *e; X X while ((ch =3D getopt(argc, argv, "ABdotx")) !=3D -1) { X switch (ch) { X case 'A': X /* SysV style output */ X if (sysvflg =3D=3D 0) X usage(); X sysvflg =3D 1; X break; X case 'B': X /* Berkeley style output */ X if (sysvflg =3D=3D 1) X usage(); X sysvflg =3D 0; X break; X case 'd': X /* Decimal output */ X if (radixflg !=3D -1 && radixflg !=3D 10) X usage(); X radixflg =3D 10; X break; X case 'o': X /* Octal output */ X if (radixflg !=3D -1 && radixflg !=3D 8) X usage(); X radixflg =3D 8; X break; X case 'x': X /* Hexadecimal output */ X if (radixflg !=3D -1 && radixflg !=3D 8) X usage(); X radixflg =3D 16; X break; X case 't': X /* List sum of sizes */ X totalflg =3D 1; X break; X default: X usage(); X } X } X argc -=3D optind; X argv +=3D optind; X X if (argc =3D=3D 0) X usage(); X=09 X if (elf_version(EV_CURRENT) =3D=3D EV_NONE) { X errx(1, "ELF library initialization failed: %s", X elf_errmsg(-1)); X } X=09 X X do { X /* X * Already parse the ELF file and hand it over to the print X * routines. X */ X fd =3D open(*argv, O_RDONLY); X if (fd < 0) { X warn("%s", *argv); X exitcode =3D 1; X continue; X } X X e =3D elf_begin(fd, ELF_C_READ, NULL); X if (e =3D=3D NULL) { X warnx("%s: %s", *argv, elf_errmsg(-1)); X exitcode =3D 1; X continue; X } X X if (sysvflg =3D=3D 1) X print_entry_sysv(e, *argv); X else X print_entry_berkeley(e, *argv); X X elf_end(e); X close(fd); X } while (argv++, --argc !=3D 0); X X if (sysvflg !=3D 1) X print_footer_berkeley(); X X return (exitcode); X} END-of-size.c exit --jtcAeju3WzRmRF+o-- --yaap9KN+GmBP785v Content-Type: application/pgp-signature Content-Disposition: inline -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.8 (FreeBSD) iEYEARECAAYFAkfM/LQACgkQ52SDGA2eCwWwQQCfRyNC8Tb6PIOmgXzgk/QOUQwG LWMAninpb2Z+Zrx8TEbEe2/ypYXw+lR7 =ADIA -----END PGP SIGNATURE----- --yaap9KN+GmBP785v--
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20080304073932.GJ80576>