31#include "llvm/Config/llvm-config.h"
107 if (
const auto *MAV = dyn_cast<MetadataAsValue>(V))
108 if (
const auto *VAM = dyn_cast<ValueAsMetadata>(MAV->getMetadata()))
109 return VAM->getValue();
117 if (
const Constant *
C = dyn_cast<Constant>(V))
118 if (
C->getNumOperands() && !isa<GlobalValue>(
C))
119 for (
const Value *
Op :
C->operands())
120 if (!isa<BasicBlock>(
Op) && !isa<GlobalValue>(
Op))
125 unsigned ID = OM.size() + 1;
133 if (
G.hasInitializer())
134 if (!isa<GlobalValue>(
G.getInitializer()))
139 if (!isa<GlobalValue>(
A.getAliasee()))
144 if (!isa<GlobalValue>(
I.getResolver()))
149 for (
const Use &U :
F.operands())
150 if (!isa<GlobalValue>(U.get()))
155 if (
F.isDeclaration())
163 for (
const Value *
Op :
I.operands()) {
165 if ((isa<Constant>(*
Op) && !isa<GlobalValue>(*
Op)) ||
176static std::vector<unsigned>
179 using Entry = std::pair<const Use *, unsigned>;
181 for (
const Use &U : V->uses())
183 if (OM.lookup(U.getUser()))
184 List.push_back(std::make_pair(&U,
List.size()));
193 bool GetsReversed = !isa<BasicBlock>(V);
194 if (
auto *BA = dyn_cast<BlockAddress>(V))
195 ID = OM.lookup(BA->getBasicBlock());
197 const Use *LU = L.first;
198 const Use *RU = R.first;
202 auto LID = OM.lookup(LU->getUser());
203 auto RID = OM.lookup(RU->getUser());
223 return LU->getOperandNo() < RU->getOperandNo();
224 return LU->getOperandNo() > RU->getOperandNo();
232 std::vector<unsigned> Shuffle(
List.size());
233 for (
size_t I = 0, E =
List.size();
I != E; ++
I)
234 Shuffle[
I] =
List[
I].second;
241 for (
const auto &Pair : OM) {
242 const Value *V = Pair.first;
243 if (V->use_empty() || std::next(V->use_begin()) == V->use_end())
246 std::vector<unsigned> Shuffle =
252 if (
auto *
I = dyn_cast<Instruction>(V))
253 F =
I->getFunction();
254 if (
auto *
A = dyn_cast<Argument>(V))
256 if (
auto *BB = dyn_cast<BasicBlock>(V))
258 ULOM[
F][V] = std::move(Shuffle);
264 if (
const Argument *MA = dyn_cast<Argument>(V))
265 return MA->getParent() ? MA->getParent()->getParent() :
nullptr;
267 if (
const BasicBlock *BB = dyn_cast<BasicBlock>(V))
268 return BB->getParent() ? BB->getParent()->getParent() :
nullptr;
271 const Function *M =
I->getParent() ?
I->getParent()->getParent() :
nullptr;
272 return M ? M->getParent() :
nullptr;
275 if (
const GlobalValue *GV = dyn_cast<GlobalValue>(V))
276 return GV->getParent();
278 if (
const auto *MAV = dyn_cast<MetadataAsValue>(V)) {
279 for (
const User *U : MAV->users())
280 if (isa<Instruction>(U))
292 return M ? M->getParent() :
nullptr;
301 default: Out <<
"cc" << cc;
break;
324 Out <<
"aarch64_sve_vector_pcs";
327 Out <<
"aarch64_sme_preservemost_from_x0";
330 Out <<
"aarch64_sme_preservemost_from_x1";
333 Out <<
"aarch64_sme_preservemost_from_x2";
361 Out <<
"amdgpu_cs_chain";
364 Out <<
"amdgpu_cs_chain_preserve";
370 Out <<
"riscv_vector_cc";
384 assert(!
Name.empty() &&
"Cannot get empty name!");
387 bool NeedsQuotes = isdigit(
static_cast<unsigned char>(
Name[0]));
389 for (
unsigned char C :
Name) {
394 if (!isalnum(
static_cast<unsigned char>(
C)) &&
C !=
'-' &&
C !=
'.' &&
411 printEscapedString(
Name,
OS);
447 if (isa<ScalableVectorType>(Ty))
449 Out << Mask.size() <<
" x i32> ";
450 bool FirstElt =
true;
451 if (
all_of(Mask, [](
int Elt) {
return Elt == 0; })) {
452 Out <<
"zeroinitializer";
457 for (
int Elt : Mask) {
476 TypePrinting(
const Module *M =
nullptr) : DeferredM(
M) {}
478 TypePrinting(
const TypePrinting &) =
delete;
479 TypePrinting &operator=(
const TypePrinting &) =
delete;
485 std::vector<StructType *> &getNumberedTypes();
494 void incorporateTypes();
504 std::vector<StructType *> NumberedTypes;
514std::vector<StructType *> &TypePrinting::getNumberedTypes() {
520 if (NumberedTypes.size() == Type2Number.size())
521 return NumberedTypes;
523 NumberedTypes.resize(Type2Number.size());
524 for (
const auto &
P : Type2Number) {
525 assert(
P.second < NumberedTypes.size() &&
"Didn't get a dense numbering?");
526 assert(!NumberedTypes[
P.second] &&
"Didn't get a unique numbering?");
527 NumberedTypes[
P.second] =
P.first;
529 return NumberedTypes;
532bool TypePrinting::empty() {
534 return NamedTypes.empty() && Type2Number.empty();
537void TypePrinting::incorporateTypes() {
541 NamedTypes.run(*DeferredM,
false);
546 unsigned NextNumber = 0;
548 std::vector<StructType *>::iterator NextToUse = NamedTypes.begin();
551 if (STy->isLiteral())
554 if (STy->getName().empty())
555 Type2Number[STy] = NextNumber++;
560 NamedTypes.erase(NextToUse, NamedTypes.end());
582 OS << 'i' << cast<IntegerType>(Ty)->getBitWidth();
587 print(FTy->getReturnType(),
OS);
590 for (
Type *Ty : FTy->params()) {
603 return printStructBody(STy,
OS);
609 const auto I = Type2Number.find(STy);
610 if (
I != Type2Number.end())
611 OS <<
'%' <<
I->second;
613 OS <<
"%\"type " << STy <<
'\"';
625 OS <<
'[' << ATy->getNumElements() <<
" x ";
626 print(ATy->getElementType(),
OS);
637 OS <<
EC.getKnownMinValue() <<
" x ";
638 print(PTy->getElementType(),
OS);
654 OS <<
", " << *Inner;
656 OS <<
", " << IntParam;
707 const Function* TheFunction =
nullptr;
708 bool FunctionProcessed =
false;
709 bool ShouldInitializeAllMetadata;
714 ProcessFunctionHookFn;
729 unsigned mdnNext = 0;
737 unsigned ModulePathNext = 0;
741 unsigned GUIDNext = 0;
745 unsigned TypeIdNext = 0;
750 unsigned TypeIdCompatibleVtableNext = 0;
759 bool ShouldInitializeAllMetadata =
false);
767 bool ShouldInitializeAllMetadata =
false);
801 FunctionProcessed =
false;
840 void CreateMetadataSlot(
const MDNode *
N);
843 void CreateFunctionSlot(
const Value *V);
848 inline void CreateModulePathSlot(
StringRef Path);
851 void CreateTypeIdCompatibleVtableSlot(
StringRef Id);
855 void processModule();
860 void processFunction();
863 void processGlobalObjectMetadata(
const GlobalObject &GO);
866 void processFunctionMetadata(
const Function &
F);
872 void processDbgRecordMetadata(
const DbgRecord &DVR);
882 bool ShouldInitializeAllMetadata)
883 : ShouldCreateStorage(M),
884 ShouldInitializeAllMetadata(ShouldInitializeAllMetadata), M(M) {}
889 if (!ShouldCreateStorage)
892 ShouldCreateStorage =
false;
894 std::make_unique<SlotTracker>(M, ShouldInitializeAllMetadata);
895 Machine = MachineStorage.get();
896 if (ProcessModuleHookFn)
898 if (ProcessFunctionHookFn)
918 assert(F &&
"No function incorporated");
925 ProcessModuleHookFn = Fn;
931 ProcessFunctionHookFn = Fn;
935 if (
const Argument *FA = dyn_cast<Argument>(V))
942 if (
const BasicBlock *BB = dyn_cast<BasicBlock>(V))
948 if (
const GlobalAlias *GA = dyn_cast<GlobalAlias>(V))
951 if (
const GlobalIFunc *GIF = dyn_cast<GlobalIFunc>(V))
954 if (
const Function *Func = dyn_cast<Function>(V))
961#define ST_DEBUG(X) dbgs() << X
969 : TheModule(M), ShouldInitializeAllMetadata(ShouldInitializeAllMetadata) {}
974 : TheModule(
F ?
F->
getParent() : nullptr), TheFunction(
F),
975 ShouldInitializeAllMetadata(ShouldInitializeAllMetadata) {}
978 : TheModule(nullptr), ShouldInitializeAllMetadata(
false), TheIndex(
Index) {}
986 if (TheFunction && !FunctionProcessed)
993 int NumSlots = processIndex();
1000void SlotTracker::processModule() {
1001 ST_DEBUG(
"begin processModule!\n");
1006 CreateModuleSlot(&Var);
1007 processGlobalObjectMetadata(Var);
1008 auto Attrs = Var.getAttributes();
1009 if (Attrs.hasAttributes())
1010 CreateAttributeSetSlot(Attrs);
1015 CreateModuleSlot(&
A);
1020 CreateModuleSlot(&
I);
1025 for (
const MDNode *
N : NMD.operands())
1026 CreateMetadataSlot(
N);
1032 CreateModuleSlot(&
F);
1034 if (ShouldInitializeAllMetadata)
1035 processFunctionMetadata(
F);
1041 CreateAttributeSetSlot(FnAttrs);
1044 if (ProcessModuleHookFn)
1045 ProcessModuleHookFn(
this, TheModule, ShouldInitializeAllMetadata);
1051void SlotTracker::processFunction() {
1052 ST_DEBUG(
"begin processFunction!\n");
1056 if (!ShouldInitializeAllMetadata)
1057 processFunctionMetadata(*TheFunction);
1061 AE = TheFunction->
arg_end(); AI != AE; ++AI)
1063 CreateFunctionSlot(&*AI);
1065 ST_DEBUG(
"Inserting Instructions:\n");
1068 for (
auto &BB : *TheFunction) {
1070 CreateFunctionSlot(&BB);
1072 for (
auto &
I : BB) {
1073 if (!
I.getType()->isVoidTy() && !
I.hasName())
1074 CreateFunctionSlot(&
I);
1078 if (
const auto *Call = dyn_cast<CallBase>(&
I)) {
1081 if (
Attrs.hasAttributes())
1082 CreateAttributeSetSlot(Attrs);
1087 if (ProcessFunctionHookFn)
1088 ProcessFunctionHookFn(
this, TheFunction, ShouldInitializeAllMetadata);
1090 FunctionProcessed =
true;
1092 ST_DEBUG(
"end processFunction!\n");
1096int SlotTracker::processIndex() {
1103 std::vector<StringRef> ModulePaths;
1105 ModulePaths.push_back(ModPath);
1106 llvm::sort(ModulePaths.begin(), ModulePaths.end());
1107 for (
auto &ModPath : ModulePaths)
1108 CreateModulePathSlot(ModPath);
1111 GUIDNext = ModulePathNext;
1113 for (
auto &GlobalList : *TheIndex)
1114 CreateGUIDSlot(GlobalList.first);
1117 TypeIdCompatibleVtableNext = GUIDNext;
1118 for (
auto &TId : TheIndex->typeIdCompatibleVtableMap())
1119 CreateTypeIdCompatibleVtableSlot(TId.first);
1122 TypeIdNext = TypeIdCompatibleVtableNext;
1123 for (
const auto &TID : TheIndex->typeIds())
1124 CreateTypeIdSlot(TID.second.first);
1130void SlotTracker::processGlobalObjectMetadata(
const GlobalObject &GO) {
1133 for (
auto &MD : MDs)
1134 CreateMetadataSlot(MD.second);
1137void SlotTracker::processFunctionMetadata(
const Function &
F) {
1138 processGlobalObjectMetadata(
F);
1139 for (
auto &BB :
F) {
1140 for (
auto &
I : BB) {
1141 for (
const DbgRecord &DR :
I.getDbgRecordRange())
1142 processDbgRecordMetadata(DR);
1143 processInstructionMetadata(
I);
1148void SlotTracker::processDbgRecordMetadata(
const DbgRecord &DR) {
1154 if (
auto *
Empty = dyn_cast<MDNode>(DVR->getRawLocation()))
1155 CreateMetadataSlot(
Empty);
1156 CreateMetadataSlot(DVR->getRawVariable());
1157 if (DVR->isDbgAssign()) {
1158 CreateMetadataSlot(cast<MDNode>(DVR->getRawAssignID()));
1159 if (
auto *
Empty = dyn_cast<MDNode>(DVR->getRawAddress()))
1160 CreateMetadataSlot(
Empty);
1162 }
else if (
const DbgLabelRecord *DLR = dyn_cast<const DbgLabelRecord>(&DR)) {
1163 CreateMetadataSlot(DLR->getRawLabel());
1170void SlotTracker::processInstructionMetadata(
const Instruction &
I) {
1172 if (
const CallInst *CI = dyn_cast<CallInst>(&
I))
1173 if (
Function *
F = CI->getCalledFunction())
1174 if (
F->isIntrinsic())
1175 for (
auto &
Op :
I.operands())
1176 if (
auto *V = dyn_cast_or_null<MetadataAsValue>(
Op))
1177 if (
MDNode *
N = dyn_cast<MDNode>(
V->getMetadata()))
1178 CreateMetadataSlot(
N);
1182 I.getAllMetadata(MDs);
1183 for (
auto &MD : MDs)
1184 CreateMetadataSlot(MD.second);
1191 ST_DEBUG(
"begin purgeFunction!\n");
1193 TheFunction =
nullptr;
1194 FunctionProcessed =
false;
1205 return MI == mMap.
end() ? -1 : (int)
MI->second;
1211 ProcessModuleHookFn = Fn;
1217 ProcessFunctionHookFn = Fn;
1230 return MI == mdnMap.
end() ? -1 : (int)
MI->second;
1235 assert(!isa<Constant>(V) &&
"Can't get a constant or global slot with this!");
1241 return FI == fMap.
end() ? -1 : (int)FI->second;
1250 return AI == asMap.
end() ? -1 : (int)AI->second;
1258 auto I = ModulePathMap.
find(Path);
1259 return I == ModulePathMap.
end() ? -1 : (int)
I->second;
1268 return I == GUIDMap.
end() ? -1 : (int)
I->second;
1276 auto I = TypeIdMap.
find(Id);
1277 return I == TypeIdMap.
end() ? -1 : (int)
I->second;
1285 auto I = TypeIdCompatibleVtableMap.
find(Id);
1286 return I == TypeIdCompatibleVtableMap.
end() ? -1 : (int)
I->second;
1290void SlotTracker::CreateModuleSlot(
const GlobalValue *V) {
1291 assert(V &&
"Can't insert a null Value into SlotTracker!");
1292 assert(!V->getType()->isVoidTy() &&
"Doesn't need a slot!");
1293 assert(!V->hasName() &&
"Doesn't need a slot!");
1295 unsigned DestSlot = mNext++;
1298 ST_DEBUG(
" Inserting value [" << V->getType() <<
"] = " << V <<
" slot=" <<
1301 ST_DEBUG((isa<GlobalVariable>(V) ?
'G' :
1302 (isa<Function>(V) ?
'F' :
1303 (isa<GlobalAlias>(V) ?
'A' :
1304 (isa<GlobalIFunc>(V) ?
'I' :
'o')))) <<
"]\n");
1308void SlotTracker::CreateFunctionSlot(
const Value *V) {
1309 assert(!V->getType()->isVoidTy() && !V->hasName() &&
"Doesn't need a slot!");
1311 unsigned DestSlot = fNext++;
1315 ST_DEBUG(
" Inserting value [" << V->getType() <<
"] = " << V <<
" slot=" <<
1316 DestSlot <<
" [o]\n");
1320void SlotTracker::CreateMetadataSlot(
const MDNode *
N) {
1321 assert(
N &&
"Can't insert a null Value into SlotTracker!");
1324 if (isa<DIExpression>(
N))
1327 unsigned DestSlot = mdnNext;
1328 if (!mdnMap.
insert(std::make_pair(
N, DestSlot)).second)
1333 for (
unsigned i = 0, e =
N->getNumOperands(); i != e; ++i)
1334 if (
const MDNode *
Op = dyn_cast_or_null<MDNode>(
N->getOperand(i)))
1335 CreateMetadataSlot(
Op);
1338void SlotTracker::CreateAttributeSetSlot(
AttributeSet AS) {
1342 if (
I != asMap.
end())
1345 unsigned DestSlot = asNext++;
1346 asMap[AS] = DestSlot;
1350void SlotTracker::CreateModulePathSlot(
StringRef Path) {
1351 ModulePathMap[
Path] = ModulePathNext++;
1356 GUIDMap[
GUID] = GUIDNext++;
1360void SlotTracker::CreateTypeIdSlot(
StringRef Id) {
1361 TypeIdMap[
Id] = TypeIdNext++;
1365void SlotTracker::CreateTypeIdCompatibleVtableSlot(
StringRef Id) {
1366 TypeIdCompatibleVtableMap[
Id] = TypeIdCompatibleVtableNext++;
1371struct AsmWriterContext {
1372 TypePrinting *TypePrinter =
nullptr;
1374 const Module *Context =
nullptr;
1377 : TypePrinter(TP),
Machine(
ST), Context(
M) {}
1379 static AsmWriterContext &getEmpty() {
1380 static AsmWriterContext EmptyCtx(
nullptr,
nullptr);
1386 virtual void onWriteMetadataAsOperand(
const Metadata *) {}
1388 virtual ~AsmWriterContext() =
default;
1397 AsmWriterContext &WriterCtx);
1400 AsmWriterContext &WriterCtx,
1401 bool FromValue =
false);
1404 if (
const FPMathOperator *FPO = dyn_cast<const FPMathOperator>(U))
1405 Out << FPO->getFastMathFlags();
1408 dyn_cast<OverflowingBinaryOperator>(U)) {
1409 if (OBO->hasNoUnsignedWrap())
1411 if (OBO->hasNoSignedWrap())
1414 dyn_cast<PossiblyExactOperator>(U)) {
1418 dyn_cast<PossiblyDisjointInst>(U)) {
1419 if (PDI->isDisjoint())
1422 if (
GEP->isInBounds())
1424 else if (
GEP->hasNoUnsignedSignedWrap())
1426 if (
GEP->hasNoUnsignedWrap())
1429 Out <<
" inrange(" <<
InRange->getLower() <<
", " <<
InRange->getUpper()
1432 }
else if (
const auto *NNI = dyn_cast<PossiblyNonNegInst>(U)) {
1433 if (NNI->hasNonNeg())
1435 }
else if (
const auto *TI = dyn_cast<TruncInst>(U)) {
1436 if (TI->hasNoUnsignedWrap())
1438 if (TI->hasNoSignedWrap())
1454 bool isNaN = APF.
isNaN();
1456 if (!isInf && !isNaN) {
1465 ((StrVal[0] ==
'-' || StrVal[0] ==
'+') &&
isDigit(StrVal[1]))) &&
1466 "[-+]?[0-9] regex does not match!");
1478 static_assert(
sizeof(double) ==
sizeof(
uint64_t),
1479 "assuming that double is 64 bits!");
1537 AsmWriterContext &WriterCtx) {
1538 if (
const ConstantInt *CI = dyn_cast<ConstantInt>(CV)) {
1539 Type *Ty = CI->getType();
1548 Out << (CI->getZExtValue() ?
"true" :
"false");
1550 Out << CI->getValue();
1558 if (
const ConstantFP *CFP = dyn_cast<ConstantFP>(CV)) {
1559 Type *Ty = CFP->getType();
1575 if (isa<ConstantAggregateZero>(CV) || isa<ConstantTargetNone>(CV)) {
1576 Out <<
"zeroinitializer";
1580 if (
const BlockAddress *BA = dyn_cast<BlockAddress>(CV)) {
1581 Out <<
"blockaddress(";
1589 if (
const auto *Equiv = dyn_cast<DSOLocalEquivalent>(CV)) {
1590 Out <<
"dso_local_equivalent ";
1595 if (
const auto *
NC = dyn_cast<NoCFIValue>(CV)) {
1605 unsigned NumOpsToWrite = 2;
1606 if (!CPA->getOperand(2)->isNullValue())
1608 if (!CPA->getOperand(3)->isNullValue())
1612 for (
unsigned i = 0, e = NumOpsToWrite; i != e; ++i) {
1614 WriterCtx.TypePrinter->print(CPA->getOperand(i)->getType(), Out);
1622 if (
const ConstantArray *CA = dyn_cast<ConstantArray>(CV)) {
1623 Type *ETy = CA->getType()->getElementType();
1625 WriterCtx.TypePrinter->print(ETy, Out);
1628 for (
unsigned i = 1, e = CA->getNumOperands(); i != e; ++i) {
1630 WriterCtx.TypePrinter->print(ETy, Out);
1641 if (CA->isString()) {
1643 printEscapedString(CA->getAsString(), Out);
1648 Type *ETy = CA->getType()->getElementType();
1650 WriterCtx.TypePrinter->print(ETy, Out);
1653 for (
unsigned i = 1, e = CA->getNumElements(); i != e; ++i) {
1655 WriterCtx.TypePrinter->print(ETy, Out);
1664 if (CS->getType()->isPacked())
1667 unsigned N = CS->getNumOperands();
1670 WriterCtx.TypePrinter->print(CS->getOperand(0)->getType(), Out);
1675 for (
unsigned i = 1; i <
N; i++) {
1677 WriterCtx.TypePrinter->print(CS->getOperand(i)->getType(), Out);
1686 if (CS->getType()->isPacked())
1691 if (isa<ConstantVector>(CV) || isa<ConstantDataVector>(CV)) {
1692 auto *CVVTy = cast<FixedVectorType>(CV->
getType());
1693 Type *ETy = CVVTy->getElementType();
1695 WriterCtx.TypePrinter->print(ETy, Out);
1698 for (
unsigned i = 1, e = CVVTy->getNumElements(); i != e; ++i) {
1700 WriterCtx.TypePrinter->print(ETy, Out);
1708 if (isa<ConstantPointerNull>(CV)) {
1713 if (isa<ConstantTokenNone>(CV)) {
1718 if (isa<PoisonValue>(CV)) {
1723 if (isa<UndefValue>(CV)) {
1728 if (
const ConstantExpr *CE = dyn_cast<ConstantExpr>(CV)) {
1729 Out << CE->getOpcodeName();
1734 WriterCtx.TypePrinter->print(
GEP->getSourceElementType(), Out);
1740 WriterCtx.TypePrinter->print((*OI)->getType(), Out);
1743 if (OI+1 != CE->op_end())
1749 WriterCtx.TypePrinter->print(CE->getType(), Out);
1752 if (CE->getOpcode() == Instruction::ShuffleVector)
1759 Out <<
"<placeholder or erroneous Constant>";
1763 AsmWriterContext &WriterCtx) {
1765 for (
unsigned mi = 0, me =
Node->getNumOperands(); mi != me; ++mi) {
1769 else if (
auto *MDV = dyn_cast<ValueAsMetadata>(MD)) {
1770 Value *V = MDV->getValue();
1771 WriterCtx.TypePrinter->print(V->getType(), Out);
1776 WriterCtx.onWriteMetadataAsOperand(MD);
1787struct FieldSeparator {
1791 FieldSeparator(
const char *Sep =
", ") : Sep(Sep) {}
1799 return OS <<
FS.Sep;
1802struct MDFieldPrinter {
1805 AsmWriterContext &WriterCtx;
1808 : Out(Out), WriterCtx(AsmWriterContext::getEmpty()) {}
1809 MDFieldPrinter(
raw_ostream &Out, AsmWriterContext &Ctx)
1810 : Out(Out), WriterCtx(Ctx) {}
1812 void printTag(
const DINode *
N);
1816 bool ShouldSkipEmpty =
true);
1818 bool ShouldSkipNull =
true);
1819 template <
class IntTy>
1822 bool ShouldSkipZero);
1824 std::optional<bool>
Default = std::nullopt);
1827 template <
class IntTy,
class Stringifier>
1829 bool ShouldSkipZero =
true);
1837void MDFieldPrinter::printTag(
const DINode *
N) {
1838 Out <<
FS <<
"tag: ";
1846void MDFieldPrinter::printMacinfoType(
const DIMacroNode *
N) {
1847 Out <<
FS <<
"type: ";
1852 Out <<
N->getMacinfoType();
1855void MDFieldPrinter::printChecksum(
1858 printString(
"checksum", Checksum.
Value,
false);
1862 bool ShouldSkipEmpty) {
1863 if (ShouldSkipEmpty &&
Value.empty())
1866 Out <<
FS <<
Name <<
": \"";
1867 printEscapedString(
Value, Out);
1872 AsmWriterContext &WriterCtx) {
1878 WriterCtx.onWriteMetadataAsOperand(MD);
1882 bool ShouldSkipNull) {
1883 if (ShouldSkipNull && !MD)
1886 Out <<
FS <<
Name <<
": ";
1890template <
class IntTy>
1891void MDFieldPrinter::printInt(
StringRef Name, IntTy
Int,
bool ShouldSkipZero) {
1892 if (ShouldSkipZero && !
Int)
1899 bool IsUnsigned,
bool ShouldSkipZero) {
1900 if (ShouldSkipZero &&
Int.isZero())
1903 Out <<
FS <<
Name <<
": ";
1904 Int.print(Out, !IsUnsigned);
1908 std::optional<bool>
Default) {
1911 Out <<
FS <<
Name <<
": " << (
Value ?
"true" :
"false");
1918 Out <<
FS <<
Name <<
": ";
1923 FieldSeparator FlagsFS(
" | ");
1924 for (
auto F : SplitFlags) {
1926 assert(!StringF.empty() &&
"Expected valid flag");
1927 Out << FlagsFS << StringF;
1929 if (Extra || SplitFlags.empty())
1930 Out << FlagsFS << Extra;
1937 Out <<
FS <<
Name <<
": ";
1947 FieldSeparator FlagsFS(
" | ");
1948 for (
auto F : SplitFlags) {
1950 assert(!StringF.empty() &&
"Expected valid flag");
1951 Out << FlagsFS << StringF;
1953 if (Extra || SplitFlags.empty())
1954 Out << FlagsFS << Extra;
1969template <
class IntTy,
class Stringifier>
1971 Stringifier
toString,
bool ShouldSkipZero) {
1975 Out <<
FS <<
Name <<
": ";
1984 AsmWriterContext &WriterCtx) {
1985 Out <<
"!GenericDINode(";
1986 MDFieldPrinter
Printer(Out, WriterCtx);
1988 Printer.printString(
"header",
N->getHeader());
1989 if (
N->getNumDwarfOperands()) {
1990 Out <<
Printer.FS <<
"operands: {";
1992 for (
auto &
I :
N->dwarf_operands()) {
2002 AsmWriterContext &WriterCtx) {
2003 Out <<
"!DILocation(";
2004 MDFieldPrinter
Printer(Out, WriterCtx);
2006 Printer.printInt(
"line",
DL->getLine(),
false);
2007 Printer.printInt(
"column",
DL->getColumn());
2008 Printer.printMetadata(
"scope",
DL->getRawScope(),
false);
2009 Printer.printMetadata(
"inlinedAt",
DL->getRawInlinedAt());
2010 Printer.printBool(
"isImplicitCode",
DL->isImplicitCode(),
2016 AsmWriterContext &WriterCtx) {
2017 Out <<
"!DIAssignID()";
2018 MDFieldPrinter
Printer(Out, WriterCtx);
2022 AsmWriterContext &WriterCtx) {
2023 Out <<
"!DISubrange(";
2024 MDFieldPrinter
Printer(Out, WriterCtx);
2026 auto *Count =
N->getRawCountNode();
2027 if (
auto *CE = dyn_cast_or_null<ConstantAsMetadata>(Count)) {
2028 auto *CV = cast<ConstantInt>(CE->getValue());
2029 Printer.printInt(
"count", CV->getSExtValue(),
2032 Printer.printMetadata(
"count", Count,
true);
2036 auto *LBound =
N->getRawLowerBound();
2037 if (
auto *LE = dyn_cast_or_null<ConstantAsMetadata>(LBound)) {
2038 auto *LV = cast<ConstantInt>(LE->getValue());
2039 Printer.printInt(
"lowerBound", LV->getSExtValue(),
2042 Printer.printMetadata(
"lowerBound", LBound,
true);
2044 auto *UBound =
N->getRawUpperBound();
2045 if (
auto *UE = dyn_cast_or_null<ConstantAsMetadata>(UBound)) {
2046 auto *UV = cast<ConstantInt>(UE->getValue());
2047 Printer.printInt(
"upperBound", UV->getSExtValue(),
2050 Printer.printMetadata(
"upperBound", UBound,
true);
2052 auto *Stride =
N->getRawStride();
2053 if (
auto *SE = dyn_cast_or_null<ConstantAsMetadata>(Stride)) {
2054 auto *SV = cast<ConstantInt>(SE->getValue());
2055 Printer.printInt(
"stride", SV->getSExtValue(),
false);
2057 Printer.printMetadata(
"stride", Stride,
true);
2063 AsmWriterContext &WriterCtx) {
2064 Out <<
"!DIGenericSubrange(";
2065 MDFieldPrinter
Printer(Out, WriterCtx);
2067 auto IsConstant = [&](
Metadata *Bound) ->
bool {
2068 if (
auto *BE = dyn_cast_or_null<DIExpression>(Bound)) {
2069 return BE->isConstant() &&
2076 auto GetConstant = [&](
Metadata *Bound) -> int64_t {
2077 assert(IsConstant(Bound) &&
"Expected constant");
2078 auto *BE = dyn_cast_or_null<DIExpression>(Bound);
2079 return static_cast<int64_t
>(BE->getElement(1));
2082 auto *Count =
N->getRawCountNode();
2083 if (IsConstant(Count))
2084 Printer.printInt(
"count", GetConstant(Count),
2087 Printer.printMetadata(
"count", Count,
true);
2089 auto *LBound =
N->getRawLowerBound();
2090 if (IsConstant(LBound))
2091 Printer.printInt(
"lowerBound", GetConstant(LBound),
2094 Printer.printMetadata(
"lowerBound", LBound,
true);
2096 auto *UBound =
N->getRawUpperBound();
2097 if (IsConstant(UBound))
2098 Printer.printInt(
"upperBound", GetConstant(UBound),
2101 Printer.printMetadata(
"upperBound", UBound,
true);
2103 auto *Stride =
N->getRawStride();
2104 if (IsConstant(Stride))
2105 Printer.printInt(
"stride", GetConstant(Stride),
2108 Printer.printMetadata(
"stride", Stride,
true);
2114 AsmWriterContext &) {
2115 Out <<
"!DIEnumerator(";
2117 Printer.printString(
"name",
N->getName(),
false);
2118 Printer.printAPInt(
"value",
N->getValue(),
N->isUnsigned(),
2120 if (
N->isUnsigned())
2121 Printer.printBool(
"isUnsigned",
true);
2126 AsmWriterContext &) {
2127 Out <<
"!DIBasicType(";
2129 if (
N->getTag() != dwarf::DW_TAG_base_type)
2131 Printer.printString(
"name",
N->getName());
2132 Printer.printInt(
"size",
N->getSizeInBits());
2133 Printer.printInt(
"align",
N->getAlignInBits());
2134 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2136 Printer.printDIFlags(
"flags",
N->getFlags());
2141 AsmWriterContext &WriterCtx) {
2142 Out <<
"!DIStringType(";
2143 MDFieldPrinter
Printer(Out, WriterCtx);
2144 if (
N->getTag() != dwarf::DW_TAG_string_type)
2146 Printer.printString(
"name",
N->getName());
2147 Printer.printMetadata(
"stringLength",
N->getRawStringLength());
2148 Printer.printMetadata(
"stringLengthExpression",
N->getRawStringLengthExp());
2149 Printer.printMetadata(
"stringLocationExpression",
2150 N->getRawStringLocationExp());
2151 Printer.printInt(
"size",
N->getSizeInBits());
2152 Printer.printInt(
"align",
N->getAlignInBits());
2153 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2159 AsmWriterContext &WriterCtx) {
2160 Out <<
"!DIDerivedType(";
2161 MDFieldPrinter
Printer(Out, WriterCtx);
2163 Printer.printString(
"name",
N->getName());
2164 Printer.printMetadata(
"scope",
N->getRawScope());
2165 Printer.printMetadata(
"file",
N->getRawFile());
2166 Printer.printInt(
"line",
N->getLine());
2167 Printer.printMetadata(
"baseType",
N->getRawBaseType(),
2169 Printer.printInt(
"size",
N->getSizeInBits());
2170 Printer.printInt(
"align",
N->getAlignInBits());
2171 Printer.printInt(
"offset",
N->getOffsetInBits());
2172 Printer.printDIFlags(
"flags",
N->getFlags());
2173 Printer.printMetadata(
"extraData",
N->getRawExtraData());
2174 if (
const auto &DWARFAddressSpace =
N->getDWARFAddressSpace())
2175 Printer.printInt(
"dwarfAddressSpace", *DWARFAddressSpace,
2177 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2178 if (
auto PtrAuthData =
N->getPtrAuthData()) {
2179 Printer.printInt(
"ptrAuthKey", PtrAuthData->key());
2180 Printer.printBool(
"ptrAuthIsAddressDiscriminated",
2181 PtrAuthData->isAddressDiscriminated());
2182 Printer.printInt(
"ptrAuthExtraDiscriminator",
2183 PtrAuthData->extraDiscriminator());
2184 Printer.printBool(
"ptrAuthIsaPointer", PtrAuthData->isaPointer());
2185 Printer.printBool(
"ptrAuthAuthenticatesNullValues",
2186 PtrAuthData->authenticatesNullValues());
2192 AsmWriterContext &WriterCtx) {
2193 Out <<
"!DICompositeType(";
2194 MDFieldPrinter
Printer(Out, WriterCtx);
2196 Printer.printString(
"name",
N->getName());
2197 Printer.printMetadata(
"scope",
N->getRawScope());
2198 Printer.printMetadata(
"file",
N->getRawFile());
2199 Printer.printInt(
"line",
N->getLine());
2200 Printer.printMetadata(
"baseType",
N->getRawBaseType());
2201 Printer.printInt(
"size",
N->getSizeInBits());
2202 Printer.printInt(
"align",
N->getAlignInBits());
2203 Printer.printInt(
"offset",
N->getOffsetInBits());
2204 Printer.printDIFlags(
"flags",
N->getFlags());
2205 Printer.printMetadata(
"elements",
N->getRawElements());
2206 Printer.printDwarfEnum(
"runtimeLang",
N->getRuntimeLang(),
2208 Printer.printMetadata(
"vtableHolder",
N->getRawVTableHolder());
2209 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2210 Printer.printString(
"identifier",
N->getIdentifier());
2211 Printer.printMetadata(
"discriminator",
N->getRawDiscriminator());
2212 Printer.printMetadata(
"dataLocation",
N->getRawDataLocation());
2213 Printer.printMetadata(
"associated",
N->getRawAssociated());
2214 Printer.printMetadata(
"allocated",
N->getRawAllocated());
2215 if (
auto *RankConst =
N->getRankConst())
2216 Printer.printInt(
"rank", RankConst->getSExtValue(),
2219 Printer.printMetadata(
"rank",
N->getRawRank(),
true);
2220 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2225 AsmWriterContext &WriterCtx) {
2226 Out <<
"!DISubroutineType(";
2227 MDFieldPrinter
Printer(Out, WriterCtx);
2228 Printer.printDIFlags(
"flags",
N->getFlags());
2230 Printer.printMetadata(
"types",
N->getRawTypeArray(),
2238 Printer.printString(
"filename",
N->getFilename(),
2240 Printer.printString(
"directory",
N->getDirectory(),
2243 if (
N->getChecksum())
2244 Printer.printChecksum(*
N->getChecksum());
2251 AsmWriterContext &WriterCtx) {
2252 Out <<
"!DICompileUnit(";
2253 MDFieldPrinter
Printer(Out, WriterCtx);
2254 Printer.printDwarfEnum(
"language",
N->getSourceLanguage(),
2256 Printer.printMetadata(
"file",
N->getRawFile(),
false);
2257 Printer.printString(
"producer",
N->getProducer());
2258 Printer.printBool(
"isOptimized",
N->isOptimized());
2259 Printer.printString(
"flags",
N->getFlags());
2260 Printer.printInt(
"runtimeVersion",
N->getRuntimeVersion(),
2262 Printer.printString(
"splitDebugFilename",
N->getSplitDebugFilename());
2263 Printer.printEmissionKind(
"emissionKind",
N->getEmissionKind());
2264 Printer.printMetadata(
"enums",
N->getRawEnumTypes());
2265 Printer.printMetadata(
"retainedTypes",
N->getRawRetainedTypes());
2266 Printer.printMetadata(
"globals",
N->getRawGlobalVariables());
2267 Printer.printMetadata(
"imports",
N->getRawImportedEntities());
2268 Printer.printMetadata(
"macros",
N->getRawMacros());
2269 Printer.printInt(
"dwoId",
N->getDWOId());
2270 Printer.printBool(
"splitDebugInlining",
N->getSplitDebugInlining(),
true);
2271 Printer.printBool(
"debugInfoForProfiling",
N->getDebugInfoForProfiling(),
2273 Printer.printNameTableKind(
"nameTableKind",
N->getNameTableKind());
2274 Printer.printBool(
"rangesBaseAddress",
N->getRangesBaseAddress(),
false);
2275 Printer.printString(
"sysroot",
N->getSysRoot());
2276 Printer.printString(
"sdk",
N->getSDK());
2281 AsmWriterContext &WriterCtx) {
2282 Out <<
"!DISubprogram(";
2283 MDFieldPrinter
Printer(Out, WriterCtx);
2284 Printer.printString(
"name",
N->getName());
2285 Printer.printString(
"linkageName",
N->getLinkageName());
2286 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2287 Printer.printMetadata(
"file",
N->getRawFile());
2288 Printer.printInt(
"line",
N->getLine());
2289 Printer.printMetadata(
"type",
N->getRawType());
2290 Printer.printInt(
"scopeLine",
N->getScopeLine());
2291 Printer.printMetadata(
"containingType",
N->getRawContainingType());
2292 if (
N->getVirtuality() != dwarf::DW_VIRTUALITY_none ||
2293 N->getVirtualIndex() != 0)
2294 Printer.printInt(
"virtualIndex",
N->getVirtualIndex(),
false);
2295 Printer.printInt(
"thisAdjustment",
N->getThisAdjustment());
2296 Printer.printDIFlags(
"flags",
N->getFlags());
2297 Printer.printDISPFlags(
"spFlags",
N->getSPFlags());
2298 Printer.printMetadata(
"unit",
N->getRawUnit());
2299 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2300 Printer.printMetadata(
"declaration",
N->getRawDeclaration());
2301 Printer.printMetadata(
"retainedNodes",
N->getRawRetainedNodes());
2302 Printer.printMetadata(
"thrownTypes",
N->getRawThrownTypes());
2303 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2304 Printer.printString(
"targetFuncName",
N->getTargetFuncName());
2309 AsmWriterContext &WriterCtx) {
2310 Out <<
"!DILexicalBlock(";
2311 MDFieldPrinter
Printer(Out, WriterCtx);
2312 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2313 Printer.printMetadata(
"file",
N->getRawFile());
2314 Printer.printInt(
"line",
N->getLine());
2315 Printer.printInt(
"column",
N->getColumn());
2321 AsmWriterContext &WriterCtx) {
2322 Out <<
"!DILexicalBlockFile(";
2323 MDFieldPrinter
Printer(Out, WriterCtx);
2324 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2325 Printer.printMetadata(
"file",
N->getRawFile());
2326 Printer.printInt(
"discriminator",
N->getDiscriminator(),
2332 AsmWriterContext &WriterCtx) {
2333 Out <<
"!DINamespace(";
2334 MDFieldPrinter
Printer(Out, WriterCtx);
2335 Printer.printString(
"name",
N->getName());
2336 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2337 Printer.printBool(
"exportSymbols",
N->getExportSymbols(),
false);
2342 AsmWriterContext &WriterCtx) {
2343 Out <<
"!DICommonBlock(";
2344 MDFieldPrinter
Printer(Out, WriterCtx);
2345 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2346 Printer.printMetadata(
"declaration",
N->getRawDecl(),
false);
2347 Printer.printString(
"name",
N->getName());
2348 Printer.printMetadata(
"file",
N->getRawFile());
2349 Printer.printInt(
"line",
N->getLineNo());
2354 AsmWriterContext &WriterCtx) {
2356 MDFieldPrinter
Printer(Out, WriterCtx);
2358 Printer.printInt(
"line",
N->getLine());
2359 Printer.printString(
"name",
N->getName());
2360 Printer.printString(
"value",
N->getValue());
2365 AsmWriterContext &WriterCtx) {
2366 Out <<
"!DIMacroFile(";
2367 MDFieldPrinter
Printer(Out, WriterCtx);
2368 Printer.printInt(
"line",
N->getLine());
2369 Printer.printMetadata(
"file",
N->getRawFile(),
false);
2370 Printer.printMetadata(
"nodes",
N->getRawElements());
2375 AsmWriterContext &WriterCtx) {
2376 Out <<
"!DIModule(";
2377 MDFieldPrinter
Printer(Out, WriterCtx);
2378 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2379 Printer.printString(
"name",
N->getName());
2380 Printer.printString(
"configMacros",
N->getConfigurationMacros());
2381 Printer.printString(
"includePath",
N->getIncludePath());
2382 Printer.printString(
"apinotes",
N->getAPINotesFile());
2383 Printer.printMetadata(
"file",
N->getRawFile());
2384 Printer.printInt(
"line",
N->getLineNo());
2385 Printer.printBool(
"isDecl",
N->getIsDecl(),
false);
2391 AsmWriterContext &WriterCtx) {
2392 Out <<
"!DITemplateTypeParameter(";
2393 MDFieldPrinter
Printer(Out, WriterCtx);
2394 Printer.printString(
"name",
N->getName());
2395 Printer.printMetadata(
"type",
N->getRawType(),
false);
2396 Printer.printBool(
"defaulted",
N->isDefault(),
false);
2402 AsmWriterContext &WriterCtx) {
2403 Out <<
"!DITemplateValueParameter(";
2404 MDFieldPrinter
Printer(Out, WriterCtx);
2405 if (
N->getTag() != dwarf::DW_TAG_template_value_parameter)
2407 Printer.printString(
"name",
N->getName());
2408 Printer.printMetadata(
"type",
N->getRawType());
2409 Printer.printBool(
"defaulted",
N->isDefault(),
false);
2410 Printer.printMetadata(
"value",
N->getValue(),
false);
2415 AsmWriterContext &WriterCtx) {
2416 Out <<
"!DIGlobalVariable(";
2417 MDFieldPrinter
Printer(Out, WriterCtx);
2418 Printer.printString(
"name",
N->getName());
2419 Printer.printString(
"linkageName",
N->getLinkageName());
2420 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2421 Printer.printMetadata(
"file",
N->getRawFile());
2422 Printer.printInt(
"line",
N->getLine());
2423 Printer.printMetadata(
"type",
N->getRawType());
2424 Printer.printBool(
"isLocal",
N->isLocalToUnit());
2425 Printer.printBool(
"isDefinition",
N->isDefinition());
2426 Printer.printMetadata(
"declaration",
N->getRawStaticDataMemberDeclaration());
2427 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2428 Printer.printInt(
"align",
N->getAlignInBits());
2429 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2434 AsmWriterContext &WriterCtx) {
2435 Out <<
"!DILocalVariable(";
2436 MDFieldPrinter
Printer(Out, WriterCtx);
2437 Printer.printString(
"name",
N->getName());
2438 Printer.printInt(
"arg",
N->getArg());
2439 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2440 Printer.printMetadata(
"file",
N->getRawFile());
2441 Printer.printInt(
"line",
N->getLine());
2442 Printer.printMetadata(
"type",
N->getRawType());
2443 Printer.printDIFlags(
"flags",
N->getFlags());
2444 Printer.printInt(
"align",
N->getAlignInBits());
2445 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2450 AsmWriterContext &WriterCtx) {
2452 MDFieldPrinter
Printer(Out, WriterCtx);
2453 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2454 Printer.printString(
"name",
N->getName());
2455 Printer.printMetadata(
"file",
N->getRawFile());
2456 Printer.printInt(
"line",
N->getLine());
2461 AsmWriterContext &WriterCtx) {
2462 Out <<
"!DIExpression(";
2467 assert(!OpStr.empty() &&
"Expected valid opcode");
2471 Out << FS <<
Op.getArg(0);
2474 for (
unsigned A = 0, AE =
Op.getNumArgs();
A != AE; ++
A)
2475 Out << FS <<
Op.getArg(
A);
2479 for (
const auto &
I :
N->getElements())
2486 AsmWriterContext &WriterCtx,
2487 bool FromValue =
false) {
2489 "Unexpected DIArgList metadata outside of value argument");
2490 Out <<
"!DIArgList(";
2492 MDFieldPrinter
Printer(Out, WriterCtx);
2502 AsmWriterContext &WriterCtx) {
2503 Out <<
"!DIGlobalVariableExpression(";
2504 MDFieldPrinter
Printer(Out, WriterCtx);
2505 Printer.printMetadata(
"var",
N->getVariable());
2506 Printer.printMetadata(
"expr",
N->getExpression());
2511 AsmWriterContext &WriterCtx) {
2512 Out <<
"!DIObjCProperty(";
2513 MDFieldPrinter
Printer(Out, WriterCtx);
2514 Printer.printString(
"name",
N->getName());
2515 Printer.printMetadata(
"file",
N->getRawFile());
2516 Printer.printInt(
"line",
N->getLine());
2517 Printer.printString(
"setter",
N->getSetterName());
2518 Printer.printString(
"getter",
N->getGetterName());
2519 Printer.printInt(
"attributes",
N->getAttributes());
2520 Printer.printMetadata(
"type",
N->getRawType());
2525 AsmWriterContext &WriterCtx) {
2526 Out <<
"!DIImportedEntity(";
2527 MDFieldPrinter
Printer(Out, WriterCtx);
2529 Printer.printString(
"name",
N->getName());
2530 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2531 Printer.printMetadata(
"entity",
N->getRawEntity());
2532 Printer.printMetadata(
"file",
N->getRawFile());
2533 Printer.printInt(
"line",
N->getLine());
2534 Printer.printMetadata(
"elements",
N->getRawElements());
2539 AsmWriterContext &Ctx) {
2540 if (
Node->isDistinct())
2542 else if (
Node->isTemporary())
2543 Out <<
"<temporary!> ";
2545 switch (
Node->getMetadataID()) {
2548#define HANDLE_MDNODE_LEAF(CLASS) \
2549 case Metadata::CLASS##Kind: \
2550 write##CLASS(Out, cast<CLASS>(Node), Ctx); \
2552#include "llvm/IR/Metadata.def"
2559 AsmWriterContext &WriterCtx) {
2565 const Constant *CV = dyn_cast<Constant>(V);
2566 if (CV && !isa<GlobalValue>(CV)) {
2567 assert(WriterCtx.TypePrinter &&
"Constants require TypePrinting!");
2572 if (
const InlineAsm *IA = dyn_cast<InlineAsm>(V)) {
2574 if (IA->hasSideEffects())
2575 Out <<
"sideeffect ";
2576 if (IA->isAlignStack())
2577 Out <<
"alignstack ";
2580 Out <<
"inteldialect ";
2584 printEscapedString(IA->getAsmString(), Out);
2586 printEscapedString(IA->getConstraintString(), Out);
2591 if (
auto *MD = dyn_cast<MetadataAsValue>(V)) {
2599 auto *
Machine = WriterCtx.Machine;
2602 if (
const GlobalValue *GV = dyn_cast<GlobalValue>(V)) {
2603 Slot =
Machine->getGlobalSlot(GV);
2606 Slot =
Machine->getLocalSlot(V);
2613 Slot =
Machine->getLocalSlot(V);
2619 if (
const GlobalValue *GV = dyn_cast<GlobalValue>(V)) {
2620 Slot =
Machine->getGlobalSlot(GV);
2623 Slot =
Machine->getLocalSlot(V);
2632 Out << Prefix << Slot;
2638 AsmWriterContext &WriterCtx,
2642 if (
const DIExpression *Expr = dyn_cast<DIExpression>(MD)) {
2646 if (
const DIArgList *ArgList = dyn_cast<DIArgList>(MD)) {
2651 if (
const MDNode *
N = dyn_cast<MDNode>(MD)) {
2652 std::unique_ptr<SlotTracker> MachineStorage;
2654 if (!WriterCtx.Machine) {
2655 MachineStorage = std::make_unique<SlotTracker>(WriterCtx.Context);
2656 WriterCtx.Machine = MachineStorage.get();
2658 int Slot = WriterCtx.Machine->getMetadataSlot(
N);
2660 if (
const DILocation *Loc = dyn_cast<DILocation>(
N)) {
2666 Out <<
"<" <<
N <<
">";
2672 if (
const MDString *MDS = dyn_cast<MDString>(MD)) {
2674 printEscapedString(MDS->getString(), Out);
2679 auto *V = cast<ValueAsMetadata>(MD);
2680 assert(WriterCtx.TypePrinter &&
"TypePrinter required for metadata values");
2681 assert((FromValue || !isa<LocalAsMetadata>(V)) &&
2682 "Unexpected function-local metadata outside of value argument");
2684 WriterCtx.TypePrinter->print(V->getValue()->getType(), Out);
2691class AssemblyWriter {
2693 const Module *TheModule =
nullptr;
2695 std::unique_ptr<SlotTracker> SlotTrackerStorage;
2697 TypePrinting TypePrinter;
2701 bool ShouldPreserveUseListOrder;
2712 bool ShouldPreserveUseListOrder =
false);
2717 AsmWriterContext getContext() {
2718 return AsmWriterContext(&TypePrinter, &
Machine, TheModule);
2721 void printMDNodeBody(
const MDNode *MD);
2724 void printModule(
const Module *M);
2726 void writeOperand(
const Value *
Op,
bool PrintType);
2728 void writeOperandBundles(
const CallBase *Call);
2734 void writeAtomicCmpXchg(
const LLVMContext &Context,
2739 void writeAllMDNodes();
2740 void writeMDNode(
unsigned Slot,
const MDNode *
Node);
2741 void writeAttribute(
const Attribute &Attr,
bool InAttrGroup =
false);
2742 void writeAttributeSet(
const AttributeSet &AttrSet,
bool InAttrGroup =
false);
2743 void writeAllAttributeGroups();
2745 void printTypeIdentities();
2749 void printComdat(
const Comdat *
C);
2755 void printDbgMarker(
const DbgMarker &DPI);
2758 void printDbgRecord(
const DbgRecord &DR);
2759 void printDbgRecordLine(
const DbgRecord &DR);
2761 void printUseListOrder(
const Value *V,
const std::vector<unsigned> &Shuffle);
2764 void printModuleSummaryIndex();
2765 void printSummaryInfo(
unsigned Slot,
const ValueInfo &VI);
2773 void printArgs(
const std::vector<uint64_t> &Args);
2778 printNonConstVCalls(
const std::vector<FunctionSummary::VFuncId> &VCallList,
2781 printConstVCalls(
const std::vector<FunctionSummary::ConstVCall> &VCallList,
2786 void printMetadataAttachments(
2792 void printInfoComment(
const Value &V);
2803 bool IsForDebug,
bool ShouldPreserveUseListOrder)
2804 : Out(
o), TheModule(
M),
Machine(Mac), TypePrinter(
M), AnnotationWriter(AAW),
2805 IsForDebug(IsForDebug),
2806 ShouldPreserveUseListOrder(ShouldPreserveUseListOrder) {
2809 for (
const GlobalObject &GO : TheModule->global_objects())
2816 : Out(
o), TheIndex(
Index),
Machine(Mac), TypePrinter(nullptr),
2817 IsForDebug(IsForDebug), ShouldPreserveUseListOrder(
false) {}
2819void AssemblyWriter::writeOperand(
const Value *Operand,
bool PrintType) {
2821 Out <<
"<null operand!>";
2825 TypePrinter.print(Operand->
getType(), Out);
2828 auto WriterCtx = getContext();
2832void AssemblyWriter::writeSyncScope(
const LLVMContext &Context,
2842 Out <<
" syncscope(\"";
2843 printEscapedString(SSNs[SSID], Out);
2850void AssemblyWriter::writeAtomic(
const LLVMContext &Context,
2853 if (Ordering == AtomicOrdering::NotAtomic)
2856 writeSyncScope(Context, SSID);
2860void AssemblyWriter::writeAtomicCmpXchg(
const LLVMContext &Context,
2864 assert(SuccessOrdering != AtomicOrdering::NotAtomic &&
2865 FailureOrdering != AtomicOrdering::NotAtomic);
2867 writeSyncScope(Context, SSID);
2872void AssemblyWriter::writeParamOperand(
const Value *Operand,
2875 Out <<
"<null operand!>";
2880 TypePrinter.print(Operand->
getType(), Out);
2882 if (
Attrs.hasAttributes()) {
2884 writeAttributeSet(Attrs);
2888 auto WriterCtx = getContext();
2892void AssemblyWriter::writeOperandBundles(
const CallBase *Call) {
2893 if (!
Call->hasOperandBundles())
2898 bool FirstBundle =
true;
2899 for (
unsigned i = 0, e =
Call->getNumOperandBundles(); i != e; ++i) {
2904 FirstBundle =
false;
2912 bool FirstInput =
true;
2913 auto WriterCtx = getContext();
2914 for (
const auto &Input : BU.
Inputs) {
2919 if (Input ==
nullptr)
2920 Out <<
"<null operand bundle!>";
2922 TypePrinter.print(Input->getType(), Out);
2934void AssemblyWriter::printModule(
const Module *M) {
2937 if (ShouldPreserveUseListOrder)
2940 if (!
M->getModuleIdentifier().empty() &&
2943 M->getModuleIdentifier().find(
'\n') == std::string::npos)
2944 Out <<
"; ModuleID = '" <<
M->getModuleIdentifier() <<
"'\n";
2946 if (!
M->getSourceFileName().empty()) {
2947 Out <<
"source_filename = \"";
2948 printEscapedString(
M->getSourceFileName(), Out);
2952 const std::string &
DL =
M->getDataLayoutStr();
2954 Out <<
"target datalayout = \"" <<
DL <<
"\"\n";
2955 if (!
M->getTargetTriple().empty())
2956 Out <<
"target triple = \"" <<
M->getTargetTriple() <<
"\"\n";
2958 if (!
M->getModuleInlineAsm().empty()) {
2965 std::tie(Front, Asm) =
Asm.split(
'\n');
2969 Out <<
"module asm \"";
2970 printEscapedString(Front, Out);
2972 }
while (!
Asm.empty());
2975 printTypeIdentities();
2978 if (!Comdats.empty())
2980 for (
const Comdat *
C : Comdats) {
2982 if (
C != Comdats.back())
2987 if (!
M->global_empty()) Out <<
'\n';
2989 printGlobal(&GV); Out <<
'\n';
2993 if (!
M->alias_empty()) Out <<
"\n";
2998 if (!
M->ifunc_empty()) Out <<
"\n";
3009 printUseLists(
nullptr);
3014 writeAllAttributeGroups();
3018 if (!
M->named_metadata_empty()) Out <<
'\n';
3021 printNamedMDNode(&
Node);
3030void AssemblyWriter::printModuleSummaryIndex() {
3032 int NumSlots =
Machine.initializeIndexIfNeeded();
3038 std::vector<std::pair<std::string, ModuleHash>> moduleVec;
3039 std::string RegularLTOModuleName =
3041 moduleVec.resize(TheIndex->modulePaths().size());
3042 for (
auto &[ModPath, ModHash] : TheIndex->modulePaths())
3043 moduleVec[
Machine.getModulePathSlot(ModPath)] = std::make_pair(
3046 ModPath.empty() ? RegularLTOModuleName : std::string(ModPath), ModHash);
3049 for (
auto &ModPair : moduleVec) {
3050 Out <<
"^" << i++ <<
" = module: (";
3052 printEscapedString(ModPair.first, Out);
3053 Out <<
"\", hash: (";
3055 for (
auto Hash : ModPair.second)
3062 for (
auto &GlobalList : *TheIndex) {
3063 auto GUID = GlobalList.first;
3064 for (
auto &Summary : GlobalList.second.SummaryList)
3069 for (
auto &GlobalList : *TheIndex) {
3070 auto GUID = GlobalList.first;
3071 auto VI = TheIndex->getValueInfo(GlobalList);
3072 printSummaryInfo(
Machine.getGUIDSlot(GUID), VI);
3076 for (
const auto &TID : TheIndex->typeIds()) {
3077 Out <<
"^" <<
Machine.getTypeIdSlot(TID.second.first)
3078 <<
" = typeid: (name: \"" << TID.second.first <<
"\"";
3079 printTypeIdSummary(TID.second.second);
3080 Out <<
") ; guid = " << TID.first <<
"\n";
3084 for (
auto &TId : TheIndex->typeIdCompatibleVtableMap()) {
3086 Out <<
"^" <<
Machine.getTypeIdCompatibleVtableSlot(TId.first)
3087 <<
" = typeidCompatibleVTable: (name: \"" << TId.first <<
"\"";
3088 printTypeIdCompatibleVtableSummary(TId.second);
3089 Out <<
") ; guid = " <<
GUID <<
"\n";
3093 if (TheIndex->getFlags()) {
3094 Out <<
"^" << NumSlots <<
" = flags: " << TheIndex->getFlags() <<
"\n";
3098 Out <<
"^" << NumSlots <<
" = blockcount: " << TheIndex->getBlockCount()
3108 return "singleImpl";
3110 return "branchFunnel";
3121 return "uniformRetVal";
3123 return "uniqueRetVal";
3125 return "virtualConstProp";
3155 Out <<
", alignLog2: " << TTRes.
AlignLog2;
3157 Out <<
", sizeM1: " << TTRes.
SizeM1;
3167void AssemblyWriter::printTypeIdSummary(
const TypeIdSummary &TIS) {
3168 Out <<
", summary: (";
3169 printTypeTestResolution(TIS.
TTRes);
3170 if (!TIS.
WPDRes.empty()) {
3171 Out <<
", wpdResolutions: (";
3173 for (
auto &WPDRes : TIS.
WPDRes) {
3175 Out <<
"(offset: " << WPDRes.first <<
", ";
3176 printWPDRes(WPDRes.second);
3184void AssemblyWriter::printTypeIdCompatibleVtableSummary(
3186 Out <<
", summary: (";
3188 for (
auto &
P : TI) {
3190 Out <<
"(offset: " <<
P.AddressPointOffset <<
", ";
3191 Out <<
"^" <<
Machine.getGUIDSlot(
P.VTableVI.getGUID());
3197void AssemblyWriter::printArgs(
const std::vector<uint64_t> &Args) {
3200 for (
auto arg : Args) {
3208 Out <<
"wpdRes: (kind: ";
3215 Out <<
", resByArg: (";
3217 for (
auto &ResByArg : WPDRes.
ResByArg) {
3219 printArgs(ResByArg.first);
3220 Out <<
", byArg: (kind: ";
3222 if (ResByArg.second.TheKind ==
3224 ResByArg.second.TheKind ==
3226 Out <<
", info: " << ResByArg.second.Info;
3230 if (ResByArg.second.Byte || ResByArg.second.Bit)
3231 Out <<
", byte: " << ResByArg.second.Byte
3232 <<
", bit: " << ResByArg.second.Bit;
3253void AssemblyWriter::printAliasSummary(
const AliasSummary *AS) {
3254 Out <<
", aliasee: ";
3265 auto VTableFuncs =
GS->vTableFuncs();
3266 Out <<
", varFlags: (readonly: " <<
GS->VarFlags.MaybeReadOnly <<
", "
3267 <<
"writeonly: " <<
GS->VarFlags.MaybeWriteOnly <<
", "
3268 <<
"constant: " <<
GS->VarFlags.Constant;
3269 if (!VTableFuncs.empty())
3271 <<
"vcall_visibility: " <<
GS->VarFlags.VCallVisibility;
3274 if (!VTableFuncs.empty()) {
3275 Out <<
", vTableFuncs: (";
3277 for (
auto &
P : VTableFuncs) {
3279 Out <<
"(virtFunc: ^" <<
Machine.getGUIDSlot(
P.FuncVI.getGUID())
3280 <<
", offset: " <<
P.VTableOffset;
3298 return "linkonce_odr";
3308 return "extern_weak";
3310 return "available_externally";
3339 return "definition";
3341 return "declaration";
3347 Out <<
", insts: " <<
FS->instCount();
3348 if (
FS->fflags().anyFlagSet())
3349 Out <<
", " <<
FS->fflags();
3351 if (!
FS->calls().empty()) {
3352 Out <<
", calls: (";
3354 for (
auto &Call :
FS->calls()) {
3356 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(
Call.first.getGUID());
3357 if (
Call.second.getHotness() != CalleeInfo::HotnessType::Unknown)
3359 else if (
Call.second.RelBlockFreq)
3360 Out <<
", relbf: " <<
Call.second.RelBlockFreq;
3363 if (
Call.second.HasTailCall)
3370 if (
const auto *TIdInfo =
FS->getTypeIdInfo())
3371 printTypeIdInfo(*TIdInfo);
3375 auto AllocTypeName = [](uint8_t
Type) ->
const char * {
3377 case (uint8_t)AllocationType::None:
3379 case (uint8_t)AllocationType::NotCold:
3381 case (uint8_t)AllocationType::Cold:
3383 case (uint8_t)AllocationType::Hot:
3389 if (!
FS->allocs().empty()) {
3390 Out <<
", allocs: (";
3392 for (
auto &AI :
FS->allocs()) {
3394 Out <<
"(versions: (";
3396 for (
auto V : AI.Versions) {
3398 Out << AllocTypeName(V);
3400 Out <<
"), memProf: (";
3401 FieldSeparator MIBFS;
3402 for (
auto &MIB : AI.MIBs) {
3404 Out <<
"(type: " << AllocTypeName((uint8_t)MIB.AllocType);
3405 Out <<
", stackIds: (";
3406 FieldSeparator SIDFS;
3407 for (
auto Id : MIB.StackIdIndices) {
3409 Out << TheIndex->getStackIdAtIndex(Id);
3418 if (!
FS->callsites().empty()) {
3419 Out <<
", callsites: (";
3420 FieldSeparator SNFS;
3421 for (
auto &CI :
FS->callsites()) {
3424 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(CI.Callee.getGUID());
3426 Out <<
"(callee: null";
3427 Out <<
", clones: (";
3429 for (
auto V : CI.Clones) {
3433 Out <<
"), stackIds: (";
3434 FieldSeparator SIDFS;
3435 for (
auto Id : CI.StackIdIndices) {
3437 Out << TheIndex->getStackIdAtIndex(Id);
3448 if (!
FS->paramAccesses().empty()) {
3449 Out <<
", params: (";
3451 for (
auto &PS :
FS->paramAccesses()) {
3453 Out <<
"(param: " << PS.ParamNo;
3454 Out <<
", offset: ";
3456 if (!PS.Calls.empty()) {
3457 Out <<
", calls: (";
3459 for (
auto &Call : PS.Calls) {
3461 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(
Call.Callee.getGUID());
3462 Out <<
", param: " <<
Call.ParamNo;
3463 Out <<
", offset: ";
3464 PrintRange(
Call.Offsets);
3475void AssemblyWriter::printTypeIdInfo(
3477 Out <<
", typeIdInfo: (";
3478 FieldSeparator TIDFS;
3481 Out <<
"typeTests: (";
3484 auto TidIter = TheIndex->typeIds().equal_range(GUID);
3485 if (TidIter.first == TidIter.second) {
3491 for (
auto It = TidIter.first; It != TidIter.second; ++It) {
3493 auto Slot =
Machine.getTypeIdSlot(It->second.first);
3511 "typeTestAssumeConstVCalls");
3516 "typeCheckedLoadConstVCalls");
3522 auto TidIter = TheIndex->typeIds().equal_range(VFId.
GUID);
3523 if (TidIter.first == TidIter.second) {
3524 Out <<
"vFuncId: (";
3525 Out <<
"guid: " << VFId.
GUID;
3526 Out <<
", offset: " << VFId.
Offset;
3532 for (
auto It = TidIter.first; It != TidIter.second; ++It) {
3534 Out <<
"vFuncId: (";
3535 auto Slot =
Machine.getTypeIdSlot(It->second.first);
3538 Out <<
", offset: " << VFId.
Offset;
3543void AssemblyWriter::printNonConstVCalls(
3544 const std::vector<FunctionSummary::VFuncId> &VCallList,
const char *
Tag) {
3545 Out <<
Tag <<
": (";
3547 for (
auto &VFuncId : VCallList) {
3549 printVFuncId(VFuncId);
3554void AssemblyWriter::printConstVCalls(
3555 const std::vector<FunctionSummary::ConstVCall> &VCallList,
3557 Out <<
Tag <<
": (";
3559 for (
auto &ConstVCall : VCallList) {
3562 printVFuncId(ConstVCall.VFunc);
3563 if (!ConstVCall.Args.empty()) {
3565 printArgs(ConstVCall.Args);
3576 Out <<
"(module: ^" <<
Machine.getModulePathSlot(
Summary.modulePath())
3579 Out <<
", visibility: "
3582 Out <<
", live: " << GVFlags.
Live;
3583 Out <<
", dsoLocal: " << GVFlags.
DSOLocal;
3585 Out <<
", importType: "
3590 printAliasSummary(cast<AliasSummary>(&Summary));
3592 printFunctionSummary(cast<FunctionSummary>(&Summary));
3594 printGlobalVarSummary(cast<GlobalVarSummary>(&Summary));
3596 auto RefList =
Summary.refs();
3597 if (!RefList.empty()) {
3600 for (
auto &
Ref : RefList) {
3602 if (
Ref.isReadOnly())
3604 else if (
Ref.isWriteOnly())
3605 Out <<
"writeonly ";
3606 Out <<
"^" <<
Machine.getGUIDSlot(
Ref.getGUID());
3614void AssemblyWriter::printSummaryInfo(
unsigned Slot,
const ValueInfo &VI) {
3615 Out <<
"^" <<
Slot <<
" = gv: (";
3616 if (!
VI.name().empty())
3617 Out <<
"name: \"" <<
VI.name() <<
"\"";
3619 Out <<
"guid: " <<
VI.getGUID();
3620 if (!
VI.getSummaryList().empty()) {
3621 Out <<
", summaries: (";
3623 for (
auto &Summary :
VI.getSummaryList()) {
3625 printSummary(*Summary);
3630 if (!
VI.name().empty())
3631 Out <<
" ; guid = " <<
VI.getGUID();
3638 Out <<
"<empty name> ";
3640 unsigned char FirstC =
static_cast<unsigned char>(
Name[0]);
3641 if (isalpha(FirstC) || FirstC ==
'-' || FirstC ==
'$' || FirstC ==
'.' ||
3645 Out <<
'\\' << hexdigit(FirstC >> 4) << hexdigit(FirstC & 0x0F);
3646 for (
unsigned i = 1, e =
Name.size(); i != e; ++i) {
3647 unsigned char C =
Name[i];
3648 if (isalnum(
C) ||
C ==
'-' ||
C ==
'$' ||
C ==
'.' ||
C ==
'_')
3651 Out <<
'\\' << hexdigit(
C >> 4) << hexdigit(
C & 0x0F);
3656void AssemblyWriter::printNamedMDNode(
const NamedMDNode *NMD) {
3667 if (
auto *Expr = dyn_cast<DIExpression>(
Op)) {
3693 Out <<
"dso_local ";
3708 case GlobalVariable::NotThreadLocal:
3710 case GlobalVariable::GeneralDynamicTLSModel:
3711 Out <<
"thread_local ";
3713 case GlobalVariable::LocalDynamicTLSModel:
3714 Out <<
"thread_local(localdynamic) ";
3716 case GlobalVariable::InitialExecTLSModel:
3717 Out <<
"thread_local(initialexec) ";
3719 case GlobalVariable::LocalExecTLSModel:
3720 Out <<
"thread_local(localexec) ";
3727 case GlobalVariable::UnnamedAddr::None:
3729 case GlobalVariable::UnnamedAddr::Local:
3730 return "local_unnamed_addr";
3731 case GlobalVariable::UnnamedAddr::Global:
3732 return "unnamed_addr";
3743 if (isa<GlobalVariable>(GO))
3757 Out <<
"; Materializable\n";
3778 Out << (GV->
isConstant() ?
"constant " :
"global ");
3787 Out <<
", section \"";
3792 Out <<
", partition \"";
3797 Out <<
", code_model \"";
3822 Out <<
", no_sanitize_address";
3824 Out <<
", no_sanitize_hwaddress";
3826 Out <<
", sanitize_memtag";
3828 Out <<
", sanitize_address_dyninit";
3833 Out <<
", align " <<
A->value();
3837 printMetadataAttachments(MDs,
", ");
3840 if (
Attrs.hasAttributes())
3841 Out <<
" #" <<
Machine.getAttributeGroupSlot(Attrs);
3843 printInfoComment(*GV);
3846void AssemblyWriter::printAlias(
const GlobalAlias *GA) {
3848 Out <<
"; Materializable\n";
3869 writeOperand(Aliasee, !isa<ConstantExpr>(Aliasee));
3871 TypePrinter.print(GA->
getType(), Out);
3872 Out <<
" <<NULL ALIASEE>>";
3876 Out <<
", partition \"";
3881 printInfoComment(*GA);
3885void AssemblyWriter::printIFunc(
const GlobalIFunc *GI) {
3887 Out <<
"; Materializable\n";
3905 TypePrinter.print(GI->
getType(), Out);
3906 Out <<
" <<NULL RESOLVER>>";
3910 Out <<
", partition \"";
3915 printInfoComment(*GI);
3919void AssemblyWriter::printComdat(
const Comdat *
C) {
3923void AssemblyWriter::printTypeIdentities() {
3924 if (TypePrinter.empty())
3930 auto &NumberedTypes = TypePrinter.getNumberedTypes();
3931 for (
unsigned I = 0, E = NumberedTypes.size();
I != E; ++
I) {
3932 Out <<
'%' <<
I <<
" = type ";
3936 TypePrinter.printStructBody(NumberedTypes[
I], Out);
3940 auto &NamedTypes = TypePrinter.getNamedTypes();
3947 TypePrinter.printStructBody(NamedType, Out);
3953void AssemblyWriter::printFunction(
const Function *
F) {
3954 if (AnnotationWriter) AnnotationWriter->emitFunctionAnnot(
F, Out);
3956 if (
F->isMaterializable())
3957 Out <<
"; Materializable\n";
3960 if (
Attrs.hasFnAttrs()) {
3962 std::string AttrStr;
3965 if (!Attr.isStringAttribute()) {
3966 if (!AttrStr.empty()) AttrStr +=
' ';
3967 AttrStr += Attr.getAsString();
3971 if (!AttrStr.empty())
3972 Out <<
"; Function Attrs: " << AttrStr <<
'\n';
3977 if (
F->isDeclaration()) {
3980 F->getAllMetadata(MDs);
3981 printMetadataAttachments(MDs,
" ");
3998 if (
Attrs.hasRetAttrs())
4000 TypePrinter.print(
F->getReturnType(), Out);
4001 AsmWriterContext WriterCtx(&TypePrinter, &
Machine,
F->getParent());
4007 if (
F->isDeclaration() && !IsForDebug) {
4009 for (
unsigned I = 0, E = FT->getNumParams();
I != E; ++
I) {
4014 TypePrinter.print(FT->getParamType(
I), Out);
4019 writeAttributeSet(ArgAttrs);
4026 if (Arg.getArgNo() != 0)
4028 printArgument(&Arg,
Attrs.getParamAttrs(Arg.getArgNo()));
4033 if (FT->isVarArg()) {
4034 if (FT->getNumParams()) Out <<
", ";
4045 if (
F->getAddressSpace() != 0 || !
Mod ||
4046 Mod->getDataLayout().getProgramAddressSpace() != 0)
4047 Out <<
" addrspace(" <<
F->getAddressSpace() <<
")";
4048 if (
Attrs.hasFnAttrs())
4049 Out <<
" #" <<
Machine.getAttributeGroupSlot(
Attrs.getFnAttrs());
4050 if (
F->hasSection()) {
4051 Out <<
" section \"";
4052 printEscapedString(
F->getSection(), Out);
4055 if (
F->hasPartition()) {
4056 Out <<
" partition \"";
4057 printEscapedString(
F->getPartition(), Out);
4062 Out <<
" align " <<
A->value();
4064 Out <<
" gc \"" <<
F->getGC() <<
'"';
4065 if (
F->hasPrefixData()) {
4067 writeOperand(
F->getPrefixData(),
true);
4069 if (
F->hasPrologueData()) {
4070 Out <<
" prologue ";
4071 writeOperand(
F->getPrologueData(),
true);
4073 if (
F->hasPersonalityFn()) {
4074 Out <<
" personality ";
4075 writeOperand(
F->getPersonalityFn(),
true);
4078 if (
F->isDeclaration()) {
4082 F->getAllMetadata(MDs);
4083 printMetadataAttachments(MDs,
" ");
4088 printBasicBlock(&BB);
4103 TypePrinter.print(Arg->
getType(), Out);
4106 if (
Attrs.hasAttributes()) {
4108 writeAttributeSet(Attrs);
4117 assert(Slot != -1 &&
"expect argument in function here");
4118 Out <<
" %" <<
Slot;
4123void AssemblyWriter::printBasicBlock(
const BasicBlock *BB) {
4129 }
else if (!IsEntryBlock) {
4138 if (!IsEntryBlock) {
4140 Out.PadToColumn(50);
4145 Out <<
" No predecessors!";
4148 writeOperand(*PI,
false);
4149 for (++PI; PI != PE; ++PI) {
4151 writeOperand(*PI,
false);
4158 if (AnnotationWriter) AnnotationWriter->emitBasicBlockStartAnnot(BB, Out);
4162 for (
const DbgRecord &DR :
I.getDbgRecordRange())
4163 printDbgRecordLine(DR);
4164 printInstructionLine(
I);
4167 if (AnnotationWriter) AnnotationWriter->emitBasicBlockEndAnnot(BB, Out);
4171void AssemblyWriter::printInstructionLine(
const Instruction &
I) {
4172 printInstruction(
I);
4178void AssemblyWriter::printGCRelocateComment(
const GCRelocateInst &Relocate) {
4188void AssemblyWriter::printInfoComment(
const Value &V) {
4189 if (
const auto *Relocate = dyn_cast<GCRelocateInst>(&V))
4190 printGCRelocateComment(*Relocate);
4192 if (AnnotationWriter) {
4193 AnnotationWriter->printInfoComment(V, Out);
4200 if (Operand ==
nullptr) {
4201 Out <<
" <cannot get addrspace!>";
4205 bool PrintAddrSpace = CallAddrSpace != 0;
4206 if (!PrintAddrSpace) {
4211 if (!
Mod ||
Mod->getDataLayout().getProgramAddressSpace() != 0)
4212 PrintAddrSpace =
true;
4215 Out <<
" addrspace(" << CallAddrSpace <<
")";
4219void AssemblyWriter::printInstruction(
const Instruction &
I) {
4220 if (AnnotationWriter) AnnotationWriter->emitInstructionAnnot(&
I, Out);
4229 }
else if (!
I.getType()->isVoidTy()) {
4231 int SlotNum =
Machine.getLocalSlot(&
I);
4233 Out <<
"<badref> = ";
4235 Out <<
'%' << SlotNum <<
" = ";
4238 if (
const CallInst *CI = dyn_cast<CallInst>(&
I)) {
4239 if (CI->isMustTailCall())
4241 else if (CI->isTailCall())
4243 else if (CI->isNoTailCall())
4248 Out <<
I.getOpcodeName();
4251 if ((isa<LoadInst>(
I) && cast<LoadInst>(
I).isAtomic()) ||
4252 (isa<StoreInst>(
I) && cast<StoreInst>(
I).isAtomic()))
4255 if (isa<AtomicCmpXchgInst>(
I) && cast<AtomicCmpXchgInst>(
I).isWeak())
4259 if ((isa<LoadInst>(
I) && cast<LoadInst>(
I).isVolatile()) ||
4260 (isa<StoreInst>(
I) && cast<StoreInst>(
I).isVolatile()) ||
4261 (isa<AtomicCmpXchgInst>(
I) && cast<AtomicCmpXchgInst>(
I).isVolatile()) ||
4262 (isa<AtomicRMWInst>(
I) && cast<AtomicRMWInst>(
I).isVolatile()))
4269 if (
const CmpInst *CI = dyn_cast<CmpInst>(&
I))
4270 Out <<
' ' << CI->getPredicate();
4277 const Value *Operand =
I.getNumOperands() ?
I.getOperand(0) :
nullptr;
4280 if (isa<BranchInst>(
I) && cast<BranchInst>(
I).isConditional()) {
4283 writeOperand(BI.getCondition(),
true);
4285 writeOperand(BI.getSuccessor(0),
true);
4287 writeOperand(BI.getSuccessor(1),
true);
4289 }
else if (isa<SwitchInst>(
I)) {
4293 writeOperand(
SI.getCondition(),
true);
4295 writeOperand(
SI.getDefaultDest(),
true);
4297 for (
auto Case :
SI.cases()) {
4299 writeOperand(Case.getCaseValue(),
true);
4301 writeOperand(Case.getCaseSuccessor(),
true);
4304 }
else if (isa<IndirectBrInst>(
I)) {
4307 writeOperand(Operand,
true);
4310 for (
unsigned i = 1, e =
I.getNumOperands(); i != e; ++i) {
4313 writeOperand(
I.getOperand(i),
true);
4316 }
else if (
const PHINode *PN = dyn_cast<PHINode>(&
I)) {
4318 TypePrinter.print(
I.getType(), Out);
4321 for (
unsigned op = 0, Eop = PN->getNumIncomingValues();
op < Eop; ++
op) {
4322 if (
op) Out <<
", ";
4324 writeOperand(PN->getIncomingValue(
op),
false); Out <<
", ";
4325 writeOperand(PN->getIncomingBlock(
op),
false); Out <<
" ]";
4329 writeOperand(
I.getOperand(0),
true);
4330 for (
unsigned i : EVI->indices())
4334 writeOperand(
I.getOperand(0),
true); Out <<
", ";
4335 writeOperand(
I.getOperand(1),
true);
4336 for (
unsigned i : IVI->indices())
4338 }
else if (
const LandingPadInst *LPI = dyn_cast<LandingPadInst>(&
I)) {
4340 TypePrinter.print(
I.getType(), Out);
4341 if (LPI->isCleanup() || LPI->getNumClauses() != 0)
4344 if (LPI->isCleanup())
4347 for (
unsigned i = 0, e = LPI->getNumClauses(); i != e; ++i) {
4348 if (i != 0 || LPI->isCleanup()) Out <<
"\n";
4349 if (LPI->isCatch(i))
4354 writeOperand(LPI->getClause(i),
true);
4356 }
else if (
const auto *CatchSwitch = dyn_cast<CatchSwitchInst>(&
I)) {
4358 writeOperand(CatchSwitch->getParentPad(),
false);
4361 for (
const BasicBlock *PadBB : CatchSwitch->handlers()) {
4364 writeOperand(PadBB,
true);
4368 if (
const BasicBlock *UnwindDest = CatchSwitch->getUnwindDest())
4369 writeOperand(UnwindDest,
true);
4372 }
else if (
const auto *FPI = dyn_cast<FuncletPadInst>(&
I)) {
4374 writeOperand(FPI->getParentPad(),
false);
4376 for (
unsigned Op = 0, NumOps = FPI->arg_size();
Op < NumOps; ++
Op) {
4379 writeOperand(FPI->getArgOperand(
Op),
true);
4382 }
else if (isa<ReturnInst>(
I) && !Operand) {
4384 }
else if (
const auto *CRI = dyn_cast<CatchReturnInst>(&
I)) {
4386 writeOperand(CRI->getOperand(0),
false);
4389 writeOperand(CRI->getOperand(1),
true);
4390 }
else if (
const auto *CRI = dyn_cast<CleanupReturnInst>(&
I)) {
4392 writeOperand(CRI->getOperand(0),
false);
4395 if (CRI->hasUnwindDest())
4396 writeOperand(CRI->getOperand(1),
true);
4399 }
else if (
const CallInst *CI = dyn_cast<CallInst>(&
I)) {
4406 Operand = CI->getCalledOperand();
4421 TypePrinter.print(FTy->isVarArg() ? FTy :
RetTy, Out);
4423 writeOperand(Operand,
false);
4425 for (
unsigned op = 0, Eop = CI->arg_size();
op < Eop; ++
op) {
4433 if (CI->isMustTailCall() && CI->getParent() &&
4434 CI->getParent()->getParent() &&
4435 CI->getParent()->getParent()->isVarArg()) {
4436 if (CI->arg_size() > 0)
4445 writeOperandBundles(CI);
4446 }
else if (
const InvokeInst *
II = dyn_cast<InvokeInst>(&
I)) {
4447 Operand =
II->getCalledOperand();
4469 TypePrinter.print(FTy->isVarArg() ? FTy :
RetTy, Out);
4471 writeOperand(Operand,
false);
4473 for (
unsigned op = 0, Eop =
II->arg_size();
op < Eop; ++
op) {
4483 writeOperandBundles(
II);
4486 writeOperand(
II->getNormalDest(),
true);
4488 writeOperand(
II->getUnwindDest(),
true);
4489 }
else if (
const CallBrInst *CBI = dyn_cast<CallBrInst>(&
I)) {
4490 Operand = CBI->getCalledOperand();
4509 TypePrinter.print(FTy->isVarArg() ? FTy :
RetTy, Out);
4511 writeOperand(Operand,
false);
4513 for (
unsigned op = 0, Eop = CBI->arg_size();
op < Eop; ++
op) {
4523 writeOperandBundles(CBI);
4526 writeOperand(CBI->getDefaultDest(),
true);
4528 for (
unsigned i = 0, e = CBI->getNumIndirectDests(); i != e; ++i) {
4531 writeOperand(CBI->getIndirectDest(i),
true);
4534 }
else if (
const AllocaInst *AI = dyn_cast<AllocaInst>(&
I)) {
4536 if (AI->isUsedWithInAlloca())
4538 if (AI->isSwiftError())
4539 Out <<
"swifterror ";
4540 TypePrinter.print(AI->getAllocatedType(), Out);
4546 if (!AI->getArraySize() || AI->isArrayAllocation() ||
4547 !AI->getArraySize()->getType()->isIntegerTy(32)) {
4549 writeOperand(AI->getArraySize(),
true);
4552 Out <<
", align " <<
A->value();
4555 unsigned AddrSpace = AI->getAddressSpace();
4556 if (AddrSpace != 0) {
4557 Out <<
", addrspace(" << AddrSpace <<
')';
4559 }
else if (isa<CastInst>(
I)) {
4562 writeOperand(Operand,
true);
4565 TypePrinter.print(
I.getType(), Out);
4566 }
else if (isa<VAArgInst>(
I)) {
4569 writeOperand(Operand,
true);
4572 TypePrinter.print(
I.getType(), Out);
4573 }
else if (Operand) {
4574 if (
const auto *
GEP = dyn_cast<GetElementPtrInst>(&
I)) {
4576 TypePrinter.print(
GEP->getSourceElementType(), Out);
4578 }
else if (
const auto *LI = dyn_cast<LoadInst>(&
I)) {
4580 TypePrinter.print(LI->getType(), Out);
4587 bool PrintAllTypes =
false;
4592 if (isa<SelectInst>(
I) || isa<StoreInst>(
I) || isa<ShuffleVectorInst>(
I) ||
4593 isa<ReturnInst>(
I) || isa<AtomicCmpXchgInst>(
I) ||
4594 isa<AtomicRMWInst>(
I)) {
4595 PrintAllTypes =
true;
4597 for (
unsigned i = 1, E =
I.getNumOperands(); i != E; ++i) {
4598 Operand =
I.getOperand(i);
4601 if (Operand && Operand->
getType() != TheType) {
4602 PrintAllTypes =
true;
4608 if (!PrintAllTypes) {
4610 TypePrinter.print(TheType, Out);
4614 for (
unsigned i = 0, E =
I.getNumOperands(); i != E; ++i) {
4616 writeOperand(
I.getOperand(i), PrintAllTypes);
4621 if (
const LoadInst *LI = dyn_cast<LoadInst>(&
I)) {
4623 writeAtomic(LI->getContext(), LI->getOrdering(), LI->getSyncScopeID());
4625 Out <<
", align " <<
A->value();
4626 }
else if (
const StoreInst *SI = dyn_cast<StoreInst>(&
I)) {
4628 writeAtomic(
SI->getContext(),
SI->getOrdering(),
SI->getSyncScopeID());
4630 Out <<
", align " <<
A->value();
4632 writeAtomicCmpXchg(CXI->getContext(), CXI->getSuccessOrdering(),
4633 CXI->getFailureOrdering(), CXI->getSyncScopeID());
4634 Out <<
", align " << CXI->getAlign().value();
4635 }
else if (
const AtomicRMWInst *RMWI = dyn_cast<AtomicRMWInst>(&
I)) {
4636 writeAtomic(RMWI->getContext(), RMWI->getOrdering(),
4637 RMWI->getSyncScopeID());
4638 Out <<
", align " << RMWI->getAlign().value();
4639 }
else if (
const FenceInst *FI = dyn_cast<FenceInst>(&
I)) {
4640 writeAtomic(FI->getContext(), FI->getOrdering(), FI->getSyncScopeID());
4647 I.getAllMetadata(InstMD);
4648 printMetadataAttachments(InstMD,
", ");
4651 printInfoComment(
I);
4654void AssemblyWriter::printDbgMarker(
const DbgMarker &Marker) {
4658 printDbgRecord(DPR);
4662 Out <<
" DbgMarker -> { ";
4668void AssemblyWriter::printDbgRecord(
const DbgRecord &DR) {
4669 if (
auto *DVR = dyn_cast<DbgVariableRecord>(&DR))
4670 printDbgVariableRecord(*DVR);
4671 else if (
auto *DLR = dyn_cast<DbgLabelRecord>(&DR))
4672 printDbgLabelRecord(*DLR);
4678 auto WriterCtx = getContext();
4681 case DbgVariableRecord::LocationType::Value:
4684 case DbgVariableRecord::LocationType::Declare:
4687 case DbgVariableRecord::LocationType::Assign:
4692 "Tried to print a DbgVariableRecord with an invalid LocationType!");
4715void AssemblyWriter::printDbgRecordLine(
const DbgRecord &DR) {
4722void AssemblyWriter::printDbgLabelRecord(
const DbgLabelRecord &Label) {
4723 auto WriterCtx = getContext();
4724 Out <<
"#dbg_label(";
4731void AssemblyWriter::printMetadataAttachments(
4737 if (MDNames.empty())
4738 MDs[0].second->getContext().getMDKindNames(MDNames);
4740 auto WriterCtx = getContext();
4741 for (
const auto &
I : MDs) {
4742 unsigned Kind =
I.first;
4744 if (Kind < MDNames.size()) {
4748 Out <<
"!<unknown kind #" <<
Kind <<
">";
4754void AssemblyWriter::writeMDNode(
unsigned Slot,
const MDNode *
Node) {
4755 Out <<
'!' <<
Slot <<
" = ";
4756 printMDNodeBody(
Node);
4760void AssemblyWriter::writeAllMDNodes() {
4764 Nodes[
I.second] = cast<MDNode>(
I.first);
4766 for (
unsigned i = 0, e = Nodes.
size(); i != e; ++i) {
4767 writeMDNode(i, Nodes[i]);
4771void AssemblyWriter::printMDNodeBody(
const MDNode *
Node) {
4772 auto WriterCtx = getContext();
4776void AssemblyWriter::writeAttribute(
const Attribute &Attr,
bool InAttrGroup) {
4785 TypePrinter.print(Ty, Out);
4790void AssemblyWriter::writeAttributeSet(
const AttributeSet &AttrSet,
4792 bool FirstAttr =
true;
4793 for (
const auto &Attr : AttrSet) {
4796 writeAttribute(Attr, InAttrGroup);
4801void AssemblyWriter::writeAllAttributeGroups() {
4802 std::vector<std::pair<AttributeSet, unsigned>> asVec;
4803 asVec.resize(
Machine.as_size());
4806 asVec[
I.second] =
I;
4808 for (
const auto &
I : asVec)
4809 Out <<
"attributes #" <<
I.second <<
" = { "
4810 <<
I.first.getAsString(
true) <<
" }\n";
4813void AssemblyWriter::printUseListOrder(
const Value *V,
4814 const std::vector<unsigned> &Shuffle) {
4815 bool IsInFunction =
Machine.getFunction();
4819 Out <<
"uselistorder";
4820 if (
const BasicBlock *BB = IsInFunction ?
nullptr : dyn_cast<BasicBlock>(V)) {
4822 writeOperand(BB->getParent(),
false);
4824 writeOperand(BB,
false);
4827 writeOperand(V,
true);
4831 assert(Shuffle.size() >= 2 &&
"Shuffle too small");
4833 for (
unsigned I = 1, E = Shuffle.size();
I != E; ++
I)
4834 Out <<
", " << Shuffle[
I];
4838void AssemblyWriter::printUseLists(
const Function *
F) {
4839 auto It = UseListOrders.find(
F);
4840 if (It == UseListOrders.end())
4843 Out <<
"\n; uselistorder directives\n";
4844 for (
const auto &Pair : It->second)
4845 printUseListOrder(Pair.first, Pair.second);
4853 bool ShouldPreserveUseListOrder,
4854 bool IsForDebug)
const {
4857 AssemblyWriter W(
OS, SlotTable, this->
getParent(), AAW,
4859 ShouldPreserveUseListOrder);
4860 W.printFunction(
this);
4864 bool ShouldPreserveUseListOrder,
4865 bool IsForDebug)
const {
4868 AssemblyWriter W(
OS, SlotTable, this->
getModule(), AAW,
4870 ShouldPreserveUseListOrder);
4871 W.printBasicBlock(
this);
4875 bool ShouldPreserveUseListOrder,
bool IsForDebug)
const {
4878 AssemblyWriter W(
OS, SlotTable,
this, AAW, IsForDebug,
4879 ShouldPreserveUseListOrder);
4880 W.printModule(
this);
4886 AssemblyWriter W(
OS, SlotTable,
getParent(),
nullptr, IsForDebug);
4887 W.printNamedMDNode(
this);
4891 bool IsForDebug)
const {
4892 std::optional<SlotTracker> LocalST;
4898 SlotTable = &*LocalST;
4902 AssemblyWriter W(
OS, *SlotTable,
getParent(),
nullptr, IsForDebug);
4903 W.printNamedMDNode(
this);
4908 ROS <<
" = comdat ";
4915 ROS <<
"exactmatch";
4921 ROS <<
"nodeduplicate";
4933 TP.print(
const_cast<Type*
>(
this),
OS);
4939 if (
StructType *STy = dyn_cast<StructType>(
const_cast<Type*
>(
this)))
4942 TP.printStructBody(STy,
OS);
4947 if (
const auto *CI = dyn_cast<CallInst>(&
I))
4948 if (
Function *
F = CI->getCalledFunction())
4949 if (
F->isIntrinsic())
4950 for (
auto &
Op :
I.operands())
4951 if (
auto *V = dyn_cast_or_null<MetadataAsValue>(
Op))
4952 if (isa<MDNode>(V->getMetadata()))
4960 print(ROS, MST, IsForDebug);
4966 print(ROS, MST, IsForDebug);
4970 bool IsForDebug)
const {
4975 auto incorporateFunction = [&](
const Function *
F) {
4981 W.printDbgMarker(*
this);
4987 print(ROS, MST, IsForDebug);
4991 bool IsForDebug)
const {
4996 auto incorporateFunction = [&](
const Function *
F) {
5004 W.printDbgVariableRecord(*
this);
5008 bool IsForDebug)
const {
5013 auto incorporateFunction = [&](
const Function *
F) {
5020 W.printDbgLabelRecord(*
this);
5024 bool ShouldInitializeAllMetadata =
false;
5025 if (
auto *
I = dyn_cast<Instruction>(
this))
5027 else if (isa<Function>(
this) || isa<MetadataAsValue>(
this))
5028 ShouldInitializeAllMetadata =
true;
5031 print(ROS, MST, IsForDebug);
5035 bool IsForDebug)
const {
5040 auto incorporateFunction = [&](
const Function *
F) {
5045 if (
const Instruction *
I = dyn_cast<Instruction>(
this)) {
5046 incorporateFunction(
I->getParent() ?
I->getParent()->getParent() :
nullptr);
5048 W.printInstruction(*
I);
5049 }
else if (
const BasicBlock *BB = dyn_cast<BasicBlock>(
this)) {
5050 incorporateFunction(BB->getParent());
5052 W.printBasicBlock(BB);
5053 }
else if (
const GlobalValue *GV = dyn_cast<GlobalValue>(
this)) {
5054 AssemblyWriter W(
OS, SlotTable, GV->
getParent(),
nullptr, IsForDebug);
5057 else if (
const Function *
F = dyn_cast<Function>(GV))
5059 else if (
const GlobalAlias *
A = dyn_cast<GlobalAlias>(GV))
5061 else if (
const GlobalIFunc *
I = dyn_cast<GlobalIFunc>(GV))
5065 }
else if (
const MetadataAsValue *V = dyn_cast<MetadataAsValue>(
this)) {
5067 }
else if (
const Constant *
C = dyn_cast<Constant>(
this)) {
5068 TypePrinting TypePrinter;
5069 TypePrinter.print(
C->getType(),
OS);
5071 AsmWriterContext WriterCtx(&TypePrinter, MST.
getMachine());
5073 }
else if (isa<InlineAsm>(
this) || isa<Argument>(
this)) {
5085 if (V.hasName() || isa<GlobalValue>(V) ||
5086 (!isa<Constant>(V) && !isa<MetadataAsValue>(V))) {
5087 AsmWriterContext WriterCtx(
nullptr,
Machine, M);
5096 TypePrinting TypePrinter(MST.
getModule());
5098 TypePrinter.print(V.getType(), O);
5116 M, isa<MetadataAsValue>(
this));
5132 AsmWriterContext &WriterCtx) {
5136 auto *
N = dyn_cast<MDNode>(&MD);
5137 if (!
N || isa<DIExpression>(MD))
5145struct MDTreeAsmWriterContext :
public AsmWriterContext {
5148 using EntryTy = std::pair<unsigned, std::string>;
5158 : AsmWriterContext(TP,
ST,
M), Level(0
U), Visited({InitMD}), MainOS(
OS) {}
5160 void onWriteMetadataAsOperand(
const Metadata *MD)
override {
5161 if (!Visited.
insert(MD).second)
5170 unsigned InsertIdx = Buffer.
size() - 1;
5173 Buffer[InsertIdx].second = std::move(
SS.str());
5177 ~MDTreeAsmWriterContext() {
5178 for (
const auto &Entry : Buffer) {
5180 unsigned NumIndent =
Entry.first * 2U;
5189 bool OnlyAsOperand,
bool PrintAsTree =
false) {
5192 TypePrinting TypePrinter(M);
5194 std::unique_ptr<AsmWriterContext> WriterCtx;
5195 if (PrintAsTree && !OnlyAsOperand)
5196 WriterCtx = std::make_unique<MDTreeAsmWriterContext>(
5200 std::make_unique<AsmWriterContext>(&TypePrinter, MST.
getMachine(), M);
5204 auto *
N = dyn_cast<MDNode>(&MD);
5205 if (OnlyAsOperand || !
N || isa<DIExpression>(MD))
5229 const Module *M,
bool )
const {
5248 AssemblyWriter W(
OS, SlotTable,
this, IsForDebug);
5249 W.printModuleSummaryIndex();
5253 unsigned UB)
const {
5259 if (
I.second >= LB &&
I.second < UB)
5260 L.push_back(std::make_pair(
I.second,
I.first));
5263#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
This file declares a class to represent arbitrary precision floating point values and provide a varie...
This file implements a class to represent arbitrary precision integral constant values and operations...
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
static void print(raw_ostream &Out, object::Archive::Kind Kind, T Val)
static void writeDIMacro(raw_ostream &Out, const DIMacro *N, AsmWriterContext &WriterCtx)
static void writeMetadataAsOperand(raw_ostream &Out, const Metadata *MD, AsmWriterContext &WriterCtx)
static void writeDIGlobalVariableExpression(raw_ostream &Out, const DIGlobalVariableExpression *N, AsmWriterContext &WriterCtx)
MapVector< const Value *, unsigned > OrderMap
static void PrintCallingConv(unsigned cc, raw_ostream &Out)
static void writeDICompositeType(raw_ostream &Out, const DICompositeType *N, AsmWriterContext &WriterCtx)
static const char * getWholeProgDevirtResKindName(WholeProgramDevirtResolution::Kind K)
static void printMetadataImpl(raw_ostream &ROS, const Metadata &MD, ModuleSlotTracker &MST, const Module *M, bool OnlyAsOperand, bool PrintAsTree=false)
static void WriteOptimizationInfo(raw_ostream &Out, const User *U)
static void writeDIStringType(raw_ostream &Out, const DIStringType *N, AsmWriterContext &WriterCtx)
static std::string getLinkageNameWithSpace(GlobalValue::LinkageTypes LT)
static std::vector< unsigned > predictValueUseListOrder(const Value *V, unsigned ID, const OrderMap &OM)
static void writeDIGlobalVariable(raw_ostream &Out, const DIGlobalVariable *N, AsmWriterContext &WriterCtx)
static void orderValue(const Value *V, OrderMap &OM)
static void PrintThreadLocalModel(GlobalVariable::ThreadLocalMode TLM, formatted_raw_ostream &Out)
static void PrintLLVMName(raw_ostream &OS, StringRef Name, PrefixType Prefix)
Turn the specified name into an 'LLVM name', which is either prefixed with % (if the string only cont...
static StringRef getUnnamedAddrEncoding(GlobalVariable::UnnamedAddr UA)
static const char * getWholeProgDevirtResByArgKindName(WholeProgramDevirtResolution::ByArg::Kind K)
static void PrintShuffleMask(raw_ostream &Out, Type *Ty, ArrayRef< int > Mask)
static void writeDIModule(raw_ostream &Out, const DIModule *N, AsmWriterContext &WriterCtx)
static void writeDIFile(raw_ostream &Out, const DIFile *N, AsmWriterContext &)
static void writeDISubroutineType(raw_ostream &Out, const DISubroutineType *N, AsmWriterContext &WriterCtx)
static bool isReferencingMDNode(const Instruction &I)
static void writeDILabel(raw_ostream &Out, const DILabel *N, AsmWriterContext &WriterCtx)
static void WriteMDNodeBodyInternal(raw_ostream &Out, const MDNode *Node, AsmWriterContext &Ctx)
static void writeDIDerivedType(raw_ostream &Out, const DIDerivedType *N, AsmWriterContext &WriterCtx)
static void printMetadataIdentifier(StringRef Name, formatted_raw_ostream &Out)
static void writeDIImportedEntity(raw_ostream &Out, const DIImportedEntity *N, AsmWriterContext &WriterCtx)
static const Module * getModuleFromDPI(const DbgMarker *Marker)
static void printAsOperandImpl(const Value &V, raw_ostream &O, bool PrintType, ModuleSlotTracker &MST)
static void writeDIObjCProperty(raw_ostream &Out, const DIObjCProperty *N, AsmWriterContext &WriterCtx)
static void PrintDLLStorageClass(GlobalValue::DLLStorageClassTypes SCT, formatted_raw_ostream &Out)
static void writeDISubprogram(raw_ostream &Out, const DISubprogram *N, AsmWriterContext &WriterCtx)
static const char * getSummaryKindName(GlobalValueSummary::SummaryKind SK)
static OrderMap orderModule(const Module *M)
static const char * getVisibilityName(GlobalValue::VisibilityTypes Vis)
static void printMetadataImplRec(raw_ostream &ROS, const Metadata &MD, AsmWriterContext &WriterCtx)
Recursive version of printMetadataImpl.
static SlotTracker * createSlotTracker(const Value *V)
static void WriteAPFloatInternal(raw_ostream &Out, const APFloat &APF)
static void writeDILocation(raw_ostream &Out, const DILocation *DL, AsmWriterContext &WriterCtx)
static void writeDINamespace(raw_ostream &Out, const DINamespace *N, AsmWriterContext &WriterCtx)
static void writeDICommonBlock(raw_ostream &Out, const DICommonBlock *N, AsmWriterContext &WriterCtx)
static UseListOrderMap predictUseListOrder(const Module *M)
static void WriteConstantInternal(raw_ostream &Out, const Constant *CV, AsmWriterContext &WriterCtx)
static void WriteAsOperandInternal(raw_ostream &Out, const Value *V, AsmWriterContext &WriterCtx)
static std::string getLinkageName(GlobalValue::LinkageTypes LT)
static void writeDIBasicType(raw_ostream &Out, const DIBasicType *N, AsmWriterContext &)
static void writeGenericDINode(raw_ostream &Out, const GenericDINode *N, AsmWriterContext &WriterCtx)
static void writeDILocalVariable(raw_ostream &Out, const DILocalVariable *N, AsmWriterContext &WriterCtx)
static const char * getTTResKindName(TypeTestResolution::Kind K)
static void writeDITemplateTypeParameter(raw_ostream &Out, const DITemplateTypeParameter *N, AsmWriterContext &WriterCtx)
static const char * getImportTypeName(GlobalValueSummary::ImportKind IK)
static void writeDICompileUnit(raw_ostream &Out, const DICompileUnit *N, AsmWriterContext &WriterCtx)
static const Module * getModuleFromVal(const Value *V)
static void maybePrintCallAddrSpace(const Value *Operand, const Instruction *I, raw_ostream &Out)
static void writeDIGenericSubrange(raw_ostream &Out, const DIGenericSubrange *N, AsmWriterContext &WriterCtx)
static void writeDISubrange(raw_ostream &Out, const DISubrange *N, AsmWriterContext &WriterCtx)
static void PrintVisibility(GlobalValue::VisibilityTypes Vis, formatted_raw_ostream &Out)
static void writeDILexicalBlockFile(raw_ostream &Out, const DILexicalBlockFile *N, AsmWriterContext &WriterCtx)
static void writeDIEnumerator(raw_ostream &Out, const DIEnumerator *N, AsmWriterContext &)
static void writeMDTuple(raw_ostream &Out, const MDTuple *Node, AsmWriterContext &WriterCtx)
static void writeDIExpression(raw_ostream &Out, const DIExpression *N, AsmWriterContext &WriterCtx)
static void PrintDSOLocation(const GlobalValue &GV, formatted_raw_ostream &Out)
static void writeDIAssignID(raw_ostream &Out, const DIAssignID *DL, AsmWriterContext &WriterCtx)
static void writeDILexicalBlock(raw_ostream &Out, const DILexicalBlock *N, AsmWriterContext &WriterCtx)
static void maybePrintComdat(formatted_raw_ostream &Out, const GlobalObject &GO)
static bool printWithoutType(const Value &V, raw_ostream &O, SlotTracker *Machine, const Module *M)
Print without a type, skipping the TypePrinting object.
static void writeDIArgList(raw_ostream &Out, const DIArgList *N, AsmWriterContext &WriterCtx, bool FromValue=false)
static void writeDITemplateValueParameter(raw_ostream &Out, const DITemplateValueParameter *N, AsmWriterContext &WriterCtx)
static const Value * skipMetadataWrapper(const Value *V)
Look for a value that might be wrapped as metadata, e.g.
static void writeDIMacroFile(raw_ostream &Out, const DIMacroFile *N, AsmWriterContext &WriterCtx)
Atomic ordering constants.
This file contains the simple types necessary to represent the attributes associated with functions a...
static const Function * getParent(const Value *V)
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
COFF::MachineTypes Machine
#define LLVM_DUMP_METHOD
Mark debug helper function definitions like dump() that should not be stripped from debug builds.
This file contains the declarations for the subclasses of Constant, which represent the different fla...
dxil pretty DXIL Metadata Pretty Printer
Looks at all the uses of the given value Returns the Liveness deduced from the uses of this value Adds all uses that cause the result to be MaybeLive to MaybeLiveRetUses If the result is MaybeLiveUses might be modified but its content should be ignored(since it might not be complete). DeadArgumentEliminationPass
This file defines the DenseMap class.
This file contains constants used for implementing Dwarf debug support.
This file contains the declaration of the GlobalIFunc class, which represents a single indirect funct...
This file provides various utilities for inspecting and working with the control flow graph in LLVM I...
This file contains an interface for creating legacy passes to print out IR in various granularities.
This defines the Use class.
static bool InRange(int64_t Value, unsigned short Shift, int LBound, int HBound)
ModuleSummaryIndex.h This file contains the declarations the classes that hold the module index and s...
Module.h This file contains the declarations for the Module class.
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
uint64_t IntrinsicInst * II
if(auto Err=PB.parsePassPipeline(MPM, Passes)) return wrap(std MPM run * Mod
static bool isDigit(const char C)
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file provides utility classes that use RAII to save and restore values.
This file implements a set that has insertion order iteration characteristics.
This file defines the SmallPtrSet class.
This file defines the SmallString class.
This file defines the SmallVector class.
static APFloat getSNaN(const fltSemantics &Sem, bool Negative=false, const APInt *payload=nullptr)
Factory for SNaN values.
opStatus convert(const fltSemantics &ToSemantics, roundingMode RM, bool *losesInfo)
double convertToDouble() const
Converts this APFloat to host double value.
void toString(SmallVectorImpl< char > &Str, unsigned FormatPrecision=0, unsigned FormatMaxPadding=3, bool TruncateZero=true) const
const fltSemantics & getSemantics() const
APInt bitcastToAPInt() const
Class for arbitrary precision integers.
APInt getLoBits(unsigned numBits) const
Compute an APInt containing numBits lowbits from this APInt.
uint64_t getZExtValue() const
Get zero extended value.
APInt getHiBits(unsigned numBits) const
Compute an APInt containing numBits highbits from this APInt.
Abstract interface of slot tracker storage.
virtual ~AbstractSlotTrackerStorage()
Alias summary information.
const GlobalValueSummary & getAliasee() const
an instruction to allocate memory on the stack
This class represents an incoming formal argument to a Function.
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
virtual ~AssemblyAnnotationWriter()
An instruction that atomically checks whether a specified value is in a memory location,...
an instruction that atomically reads a memory location, combines it with another value,...
static StringRef getOperationName(BinOp Op)
AttributeSet getFnAttrs() const
The function attributes are returned.
std::string getAsString(unsigned Index, bool InAttrGrp=false) const
Return the attributes at the index as a string.
bool hasRetAttrs() const
Return true if attributes exist for the return value.
AttributeSet getAttributes(unsigned Index) const
The attributes for the specified index are returned.
bool hasFnAttrs() const
Return true the attributes exist for the function.
AttributeSet getParamAttrs(unsigned ArgNo) const
The attributes for the argument or parameter at the given index are returned.
bool hasAttributes() const
Return true if attributes exists in this set.
std::string getAsString(bool InAttrGrp=false) const
The Attribute is converted to a string of equivalent mnemonic.
Attribute::AttrKind getKindAsEnum() const
Return the attribute's kind as an enum (Attribute::AttrKind).
static StringRef getNameFromAttrKind(Attribute::AttrKind AttrKind)
bool isTypeAttribute() const
Return true if the attribute is a type attribute.
Type * getValueAsType() const
Return the attribute's value as a Type.
LLVM Basic Block Representation.
void print(raw_ostream &OS, AssemblyAnnotationWriter *AAW=nullptr, bool ShouldPreserveUseListOrder=false, bool IsForDebug=false) const
Print the basic block to an output stream with an optional AssemblyAnnotationWriter.
bool isEntryBlock() const
Return true if this is the entry block of the containing function.
const Function * getParent() const
Return the enclosing method, or null if none.
const Module * getModule() const
Return the module owning the function this basic block belongs to, or nullptr if the function does no...
The address of a basic block.
Conditional or Unconditional Branch instruction.
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
CallBr instruction, tracking function calls that may not return control but instead transfer it to a ...
This class represents a function call, abstracting a target machine's calling convention.
This class is the base class for the comparison instructions.
void print(raw_ostream &OS, bool IsForDebug=false) const
StringRef getName() const
@ Largest
The linker will choose the largest COMDAT.
@ SameSize
The data referenced by the COMDAT must be the same size.
@ Any
The linker may choose any COMDAT.
@ NoDeduplicate
No deduplication is performed.
@ ExactMatch
The data referenced by the COMDAT must be the same.
SelectionKind getSelectionKind() const
ConstantArray - Constant Array Declarations.
An array constant whose element type is a simple 1/2/4/8-byte integer or float/double,...
A constant value that is initialized with an expression using other constant values.
ConstantFP - Floating Point Values [float, double].
This is the shared class of boolean and integer constants.
A signed pointer, in the ptrauth sense.
This class represents a range of values.
APInt getSignedMin() const
Return the smallest signed value contained in the ConstantRange.
APInt getSignedMax() const
Return the largest signed value contained in the ConstantRange.
This is an important base class in LLVM.
Constant * getAggregateElement(unsigned Elt) const
For aggregates (struct/array/vector) return the constant that corresponds to the specified element if...
List of ValueAsMetadata, to be used as an argument to a dbg.value intrinsic.
Basic type, like 'int' or 'float'.
static const char * nameTableKindString(DebugNameTableKind PK)
static const char * emissionKindString(DebugEmissionKind EK)
A lightweight wrapper around an expression operand.
A pair of DIGlobalVariable and DIExpression.
An imported module (C++ using directive or similar).
Macro Info DWARF-like metadata node.
Represents a module in the programming language, for example, a Clang module, or a Fortran module.
Tagged DWARF-like metadata node.
static DIFlags splitFlags(DIFlags Flags, SmallVectorImpl< DIFlags > &SplitFlags)
Split up a flags bitfield.
static StringRef getFlagString(DIFlags Flag)
String type, Fortran CHARACTER(n)
static DISPFlags splitFlags(DISPFlags Flags, SmallVectorImpl< DISPFlags > &SplitFlags)
Split up a flags bitfield for easier printing.
static StringRef getFlagString(DISPFlags Flag)
DISPFlags
Debug info subprogram flags.
Type array for a subprogram.
This class represents an Operation in the Expression.
Records a position in IR for a source label (DILabel).
void print(raw_ostream &O, bool IsForDebug=false) const
Per-instruction record of debug-info.
Instruction * MarkedInstr
Link back to the Instruction that owns this marker.
void print(raw_ostream &O, bool IsForDebug=false) const
Implement operator<< on DbgMarker.
const BasicBlock * getParent() const
simple_ilist< DbgRecord > StoredDbgRecords
List of DbgRecords, the non-instruction equivalent of llvm.dbg.
Base class for non-instruction debug metadata records that have positions within IR.
void print(raw_ostream &O, bool IsForDebug=false) const
DebugLoc getDebugLoc() const
DbgMarker * Marker
Marker that this DbgRecord is linked into.
Record of a variable value-assignment, aka a non instruction representation of the dbg....
LocationType getType() const
void print(raw_ostream &O, bool IsForDebug=false) const
MDNode * getRawExpression() const
MDNode * getRawAddressExpression() const
Metadata * getRawAssignID() const
MDNode * getRawVariable() const
Metadata * getRawLocation() const
Returns the metadata operand for the first location description.
Metadata * getRawAddress() const
MDNode * getAsMDNode() const
Return this as a bar MDNode.
iterator find(const_arg_type_t< KeyT > Val)
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
Utility class for floating point operations which can have information about relaxed accuracy require...
An instruction for ordering other memory operations.
Function summary information to aid decisions and implementation of importing.
void print(raw_ostream &OS, AssemblyAnnotationWriter *AAW=nullptr, bool ShouldPreserveUseListOrder=false, bool IsForDebug=false) const
Print the function to an output stream with an optional AssemblyAnnotationWriter.
Represents calls to the gc.relocate intrinsic.
Value * getBasePtr() const
Value * getDerivedPtr() const
Generic tagged DWARF-like metadata node.
const Constant * getAliasee() const
const Constant * getResolver() const
StringRef getSection() const
Get the custom section of this global if it has one.
MaybeAlign getAlign() const
Returns the alignment of the given variable or function.
void getAllMetadata(SmallVectorImpl< std::pair< unsigned, MDNode * > > &MDs) const
Appends all metadata attached to this value to MDs, sorting by KindID.
const Comdat * getComdat() const
bool hasSection() const
Check if this global has a custom object file section.
Function and variable summary information to aid decisions and implementation of importing.
SummaryKind
Sububclass discriminator (for dyn_cast<> et al.)
bool hasPartition() const
const SanitizerMetadata & getSanitizerMetadata() const
bool hasExternalLinkage() const
VisibilityTypes getVisibility() const
bool isImplicitDSOLocal() const
LinkageTypes getLinkage() const
ThreadLocalMode getThreadLocalMode() const
DLLStorageClassTypes
Storage classes of global values for PE targets.
@ DLLExportStorageClass
Function to be accessible from DLL.
@ DLLImportStorageClass
Function to be imported from DLL.
bool hasSanitizerMetadata() const
GUID getGUID() const
Return a 64-bit global unique ID constructed from global value name (i.e.
StringRef getPartition() const
Module * getParent()
Get the module that this global value is contained inside of...
PointerType * getType() const
Global values are always pointers.
VisibilityTypes
An enumeration for the kinds of visibility of global values.
@ DefaultVisibility
The GV is visible.
@ HiddenVisibility
The GV is hidden.
@ ProtectedVisibility
The GV is protected.
bool isMaterializable() const
If this function's Module is being lazily streamed in functions from disk or some other source,...
UnnamedAddr getUnnamedAddr() const
LinkageTypes
An enumeration for the kinds of linkage for global values.
@ PrivateLinkage
Like Internal, but omit from symbol table.
@ CommonLinkage
Tentative definitions.
@ InternalLinkage
Rename collisions when linking (static functions).
@ LinkOnceAnyLinkage
Keep one copy of function when linking (inline)
@ WeakODRLinkage
Same, but only replaced by something equivalent.
@ ExternalLinkage
Externally visible function.
@ WeakAnyLinkage
Keep one copy of named function when linking (weak)
@ AppendingLinkage
Special purpose, only applies to global arrays.
@ AvailableExternallyLinkage
Available for inspection, not emission.
@ ExternalWeakLinkage
ExternalWeak linkage description.
@ LinkOnceODRLinkage
Same, but only replaced by something equivalent.
DLLStorageClassTypes getDLLStorageClass() const
Type * getValueType() const
Global variable summary information to aid decisions and implementation of importing.
const Constant * getInitializer() const
getInitializer - Return the initializer for this global variable.
bool isExternallyInitialized() const
bool hasInitializer() const
Definitions have initializers, declarations don't.
AttributeSet getAttributes() const
Return the attribute set for this global.
std::optional< CodeModel::Model > getCodeModel() const
Get the custom code model of this global if it has one.
bool isConstant() const
If the value is a global constant, its value is immutable throughout the runtime execution of the pro...
This instruction inserts a struct field of array element value into an aggregate value.
This is an important class for using LLVM in a threaded context.
void getSyncScopeNames(SmallVectorImpl< StringRef > &SSNs) const
getSyncScopeNames - Populates client supplied SmallVector with synchronization scope names registered...
The landingpad instruction holds all of the information necessary to generate correct exception handl...
An instruction for reading from memory.
void printTree(raw_ostream &OS, const Module *M=nullptr) const
Print in tree shape.
void dumpTree() const
User-friendly dump in tree shape.
This class implements a map that also provides access to all stored values in a deterministic order.
Manage lifetime of a slot tracker for printing IR.
std::vector< std::pair< unsigned, const MDNode * > > MachineMDNodeListType
const Module * getModule() const
ModuleSlotTracker(SlotTracker &Machine, const Module *M, const Function *F=nullptr)
Wrap a preinitialized SlotTracker.
virtual ~ModuleSlotTracker()
Destructor to clean up storage.
int getLocalSlot(const Value *V)
Return the slot number of the specified local value.
void collectMDNodes(MachineMDNodeListType &L, unsigned LB, unsigned UB) const
SlotTracker * getMachine()
Lazily creates a slot tracker.
void setProcessHook(std::function< void(AbstractSlotTrackerStorage *, const Module *, bool)>)
void incorporateFunction(const Function &F)
Incorporate the given function.
Class to hold module path string table and global value map, and encapsulate methods for operating on...
static constexpr const char * getRegularLTOModuleName()
const StringMap< ModuleHash > & modulePaths() const
Table of modules, containing module hash and id.
void dump() const
Dump to stderr (for debugging).
void print(raw_ostream &OS, bool IsForDebug=false) const
Print to an output stream.
A Module instance is used to store all the information related to an LLVM module.
iterator_range< ifunc_iterator > ifuncs()
iterator_range< named_metadata_iterator > named_metadata()
iterator_range< alias_iterator > aliases()
iterator_range< global_iterator > globals()
void print(raw_ostream &OS, AssemblyAnnotationWriter *AAW, bool ShouldPreserveUseListOrder=false, bool IsForDebug=false) const
Print the module to an output stream with an optional AssemblyAnnotationWriter.
void dump() const
Dump the module to stderr (for debugging).
StringRef getName() const
void print(raw_ostream &ROS, bool IsForDebug=false) const
MDNode * getOperand(unsigned i) const
unsigned getNumOperands() const
Module * getParent()
Get the module that holds this named metadata collection.
Utility class for integer operators which may exhibit overflow - Add, Sub, Mul, and Shl.
unsigned getAddressSpace() const
Return the address space of the Pointer type.
An or instruction, which can be marked as "disjoint", indicating that the inputs don't have a 1 in th...
A udiv or sdiv instruction, which can be marked as "exact", indicating that no bits are destroyed.
Interface for looking up the initializer for a variable name, used by Init::resolveReferences.
A vector that has set insertion semantics.
This instruction constructs a fixed permutation of two input vectors.
This class provides computation of slot numbers for LLVM Assembly writing.
int getMetadataSlot(const MDNode *N) override
getMetadataSlot - Get the slot number of a MDNode.
int getTypeIdCompatibleVtableSlot(StringRef Id)
int getModulePathSlot(StringRef Path)
unsigned mdn_size() const
SlotTracker(const SlotTracker &)=delete
void purgeFunction()
After calling incorporateFunction, use this method to remove the most recently incorporated function ...
int getTypeIdSlot(StringRef Id)
void initializeIfNeeded()
These functions do the actual initialization.
int getGlobalSlot(const GlobalValue *V)
getGlobalSlot - Get the slot number of a global value.
const Function * getFunction() const
unsigned getNextMetadataSlot() override
void incorporateFunction(const Function *F)
If you'd like to deal with a function instead of just a module, use this method to get its data into ...
int getLocalSlot(const Value *V)
Return the slot number of the specified value in it's type plane.
int getAttributeGroupSlot(AttributeSet AS)
SlotTracker(const Module *M, bool ShouldInitializeAllMetadata=false)
Construct from a module.
void createMetadataSlot(const MDNode *N) override
getMetadataSlot - Get the slot number of a MDNode.
void setProcessHook(std::function< void(AbstractSlotTrackerStorage *, const Module *, bool)>)
SlotTracker & operator=(const SlotTracker &)=delete
int getGUIDSlot(GlobalValue::GUID GUID)
int initializeIndexIfNeeded()
DenseMap< AttributeSet, unsigned >::iterator as_iterator
AttributeSet map iterators.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
An instruction for storing to memory.
StringMap - This is an unconventional map that is specialized for handling keys that are "strings",...
iterator find(StringRef Key)
StringRef - Represent a constant reference to a string, i.e.
constexpr bool empty() const
empty - Check if the string is empty.
Class to represent struct types.
ArrayRef< Type * > elements() const
unsigned getNumElements() const
Random access to the elements.
bool isLiteral() const
Return true if this type is uniqued by structural equivalence, false if it is a struct definition.
bool isOpaque() const
Return true if this is a type with an identity that has no body specified yet.
StringRef getName() const
Return the name for this struct type if it has an identity.
Class to represent target extensions types, which are generally unintrospectable from target-independ...
ArrayRef< Type * > type_params() const
Return the type parameters for this particular target extension type.
ArrayRef< unsigned > int_params() const
Return the integer parameters for this particular target extension type.
TypeFinder - Walk over a module, identifying all of the types that are used by the module.
The instances of the Type class are immutable: once they are created, they are never changed.
bool isVectorTy() const
True if this is an instance of VectorType.
unsigned getPointerAddressSpace() const
Get the address space of this pointer or pointer vector type.
@ X86_AMXTyID
AMX vectors (8192 bits, X86 specific)
@ TypedPointerTyID
Typed pointer used by some GPU targets.
@ HalfTyID
16-bit floating point type
@ TargetExtTyID
Target extension type.
@ VoidTyID
type with no size
@ ScalableVectorTyID
Scalable SIMD vector type.
@ FloatTyID
32-bit floating point type
@ IntegerTyID
Arbitrary bit width integers.
@ FixedVectorTyID
Fixed width SIMD vector type.
@ BFloatTyID
16-bit floating point type (7-bit significand)
@ DoubleTyID
64-bit floating point type
@ X86_FP80TyID
80-bit floating point type (X87)
@ PPC_FP128TyID
128-bit floating point type (two 64-bits, PowerPC)
@ FP128TyID
128-bit floating point type (112-bit significand)
void print(raw_ostream &O, bool IsForDebug=false, bool NoDetails=false) const
Print the current type.
StringRef getTargetExtName() const
bool isIntegerTy() const
True if this is an instance of IntegerType.
TypeID getTypeID() const
Return the type id for the type.
Type * getScalarType() const
If this is a vector type, return the element type, otherwise return 'this'.
A few GPU targets, such as DXIL and SPIR-V, have typed pointers.
Type * getElementType() const
unsigned getAddressSpace() const
Return the address space of the Pointer type.
A Use represents the edge between a Value definition and its users.
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
void print(raw_ostream &O, bool IsForDebug=false) const
Implement operator<< on Value.
void printAsOperand(raw_ostream &O, bool PrintType=true, const Module *M=nullptr) const
Print the name of this Value out to the specified raw_ostream.
StringRef getName() const
Return a constant reference to the value's name.
void dump() const
Support for debugging, callable in GDB: V->dump()
This class implements an extremely fast bulk output stream that can only output to a stream.
raw_ostream & indent(unsigned NumSpaces)
indent - Insert 'NumSpaces' spaces.
A raw_ostream that writes to an std::string.
StringRef LanguageString(unsigned Language)
StringRef AttributeEncodingString(unsigned Encoding)
StringRef ConventionString(unsigned Convention)
StringRef MacinfoString(unsigned Encoding)
StringRef OperationEncodingString(unsigned Encoding)
StringRef TagString(unsigned Tag)
This provides a very simple, boring adaptor for a begin and end iterator into a range type.
This file contains the declaration of the Comdat class, which represents a single COMDAT in LLVM.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
constexpr char Attrs[]
Key for Kernel::Metadata::mAttrs.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ AArch64_VectorCall
Used between AArch64 Advanced SIMD functions.
@ X86_64_SysV
The C convention as specified in the x86-64 supplement to the System V ABI, used on most non-Windows ...
@ RISCV_VectorCall
Calling convention used for RISC-V V-extension.
@ AMDGPU_CS
Used for Mesa/AMDPAL compute shaders.
@ AMDGPU_VS
Used for Mesa vertex shaders, or AMDPAL last shader stage before rasterization (vertex shader if tess...
@ AVR_SIGNAL
Used for AVR signal routines.
@ Swift
Calling convention for Swift.
@ AMDGPU_KERNEL
Used for AMDGPU code object kernels.
@ AArch64_SVE_VectorCall
Used between AArch64 SVE functions.
@ ARM_APCS
ARM Procedure Calling Standard (obsolete, but still used on some targets).
@ CFGuard_Check
Special calling convention on Windows for calling the Control Guard Check ICall funtion.
@ AVR_INTR
Used for AVR interrupt routines.
@ PreserveMost
Used for runtime calls that preserves most registers.
@ AnyReg
OBSOLETED - Used for stack based JavaScript calls.
@ AMDGPU_Gfx
Used for AMD graphics targets.
@ DUMMY_HHVM
Placeholders for HHVM calling conventions (deprecated, removed).
@ AMDGPU_CS_ChainPreserve
Used on AMDGPUs to give the middle-end more control over argument placement.
@ AMDGPU_HS
Used for Mesa/AMDPAL hull shaders (= tessellation control shaders).
@ ARM_AAPCS
ARM Architecture Procedure Calling Standard calling convention (aka EABI).
@ AMDGPU_GS
Used for Mesa/AMDPAL geometry shaders.
@ AArch64_SME_ABI_Support_Routines_PreserveMost_From_X2
Preserve X2-X15, X19-X29, SP, Z0-Z31, P0-P15.
@ CXX_FAST_TLS
Used for access functions.
@ X86_INTR
x86 hardware interrupt context.
@ AArch64_SME_ABI_Support_Routines_PreserveMost_From_X0
Preserve X0-X13, X19-X29, SP, Z0-Z31, P0-P15.
@ AMDGPU_CS_Chain
Used on AMDGPUs to give the middle-end more control over argument placement.
@ GHC
Used by the Glasgow Haskell Compiler (GHC).
@ AMDGPU_PS
Used for Mesa/AMDPAL pixel shaders.
@ Cold
Attempts to make code in the caller as efficient as possible under the assumption that the call is no...
@ AArch64_SME_ABI_Support_Routines_PreserveMost_From_X1
Preserve X1-X15, X19-X29, SP, Z0-Z31, P0-P15.
@ X86_ThisCall
Similar to X86_StdCall.
@ PTX_Device
Call to a PTX device function.
@ SPIR_KERNEL
Used for SPIR kernel functions.
@ PreserveAll
Used for runtime calls that preserves (almost) all registers.
@ X86_StdCall
stdcall is mostly used by the Win32 API.
@ SPIR_FUNC
Used for SPIR non-kernel device functions.
@ Fast
Attempts to make calls as fast as possible (e.g.
@ MSP430_INTR
Used for MSP430 interrupt routines.
@ X86_VectorCall
MSVC calling convention that passes vectors and vector aggregates in SSE registers.
@ Intel_OCL_BI
Used for Intel OpenCL built-ins.
@ PreserveNone
Used for runtime calls that preserves none general registers.
@ AMDGPU_ES
Used for AMDPAL shader stage before geometry shader if geometry is in use.
@ Tail
Attemps to make calls as fast as possible while guaranteeing that tail call optimization can always b...
@ Win64
The C convention as implemented on Windows/x86-64 and AArch64.
@ PTX_Kernel
Call to a PTX kernel. Passes all arguments in parameter space.
@ SwiftTail
This follows the Swift calling convention in how arguments are passed but guarantees tail calls will ...
@ GRAAL
Used by GraalVM. Two additional registers are reserved.
@ AMDGPU_LS
Used for AMDPAL vertex shader if tessellation is in use.
@ ARM_AAPCS_VFP
Same as ARM_AAPCS, but uses hard floating point ABI.
@ X86_RegCall
Register calling convention used for parameters transfer optimization.
@ M68k_RTD
Used for M68k rtd-based CC (similar to X86's stdcall).
@ C
The default llvm calling convention, compatible with C.
@ X86_FastCall
'fast' analog of X86_StdCall.
@ System
Synchronized with respect to all concurrently executing threads.
std::optional< const char * > toString(const std::optional< DWARFFormValue > &V)
Take an optional DWARFFormValue and try to extract a string value from it.
@ DW_OP_LLVM_convert
Only used in LLVM metadata.
This is an optimization pass for GlobalISel generic memory operations.
pred_iterator pred_end(BasicBlock *BB)
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
const char * getHotnessName(CalleeInfo::HotnessType HT)
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
const char * toIRString(AtomicOrdering ao)
String used by LLVM IR to represent atomic ordering.
pred_iterator pred_begin(BasicBlock *BB)
void sort(IteratorTy Start, IteratorTy End)
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool is_sorted(R &&Range, Compare C)
Wrapper function around std::is_sorted to check if elements in a range R are sorted with respect to a...
FormattedNumber format_hex(uint64_t N, unsigned Width, bool Upper=false)
format_hex - Output N as a fixed width hexadecimal.
FormattedNumber format_hex_no_prefix(uint64_t N, unsigned Width, bool Upper=false)
format_hex_no_prefix - Output N as a fixed width hexadecimal.
constexpr int PoisonMaskElem
AtomicOrdering
Atomic ordering for LLVM's memory model.
@ Ref
The access may reference the value stored in memory.
DWARFExpression::Operation Op
raw_ostream & operator<<(raw_ostream &OS, const APFixedPoint &FX)
@ Default
The result values are uniform if and only if all operands are uniform.
std::vector< TypeIdOffsetVtableInfo > TypeIdCompatibleVtableInfo
List of vtable definitions decorated by a particular type identifier, and their corresponding offsets...
void printLLVMNameWithoutPrefix(raw_ostream &OS, StringRef Name)
Print out a name of an LLVM value without any prefixes.
static const fltSemantics & IEEEsingle() LLVM_READNONE
static constexpr roundingMode rmNearestTiesToEven
static const fltSemantics & PPCDoubleDouble() LLVM_READNONE
static const fltSemantics & x87DoubleExtended() LLVM_READNONE
static const fltSemantics & IEEEquad() LLVM_READNONE
static const fltSemantics & IEEEdouble() LLVM_READNONE
static const fltSemantics & IEEEhalf() LLVM_READNONE
static const fltSemantics & BFloat() LLVM_READNONE
A single checksum, represented by a Kind and a Value (a string).
T Value
The string value of the checksum.
StringRef getKindAsString() const
All type identifier related information.
std::vector< ConstVCall > TypeCheckedLoadConstVCalls
std::vector< VFuncId > TypeCheckedLoadVCalls
std::vector< ConstVCall > TypeTestAssumeConstVCalls
List of virtual calls made by this function using (respectively) llvm.assume(llvm....
std::vector< GlobalValue::GUID > TypeTests
List of type identifiers used by this function in llvm.type.test intrinsics referenced by something o...
std::vector< VFuncId > TypeTestAssumeVCalls
List of virtual calls made by this function using (respectively) llvm.assume(llvm....
An "identifier" for a virtual function.
Group flags (Linkage, NotEligibleToImport, etc.) as a bitfield.
unsigned DSOLocal
Indicates that the linker resolved the symbol to a definition from within the same linkage unit.
unsigned CanAutoHide
In the per-module summary, indicates that the global value is linkonce_odr and global unnamed addr (s...
unsigned ImportType
This field is written by the ThinLTO indexing step to postlink combined summary.
unsigned NotEligibleToImport
Indicate if the global value cannot be imported (e.g.
unsigned Linkage
The linkage type of the associated global value.
unsigned Visibility
Indicates the visibility.
unsigned Live
In per-module summary, indicate that the global value must be considered a live root for index-based ...
This struct is a compact representation of a valid (power of two) or undefined (0) alignment.
A lightweight accessor for an operand bundle meant to be passed around by value.
StringRef getTagName() const
Return the tag of this operand bundle as a string.
A utility class that uses RAII to save and restore the value of a variable.
std::map< uint64_t, WholeProgramDevirtResolution > WPDRes
Mapping from byte offset to whole-program devirt resolution for that (typeid, byte offset) pair.
Kind
Specifies which kind of type check we should emit for this byte array.
@ Unknown
Unknown (analysis not performed, don't lower)
@ Single
Single element (last example in "Short Inline Bit Vectors")
@ Inline
Inlined bit vector ("Short Inline Bit Vectors")
@ Unsat
Unsatisfiable type (i.e. no global has this type metadata)
@ AllOnes
All-ones bit vector ("Eliminating Bit Vector Checks for All-Ones Bit Vectors")
@ ByteArray
Test a byte array (first example)
unsigned SizeM1BitWidth
Range of size-1 expressed as a bit width.
enum llvm::TypeTestResolution::Kind TheKind
Struct that holds a reference to a particular GUID in a global value summary.
@ UniformRetVal
Uniform return value optimization.
@ VirtualConstProp
Virtual constant propagation.
@ UniqueRetVal
Unique return value optimization.
@ Indir
Just do a regular virtual call.
enum llvm::WholeProgramDevirtResolution::Kind TheKind
std::map< std::vector< uint64_t >, ByArg > ResByArg
Resolutions for calls with all constant integer arguments (excluding the first argument,...
std::string SingleImplName
@ SingleImpl
Single implementation devirtualization.
@ Indir
Just do a regular virtual call.
@ BranchFunnel
When retpoline mitigation is enabled, use a branch funnel that is defined in the merged module.
Function object to check whether the second component of a container supported by std::get (like std:...