2 // ELFManager.cpp: ELF format manager
6 // JPM = Jean-Paul Mari <djipi.mari@gmail.com>
9 // --- ---------- ------------------------------------------------------------
10 // JPM Jan./2016 Created this file and added ELF format support
11 // JPM 07/13/2017 ELF DWARF format support improvement
12 // JPM 10/20/2018 Added function name support from ELF structure
13 // JPM 03/13/2020 Added ELF & DWARF .debug* types
18 #include "libelf/libelf.h"
19 #include "libelf/gelf.h"
22 #include "ELFManager.h"
23 #include "DwarfManager.h"
26 //#define LOG_SUPPORT // Support log
30 const char *SectionName
;
45 ELFSectionType ELFTabSectionType
[] = {
46 { "", ELF_NULL_TYPE
},
47 { ".text", ELF_text_TYPE
},
48 { ".rodata", ELF_rodata_TYPE
},
49 { ".data", ELF_data_TYPE
},
50 { ".bss", ELF_bss_TYPE
},
51 { ".heap", ELF_heap_TYPE
},
52 { ".debug", ELF_debug_TYPE
},
53 { ".comment", ELF_comment_TYPE
},
54 { ".shstrtab", ELF_shstrtab_TYPE
},
55 { ".symtab", ELF_symtab_TYPE
},
56 { ".strtab", ELF_strtab_TYPE
},
57 { ".debug_abbrev", ELF_debug_abbrev_TYPE
},
58 { ".debug_aranges", ELF_debug_aranges_TYPE
},
59 { ".debug_frame", ELF_debug_frame_TYPE
},
60 { ".debug_info", ELF_debug_info_TYPE
},
61 { ".debug_line", ELF_debug_line_TYPE
},
62 { ".debug_loc", ELF_debug_loc_TYPE
},
63 { ".debug_macinfo", ELF_debug_macinfo_TYPE
},
64 { ".debug_pubnames", ELF_debug_pubnames_TYPE
},
65 { ".debug_pubtypes", ELF_debug_pubtypes_TYPE
},
66 { ".debug_ranges", ELF_debug_ranges_TYPE
},
67 { ".debug_str", ELF_debug_str_TYPE
},
68 { ".debug_types", ELF_debug_types_TYPE
}
74 void *PtrExec
; // ELF executable
76 // ELF Dwarf management
80 size_t NbELFtabStruct
;
84 char *ELFManager_GetSymbolnameFromSymbolindex(size_t Index
);
87 // ELF section type detection
88 size_t ELFManager_GetSectionType(char *SectionName
)
92 for (i
= 0; i
< ELF_END_TYPE
; i
++)
94 if (!strcmp(ELFTabSectionType
[i
].SectionName
, SectionName
))
96 return (ELFTabSectionType
[i
].SectionType
);
100 return (size_t)ELF_NO_TYPE
;
104 // ELF manager executable copy
105 void *ELFManager_ExeCopy(void *src
, size_t size
)
107 if ((PtrExec
= malloc(size
)) != NULL
)
109 memcpy(PtrExec
, src
, size
);
116 Elf
*ELFManager_MemOpen(char *PtrELFExe
, size_t Size
)
118 return(ElfMem
= elf_memory(PtrELFExe
, Size
));
122 int ELFManager_MemEnd(void)
128 err
= elf_end(ElfMem
);
136 // ELF manager Initialisation
137 void ELFManager_Init(void)
147 // ELF manager Dwarf Initialisation
148 bool ELFManager_DwarfInit(Elf
*PtrElfMem
)
150 return (ElfDwarf
= (DWARFManager_ElfInit(PtrElfMem
) == DW_DLV_OK
) ? true : false);
155 void ELFManager_Reset(void)
159 DWARFManager_Reset();
171 while (NbELFtabStruct
)
173 free(ELFtab
[--NbELFtabStruct
]);
184 // ELF manager Closing
185 void ELFManager_Close(void)
191 // Save the ELF tab in memory
192 bool ELFManager_AddTab(void *Ptr
, size_t type
)
196 if ((ELFtab
= (ELFTab
**)calloc(1, sizeof(ELFTab
*))) == NULL
)
203 if ((ELFtab
= (ELFTab
**)realloc(ELFtab
, sizeof(ELFTab
*)*(NbELFtabStruct
+1))) == NULL
)
209 if ((ELFtab
[NbELFtabStruct
] = (ELFTab
*)calloc(1, sizeof(ELFTab
))) == NULL
)
215 ELFtab
[NbELFtabStruct
]->Type
= type
;
216 ELFtab
[NbELFtabStruct
]->PtrTab
= Ptr
;
223 // Get Address from his Symbol Name
224 // Return 0 if Symbol name is not found
225 size_t ELFManager_GetAdrFromSymbolName(char *SymbolName
)
230 if (ELFtab
&& SymbolName
)
232 for (size_t i
= 0; i
< NbELFtabStruct
; i
++)
234 if ((ELFtab
[i
]->Type
== ELF_symtab_TYPE
) && ((ELFtab
[i
]->PtrDataTab
) != NULL
))
238 while ((PtrST
= gelf_getsym(ELFtab
[i
]->PtrDataTab
, j
++, &ST
)) != NULL
)
240 if (!strcmp(ELFManager_GetSymbolnameFromSymbolindex(PtrST
->st_name
), SymbolName
))
242 Adr
= PtrST
->st_value
;
253 // Get function name from his address
254 // Return NULL if function name is not found
255 char *ELFManager_GetFunctionName(size_t Adr
)
257 char *SymbolName
= NULL
;
262 for (size_t i
= 0; i
< NbELFtabStruct
; i
++)
264 if ((ELFtab
[i
]->Type
== ELF_symtab_TYPE
) && ((ELFtab
[i
]->PtrDataTab
) != NULL
))
268 while ((PtrST
= gelf_getsym(ELFtab
[i
]->PtrDataTab
, j
++, &ST
)) != NULL
)
270 if (PtrST
->st_value
== Adr
)
272 if (ELF32_ST_TYPE(PtrST
->st_info
) == STT_FUNC
)
274 SymbolName
= ELFManager_GetSymbolnameFromSymbolindex(PtrST
->st_name
);
286 // Get Symbol name from his address
287 // Return NULL if Symbol name is not found
288 char *ELFManager_GetSymbolnameFromAdr(size_t Adr
)
290 char *SymbolName
= NULL
;
295 for (size_t i
= 0; i
< NbELFtabStruct
; i
++)
297 if ((ELFtab
[i
]->Type
== ELF_symtab_TYPE
) && ((ELFtab
[i
]->PtrDataTab
) != NULL
))
301 while ((PtrST
= gelf_getsym(ELFtab
[i
]->PtrDataTab
, j
++, &ST
)) != NULL
)
303 if (PtrST
->st_value
== Adr
)
306 WriteLog("ELF: .symtab: DATA: st_info=%0x, st_name=%0x, st_other=%0x, st_shndx=%0x, st_size=%0x, st_value=%0x\n", PtrST
->st_info
, PtrST
->st_name
, PtrST
->st_other
, PtrST
->st_shndx
, PtrST
->st_size
, PtrST
->st_value
);
308 SymbolName
= ELFManager_GetSymbolnameFromSymbolindex(PtrST
->st_name
);
319 // Get Symbol name from his Symbol index
320 char *ELFManager_GetSymbolnameFromSymbolindex(size_t Index
)
324 for (size_t i
= 0; i
< NbELFtabStruct
; i
++)
326 if ((ELFtab
[i
]->Type
== ELF_strtab_TYPE
) && ((ELFtab
[i
]->PtrDataTab
) != NULL
))
328 return ((char *)(ELFtab
[i
]->PtrDataTab
->d_buf
) + Index
);