--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/bintools/elftools/elfdump/elfdump.cpp Mon May 10 19:54:49 2010 +0100
@@ -0,0 +1,482 @@
+/*
+* Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
+* All rights reserved.
+* This component and the accompanying materials are made available
+* under the terms of the License "Eclipse Public License v1.0"
+* which accompanies this distribution, and is available
+* at the URL "http://www.eclipse.org/legal/epl-v10.html".
+*
+* Initial Contributors:
+* Nokia Corporation - initial contribution.
+*
+* Contributors:
+*
+* Description:
+*
+*/
+
+
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <elfdefs.h>
+#include <sys/stat.h>
+
+#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
+#include <iostream>
+#include <iomanip>
+using namespace std;
+#else //(!__MSVCDOTNET__ && !__TOOLS2__)
+#include <iostream.h>
+#include <iomanip.h>
+#endif //__MSVCDOTNET__
+
+#define ADDR(rtype, p, o) (rtype *)(((char *)p) + o)
+
+bool ignoreSomeSections;
+
+void hexdump(unsigned char* data, int aSize, int offset)
+ // print hex dump of relevant sections
+ {
+ int i=0;
+ int p=0;
+ while (i<aSize)
+ {
+ int count=0;
+ if(p==0){printf("\t%06x ",offset);} // offset into section
+ while (i<aSize && count<4)
+ {
+ printf("%02X", *data); // print 4 lots of %08x for the data expresed as 32-bit word
+ data++;
+ i++;
+ count++;
+ offset++;
+ }
+
+ printf(" ");
+ p++;
+ if (p==4)
+ {
+ data=data-16;
+ for (int i=0;i<16;i++) //print 16 bytes of memory interpreted
+ { //as ASCII characters with all non-printing
+ if (*data>32 && *data <127) //characters converted to dots
+ {
+ printf("%1c",*data);
+ }
+ else
+ {
+ printf(".");
+ }
+ data++;
+ }
+ p=0;
+ printf(" \n ");
+ }
+ }
+ printf(" \n\n ");
+ }
+
+void print_directive(unsigned char* data, int size)
+ // print formatted text of directive section
+ {
+ printf ("\t");
+
+ for (int i=0; i<size; i++)
+ {
+ if ((char)data[i]>31 && (char)data[i]<127)
+ {
+ printf ("%c", (char)data[i]);
+ }
+
+ if ((char)data[i]=='\n')
+ {
+ printf ("\n\t");
+ }
+ }
+
+ printf ("\n");
+ }
+
+void print_reloc(Elf32_Ehdr* eh, Elf32_Sym* symT, unsigned char* strtab)
+ // print relocation section
+ {
+ Elf32_Shdr* shdr = ADDR(Elf32_Shdr, eh, eh->e_shoff);
+ for (int j=0;j< eh->e_shnum;j++)
+ {
+ char* sname = ADDR(char, eh, shdr[eh->e_shstrndx].sh_offset);
+ if ( (shdr[j].sh_type==9) &&
+ ( (!ignoreSomeSections) ||
+ (strncmp(".rel.debug_", &sname[shdr[j].sh_name], 11))
+ )
+ )
+ {
+ unsigned char* data = ADDR(unsigned char, eh, shdr[j].sh_offset);
+ int noOfReloc=shdr[j].sh_size / shdr[j].sh_entsize;
+ printf("\n\n\n\t\t\t%s\n", &sname[shdr[j].sh_name]);
+ Elf32_Rel* rl=(Elf32_Rel*)data; // pointer to relocation section
+ for (int i=0;i<noOfReloc;i++)
+ {
+ unsigned char* symbolName = strtab; // pointer to firest element of string // table which holds symbol names
+ Elf32_Sym* sym = symT; // pointer to symbol table
+ int symTIndx= ELF32_R_SYM(rl->r_info); // symbol Tableindex
+ sym=sym+symTIndx;
+ symbolName=symbolName+sym->st_name; // index into string table section
+ // with symbol names
+ printf("\t0x%08x \t", rl->r_offset); // prints offset into relocation section
+ printf("%d", symTIndx); // symbol table index
+ printf("\t%s\n",symbolName); // symbol name
+ rl++;
+ }
+ }
+ }
+ }
+
+void print_GlSymbols(Elf32_Ehdr* eh, Elf32_Sym* symT, unsigned char* strtab)
+ // print global symbols from Symbol Table
+ {
+ Elf32_Shdr* shdr = ADDR(Elf32_Shdr, eh, eh->e_shoff);
+ char* sname = ADDR(char, eh, shdr[eh->e_shstrndx].sh_offset);
+ for (int i=0;i< eh->e_shnum;i++)
+ {
+ if (!strcmp(".symtab", &sname[shdr[i].sh_name]))
+ {
+ int noOfSym=shdr[i].sh_size / shdr[i].sh_entsize; // number of symbols
+ const char *symName =(const char *)strtab;
+ int count = 1;
+ printf("Global symbols:\n");
+ printf("=================\n\n");
+ for (int l=0;l< noOfSym ;l++)
+ {
+ symT=symT+1;
+ if( ELF32_ST_BIND(symT->st_info) == 1) // searching for global symbols
+ {
+ symName = symName + symT->st_name; // index into string table section
+ printf("%d ",count);
+ printf(symName);
+ printf("\n");
+ symName = symName - symT->st_name; // back to pointing to first byte of string table
+ count++;
+ }
+
+ }
+ }
+ }
+ }
+
+void print_elf_header(Elf32_Ehdr* eh)
+ {
+ // print elf header
+ if (eh->e_version==1)
+ printf("\tHeader version: EV_CURRENT (Current version)\n");
+ else
+ printf("\tInvalid version: EV_NONE (Invalid version)\n");
+
+ if (eh->e_type==0)
+ printf("\tFile Type: ET_NONE (No file type) (0)\n");
+ else if (eh->e_type==1)
+ printf("\tFile Type: ET_REL (Relocatable object) (1)\n");
+ else if (eh->e_type==2)
+ printf("\tFile Type: ET_EXEC (Executable file) (2)\n");
+ else if (eh->e_type==3)
+ printf("\tFile Type: ET_DYN (Shared object file) (3)\n");
+ else if (eh->e_type==4)
+ printf("\tFile Type: ET_CORE (Core File) (4)\n");
+ else if (eh->e_type==65280)
+ printf("\tFile Type: ET_LOPROC (Precessor Specific) (ff00)\n");
+ else
+ printf("\tFile Type: ET_HIPROC (Precessor Specific) (ffff)\n");
+ if (eh->e_machine==40)
+ printf("\tMachine: EM_ARM (ARM)\n");
+ else
+ printf("\tERROR:\tUnexpected machine\n");
+
+ printf("\tEntry offset (in SHF_ENTRYSECT section):0x%08x \n",eh->e_entry);
+ printf("\tProgram header entries : %d\n",eh->e_phnum);
+ printf("\tSection header entries : %d\n",eh->e_shnum);
+
+ printf("\tProgram header offset : %d",eh->e_phoff);
+ printf(" bytes (0x%08X",eh->e_phoff);
+ printf(")\n");
+ printf("\tSection header offset : %d",eh->e_shoff);
+ printf(" bytes (0x%08X",eh->e_shoff);
+ printf(")\n");
+
+ printf("\tProgram header entry size : %d",eh->e_phentsize);
+ printf(" bytes (0x%02X",eh->e_phentsize);
+ printf(")\n");
+ printf("\tSection header entry size : %d",eh->e_shentsize);
+ printf(" bytes (0x%02X",eh->e_shentsize);
+ printf(")\n");
+ printf("\tSection header string table index: %d \n", eh->e_shstrndx);
+ printf("\tHeader size: %d", eh->e_ehsize);
+ printf(" bytes (0x%02X",eh->e_ehsize);
+ printf(")\n");
+ }
+
+void print_sect_header(char* sname, Elf32_Shdr* shdr, int count)
+ // print section header names
+ {
+ static const char* KtypeName[]={"0","SHT_PROGBITS (1)","SHT_SYMTAB (2)","SHT_STRTAB (3)",
+ "SHT_RELA (4)","5", "SHT_DINAMIC (6)","7","8","SHT_REL (9)",
+ "10","SHT_DINSYM (11)"};
+
+ printf("\n\n\tName\t\t:%1s\n ",&sname[shdr[count].sh_name]);
+ printf("\tType\t\t: %s\n", KtypeName[shdr[count].sh_type]);
+ printf("\tAddr\t\t: 0x%08X\n",shdr[count].sh_addr);
+ printf("\tSize\t\t: %1d", shdr[count].sh_size);
+ printf(" bytes (0x%X",shdr[count].sh_size);
+ printf(")\n");
+ printf("\tEntry Size\t: %1d\n",shdr[count].sh_entsize);
+ printf("\tAligment\t: %1d\n\n\n",shdr[count].sh_addralign);
+ }
+
+unsigned char* findSymbolStringT(Elf32_Ehdr* eh)
+ //calculate and return pointer to the first byte of string table(the one with symbol names)
+ {
+ Elf32_Shdr* shdr = ADDR(Elf32_Shdr, eh, eh->e_shoff);
+ char* sname = ADDR(char, eh, shdr[eh->e_shstrndx].sh_offset);
+ for (int i=0;i < eh->e_shnum; i++)
+ {
+ if (!strcmp(".strtab", &sname[shdr[i].sh_name]))
+ {
+ unsigned char* data = ADDR(unsigned char, eh, shdr[i].sh_offset);
+ return data; //pointer to the first byte of string table section
+ }
+ }
+ return NULL; //if not found
+ }
+
+Elf32_Sym* findSymbolT(Elf32_Ehdr* eh)
+ //calculate and return pointer to the first element of symbol table
+ {
+ Elf32_Shdr* shdr = ADDR(Elf32_Shdr, eh, eh->e_shoff);
+ for (int i=0;i < eh->e_shnum;i++)
+ {
+ if (shdr[i].sh_type==2)
+ {
+ unsigned char* data = ADDR(unsigned char, eh, shdr[i].sh_offset);
+ Elf32_Sym* sym=(Elf32_Sym*)data;
+ return sym; //pointer to the first element of symbol table.
+ }
+ }
+ return NULL; // if not found
+ }
+
+void print_Summary(Elf32_Ehdr* eh)
+ {
+ //print section names
+ Elf32_Shdr* shdr = ADDR(Elf32_Shdr, eh, eh->e_shoff);
+ char* sname = ADDR(char, eh, shdr[eh->e_shstrndx].sh_offset);
+ printf("\nSummary: \n");
+ printf("==========\n");
+ for (int i=0;i< eh->e_shnum;i++)
+ {
+ printf(&sname[shdr[i].sh_name]);
+ printf("\n");
+ }
+ }
+
+bool printAll;
+
+int do_elf_file(char* buffer, char* name)
+ {
+ Elf32_Ehdr* eh=(Elf32_Ehdr *)buffer; //elf header
+ if (eh->e_ident[EI_MAG0] !=0x7f || eh->e_ident[EI_MAG1] != 0x45 || eh->e_ident[EI_MAG2] !=0x4c || eh->e_ident[EI_MAG3] != 0x46)
+ {
+ // EI_MAG0 to EI_MAG3 - A files' first 4 bytes hold a 'magic number', identifying the file as an ELF object file.
+ cout << "Error: " << name << " is not a valid ELF file";
+ return 1;
+ }
+ if (eh->e_ident[EI_DATA] == 2)
+ {
+ // ELF Header size should be 52 bytes or converted into Big-Endian system 13312
+ if (eh->e_ehsize != 13312)
+ {
+ printf("\tERROR:\tELF Header contains invalid file type\n");
+ exit(1);
+ }
+ // e_ident[EI_DATA] specifies the data encoding of the processor-specific data in the object file.
+ printf("\tERROR:\tData encoding ELFDATA2MSB (Big-Endian) not supported\n");
+ exit(1);
+ }
+ if (eh->e_ehsize != 52)
+ {
+ // ELF Header size should be 52 bytes
+ printf("\tERROR:\tELF Header contains invalid file type\n");
+ exit(1);
+ }
+ int shoff = eh->e_shoff; // offset of section header table
+ Elf32_Shdr* shdr = ADDR(Elf32_Shdr, eh, shoff); // calculating pointer to Secton Header Table
+ // Elf32_Shdr * shdr = (Elf32_Shdr *)(buffer+shoff);
+ int shnum = eh->e_shnum; // number of section headers
+ int shstrndx = eh->e_shstrndx;
+ int snameoffset = shdr[shstrndx].sh_offset; // offset in file of sections' names
+ char* sname = ADDR(char, eh, snameoffset); // pointer to String Table which holds section names
+ // char * sname = (char *)(buffer+snameoffset);
+ print_elf_header(eh); // print Elf Header
+
+ Elf32_Sym* symT= findSymbolT(eh); // pointer to Symbol table
+ if (symT==NULL)
+ {
+ printf("\nSymbol table not found\n");
+ }
+ unsigned char* strtab=findSymbolStringT(eh); // pointer to String table which holds symbol names
+ if (strtab==NULL)
+ {
+ printf("\nString (the one which holds symbol names) table not found\n");
+ }
+ print_reloc(eh,symT, strtab); // print relocation info showing symbol names and
+ // and the name of section in which the relocaton occurs.
+ for(int i = 0; i < shnum; i++)
+ {
+ unsigned char* data = ADDR(unsigned char, eh, shdr[i].sh_offset); //pointer to the first byte in the section
+ //unsigned char * data = (unsigned char * )(buffer+shdr[i].sh_offset);
+ int size = shdr[i].sh_size; // section size in bytes
+
+ //print directive section
+ if (!strcmp(".directive", &sname[shdr[i].sh_name]))
+ {
+ print_sect_header(sname, shdr, i);
+ print_directive(data,size);
+ }
+
+ if (!strcmp(".symtab", &sname[shdr[i].sh_name]))
+ {
+ print_sect_header(sname, shdr, i);
+ // print global symbols
+ print_GlSymbols(eh,symT, strtab);
+ }
+
+ //print relevant section header names
+ //print hex dump of relevant sections
+ if (shdr[i].sh_type==1 || shdr[i].sh_type==4 || shdr[i].sh_type==6 ||
+ shdr[i].sh_type==9 || shdr[i].sh_type==11)
+ {
+ if (strcmp(".comment", &sname[shdr[i].sh_name])&&
+ strcmp(".line", &sname[shdr[i].sh_name]) &&
+ strcmp(".hash", &sname[shdr[i].sh_name]) &&
+ strcmp(".note", &sname[shdr[i].sh_name]) &&
+ strcmp(".directive", &sname[shdr[i].sh_name]) &&
+ strncmp(".debug",&sname[shdr[i].sh_name] ,6))
+ {
+ if ( ! ( (ignoreSomeSections) &&
+ (strncmp(".rel.debug_", &sname[shdr[i].sh_name], 11)==0)
+ )
+ )
+ {
+ print_sect_header(sname, shdr, i);
+ hexdump(data,size,i);
+ }
+ }
+ }
+ if (printAll) // displays extra information
+ {
+ if(i!=0)
+ {
+ print_sect_header(sname, shdr, i);
+ hexdump(data,size,i);
+ }
+ }
+ }
+ print_Summary(eh); // print section names
+ return 0;
+}
+
+int read_ar_element_header(char* ptr)
+ {
+ int length = strtol(ptr+48,0,10);
+
+ if (strncmp(ptr+58, "\x60\x0A", 2) != 0)
+ {
+ return -1;
+ }
+ return length;
+ }
+
+int main(int argc, char* argv[])
+ {
+ char* arg;
+ int numberOfOptions=2;
+ printAll=0;
+ ignoreSomeSections=0;
+ if (argc<2)
+ {
+ cout << "File not specified";
+ exit(1);
+ }
+ else if (argc>numberOfOptions+2)
+ {
+ cout << "Too many arguments";
+ exit(1);
+ }
+ else
+ {
+ for (int i=1;i<=argc-2;i++)
+ {
+ if ( strcmp("-i", argv[i]) ==0 )
+ {
+ ignoreSomeSections=1;
+ }
+ else if ( strcmp("-a", argv[i]) ==0 )
+ {
+ printAll=1;
+ }
+ }
+ arg=argv[argc-1];
+ }
+
+ struct stat results;
+ stat(arg, &results);
+ FILE *elffile;
+ if((elffile = fopen(arg, "rb" )) == NULL)
+ {
+ cout << "Error opening file " << arg;
+ exit (1);
+ }
+ char* buffer=new char [results.st_size];//allocating enough memory
+ fread( buffer, sizeof( char ), results.st_size, elffile);
+ fclose(elffile);
+
+ if (strncmp(buffer, "!<arch>\x0A", 8) != 0)
+ {
+ // plain ELF file
+ if (do_elf_file(buffer, arg) != 0)
+ {
+ return 1;
+ }
+ return 0;
+ }
+
+ // library file
+ char* nextfile = buffer;
+ int remainder = results.st_size;
+
+#define ADVANCE(n) nextfile+=(n); remainder-=(n);
+
+ ADVANCE(8);
+
+ while (remainder > 0)
+ {
+ int element_length = read_ar_element_header(nextfile);
+ ADVANCE(60);
+
+ if (element_length < 0 || element_length > remainder)
+ {
+ cout << "Error: archive file corrupt";
+ return 1;
+ }
+
+ if (strncmp(nextfile, "\x7F\x45\x4C\x46",4) == 0)
+ {
+ if (do_elf_file(nextfile, "archive_element") != 0)
+ {
+ return 1;
+ }
+ }
+ element_length += element_length&1; // round up to a multiple of 2
+ ADVANCE(element_length);
+ }
+
+ return 0;
+ }