net.fornwall.jelf.ElfSection Maven / Gradle / Ivy
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);
}
}
}