ELF.cpp 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. //===- ELF.cpp - ELF object file implementation ---------------------------===//
  2. //
  3. // The LLVM Compiler Infrastructure
  4. //
  5. // This file is distributed under the University of Illinois Open Source
  6. // License. See LICENSE.TXT for details.
  7. //
  8. //===----------------------------------------------------------------------===//
  9. #include "llvm/Object/ELF.h"
  10. #include "llvm/BinaryFormat/ELF.h"
  11. #include "llvm/Support/LEB128.h"
  12. using namespace llvm;
  13. using namespace object;
  14. #define STRINGIFY_ENUM_CASE(ns, name) \
  15. case ns::name: \
  16. return #name;
  17. #define ELF_RELOC(name, value) STRINGIFY_ENUM_CASE(ELF, name)
  18. StringRef llvm::object::getELFRelocationTypeName(uint32_t Machine,
  19. uint32_t Type) {
  20. switch (Machine) {
  21. case ELF::EM_X86_64:
  22. switch (Type) {
  23. #include "llvm/BinaryFormat/ELFRelocs/x86_64.def"
  24. default:
  25. break;
  26. }
  27. break;
  28. case ELF::EM_386:
  29. case ELF::EM_IAMCU:
  30. switch (Type) {
  31. #include "llvm/BinaryFormat/ELFRelocs/i386.def"
  32. default:
  33. break;
  34. }
  35. break;
  36. case ELF::EM_MIPS:
  37. switch (Type) {
  38. #include "llvm/BinaryFormat/ELFRelocs/Mips.def"
  39. default:
  40. break;
  41. }
  42. break;
  43. case ELF::EM_AARCH64:
  44. switch (Type) {
  45. #include "llvm/BinaryFormat/ELFRelocs/AArch64.def"
  46. default:
  47. break;
  48. }
  49. break;
  50. case ELF::EM_ARM:
  51. switch (Type) {
  52. #include "llvm/BinaryFormat/ELFRelocs/ARM.def"
  53. default:
  54. break;
  55. }
  56. break;
  57. case ELF::EM_ARC_COMPACT:
  58. case ELF::EM_ARC_COMPACT2:
  59. switch (Type) {
  60. #include "llvm/BinaryFormat/ELFRelocs/ARC.def"
  61. default:
  62. break;
  63. }
  64. break;
  65. case ELF::EM_AVR:
  66. switch (Type) {
  67. #include "llvm/BinaryFormat/ELFRelocs/AVR.def"
  68. default:
  69. break;
  70. }
  71. break;
  72. case ELF::EM_HEXAGON:
  73. switch (Type) {
  74. #include "llvm/BinaryFormat/ELFRelocs/Hexagon.def"
  75. default:
  76. break;
  77. }
  78. break;
  79. case ELF::EM_LANAI:
  80. switch (Type) {
  81. #include "llvm/BinaryFormat/ELFRelocs/Lanai.def"
  82. default:
  83. break;
  84. }
  85. break;
  86. case ELF::EM_PPC:
  87. switch (Type) {
  88. #include "llvm/BinaryFormat/ELFRelocs/PowerPC.def"
  89. default:
  90. break;
  91. }
  92. break;
  93. case ELF::EM_PPC64:
  94. switch (Type) {
  95. #include "llvm/BinaryFormat/ELFRelocs/PowerPC64.def"
  96. default:
  97. break;
  98. }
  99. break;
  100. case ELF::EM_RISCV:
  101. switch (Type) {
  102. #include "llvm/BinaryFormat/ELFRelocs/RISCV.def"
  103. default:
  104. break;
  105. }
  106. break;
  107. case ELF::EM_S390:
  108. switch (Type) {
  109. #include "llvm/BinaryFormat/ELFRelocs/SystemZ.def"
  110. default:
  111. break;
  112. }
  113. break;
  114. case ELF::EM_SPARC:
  115. case ELF::EM_SPARC32PLUS:
  116. case ELF::EM_SPARCV9:
  117. switch (Type) {
  118. #include "llvm/BinaryFormat/ELFRelocs/Sparc.def"
  119. default:
  120. break;
  121. }
  122. break;
  123. case ELF::EM_WEBASSEMBLY:
  124. switch (Type) {
  125. #include "llvm/BinaryFormat/ELFRelocs/WebAssembly.def"
  126. default:
  127. break;
  128. }
  129. break;
  130. case ELF::EM_AMDGPU:
  131. switch (Type) {
  132. #include "llvm/BinaryFormat/ELFRelocs/AMDGPU.def"
  133. default:
  134. break;
  135. }
  136. break;
  137. case ELF::EM_BPF:
  138. switch (Type) {
  139. #include "llvm/BinaryFormat/ELFRelocs/BPF.def"
  140. default:
  141. break;
  142. }
  143. break;
  144. default:
  145. break;
  146. }
  147. return "Unknown";
  148. }
  149. #undef ELF_RELOC
  150. StringRef llvm::object::getELFSectionTypeName(uint32_t Machine, unsigned Type) {
  151. switch (Machine) {
  152. case ELF::EM_ARM:
  153. switch (Type) {
  154. STRINGIFY_ENUM_CASE(ELF, SHT_ARM_EXIDX);
  155. STRINGIFY_ENUM_CASE(ELF, SHT_ARM_PREEMPTMAP);
  156. STRINGIFY_ENUM_CASE(ELF, SHT_ARM_ATTRIBUTES);
  157. STRINGIFY_ENUM_CASE(ELF, SHT_ARM_DEBUGOVERLAY);
  158. STRINGIFY_ENUM_CASE(ELF, SHT_ARM_OVERLAYSECTION);
  159. }
  160. break;
  161. case ELF::EM_HEXAGON:
  162. switch (Type) { STRINGIFY_ENUM_CASE(ELF, SHT_HEX_ORDERED); }
  163. break;
  164. case ELF::EM_X86_64:
  165. switch (Type) { STRINGIFY_ENUM_CASE(ELF, SHT_X86_64_UNWIND); }
  166. break;
  167. case ELF::EM_MIPS:
  168. case ELF::EM_MIPS_RS3_LE:
  169. switch (Type) {
  170. STRINGIFY_ENUM_CASE(ELF, SHT_MIPS_REGINFO);
  171. STRINGIFY_ENUM_CASE(ELF, SHT_MIPS_OPTIONS);
  172. STRINGIFY_ENUM_CASE(ELF, SHT_MIPS_ABIFLAGS);
  173. STRINGIFY_ENUM_CASE(ELF, SHT_MIPS_DWARF);
  174. }
  175. break;
  176. default:
  177. break;
  178. }
  179. switch (Type) {
  180. STRINGIFY_ENUM_CASE(ELF, SHT_NULL);
  181. STRINGIFY_ENUM_CASE(ELF, SHT_PROGBITS);
  182. STRINGIFY_ENUM_CASE(ELF, SHT_SYMTAB);
  183. STRINGIFY_ENUM_CASE(ELF, SHT_STRTAB);
  184. STRINGIFY_ENUM_CASE(ELF, SHT_RELA);
  185. STRINGIFY_ENUM_CASE(ELF, SHT_HASH);
  186. STRINGIFY_ENUM_CASE(ELF, SHT_DYNAMIC);
  187. STRINGIFY_ENUM_CASE(ELF, SHT_NOTE);
  188. STRINGIFY_ENUM_CASE(ELF, SHT_NOBITS);
  189. STRINGIFY_ENUM_CASE(ELF, SHT_REL);
  190. STRINGIFY_ENUM_CASE(ELF, SHT_SHLIB);
  191. STRINGIFY_ENUM_CASE(ELF, SHT_DYNSYM);
  192. STRINGIFY_ENUM_CASE(ELF, SHT_INIT_ARRAY);
  193. STRINGIFY_ENUM_CASE(ELF, SHT_FINI_ARRAY);
  194. STRINGIFY_ENUM_CASE(ELF, SHT_PREINIT_ARRAY);
  195. STRINGIFY_ENUM_CASE(ELF, SHT_GROUP);
  196. STRINGIFY_ENUM_CASE(ELF, SHT_SYMTAB_SHNDX);
  197. STRINGIFY_ENUM_CASE(ELF, SHT_ANDROID_REL);
  198. STRINGIFY_ENUM_CASE(ELF, SHT_ANDROID_RELA);
  199. STRINGIFY_ENUM_CASE(ELF, SHT_LLVM_ODRTAB);
  200. STRINGIFY_ENUM_CASE(ELF, SHT_GNU_ATTRIBUTES);
  201. STRINGIFY_ENUM_CASE(ELF, SHT_GNU_HASH);
  202. STRINGIFY_ENUM_CASE(ELF, SHT_GNU_verdef);
  203. STRINGIFY_ENUM_CASE(ELF, SHT_GNU_verneed);
  204. STRINGIFY_ENUM_CASE(ELF, SHT_GNU_versym);
  205. default:
  206. return "Unknown";
  207. }
  208. }
  209. template <class ELFT>
  210. Expected<std::vector<typename ELFT::Rela>>
  211. ELFFile<ELFT>::android_relas(const Elf_Shdr *Sec) const {
  212. // This function reads relocations in Android's packed relocation format,
  213. // which is based on SLEB128 and delta encoding.
  214. Expected<ArrayRef<uint8_t>> ContentsOrErr = getSectionContents(Sec);
  215. if (!ContentsOrErr)
  216. return ContentsOrErr.takeError();
  217. const uint8_t *Cur = ContentsOrErr->begin();
  218. const uint8_t *End = ContentsOrErr->end();
  219. if (ContentsOrErr->size() < 4 || Cur[0] != 'A' || Cur[1] != 'P' ||
  220. Cur[2] != 'S' || Cur[3] != '2')
  221. return createError("invalid packed relocation header");
  222. Cur += 4;
  223. const char *ErrStr = nullptr;
  224. auto ReadSLEB = [&]() -> int64_t {
  225. if (ErrStr)
  226. return 0;
  227. unsigned Len;
  228. int64_t Result = decodeSLEB128(Cur, &Len, End, &ErrStr);
  229. Cur += Len;
  230. return Result;
  231. };
  232. uint64_t NumRelocs = ReadSLEB();
  233. uint64_t Offset = ReadSLEB();
  234. uint64_t Addend = 0;
  235. if (ErrStr)
  236. return createError(ErrStr);
  237. std::vector<Elf_Rela> Relocs;
  238. Relocs.reserve(NumRelocs);
  239. while (NumRelocs) {
  240. uint64_t NumRelocsInGroup = ReadSLEB();
  241. if (NumRelocsInGroup > NumRelocs)
  242. return createError("relocation group unexpectedly large");
  243. NumRelocs -= NumRelocsInGroup;
  244. uint64_t GroupFlags = ReadSLEB();
  245. bool GroupedByInfo = GroupFlags & ELF::RELOCATION_GROUPED_BY_INFO_FLAG;
  246. bool GroupedByOffsetDelta = GroupFlags & ELF::RELOCATION_GROUPED_BY_OFFSET_DELTA_FLAG;
  247. bool GroupedByAddend = GroupFlags & ELF::RELOCATION_GROUPED_BY_ADDEND_FLAG;
  248. bool GroupHasAddend = GroupFlags & ELF::RELOCATION_GROUP_HAS_ADDEND_FLAG;
  249. uint64_t GroupOffsetDelta;
  250. if (GroupedByOffsetDelta)
  251. GroupOffsetDelta = ReadSLEB();
  252. uint64_t GroupRInfo;
  253. if (GroupedByInfo)
  254. GroupRInfo = ReadSLEB();
  255. if (GroupedByAddend && GroupHasAddend)
  256. Addend += ReadSLEB();
  257. for (uint64_t I = 0; I != NumRelocsInGroup; ++I) {
  258. Elf_Rela R;
  259. Offset += GroupedByOffsetDelta ? GroupOffsetDelta : ReadSLEB();
  260. R.r_offset = Offset;
  261. R.r_info = GroupedByInfo ? GroupRInfo : ReadSLEB();
  262. if (GroupHasAddend) {
  263. if (!GroupedByAddend)
  264. Addend += ReadSLEB();
  265. R.r_addend = Addend;
  266. } else {
  267. R.r_addend = 0;
  268. }
  269. Relocs.push_back(R);
  270. if (ErrStr)
  271. return createError(ErrStr);
  272. }
  273. if (ErrStr)
  274. return createError(ErrStr);
  275. }
  276. return Relocs;
  277. }
  278. template class llvm::object::ELFFile<ELF32LE>;
  279. template class llvm::object::ELFFile<ELF32BE>;
  280. template class llvm::object::ELFFile<ELF64LE>;
  281. template class llvm::object::ELFFile<ELF64BE>;