bintools/elftools/getexports/geninf.cpp
changeset 2 39c28ec933dd
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/bintools/elftools/getexports/geninf.cpp	Mon May 10 19:54:49 2010 +0100
@@ -0,0 +1,433 @@
+/*
+* Copyright (c) 2006-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>
+#define DEFAULT_VERSION 2
+
+#define ELFADDR(rtype, p, o) (rtype *)(((char *)p) + o)
+
+FILE *OUTFILE;
+
+void croak(char * s)
+{
+	printf("GENINF ERROR: %s\n", s);
+	exit(EXIT_FAILURE);
+}
+
+void croak2(char * s1, char * s2)
+{
+	printf("GENINF ERROR: %s %s\n", s1, s2);
+	exit(EXIT_FAILURE);
+}
+
+int GetFileSize(FILE * f, char * file)
+{
+	int r = fseek(f, 0, SEEK_END);
+	if (r) croak2("can't seek to end of file:", file);
+	r = ftell(f);
+	rewind(f);
+	return r;
+}
+
+void EnsureElf(Elf32_Ehdr* eh)
+{
+	if (!(eh->e_ident[EI_MAG0] == ELFMAG0 &&
+	      eh->e_ident[EI_MAG1] == ELFMAG1 &&
+	      eh->e_ident[EI_MAG2] == ELFMAG2 &&
+	      eh->e_ident[EI_MAG3] == ELFMAG3)) 
+		croak("Invalid ELF magic.\n");
+	if (eh->e_ident[EI_CLASS] != ELFCLASS32) 
+		croak("File is not a 32 bit ELF object file.\n");
+	if (eh->e_ident[EI_DATA] != ELFDATA2LSB) 
+		croak("File data encoding is not Little Endian.\n");
+  	if (eh->e_machine != EM_ARM) 
+		croak("File does not target ARM/THUMB processors.\n");
+
+}
+
+
+Elf32_Ehdr* OpenElfFile (char* file)
+{
+	Elf32_Ehdr* eh;
+	FILE* f = fopen(file, "rb");
+	if (!f)
+		croak2("can't open file:", file);
+
+	int fsize = GetFileSize(f, file);
+
+	eh = (Elf32_Ehdr*)malloc(fsize);
+	if (!eh)
+		croak("Out of memory");
+
+	if (fread(eh, fsize, 1, f) != 1)
+		croak2("Can't read file", file);
+
+	EnsureElf(eh);
+
+	return eh;
+}
+
+typedef struct NameList 
+{
+	NameList * nl_next;
+	char * nl_name;
+	unsigned int nl_size;
+} NameList;
+
+
+static NameList * ExportedData = 0;
+static NameList * ExportedCode = 0;
+//Workaround  for compiler defect(To avoid export of static symbols)
+static NameList * LocalSymbols = 0;
+
+
+NameList* IsPresentInList(char * name, NameList * list)
+{
+	NameList * e = list;
+	while (e) {
+		if (!strcmp(name, e->nl_name))
+			return e;
+		else
+			e = e->nl_next;
+	}
+	return 0;
+}
+
+NameList* IsExportedData(char * name) 
+{
+	return IsPresentInList(name,ExportedData);
+}
+
+NameList* IsExportedCode(char * name) 
+{
+	return IsPresentInList(name,ExportedCode);
+}
+
+//Workaround  for compiler defect(To avoid export of local symbols)
+NameList* IsLocalSymbol(char * name) 
+{
+ 	return IsPresentInList(name,LocalSymbols);
+}
+
+void AddToList(char * name, NameList *& list, unsigned int aSymSz)
+{
+	NameList * ed = new NameList;
+	ed->nl_name = name;
+	ed->nl_size = aSymSz;
+	ed->nl_next = list;
+	list = ed;
+}
+
+void AddExportedData(char * name, unsigned int aSymSz)
+{
+	AddToList(name, ExportedData, aSymSz);
+}
+
+void AddExportedCode(char * name, unsigned int aSymSz)
+{
+	AddToList(name, ExportedCode, aSymSz);
+}
+
+//Workaround  for compiler defect(To avoid export of local symbols)
+void AddLocalSymbols(char * name)
+{
+	AddToList(name, LocalSymbols, 0);
+}
+
+void InitSymbolsLists(Elf32_Ehdr * eh,Elf32_Shdr * shdr,int shnum)
+{
+	Elf32_Sym * symtab = 0;
+	int nSyms = 0;
+	char * strtab = 0;
+	int i=0;
+
+	for (i = 0; (i < shnum); i++) {
+		if (shdr[i].sh_type == SHT_SYMTAB) {
+			symtab = ELFADDR(Elf32_Sym, eh, shdr[i].sh_offset);
+			nSyms = shdr[i].sh_size / shdr[i].sh_entsize;
+			strtab = ELFADDR(char, eh, shdr[shdr[i].sh_link].sh_offset);
+			break;
+		}
+	}
+	for (i = 0; i < nSyms; i++) {
+		if (symtab[i].st_info == ELF32_ST_INFO(STB_GLOBAL, STT_OBJECT)) {
+			char * name = ELFADDR(char, strtab, symtab[i].st_name);
+			AddExportedData(name, symtab[i].st_size);
+		} else if (symtab[i].st_info == ELF32_ST_INFO(STB_GLOBAL, STT_FUNC)) {
+			char * name = ELFADDR(char, strtab, symtab[i].st_name);
+			AddExportedCode(name, symtab[i].st_size);
+		}
+		else if (symtab[i].st_info == ELF32_ST_INFO(STB_LOCAL, STT_FUNC) || 
+ 			symtab[i].st_info == ELF32_ST_INFO(STB_LOCAL, STT_OBJECT)) {
+		// Workaround for compiler defect to avoid export of local symbols
+ 			char * name = ELFADDR(char, strtab, symtab[i].st_name);
+ 			AddLocalSymbols(name);
+ 		}
+	}
+}
+
+#define DIRECTIVE_TAG "#<SYMEDIT>#"
+#define DIRECTIVE_TAG_SIZE (sizeof(DIRECTIVE_TAG) - 1)
+#define EXPORT_DIRECTIVE "EXPORT "
+#define EXPORT_DIRECTIVE_SIZE (sizeof(EXPORT_DIRECTIVE) - 1)
+
+void PrintExportsFromSection(char * section, int size)
+{
+	if (strncmp(DIRECTIVE_TAG, section, DIRECTIVE_TAG_SIZE) != 0) 
+		croak("Unrecognized .directive tag");
+	section += (DIRECTIVE_TAG_SIZE + 1);
+	size -= (DIRECTIVE_TAG_SIZE + 1);
+	// The separator for the following entries is 0x0A (i.e. \n). 
+	// We're only interested in lines starting with EXPORT
+	char eolchar = '\n';
+	NameList* aDataSymbol;
+	while (size > 0) {
+		
+		char * eolp = (char *)memchr(section, eolchar, size);
+		int linelength = (eolp - section) + 1;
+		if (!strncmp(EXPORT_DIRECTIVE, section, EXPORT_DIRECTIVE_SIZE)) {
+			char * symbol = section + EXPORT_DIRECTIVE_SIZE;
+			int symbolsize = linelength - EXPORT_DIRECTIVE_SIZE - 1;
+			// null-terminate the string - doesn't matter that we side effect the
+			// section since we won't see it again and saves making a copy.
+			symbol[symbolsize] = 0;
+			if (IsExportedCode(symbol)) {
+				fprintf(OUTFILE, " %s\n", symbol);
+
+			} else if ( (aDataSymbol = IsExportedData(symbol)) != 0) {
+				fprintf(OUTFILE, " %s DATA %d\n", symbol, aDataSymbol->nl_size);
+			}
+			else {
+				//All those symbols included from static libs that are
+				//treated as exported because of the dll_runtime compiler
+				//option.Such symbols donot figure out in the symbol table.
+				//Hence are handled separately here.
+				
+				//Workaround  for compiler defect - To avoid export of local symbols
+				if(!IsLocalSymbol(symbol)) 
+ 				{																	
+					fprintf(OUTFILE, " %s\n", symbol);								
+				}
+
+			}
+		}
+		size -= linelength;
+		section += linelength;
+	}
+}
+
+void PrintABIv1ExportSymbols (Elf32_Ehdr * eh)
+{
+  int shoff = eh->e_shoff;                      // offset of section header table
+  if (shoff) {
+    Elf32_Shdr * shdr = ELFADDR(Elf32_Shdr, eh, shoff);
+
+    int shnum = eh->e_shnum;                    // number of section headers
+
+	// e_shnum will be 0 if the number of sections is >= SHN_LORESERVE (0xff00)
+	// If this is the case, sh_size contains the actual number of section headers.
+	// If sh_size is 0, there really aren't any section headers.
+	if (!shnum)
+		shnum = shdr->sh_size;
+
+    int shstrndx = eh->e_shstrndx;
+
+	// If the section name string table index is >= SHN_LORESERVE (0xff00), shstrndx
+	// contains SHN_XINDEX (0xffff).
+	// If this is the case, sh_link contains the actual index of the section name string
+	// table, otherwise sh_link is 0.
+
+	if (shstrndx >= 65535)
+		shstrndx = shdr->sh_link;
+
+	// Initialize list of global data symbols
+	InitSymbolsLists(eh,shdr,shnum);
+
+    int snameoffset = shdr[shstrndx].sh_offset; // offset in file of sections' names
+    char * sname = ELFADDR(char, eh, snameoffset);
+    for (int i = 0; i < shnum; i++) {
+      if (i != shstrndx) {
+		if (!strcmp(".directive", &sname[shdr[i].sh_name])) {
+			// we're in business. print the section to stdout
+			char * data = ELFADDR(char, eh, shdr[i].sh_offset);
+			int size = shdr[i].sh_size;
+			PrintExportsFromSection(data, size);
+		}
+      }
+    }
+  }
+}
+
+void PrintABIv2ExportSymbols(Elf32_Ehdr *eh)
+{
+	int shoff = eh->e_shoff;                      // offset of section header table
+	if (shoff) {
+	Elf32_Shdr * shdr = ELFADDR(Elf32_Shdr, eh, shoff);
+	int i=0;
+
+	int shnum = eh->e_shnum;                    // number of section headers
+
+	// e_shnum will be 0 if the number of sections is >= SHN_LORESERVE (0xff00)
+	// If this is the case, sh_size contains the actual number of section headers.
+	// If sh_size is 0, there really aren't any section headers.
+	if (!shnum)
+		shnum = shdr->sh_size;
+
+	int shstrndx = eh->e_shstrndx;
+
+	// If the section name string table index is >= SHN_LORESERVE (0xff00), shstrndx
+	// contains SHN_XINDEX (0xffff).
+	// If this is the case, sh_link contains the actual index of the section name string
+	// table, otherwise sh_link is 0.
+
+	if (shstrndx >= 65535)
+		shstrndx = shdr->sh_link;
+
+	//Get the symbol table
+	Elf32_Sym * symtab = 0;
+	int nSyms = 0;
+	char * strtab = 0;
+
+	for (i = 0; (i < shnum); i++) {
+		if (shdr[i].sh_type == SHT_DYNSYM) {
+			symtab = ELFADDR(Elf32_Sym, eh, shdr[i].sh_offset);
+			nSyms = shdr[i].sh_size / shdr[i].sh_entsize;
+			strtab = ELFADDR(char, eh, shdr[shdr[i].sh_link].sh_offset);
+			break;
+		}
+	}
+	for (i = 0; i < nSyms; i++) {
+		if (symtab[i].st_info == ELF32_ST_INFO(STB_GLOBAL, STT_OBJECT)) {
+			char * name = ELFADDR(char, strtab, symtab[i].st_name);
+			char * CodeSection = ELFADDR(char, eh, shdr[symtab[i].st_shndx].sh_offset);
+			Elf32_Word *aLocation = ELFADDR(Elf32_Word, CodeSection, symtab[i].st_value);
+			int ordinal = *aLocation;
+			fprintf(OUTFILE, "%s %d DATA %d %d\n",name, ordinal, symtab[i].st_size);
+		} else if (symtab[i].st_info == ELF32_ST_INFO(STB_GLOBAL, STT_FUNC)) {
+			char * name = ELFADDR(char, strtab, symtab[i].st_name);
+			char * CodeSection = ELFADDR(char, eh, shdr[symtab[i].st_shndx].sh_offset);
+			Elf32_Word *aLocation = ELFADDR(Elf32_Word, CodeSection, symtab[i].st_value);
+			int ordinal = *aLocation;
+			fprintf(OUTFILE, "%s %d \n",name, ordinal);
+		}
+	}
+	}
+}
+
+void PrintSONAME(Elf32_Ehdr *eh)
+{
+	int shoff = eh->e_shoff;                      // offset of section header table
+	if (shoff) {
+		Elf32_Shdr * shdr = ELFADDR(Elf32_Shdr, eh, shoff);
+
+		int shnum = eh->e_shnum;                    // number of section headers
+
+		// e_shnum will be 0 if the number of sections is >= SHN_LORESERVE (0xff00)
+		// If this is the case, sh_size contains the actual number of section headers.
+		// If sh_size is 0, there really aren't any section headers.
+		if (!shnum)
+			shnum = shdr->sh_size;
+
+		int shstrndx = eh->e_shstrndx;
+		char *aSHdrStrTab = ELFADDR(char, eh, shdr[shstrndx].sh_offset);
+		int i=0;
+		Elf32_Verdef *aVerDef = 0;
+		char *aStringTab = 0;
+		for(i = 0; i < shnum; i++) {
+			if(strcmp(aSHdrStrTab + shdr[i].sh_name, ".version_d") == 0) {
+				aVerDef = ELFADDR(Elf32_Verdef,eh , shdr[i].sh_offset );
+				aStringTab = ELFADDR(char, eh, shdr[shdr[i].sh_link].sh_offset);
+				break;
+			}
+		}
+		if(!aVerDef)
+			return;
+		char *aLinkAsName;
+		while(aVerDef) {
+			if(aVerDef->vd_ndx == DEFAULT_VERSION){
+				Elf32_Verdaux *aVerAux = ELFADDR(Elf32_Verdaux, aVerDef, aVerDef->vd_aux);
+				aLinkAsName = ELFADDR(char, aStringTab, aVerAux->vda_name);
+				fprintf(OUTFILE, "%s \n", aLinkAsName);
+				break;
+			}
+			aVerDef = ELFADDR(Elf32_Verdef, aVerDef, aVerDef->vd_next);
+		}
+	}
+}
+
+int main(int argc, char** argv)
+{
+	if( argc < 2 )
+	{
+		return EXIT_FAILURE;
+	}
+
+	char* file = argv[argc - 1];//The last arg is the file name
+	int idx = 0;
+	bool outFileOpt = false;
+	bool proxyDso = false;
+	bool soname = false;
+	while(idx < argc )
+	{
+		if((stricmp(argv[idx], "-o") == 0) && ((idx + 1) < argc) )
+		{
+			idx++;
+			char *outfilename = argv[idx];
+			OUTFILE = fopen(outfilename, "wb");
+			outFileOpt = true;
+		}
+		else if(stricmp(argv[idx], "-d") == 0)
+		{
+			proxyDso = true;
+		}
+		else if(stricmp(argv[idx], "-s") == 0)
+		{
+			soname = true;
+		}
+		idx++;
+	}
+
+	if(!outFileOpt)
+	{
+		OUTFILE = stdout;
+	}
+
+	Elf32_Ehdr * eh = OpenElfFile(file);
+
+	if( soname ) {
+		PrintSONAME(eh);
+	}
+	else if( !proxyDso ) {
+		PrintABIv1ExportSymbols(eh);
+	}
+	else {
+		PrintABIv2ExportSymbols(eh);
+	}
+	return EXIT_SUCCESS;
+}
+  
+
+  
+
+  
+