46 #include <sys/types.h> 56 #include "debug/Loader.hh" 67 if (elf_version(EV_CURRENT) == EV_NONE)
68 panic(
"wrong elf version number!");
75 elf_memory((
char *)const_cast<uint8_t *>(ifd->data()), ifd->len());
79 if (gelf_getehdr(elf, &ehdr) == 0)
93 std::string interpDir;
101 "Error: setInterpDir has already been called once\n");
108 elf = elf_memory((
char *)const_cast<uint8_t *>(
imageData->data()),
121 for (
int i = 0;
i <
ehdr.e_phnum; ++
i) {
123 if (gelf_getphdr(
elf,
i, &phdr) == 0) {
124 panic(
"gelf_getphdr failed for segment %d.",
i);
127 if (phdr.p_type == PT_LOAD)
129 if (phdr.p_type == PT_INTERP) {
140 "No loadable segments in '%s'. ELF file corrupted?\n",
153 const std::string elf_path = (
char *)
imageData->data() + phdr.p_offset;
154 if (!interpDir.empty())
155 return interpDir + elf_path;
163 auto &emach =
ehdr.e_machine;
164 auto &eclass =
ehdr.e_ident[EI_CLASS];
165 auto &edata =
ehdr.e_ident[EI_DATA];
168 if (emach == EM_SPARC64 || (emach == EM_SPARC && eclass == ELFCLASS64) ||
169 emach == EM_SPARCV9) {
171 }
else if (emach == EM_SPARC32PLUS ||
172 (emach == EM_SPARC && eclass == ELFCLASS32)) {
174 }
else if (emach == EM_MIPS && eclass == ELFCLASS32) {
176 if (edata != ELFDATA2LSB) {
177 fatal(
"The binary you're trying to load is compiled for big " 178 "endian MIPS. gem5\nonly supports little endian MIPS. " 179 "Please recompile your binary.\n");
181 }
else if (emach == EM_X86_64 && eclass == ELFCLASS64) {
183 }
else if (emach == EM_386 && eclass == ELFCLASS32) {
185 }
else if (emach == EM_ARM && eclass == ELFCLASS32) {
187 }
else if (emach == EM_AARCH64 && eclass == ELFCLASS64) {
189 }
else if (emach == EM_RISCV) {
191 }
else if (emach == EM_PPC && eclass == ELFCLASS32) {
193 if (edata != ELFDATA2MSB) {
194 fatal(
"The binary you're trying to load is compiled for " 195 "little endian Power.\ngem5 only supports big " 196 "endian Power. Please recompile your binary.\n");
198 }
else if (emach == EM_PPC64) {
199 fatal(
"The binary you're trying to load is compiled for 64-bit " 200 "Power. M5\n only supports 32-bit Power. Please " 201 "recompile your binary.\n");
203 warn(
"Unknown architecture: %d\n", emach);
211 switch (
ehdr.e_ident[EI_OSABI]) {
215 case ELFOSABI_SOLARIS:
224 case ELFOSABI_FREEBSD:
231 Elf_Scn *section = elf_getscn(
elf, 1);
232 for (
int sec_idx = 1; section; section = elf_getscn(
elf, ++sec_idx)) {
234 gelf_getshdr(section, &shdr);
236 char *e_str = elf_strptr(
elf,
ehdr.e_shstrndx, shdr.sh_name);
237 if (shdr.sh_type == SHT_NOTE && !strcmp(
".note.ABI-tag", e_str)) {
241 Elf_Data *raw_data = elf_rawdata(section,
nullptr);
242 assert(raw_data && raw_data->d_buf);
244 uint32_t raw_abi = ((uint32_t *)raw_data->d_buf)[4];
245 bool is_le =
ehdr.e_ident[EI_DATA] == ELFDATA2LSB;
246 uint32_t os_abi = is_le ?
htole(raw_abi) :
htobe(raw_abi);
253 fatal(
"gem5 does not support the HURD ABI.\n");
263 if (!strcmp(
".SUNW_version", e_str) || !strcmp(
".stab.index", e_str)) {
276 phdr.p_offset, phdr.p_filesz });
277 Addr uninitialized = phdr.p_memsz - phdr.p_filesz;
284 phdr.p_paddr + phdr.p_filesz, uninitialized });
287 const Addr file_start = phdr.p_offset;
288 const Addr file_end = file_start + phdr.p_filesz;
294 if (file_start <= ehdr.e_phoff && file_end >
ehdr.e_phoff)
311 if (elf_version(EV_CURRENT) == EV_NONE)
312 panic(
"wrong elf version number!");
315 Elf *
elf = elf_memory((
char *)const_cast<uint8_t *>(
321 Elf_Scn *section = elf_getscn(elf, sec_idx);
325 while (section != NULL) {
327 gelf_getshdr(section, &shdr);
329 if (shdr.sh_type == SHT_SYMTAB) {
331 Elf_Data *
data = elf_getdata(section, NULL);
332 int count = shdr.sh_size / shdr.sh_entsize;
333 DPRINTF(
Loader,
"Found Symbol Table, %d symbols present\n", count);
338 gelf_getsym(data,
i, &sym);
339 if (GELF_ST_BIND(sym.st_info) == binding) {
341 elf_strptr(elf, shdr.sh_link, sym.st_name);
342 if (sym_name && sym_name[0] !=
'$') {
344 if (symtab->
insert(value & mask, sym_name)) {
353 section = elf_getscn(elf, sec_idx);
409 if (elf_version(EV_CURRENT) == EV_NONE)
410 panic(
"wrong elf version number!");
414 elf_memory((
char *)const_cast<uint8_t *>(
imageData->data()),
420 if (gelf_getehdr(elf, &ehdr) ==0) {
421 panic(
"Not ELF, shouldn't be here");
426 Elf_Scn *section = elf_getscn(elf, sec_idx);
431 gelf_getshdr(section, &shdr);
432 sectionNames.insert(elf_strptr(elf, ehdr.e_shstrndx, shdr.sh_name));
433 section = elf_getscn(elf, ++sec_idx);
#define panic(...)
This implements a cprintf based panic() function.
bool loadSomeSymbols(SymbolTable *symtab, int binding, Addr mask, Addr base, Addr offset)
Helper functions for loadGlobalSymbols() and loadLocalSymbols().
ObjectFile * createObjectFile(const std::string &fname, bool raw)
#define fatal(...)
This implements a cprintf based fatal() function.
const std::string & name()
ElfObject(ImageFileDataPtr ifd)
ImageFileDataPtr imageData
std::shared_ptr< ImageFileData > ImageFileDataPtr
bool loadAllSymbols(SymbolTable *symtab, Addr base=0, Addr offset=0, Addr addr_mask=MaxAddr) override
void setInterpDir(const std::string &dirname)
This is the interface for setting up a base path for the elf interpreter.
uint16_t _programHeaderSize
bool insert(Addr address, std::string symbol)
bool loadWeakSymbols(SymbolTable *symtab, Addr base=0, Addr offset=0, Addr addr_mask=MaxAddr) override
MemoryImage & offset(Addr by)
bool sectionExists(std::string sec)
#define warn_if(cond,...)
Conditional warning macro that checks the supplied condition and only prints a warning if the conditi...
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
void updateBias(Addr bias_addr) override
uint16_t _programHeaderCount
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
std::string getInterpPath(const GElf_Phdr &phdr) const
void handleLoadableSegment(GElf_Phdr phdr, int seg_num)
const std::vector< Segment > & segments() const
void addSegment(const Segment &seg)
bool loadGlobalSymbols(SymbolTable *symtab, Addr base=0, Addr offset=0, Addr addr_mask=MaxAddr) override
bool loadLocalSymbols(SymbolTable *symtab, Addr base=0, Addr offset=0, Addr addr_mask=MaxAddr) override
std::set< std::string > sectionNames
T bits(T val, int first, int last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it...
const std::string to_string(sc_enc enc)