FormatUtil.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. //===- FormatUtil.cpp ----------------------------------------- *- C++ --*-===//
  2. //
  3. // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
  4. // See https://llvm.org/LICENSE.txt for license information.
  5. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  6. //
  7. //===----------------------------------------------------------------------===//
  8. #include "llvm/DebugInfo/PDB/Native/FormatUtil.h"
  9. #include "llvm/ADT/STLExtras.h"
  10. #include "llvm/ADT/StringExtras.h"
  11. #include "llvm/BinaryFormat/COFF.h"
  12. #include "llvm/DebugInfo/CodeView/CodeView.h"
  13. #include "llvm/Support/FormatAdapters.h"
  14. #include "llvm/Support/FormatVariadic.h"
  15. using namespace llvm;
  16. using namespace llvm::codeview;
  17. using namespace llvm::pdb;
  18. std::string llvm::pdb::typesetItemList(ArrayRef<std::string> Opts,
  19. uint32_t IndentLevel, uint32_t GroupSize,
  20. StringRef Sep) {
  21. std::string Result;
  22. while (!Opts.empty()) {
  23. ArrayRef<std::string> ThisGroup;
  24. ThisGroup = Opts.take_front(GroupSize);
  25. Opts = Opts.drop_front(ThisGroup.size());
  26. Result += join(ThisGroup, Sep);
  27. if (!Opts.empty()) {
  28. Result += Sep;
  29. Result += "\n";
  30. Result += std::string(formatv("{0}", fmt_repeat(' ', IndentLevel)));
  31. }
  32. }
  33. return Result;
  34. }
  35. std::string llvm::pdb::typesetStringList(uint32_t IndentLevel,
  36. ArrayRef<StringRef> Strings) {
  37. std::string Result = "[";
  38. for (const auto &S : Strings) {
  39. Result += std::string(formatv("\n{0}{1}", fmt_repeat(' ', IndentLevel), S));
  40. }
  41. Result += "]";
  42. return Result;
  43. }
  44. std::string llvm::pdb::formatChunkKind(DebugSubsectionKind Kind,
  45. bool Friendly) {
  46. if (Friendly) {
  47. switch (Kind) {
  48. RETURN_CASE(DebugSubsectionKind, None, "none");
  49. RETURN_CASE(DebugSubsectionKind, Symbols, "symbols");
  50. RETURN_CASE(DebugSubsectionKind, Lines, "lines");
  51. RETURN_CASE(DebugSubsectionKind, StringTable, "strings");
  52. RETURN_CASE(DebugSubsectionKind, FileChecksums, "checksums");
  53. RETURN_CASE(DebugSubsectionKind, FrameData, "frames");
  54. RETURN_CASE(DebugSubsectionKind, InlineeLines, "inlinee lines");
  55. RETURN_CASE(DebugSubsectionKind, CrossScopeImports, "xmi");
  56. RETURN_CASE(DebugSubsectionKind, CrossScopeExports, "xme");
  57. RETURN_CASE(DebugSubsectionKind, ILLines, "il lines");
  58. RETURN_CASE(DebugSubsectionKind, FuncMDTokenMap, "func md token map");
  59. RETURN_CASE(DebugSubsectionKind, TypeMDTokenMap, "type md token map");
  60. RETURN_CASE(DebugSubsectionKind, MergedAssemblyInput,
  61. "merged assembly input");
  62. RETURN_CASE(DebugSubsectionKind, CoffSymbolRVA, "coff symbol rva");
  63. RETURN_CASE(DebugSubsectionKind, XfgHashType, "xfg hash type");
  64. RETURN_CASE(DebugSubsectionKind, XfgHashVirtual, "xfg hash virtual");
  65. }
  66. } else {
  67. switch (Kind) {
  68. RETURN_CASE(DebugSubsectionKind, None, "none");
  69. RETURN_CASE(DebugSubsectionKind, Symbols, "DEBUG_S_SYMBOLS");
  70. RETURN_CASE(DebugSubsectionKind, Lines, "DEBUG_S_LINES");
  71. RETURN_CASE(DebugSubsectionKind, StringTable, "DEBUG_S_STRINGTABLE");
  72. RETURN_CASE(DebugSubsectionKind, FileChecksums, "DEBUG_S_FILECHKSMS");
  73. RETURN_CASE(DebugSubsectionKind, FrameData, "DEBUG_S_FRAMEDATA");
  74. RETURN_CASE(DebugSubsectionKind, InlineeLines, "DEBUG_S_INLINEELINES");
  75. RETURN_CASE(DebugSubsectionKind, CrossScopeImports,
  76. "DEBUG_S_CROSSSCOPEIMPORTS");
  77. RETURN_CASE(DebugSubsectionKind, CrossScopeExports,
  78. "DEBUG_S_CROSSSCOPEEXPORTS");
  79. RETURN_CASE(DebugSubsectionKind, ILLines, "DEBUG_S_IL_LINES");
  80. RETURN_CASE(DebugSubsectionKind, FuncMDTokenMap,
  81. "DEBUG_S_FUNC_MDTOKEN_MAP");
  82. RETURN_CASE(DebugSubsectionKind, TypeMDTokenMap,
  83. "DEBUG_S_TYPE_MDTOKEN_MAP");
  84. RETURN_CASE(DebugSubsectionKind, MergedAssemblyInput,
  85. "DEBUG_S_MERGED_ASSEMBLYINPUT");
  86. RETURN_CASE(DebugSubsectionKind, CoffSymbolRVA,
  87. "DEBUG_S_COFF_SYMBOL_RVA");
  88. RETURN_CASE(DebugSubsectionKind, XfgHashType,
  89. "DEBUG_S_XFGHASH_TYPE");
  90. RETURN_CASE(DebugSubsectionKind, XfgHashVirtual,
  91. "DEBUG_S_XFGHASH_VIRTUAL");
  92. }
  93. }
  94. return formatUnknownEnum(Kind);
  95. }
  96. std::string llvm::pdb::formatSymbolKind(SymbolKind K) {
  97. switch (uint32_t(K)) {
  98. #define SYMBOL_RECORD(EnumName, value, name) \
  99. case EnumName: \
  100. return #EnumName;
  101. #define CV_SYMBOL(EnumName, value) SYMBOL_RECORD(EnumName, value, EnumName)
  102. #include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
  103. }
  104. return formatUnknownEnum(K);
  105. }
  106. std::string llvm::pdb::formatTypeLeafKind(TypeLeafKind K) {
  107. switch (K) {
  108. #define TYPE_RECORD(EnumName, value, name) \
  109. case EnumName: \
  110. return #EnumName;
  111. #include "llvm/DebugInfo/CodeView/CodeViewTypes.def"
  112. default:
  113. return formatv("UNKNOWN RECORD ({0:X})",
  114. static_cast<std::underlying_type_t<TypeLeafKind>>(K))
  115. .str();
  116. }
  117. }
  118. std::string llvm::pdb::formatSegmentOffset(uint16_t Segment, uint32_t Offset) {
  119. return std::string(formatv("{0:4}:{1:4}", Segment, Offset));
  120. }
  121. #define PUSH_CHARACTERISTIC_FLAG(Enum, TheOpt, Value, Style, Descriptive) \
  122. PUSH_FLAG(Enum, TheOpt, Value, \
  123. ((Style == CharacteristicStyle::HeaderDefinition) ? #TheOpt \
  124. : Descriptive))
  125. #define PUSH_MASKED_CHARACTERISTIC_FLAG(Enum, Mask, TheOpt, Value, Style, \
  126. Descriptive) \
  127. PUSH_MASKED_FLAG(Enum, Mask, TheOpt, Value, \
  128. ((Style == CharacteristicStyle::HeaderDefinition) \
  129. ? #TheOpt \
  130. : Descriptive))
  131. std::string llvm::pdb::formatSectionCharacteristics(uint32_t IndentLevel,
  132. uint32_t C,
  133. uint32_t FlagsPerLine,
  134. StringRef Separator,
  135. CharacteristicStyle Style) {
  136. using SC = COFF::SectionCharacteristics;
  137. std::vector<std::string> Opts;
  138. if (C == COFF::SC_Invalid)
  139. return "invalid";
  140. if (C == 0)
  141. return "none";
  142. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_TYPE_NOLOAD, C, Style, "noload");
  143. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_TYPE_NO_PAD, C, Style, "no padding");
  144. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_CNT_CODE, C, Style, "code");
  145. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_CNT_INITIALIZED_DATA, C, Style,
  146. "initialized data");
  147. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_CNT_UNINITIALIZED_DATA, C, Style,
  148. "uninitialized data");
  149. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_LNK_OTHER, C, Style, "other");
  150. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_LNK_INFO, C, Style, "info");
  151. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_LNK_REMOVE, C, Style, "remove");
  152. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_LNK_COMDAT, C, Style, "comdat");
  153. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_GPREL, C, Style, "gp rel");
  154. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_PURGEABLE, C, Style, "purgeable");
  155. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_16BIT, C, Style, "16-bit");
  156. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_LOCKED, C, Style, "locked");
  157. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_PRELOAD, C, Style, "preload");
  158. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_1BYTES, C,
  159. Style, "1 byte align");
  160. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_2BYTES, C,
  161. Style, "2 byte align");
  162. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_4BYTES, C,
  163. Style, "4 byte align");
  164. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_8BYTES, C,
  165. Style, "8 byte align");
  166. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_16BYTES, C,
  167. Style, "16 byte align");
  168. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_32BYTES, C,
  169. Style, "32 byte align");
  170. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_64BYTES, C,
  171. Style, "64 byte align");
  172. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_128BYTES, C,
  173. Style, "128 byte align");
  174. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_256BYTES, C,
  175. Style, "256 byte align");
  176. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_512BYTES, C,
  177. Style, "512 byte align");
  178. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_1024BYTES, C,
  179. Style, "1024 byte align");
  180. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_2048BYTES, C,
  181. Style, "2048 byte align");
  182. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_4096BYTES, C,
  183. Style, "4096 byte align");
  184. PUSH_MASKED_CHARACTERISTIC_FLAG(SC, 0xF00000, IMAGE_SCN_ALIGN_8192BYTES, C,
  185. Style, "8192 byte align");
  186. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_LNK_NRELOC_OVFL, C, Style,
  187. "noreloc overflow");
  188. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_DISCARDABLE, C, Style,
  189. "discardable");
  190. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_NOT_CACHED, C, Style,
  191. "not cached");
  192. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_NOT_PAGED, C, Style, "not paged");
  193. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_SHARED, C, Style, "shared");
  194. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_EXECUTE, C, Style,
  195. "execute permissions");
  196. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_READ, C, Style,
  197. "read permissions");
  198. PUSH_CHARACTERISTIC_FLAG(SC, IMAGE_SCN_MEM_WRITE, C, Style,
  199. "write permissions");
  200. return typesetItemList(Opts, IndentLevel, FlagsPerLine, Separator);
  201. }