#include "AArch64WinCOFFStreamer.h"
#include "llvm/MC/MCAsmBackend.h"
#include "llvm/MC/MCAssembler.h"
#include "llvm/MC/MCCodeEmitter.h"
#include "llvm/MC/MCObjectWriter.h"
#include "llvm/MC/MCWin64EH.h"
#include "llvm/MC/MCWinCOFFStreamer.h"
using namespace llvm;
namespace {
class AArch64WinCOFFStreamer : public MCWinCOFFStreamer {
Win64EH::ARM64UnwindEmitter EHStreamer;
public:
AArch64WinCOFFStreamer(MCContext &C, std::unique_ptr<MCAsmBackend> AB,
std::unique_ptr<MCCodeEmitter> CE,
std::unique_ptr<MCObjectWriter> OW)
: MCWinCOFFStreamer(C, std::move(AB), std::move(CE), std::move(OW)) {}
void emitWinEHHandlerData(SMLoc Loc) override;
void emitWindowsUnwindTables() override;
void emitWindowsUnwindTables(WinEH::FrameInfo *Frame) override;
void finishImpl() override;
};
void AArch64WinCOFFStreamer::emitWinEHHandlerData(SMLoc Loc) {
MCStreamer::emitWinEHHandlerData(Loc);
EHStreamer.EmitUnwindInfo(*this, getCurrentWinFrameInfo(),
true);
}
void AArch64WinCOFFStreamer::emitWindowsUnwindTables(WinEH::FrameInfo *Frame) {
EHStreamer.EmitUnwindInfo(*this, Frame, false);
}
void AArch64WinCOFFStreamer::emitWindowsUnwindTables() {
if (!getNumWinFrameInfos())
return;
EHStreamer.Emit(*this);
}
void AArch64WinCOFFStreamer::finishImpl() {
emitFrames(nullptr);
emitWindowsUnwindTables();
MCWinCOFFStreamer::finishImpl();
}
}
void AArch64TargetWinCOFFStreamer::emitARM64WinUnwindCode(unsigned UnwindCode,
int Reg, int Offset) {
auto &S = getStreamer();
WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
if (!CurFrame)
return;
auto Inst = WinEH::Instruction(UnwindCode, nullptr, Reg, Offset);
if (InEpilogCFI)
CurFrame->EpilogMap[CurrentEpilog].Instructions.push_back(Inst);
else
CurFrame->Instructions.push_back(Inst);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIAllocStack(unsigned Size) {
unsigned Op = Win64EH::UOP_AllocSmall;
if (Size >= 16384)
Op = Win64EH::UOP_AllocLarge;
else if (Size >= 512)
Op = Win64EH::UOP_AllocMedium;
emitARM64WinUnwindCode(Op, -1, Size);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveR19R20X(int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveR19R20X, -1, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFPLR(int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveFPLR, -1, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFPLRX(int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveFPLRX, -1, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveReg(unsigned Reg,
int Offset) {
assert(Offset >= 0 && Offset <= 504 &&
"Offset for save reg should be >= 0 && <= 504");
emitARM64WinUnwindCode(Win64EH::UOP_SaveReg, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveRegX(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveRegX, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveRegP(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveRegP, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveRegPX(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveRegPX, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveLRPair(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveLRPair, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFReg(unsigned Reg,
int Offset) {
assert(Offset >= 0 && Offset <= 504 &&
"Offset for save reg should be >= 0 && <= 504");
emitARM64WinUnwindCode(Win64EH::UOP_SaveFReg, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFRegX(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveFRegX, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFRegP(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveFRegP, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveFRegPX(unsigned Reg,
int Offset) {
emitARM64WinUnwindCode(Win64EH::UOP_SaveFRegPX, Reg, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISetFP() {
emitARM64WinUnwindCode(Win64EH::UOP_SetFP, -1, 0);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIAddFP(unsigned Offset) {
assert(Offset <= 2040 && "UOP_AddFP must have offset <= 2040");
emitARM64WinUnwindCode(Win64EH::UOP_AddFP, -1, Offset);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFINop() {
emitARM64WinUnwindCode(Win64EH::UOP_Nop, -1, 0);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFISaveNext() {
emitARM64WinUnwindCode(Win64EH::UOP_SaveNext, -1, 0);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIPrologEnd() {
auto &S = getStreamer();
WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
if (!CurFrame)
return;
MCSymbol *Label = S.emitCFILabel();
CurFrame->PrologEnd = Label;
WinEH::Instruction Inst =
WinEH::Instruction(Win64EH::UOP_End, nullptr, -1, 0);
auto it = CurFrame->Instructions.begin();
CurFrame->Instructions.insert(it, Inst);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIEpilogStart() {
auto &S = getStreamer();
WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
if (!CurFrame)
return;
InEpilogCFI = true;
CurrentEpilog = S.emitCFILabel();
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIEpilogEnd() {
auto &S = getStreamer();
WinEH::FrameInfo *CurFrame = S.EnsureValidWinFrameInfo(SMLoc());
if (!CurFrame)
return;
InEpilogCFI = false;
WinEH::Instruction Inst =
WinEH::Instruction(Win64EH::UOP_End, nullptr, -1, 0);
CurFrame->EpilogMap[CurrentEpilog].Instructions.push_back(Inst);
CurrentEpilog = nullptr;
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFITrapFrame() {
emitARM64WinUnwindCode(Win64EH::UOP_TrapFrame, -1, 0);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIMachineFrame() {
emitARM64WinUnwindCode(Win64EH::UOP_PushMachFrame, -1, 0);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIContext() {
emitARM64WinUnwindCode(Win64EH::UOP_Context, -1, 0);
}
void AArch64TargetWinCOFFStreamer::emitARM64WinCFIClearUnwoundToCall() {
emitARM64WinUnwindCode(Win64EH::UOP_ClearUnwoundToCall, -1, 0);
}
MCWinCOFFStreamer *llvm::createAArch64WinCOFFStreamer(
MCContext &Context, std::unique_ptr<MCAsmBackend> MAB,
std::unique_ptr<MCObjectWriter> OW, std::unique_ptr<MCCodeEmitter> Emitter,
bool RelaxAll, bool IncrementalLinkerCompatible) {
auto *S = new AArch64WinCOFFStreamer(Context, std::move(MAB),
std::move(Emitter), std::move(OW));
S->getAssembler().setIncrementalLinkerCompatible(IncrementalLinkerCompatible);
return S;
}