MachOYAML.h 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  1. #pragma once
  2. #ifdef __GNUC__
  3. #pragma GCC diagnostic push
  4. #pragma GCC diagnostic ignored "-Wunused-parameter"
  5. #endif
  6. //===- MachOYAML.h - Mach-O YAMLIO implementation ---------------*- C++ -*-===//
  7. //
  8. // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
  9. // See https://llvm.org/LICENSE.txt for license information.
  10. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  11. //
  12. //===----------------------------------------------------------------------===//
  13. ///
  14. /// \file
  15. /// This file declares classes for handling the YAML representation
  16. /// of Mach-O.
  17. ///
  18. //===----------------------------------------------------------------------===//
  19. #ifndef LLVM_OBJECTYAML_MACHOYAML_H
  20. #define LLVM_OBJECTYAML_MACHOYAML_H
  21. #include "llvm/ADT/StringRef.h"
  22. #include "llvm/BinaryFormat/MachO.h"
  23. #include "llvm/ObjectYAML/DWARFYAML.h"
  24. #include "llvm/ObjectYAML/YAML.h"
  25. #include "llvm/Support/YAMLTraits.h"
  26. #include <cstdint>
  27. #include <optional>
  28. #include <string>
  29. #include <vector>
  30. namespace llvm {
  31. namespace MachOYAML {
  32. struct Relocation {
  33. // Offset in the section to what is being relocated.
  34. llvm::yaml::Hex32 address;
  35. // Symbol index if r_extern == 1 else section index.
  36. uint32_t symbolnum;
  37. bool is_pcrel;
  38. // Real length = 2 ^ length.
  39. uint8_t length;
  40. bool is_extern;
  41. uint8_t type;
  42. bool is_scattered;
  43. int32_t value;
  44. };
  45. struct Section {
  46. char sectname[16];
  47. char segname[16];
  48. llvm::yaml::Hex64 addr;
  49. uint64_t size;
  50. llvm::yaml::Hex32 offset;
  51. uint32_t align;
  52. llvm::yaml::Hex32 reloff;
  53. uint32_t nreloc;
  54. llvm::yaml::Hex32 flags;
  55. llvm::yaml::Hex32 reserved1;
  56. llvm::yaml::Hex32 reserved2;
  57. llvm::yaml::Hex32 reserved3;
  58. std::optional<llvm::yaml::BinaryRef> content;
  59. std::vector<Relocation> relocations;
  60. };
  61. struct FileHeader {
  62. llvm::yaml::Hex32 magic;
  63. llvm::yaml::Hex32 cputype;
  64. llvm::yaml::Hex32 cpusubtype;
  65. llvm::yaml::Hex32 filetype;
  66. uint32_t ncmds;
  67. uint32_t sizeofcmds;
  68. llvm::yaml::Hex32 flags;
  69. llvm::yaml::Hex32 reserved;
  70. };
  71. struct LoadCommand {
  72. virtual ~LoadCommand();
  73. llvm::MachO::macho_load_command Data;
  74. std::vector<Section> Sections;
  75. std::vector<MachO::build_tool_version> Tools;
  76. std::vector<llvm::yaml::Hex8> PayloadBytes;
  77. std::string Content;
  78. uint64_t ZeroPadBytes;
  79. };
  80. struct NListEntry {
  81. uint32_t n_strx;
  82. llvm::yaml::Hex8 n_type;
  83. uint8_t n_sect;
  84. uint16_t n_desc;
  85. uint64_t n_value;
  86. };
  87. struct RebaseOpcode {
  88. MachO::RebaseOpcode Opcode;
  89. uint8_t Imm;
  90. std::vector<yaml::Hex64> ExtraData;
  91. };
  92. struct BindOpcode {
  93. MachO::BindOpcode Opcode;
  94. uint8_t Imm;
  95. std::vector<yaml::Hex64> ULEBExtraData;
  96. std::vector<int64_t> SLEBExtraData;
  97. StringRef Symbol;
  98. };
  99. struct ExportEntry {
  100. uint64_t TerminalSize = 0;
  101. uint64_t NodeOffset = 0;
  102. std::string Name;
  103. llvm::yaml::Hex64 Flags = 0;
  104. llvm::yaml::Hex64 Address = 0;
  105. llvm::yaml::Hex64 Other = 0;
  106. std::string ImportName;
  107. std::vector<MachOYAML::ExportEntry> Children;
  108. };
  109. struct DataInCodeEntry {
  110. llvm::yaml::Hex32 Offset;
  111. uint16_t Length;
  112. llvm::yaml::Hex16 Kind;
  113. };
  114. struct LinkEditData {
  115. std::vector<MachOYAML::RebaseOpcode> RebaseOpcodes;
  116. std::vector<MachOYAML::BindOpcode> BindOpcodes;
  117. std::vector<MachOYAML::BindOpcode> WeakBindOpcodes;
  118. std::vector<MachOYAML::BindOpcode> LazyBindOpcodes;
  119. MachOYAML::ExportEntry ExportTrie;
  120. std::vector<NListEntry> NameList;
  121. std::vector<StringRef> StringTable;
  122. std::vector<yaml::Hex32> IndirectSymbols;
  123. std::vector<yaml::Hex64> FunctionStarts;
  124. std::vector<DataInCodeEntry> DataInCode;
  125. std::vector<yaml::Hex8> ChainedFixups;
  126. bool isEmpty() const;
  127. };
  128. struct Object {
  129. bool IsLittleEndian;
  130. FileHeader Header;
  131. std::vector<LoadCommand> LoadCommands;
  132. std::vector<Section> Sections;
  133. LinkEditData LinkEdit;
  134. std::optional<llvm::yaml::BinaryRef> RawLinkEditSegment;
  135. DWARFYAML::Data DWARF;
  136. };
  137. struct FatHeader {
  138. llvm::yaml::Hex32 magic;
  139. uint32_t nfat_arch;
  140. };
  141. struct FatArch {
  142. llvm::yaml::Hex32 cputype;
  143. llvm::yaml::Hex32 cpusubtype;
  144. llvm::yaml::Hex64 offset;
  145. uint64_t size;
  146. uint32_t align;
  147. llvm::yaml::Hex32 reserved;
  148. };
  149. struct UniversalBinary {
  150. FatHeader Header;
  151. std::vector<FatArch> FatArchs;
  152. std::vector<Object> Slices;
  153. };
  154. } // end namespace MachOYAML
  155. } // end namespace llvm
  156. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::LoadCommand)
  157. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::Relocation)
  158. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::Section)
  159. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::RebaseOpcode)
  160. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::BindOpcode)
  161. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::ExportEntry)
  162. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::NListEntry)
  163. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::Object)
  164. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::FatArch)
  165. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachOYAML::DataInCodeEntry)
  166. LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::MachO::build_tool_version)
  167. namespace llvm {
  168. class raw_ostream;
  169. namespace yaml {
  170. template <> struct MappingTraits<MachOYAML::FileHeader> {
  171. static void mapping(IO &IO, MachOYAML::FileHeader &FileHeader);
  172. };
  173. template <> struct MappingTraits<MachOYAML::Object> {
  174. static void mapping(IO &IO, MachOYAML::Object &Object);
  175. };
  176. template <> struct MappingTraits<MachOYAML::FatHeader> {
  177. static void mapping(IO &IO, MachOYAML::FatHeader &FatHeader);
  178. };
  179. template <> struct MappingTraits<MachOYAML::FatArch> {
  180. static void mapping(IO &IO, MachOYAML::FatArch &FatArch);
  181. };
  182. template <> struct MappingTraits<MachOYAML::UniversalBinary> {
  183. static void mapping(IO &IO, MachOYAML::UniversalBinary &UniversalBinary);
  184. };
  185. template <> struct MappingTraits<MachOYAML::LoadCommand> {
  186. static void mapping(IO &IO, MachOYAML::LoadCommand &LoadCommand);
  187. };
  188. template <> struct MappingTraits<MachOYAML::LinkEditData> {
  189. static void mapping(IO &IO, MachOYAML::LinkEditData &LinkEditData);
  190. };
  191. template <> struct MappingTraits<MachOYAML::RebaseOpcode> {
  192. static void mapping(IO &IO, MachOYAML::RebaseOpcode &RebaseOpcode);
  193. };
  194. template <> struct MappingTraits<MachOYAML::BindOpcode> {
  195. static void mapping(IO &IO, MachOYAML::BindOpcode &BindOpcode);
  196. };
  197. template <> struct MappingTraits<MachOYAML::ExportEntry> {
  198. static void mapping(IO &IO, MachOYAML::ExportEntry &ExportEntry);
  199. };
  200. template <> struct MappingTraits<MachOYAML::Relocation> {
  201. static void mapping(IO &IO, MachOYAML::Relocation &R);
  202. };
  203. template <> struct MappingTraits<MachOYAML::Section> {
  204. static void mapping(IO &IO, MachOYAML::Section &Section);
  205. static std::string validate(IO &io, MachOYAML::Section &Section);
  206. };
  207. template <> struct MappingTraits<MachOYAML::NListEntry> {
  208. static void mapping(IO &IO, MachOYAML::NListEntry &NListEntry);
  209. };
  210. template <> struct MappingTraits<MachO::build_tool_version> {
  211. static void mapping(IO &IO, MachO::build_tool_version &tool);
  212. };
  213. template <> struct MappingTraits<MachOYAML::DataInCodeEntry> {
  214. static void mapping(IO &IO, MachOYAML::DataInCodeEntry &DataInCodeEntry);
  215. };
  216. #define HANDLE_LOAD_COMMAND(LCName, LCValue, LCStruct) \
  217. io.enumCase(value, #LCName, MachO::LCName);
  218. template <> struct ScalarEnumerationTraits<MachO::LoadCommandType> {
  219. static void enumeration(IO &io, MachO::LoadCommandType &value) {
  220. #include "llvm/BinaryFormat/MachO.def"
  221. io.enumFallback<Hex32>(value);
  222. }
  223. };
  224. #define ENUM_CASE(Enum) io.enumCase(value, #Enum, MachO::Enum);
  225. template <> struct ScalarEnumerationTraits<MachO::RebaseOpcode> {
  226. static void enumeration(IO &io, MachO::RebaseOpcode &value) {
  227. ENUM_CASE(REBASE_OPCODE_DONE)
  228. ENUM_CASE(REBASE_OPCODE_SET_TYPE_IMM)
  229. ENUM_CASE(REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB)
  230. ENUM_CASE(REBASE_OPCODE_ADD_ADDR_ULEB)
  231. ENUM_CASE(REBASE_OPCODE_ADD_ADDR_IMM_SCALED)
  232. ENUM_CASE(REBASE_OPCODE_DO_REBASE_IMM_TIMES)
  233. ENUM_CASE(REBASE_OPCODE_DO_REBASE_ULEB_TIMES)
  234. ENUM_CASE(REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB)
  235. ENUM_CASE(REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB)
  236. io.enumFallback<Hex8>(value);
  237. }
  238. };
  239. template <> struct ScalarEnumerationTraits<MachO::BindOpcode> {
  240. static void enumeration(IO &io, MachO::BindOpcode &value) {
  241. ENUM_CASE(BIND_OPCODE_DONE)
  242. ENUM_CASE(BIND_OPCODE_SET_DYLIB_ORDINAL_IMM)
  243. ENUM_CASE(BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB)
  244. ENUM_CASE(BIND_OPCODE_SET_DYLIB_SPECIAL_IMM)
  245. ENUM_CASE(BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM)
  246. ENUM_CASE(BIND_OPCODE_SET_TYPE_IMM)
  247. ENUM_CASE(BIND_OPCODE_SET_ADDEND_SLEB)
  248. ENUM_CASE(BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB)
  249. ENUM_CASE(BIND_OPCODE_ADD_ADDR_ULEB)
  250. ENUM_CASE(BIND_OPCODE_DO_BIND)
  251. ENUM_CASE(BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
  252. ENUM_CASE(BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED)
  253. ENUM_CASE(BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB)
  254. io.enumFallback<Hex8>(value);
  255. }
  256. };
  257. // This trait is used for 16-byte chars in Mach structures used for strings
  258. using char_16 = char[16];
  259. template <> struct ScalarTraits<char_16> {
  260. static void output(const char_16 &Val, void *, raw_ostream &Out);
  261. static StringRef input(StringRef Scalar, void *, char_16 &Val);
  262. static QuotingType mustQuote(StringRef S);
  263. };
  264. // This trait is used for UUIDs. It reads and writes them matching otool's
  265. // formatting style.
  266. using uuid_t = raw_ostream::uuid_t;
  267. template <> struct ScalarTraits<uuid_t> {
  268. static void output(const uuid_t &Val, void *, raw_ostream &Out);
  269. static StringRef input(StringRef Scalar, void *, uuid_t &Val);
  270. static QuotingType mustQuote(StringRef S);
  271. };
  272. // Load Command struct mapping traits
  273. #define LOAD_COMMAND_STRUCT(LCStruct) \
  274. template <> struct MappingTraits<MachO::LCStruct> { \
  275. static void mapping(IO &IO, MachO::LCStruct &LoadCommand); \
  276. };
  277. #include "llvm/BinaryFormat/MachO.def"
  278. // Extra structures used by load commands
  279. template <> struct MappingTraits<MachO::dylib> {
  280. static void mapping(IO &IO, MachO::dylib &LoadCommand);
  281. };
  282. template <> struct MappingTraits<MachO::fvmlib> {
  283. static void mapping(IO &IO, MachO::fvmlib &LoadCommand);
  284. };
  285. template <> struct MappingTraits<MachO::section> {
  286. static void mapping(IO &IO, MachO::section &LoadCommand);
  287. };
  288. template <> struct MappingTraits<MachO::section_64> {
  289. static void mapping(IO &IO, MachO::section_64 &LoadCommand);
  290. };
  291. } // end namespace yaml
  292. } // end namespace llvm
  293. #endif // LLVM_OBJECTYAML_MACHOYAML_H
  294. #ifdef __GNUC__
  295. #pragma GCC diagnostic pop
  296. #endif