LLVM 23.0.0git
X86AsmPrinter.cpp
Go to the documentation of this file.
1//===-- X86AsmPrinter.cpp - Convert X86 LLVM code to AT&T assembly --------===//
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//
9// This file contains a printer that converts from our internal representation
10// of machine-dependent LLVM code to X86 machine code.
11//
12//===----------------------------------------------------------------------===//
13
14#include "X86AsmPrinter.h"
20#include "X86.h"
21#include "X86InstrInfo.h"
23#include "X86Subtarget.h"
24#include "llvm-c/Visibility.h"
33#include "llvm/IR/InlineAsm.h"
35#include "llvm/IR/Mangler.h"
36#include "llvm/IR/Module.h"
37#include "llvm/IR/Type.h"
38#include "llvm/MC/MCAsmInfo.h"
40#include "llvm/MC/MCContext.h"
41#include "llvm/MC/MCExpr.h"
42#include "llvm/MC/MCInst.h"
47#include "llvm/MC/MCStreamer.h"
48#include "llvm/MC/MCSymbol.h"
50#include "llvm/Support/Debug.h"
53
54using namespace llvm;
55
57 std::unique_ptr<MCStreamer> Streamer)
58 : AsmPrinter(TM, std::move(Streamer), ID), FM(*this) {
59 GetPSI = [this](Module &M) -> ProfileSummaryInfo * {
61 return &PSIW->getPSI();
62 return nullptr;
63 };
64 GetSDPI = [this](Module &M) -> StaticDataProfileInfo * {
65 if (auto *SDPIW =
67 return &SDPIW->getStaticDataProfileInfo();
68 return nullptr;
69 };
70}
71
72//===----------------------------------------------------------------------===//
73// Primitive Helper Functions.
74//===----------------------------------------------------------------------===//
75
76/// runOnMachineFunction - Emit the function body.
77///
79 PSI = GetPSI(*MF.getFunction().getParent());
80 SDPI = GetSDPI(*MF.getFunction().getParent());
81
82 Subtarget = &MF.getSubtarget<X86Subtarget>();
83
84 SMShadowTracker.startFunction(MF);
85 CodeEmitter.reset(TM.getTarget().createMCCodeEmitter(
86 *Subtarget->getInstrInfo(), MF.getContext()));
87
88 const Module *M = MF.getFunction().getParent();
89 EmitFPOData = Subtarget->isTargetWin32() && M->getCodeViewFlag();
90
91 IndCSPrefix = M->getModuleFlag("indirect_branch_cs_prefix");
92
94
95 if (Subtarget->isTargetCOFF()) {
96 bool Local = MF.getFunction().hasLocalLinkage();
97 OutStreamer->beginCOFFSymbolDef(CurrentFnSym);
98 OutStreamer->emitCOFFSymbolStorageClass(
102 OutStreamer->endCOFFSymbolDef();
103 }
104
105 // Emit the rest of the function body.
107
108 // Emit the XRay table for this function.
110
111 EmitFPOData = false;
112
113 IndCSPrefix = false;
114
115 // We didn't modify anything.
116 return false;
117}
118
120 if (EmitFPOData) {
121 auto *XTS =
122 static_cast<X86TargetStreamer *>(OutStreamer->getTargetStreamer());
123 XTS->emitFPOProc(
126 }
127}
128
130 if (EmitFPOData) {
131 auto *XTS =
132 static_cast<X86TargetStreamer *>(OutStreamer->getTargetStreamer());
133 XTS->emitFPOEndProc();
134 }
135}
136
137uint32_t X86AsmPrinter::MaskKCFIType(uint32_t Value) {
138 // If the type hash matches an invalid pattern, mask the value.
139 const uint32_t InvalidValues[] = {
140 0xFA1E0FF3, /* ENDBR64 */
141 0xFB1E0FF3, /* ENDBR32 */
142 };
143 for (uint32_t N : InvalidValues) {
144 // LowerKCFI_CHECK emits -Value for indirect call checks, so we must also
145 // mask that. Note that -(Value + 1) == ~Value.
146 if (N == Value || -N == Value)
147 return Value + 1;
148 }
149 return Value;
150}
151
152void X86AsmPrinter::EmitKCFITypePadding(const MachineFunction &MF,
153 bool HasType) {
154 // Keep the function entry aligned, taking patchable-function-prefix into
155 // account if set.
156 int64_t PrefixBytes = 0;
157 (void)MF.getFunction()
158 .getFnAttribute("patchable-function-prefix")
159 .getValueAsString()
160 .getAsInteger(10, PrefixBytes);
161
162 // Also take the type identifier into account if we're emitting
163 // one. Otherwise, just pad with nops. The X86::MOV32ri instruction emitted
164 // in X86AsmPrinter::emitKCFITypeId is 5 bytes long.
165 if (HasType)
166 PrefixBytes += 5;
167
168 emitNops(offsetToAlignment(PrefixBytes, MF.getPreferredAlignment()));
169}
170
171/// emitKCFITypeId - Emit the KCFI type information in architecture specific
172/// format.
174 const Function &F = MF.getFunction();
175 if (!F.getParent()->getModuleFlag("kcfi"))
176 return;
177
178 ConstantInt *Type = nullptr;
179 if (const MDNode *MD = F.getMetadata(LLVMContext::MD_kcfi_type))
180 Type = mdconst::extract<ConstantInt>(MD->getOperand(0));
181
182 // If we don't have a type to emit, just emit padding if needed to maintain
183 // the same alignment for all functions.
184 if (!Type) {
185 EmitKCFITypePadding(MF, /*HasType=*/false);
186 return;
187 }
188
189 // Emit a function symbol for the type data to avoid unreachable instruction
190 // warnings from binary validation tools, and use the same linkage as the
191 // parent function. Note that using local linkage would result in duplicate
192 // symbols for weak parent functions.
193 MCSymbol *FnSym = OutContext.getOrCreateSymbol("__cfi_" + MF.getName());
194 emitLinkage(&MF.getFunction(), FnSym);
195 if (MAI->hasDotTypeDotSizeDirective())
196 OutStreamer->emitSymbolAttribute(FnSym, MCSA_ELF_TypeFunction);
197 OutStreamer->emitLabel(FnSym);
198
199 // Embed the type hash in the X86::MOV32ri instruction to avoid special
200 // casing object file parsers.
201 EmitKCFITypePadding(MF);
202 unsigned DestReg = X86::EAX;
203
204 if (F.getParent()->getModuleFlag("kcfi-arity")) {
205 // The ArityToRegMap assumes the 64-bit SysV ABI.
206 [[maybe_unused]] const auto &Triple = MF.getTarget().getTargetTriple();
208
209 // Determine the function's arity (i.e., the number of arguments) at the ABI
210 // level by counting the number of parameters that are passed
211 // as registers, such as pointers and 64-bit (or smaller) integers. The
212 // Linux x86-64 ABI allows up to 6 integer parameters to be passed in GPRs.
213 // Additional parameters or parameters larger than 64 bits may be passed on
214 // the stack, in which case the arity is denoted as 7. Floating-point
215 // arguments passed in XMM0-XMM7 are not counted toward arity because
216 // floating-point values are not relevant to enforcing kCFI at this time.
217 const unsigned ArityToRegMap[8] = {X86::EAX, X86::ECX, X86::EDX, X86::EBX,
218 X86::ESP, X86::EBP, X86::ESI, X86::EDI};
219 int Arity;
220 if (MF.getInfo<X86MachineFunctionInfo>()->getArgumentStackSize() > 0) {
221 Arity = 7;
222 } else {
223 Arity = 0;
224 for (const auto &LI : MF.getRegInfo().liveins()) {
225 auto Reg = LI.first;
226 if (X86::GR8RegClass.contains(Reg) || X86::GR16RegClass.contains(Reg) ||
227 X86::GR32RegClass.contains(Reg) ||
228 X86::GR64RegClass.contains(Reg)) {
229 ++Arity;
230 }
231 }
232 }
233 DestReg = ArityToRegMap[Arity];
234 }
235
236 EmitAndCountInstruction(MCInstBuilder(X86::MOV32ri)
237 .addReg(DestReg)
238 .addImm(MaskKCFIType(Type->getZExtValue())));
239
240 if (MAI->hasDotTypeDotSizeDirective()) {
241 MCSymbol *EndSym = OutContext.createTempSymbol("cfi_func_end");
242 OutStreamer->emitLabel(EndSym);
243
244 const MCExpr *SizeExp = MCBinaryExpr::createSub(
247 OutStreamer->emitELFSize(FnSym, SizeExp);
248 }
249}
250
251/// PrintSymbolOperand - Print a raw symbol reference operand. This handles
252/// jump tables, constant pools, global address and external symbols, all of
253/// which print to a label with various suffixes for relocation types etc.
254void X86AsmPrinter::PrintSymbolOperand(const MachineOperand &MO,
255 raw_ostream &O) {
256 switch (MO.getType()) {
257 default: llvm_unreachable("unknown symbol type!");
259 GetCPISymbol(MO.getIndex())->print(O, MAI);
260 printOffset(MO.getOffset(), O);
261 break;
263 const GlobalValue *GV = MO.getGlobal();
264
265 MCSymbol *GVSym;
268 GVSym = getSymbolWithGlobalValueBase(GV, "$non_lazy_ptr");
269 else
270 GVSym = getSymbolPreferLocal(*GV);
271
272 // Handle dllimport linkage.
274 GVSym = OutContext.getOrCreateSymbol(Twine("__imp_") + GVSym->getName());
275 else if (MO.getTargetFlags() == X86II::MO_COFFSTUB)
276 GVSym =
277 OutContext.getOrCreateSymbol(Twine(".refptr.") + GVSym->getName());
278
281 MCSymbol *Sym = getSymbolWithGlobalValueBase(GV, "$non_lazy_ptr");
283 MMI->getObjFileInfo<MachineModuleInfoMachO>().getGVStubEntry(Sym);
284 if (!StubSym.getPointer())
286 !GV->hasInternalLinkage());
287 }
288
289 // If the name begins with a dollar-sign, enclose it in parens. We do this
290 // to avoid having it look like an integer immediate to the assembler.
291 if (GVSym->getName()[0] != '$')
292 GVSym->print(O, MAI);
293 else {
294 O << '(';
295 GVSym->print(O, MAI);
296 O << ')';
297 }
298 printOffset(MO.getOffset(), O);
299 break;
300 }
301 }
302
303 switch (MO.getTargetFlags()) {
304 default:
305 llvm_unreachable("Unknown target flag on GV operand");
306 case X86II::MO_NO_FLAG: // No flag.
307 break;
311 // These affect the name of the symbol, not any suffix.
312 break;
314 O << " + [.-";
315 MF->getPICBaseSymbol()->print(O, MAI);
316 O << ']';
317 break;
320 O << '-';
321 MF->getPICBaseSymbol()->print(O, MAI);
322 break;
323 case X86II::MO_TLSGD: O << "@TLSGD"; break;
324 case X86II::MO_TLSLD: O << "@TLSLD"; break;
325 case X86II::MO_TLSLDM: O << "@TLSLDM"; break;
326 case X86II::MO_GOTTPOFF: O << "@GOTTPOFF"; break;
327 case X86II::MO_INDNTPOFF: O << "@INDNTPOFF"; break;
328 case X86II::MO_TPOFF: O << "@TPOFF"; break;
329 case X86II::MO_DTPOFF: O << "@DTPOFF"; break;
330 case X86II::MO_NTPOFF: O << "@NTPOFF"; break;
331 case X86II::MO_GOTNTPOFF: O << "@GOTNTPOFF"; break;
332 case X86II::MO_GOTPCREL: O << "@GOTPCREL"; break;
333 case X86II::MO_GOTPCREL_NORELAX: O << "@GOTPCREL_NORELAX"; break;
334 case X86II::MO_GOT: O << "@GOT"; break;
335 case X86II::MO_GOTOFF: O << "@GOTOFF"; break;
336 case X86II::MO_PLT: O << "@PLT"; break;
337 case X86II::MO_TLVP: O << "@TLVP"; break;
339 O << "@TLVP" << '-';
340 MF->getPICBaseSymbol()->print(O, MAI);
341 break;
342 case X86II::MO_SECREL: O << "@SECREL32"; break;
343 }
344}
345
346void X86AsmPrinter::PrintOperand(const MachineInstr *MI, unsigned OpNo,
347 raw_ostream &O) {
348 const MachineOperand &MO = MI->getOperand(OpNo);
349 const bool IsATT = MI->getInlineAsmDialect() == InlineAsm::AD_ATT;
350 switch (MO.getType()) {
351 default: llvm_unreachable("unknown operand type!");
353 if (IsATT)
354 O << '%';
356 return;
357 }
358
360 if (IsATT)
361 O << '$';
362 O << MO.getImm();
363 return;
364
367 switch (MI->getInlineAsmDialect()) {
369 O << '$';
370 break;
372 O << "offset ";
373 break;
374 }
375 PrintSymbolOperand(MO, O);
376 break;
377 }
380 Sym->print(O, MAI);
381 break;
382 }
383 }
384}
385
386/// PrintModifiedOperand - Print subregisters based on supplied modifier,
387/// deferring to PrintOperand() if no modifier was supplied or if operand is not
388/// a register.
389void X86AsmPrinter::PrintModifiedOperand(const MachineInstr *MI, unsigned OpNo,
390 raw_ostream &O, StringRef Modifier) {
391 const MachineOperand &MO = MI->getOperand(OpNo);
392 if (Modifier.empty() || !MO.isReg())
393 return PrintOperand(MI, OpNo, O);
394 if (MI->getInlineAsmDialect() == InlineAsm::AD_ATT)
395 O << '%';
396 Register Reg = MO.getReg();
397 if (Modifier.consume_front("subreg")) {
398 unsigned Size = (Modifier == "64") ? 64
399 : (Modifier == "32") ? 32
400 : (Modifier == "16") ? 16
401 : 8;
403 }
405}
406
407/// PrintPCRelImm - This is used to print an immediate value that ends up
408/// being encoded as a pc-relative value. These print slightly differently, for
409/// example, a $ is not emitted.
410void X86AsmPrinter::PrintPCRelImm(const MachineInstr *MI, unsigned OpNo,
411 raw_ostream &O) {
412 const MachineOperand &MO = MI->getOperand(OpNo);
413 switch (MO.getType()) {
414 default: llvm_unreachable("Unknown pcrel immediate operand");
416 // pc-relativeness was handled when computing the value in the reg.
417 PrintOperand(MI, OpNo, O);
418 return;
420 O << MO.getImm();
421 return;
423 PrintSymbolOperand(MO, O);
424 return;
425 }
426}
427
428void X86AsmPrinter::PrintLeaMemReference(const MachineInstr *MI, unsigned OpNo,
429 raw_ostream &O, StringRef Modifier) {
430 const MachineOperand &BaseReg = MI->getOperand(OpNo + X86::AddrBaseReg);
431 const MachineOperand &IndexReg = MI->getOperand(OpNo + X86::AddrIndexReg);
432 const MachineOperand &DispSpec = MI->getOperand(OpNo + X86::AddrDisp);
433
434 // If we really don't want to print out (rip), don't.
435 bool HasBaseReg = BaseReg.getReg() != 0;
436 if (HasBaseReg && Modifier == "no-rip" && BaseReg.getReg() == X86::RIP)
437 HasBaseReg = false;
438
439 // HasParenPart - True if we will print out the () part of the mem ref.
440 bool HasParenPart = IndexReg.getReg() || HasBaseReg;
441
442 switch (DispSpec.getType()) {
443 default:
444 llvm_unreachable("unknown operand type!");
446 int DispVal = DispSpec.getImm();
447 if (DispVal || !HasParenPart)
448 O << DispVal;
449 break;
450 }
453 PrintSymbolOperand(DispSpec, O);
454 break;
455 }
456
457 if (Modifier == "H")
458 O << "+8";
459
460 if (HasParenPart) {
461 assert(IndexReg.getReg() != X86::ESP &&
462 "X86 doesn't allow scaling by ESP");
463
464 O << '(';
465 if (HasBaseReg)
466 PrintModifiedOperand(MI, OpNo + X86::AddrBaseReg, O, Modifier);
467
468 if (IndexReg.getReg()) {
469 O << ',';
470 PrintModifiedOperand(MI, OpNo + X86::AddrIndexReg, O, Modifier);
471 unsigned ScaleVal = MI->getOperand(OpNo + X86::AddrScaleAmt).getImm();
472 if (ScaleVal != 1)
473 O << ',' << ScaleVal;
474 }
475 O << ')';
476 }
477}
478
479static bool isSimpleReturn(const MachineInstr &MI) {
480 // We exclude all tail calls here which set both isReturn and isCall.
481 return MI.getDesc().isReturn() && !MI.getDesc().isCall();
482}
483
485 unsigned Opc = MI.getOpcode();
486 return MI.getDesc().isIndirectBranch() /*Make below code in a good shape*/ ||
487 Opc == X86::TAILJMPr || Opc == X86::TAILJMPm ||
488 Opc == X86::TAILJMPr64 || Opc == X86::TAILJMPm64 ||
489 Opc == X86::TCRETURNri || Opc == X86::TCRETURN_WIN64ri ||
490 Opc == X86::TCRETURN_HIPE32ri || Opc == X86::TCRETURNmi ||
491 Opc == X86::TCRETURN_WINmi64 || Opc == X86::TCRETURNri64 ||
492 Opc == X86::TCRETURNmi64 || Opc == X86::TCRETURNri64_ImpCall ||
493 Opc == X86::TAILJMPr64_REX || Opc == X86::TAILJMPm64_REX;
494}
495
497 if (Subtarget->hardenSlsRet() || Subtarget->hardenSlsIJmp()) {
498 auto I = MBB.getLastNonDebugInstr();
499 if (I != MBB.end()) {
500 if ((Subtarget->hardenSlsRet() && isSimpleReturn(*I)) ||
501 (Subtarget->hardenSlsIJmp() && isIndirectBranchOrTailCall(*I))) {
502 MCInst TmpInst;
503 TmpInst.setOpcode(X86::INT3);
504 EmitToStreamer(*OutStreamer, TmpInst);
505 }
506 }
507 }
508 if (SplitChainedAtEndOfBlock) {
509 OutStreamer->emitWinCFISplitChained();
510 // Splitting into a new unwind info implicitly starts a prolog. We have no
511 // instructions to add to the prolog, so immediately end it.
512 OutStreamer->emitWinCFIEndProlog();
513 SplitChainedAtEndOfBlock = false;
514 }
516 SMShadowTracker.emitShadowPadding(*OutStreamer, getSubtargetInfo());
517}
518
519void X86AsmPrinter::PrintMemReference(const MachineInstr *MI, unsigned OpNo,
520 raw_ostream &O, StringRef Modifier) {
521 assert(isMem(*MI, OpNo) && "Invalid memory reference!");
522 const MachineOperand &Segment = MI->getOperand(OpNo + X86::AddrSegmentReg);
523 if (Segment.getReg()) {
524 PrintModifiedOperand(MI, OpNo + X86::AddrSegmentReg, O, Modifier);
525 O << ':';
526 }
527 PrintLeaMemReference(MI, OpNo, O, Modifier);
528}
529
530void X86AsmPrinter::PrintIntelMemReference(const MachineInstr *MI,
531 unsigned OpNo, raw_ostream &O,
532 StringRef Modifier) {
533 const MachineOperand &BaseReg = MI->getOperand(OpNo + X86::AddrBaseReg);
534 unsigned ScaleVal = MI->getOperand(OpNo + X86::AddrScaleAmt).getImm();
535 const MachineOperand &IndexReg = MI->getOperand(OpNo + X86::AddrIndexReg);
536 const MachineOperand &DispSpec = MI->getOperand(OpNo + X86::AddrDisp);
537 const MachineOperand &SegReg = MI->getOperand(OpNo + X86::AddrSegmentReg);
538
539 // If we really don't want to print out (rip), don't.
540 bool HasBaseReg = BaseReg.getReg() != 0;
541 if (HasBaseReg && Modifier == "no-rip" && BaseReg.getReg() == X86::RIP)
542 HasBaseReg = false;
543
544 // If we really just want to print out displacement.
545 if ((DispSpec.isGlobal() || DispSpec.isSymbol()) && Modifier == "disp-only") {
546 HasBaseReg = false;
547 }
548
549 // If this has a segment register, print it.
550 if (SegReg.getReg()) {
551 PrintOperand(MI, OpNo + X86::AddrSegmentReg, O);
552 O << ':';
553 }
554
555 O << '[';
556
557 bool NeedPlus = false;
558 if (HasBaseReg) {
559 PrintOperand(MI, OpNo + X86::AddrBaseReg, O);
560 NeedPlus = true;
561 }
562
563 if (IndexReg.getReg()) {
564 if (NeedPlus) O << " + ";
565 if (ScaleVal != 1)
566 O << ScaleVal << '*';
567 PrintOperand(MI, OpNo + X86::AddrIndexReg, O);
568 NeedPlus = true;
569 }
570
571 if (!DispSpec.isImm()) {
572 if (NeedPlus) O << " + ";
573 // Do not add `offset` operator. Matches the behaviour of
574 // X86IntelInstPrinter::printMemReference.
575 PrintSymbolOperand(DispSpec, O);
576 } else {
577 int64_t DispVal = DispSpec.getImm();
578 if (DispVal || (!IndexReg.getReg() && !HasBaseReg)) {
579 if (NeedPlus) {
580 if (DispVal > 0)
581 O << " + ";
582 else {
583 O << " - ";
584 DispVal = -DispVal;
585 }
586 }
587 O << DispVal;
588 }
589 }
590 O << ']';
591}
592
594 assert(Subtarget);
595 return Subtarget;
596}
597
598void X86AsmPrinter::emitMachOIFuncStubBody(Module &M, const GlobalIFunc &GI,
599 MCSymbol *LazyPointer) {
600 // _ifunc:
601 // jmpq *lazy_pointer(%rip)
602
603 OutStreamer->emitInstruction(
604 MCInstBuilder(X86::JMP32m)
605 .addReg(X86::RIP)
606 .addImm(1)
607 .addReg(0)
609 MCSymbolRefExpr::create(LazyPointer, OutContext)))
610 .addReg(0),
611 *Subtarget);
612}
613
614void X86AsmPrinter::emitMachOIFuncStubHelperBody(Module &M,
615 const GlobalIFunc &GI,
616 MCSymbol *LazyPointer) {
617 // _ifunc.stub_helper:
618 // push %rax
619 // push %rdi
620 // push %rsi
621 // push %rdx
622 // push %rcx
623 // push %r8
624 // push %r9
625 // callq foo
626 // movq %rax,lazy_pointer(%rip)
627 // pop %r9
628 // pop %r8
629 // pop %rcx
630 // pop %rdx
631 // pop %rsi
632 // pop %rdi
633 // pop %rax
634 // jmpq *lazy_pointer(%rip)
635
636 for (int Reg :
637 {X86::RAX, X86::RDI, X86::RSI, X86::RDX, X86::RCX, X86::R8, X86::R9})
638 OutStreamer->emitInstruction(MCInstBuilder(X86::PUSH64r).addReg(Reg),
639 *Subtarget);
640
641 OutStreamer->emitInstruction(
642 MCInstBuilder(X86::CALL64pcrel32)
644 *Subtarget);
645
646 OutStreamer->emitInstruction(
647 MCInstBuilder(X86::MOV64mr)
648 .addReg(X86::RIP)
649 .addImm(1)
650 .addReg(0)
652 MCSymbolRefExpr::create(LazyPointer, OutContext)))
653 .addReg(0)
654 .addReg(X86::RAX),
655 *Subtarget);
656
657 for (int Reg :
658 {X86::R9, X86::R8, X86::RCX, X86::RDX, X86::RSI, X86::RDI, X86::RAX})
659 OutStreamer->emitInstruction(MCInstBuilder(X86::POP64r).addReg(Reg),
660 *Subtarget);
661
662 OutStreamer->emitInstruction(
663 MCInstBuilder(X86::JMP32m)
664 .addReg(X86::RIP)
665 .addImm(1)
666 .addReg(0)
668 MCSymbolRefExpr::create(LazyPointer, OutContext)))
669 .addReg(0),
670 *Subtarget);
671}
672
673static bool printAsmMRegister(const X86AsmPrinter &P, const MachineOperand &MO,
674 char Mode, raw_ostream &O) {
675 Register Reg = MO.getReg();
676 bool EmitPercent = MO.getParent()->getInlineAsmDialect() == InlineAsm::AD_ATT;
677
678 if (!X86::GR8RegClass.contains(Reg) &&
679 !X86::GR16RegClass.contains(Reg) &&
680 !X86::GR32RegClass.contains(Reg) &&
681 !X86::GR64RegClass.contains(Reg))
682 return true;
683
684 switch (Mode) {
685 default: return true; // Unknown mode.
686 case 'b': // Print QImode register
688 break;
689 case 'h': // Print QImode high register
690 Reg = getX86SubSuperRegister(Reg, 8, true);
691 if (!Reg.isValid())
692 return true;
693 break;
694 case 'w': // Print HImode register
696 break;
697 case 'k': // Print SImode register
699 break;
700 case 'V':
701 EmitPercent = false;
702 [[fallthrough]];
703 case 'q':
704 // Print 64-bit register names if 64-bit integer registers are available.
705 // Otherwise, print 32-bit register names.
706 Reg = getX86SubSuperRegister(Reg, P.getSubtarget().is64Bit() ? 64 : 32);
707 break;
708 }
709
710 if (EmitPercent)
711 O << '%';
712
714 return false;
715}
716
717static bool printAsmVRegister(const MachineOperand &MO, char Mode,
718 raw_ostream &O) {
719 Register Reg = MO.getReg();
720 bool EmitPercent = MO.getParent()->getInlineAsmDialect() == InlineAsm::AD_ATT;
721
722 unsigned Index;
723 if (X86::VR128XRegClass.contains(Reg))
724 Index = Reg - X86::XMM0;
725 else if (X86::VR256XRegClass.contains(Reg))
726 Index = Reg - X86::YMM0;
727 else if (X86::VR512RegClass.contains(Reg))
728 Index = Reg - X86::ZMM0;
729 else
730 return true;
731
732 switch (Mode) {
733 default: // Unknown mode.
734 return true;
735 case 'x': // Print V4SFmode register
736 Reg = X86::XMM0 + Index;
737 break;
738 case 't': // Print V8SFmode register
739 Reg = X86::YMM0 + Index;
740 break;
741 case 'g': // Print V16SFmode register
742 Reg = X86::ZMM0 + Index;
743 break;
744 }
745
746 if (EmitPercent)
747 O << '%';
748
750 return false;
751}
752
753/// PrintAsmOperand - Print out an operand for an inline asm expression.
754///
756 const char *ExtraCode, raw_ostream &O) {
757 // Does this asm operand have a single letter operand modifier?
758 if (ExtraCode && ExtraCode[0]) {
759 if (ExtraCode[1] != 0) return true; // Unknown modifier.
760
761 const MachineOperand &MO = MI->getOperand(OpNo);
762
763 switch (ExtraCode[0]) {
764 default:
765 // See if this is a generic print operand
766 return AsmPrinter::PrintAsmOperand(MI, OpNo, ExtraCode, O);
767 case 'a': // This is an address. Currently only 'i' and 'r' are expected.
768 switch (MO.getType()) {
769 default:
770 return true;
772 O << MO.getImm();
773 return false;
777 llvm_unreachable("unexpected operand type!");
779 PrintSymbolOperand(MO, O);
780 if (Subtarget->is64Bit())
781 O << "(%rip)";
782 return false;
784 O << '(';
785 PrintOperand(MI, OpNo, O);
786 O << ')';
787 return false;
788 }
789
790 case 'c': // Don't print "$" before a global var name or constant.
791 switch (MO.getType()) {
792 default:
793 PrintOperand(MI, OpNo, O);
794 break;
796 O << MO.getImm();
797 break;
801 llvm_unreachable("unexpected operand type!");
803 PrintSymbolOperand(MO, O);
804 break;
805 }
806 return false;
807
808 case 'A': // Print '*' before a register (it must be a register)
809 if (MO.isReg()) {
810 O << '*';
811 PrintOperand(MI, OpNo, O);
812 return false;
813 }
814 return true;
815
816 case 'b': // Print QImode register
817 case 'h': // Print QImode high register
818 case 'w': // Print HImode register
819 case 'k': // Print SImode register
820 case 'q': // Print DImode register
821 case 'V': // Print native register without '%'
822 if (MO.isReg())
823 return printAsmMRegister(*this, MO, ExtraCode[0], O);
824 PrintOperand(MI, OpNo, O);
825 return false;
826
827 case 'x': // Print V4SFmode register
828 case 't': // Print V8SFmode register
829 case 'g': // Print V16SFmode register
830 if (MO.isReg())
831 return printAsmVRegister(MO, ExtraCode[0], O);
832 PrintOperand(MI, OpNo, O);
833 return false;
834
835 case 'p': {
836 const MachineOperand &MO = MI->getOperand(OpNo);
838 return true;
839 PrintSymbolOperand(MO, O);
840 return false;
841 }
842
843 case 'P': // This is the operand of a call, treat specially.
844 PrintPCRelImm(MI, OpNo, O);
845 return false;
846
847 case 'n': // Negate the immediate or print a '-' before the operand.
848 // Note: this is a temporary solution. It should be handled target
849 // independently as part of the 'MC' work.
850 if (MO.isImm()) {
851 O << -MO.getImm();
852 return false;
853 }
854 O << '-';
855 }
856 }
857
858 PrintOperand(MI, OpNo, O);
859 return false;
860}
861
863 const char *ExtraCode,
864 raw_ostream &O) {
865 if (ExtraCode && ExtraCode[0]) {
866 if (ExtraCode[1] != 0) return true; // Unknown modifier.
867
868 switch (ExtraCode[0]) {
869 default: return true; // Unknown modifier.
870 case 'b': // Print QImode register
871 case 'h': // Print QImode high register
872 case 'w': // Print HImode register
873 case 'k': // Print SImode register
874 case 'q': // Print SImode register
875 // These only apply to registers, ignore on mem.
876 break;
877 case 'H':
878 if (MI->getInlineAsmDialect() == InlineAsm::AD_Intel) {
879 return true; // Unsupported modifier in Intel inline assembly.
880 } else {
881 PrintMemReference(MI, OpNo, O, "H");
882 }
883 return false;
884 // Print memory only with displacement. The Modifer 'P' is used in inline
885 // asm to present a call symbol or a global symbol which can not use base
886 // reg or index reg.
887 case 'P':
888 if (MI->getInlineAsmDialect() == InlineAsm::AD_Intel) {
889 PrintIntelMemReference(MI, OpNo, O, "disp-only");
890 } else {
891 PrintMemReference(MI, OpNo, O, "disp-only");
892 }
893 return false;
894 }
895 }
896 if (MI->getInlineAsmDialect() == InlineAsm::AD_Intel) {
897 PrintIntelMemReference(MI, OpNo, O);
898 } else {
899 PrintMemReference(MI, OpNo, O);
900 }
901 return false;
902}
903
905 const Triple &TT = TM.getTargetTriple();
906
907 if (TT.isOSBinFormatELF()) {
908 // Assemble feature flags that may require creation of a note section.
909 unsigned FeatureFlagsAnd = 0;
910 if (M.getModuleFlag("cf-protection-branch"))
911 FeatureFlagsAnd |= ELF::GNU_PROPERTY_X86_FEATURE_1_IBT;
912 if (M.getModuleFlag("cf-protection-return"))
913 FeatureFlagsAnd |= ELF::GNU_PROPERTY_X86_FEATURE_1_SHSTK;
914
915 if (FeatureFlagsAnd) {
916 // Emit a .note.gnu.property section with the flags.
917 assert((TT.isX86_32() || TT.isX86_64()) &&
918 "CFProtection used on invalid architecture!");
919 MCSection *Cur = OutStreamer->getCurrentSectionOnly();
920 MCSection *Nt = MMI->getContext().getELFSection(
921 ".note.gnu.property", ELF::SHT_NOTE, ELF::SHF_ALLOC);
922 OutStreamer->switchSection(Nt);
923
924 // Emitting note header.
925 const int WordSize = TT.isX86_64() && !TT.isX32() ? 8 : 4;
926 emitAlignment(WordSize == 4 ? Align(4) : Align(8));
927 OutStreamer->emitIntValue(4, 4 /*size*/); // data size for "GNU\0"
928 OutStreamer->emitIntValue(8 + WordSize, 4 /*size*/); // Elf_Prop size
929 OutStreamer->emitIntValue(ELF::NT_GNU_PROPERTY_TYPE_0, 4 /*size*/);
930 OutStreamer->emitBytes(StringRef("GNU", 4)); // note name
931
932 // Emitting an Elf_Prop for the CET properties.
934 OutStreamer->emitInt32(4); // data size
935 OutStreamer->emitInt32(FeatureFlagsAnd); // data
936 emitAlignment(WordSize == 4 ? Align(4) : Align(8)); // padding
937
938 OutStreamer->switchSection(Cur);
939 }
940 }
941
942 if (TT.isOSBinFormatMachO())
943 OutStreamer->switchSection(getObjFileLowering().getTextSection());
944
945 if (TT.isOSBinFormatCOFF()) {
948
949 if (M.getModuleFlag("import-call-optimization"))
950 EnableImportCallOptimization = true;
951 }
952
953 // TODO: Support prefixed registers for the Intel syntax.
954 const bool IntelSyntax = MAI->getAssemblerDialect() == InlineAsm::AD_Intel;
955 OutStreamer->emitSyntaxDirective(IntelSyntax ? "intel" : "att",
956 IntelSyntax ? "noprefix" : "");
957
958 // If this is not inline asm and we're in 16-bit
959 // mode prefix assembly with .code16.
960 bool is16 = TT.getEnvironment() == Triple::CODE16;
961 if (M.getModuleInlineAsm().empty() && is16) {
962 auto *XTS =
963 static_cast<X86TargetStreamer *>(OutStreamer->getTargetStreamer());
964 XTS->emitCode16();
965 }
966}
967
968static void
971 // L_foo$stub:
972 OutStreamer.emitLabel(StubLabel);
973 // .indirect_symbol _foo
975
976 if (MCSym.getInt())
977 // External to current translation unit.
978 OutStreamer.emitIntValue(0, 4/*size*/);
979 else
980 // Internal to current translation unit.
981 //
982 // When we place the LSDA into the TEXT section, the type info
983 // pointers need to be indirect and pc-rel. We accomplish this by
984 // using NLPs; however, sometimes the types are local to the file.
985 // We need to fill in the value for the NLP in those cases.
986 OutStreamer.emitValue(
987 MCSymbolRefExpr::create(MCSym.getPointer(), OutStreamer.getContext()),
988 4 /*size*/);
989}
990
991static void emitNonLazyStubs(MachineModuleInfo *MMI, MCStreamer &OutStreamer) {
992
993 MachineModuleInfoMachO &MMIMacho =
995
996 // Output stubs for dynamically-linked functions.
998
999 // Output stubs for external and common global variables.
1000 Stubs = MMIMacho.GetGVStubList();
1001 if (!Stubs.empty()) {
1002 OutStreamer.switchSection(MMI->getContext().getMachOSection(
1003 "__IMPORT", "__pointers", MachO::S_NON_LAZY_SYMBOL_POINTERS,
1005
1006 for (auto &Stub : Stubs)
1007 emitNonLazySymbolPointer(OutStreamer, Stub.first, Stub.second);
1008
1009 Stubs.clear();
1010 OutStreamer.addBlankLine();
1011 }
1012}
1013
1014/// True if this module is being built for windows/msvc, and uses floating
1015/// point. This is used to emit an undefined reference to _fltused. This is
1016/// needed in Windows kernel or driver contexts to find and prevent code from
1017/// modifying non-GPR registers.
1018///
1019/// TODO: It would be better if this was computed from MIR by looking for
1020/// selected floating-point instructions.
1021static bool usesMSVCFloatingPoint(const Triple &TT, const Module &M) {
1022 // Only needed for MSVC
1023 if (!TT.isWindowsMSVCEnvironment())
1024 return false;
1025
1026 for (const Function &F : M) {
1027 for (const Instruction &I : instructions(F)) {
1028 if (I.getType()->isFloatingPointTy())
1029 return true;
1030
1031 for (const auto &Op : I.operands()) {
1032 if (Op->getType()->isFloatingPointTy())
1033 return true;
1034 }
1035 }
1036 }
1037
1038 return false;
1039}
1040
1042 const Triple &TT = TM.getTargetTriple();
1043
1044 if (TT.isOSBinFormatMachO()) {
1045 // Mach-O uses non-lazy symbol stubs to encode per-TU information into
1046 // global table for symbol lookup.
1048
1049 // Emit fault map information.
1050 FM.serializeToFaultMapSection();
1051
1052 // This flag tells the linker that no global symbols contain code that fall
1053 // through to other global symbols (e.g. an implementation of multiple entry
1054 // points). If this doesn't occur, the linker can safely perform dead code
1055 // stripping. Since LLVM never generates code that does this, it is always
1056 // safe to set.
1057 OutStreamer->emitSubsectionsViaSymbols();
1058 } else if (TT.isOSBinFormatCOFF()) {
1059 // If import call optimization is enabled, emit the appropriate section.
1060 // We do this whether or not we recorded any items.
1061 if (EnableImportCallOptimization) {
1062 OutStreamer->switchSection(getObjFileLowering().getImportCallSection());
1063
1064 // Section always starts with some magic.
1065 constexpr char ImpCallMagic[12] = "RetpolineV1";
1066 OutStreamer->emitBytes(StringRef{ImpCallMagic, sizeof(ImpCallMagic)});
1067
1068 // Layout of this section is:
1069 // Per section that contains an item to record:
1070 // uint32_t SectionSize: Size in bytes for information in this section.
1071 // uint32_t Section Number
1072 // Per call to imported function in section:
1073 // uint32_t Kind: the kind of item.
1074 // uint32_t InstOffset: the offset of the instr in its parent section.
1075 for (auto &[Section, CallsToImportedFuncs] :
1076 SectionToImportedFunctionCalls) {
1077 unsigned SectionSize =
1078 sizeof(uint32_t) * (2 + 2 * CallsToImportedFuncs.size());
1079 OutStreamer->emitInt32(SectionSize);
1080 OutStreamer->emitCOFFSecNumber(Section->getBeginSymbol());
1081 for (auto &[CallsiteSymbol, Kind] : CallsToImportedFuncs) {
1082 OutStreamer->emitInt32(Kind);
1083 OutStreamer->emitCOFFSecOffset(CallsiteSymbol);
1084 }
1085 }
1086 }
1087
1088 if (usesMSVCFloatingPoint(TT, M)) {
1089 // In Windows' libcmt.lib, there is a file which is linked in only if the
1090 // symbol _fltused is referenced. Linking this in causes some
1091 // side-effects:
1092 //
1093 // 1. For x86-32, it will set the x87 rounding mode to 53-bit instead of
1094 // 64-bit mantissas at program start.
1095 //
1096 // 2. It links in support routines for floating-point in scanf and printf.
1097 //
1098 // MSVC emits an undefined reference to _fltused when there are any
1099 // floating point operations in the program (including calls). A program
1100 // that only has: `scanf("%f", &global_float);` may fail to trigger this,
1101 // but oh well...that's a documented issue.
1102 StringRef SymbolName =
1103 (TT.getArch() == Triple::x86) ? "__fltused" : "_fltused";
1104 MCSymbol *S = MMI->getContext().getOrCreateSymbol(SymbolName);
1105 OutStreamer->emitSymbolAttribute(S, MCSA_Global);
1106 return;
1107 }
1108 } else if (TT.isOSBinFormatELF()) {
1109 FM.serializeToFaultMapSection();
1110 }
1111
1112 // Emit __morestack address if needed for indirect calls.
1113 if (TT.isX86_64() && TM.getCodeModel() == CodeModel::Large) {
1114 if (MCSymbol *AddrSymbol = OutContext.lookupSymbol("__morestack_addr")) {
1115 Align Alignment(1);
1118 /*C=*/nullptr, Alignment);
1119 OutStreamer->switchSection(ReadOnlySection);
1120 OutStreamer->emitLabel(AddrSymbol);
1121
1122 unsigned PtrSize = MAI->getCodePointerSize();
1123 OutStreamer->emitSymbolValue(GetExternalSymbolSymbol("__morestack"),
1124 PtrSize);
1125 }
1126 }
1127}
1128
1129char X86AsmPrinter::ID = 0;
1130
1131INITIALIZE_PASS(X86AsmPrinter, "x86-asm-printer", "X86 Assembly Printer", false,
1132 false)
1133
1134//===----------------------------------------------------------------------===//
1135// Target Registry Stuff
1136//===----------------------------------------------------------------------===//
1137
1138// Force static initialization.
1139extern "C" LLVM_C_ABI void LLVMInitializeX86AsmPrinter() {
1142}
1143
1146 Expected<std::unique_ptr<MCStreamer>> Streamer = CreateStreamer(TM);
1147 if (!Streamer)
1148 reportFatalInternalError("Failed to create MCStreamer");
1149 X86AsmPrinter AsmPrinter(TM, std::move(*Streamer));
1150 AsmPrinter.GetPSI = [&MAM](Module &M) {
1151 return &MAM.getResult<ProfileSummaryAnalysis>(M);
1152 };
1153 AsmPrinter.GetSDPI = [](Module &M) { return nullptr; };
1156 return PreservedAnalyses::all();
1157}
1158
1161 Expected<std::unique_ptr<MCStreamer>> Streamer = CreateStreamer(TM);
1162 if (!Streamer)
1163 reportFatalInternalError("Failed to create MCStreamer");
1164 X86AsmPrinter AsmPrinter(TM, std::move(*Streamer));
1165 AsmPrinter.GetPSI = [&MFAM, &MF](Module &M) {
1167 .getCachedResult<ProfileSummaryAnalysis>(M);
1168 };
1169 AsmPrinter.GetSDPI = [](Module &M) { return nullptr; };
1172 return PreservedAnalyses::all();
1173}
1174
1177 Expected<std::unique_ptr<MCStreamer>> Streamer = CreateStreamer(TM);
1178 if (!Streamer)
1179 reportFatalInternalError("Failed to create MCStreamer");
1180 X86AsmPrinter AsmPrinter(TM, std::move(*Streamer));
1181 AsmPrinter.GetPSI = [&MAM](Module &M) {
1182 return &MAM.getResult<ProfileSummaryAnalysis>(M);
1183 };
1184 AsmPrinter.GetSDPI = [](Module &M) { return nullptr; };
1187 return PreservedAnalyses::all();
1188}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static MCDisassembler::DecodeStatus addOperand(MCInst &Inst, const MCOperand &Opnd)
static void emitNonLazySymbolPointer(MCStreamer &OutStreamer, MCSymbol *StubLabel, MachineModuleInfoImpl::StubValueTy &MCSym)
MachineBasicBlock & MBB
Expand Atomic instructions
IRTranslator LLVM IR MI
Module.h This file contains the declarations for the Module class.
#define F(x, y, z)
Definition MD5.cpp:54
#define I(x, y, z)
Definition MD5.cpp:57
This file declares the MachineConstantPool class which is an abstract constant pool to keep track of ...
Register Reg
Promote Memory to Register
Definition Mem2Reg.cpp:110
#define P(N)
ModuleAnalysisManager MAM
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition PassSupport.h:56
static cl::opt< RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode > Mode("regalloc-enable-advisor", cl::Hidden, cl::init(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Default), cl::desc("Enable regalloc advisor mode"), cl::values(clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Default, "default", "Default"), clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Release, "release", "precompiled"), clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Development, "development", "for training")))
static bool contains(SmallPtrSetImpl< ConstantExpr * > &Cache, ConstantExpr *Expr, Constant *C)
Definition Value.cpp:487
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static TableGen::Emitter::OptClass< SkeletonEmitter > X("gen-skeleton-class", "Generate example skeleton class")
#define LLVM_C_ABI
LLVM_C_ABI is the export/visibility macro used to mark symbols declared in llvm-c as exported when bu...
Definition Visibility.h:40
static bool printAsmMRegister(const X86AsmPrinter &P, const MachineOperand &MO, char Mode, raw_ostream &O)
static bool isSimpleReturn(const MachineInstr &MI)
static bool usesMSVCFloatingPoint(const Triple &TT, const Module &M)
True if this module is being built for windows/msvc, and uses floating point.
static bool isIndirectBranchOrTailCall(const MachineInstr &MI)
static bool printAsmVRegister(const MachineOperand &MO, char Mode, raw_ostream &O)
static void emitNonLazyStubs(MachineModuleInfo *MMI, MCStreamer &OutStreamer)
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
This class is intended to be used as a driving class for all asm writers.
Definition AsmPrinter.h:91
const TargetLoweringObjectFile & getObjFileLowering() const
Return information about object file lowering.
MCSymbol * getSymbolWithGlobalValueBase(const GlobalValue *GV, StringRef Suffix) const
Return the MCSymbol for a private symbol with global value name as its base, with the specified suffi...
MCSymbol * getSymbol(const GlobalValue *GV) const
void emitNops(unsigned N)
Emit N NOP instructions.
void EmitToStreamer(MCStreamer &S, const MCInst &Inst)
TargetMachine & TM
Target machine description.
Definition AsmPrinter.h:94
void emitXRayTable()
Emit a table with all XRay instrumentation points.
virtual void emitBasicBlockEnd(const MachineBasicBlock &MBB)
Targets can override this to emit stuff at the end of a basic block.
virtual MCSymbol * GetCPISymbol(unsigned CPID) const
Return the symbol for the specified constant pool entry.
const MCAsmInfo * MAI
Target Asm Printer information.
Definition AsmPrinter.h:97
MachineFunction * MF
The current machine function.
Definition AsmPrinter.h:109
virtual const MCExpr * lowerConstant(const Constant *CV, const Constant *BaseCV=nullptr, uint64_t Offset=0)
Lower the specified LLVM Constant to an MCExpr.
virtual void SetupMachineFunction(MachineFunction &MF)
This should be called when a new MachineFunction is being processed from runOnMachineFunction.
void emitFunctionBody()
This method emits the body and trailer for a function.
bool doInitialization(Module &M) override
Set up the AsmPrinter when we are working on a new module.
virtual void emitLinkage(const GlobalValue *GV, MCSymbol *GVSym) const
This emits linkage information about GVSym based on GV, if this is supported by the target.
AsmPrinter(TargetMachine &TM, std::unique_ptr< MCStreamer > Streamer, char &ID=AsmPrinter::ID)
void printOffset(int64_t Offset, raw_ostream &OS) const
This is just convenient handler for printing offsets.
MCSymbol * getSymbolPreferLocal(const GlobalValue &GV) const
Similar to getSymbol() but preferred for references.
MCSymbol * CurrentFnSym
The symbol for the current function.
Definition AsmPrinter.h:128
MachineModuleInfo * MMI
This is a pointer to the current MachineModuleInfo.
Definition AsmPrinter.h:112
void emitAlignment(Align Alignment, const GlobalObject *GV=nullptr, unsigned MaxBytesToEmit=0) const
Emit an alignment directive to the specified power of two boundary.
MCContext & OutContext
This is the context for the output file that we are streaming.
Definition AsmPrinter.h:101
const StaticDataProfileInfo * SDPI
Provides the profile information for constants.
Definition AsmPrinter.h:147
bool doFinalization(Module &M) override
Shut down the asmprinter.
virtual const MCSubtargetInfo * getIFuncMCSubtargetInfo() const
getSubtargetInfo() cannot be used where this is needed because we don't have a MachineFunction when w...
Definition AsmPrinter.h:675
bool runOnMachineFunction(MachineFunction &MF) override
Emit the specified function out to the OutStreamer.
Definition AsmPrinter.h:458
std::unique_ptr< MCStreamer > OutStreamer
This is the MCStreamer object for the file we are generating.
Definition AsmPrinter.h:106
const ProfileSummaryInfo * PSI
The profile summary information.
Definition AsmPrinter.h:150
MCSymbol * GetBlockAddressSymbol(const BlockAddress *BA) const
Return the MCSymbol used to satisfy BlockAddress uses of the specified basic block.
const DataLayout & getDataLayout() const
Return information about data layout.
void emitCOFFFeatureSymbol(Module &M)
Emits the @feat.00 symbol indicating the features enabled in this module.
MCSymbol * GetExternalSymbolSymbol(const Twine &Sym) const
Return the MCSymbol for the specified ExternalSymbol.
const MCSubtargetInfo & getSubtargetInfo() const
Return information about subtarget.
void emitCOFFReplaceableFunctionData(Module &M)
Emits symbols and data to allow functions marked with the loader-replaceable attribute to be replacea...
virtual bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &OS)
Print the specified operand of MI, an INLINEASM instruction, using the specified assembler variant.
This is the shared class of boolean and integer constants.
Definition Constants.h:87
Tagged union holding either a T or a Error.
Definition Error.h:485
const Constant * getResolver() const
Definition GlobalIFunc.h:73
bool hasInternalLinkage() const
static const MCBinaryExpr * createSub(const MCExpr *LHS, const MCExpr *RHS, MCContext &Ctx)
Definition MCExpr.h:428
LLVM_ABI MCSectionMachO * getMachOSection(StringRef Segment, StringRef Section, unsigned TypeAndAttributes, unsigned Reserved2, SectionKind K, const char *BeginSymName=nullptr)
Return the MCSection for the specified mach-o section.
LLVM_ABI MCSymbol * getOrCreateSymbol(const Twine &Name)
Lookup the symbol inside with the specified Name.
Base class for the full range of assembler expressions which are needed for parsing.
Definition MCExpr.h:34
Instances of this class represent a single low-level machine instruction.
Definition MCInst.h:188
void setOpcode(unsigned Op)
Definition MCInst.h:201
static MCOperand createExpr(const MCExpr *Val)
Definition MCInst.h:166
Instances of this class represent a uniqued identifier for a section in the current translation unit.
Definition MCSection.h:516
Streaming machine code generation interface.
Definition MCStreamer.h:221
virtual void addBlankLine()
Emit a blank line to a .s file to pretty it up.
Definition MCStreamer.h:414
virtual bool emitSymbolAttribute(MCSymbol *Symbol, MCSymbolAttr Attribute)=0
Add the given Attribute to Symbol.
MCContext & getContext() const
Definition MCStreamer.h:322
void emitValue(const MCExpr *Value, unsigned Size, SMLoc Loc=SMLoc())
virtual void emitLabel(MCSymbol *Symbol, SMLoc Loc=SMLoc())
Emit a label for Symbol into the current section.
virtual void emitIntValue(uint64_t Value, unsigned Size)
Special case of EmitValue that avoids the client having to pass in a MCExpr for constant integers.
virtual void switchSection(MCSection *Section, uint32_t Subsec=0)
Set the current section where code is being emitted to Section.
Generic base class for all target subtargets.
static const MCSymbolRefExpr * create(const MCSymbol *Symbol, MCContext &Ctx, SMLoc Loc=SMLoc())
Definition MCExpr.h:214
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
Definition MCSymbol.h:42
LLVM_ABI void print(raw_ostream &OS, const MCAsmInfo *MAI) const
print - Print the value to the stream OS.
Definition MCSymbol.cpp:59
StringRef getName() const
getName - Get the symbol name.
Definition MCSymbol.h:188
Metadata node.
Definition Metadata.h:1080
Representation of each machine instruction.
LLVM_ABI InlineAsm::AsmDialect getInlineAsmDialect() const
std::vector< std::pair< MCSymbol *, StubValueTy > > SymbolListTy
PointerIntPair< MCSymbol *, 1, bool > StubValueTy
MachineModuleInfoMachO - This is a MachineModuleInfoImpl implementation for MachO targets.
SymbolListTy GetGVStubList()
Accessor methods to return the set of stubs in sorted order.
This class contains meta information specific to a module.
const MCContext & getContext() const
Ty & getObjFileInfo()
Keep track of various per-module pieces of information for backends that would like to do so.
MachineOperand class - Representation of each machine instruction operand.
const GlobalValue * getGlobal() const
int64_t getImm() const
bool isReg() const
isReg - Tests if this is a MO_Register operand.
bool isImm() const
isImm - Tests if this is a MO_Immediate operand.
bool isSymbol() const
isSymbol - Tests if this is a MO_ExternalSymbol operand.
const BlockAddress * getBlockAddress() const
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
unsigned getTargetFlags() const
bool isGlobal() const
isGlobal - Tests if this is a MO_GlobalAddress operand.
MachineOperandType getType() const
getType - Returns the MachineOperandType for this operand.
Register getReg() const
getReg - Returns the register number.
@ MO_Immediate
Immediate operand.
@ MO_ConstantPoolIndex
Address of indexed Constant in Constant Pool.
@ MO_GlobalAddress
Address of a global value.
@ MO_BlockAddress
Address of a basic block.
@ MO_Register
Register operand.
@ MO_ExternalSymbol
Name of external global symbol.
@ MO_JumpTableIndex
Address of indexed Jump Table for switch.
int64_t getOffset() const
Return the offset from the symbol in this operand.
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
AnalysisType * getAnalysisIfAvailable() const
getAnalysisIfAvailable<AnalysisType>() - Subclasses use this function to get analysis information tha...
IntType getInt() const
PointerTy getPointer() const
A set of analyses that are preserved following a run of a transformation pass.
Definition Analysis.h:112
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Definition Analysis.h:118
An analysis pass based on the new PM to deliver ProfileSummaryInfo.
Analysis providing profile information.
Wrapper class representing virtual and physical registers.
Definition Register.h:20
static SectionKind getMetadata()
static SectionKind getReadOnly()
A class that holds the constants that represent static data and their profile information and provide...
StringRef - Represent a constant reference to a string, i.e.
Definition StringRef.h:55
constexpr bool empty() const
empty - Check if the string is empty.
Definition StringRef.h:140
bool consume_front(char Prefix)
Returns true if this StringRef has the given prefix and removes that prefix.
Definition StringRef.h:655
virtual MCSection * getSectionForConstant(const DataLayout &DL, SectionKind Kind, const Constant *C, Align &Alignment) const
Given a constant with the SectionKind, return a section that it should be placed in.
Primary interface to the complete machine description for the target machine.
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
bool isX86_64() const
Tests whether the target is x86 (64-bit).
Definition Triple.h:1158
bool isOSWindows() const
Tests whether the OS is Windows.
Definition Triple.h:708
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
Definition Twine.h:82
The instances of the Type class are immutable: once they are created, they are never changed.
Definition Type.h:45
LLVM Value Representation.
Definition Value.h:75
static const char * getRegisterName(MCRegister Reg)
PreservedAnalyses run(Module &M, ModuleAnalysisManager &MAM)
PreservedAnalyses run(Module &M, ModuleAnalysisManager &MAM)
PreservedAnalyses run(MachineFunction &MF, MachineFunctionAnalysisManager &MFAM)
bool runOnMachineFunction(MachineFunction &MF) override
runOnMachineFunction - Emit the function body.
void emitKCFITypeId(const MachineFunction &MF) override
emitKCFITypeId - Emit the KCFI type information in architecture specific format.
void emitStartOfAsmFile(Module &M) override
This virtual method can be overridden by targets that want to emit something at the start of their fi...
void emitEndOfAsmFile(Module &M) override
This virtual method can be overridden by targets that want to emit something at the end of their file...
std::function< ProfileSummaryInfo *(Module &)> GetPSI
void emitFunctionBodyEnd() override
Targets can override this to emit stuff after the last basic block in the function.
std::function< StaticDataProfileInfo *(Module &)> GetSDPI
bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &O) override
Print the specified operand of MI, an INLINEASM instruction, using the specified assembler variant as...
void emitBasicBlockEnd(const MachineBasicBlock &MBB) override
Targets can override this to emit stuff at the end of a basic block.
X86AsmPrinter(TargetMachine &TM, std::unique_ptr< MCStreamer > Streamer)
bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &O) override
PrintAsmOperand - Print out an operand for an inline asm expression.
void emitFunctionBodyStart() override
Targets can override this to emit stuff before the first basic block in the function.
X86MachineFunctionInfo - This class is derived from MachineFunction and contains private X86 target-s...
X86 target streamer implementing x86-only assembly directives.
virtual bool emitFPOProc(const MCSymbol *ProcSym, unsigned ParamsSize, SMLoc L={})
virtual bool emitFPOEndProc(SMLoc L={})
This class implements an extremely fast bulk output stream that can only output to a stream.
Definition raw_ostream.h:53
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ IMAGE_SYM_CLASS_EXTERNAL
External symbol.
Definition COFF.h:224
@ IMAGE_SYM_CLASS_STATIC
Static.
Definition COFF.h:225
@ IMAGE_SYM_DTYPE_FUNCTION
A function that returns a base type.
Definition COFF.h:276
@ SCT_COMPLEX_TYPE_SHIFT
Type is formed as (base + (derived << SCT_COMPLEX_TYPE_SHIFT))
Definition COFF.h:280
@ NT_GNU_PROPERTY_TYPE_0
Definition ELF.h:1812
@ SHF_ALLOC
Definition ELF.h:1250
@ SHT_NOTE
Definition ELF.h:1155
@ GNU_PROPERTY_X86_FEATURE_1_AND
Definition ELF.h:1847
@ GNU_PROPERTY_X86_FEATURE_1_SHSTK
Definition ELF.h:1894
@ GNU_PROPERTY_X86_FEATURE_1_IBT
Definition ELF.h:1893
@ S_NON_LAZY_SYMBOL_POINTERS
S_NON_LAZY_SYMBOL_POINTERS - Section with non-lazy symbol pointers.
Definition MachO.h:139
@ MO_TLSLD
MO_TLSLD - On a symbol operand this indicates that the immediate is the offset of the GOT entry with ...
@ MO_GOTPCREL_NORELAX
MO_GOTPCREL_NORELAX - Same as MO_GOTPCREL except that R_X86_64_GOTPCREL relocations are guaranteed to...
@ MO_GOTOFF
MO_GOTOFF - On a symbol operand this indicates that the immediate is the offset to the location of th...
@ MO_DARWIN_NONLAZY_PIC_BASE
MO_DARWIN_NONLAZY_PIC_BASE - On a symbol operand "FOO", this indicates that the reference is actually...
@ MO_GOT_ABSOLUTE_ADDRESS
MO_GOT_ABSOLUTE_ADDRESS - On a symbol operand, this represents a relocation of: SYMBOL_LABEL + [.
@ MO_COFFSTUB
MO_COFFSTUB - On a symbol operand "FOO", this indicates that the reference is actually to the "....
@ MO_NTPOFF
MO_NTPOFF - On a symbol operand this indicates that the immediate is the negative thread-pointer offs...
@ MO_DARWIN_NONLAZY
MO_DARWIN_NONLAZY - On a symbol operand "FOO", this indicates that the reference is actually to the "...
@ MO_INDNTPOFF
MO_INDNTPOFF - On a symbol operand this indicates that the immediate is the absolute address of the G...
@ MO_GOTNTPOFF
MO_GOTNTPOFF - On a symbol operand this indicates that the immediate is the offset of the GOT entry w...
@ MO_TPOFF
MO_TPOFF - On a symbol operand this indicates that the immediate is the thread-pointer offset for the...
@ MO_TLVP_PIC_BASE
MO_TLVP_PIC_BASE - On a symbol operand this indicates that the immediate is some TLS offset from the ...
@ MO_GOT
MO_GOT - On a symbol operand this indicates that the immediate is the offset to the GOT entry for the...
@ MO_PLT
MO_PLT - On a symbol operand this indicates that the immediate is offset to the PLT entry of symbol n...
@ MO_TLSGD
MO_TLSGD - On a symbol operand this indicates that the immediate is the offset of the GOT entry with ...
@ MO_NO_FLAG
MO_NO_FLAG - No flag for the operand.
@ MO_TLVP
MO_TLVP - On a symbol operand this indicates that the immediate is some TLS offset.
@ MO_DLLIMPORT
MO_DLLIMPORT - On a symbol operand "FOO", this indicates that the reference is actually to the "__imp...
@ MO_GOTTPOFF
MO_GOTTPOFF - On a symbol operand this indicates that the immediate is the offset of the GOT entry wi...
@ MO_SECREL
MO_SECREL - On a symbol operand this indicates that the immediate is the offset from beginning of sec...
@ MO_DTPOFF
MO_DTPOFF - On a symbol operand this indicates that the immediate is the offset of the GOT entry with...
@ MO_PIC_BASE_OFFSET
MO_PIC_BASE_OFFSET - On a symbol operand this indicates that the immediate should get the value of th...
@ MO_TLSLDM
MO_TLSLDM - On a symbol operand this indicates that the immediate is the offset of the GOT entry with...
@ MO_GOTPCREL
MO_GOTPCREL - On a symbol operand this indicates that the immediate is offset to the GOT entry for th...
std::enable_if_t< detail::IsValidPointer< X, Y >::value, X * > extract(Y &&MD)
Extract a Value from Metadata.
Definition Metadata.h:668
BaseReg
Stack frame base register. Bit 0 of FREInfo.Info.
Definition SFrame.h:77
This is an optimization pass for GlobalISel generic memory operations.
Definition Types.h:26
FunctionAddr VTableAddr Value
Definition InstrProf.h:137
OuterAnalysisManagerProxy< ModuleAnalysisManager, MachineFunction > ModuleAnalysisManagerMachineFunctionProxy
Provide the ModuleAnalysisManager to Function proxy.
static bool isMem(const MachineInstr &MI, unsigned Op)
MCRegister getX86SubSuperRegister(MCRegister Reg, unsigned Size, bool High=false)
Target & getTheX86_32Target()
AnalysisManager< MachineFunction > MachineFunctionAnalysisManager
LLVM_ABI void reportFatalInternalError(Error Err)
Report a fatal error that indicates a bug in LLVM.
Definition Error.cpp:173
void setupModuleAsmPrinter(Module &M, ModuleAnalysisManager &MAM, AsmPrinter &AsmPrinter)
uint64_t offsetToAlignment(uint64_t Value, Align Alignment)
Returns the offset to the next integer (mod 2**64) that is greater than or equal to Value and is a mu...
Definition Alignment.h:186
DWARFExpression::Operation Op
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
Definition STLExtras.h:1917
void setupMachineFunctionAsmPrinter(MachineFunctionAnalysisManager &MFAM, MachineFunction &MF, AsmPrinter &AsmPrinter)
Target & getTheX86_64Target()
@ MCSA_IndirectSymbol
.indirect_symbol (MachO)
@ MCSA_Global
.type _foo, @gnu_unique_object
@ MCSA_ELF_TypeFunction
.type _foo, STT_FUNC # aka @function
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
Definition MIRParser.h:39
Implement std::hash so that hash_code can be used in STL containers.
Definition BitVector.h:870
#define N
This struct is a compact representation of a valid (non-zero power of two) alignment.
Definition Alignment.h:39
RegisterAsmPrinter - Helper template for registering a target specific assembly printer,...