42class X86AlignBranchKind {
44 uint8_t AlignBranchKind = 0;
47 void operator=(
const std::string &Val) {
51 StringRef(Val).split(BranchTypes,
'+', -1,
false);
52 for (
auto BranchType : BranchTypes) {
53 if (BranchType ==
"fused")
55 else if (BranchType ==
"jcc")
57 else if (BranchType ==
"jmp")
59 else if (BranchType ==
"call")
61 else if (BranchType ==
"ret")
63 else if (BranchType ==
"indirect")
67 <<
" to -x86-align-branch=; each element must be one of: fused, "
68 "jcc, jmp, call, ret, indirect.(plus separated)\n";
73 operator uint8_t()
const {
return AlignBranchKind; }
77X86AlignBranchKind X86AlignBranchKindLoc;
80 "x86-align-branch-boundary",
cl::init(0),
82 "Control how the assembler should align branches with NOP. If the "
83 "boundary's size is not 0, it should be a power of 2 and no less "
84 "than 32. Branches will be aligned to prevent from being across or "
85 "against the boundary of specified size. The default value 0 does not "
91 "Specify types of branches to align (plus separated list of types):"
92 "\njcc indicates conditional jumps"
93 "\nfused indicates fused conditional jumps"
94 "\njmp indicates direct unconditional jumps"
95 "\ncall indicates direct and indirect calls"
96 "\nret indicates rets"
97 "\nindirect indicates indirect unconditional jumps"),
101 "x86-branches-within-32B-boundaries",
cl::init(
false),
103 "Align selected instructions to mitigate negative performance impact "
104 "of Intel's micro code update for errata skx102. May break "
105 "assumptions about labels corresponding to particular instructions, "
106 "and should be used with caution."));
109 "x86-pad-max-prefix-size",
cl::init(0),
110 cl::desc(
"Maximum number of prefixes to use for padding"));
114 cl::desc(
"Pad previous instructions to implement align directives"));
118 cl::desc(
"Pad previous instructions to implement branch alignment"));
121 const MCSubtargetInfo &STI;
122 std::unique_ptr<const MCInstrInfo> MCII;
123 X86AlignBranchKind AlignBranchType;
125 unsigned TargetPrefixMax = 0;
128 unsigned PrevInstOpcode = 0;
129 MCBoundaryAlignFragment *PendingBA =
nullptr;
130 std::pair<MCFragment *, size_t> PrevInstPosition;
132 uint8_t determinePaddingPrefix(
const MCInst &Inst)
const;
133 bool isMacroFused(
const MCInst &Cmp,
const MCInst &Jcc)
const;
134 bool needAlign(
const MCInst &Inst)
const;
135 bool canPadBranches(MCObjectStreamer &OS)
const;
136 bool canPadInst(
const MCInst &Inst, MCObjectStreamer &OS)
const;
139 X86AsmBackend(
const Target &
T,
const MCSubtargetInfo &STI)
142 if (X86AlignBranchWithin32BBoundaries) {
153 if (X86AlignBranchBoundary.getNumOccurrences())
155 if (X86AlignBranch.getNumOccurrences())
156 AlignBranchType = X86AlignBranchKindLoc;
157 if (X86PadMaxPrefixSize.getNumOccurrences())
158 TargetPrefixMax = X86PadMaxPrefixSize;
162 AllowEnhancedRelaxation =
163 AllowAutoPadding && TargetPrefixMax != 0 && X86PadForBranchAlign;
166 void emitInstructionBegin(MCObjectStreamer &OS,
const MCInst &Inst,
167 const MCSubtargetInfo &STI);
168 void emitInstructionEnd(MCObjectStreamer &OS,
const MCInst &Inst);
171 std::optional<MCFixupKind>
getFixupKind(StringRef Name)
const override;
173 MCFixupKindInfo getFixupKindInfo(
MCFixupKind Kind)
const override;
175 std::optional<bool> evaluateFixup(
const MCFragment &, MCFixup &, MCValue &,
176 uint64_t &)
override;
177 void applyFixup(
const MCFragment &,
const MCFixup &,
const MCValue &Target,
178 uint8_t *
Data, uint64_t
Value,
bool IsResolved)
override;
181 const MCSubtargetInfo &STI)
const override;
183 bool fixupNeedsRelaxationAdvanced(
const MCFragment &,
const MCFixup &,
184 const MCValue &, uint64_t,
185 bool)
const override;
187 void relaxInstruction(MCInst &Inst,
188 const MCSubtargetInfo &STI)
const override;
190 bool padInstructionViaRelaxation(MCFragment &RF, MCCodeEmitter &
Emitter,
191 unsigned &RemainingSize)
const;
193 bool padInstructionViaPrefix(MCFragment &RF, MCCodeEmitter &
Emitter,
194 unsigned &RemainingSize)
const;
196 bool padInstructionEncoding(MCFragment &RF, MCCodeEmitter &
Emitter,
197 unsigned &RemainingSize)
const;
199 bool finishLayout()
const override;
201 unsigned getMaximumNopSize(
const MCSubtargetInfo &STI)
const override;
203 bool writeNopData(raw_ostream &OS, uint64_t
Count,
204 const MCSubtargetInfo *STI)
const override;
209 return Opcode == X86::JCC_1 || Opcode == X86::JMP_1;
213 bool Is16BitMode =
false) {
218 return (Is16BitMode) ? X86::JCC_2 : X86::JCC_4;
220 return (Is16BitMode) ? X86::JMP_2 : X86::JMP_4;
225 unsigned Opcode =
MI.getOpcode();
232 unsigned Opcode =
MI.getOpcode();
239 MI.getOperand(
Desc.getNumOperands() - 1).getImm());
247 return classifySecondCondCodeInMacroFusion(CC);
252 unsigned Opcode =
MI.getOpcode();
257 if (MemoryOperand < 0)
260 MCRegister BaseReg =
MI.getOperand(BaseRegNum).getReg();
261 return (BaseReg == X86::RIP);
289uint8_t X86AsmBackend::determinePaddingPrefix(
const MCInst &Inst)
const {
291 "Prefixes can be added only in 32-bit or 64-bit mode.");
293 uint64_t TSFlags =
Desc.TSFlags;
297 if (MemoryOperand != -1)
300 MCRegister SegmentReg;
301 if (MemoryOperand >= 0) {
334 if (MemoryOperand >= 0) {
337 if (BaseReg == X86::ESP || BaseReg == X86::EBP)
344bool X86AsmBackend::isMacroFused(
const MCInst &Cmp,
const MCInst &Jcc)
const {
345 const MCInstrDesc &InstDesc = MCII->get(Jcc.
getOpcode());
359 for (
auto &Operand :
MI) {
360 if (!Operand.isExpr())
362 const MCExpr &Expr = *Operand.getExpr();
374 switch (InstOpcode) {
393bool X86AsmBackend::canPadInst(
const MCInst &Inst, MCObjectStreamer &OS)
const {
404 if (
isPrefix(PrevInstOpcode, *MCII))
418 Offset != PrevInstPosition.second))
424bool X86AsmBackend::canPadBranches(MCObjectStreamer &OS)
const {
427 assert(allowAutoPadding() &&
"incorrect initialization!");
441bool X86AsmBackend::needAlign(
const MCInst &Inst)
const {
443 return (
Desc.isConditionalBranch() &&
445 (
Desc.isUnconditionalBranch() &&
449 (
Desc.isIndirectBranch() &&
456 if (
LLVM_LIKELY(!AutoPadding && !X86PadForAlign)) {
457 S.MCObjectStreamer::emitInstruction(Inst, STI);
461 auto &Backend =
static_cast<X86AsmBackend &
>(S.
getAssembler().getBackend());
462 Backend.emitInstructionBegin(S, Inst, STI);
463 S.MCObjectStreamer::emitInstruction(Inst, STI);
464 Backend.emitInstructionEnd(S, Inst);
470 bool CanPadInst = canPadInst(Inst, OS);
474 if (!canPadBranches(OS))
478 if (!isMacroFused(PrevInst, Inst))
490 auto *NextFragment = PendingBA->
getNext();
491 assert(NextFragment &&
"NextFragment should not be null");
530void X86AsmBackend::emitInstructionEnd(MCObjectStreamer &OS,
531 const MCInst &Inst) {
537 if (!canPadBranches(OS))
543 if (!needAlign(Inst) || !PendingBA)
560std::optional<MCFixupKind> X86AsmBackend::getFixupKind(StringRef Name)
const {
564 Type = llvm::StringSwitch<unsigned>(Name)
565#define ELF_RELOC(X, Y) .Case(#X, Y)
566#include "llvm/BinaryFormat/ELFRelocs/x86_64.def"
568 .Case(
"BFD_RELOC_NONE", ELF::R_X86_64_NONE)
569 .Case(
"BFD_RELOC_8", ELF::R_X86_64_8)
570 .Case(
"BFD_RELOC_16", ELF::R_X86_64_16)
571 .Case(
"BFD_RELOC_32", ELF::R_X86_64_32)
572 .Case(
"BFD_RELOC_64", ELF::R_X86_64_64)
575 Type = llvm::StringSwitch<unsigned>(Name)
576#define ELF_RELOC(X, Y) .Case(#X, Y)
577#include "llvm/BinaryFormat/ELFRelocs/i386.def"
579 .Case(
"BFD_RELOC_NONE", ELF::R_386_NONE)
580 .Case(
"BFD_RELOC_8", ELF::R_386_8)
581 .Case(
"BFD_RELOC_16", ELF::R_386_16)
582 .Case(
"BFD_RELOC_32", ELF::R_386_32)
592MCFixupKindInfo X86AsmBackend::getFixupKindInfo(
MCFixupKind Kind)
const {
595 {
"reloc_riprel_4byte", 0, 32, 0},
596 {
"reloc_riprel_4byte_movq_load", 0, 32, 0},
597 {
"reloc_riprel_4byte_movq_load_rex2", 0, 32, 0},
598 {
"reloc_riprel_4byte_relax", 0, 32, 0},
599 {
"reloc_riprel_4byte_relax_rex", 0, 32, 0},
600 {
"reloc_riprel_4byte_relax_rex2", 0, 32, 0},
601 {
"reloc_riprel_4byte_relax_evex", 0, 32, 0},
602 {
"reloc_signed_4byte", 0, 32, 0},
603 {
"reloc_signed_4byte_relax", 0, 32, 0},
604 {
"reloc_global_offset_table", 0, 32, 0},
605 {
"reloc_branch_4byte_pcrel", 0, 32, 0},
659std::optional<bool> X86AsmBackend::evaluateFixup(
const MCFragment &,
662 if (
Fixup.isPCRel()) {
663 switch (
Fixup.getKind()) {
686void X86AsmBackend::applyFixup(
const MCFragment &
F,
const MCFixup &
Fixup,
687 const MCValue &Target, uint8_t *
Data,
688 uint64_t
Value,
bool IsResolved) {
691 if (
Target.getSpecifier())
693 maybeAddReloc(
F,
Fixup, Target,
Value, IsResolved);
716 "value of " + Twine(int64_t(
Value)) +
717 " is too large for field of " + Twine(
Size) +
718 (
Size == 1 ?
" byte" :
" bytes"));
721 for (
unsigned i = 0; i !=
Size; ++i)
725bool X86AsmBackend::mayNeedRelaxation(
unsigned Opcode,
727 const MCSubtargetInfo &STI)
const {
728 unsigned SkipOperands = X86::isCCMPCC(Opcode) ? 2 : 0;
731 Operands[Operands.
size() - 1 - SkipOperands].isExpr());
734bool X86AsmBackend::fixupNeedsRelaxationAdvanced(
const MCFragment &,
735 const MCFixup &
Fixup,
736 const MCValue &Target,
738 bool Resolved)
const {
755void X86AsmBackend::relaxInstruction(MCInst &Inst,
756 const MCSubtargetInfo &STI)
const {
758 bool Is16BitMode = STI.
hasFeature(X86::Is16Bit);
764bool X86AsmBackend::padInstructionViaPrefix(MCFragment &RF,
766 unsigned &RemainingSize)
const {
781 const unsigned MaxPossiblePad = std::min(15 - OldSize, RemainingSize);
782 const unsigned RemainingPrefixSize = [&]() ->
unsigned {
783 SmallString<15>
Code;
785 assert(
Code.size() < 15 &&
"The number of prefixes must be less than 15.");
792 unsigned ExistingPrefixSize =
Code.size();
793 if (TargetPrefixMax <= ExistingPrefixSize)
795 return TargetPrefixMax - ExistingPrefixSize;
797 const unsigned PrefixBytesToAdd =
798 std::min(MaxPossiblePad, RemainingPrefixSize);
799 if (PrefixBytesToAdd == 0)
804 SmallString<256>
Code;
805 Code.append(PrefixBytesToAdd, Prefix);
811 F.setOffset(PrefixBytesToAdd +
F.getOffset());
813 RemainingSize -= PrefixBytesToAdd;
817bool X86AsmBackend::padInstructionViaRelaxation(MCFragment &RF,
819 unsigned &RemainingSize)
const {
830 SmallString<15>
Code;
833 const unsigned NewSize =
Code.size();
834 assert(NewSize >= OldSize &&
"size decrease during relaxation?");
835 unsigned Delta = NewSize - OldSize;
836 if (Delta > RemainingSize)
841 RemainingSize -= Delta;
845bool X86AsmBackend::padInstructionEncoding(MCFragment &RF,
847 unsigned &RemainingSize)
const {
849 if (RemainingSize != 0)
850 Changed |= padInstructionViaRelaxation(RF,
Emitter, RemainingSize);
851 if (RemainingSize != 0)
856bool X86AsmBackend::finishLayout()
const {
863 if (!X86PadForAlign && !X86PadForBranchAlign)
869 DenseSet<MCFragment *> LabeledFragments;
870 for (
const MCSymbol &S :
Asm->symbols())
871 LabeledFragments.
insert(S.getFragment());
874 for (MCSection &Sec : *Asm) {
879 for (MCSection::iterator
I = Sec.begin(), IE = Sec.end();
I != IE; ++
I) {
882 if (LabeledFragments.
count(&
F))
894 auto canHandle = [](MCFragment &
F) ->
bool {
895 switch (
F.getKind()) {
899 return X86PadForAlign;
901 return X86PadForBranchAlign;
914 unsigned RemainingSize =
Asm->computeFragmentSize(
F) -
F.getFixedSize();
915 while (!Relaxable.
empty() && RemainingSize != 0) {
920 Changed |= padInstructionEncoding(RF,
Asm->getEmitter(), RemainingSize);
940 const MCFragment *LastFragment = BF->getLastFragment();
943 while (&*
I != LastFragment)
952unsigned X86AsmBackend::getMaximumNopSize(
const MCSubtargetInfo &STI)
const {
972bool X86AsmBackend::writeNopData(raw_ostream &OS, uint64_t
Count,
973 const MCSubtargetInfo *STI)
const {
974 static const char Nops32Bit[10][11] = {
984 "\x0f\x1f\x44\x00\x00",
986 "\x66\x0f\x1f\x44\x00\x00",
988 "\x0f\x1f\x80\x00\x00\x00\x00",
990 "\x0f\x1f\x84\x00\x00\x00\x00\x00",
992 "\x66\x0f\x1f\x84\x00\x00\x00\x00\x00",
994 "\x66\x2e\x0f\x1f\x84\x00\x00\x00\x00\x00",
998 static const char Nops16Bit[4][11] = {
1009 const char(*Nops)[11] =
1010 STI->
hasFeature(X86::Is16Bit) ? Nops16Bit : Nops32Bit;
1012 uint64_t MaxNopLength = (uint64_t)getMaximumNopSize(*STI);
1017 const uint8_t ThisNopLength = (uint8_t) std::min(
Count, MaxNopLength);
1018 const uint8_t Prefixes = ThisNopLength <= 10 ? 0 : ThisNopLength - 10;
1019 for (uint8_t i = 0; i < Prefixes; i++)
1021 const uint8_t Rest = ThisNopLength - Prefixes;
1023 OS.
write(Nops[Rest - 1], Rest);
1024 Count -= ThisNopLength;
1025 }
while (
Count != 0);
1034class ELFX86AsmBackend :
public X86AsmBackend {
1037 ELFX86AsmBackend(
const Target &
T, uint8_t OSABI,
const MCSubtargetInfo &STI)
1038 : X86AsmBackend(
T, STI), OSABI(OSABI) {}
1041class ELFX86_32AsmBackend :
public ELFX86AsmBackend {
1043 ELFX86_32AsmBackend(
const Target &
T, uint8_t OSABI,
1044 const MCSubtargetInfo &STI)
1045 : ELFX86AsmBackend(
T, OSABI, STI) {}
1047 std::unique_ptr<MCObjectTargetWriter>
1048 createObjectTargetWriter()
const override {
1053class ELFX86_X32AsmBackend :
public ELFX86AsmBackend {
1055 ELFX86_X32AsmBackend(
const Target &
T, uint8_t OSABI,
1056 const MCSubtargetInfo &STI)
1057 : ELFX86AsmBackend(
T, OSABI, STI) {}
1059 std::unique_ptr<MCObjectTargetWriter>
1060 createObjectTargetWriter()
const override {
1066class ELFX86_IAMCUAsmBackend :
public ELFX86AsmBackend {
1068 ELFX86_IAMCUAsmBackend(
const Target &
T, uint8_t OSABI,
1069 const MCSubtargetInfo &STI)
1070 : ELFX86AsmBackend(
T, OSABI, STI) {}
1072 std::unique_ptr<MCObjectTargetWriter>
1073 createObjectTargetWriter()
const override {
1079class ELFX86_64AsmBackend :
public ELFX86AsmBackend {
1081 ELFX86_64AsmBackend(
const Target &
T, uint8_t OSABI,
1082 const MCSubtargetInfo &STI)
1083 : ELFX86AsmBackend(
T, OSABI, STI) {}
1085 std::unique_ptr<MCObjectTargetWriter>
1086 createObjectTargetWriter()
const override {
1091class WindowsX86AsmBackend :
public X86AsmBackend {
1095 WindowsX86AsmBackend(
const Target &
T,
bool is64Bit,
1096 const MCSubtargetInfo &STI)
1097 : X86AsmBackend(
T, STI)
1101 std::optional<MCFixupKind>
getFixupKind(StringRef Name)
const override {
1102 return StringSwitch<std::optional<MCFixupKind>>(
Name)
1109 std::unique_ptr<MCObjectTargetWriter>
1110 createObjectTargetWriter()
const override {
1118 enum CompactUnwindEncodings {
1121 UNWIND_MODE_BP_FRAME = 0x01000000,
1124 UNWIND_MODE_STACK_IMMD = 0x02000000,
1127 UNWIND_MODE_STACK_IND = 0x03000000,
1130 UNWIND_MODE_DWARF = 0x04000000,
1133 UNWIND_BP_FRAME_REGISTERS = 0x00007FFF,
1136 UNWIND_FRAMELESS_STACK_REG_PERMUTATION = 0x000003FF
1141class DarwinX86AsmBackend :
public X86AsmBackend {
1142 const MCRegisterInfo &MRI;
1145 enum { CU_NUM_SAVED_REGS = 6 };
1147 mutable unsigned SavedRegs[CU_NUM_SAVED_REGS];
1151 unsigned OffsetSize;
1152 unsigned MoveInstrSize;
1153 unsigned StackDivide;
1156 unsigned PushInstrSize(MCRegister
Reg)
const {
1179 int getCompactUnwindRegNum(
unsigned Reg)
const {
1180 static const MCPhysReg CU32BitRegs[7] = {
1181 X86::EBX, X86::ECX, X86::EDX, X86::EDI, X86::ESI, X86::EBP, 0
1183 static const MCPhysReg CU64BitRegs[] = {
1184 X86::RBX, X86::R12, X86::R13, X86::R14, X86::R15, X86::RBP, 0
1186 const MCPhysReg *CURegs = Is64Bit ? CU64BitRegs : CU32BitRegs;
1187 for (
int Idx = 1; *CURegs; ++CURegs, ++Idx)
1196 uint32_t encodeCompactUnwindRegistersWithFrame()
const {
1200 uint32_t RegEnc = 0;
1201 for (
int i = 0, Idx = 0; i != CU_NUM_SAVED_REGS; ++i) {
1202 unsigned Reg = SavedRegs[i];
1203 if (
Reg == 0)
break;
1205 int CURegNum = getCompactUnwindRegNum(
Reg);
1206 if (CURegNum == -1)
return ~0
U;
1210 RegEnc |= (CURegNum & 0x7) << (Idx++ * 3);
1213 assert((RegEnc & 0x3FFFF) == RegEnc &&
1214 "Invalid compact register encoding!");
1221 uint32_t encodeCompactUnwindRegistersWithoutFrame(
unsigned RegCount)
const {
1235 for (
unsigned i = 0; i < RegCount; ++i) {
1236 int CUReg = getCompactUnwindRegNum(SavedRegs[i]);
1237 if (CUReg == -1)
return ~0
U;
1238 SavedRegs[i] = CUReg;
1242 std::reverse(&SavedRegs[0], &SavedRegs[CU_NUM_SAVED_REGS]);
1244 uint32_t RenumRegs[CU_NUM_SAVED_REGS];
1245 for (
unsigned i = CU_NUM_SAVED_REGS - RegCount; i < CU_NUM_SAVED_REGS; ++i){
1246 unsigned Countless = 0;
1247 for (
unsigned j = CU_NUM_SAVED_REGS - RegCount;
j < i; ++
j)
1248 if (SavedRegs[j] < SavedRegs[i])
1251 RenumRegs[i] = SavedRegs[i] - Countless - 1;
1255 uint32_t permutationEncoding = 0;
1258 permutationEncoding |= 120 * RenumRegs[0] + 24 * RenumRegs[1]
1259 + 6 * RenumRegs[2] + 2 * RenumRegs[3]
1263 permutationEncoding |= 120 * RenumRegs[1] + 24 * RenumRegs[2]
1264 + 6 * RenumRegs[3] + 2 * RenumRegs[4]
1268 permutationEncoding |= 60 * RenumRegs[2] + 12 * RenumRegs[3]
1269 + 3 * RenumRegs[4] + RenumRegs[5];
1272 permutationEncoding |= 20 * RenumRegs[3] + 4 * RenumRegs[4]
1276 permutationEncoding |= 5 * RenumRegs[4] + RenumRegs[5];
1279 permutationEncoding |= RenumRegs[5];
1283 assert((permutationEncoding & 0x3FF) == permutationEncoding &&
1284 "Invalid compact register encoding!");
1285 return permutationEncoding;
1289 DarwinX86AsmBackend(
const Target &
T,
const MCRegisterInfo &MRI,
1290 const MCSubtargetInfo &STI)
1291 : X86AsmBackend(
T, STI), MRI(MRI),
TT(STI.getTargetTriple()),
1292 Is64Bit(
TT.isX86_64()) {
1293 memset(SavedRegs, 0,
sizeof(SavedRegs));
1294 OffsetSize = Is64Bit ? 8 : 4;
1295 MoveInstrSize = Is64Bit ? 3 : 2;
1296 StackDivide = Is64Bit ? 8 : 4;
1299 std::unique_ptr<MCObjectTargetWriter>
1300 createObjectTargetWriter()
const override {
1308 uint64_t generateCompactUnwindEncoding(
const MCDwarfFrameInfo *FI,
1309 const MCContext *Ctxt)
const override {
1311 return CU::UNWIND_MODE_DWARF;
1315 return CU::UNWIND_MODE_DWARF;
1318 if (Instrs.
empty())
return 0;
1319 if (!isDarwinCanonicalPersonality(FI->
Personality) &&
1321 return CU::UNWIND_MODE_DWARF;
1324 unsigned SavedRegIdx = 0;
1325 memset(SavedRegs, 0,
sizeof(SavedRegs));
1330 uint64_t CompactUnwindEncoding = 0;
1332 unsigned SubtractInstrIdx = Is64Bit ? 3 : 2;
1333 unsigned InstrOffset = 0;
1334 unsigned StackAdjust = 0;
1335 uint64_t StackSize = 0;
1336 int64_t MinAbsOffset = std::numeric_limits<int64_t>::max();
1338 for (
const MCCFIInstruction &Inst : Instrs) {
1339 switch (Inst.getOperation()) {
1343 return CU::UNWIND_MODE_DWARF;
1356 (Is64Bit ? X86::RBP : X86::EBP))
1357 return CU::UNWIND_MODE_DWARF;
1360 memset(SavedRegs, 0,
sizeof(SavedRegs));
1363 MinAbsOffset = std::numeric_limits<int64_t>::max();
1364 InstrOffset += MoveInstrSize;
1382 StackSize = Inst.getOffset() / StackDivide;
1398 if (SavedRegIdx == CU_NUM_SAVED_REGS)
1401 return CU::UNWIND_MODE_DWARF;
1404 SavedRegs[SavedRegIdx++] =
Reg.
id();
1405 StackAdjust += OffsetSize;
1406 MinAbsOffset = std::min(MinAbsOffset, std::abs(Inst.getOffset()));
1407 InstrOffset += PushInstrSize(
Reg);
1413 StackAdjust /= StackDivide;
1416 if ((StackAdjust & 0xFF) != StackAdjust)
1418 return CU::UNWIND_MODE_DWARF;
1422 if (SavedRegIdx != 0 && MinAbsOffset != 3 * (
int)OffsetSize)
1423 return CU::UNWIND_MODE_DWARF;
1426 uint32_t RegEnc = encodeCompactUnwindRegistersWithFrame();
1427 if (RegEnc == ~0U)
return CU::UNWIND_MODE_DWARF;
1429 CompactUnwindEncoding |= CU::UNWIND_MODE_BP_FRAME;
1430 CompactUnwindEncoding |= (StackAdjust & 0xFF) << 16;
1431 CompactUnwindEncoding |= RegEnc & CU::UNWIND_BP_FRAME_REGISTERS;
1433 SubtractInstrIdx += InstrOffset;
1436 if ((StackSize & 0xFF) == StackSize) {
1438 CompactUnwindEncoding |= CU::UNWIND_MODE_STACK_IMMD;
1441 CompactUnwindEncoding |= (StackSize & 0xFF) << 16;
1443 if ((StackAdjust & 0x7) != StackAdjust)
1445 return CU::UNWIND_MODE_DWARF;
1448 CompactUnwindEncoding |= CU::UNWIND_MODE_STACK_IND;
1452 CompactUnwindEncoding |= (SubtractInstrIdx & 0xFF) << 16;
1455 CompactUnwindEncoding |= (StackAdjust & 0x7) << 13;
1459 std::reverse(&SavedRegs[0], &SavedRegs[SavedRegIdx]);
1460 CompactUnwindEncoding |= (SavedRegIdx & 0x7) << 10;
1464 uint32_t RegEnc = encodeCompactUnwindRegistersWithoutFrame(SavedRegIdx);
1465 if (RegEnc == ~0U)
return CU::UNWIND_MODE_DWARF;
1468 CompactUnwindEncoding |=
1469 RegEnc & CU::UNWIND_FRAMELESS_STACK_REG_PERMUTATION;
1472 return CompactUnwindEncoding;
1484 return new DarwinX86AsmBackend(
T, MRI, STI);
1487 return new WindowsX86AsmBackend(
T,
false, STI);
1492 return new ELFX86_IAMCUAsmBackend(
T, OSABI, STI);
1494 return new ELFX86_32AsmBackend(
T, OSABI, STI);
1503 return new DarwinX86AsmBackend(
T, MRI, STI);
1506 return new WindowsX86AsmBackend(
T,
true, STI);
1508 if (TheTriple.
isUEFI()) {
1510 "Only COFF format is supported in UEFI environment.");
1511 return new WindowsX86AsmBackend(
T,
true, STI);
1516 if (TheTriple.
isX32())
1517 return new ELFX86_X32AsmBackend(
T, OSABI, STI);
1518 return new ELFX86_64AsmBackend(
T, OSABI, STI);
1524 X86ELFStreamer(
MCContext &Context, std::unique_ptr<MCAsmBackend> TAB,
1525 std::unique_ptr<MCObjectWriter> OW,
1526 std::unique_ptr<MCCodeEmitter>
Emitter)
1530 void emitInstruction(
const MCInst &Inst,
const MCSubtargetInfo &STI)
override;
1534void X86ELFStreamer::emitInstruction(
const MCInst &Inst,
1535 const MCSubtargetInfo &STI) {
1540 std::unique_ptr<MCAsmBackend> &&MAB,
1541 std::unique_ptr<MCObjectWriter> &&MOW,
1542 std::unique_ptr<MCCodeEmitter> &&MCE) {
1543 return new X86ELFStreamer(Context, std::move(MAB), std::move(MOW),
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
#define LLVM_LIKELY(EXPR)
dxil DXContainer Global Emitter
static unsigned getRelaxedOpcode(unsigned Opcode)
PowerPC TLS Dynamic Call Fixup
This file implements the StringSwitch template, which mimics a switch() statement whose cases are str...
static MCInstrInfo * createMCInstrInfo()
static unsigned getRelaxedOpcodeBranch(unsigned Opcode, bool Is16BitMode=false)
static X86::SecondMacroFusionInstKind classifySecondInstInMacroFusion(const MCInst &MI, const MCInstrInfo &MCII)
static bool isRIPRelative(const MCInst &MI, const MCInstrInfo &MCII)
Check if the instruction uses RIP relative addressing.
static bool mayHaveInterruptDelaySlot(unsigned InstOpcode)
X86 has certain instructions which enable interrupts exactly one instruction after the instruction wh...
static bool isFirstMacroFusibleInst(const MCInst &Inst, const MCInstrInfo &MCII)
Check if the instruction is valid as the first instruction in macro fusion.
constexpr char GotSymName[]
static X86::CondCode getCondFromBranch(const MCInst &MI, const MCInstrInfo &MCII)
static unsigned getRelaxedOpcode(const MCInst &MI, bool Is16BitMode)
static unsigned getFixupKindSize(unsigned Kind)
static bool isRelaxableBranch(unsigned Opcode)
static bool isPrefix(unsigned Opcode, const MCInstrInfo &MCII)
Check if the instruction is a prefix.
static bool hasVariantSymbol(const MCInst &MI)
Check if the instruction has a variant symbol operand.
static bool is64Bit(const char *name)
size_t size() const
Get the array size.
bool empty() const
Check if the array is empty.
Generic interface to target specific assembler backends.
virtual MCFixupKindInfo getFixupKindInfo(MCFixupKind Kind) const
Get information on a fixup kind.
virtual std::optional< MCFixupKind > getFixupKind(StringRef Name) const
Map a relocation name used in .reloc to a fixup kind.
Represents required padding such that a particular other set of fragments does not cross a particular...
void setLastFragment(const MCFragment *F)
Context object for machine code objects.
LLVM_ABI bool emitCompactUnwindNonCanonical() const
LLVM_ABI EmitDwarfUnwindType emitDwarfUnwindInfo() const
Base class for the full range of assembler expressions which are needed for parsing.
@ SymbolRef
References to labels and assigned expressions.
Encode information on a single operation to perform on a byte sequence (e.g., an encoded instruction)...
static MCFixup create(uint32_t Offset, const MCExpr *Value, MCFixupKind Kind, bool PCRel=false)
Consider bit fields if we need more flags.
bool getAllowAutoPadding() const
void setAllowAutoPadding(bool V)
unsigned getOpcode() const
MCSection * getParent() const
LLVM_ABI void setVarFixups(ArrayRef< MCFixup > Fixups)
MCFragment * getNext() const
ArrayRef< MCOperand > getOperands() const
size_t getVarSize() const
LLVM_ABI void setVarContents(ArrayRef< char > Contents)
MutableArrayRef< char > getVarContents()
const MCSubtargetInfo * getSubtargetInfo() const
Retrieve the MCSubTargetInfo in effect when the instruction was encoded.
MutableArrayRef< MCFixup > getVarFixups()
void setInst(const MCInst &Inst)
Instances of this class represent a single low-level machine instruction.
unsigned getOpcode() const
void setOpcode(unsigned Op)
const MCOperand & getOperand(unsigned i) const
Describe properties that are true of each instruction in the target description file.
bool isConditionalBranch() const
Return true if this is a branch which may fall through to the next instruction or may transfer contro...
Interface to description of machine instruction set.
const MCInstrDesc & get(unsigned Opcode) const
Return the machine instruction descriptor that corresponds to the specified instruction opcode.
Streaming object file generation interface.
FT * newSpecialFragment(Args &&...args)
MCAssembler & getAssembler()
MCRegister getReg() const
Returns the register number.
MCRegisterInfo base class - We assume that the target defines a static array of MCRegisterDesc object...
std::optional< MCRegister > getLLVMRegNum(uint64_t RegNum, bool isEH) const
Map a dwarf register back to a target register.
Wrapper class representing physical registers. Should be passed by value.
void ensureMinAlignment(Align MinAlignment)
Makes sure that Alignment is at least MinAlignment.
Streaming machine code generation interface.
MCFragment * getCurrentFragment() const
size_t getCurFragSize() const
bool getAllowAutoPadding() const
MCSection * getCurrentSectionOnly() const
Generic base class for all target subtargets.
bool hasFeature(unsigned Feature) const
const Triple & getTargetTriple() const
constexpr unsigned id() const
void push_back(const T &Elt)
Target - Wrapper for Target specific information.
Triple - Helper class for working with autoconf configuration names.
bool isX86_64() const
Tests whether the target is x86 (64-bit).
bool isX32() const
Tests whether the target is X32.
bool isOSBinFormatMachO() const
Tests whether the environment is MachO.
OSType getOS() const
Get the parsed operating system type of this triple.
bool isOSBinFormatCOFF() const
Tests whether the OS uses the COFF binary format.
bool isUEFI() const
Tests whether the OS is UEFI.
bool isOSWindows() const
Tests whether the OS is Windows.
bool isOSBinFormatELF() const
Tests whether the OS uses the ELF binary format.
std::pair< iterator, bool > insert(const ValueT &V)
size_type count(const_arg_type_t< ValueT > V) const
Return 1 if the specified key is in the set, 0 otherwise.
raw_ostream & write(unsigned char C)
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
constexpr char Align[]
Key for Kernel::Arg::Metadata::mAlign.
constexpr std::underlying_type_t< E > Mask()
Get a bitmask with 1s in all places up to the high-order bit of E's largest value.
LLVM_ABI Expected< uint32_t > getCPUSubType(const Triple &T)
LLVM_ABI Expected< uint32_t > getCPUType(const Triple &T)
VE::Fixups getFixupKind(uint8_t S)
bool isPrefix(uint64_t TSFlags)
@ RawFrmDstSrc
RawFrmDstSrc - This form is for instructions that use the source index register SI/ESI/RSI with a pos...
@ RawFrmSrc
RawFrmSrc - This form is for instructions that use the source index register SI/ESI/RSI with a possib...
@ RawFrmMemOffs
RawFrmMemOffs - This form is for instructions that store an absolute memory offset as an immediate wi...
int getMemoryOperandNo(uint64_t TSFlags)
unsigned getOperandBias(const MCInstrDesc &Desc)
Compute whether all of the def operands are repeated in the uses and therefore should be skipped.
void emitPrefix(MCCodeEmitter &MCE, const MCInst &MI, SmallVectorImpl< char > &CB, const MCSubtargetInfo &STI)
void emitInstruction(MCObjectStreamer &, const MCInst &Inst, const MCSubtargetInfo &STI)
FirstMacroFusionInstKind classifyFirstOpcodeInMacroFusion(unsigned Opcode)
AlignBranchBoundaryKind
Defines the possible values of the branch boundary alignment mask.
SecondMacroFusionInstKind
EncodingOfSegmentOverridePrefix getSegmentOverridePrefixForReg(MCRegister Reg)
Given a segment register, return the encoding of the segment override prefix for it.
unsigned getOpcodeForLongImmediateForm(unsigned Opcode)
bool isMacroFused(FirstMacroFusionInstKind FirstKind, SecondMacroFusionInstKind SecondKind)
@ reloc_riprel_4byte_movq_load_rex2
@ reloc_signed_4byte_relax
@ reloc_branch_4byte_pcrel
@ reloc_riprel_4byte_relax
@ reloc_riprel_4byte_relax_evex
@ reloc_riprel_4byte_relax_rex
@ reloc_global_offset_table
@ reloc_riprel_4byte_movq_load
@ reloc_riprel_4byte_relax_rex2
initializer< Ty > init(const Ty &Val)
LocationClass< Ty > location(Ty &L)
Error applyFixup(LinkGraph &G, Block &B, const Edge &E, const ArmConfig &ArmCfg)
Apply fixup expression for edge to block content.
bool isRelocation(MCFixupKind FixupKind)
NodeAddr< CodeNode * > Code
Context & getContext() const
BaseReg
Stack frame base register. Bit 0 of FREInfo.Info.
This is an optimization pass for GlobalISel generic memory operations.
FunctionAddr VTableAddr Value
constexpr bool isInt(int64_t x)
Checks if an integer fits into the given bit width.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
MCAsmBackend * createX86_64AsmBackend(const Target &T, const MCSubtargetInfo &STI, const MCRegisterInfo &MRI, const MCTargetOptions &Options)
std::unique_ptr< MCObjectTargetWriter > createX86WinCOFFObjectWriter(bool Is64Bit)
Construct an X86 Win COFF object writer.
uint16_t MCFixupKind
Extensible enumeration to represent the type of a fixup.
MCStreamer * createX86ELFStreamer(const Triple &T, MCContext &Context, std::unique_ptr< MCAsmBackend > &&MAB, std::unique_ptr< MCObjectWriter > &&MOW, std::unique_ptr< MCCodeEmitter > &&MCE)
FunctionAddr VTableAddr Count
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ FK_SecRel_2
A two-byte section relative fixup.
@ FirstLiteralRelocationKind
@ FK_Data_8
A eight-byte fixup.
@ FK_Data_1
A one-byte fixup.
@ FK_Data_4
A four-byte fixup.
@ FK_SecRel_8
A eight-byte section relative fixup.
@ FK_SecRel_4
A four-byte section relative fixup.
@ FK_SecRel_1
A one-byte section relative fixup.
@ FK_Data_2
A two-byte fixup.
void cantFail(Error Err, const char *Msg=nullptr)
Report a fatal error if Err is a failure value.
FunctionAddr VTableAddr uintptr_t uintptr_t Data
std::unique_ptr< MCObjectTargetWriter > createX86MachObjectWriter(bool Is64Bit, uint32_t CPUType, uint32_t CPUSubtype)
Construct an X86 Mach-O object writer.
uint16_t MCPhysReg
An unsigned integer type large enough to represent all physical registers, but not necessarily virtua...
ArrayRef(const T &OneElt) -> ArrayRef< T >
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
std::unique_ptr< MCObjectTargetWriter > createX86ELFObjectWriter(bool IsELF64, uint8_t OSABI, uint16_t EMachine)
Construct an X86 ELF object writer.
Align assumeAligned(uint64_t Value)
Treats the value 0 as a 1, so Align is always at least 1.
MCAsmBackend * createX86_32AsmBackend(const Target &T, const MCSubtargetInfo &STI, const MCRegisterInfo &MRI, const MCTargetOptions &Options)
Implement std::hash so that hash_code can be used in STL containers.
const MCSymbol * Personality
std::vector< MCCFIInstruction > Instructions