AArch64WinCOFFStreamer.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. //===-- AArch64WinCOFFStreamer.cpp - ARM Target WinCOFF Streamer ----*- 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 "AArch64WinCOFFStreamer.h"
  9. #include "llvm/MC/MCAsmBackend.h"
  10. #include "llvm/MC/MCAssembler.h"
  11. #include "llvm/MC/MCCodeEmitter.h"
  12. #include "llvm/MC/MCObjectWriter.h"
  13. #include "llvm/MC/MCWin64EH.h"
  14. #include "llvm/MC/MCWinCOFFStreamer.h"
  15. using namespace llvm;
  16. namespace {
  17. class AArch64WinCOFFStreamer : public MCWinCOFFStreamer {
  18. Win64EH::ARM64UnwindEmitter EHStreamer;
  19. public:
  20. AArch64WinCOFFStreamer(MCContext &C, std::unique_ptr<MCAsmBackend> AB,
  21. std::unique_ptr<MCCodeEmitter> CE,
  22. std::unique_ptr<MCObjectWriter> OW)
  23. : MCWinCOFFStreamer(C, std::move(AB), std::move(CE), std::move(OW)) {}
  24. void emitWinEHHandlerData(SMLoc Loc) override;
  25. void emitWindowsUnwindTables() override;
  26. void emitWindowsUnwindTables(WinEH::FrameInfo *Frame) override;
  27. void finishImpl() override;
  28. };
  29. void AArch64WinCOFFStreamer::emitWinEHHandlerData(SMLoc Loc) {
  30. MCStreamer::emitWinEHHandlerData(Loc);
  31. // We have to emit the unwind info now, because this directive
  32. // actually switches to the .xdata section!
  33. EHStreamer.EmitUnwindInfo(*this, getCurrentWinFrameInfo(),
  34. /* HandlerData = */ true);
  35. }
  36. void AArch64WinCOFFStreamer::emitWindowsUnwindTables(WinEH::FrameInfo *Frame) {
  37. EHStreamer.EmitUnwindInfo(*this, Frame, /* HandlerData = */ false);
  38. }
  39. void AArch64WinCOFFStreamer::emitWindowsUnwindTables() {
  40. if (!getNumWinFrameInfos())
  41. return;
  42. EHStreamer.Emit(*this);
  43. }
  44. void AArch64WinCOFFStreamer::finishImpl() {
  45. emitFrames(nullptr);
  46. emitWindowsUnwindTables();
  47. MCWinCOFFStreamer::finishImpl();
  48. }
  49. } // end anonymous namespace
  50. // Helper function to common out unwind code setup for those codes that can
  51. // belong to both prolog and epilog.
  52. // There are three types of Windows ARM64 SEH codes. They can
  53. // 1) take no operands: SEH_Nop, SEH_PrologEnd, SEH_EpilogStart, SEH_EpilogEnd
  54. // 2) take an offset: SEH_StackAlloc, SEH_SaveFPLR, SEH_SaveFPLR_X
  55. // 3) take a register and an offset/size: all others
  56. void AArch64TargetWinCOFFStreamer::emitARM64WinUnwindCode(unsigned UnwindCode,
  57. int Reg, int Offset) {
  58. auto &S = getStreamer();
  59. WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
  60. if (!CurFrame)
  61. return;
  62. auto Inst = WinEH::Instruction(UnwindCode, /*Label=*/nullptr, Reg, Offset);
  63. if (InEpilogCFI)
  64. CurFrame->EpilogMap[CurrentEpilog].Instructions.push_back(Inst);
  65. else
  66. CurFrame->Instructions.push_back(Inst);
  67. }
  68. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIAllocStack(unsigned Size) {
  69. unsigned Op = Win64EH::UOP_AllocSmall;
  70. if (Size >= 16384)
  71. Op = Win64EH::UOP_AllocLarge;
  72. else if (Size >= 512)
  73. Op = Win64EH::UOP_AllocMedium;
  74. emitARM64WinUnwindCode(Op, -1, Size);
  75. }
  76. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveR19R20X(int Offset) {
  77. emitARM64WinUnwindCode(Win64EH::UOP_SaveR19R20X, -1, Offset);
  78. }
  79. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFPLR(int Offset) {
  80. emitARM64WinUnwindCode(Win64EH::UOP_SaveFPLR, -1, Offset);
  81. }
  82. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFPLRX(int Offset) {
  83. emitARM64WinUnwindCode(Win64EH::UOP_SaveFPLRX, -1, Offset);
  84. }
  85. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveReg(unsigned Reg,
  86. int Offset) {
  87. assert(Offset >= 0 && Offset <= 504 &&
  88. "Offset for save reg should be >= 0 && <= 504");
  89. emitARM64WinUnwindCode(Win64EH::UOP_SaveReg, Reg, Offset);
  90. }
  91. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveRegX(unsigned Reg,
  92. int Offset) {
  93. emitARM64WinUnwindCode(Win64EH::UOP_SaveRegX, Reg, Offset);
  94. }
  95. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveRegP(unsigned Reg,
  96. int Offset) {
  97. emitARM64WinUnwindCode(Win64EH::UOP_SaveRegP, Reg, Offset);
  98. }
  99. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveRegPX(unsigned Reg,
  100. int Offset) {
  101. emitARM64WinUnwindCode(Win64EH::UOP_SaveRegPX, Reg, Offset);
  102. }
  103. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveLRPair(unsigned Reg,
  104. int Offset) {
  105. emitARM64WinUnwindCode(Win64EH::UOP_SaveLRPair, Reg, Offset);
  106. }
  107. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFReg(unsigned Reg,
  108. int Offset) {
  109. assert(Offset >= 0 && Offset <= 504 &&
  110. "Offset for save reg should be >= 0 && <= 504");
  111. emitARM64WinUnwindCode(Win64EH::UOP_SaveFReg, Reg, Offset);
  112. }
  113. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFRegX(unsigned Reg,
  114. int Offset) {
  115. emitARM64WinUnwindCode(Win64EH::UOP_SaveFRegX, Reg, Offset);
  116. }
  117. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFRegP(unsigned Reg,
  118. int Offset) {
  119. emitARM64WinUnwindCode(Win64EH::UOP_SaveFRegP, Reg, Offset);
  120. }
  121. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFRegPX(unsigned Reg,
  122. int Offset) {
  123. emitARM64WinUnwindCode(Win64EH::UOP_SaveFRegPX, Reg, Offset);
  124. }
  125. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISetFP() {
  126. emitARM64WinUnwindCode(Win64EH::UOP_SetFP, -1, 0);
  127. }
  128. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIAddFP(unsigned Offset) {
  129. assert(Offset <= 2040 && "UOP_AddFP must have offset <= 2040");
  130. emitARM64WinUnwindCode(Win64EH::UOP_AddFP, -1, Offset);
  131. }
  132. void AArch64TargetWinCOFFStreamer::emitARM64WinCFINop() {
  133. emitARM64WinUnwindCode(Win64EH::UOP_Nop, -1, 0);
  134. }
  135. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveNext() {
  136. emitARM64WinUnwindCode(Win64EH::UOP_SaveNext, -1, 0);
  137. }
  138. // The functions below handle opcodes that can end up in either a prolog or
  139. // an epilog, but not both.
  140. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIPrologEnd() {
  141. auto &S = getStreamer();
  142. WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
  143. if (!CurFrame)
  144. return;
  145. MCSymbol *Label = S.emitCFILabel();
  146. CurFrame->PrologEnd = Label;
  147. WinEH::Instruction Inst =
  148. WinEH::Instruction(Win64EH::UOP_End, /*Label=*/nullptr, -1, 0);
  149. auto it = CurFrame->Instructions.begin();
  150. CurFrame->Instructions.insert(it, Inst);
  151. }
  152. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIEpilogStart() {
  153. auto &S = getStreamer();
  154. WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
  155. if (!CurFrame)
  156. return;
  157. InEpilogCFI = true;
  158. CurrentEpilog = S.emitCFILabel();
  159. }
  160. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIEpilogEnd() {
  161. auto &S = getStreamer();
  162. WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
  163. if (!CurFrame)
  164. return;
  165. InEpilogCFI = false;
  166. WinEH::Instruction Inst =
  167. WinEH::Instruction(Win64EH::UOP_End, /*Label=*/nullptr, -1, 0);
  168. CurFrame->EpilogMap[CurrentEpilog].Instructions.push_back(Inst);
  169. MCSymbol *Label = S.emitCFILabel();
  170. CurFrame->EpilogMap[CurrentEpilog].End = Label;
  171. CurrentEpilog = nullptr;
  172. }
  173. void AArch64TargetWinCOFFStreamer::emitARM64WinCFITrapFrame() {
  174. emitARM64WinUnwindCode(Win64EH::UOP_TrapFrame, -1, 0);
  175. }
  176. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIMachineFrame() {
  177. emitARM64WinUnwindCode(Win64EH::UOP_PushMachFrame, -1, 0);
  178. }
  179. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIContext() {
  180. emitARM64WinUnwindCode(Win64EH::UOP_Context, -1, 0);
  181. }
  182. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIClearUnwoundToCall() {
  183. emitARM64WinUnwindCode(Win64EH::UOP_ClearUnwoundToCall, -1, 0);
  184. }
  185. void AArch64TargetWinCOFFStreamer::emitARM64WinCFIPACSignLR() {
  186. emitARM64WinUnwindCode(Win64EH::UOP_PACSignLR, -1, 0);
  187. }
  188. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegI(unsigned Reg,
  189. int Offset) {
  190. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegI, Reg, Offset);
  191. }
  192. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegIP(unsigned Reg,
  193. int Offset) {
  194. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegIP, Reg, Offset);
  195. }
  196. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegD(unsigned Reg,
  197. int Offset) {
  198. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegD, Reg, Offset);
  199. }
  200. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegDP(unsigned Reg,
  201. int Offset) {
  202. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegDP, Reg, Offset);
  203. }
  204. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegQ(unsigned Reg,
  205. int Offset) {
  206. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegQ, Reg, Offset);
  207. }
  208. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegQP(unsigned Reg,
  209. int Offset) {
  210. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegQP, Reg, Offset);
  211. }
  212. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegIX(unsigned Reg,
  213. int Offset) {
  214. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegIX, Reg, Offset);
  215. }
  216. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegIPX(unsigned Reg,
  217. int Offset) {
  218. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegIPX, Reg, Offset);
  219. }
  220. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegDX(unsigned Reg,
  221. int Offset) {
  222. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegDX, Reg, Offset);
  223. }
  224. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegDPX(unsigned Reg,
  225. int Offset) {
  226. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegDPX, Reg, Offset);
  227. }
  228. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegQX(unsigned Reg,
  229. int Offset) {
  230. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegQX, Reg, Offset);
  231. }
  232. void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveAnyRegQPX(unsigned Reg,
  233. int Offset) {
  234. emitARM64WinUnwindCode(Win64EH::UOP_SaveAnyRegQPX, Reg, Offset);
  235. }
  236. MCWinCOFFStreamer *llvm::createAArch64WinCOFFStreamer(
  237. MCContext &Context, std::unique_ptr<MCAsmBackend> MAB,
  238. std::unique_ptr<MCObjectWriter> OW, std::unique_ptr<MCCodeEmitter> Emitter,
  239. bool RelaxAll, bool IncrementalLinkerCompatible) {
  240. auto *S = new AArch64WinCOFFStreamer(Context, std::move(MAB),
  241. std::move(Emitter), std::move(OW));
  242. S->getAssembler().setIncrementalLinkerCompatible(IncrementalLinkerCompatible);
  243. return S;
  244. }