All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.fornwall.jelf.ElfSection Maven / Gradle / Ivy

There is a newer version: 0.9.8
Show newest version
package net.fornwall.jelf;

import java.io.IOException;

/**
 * Class corresponding to the Elf32_Shdr/Elf64_Shdr struct.
 * 
 * 

* An object file's section header table lets one locate all the file's sections. The section header table is an array * of Elf32_Shdr or Elf64_Shdr structures. A section header table index is a subscript into this array. The ELF header's * {@link ElfFile#sh_offset e_shoff member} gives the byte offset from the beginning of the file to the section header * table with each section header entry being {@link ElfFile#sh_entry_size e_shentsize} bytes big. * *

* {@link ElfFile#num_sh e_shnum} normally tells how many entries the section header table contains, but if the number * of sections is greater than or equal to SHN_LORESERVE (0xff00), e_shnum has the value SHN_UNDEF (0) and the actual * number of section header table entries is contained in the sh_size field of the section header at index 0 (otherwise, * the sh_size member of the initial entry contains 0). * *

* Some section header table indexes are reserved in contexts where index size is restricted, for example, the st_shndx * member of a symbol table entry and the e_shnum and e_shstrndx members of the ELF header. In such contexts, the * reserved values do not represent actual sections in the object file. Also in such contexts, an escape value indicates * that the actual section index is to be found elsewhere, in a larger field. */ public final class ElfSection implements SymbolLocator { /** * Marks the section header as inactive; it does not have an associated section. Other members of the section header * have undefined values. */ public static final int SHT_NULL = 0; /** Section holds information defined by the program. */ public static final int SHT_PROGBITS = 1; /** * Section holds symbol table information for link editing. It may also be used to store symbols for dynamic * linking. Only one per ELF file. The symtab contains everything, but it is non-allocable, can be stripped, and has * no runtime cost. */ public static final int SHT_SYMTAB = 2; /** Section holds string table information. */ public static final int SHT_STRTAB = 3; /** Section holds relocation entries with explicit addends. */ public static final int SHT_RELA = 4; /** Section holds symbol hash table. */ public static final int SHT_HASH = 5; /** * Section holds information for dynamic linking. Only one per ELF file. The dynsym is allocable, and contains the * symbols needed to support runtime operation. */ public static final int SHT_DYNAMIC = 6; /** Section holds information that marks the file. */ public static final int SHT_NOTE = 7; /** Section occupies no space but resembles TYPE_PROGBITS. */ public static final int SHT_NOBITS = 8; /** Section holds relocation entries without explicit addends. */ public static final int SHT_REL = 9; /** Section is reserved but has unspecified semantics. */ public static final int SHT_SHLIB = 10; /** Section holds a minimum set of dynamic linking symbols. Only one per ELF file. */ public static final int SHT_DYNSYM = 11; public static final int SHT_INIT_ARRAY = 14; public static final int SHT_FINI_ARRAY = 15; public static final int SHT_PREINIT_ARRAY = 16; public static final int SHT_GROUP = 17; public static final int SHT_SYMTAB_SHNDX = 18; public static final int SHT_GNU_verdef = 0x6ffffffd; public static final int SHT_GNU_verneed = 0x6ffffffe; public static final int SHT_GNU_versym = 0x6fffffff; /** Lower bound of the range of indexes reserved for operating system-specific semantics. */ public static final int SHT_LOOS = 0x60000000; /** Upper bound of the range of indexes reserved for operating system-specific semantics. */ public static final int SHT_HIOS = 0x6fffffff; /** Lower bound of the range of indexes reserved for processor-specific semantics. */ public static final int SHT_LOPROC = 0x70000000; /** Upper bound of the range of indexes reserved for processor-specific semantics. */ public static final int SHT_HIPROC = 0x7fffffff; /** Lower bound of the range of indexes reserved for application programs. */ public static final int SHT_LOUSER = 0x80000000; /** Upper bound of the range of indexes reserved for application programs. */ public static final int SHT_HIUSER = 0xffffffff; /** Flag informing that this section contains data that should be writable during process execution. */ public static final int FLAG_WRITE = 0x1; /** Flag informing that section occupies memory during process execution. */ public static final int FLAG_ALLOC = 0x2; /** Flag informing that section contains executable machine instructions. */ public static final int FLAG_EXEC_INSTR = 0x4; /** Flag informing that all the bits in the mask are reserved for processor specific semantics. */ public static final int FLAG_MASK = 0xf0000000; /** Section header name identifying the section as a string table. */ public static final String STRING_TABLE_NAME = ".strtab"; /** Section header name identifying the section as a dynamic string table. */ public static final String DYNAMIC_STRING_TABLE_NAME = ".dynstr"; /** Index into the section header string table which gives the name of the section. */ public final int name_ndx; // Elf32_Word or Elf64_Word - 4 bytes in both. /** Section content and semantics. */ public final int type; // Elf32_Word or Elf64_Word - 4 bytes in both. /** Flags. */ public final long flags; // Elf32_Word or Elf64_Xword. /** * sh_addr. If the section will be in the memory image of a process this will be the address at which the first byte * of section will be loaded. Otherwise, this value is 0. */ public final long address; // Elf32_Addr /** Offset from beginning of file to first byte of the section. */ public final long section_offset; // Elf32_Off /** Size in bytes of the section. TYPE_NOBITS is a special case. */ public final long size; // Elf32_Word /** Section header table index link. */ public final int link; // Elf32_Word or Elf64_Word - 4 bytes in both. /** Extra information determined by the section type. */ public final int info; // Elf32_Word or Elf64_Word - 4 bytes in both. /** Address alignment constraints for the section. */ public final long address_alignment; // Elf32_Word /** Size of a fixed-size entry, 0 if none. */ public final long entry_size; // Elf32_Word private MemoizedObject[] symbols; private MemoizedObject stringTable; private MemoizedObject hashTable; private MemoizedObject[] relocations; /** For the {@link #SHT_DYNAMIC} ".dynamic" structure. */ private MemoizedObject dynamicStructure; private MemoizedObject initArray, preInitArray; private final ElfFile elfHeader; /** Reads the section header information located at offset. */ ElfSection(final ElfParser parser, long offset) { this.elfHeader = parser.elfFile; parser.seek(offset); name_ndx = parser.readInt(); type = parser.readInt(); flags = parser.readIntOrLong(); address = parser.readIntOrLong(); section_offset = parser.readIntOrLong(); size = parser.readIntOrLong(); link = parser.readInt(); info = parser.readInt(); address_alignment = parser.readIntOrLong(); entry_size = parser.readIntOrLong(); switch (type) { case ElfSection.SHT_SYMTAB: case ElfSection.SHT_DYNSYM: int num_entries = (int) (size / entry_size); symbols = MemoizedObject.uncheckedArray(num_entries); for (int i = 0; i < num_entries; i++) { final long symbolOffset = section_offset + (i * entry_size); symbols[i] = new MemoizedObject() { @Override public ElfSymbol computeValue() { return new ElfSymbol(parser, symbolOffset, type); } }; } break; case ElfSection.SHT_STRTAB: stringTable = new MemoizedObject() { @Override public ElfStringTable computeValue() throws IOException { return new ElfStringTable(parser, section_offset, (int) size); } }; break; case ElfSection.SHT_HASH: hashTable = new MemoizedObject() { @Override public ElfHashTable computeValue() { return new ElfHashTable(parser, section_offset, (int) size); } }; break; case ElfSection.SHT_DYNAMIC: dynamicStructure = new MemoizedObject() { @Override protected ElfDynamicStructure computeValue() throws ElfException, IOException { return new ElfDynamicStructure(parser, section_offset, (int) size); } }; break; case ElfSection.SHT_RELA: case ElfSection.SHT_REL: num_entries = (int) (size / entry_size); relocations = MemoizedObject.uncheckedArray(num_entries); for (int i = 0; i < num_entries; i++) { final long relocationOffset = section_offset + (i * entry_size); relocations[i] = new MemoizedObject() { @Override public ElfRelocation computeValue() throws IOException { return new ElfRelocation(parser, relocationOffset, entry_size, parser.elfFile.getSection(link)); } }; } break; case ElfSection.SHT_INIT_ARRAY: initArray = new MemoizedObject() { @Override protected ElfInitArray computeValue() throws ElfException { return new ElfInitArray(parser, section_offset, (int) size); } }; break; case ElfSection.SHT_PREINIT_ARRAY: preInitArray = new MemoizedObject() { @Override protected ElfInitArray computeValue() throws ElfException { return new ElfInitArray(parser, section_offset, (int) size); } }; break; case ElfSection.SHT_NULL: case ElfSection.SHT_PROGBITS: case ElfSection.SHT_SHLIB: case ElfSection.SHT_NOTE: case ElfSection.SHT_NOBITS: default: break; } } public ElfInitArray getInitArray() throws IOException { return initArray.getValue(); } public ElfInitArray getPreInitArray() throws IOException { return preInitArray.getValue(); } /** Returns the number of symbols in this section or 0 if none. */ public int getNumberOfSymbols() { return (symbols != null) ? symbols.length : 0; } /** Returns the symbol at the specified index. The ELF symbol at index 0 is the undefined symbol. */ @Override public ElfSymbol getELFSymbol(int index) throws IOException { return symbols[index].getValue(); } @Override public ElfSymbol getELFSymbolByName(String name) throws IOException { for (int i = 0, m = getNumberOfSymbols(); i < m; i++) { ElfSymbol symbol = getELFSymbol(i); if (name != null && name.equals(symbol.getName())) { return symbol; } } return null; } @Override public ElfSymbol getELFSymbolByAddr(long addr) throws IOException { for (int i = 0, m = getNumberOfSymbols(); i < m; i++) { ElfSymbol symbol = getELFSymbol(i); if (addr >= symbol.value && addr < symbol.value + symbol.size) { return symbol; } } return null; } /** Returns the number of relocations in this section or 0 if none. */ public int getNumberOfRelocations() { return (relocations != null) ? relocations.length : 0; } /** Returns the relocation at the specified index. */ public ElfRelocation getELFRelocation(int index) throws IOException { return relocations[index].getValue(); } /** Returns the string table for this section or null if one does not exist. */ public ElfStringTable getStringTable() throws IOException { return (stringTable != null) ? stringTable.getValue() : null; } public ElfDynamicStructure getDynamicSection() throws IOException { return (dynamicStructure != null) ? dynamicStructure.getValue() : null; } /** * Returns the hash table for this section or null if one does not exist. NOTE: currently the ELFHashTable does not * work and this method will always return null. */ public ElfHashTable getHashTable() throws IOException { return (hashTable != null) ? hashTable.getValue() : null; } /** Returns the name of the section or null if the section has no name. */ public String getName() throws IOException { if (name_ndx == 0) return null; ElfStringTable tbl = elfHeader.getSectionNameStringTable(); return tbl.get(name_ndx); } @Override public String toString() { try { return "ElfSectionHeader[name=" + getName() + ", type=0x" + Long.toHexString(type) + "]"; } catch (IOException e) { throw new RuntimeException(e); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy