31#include "llvm/Config/llvm-config.h"
96 cl::desc(
"Print addresses of instructions when dumping"));
100 cl::desc(
"Pretty print debug locations of instructions when dumping"));
104 cl::desc(
"Pretty print perf data (branch weights, etc) when dumping"));
108 cl::desc(
"Preserve use-list order when writing LLVM assembly."));
112 cl::desc(
"Print address space names"));
131 return VAM->getValue();
144 for (
const Value *
Op :
C->operands())
151 unsigned ID = OM.size() + 1;
158 auto OrderConstantValue = [&OM](
const Value *V) {
163 auto OrderConstantFromMetadata = [&](
Metadata *MD) {
165 OrderConstantValue(VAM->getValue());
167 for (
const auto *VAM : AL->getArgs())
168 OrderConstantValue(VAM->getValue());
173 if (
G.hasInitializer())
189 for (
const Use &U :
F.operands())
195 if (
F.isDeclaration())
208 OrderConstantFromMetadata(DVR.getRawLocation());
209 if (DVR.isDbgAssign())
210 OrderConstantFromMetadata(DVR.getRawAddress());
213 for (
const Value *
Op :
I.operands()) {
226static std::vector<unsigned>
229 using Entry = std::pair<const Use *, unsigned>;
233 if (OM.lookup(U.getUser()))
234 List.
push_back(std::make_pair(&U, List.size()));
245 ID = OM.lookup(BA->getBasicBlock());
246 llvm::sort(List, [&](
const Entry &L,
const Entry &R) {
247 const Use *LU = L.first;
248 const Use *RU = R.first;
252 auto LID = OM.lookup(LU->getUser());
253 auto RID = OM.lookup(RU->getUser());
273 return LU->getOperandNo() < RU->getOperandNo();
274 return LU->getOperandNo() > RU->getOperandNo();
282 std::vector<unsigned> Shuffle(List.size());
283 for (
size_t I = 0,
E = List.size();
I !=
E; ++
I)
284 Shuffle[
I] = List[
I].second;
291 for (
const auto &Pair : OM) {
292 const Value *V = Pair.first;
293 if (V->use_empty() || std::next(V->use_begin()) == V->use_end())
296 std::vector<unsigned> Shuffle =
303 F =
I->getFunction();
308 ULOM[
F][V] = std::move(Shuffle);
315 return MA->getParent() ? MA->getParent()->getParent() :
nullptr;
318 return BB->getParent() ? BB->getParent()->getParent() :
nullptr;
321 const Function *M =
I->getParent() ?
I->getParent()->getParent() :
nullptr;
322 return M ? M->getParent() :
nullptr;
326 return GV->getParent();
351 default: Out <<
"cc" << cc;
break;
374 Out <<
"aarch64_sve_vector_pcs";
377 Out <<
"aarch64_sme_preservemost_from_x0";
380 Out <<
"aarch64_sme_preservemost_from_x1";
383 Out <<
"aarch64_sme_preservemost_from_x2";
411 Out <<
"amdgpu_cs_chain";
414 Out <<
"amdgpu_cs_chain_preserve";
419 Out <<
"amdgpu_gfx_whole_wave";
423 Out <<
"riscv_vector_cc";
425#define CC_VLS_CASE(ABI_VLEN) \
426 case CallingConv::RISCV_VLSCall_##ABI_VLEN: \
427 Out << "riscv_vls_cc(" #ABI_VLEN ")"; \
443 Out <<
"cheriot_compartmentcallcc";
446 Out <<
"cheriot_compartmentcalleecc";
449 Out <<
"cheriot_librarycallcc";
463 assert(!Name.empty() &&
"Cannot get empty name!");
466 bool NeedsQuotes = isdigit(
static_cast<unsigned char>(Name[0]));
468 for (
unsigned char C : Name) {
473 if (!isalnum(
C) &&
C !=
'-' &&
C !=
'.' &&
C !=
'_') {
527 Out << Mask.size() <<
" x i32> ";
529 Out <<
"zeroinitializer";
535 for (
int Elt : Mask) {
550 TypePrinting(
const Module *M =
nullptr)
551 : M(M), TypesIncorporated(M == nullptr) {}
553 TypePrinting(
const TypePrinting &) =
delete;
554 TypePrinting &operator=(
const TypePrinting &) =
delete;
557 TypeFinder &getNamedTypes();
560 std::vector<StructType *> &getNumberedTypes();
566 void printStructBody(StructType *Ty, raw_ostream &OS);
569 void incorporateTypes();
573 bool TypesIncorporated;
575 TypeFinder NamedTypes;
578 DenseMap<StructType *, unsigned> Type2Number;
580 std::vector<StructType *> NumberedTypes;
590std::vector<StructType *> &TypePrinting::getNumberedTypes() {
596 if (NumberedTypes.size() == Type2Number.size())
597 return NumberedTypes;
599 NumberedTypes.resize(Type2Number.size());
600 for (
const auto &
P : Type2Number) {
601 assert(
P.second < NumberedTypes.size() &&
"Didn't get a dense numbering?");
602 assert(!NumberedTypes[
P.second] &&
"Didn't get a unique numbering?");
603 NumberedTypes[
P.second] =
P.first;
605 return NumberedTypes;
608bool TypePrinting::empty() {
610 return NamedTypes.
empty() && Type2Number.empty();
613void TypePrinting::incorporateTypes() {
614 if (TypesIncorporated)
617 NamedTypes.
run(*M,
false);
618 TypesIncorporated =
true;
622 unsigned NextNumber = 0;
624 std::vector<StructType *>::iterator NextToUse = NamedTypes.
begin();
625 for (StructType *STy : NamedTypes) {
627 if (STy->isLiteral())
630 if (STy->getName().empty())
631 Type2Number[STy] = NextNumber++;
636 NamedTypes.erase(NextToUse, NamedTypes.end());
641 bool ForcePrint =
false) {
642 if (AS == 0 && !ForcePrint)
644 OS << Prefix <<
"addrspace(";
648 OS <<
"\"" << ASName <<
"\"";
656void TypePrinting::print(
Type *Ty, raw_ostream &OS) {
658 case Type::VoidTyID: OS <<
"void";
return;
659 case Type::HalfTyID: OS <<
"half";
return;
660 case Type::BFloatTyID: OS <<
"bfloat";
return;
661 case Type::FloatTyID: OS <<
"float";
return;
662 case Type::DoubleTyID: OS <<
"double";
return;
663 case Type::X86_FP80TyID: OS <<
"x86_fp80";
return;
664 case Type::FP128TyID: OS <<
"fp128";
return;
665 case Type::PPC_FP128TyID: OS <<
"ppc_fp128";
return;
666 case Type::LabelTyID: OS <<
"label";
return;
667 case Type::MetadataTyID:
670 case Type::X86_AMXTyID: OS <<
"x86_amx";
return;
671 case Type::TokenTyID: OS <<
"token";
return;
675 case Type::IntegerTyID:
676 OS << 'i' << cast<IntegerType>(Ty)->getBitWidth();
679 case Type::FunctionTyID: {
681 print(FTy->getReturnType(), OS);
684 for (
Type *Ty : FTy->params()) {
693 case Type::StructTyID: {
697 return printStructBody(STy, OS);
703 const auto I = Type2Number.find(STy);
704 if (
I != Type2Number.end())
705 OS <<
'%' <<
I->second;
707 OS <<
"%\"type " << STy <<
'\"';
710 case Type::PointerTyID: {
716 case Type::ArrayTyID: {
718 OS <<
'[' << ATy->getNumElements() <<
" x ";
719 print(ATy->getElementType(), OS);
723 case Type::FixedVectorTyID:
724 case Type::ScalableVectorTyID: {
726 ElementCount
EC = PTy->getElementCount();
730 OS <<
EC.getKnownMinValue() <<
" x ";
731 print(PTy->getElementType(), OS);
735 case Type::TypedPointerTyID: {
741 case Type::TargetExtTyID:
748 Inner->print(OS,
false,
true);
751 OS <<
", " << IntParam;
758void TypePrinting::printStructBody(StructType *STy, raw_ostream &OS) {
800 const Function* TheFunction =
nullptr;
801 bool FunctionProcessed =
false;
802 bool ShouldInitializeAllMetadata;
807 ProcessFunctionHookFn;
822 unsigned mdnNext = 0;
830 unsigned ModulePathNext = 0;
834 unsigned GUIDNext = 0;
838 unsigned TypeIdNext = 0;
843 unsigned TypeIdCompatibleVtableNext = 0;
852 bool ShouldInitializeAllMetadata =
false);
860 bool ShouldInitializeAllMetadata =
false);
877 void createMetadataSlot(
const MDNode *
N)
override;
881 int getLocalSlot(
const Value *V);
883 int getMetadataSlot(
const MDNode *
N)
override;
888 int getTypeIdCompatibleVtableSlot(
StringRef Id);
894 FunctionProcessed =
false;
902 void purgeFunction();
909 unsigned mdn_size()
const {
return mdnMap.size(); }
917 unsigned as_size()
const {
return asMap.size(); }
933 void CreateMetadataSlot(
const MDNode *
N);
936 void CreateFunctionSlot(
const Value *V);
941 inline void CreateModulePathSlot(
StringRef Path);
944 void CreateTypeIdCompatibleVtableSlot(
StringRef Id);
956 void processGlobalObjectMetadata(
const GlobalObject &GO);
959 void processFunctionMetadata(
const Function &
F);
965 void processDbgRecordMetadata(
const DbgRecord &DVR);
970 : M(M), F(F), Machine(&Machine) {}
973 bool ShouldInitializeAllMetadata)
974 : ShouldCreateStorage(M),
975 ShouldInitializeAllMetadata(ShouldInitializeAllMetadata), M(M) {}
980 if (!ShouldCreateStorage)
983 ShouldCreateStorage =
false;
985 std::make_unique<SlotTracker>(M, ShouldInitializeAllMetadata);
986 Machine = MachineStorage.get();
987 if (ProcessModuleHookFn)
988 Machine->setProcessHook(ProcessModuleHookFn);
989 if (ProcessFunctionHookFn)
990 Machine->setProcessHook(ProcessFunctionHookFn);
1003 Machine->purgeFunction();
1004 Machine->incorporateFunction(&F);
1009 assert(F &&
"No function incorporated");
1010 return Machine->getLocalSlot(V);
1016 ProcessModuleHookFn = std::move(Fn);
1022 ProcessFunctionHookFn = std::move(Fn);
1052#define ST_DEBUG(X) dbgs() << X
1060 : TheModule(M), ShouldInitializeAllMetadata(ShouldInitializeAllMetadata) {}
1065 : TheModule(
F ?
F->
getParent() : nullptr), TheFunction(
F),
1066 ShouldInitializeAllMetadata(ShouldInitializeAllMetadata) {}
1069 : TheModule(nullptr), ShouldInitializeAllMetadata(
false), TheIndex(Index) {}
1074 TheModule =
nullptr;
1077 if (TheFunction && !FunctionProcessed)
1084 int NumSlots = processIndex();
1091void SlotTracker::processModule() {
1092 ST_DEBUG(
"begin processModule!\n");
1097 CreateModuleSlot(&Var);
1098 processGlobalObjectMetadata(Var);
1099 auto Attrs = Var.getAttributes();
1100 if (Attrs.hasAttributes())
1101 CreateAttributeSetSlot(Attrs);
1106 CreateModuleSlot(&
A);
1109 for (
const GlobalIFunc &
I : TheModule->ifuncs()) {
1111 CreateModuleSlot(&
I);
1112 processGlobalObjectMetadata(
I);
1116 for (
const NamedMDNode &NMD : TheModule->named_metadata()) {
1117 for (
const MDNode *
N : NMD.operands())
1118 CreateMetadataSlot(
N);
1121 for (
const Function &
F : *TheModule) {
1124 CreateModuleSlot(&
F);
1126 if (ShouldInitializeAllMetadata)
1127 processFunctionMetadata(
F);
1131 AttributeSet FnAttrs =
F.getAttributes().getFnAttrs();
1133 CreateAttributeSetSlot(FnAttrs);
1136 if (ProcessModuleHookFn)
1137 ProcessModuleHookFn(
this, TheModule, ShouldInitializeAllMetadata);
1143void SlotTracker::processFunction() {
1144 ST_DEBUG(
"begin processFunction!\n");
1148 if (!ShouldInitializeAllMetadata)
1149 processFunctionMetadata(*TheFunction);
1153 AE = TheFunction->arg_end(); AI != AE; ++AI)
1155 CreateFunctionSlot(&*AI);
1157 ST_DEBUG(
"Inserting Instructions:\n");
1160 for (
auto &BB : *TheFunction) {
1162 CreateFunctionSlot(&BB);
1164 for (
auto &
I : BB) {
1165 if (!
I.getType()->isVoidTy() && !
I.hasName())
1166 CreateFunctionSlot(&
I);
1173 if (
Attrs.hasAttributes())
1174 CreateAttributeSetSlot(Attrs);
1179 if (ProcessFunctionHookFn)
1180 ProcessFunctionHookFn(
this, TheFunction, ShouldInitializeAllMetadata);
1182 FunctionProcessed =
true;
1184 ST_DEBUG(
"end processFunction!\n");
1188int SlotTracker::processIndex() {
1195 std::vector<StringRef> ModulePaths;
1196 for (
auto &[ModPath,
_] : TheIndex->modulePaths())
1197 ModulePaths.push_back(ModPath);
1199 for (
auto &ModPath : ModulePaths)
1200 CreateModulePathSlot(ModPath);
1203 GUIDNext = ModulePathNext;
1206 for (
const auto &GlobalList : TheIndex->sortedGlobalValueSummariesRange())
1207 CreateGUIDSlot(GlobalList.first);
1210 TypeIdCompatibleVtableNext = GUIDNext;
1211 for (
auto &TId : TheIndex->typeIdCompatibleVtableMap())
1212 CreateTypeIdCompatibleVtableSlot(TId.first);
1215 TypeIdNext = TypeIdCompatibleVtableNext;
1216 for (
const auto &TID : TheIndex->typeIds())
1217 CreateTypeIdSlot(TID.second.first);
1223void SlotTracker::processGlobalObjectMetadata(
const GlobalObject &GO) {
1226 for (
auto &MD : MDs)
1227 CreateMetadataSlot(MD.second);
1230void SlotTracker::processFunctionMetadata(
const Function &
F) {
1231 processGlobalObjectMetadata(
F);
1232 for (
auto &BB :
F) {
1233 for (
auto &
I : BB) {
1234 for (
const DbgRecord &DR :
I.getDbgRecordRange())
1235 processDbgRecordMetadata(DR);
1236 processInstructionMetadata(
I);
1241void SlotTracker::processDbgRecordMetadata(
const DbgRecord &DR) {
1252 CreateMetadataSlot(
Empty);
1253 if (DVR->getRawVariable())
1254 CreateMetadataSlot(DVR->getRawVariable());
1255 if (DVR->isDbgAssign()) {
1256 if (
auto *AssignID = DVR->getRawAssignID())
1259 CreateMetadataSlot(
Empty);
1262 CreateMetadataSlot(DLR->getRawLabel());
1270void SlotTracker::processInstructionMetadata(
const Instruction &
I) {
1273 if (Function *
F = CI->getCalledFunction())
1274 if (
F->isIntrinsic())
1275 for (
auto &
Op :
I.operands())
1278 CreateMetadataSlot(
N);
1282 I.getAllMetadata(MDs);
1283 for (
auto &MD : MDs)
1284 CreateMetadataSlot(MD.second);
1291 ST_DEBUG(
"begin purgeFunction!\n");
1293 TheFunction =
nullptr;
1294 FunctionProcessed =
false;
1305 return MI == mMap.end() ? -1 : (int)
MI->second;
1311 ProcessModuleHookFn = std::move(Fn);
1317 ProcessFunctionHookFn = std::move(Fn);
1330 return MI == mdnMap.end() ? -1 : (int)
MI->second;
1341 return FI == fMap.end() ? -1 : (int)FI->second;
1350 return AI == asMap.end() ? -1 : (int)AI->second;
1358 auto I = ModulePathMap.find(Path);
1359 return I == ModulePathMap.end() ? -1 : (int)
I->second;
1368 return I == GUIDMap.end() ? -1 : (int)
I->second;
1376 auto I = TypeIdMap.find(Id);
1377 return I == TypeIdMap.end() ? -1 : (int)
I->second;
1385 auto I = TypeIdCompatibleVtableMap.find(Id);
1386 return I == TypeIdCompatibleVtableMap.end() ? -1 : (int)
I->second;
1390void SlotTracker::CreateModuleSlot(
const GlobalValue *V) {
1391 assert(V &&
"Can't insert a null Value into SlotTracker!");
1392 assert(!V->getType()->isVoidTy() &&
"Doesn't need a slot!");
1393 assert(!V->hasName() &&
"Doesn't need a slot!");
1395 unsigned DestSlot = mNext++;
1398 ST_DEBUG(
" Inserting value [" << V->getType() <<
"] = " << V <<
" slot=" <<
1408void SlotTracker::CreateFunctionSlot(
const Value *V) {
1409 assert(!V->getType()->isVoidTy() && !V->hasName() &&
"Doesn't need a slot!");
1411 unsigned DestSlot = fNext++;
1415 ST_DEBUG(
" Inserting value [" << V->getType() <<
"] = " << V <<
" slot=" <<
1416 DestSlot <<
" [o]\n");
1420void SlotTracker::CreateMetadataSlot(
const MDNode *
N) {
1421 assert(
N &&
"Can't insert a null Value into SlotTracker!");
1427 unsigned DestSlot = mdnNext;
1428 if (!mdnMap.insert(std::make_pair(
N, DestSlot)).second)
1433 for (
unsigned i = 0, e =
N->getNumOperands(); i != e; ++i)
1435 CreateMetadataSlot(
Op);
1438void SlotTracker::CreateAttributeSetSlot(
AttributeSet AS) {
1441 if (asMap.try_emplace(AS, asNext).second)
1446void SlotTracker::CreateModulePathSlot(
StringRef Path) {
1447 ModulePathMap[
Path] = ModulePathNext++;
1452 GUIDMap[
GUID] = GUIDNext++;
1456void SlotTracker::CreateTypeIdSlot(
StringRef Id) {
1457 TypeIdMap[
Id] = TypeIdNext++;
1461void SlotTracker::CreateTypeIdCompatibleVtableSlot(
StringRef Id) {
1462 TypeIdCompatibleVtableMap[
Id] = TypeIdCompatibleVtableNext++;
1467struct AsmWriterContext {
1468 TypePrinting *TypePrinter =
nullptr;
1469 SlotTracker *
Machine =
nullptr;
1472 AsmWriterContext(TypePrinting *TP, SlotTracker *ST,
const Module *M =
nullptr)
1475 static AsmWriterContext &getEmpty() {
1476 static AsmWriterContext EmptyCtx(
nullptr,
nullptr);
1482 virtual void onWriteMetadataAsOperand(
const Metadata *) {}
1484 virtual ~AsmWriterContext() =
default;
1493 AsmWriterContext &WriterCtx,
1494 bool PrintType =
false);
1497 AsmWriterContext &WriterCtx,
1498 bool FromValue =
false);
1502 Out << FPO->getFastMathFlags();
1505 if (OBO->hasNoUnsignedWrap())
1507 if (OBO->hasNoSignedWrap())
1513 if (PDI->isDisjoint())
1516 if (
GEP->isInBounds())
1518 else if (
GEP->hasNoUnsignedSignedWrap())
1520 if (
GEP->hasNoUnsignedWrap())
1523 Out <<
" inrange(" <<
InRange->getLower() <<
", " <<
InRange->getUpper()
1527 if (NNI->hasNonNeg())
1530 if (TI->hasNoUnsignedWrap())
1532 if (TI->hasNoSignedWrap())
1535 if (ICmp->hasSameSign())
1543 unsigned NumDigits = std::max((Val.
getBitWidth() + 3) / 4, 1U);
1545 for (
unsigned i = 0; i < NumDigits - Bits.size(); i++)
1551 bool ForceBitwiseOutput =
false;
1557 ForceBitwiseOutput = !HiWord.
isZero();
1560 if (!ForceBitwiseOutput) {
1563 Out << (APF.
isNegative() ?
'-' :
'+') <<
"inf";
1605 AsmWriterContext &WriterCtx) {
1607 Type *Ty = CI->getType();
1609 if (Ty->isVectorTy()) {
1611 WriterCtx.TypePrinter->print(Ty->getScalarType(), Out);
1615 if (Ty->getScalarType()->isIntegerTy(1))
1616 Out << (CI->getZExtValue() ?
"true" :
"false");
1618 Out << CI->getValue();
1620 if (Ty->isVectorTy())
1627 Type *Ty = CB->getType();
1629 if (Ty->isVectorTy()) {
1631 WriterCtx.TypePrinter->print(Ty->getScalarType(), Out);
1635 Out << CB->getValue();
1637 if (Ty->isVectorTy())
1644 Type *Ty = CFP->getType();
1646 if (Ty->isVectorTy()) {
1647 if (CFP->getValue().bitcastToAPInt().isZero()) {
1648 Out <<
"zeroinitializer";
1653 WriterCtx.TypePrinter->print(Ty->getScalarType(), Out);
1659 if (Ty->isVectorTy())
1666 Out <<
"zeroinitializer";
1671 Out <<
"blockaddress(";
1680 Out <<
"dso_local_equivalent ";
1695 unsigned NumOpsToWrite = 2;
1696 if (!CPA->getOperand(2)->isNullValue())
1704 for (
unsigned i = 0, e = NumOpsToWrite; i != e; ++i) {
1716 for (
const Value *
Op : CA->operands()) {
1727 if (CA->isString()) {
1736 for (
uint64_t i = 0, e = CA->getNumElements(); i != e; ++i) {
1746 if (CS->getType()->isPacked())
1749 if (CS->getNumOperands() != 0) {
1752 for (
const Value *
Op : CS->operands()) {
1759 if (CS->getType()->isPacked())
1784 for (
unsigned i = 0, e = CVVTy->getNumElements(); i != e; ++i) {
1828 if (CE->getOpcode() == Instruction::ShuffleVector) {
1829 if (
auto *SplatVal = CE->getSplatValue()) {
1840 Out << CE->getOpcodeName();
1845 WriterCtx.TypePrinter->print(
GEP->getSourceElementType(), Out);
1850 for (
const Value *
Op : CE->operands()) {
1857 WriterCtx.TypePrinter->print(CE->getType(), Out);
1860 if (CE->getOpcode() == Instruction::ShuffleVector)
1867 Out <<
"<placeholder or erroneous Constant>";
1871 AsmWriterContext &WriterCtx) {
1879 Value *V = MDV->getValue();
1883 WriterCtx.onWriteMetadataAsOperand(MD);
1892struct MDFieldPrinter {
1895 AsmWriterContext &WriterCtx;
1897 explicit MDFieldPrinter(raw_ostream &Out)
1898 : Out(Out), WriterCtx(AsmWriterContext::getEmpty()) {}
1899 MDFieldPrinter(raw_ostream &Out, AsmWriterContext &Ctx)
1900 : Out(Out), WriterCtx(Ctx) {}
1902 void printTag(
const DINode *
N);
1903 void printMacinfoType(
const DIMacroNode *
N);
1904 void printChecksum(
const DIFile::ChecksumInfo<StringRef> &
N);
1905 void printString(StringRef Name, StringRef
Value,
1906 bool ShouldSkipEmpty =
true);
1907 void printMetadata(StringRef Name,
const Metadata *MD,
1908 bool ShouldSkipNull =
true);
1909 void printMetadataOrInt(StringRef Name,
const Metadata *MD,
bool IsUnsigned,
1910 bool ShouldSkipZero =
true);
1911 template <
class IntTy>
1912 void printInt(StringRef Name, IntTy
Int,
bool ShouldSkipZero =
true);
1913 void printAPInt(StringRef Name,
const APInt &
Int,
bool IsUnsigned,
1914 bool ShouldSkipZero);
1915 void printBool(StringRef Name,
bool Value,
1916 std::optional<bool>
Default = std::nullopt);
1919 template <
class IntTy,
class Stringifier>
1920 void printDwarfEnum(StringRef Name, IntTy
Value, Stringifier
toString,
1921 bool ShouldSkipZero =
true);
1923 void printNameTableKind(StringRef Name,
1930void MDFieldPrinter::printTag(
const DINode *
N) {
1931 Out <<
FS <<
"tag: ";
1939void MDFieldPrinter::printMacinfoType(
const DIMacroNode *
N) {
1940 Out <<
FS <<
"type: ";
1945 Out <<
N->getMacinfoType();
1948void MDFieldPrinter::printChecksum(
1951 printString(
"checksum", Checksum.
Value,
false);
1955 bool ShouldSkipEmpty) {
1956 if (ShouldSkipEmpty &&
Value.empty())
1959 Out <<
FS <<
Name <<
": \"";
1965 AsmWriterContext &WriterCtx) {
1971 WriterCtx.onWriteMetadataAsOperand(MD);
1975 bool ShouldSkipNull) {
1976 if (ShouldSkipNull && !MD)
1979 Out <<
FS <<
Name <<
": ";
1984 bool IsUnsigned,
bool ShouldSkipZero) {
1991 printInt(Name, CV->getZExtValue(), ShouldSkipZero);
1993 printInt(Name, CV->getSExtValue(), ShouldSkipZero);
1995 printMetadata(Name, MD);
1998template <
class IntTy>
1999void MDFieldPrinter::printInt(
StringRef Name, IntTy
Int,
bool ShouldSkipZero) {
2000 if (ShouldSkipZero && !
Int)
2007 bool IsUnsigned,
bool ShouldSkipZero) {
2008 if (ShouldSkipZero &&
Int.isZero())
2011 Out <<
FS <<
Name <<
": ";
2012 Int.print(Out, !IsUnsigned);
2016 std::optional<bool>
Default) {
2019 Out <<
FS <<
Name <<
": " << (
Value ?
"true" :
"false");
2026 Out <<
FS <<
Name <<
": ";
2032 for (
auto F : SplitFlags) {
2034 assert(!StringF.empty() &&
"Expected valid flag");
2035 Out << FlagsFS << StringF;
2037 if (Extra || SplitFlags.empty())
2038 Out << FlagsFS << Extra;
2041void MDFieldPrinter::printDISPFlags(
StringRef Name,
2045 Out <<
FS <<
Name <<
": ";
2056 for (
auto F : SplitFlags) {
2058 assert(!StringF.empty() &&
"Expected valid flag");
2059 Out << FlagsFS << StringF;
2061 if (Extra || SplitFlags.empty())
2062 Out << FlagsFS << Extra;
2065void MDFieldPrinter::printEmissionKind(
StringRef Name,
2070void MDFieldPrinter::printNameTableKind(
StringRef Name,
2077void MDFieldPrinter::printFixedPointKind(
StringRef Name,
2082template <
class IntTy,
class Stringifier>
2084 Stringifier
toString,
bool ShouldSkipZero) {
2085 if (ShouldSkipZero && !
Value)
2088 Out <<
FS <<
Name <<
": ";
2097 AsmWriterContext &WriterCtx) {
2098 Out <<
"!GenericDINode(";
2099 MDFieldPrinter
Printer(Out, WriterCtx);
2101 Printer.printString(
"header",
N->getHeader());
2102 if (
N->getNumDwarfOperands()) {
2103 Out <<
Printer.FS <<
"operands: {";
2105 for (
auto &
I :
N->dwarf_operands()) {
2115 AsmWriterContext &WriterCtx) {
2116 Out <<
"!DILocation(";
2117 MDFieldPrinter
Printer(Out, WriterCtx);
2119 Printer.printInt(
"line",
DL->getLine(),
false);
2120 Printer.printInt(
"column",
DL->getColumn());
2121 Printer.printMetadata(
"scope",
DL->getRawScope(),
false);
2122 Printer.printMetadata(
"inlinedAt",
DL->getRawInlinedAt());
2123 Printer.printBool(
"isImplicitCode",
DL->isImplicitCode(),
2125 Printer.printInt(
"atomGroup",
DL->getAtomGroup());
2126 Printer.printInt<
unsigned>(
"atomRank",
DL->getAtomRank());
2131 AsmWriterContext &WriterCtx) {
2132 Out <<
"!DIAssignID()";
2133 MDFieldPrinter
Printer(Out, WriterCtx);
2137 AsmWriterContext &WriterCtx) {
2138 Out <<
"!DISubrange(";
2139 MDFieldPrinter
Printer(Out, WriterCtx);
2141 Printer.printMetadataOrInt(
"count",
N->getRawCountNode(),
2147 Printer.printMetadataOrInt(
"lowerBound",
N->getRawLowerBound(),
2150 Printer.printMetadataOrInt(
"upperBound",
N->getRawUpperBound(),
2153 Printer.printMetadataOrInt(
"stride",
N->getRawStride(),
2161 AsmWriterContext &WriterCtx) {
2162 Out <<
"!DIGenericSubrange(";
2163 MDFieldPrinter
Printer(Out, WriterCtx);
2165 auto GetConstant = [&](
Metadata *Bound) -> std::optional<int64_t> {
2168 return std::nullopt;
2169 if (BE->isConstant() &&
2171 *BE->isConstant()) {
2172 return static_cast<int64_t
>(BE->getElement(1));
2174 return std::nullopt;
2177 auto *
Count =
N->getRawCountNode();
2178 if (
auto ConstantCount = GetConstant(
Count))
2179 Printer.printInt(
"count", *ConstantCount,
2184 auto *LBound =
N->getRawLowerBound();
2185 if (
auto ConstantLBound = GetConstant(LBound))
2186 Printer.printInt(
"lowerBound", *ConstantLBound,
2189 Printer.printMetadata(
"lowerBound", LBound,
true);
2191 auto *UBound =
N->getRawUpperBound();
2192 if (
auto ConstantUBound = GetConstant(UBound))
2193 Printer.printInt(
"upperBound", *ConstantUBound,
2196 Printer.printMetadata(
"upperBound", UBound,
true);
2198 auto *Stride =
N->getRawStride();
2199 if (
auto ConstantStride = GetConstant(Stride))
2200 Printer.printInt(
"stride", *ConstantStride,
2203 Printer.printMetadata(
"stride", Stride,
true);
2209 AsmWriterContext &) {
2210 Out <<
"!DIEnumerator(";
2212 Printer.printString(
"name",
N->getName(),
false);
2213 Printer.printAPInt(
"value",
N->getValue(),
N->isUnsigned(),
2215 if (
N->isUnsigned())
2216 Printer.printBool(
"isUnsigned",
true);
2221 AsmWriterContext &WriterCtx) {
2222 Out <<
"!DIBasicType(";
2223 MDFieldPrinter
Printer(Out, WriterCtx);
2224 if (
N->getTag() != dwarf::DW_TAG_base_type)
2226 Printer.printString(
"name",
N->getName());
2227 Printer.printMetadata(
"scope",
N->getRawScope());
2228 Printer.printMetadata(
"file",
N->getRawFile());
2229 Printer.printInt(
"line",
N->getLine());
2230 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2231 Printer.printInt(
"align",
N->getAlignInBits());
2232 Printer.printInt(
"dataSize",
N->getDataSizeInBits());
2233 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2235 Printer.printInt(
"num_extra_inhabitants",
N->getNumExtraInhabitants());
2236 Printer.printDIFlags(
"flags",
N->getFlags());
2241 AsmWriterContext &WriterCtx) {
2242 Out <<
"!DIFixedPointType(";
2243 MDFieldPrinter
Printer(Out, WriterCtx);
2244 if (
N->getTag() != dwarf::DW_TAG_base_type)
2246 Printer.printString(
"name",
N->getName());
2247 Printer.printMetadata(
"scope",
N->getRawScope());
2248 Printer.printMetadata(
"file",
N->getRawFile());
2249 Printer.printInt(
"line",
N->getLine());
2250 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2251 Printer.printInt(
"align",
N->getAlignInBits());
2252 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2254 Printer.printDIFlags(
"flags",
N->getFlags());
2255 Printer.printFixedPointKind(
"kind",
N->getKind());
2256 if (
N->isRational()) {
2257 bool IsUnsigned = !
N->isSigned();
2258 Printer.printAPInt(
"numerator",
N->getNumerator(), IsUnsigned,
false);
2259 Printer.printAPInt(
"denominator",
N->getDenominator(), IsUnsigned,
false);
2261 Printer.printInt(
"factor",
N->getFactor());
2267 AsmWriterContext &WriterCtx) {
2268 Out <<
"!DIStringType(";
2269 MDFieldPrinter
Printer(Out, WriterCtx);
2270 if (
N->getTag() != dwarf::DW_TAG_string_type)
2272 Printer.printString(
"name",
N->getName());
2273 Printer.printMetadata(
"stringLength",
N->getRawStringLength());
2274 Printer.printMetadata(
"stringLengthExpression",
N->getRawStringLengthExp());
2275 Printer.printMetadata(
"stringLocationExpression",
2276 N->getRawStringLocationExp());
2277 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2278 Printer.printInt(
"align",
N->getAlignInBits());
2279 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2285 AsmWriterContext &WriterCtx) {
2286 Out <<
"!DIDerivedType(";
2287 MDFieldPrinter
Printer(Out, WriterCtx);
2289 Printer.printString(
"name",
N->getName());
2290 Printer.printMetadata(
"scope",
N->getRawScope());
2291 Printer.printMetadata(
"file",
N->getRawFile());
2292 Printer.printInt(
"line",
N->getLine());
2293 Printer.printMetadata(
"baseType",
N->getRawBaseType(),
2295 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2296 Printer.printInt(
"align",
N->getAlignInBits());
2297 Printer.printMetadataOrInt(
"offset",
N->getRawOffsetInBits(),
true);
2298 Printer.printDIFlags(
"flags",
N->getFlags());
2299 Printer.printMetadata(
"extraData",
N->getRawExtraData());
2300 if (
const auto &DWARFAddressSpace =
N->getDWARFAddressSpace())
2301 Printer.printInt(
"dwarfAddressSpace", *DWARFAddressSpace,
2303 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2304 if (
auto PtrAuthData =
N->getPtrAuthData()) {
2305 Printer.printInt(
"ptrAuthKey", PtrAuthData->key());
2306 Printer.printBool(
"ptrAuthIsAddressDiscriminated",
2307 PtrAuthData->isAddressDiscriminated());
2308 Printer.printInt(
"ptrAuthExtraDiscriminator",
2309 PtrAuthData->extraDiscriminator());
2310 Printer.printBool(
"ptrAuthIsaPointer", PtrAuthData->isaPointer());
2311 Printer.printBool(
"ptrAuthAuthenticatesNullValues",
2312 PtrAuthData->authenticatesNullValues());
2318 AsmWriterContext &WriterCtx) {
2319 Out <<
"!DISubrangeType(";
2320 MDFieldPrinter
Printer(Out, WriterCtx);
2321 Printer.printString(
"name",
N->getName());
2322 Printer.printMetadata(
"scope",
N->getRawScope());
2323 Printer.printMetadata(
"file",
N->getRawFile());
2324 Printer.printInt(
"line",
N->getLine());
2325 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2326 Printer.printInt(
"align",
N->getAlignInBits());
2327 Printer.printDIFlags(
"flags",
N->getFlags());
2328 Printer.printMetadata(
"baseType",
N->getRawBaseType(),
2330 Printer.printMetadata(
"lowerBound",
N->getRawLowerBound());
2331 Printer.printMetadata(
"upperBound",
N->getRawUpperBound());
2332 Printer.printMetadata(
"stride",
N->getRawStride());
2333 Printer.printMetadata(
"bias",
N->getRawBias());
2338 AsmWriterContext &WriterCtx) {
2339 Out <<
"!DICompositeType(";
2340 MDFieldPrinter
Printer(Out, WriterCtx);
2342 Printer.printString(
"name",
N->getName());
2343 Printer.printMetadata(
"scope",
N->getRawScope());
2344 Printer.printMetadata(
"file",
N->getRawFile());
2345 Printer.printInt(
"line",
N->getLine());
2346 Printer.printMetadata(
"baseType",
N->getRawBaseType());
2347 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2348 Printer.printInt(
"align",
N->getAlignInBits());
2349 Printer.printMetadataOrInt(
"offset",
N->getRawOffsetInBits(),
true);
2350 Printer.printInt(
"num_extra_inhabitants",
N->getNumExtraInhabitants());
2351 Printer.printDIFlags(
"flags",
N->getFlags());
2352 Printer.printMetadata(
"elements",
N->getRawElements());
2353 Printer.printDwarfEnum(
"runtimeLang",
N->getRuntimeLang(),
2355 Printer.printMetadata(
"vtableHolder",
N->getRawVTableHolder());
2356 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2357 Printer.printString(
"identifier",
N->getIdentifier());
2358 Printer.printMetadata(
"discriminator",
N->getRawDiscriminator());
2359 Printer.printMetadata(
"dataLocation",
N->getRawDataLocation());
2360 Printer.printMetadata(
"associated",
N->getRawAssociated());
2361 Printer.printMetadata(
"allocated",
N->getRawAllocated());
2362 if (
auto *RankConst =
N->getRankConst())
2363 Printer.printInt(
"rank", RankConst->getSExtValue(),
2366 Printer.printMetadata(
"rank",
N->getRawRank(),
true);
2367 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2368 if (
auto *Specification =
N->getRawSpecification())
2369 Printer.printMetadata(
"specification", Specification);
2371 if (
auto EnumKind =
N->getEnumKind())
2375 Printer.printMetadata(
"bitStride",
N->getRawBitStride());
2380 AsmWriterContext &WriterCtx) {
2381 Out <<
"!DISubroutineType(";
2382 MDFieldPrinter
Printer(Out, WriterCtx);
2383 Printer.printDIFlags(
"flags",
N->getFlags());
2385 Printer.printMetadata(
"types",
N->getRawTypeArray(),
2393 Printer.printString(
"filename",
N->getFilename(),
2395 Printer.printString(
"directory",
N->getDirectory(),
2398 if (
N->getChecksum())
2399 Printer.printChecksum(*
N->getChecksum());
2401 Printer.printString(
"source", *
N->getSource(),
2407 AsmWriterContext &WriterCtx) {
2408 Out <<
"!DICompileUnit(";
2409 MDFieldPrinter
Printer(Out, WriterCtx);
2415 "sourceLanguageName",
2427 Printer.printMetadata(
"file",
N->getRawFile(),
false);
2428 Printer.printString(
"producer",
N->getProducer());
2429 Printer.printBool(
"isOptimized",
N->isOptimized());
2430 Printer.printString(
"flags",
N->getFlags());
2431 Printer.printInt(
"runtimeVersion",
N->getRuntimeVersion(),
2433 Printer.printString(
"splitDebugFilename",
N->getSplitDebugFilename());
2434 Printer.printEmissionKind(
"emissionKind",
N->getEmissionKind());
2435 Printer.printMetadata(
"enums",
N->getRawEnumTypes());
2436 Printer.printMetadata(
"retainedTypes",
N->getRawRetainedTypes());
2437 Printer.printMetadata(
"globals",
N->getRawGlobalVariables());
2438 Printer.printMetadata(
"imports",
N->getRawImportedEntities());
2439 Printer.printMetadata(
"macros",
N->getRawMacros());
2440 Printer.printInt(
"dwoId",
N->getDWOId());
2441 Printer.printBool(
"splitDebugInlining",
N->getSplitDebugInlining(),
true);
2442 Printer.printBool(
"debugInfoForProfiling",
N->getDebugInfoForProfiling(),
2444 Printer.printNameTableKind(
"nameTableKind",
N->getNameTableKind());
2445 Printer.printBool(
"rangesBaseAddress",
N->getRangesBaseAddress(),
false);
2446 Printer.printString(
"sysroot",
N->getSysRoot());
2447 Printer.printString(
"sdk",
N->getSDK());
2454 AsmWriterContext &WriterCtx) {
2455 Out <<
"!DISubprogram(";
2456 MDFieldPrinter
Printer(Out, WriterCtx);
2457 Printer.printString(
"name",
N->getName());
2458 Printer.printString(
"linkageName",
N->getLinkageName());
2459 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2460 Printer.printMetadata(
"file",
N->getRawFile());
2461 Printer.printInt(
"line",
N->getLine());
2462 Printer.printMetadata(
"type",
N->getRawType());
2463 Printer.printInt(
"scopeLine",
N->getScopeLine());
2464 Printer.printMetadata(
"containingType",
N->getRawContainingType());
2465 if (
N->getVirtuality() != dwarf::DW_VIRTUALITY_none ||
2466 N->getVirtualIndex() != 0)
2467 Printer.printInt(
"virtualIndex",
N->getVirtualIndex(),
false);
2468 Printer.printInt(
"thisAdjustment",
N->getThisAdjustment());
2469 Printer.printDIFlags(
"flags",
N->getFlags());
2470 Printer.printDISPFlags(
"spFlags",
N->getSPFlags());
2471 Printer.printMetadata(
"unit",
N->getRawUnit());
2472 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2473 Printer.printMetadata(
"declaration",
N->getRawDeclaration());
2474 Printer.printMetadata(
"retainedNodes",
N->getRawRetainedNodes());
2475 Printer.printMetadata(
"thrownTypes",
N->getRawThrownTypes());
2476 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2477 Printer.printString(
"targetFuncName",
N->getTargetFuncName());
2478 Printer.printBool(
"keyInstructions",
N->getKeyInstructionsEnabled(),
false);
2483 AsmWriterContext &WriterCtx) {
2484 Out <<
"!DILexicalBlock(";
2485 MDFieldPrinter
Printer(Out, WriterCtx);
2486 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2487 Printer.printMetadata(
"file",
N->getRawFile());
2488 Printer.printInt(
"line",
N->getLine());
2489 Printer.printInt(
"column",
N->getColumn());
2495 AsmWriterContext &WriterCtx) {
2496 Out <<
"!DILexicalBlockFile(";
2497 MDFieldPrinter
Printer(Out, WriterCtx);
2498 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2499 Printer.printMetadata(
"file",
N->getRawFile());
2500 Printer.printInt(
"discriminator",
N->getDiscriminator(),
2506 AsmWriterContext &WriterCtx) {
2507 Out <<
"!DINamespace(";
2508 MDFieldPrinter
Printer(Out, WriterCtx);
2509 Printer.printString(
"name",
N->getName());
2510 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2511 Printer.printBool(
"exportSymbols",
N->getExportSymbols(),
false);
2516 AsmWriterContext &WriterCtx) {
2517 Out <<
"!DICommonBlock(";
2518 MDFieldPrinter
Printer(Out, WriterCtx);
2519 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2520 Printer.printMetadata(
"declaration",
N->getRawDecl(),
false);
2521 Printer.printString(
"name",
N->getName());
2522 Printer.printMetadata(
"file",
N->getRawFile());
2523 Printer.printInt(
"line",
N->getLineNo());
2528 AsmWriterContext &WriterCtx) {
2530 MDFieldPrinter
Printer(Out, WriterCtx);
2532 Printer.printInt(
"line",
N->getLine());
2533 Printer.printString(
"name",
N->getName());
2534 Printer.printString(
"value",
N->getValue());
2539 AsmWriterContext &WriterCtx) {
2540 Out <<
"!DIMacroFile(";
2541 MDFieldPrinter
Printer(Out, WriterCtx);
2542 Printer.printInt(
"line",
N->getLine());
2543 Printer.printMetadata(
"file",
N->getRawFile(),
false);
2544 Printer.printMetadata(
"nodes",
N->getRawElements());
2549 AsmWriterContext &WriterCtx) {
2550 Out <<
"!DIModule(";
2551 MDFieldPrinter
Printer(Out, WriterCtx);
2552 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2553 Printer.printString(
"name",
N->getName());
2554 Printer.printString(
"configMacros",
N->getConfigurationMacros());
2555 Printer.printString(
"includePath",
N->getIncludePath());
2556 Printer.printString(
"apinotes",
N->getAPINotesFile());
2557 Printer.printMetadata(
"file",
N->getRawFile());
2558 Printer.printInt(
"line",
N->getLineNo());
2559 Printer.printBool(
"isDecl",
N->getIsDecl(),
false);
2565 AsmWriterContext &WriterCtx) {
2566 Out <<
"!DITemplateTypeParameter(";
2567 MDFieldPrinter
Printer(Out, WriterCtx);
2568 Printer.printString(
"name",
N->getName());
2569 Printer.printMetadata(
"type",
N->getRawType(),
false);
2570 Printer.printBool(
"defaulted",
N->isDefault(),
false);
2576 AsmWriterContext &WriterCtx) {
2577 Out <<
"!DITemplateValueParameter(";
2578 MDFieldPrinter
Printer(Out, WriterCtx);
2579 if (
N->getTag() != dwarf::DW_TAG_template_value_parameter)
2581 Printer.printString(
"name",
N->getName());
2582 Printer.printMetadata(
"type",
N->getRawType());
2583 Printer.printBool(
"defaulted",
N->isDefault(),
false);
2584 Printer.printMetadata(
"value",
N->getValue(),
false);
2589 AsmWriterContext &WriterCtx) {
2590 Out <<
"!DIGlobalVariable(";
2591 MDFieldPrinter
Printer(Out, WriterCtx);
2592 Printer.printString(
"name",
N->getName());
2593 Printer.printString(
"linkageName",
N->getLinkageName());
2594 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2595 Printer.printMetadata(
"file",
N->getRawFile());
2596 Printer.printInt(
"line",
N->getLine());
2597 Printer.printMetadata(
"type",
N->getRawType());
2598 Printer.printBool(
"isLocal",
N->isLocalToUnit());
2599 Printer.printBool(
"isDefinition",
N->isDefinition());
2600 Printer.printMetadata(
"declaration",
N->getRawStaticDataMemberDeclaration());
2601 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2602 Printer.printInt(
"align",
N->getAlignInBits());
2603 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2608 AsmWriterContext &WriterCtx) {
2609 Out <<
"!DILocalVariable(";
2610 MDFieldPrinter
Printer(Out, WriterCtx);
2611 Printer.printString(
"name",
N->getName());
2612 Printer.printInt(
"arg",
N->getArg());
2613 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2614 Printer.printMetadata(
"file",
N->getRawFile());
2615 Printer.printInt(
"line",
N->getLine());
2616 Printer.printMetadata(
"type",
N->getRawType());
2617 Printer.printDIFlags(
"flags",
N->getFlags());
2618 Printer.printInt(
"align",
N->getAlignInBits());
2619 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2624 AsmWriterContext &WriterCtx) {
2626 MDFieldPrinter
Printer(Out, WriterCtx);
2627 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2628 Printer.printString(
"name",
N->getName());
2629 Printer.printMetadata(
"file",
N->getRawFile());
2630 Printer.printInt(
"line",
N->getLine(),
false);
2631 Printer.printInt(
"column",
N->getColumn());
2632 Printer.printBool(
"isArtificial",
N->isArtificial(),
false);
2633 if (
N->getCoroSuspendIdx())
2634 Printer.printInt(
"coroSuspendIdx", *
N->getCoroSuspendIdx(),
2640 AsmWriterContext &WriterCtx) {
2641 Out <<
"!DIExpression(";
2646 assert(!OpStr.empty() &&
"Expected valid opcode");
2650 Out << FS <<
Op.getArg(0);
2653 for (
unsigned A = 0, AE =
Op.getNumArgs();
A != AE; ++
A)
2654 Out << FS <<
Op.getArg(
A);
2658 for (
const auto &
I :
N->getElements())
2665 AsmWriterContext &WriterCtx,
2666 bool FromValue =
false) {
2668 "Unexpected DIArgList metadata outside of value argument");
2669 Out <<
"!DIArgList(";
2671 MDFieldPrinter
Printer(Out, WriterCtx);
2672 for (
const Metadata *Arg :
N->getArgs()) {
2681 AsmWriterContext &WriterCtx) {
2682 Out <<
"!DIGlobalVariableExpression(";
2683 MDFieldPrinter
Printer(Out, WriterCtx);
2684 Printer.printMetadata(
"var",
N->getVariable());
2685 Printer.printMetadata(
"expr",
N->getExpression());
2690 AsmWriterContext &WriterCtx) {
2691 Out <<
"!DIObjCProperty(";
2692 MDFieldPrinter
Printer(Out, WriterCtx);
2693 Printer.printString(
"name",
N->getName());
2694 Printer.printMetadata(
"file",
N->getRawFile());
2695 Printer.printInt(
"line",
N->getLine());
2696 Printer.printString(
"setter",
N->getSetterName());
2697 Printer.printString(
"getter",
N->getGetterName());
2698 Printer.printInt(
"attributes",
N->getAttributes());
2699 Printer.printMetadata(
"type",
N->getRawType());
2704 AsmWriterContext &WriterCtx) {
2705 Out <<
"!DIImportedEntity(";
2706 MDFieldPrinter
Printer(Out, WriterCtx);
2708 Printer.printString(
"name",
N->getName());
2709 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2710 Printer.printMetadata(
"entity",
N->getRawEntity());
2711 Printer.printMetadata(
"file",
N->getRawFile());
2712 Printer.printInt(
"line",
N->getLine());
2713 Printer.printMetadata(
"elements",
N->getRawElements());
2718 AsmWriterContext &Ctx) {
2719 if (
Node->isDistinct())
2721 else if (
Node->isTemporary())
2722 Out <<
"<temporary!> ";
2724 switch (
Node->getMetadataID()) {
2727#define HANDLE_MDNODE_LEAF(CLASS) \
2728 case Metadata::CLASS##Kind: \
2729 write##CLASS(Out, cast<CLASS>(Node), Ctx); \
2731#include "llvm/IR/Metadata.def"
2738 AsmWriterContext &WriterCtx,
2741 WriterCtx.TypePrinter->print(V->getType(), Out);
2752 assert(WriterCtx.TypePrinter &&
"Constants require TypePrinting!");
2759 if (IA->hasSideEffects())
2760 Out <<
"sideeffect ";
2761 if (IA->isAlignStack())
2762 Out <<
"alignstack ";
2765 Out <<
"inteldialect ";
2784 auto *
Machine = WriterCtx.Machine;
2788 Slot =
Machine->getGlobalSlot(GV);
2791 Slot =
Machine->getLocalSlot(V);
2798 Slot =
Machine->getLocalSlot(V);
2805 Slot =
Machine->getGlobalSlot(GV);
2808 Slot =
Machine->getLocalSlot(V);
2817 Out << Prefix << Slot;
2823 AsmWriterContext &WriterCtx,
2837 std::unique_ptr<SlotTracker> MachineStorage;
2839 if (!WriterCtx.Machine) {
2840 MachineStorage = std::make_unique<SlotTracker>(WriterCtx.Context);
2841 WriterCtx.Machine = MachineStorage.get();
2851 Out <<
"<" <<
N <<
">";
2865 assert(WriterCtx.TypePrinter &&
"TypePrinter required for metadata values");
2867 "Unexpected function-local metadata outside of value argument");
2874class AssemblyWriter {
2875 formatted_raw_ostream &Out;
2876 const Module *TheModule =
nullptr;
2877 const ModuleSummaryIndex *TheIndex =
nullptr;
2878 std::unique_ptr<SlotTracker> SlotTrackerStorage;
2880 TypePrinting TypePrinter;
2881 AssemblyAnnotationWriter *AnnotationWriter =
nullptr;
2882 SetVector<const Comdat *> Comdats;
2884 bool ShouldPreserveUseListOrder;
2889 DenseMap<const GlobalValueSummary *, GlobalValue::GUID> SummaryToGUIDMap;
2893 AssemblyWriter(formatted_raw_ostream &o, SlotTracker &Mac,
const Module *M,
2894 AssemblyAnnotationWriter *AAW,
bool IsForDebug,
2895 bool ShouldPreserveUseListOrder =
false);
2897 AssemblyWriter(formatted_raw_ostream &o, SlotTracker &Mac,
2898 const ModuleSummaryIndex *Index,
bool IsForDebug);
2901 return AsmWriterContext(&TypePrinter, &
Machine, TheModule);
2904 void printMDNodeBody(
const MDNode *MD);
2905 void printNamedMDNode(
const NamedMDNode *NMD);
2907 void printModule(
const Module *M);
2909 void writeOperand(
const Value *
Op,
bool PrintType);
2910 void writeParamOperand(
const Value *Operand, AttributeSet Attrs);
2911 void writeOperandBundles(
const CallBase *
Call);
2912 void writeSyncScope(
const LLVMContext &
Context,
2914 void writeAtomic(
const LLVMContext &
Context,
2917 void writeAtomicCmpXchg(
const LLVMContext &
Context,
2922 void writeAllMDNodes();
2923 void writeMDNode(
unsigned Slot,
const MDNode *Node);
2924 void writeAttribute(
const Attribute &Attr,
bool InAttrGroup =
false);
2925 void writeAttributeSet(
const AttributeSet &AttrSet,
bool InAttrGroup =
false);
2926 void writeAllAttributeGroups();
2928 void printTypeIdentities();
2929 void printGlobal(
const GlobalVariable *GV);
2930 void printAlias(
const GlobalAlias *GA);
2931 void printIFunc(
const GlobalIFunc *GI);
2932 void printComdat(
const Comdat *
C);
2933 void printFunction(
const Function *
F);
2934 void printArgument(
const Argument *FA, AttributeSet Attrs);
2936 void printInstructionLine(
const Instruction &
I);
2937 void printInstruction(
const Instruction &
I);
2938 void printDbgMarker(
const DbgMarker &DPI);
2939 void printDbgVariableRecord(
const DbgVariableRecord &DVR);
2940 void printDbgLabelRecord(
const DbgLabelRecord &DLR);
2941 void printDbgRecord(
const DbgRecord &DR);
2942 void printDbgRecordLine(
const DbgRecord &DR);
2944 void printUseListOrder(
const Value *V, ArrayRef<unsigned> Shuffle);
2945 void printUseLists(
const Function *
F);
2947 void printModuleSummaryIndex();
2948 void printSummaryInfo(
unsigned Slot,
const ValueInfo &VI);
2949 void printSummary(
const GlobalValueSummary &Summary);
2950 void printAliasSummary(
const AliasSummary *AS);
2951 void printGlobalVarSummary(
const GlobalVarSummary *GS);
2952 void printFunctionSummary(
const FunctionSummary *FS);
2953 void printTypeIdSummary(
const TypeIdSummary &TIS);
2955 void printTypeTestResolution(
const TypeTestResolution &TTRes);
2956 void printArgs(ArrayRef<uint64_t> Args);
2957 void printWPDRes(
const WholeProgramDevirtResolution &WPDRes);
2958 void printTypeIdInfo(
const FunctionSummary::TypeIdInfo &TIDInfo);
2959 void printVFuncId(
const FunctionSummary::VFuncId VFId);
2967 void printMetadataAttachments(
2968 const SmallVectorImpl<std::pair<unsigned, MDNode *>> &MDs,
2969 StringRef Separator);
2973 void printInfoComment(
const Value &V,
bool isMaterializable =
false);
2977 void printGCRelocateComment(
const GCRelocateInst &Relocate);
2984 bool IsForDebug,
bool ShouldPreserveUseListOrder)
2985 : Out(
o), TheModule(
M),
Machine(Mac), TypePrinter(
M), AnnotationWriter(AAW),
2986 IsForDebug(IsForDebug),
2987 ShouldPreserveUseListOrder(
2990 : ShouldPreserveUseListOrder) {
2993 for (
const GlobalObject &GO : TheModule->global_objects())
3000 : Out(
o), TheIndex(Index),
Machine(Mac), TypePrinter(nullptr),
3001 IsForDebug(IsForDebug),
3004void AssemblyWriter::writeOperand(
const Value *Operand,
bool PrintType) {
3006 Out <<
"<null operand!>";
3013void AssemblyWriter::writeSyncScope(
const LLVMContext &
Context,
3021 Context.getSyncScopeNames(SSNs);
3023 Out <<
" syncscope(\"";
3031void AssemblyWriter::writeAtomic(
const LLVMContext &
Context,
3034 if (Ordering == AtomicOrdering::NotAtomic)
3037 writeSyncScope(
Context, SSID);
3041void AssemblyWriter::writeAtomicCmpXchg(
const LLVMContext &
Context,
3045 assert(SuccessOrdering != AtomicOrdering::NotAtomic &&
3046 FailureOrdering != AtomicOrdering::NotAtomic);
3048 writeSyncScope(
Context, SSID);
3053void AssemblyWriter::writeParamOperand(
const Value *Operand,
3054 AttributeSet Attrs) {
3056 Out <<
"<null operand!>";
3061 TypePrinter.print(Operand->
getType(), Out);
3063 if (
Attrs.hasAttributes()) {
3065 writeAttributeSet(Attrs);
3073void AssemblyWriter::writeOperandBundles(
const CallBase *
Call) {
3089 ListSeparator InnerLS;
3091 for (
const auto &Input : BU.
Inputs) {
3093 if (Input ==
nullptr)
3094 Out <<
"<null operand bundle!>";
3105void AssemblyWriter::printModule(
const Module *M) {
3108 if (ShouldPreserveUseListOrder)
3111 if (!
M->getModuleIdentifier().empty() &&
3114 M->getModuleIdentifier().find(
'\n') == std::string::npos)
3115 Out <<
"; ModuleID = '" <<
M->getModuleIdentifier() <<
"'\n";
3117 if (!
M->getSourceFileName().empty()) {
3118 Out <<
"source_filename = \"";
3123 const std::string &
DL =
M->getDataLayoutStr();
3125 Out <<
"target datalayout = \"" <<
DL <<
"\"\n";
3126 if (!
M->getTargetTriple().empty())
3127 Out <<
"target triple = \"" <<
M->getTargetTriple().str() <<
"\"\n";
3129 if (!
M->getModuleInlineAsm().empty()) {
3133 StringRef
Asm =
M->getModuleInlineAsm();
3136 std::tie(Front, Asm) =
Asm.split(
'\n');
3140 Out <<
"module asm \"";
3143 }
while (!
Asm.empty());
3146 printTypeIdentities();
3149 if (!Comdats.empty())
3151 for (
const Comdat *
C : Comdats) {
3153 if (
C != Comdats.back())
3158 if (!
M->global_empty()) Out <<
'\n';
3159 for (
const GlobalVariable &GV :
M->globals()) {
3160 printGlobal(&GV); Out <<
'\n';
3164 if (!
M->alias_empty()) Out <<
"\n";
3165 for (
const GlobalAlias &GA :
M->aliases())
3169 if (!
M->ifunc_empty()) Out <<
"\n";
3170 for (
const GlobalIFunc &GI :
M->ifuncs())
3174 for (
const Function &
F : *M) {
3180 printUseLists(
nullptr);
3185 writeAllAttributeGroups();
3189 if (!
M->named_metadata_empty()) Out <<
'\n';
3191 for (
const NamedMDNode &Node :
M->named_metadata())
3192 printNamedMDNode(&Node);
3201void AssemblyWriter::printModuleSummaryIndex() {
3203 int NumSlots =
Machine.initializeIndexIfNeeded();
3209 std::vector<std::pair<std::string, ModuleHash>> moduleVec;
3210 std::string RegularLTOModuleName =
3213 for (
auto &[ModPath, ModHash] : TheIndex->
modulePaths())
3214 moduleVec[
Machine.getModulePathSlot(ModPath)] = std::make_pair(
3217 ModPath.empty() ? RegularLTOModuleName : std::string(ModPath), ModHash);
3220 for (
auto &ModPair : moduleVec) {
3221 Out <<
"^" << i++ <<
" = module: (";
3224 Out <<
"\", hash: (";
3226 for (
auto Hash : ModPair.second)
3236 for (
const auto &GlobalList : SortedGVS) {
3237 auto GUID = GlobalList.first;
3238 for (
auto &Summary : GlobalList.second.getSummaryList())
3243 for (
const auto &GlobalList : SortedGVS) {
3244 auto GUID = GlobalList.first;
3246 printSummaryInfo(
Machine.getGUIDSlot(GUID), VI);
3250 for (
const auto &TID : TheIndex->
typeIds()) {
3251 Out <<
"^" <<
Machine.getTypeIdSlot(TID.second.first)
3252 <<
" = typeid: (name: \"" << TID.second.first <<
"\"";
3253 printTypeIdSummary(TID.second.second);
3254 Out <<
") ; guid = " << TID.first <<
"\n";
3260 Out <<
"^" <<
Machine.getTypeIdCompatibleVtableSlot(TId.first)
3261 <<
" = typeidCompatibleVTable: (name: \"" << TId.first <<
"\"";
3262 printTypeIdCompatibleVtableSummary(TId.second);
3263 Out <<
") ; guid = " <<
GUID <<
"\n";
3268 Out <<
"^" << NumSlots <<
" = flags: " << TheIndex->
getFlags() <<
"\n";
3272 Out <<
"^" << NumSlots <<
" = blockcount: " << TheIndex->
getBlockCount()
3282 return "singleImpl";
3284 return "branchFunnel";
3295 return "uniformRetVal";
3297 return "uniqueRetVal";
3299 return "virtualConstProp";
3322void AssemblyWriter::printTypeTestResolution(
const TypeTestResolution &TTRes) {
3329 Out <<
", alignLog2: " << TTRes.
AlignLog2;
3331 Out <<
", sizeM1: " << TTRes.
SizeM1;
3334 Out <<
", bitMask: " << (unsigned)TTRes.
BitMask;
3341void AssemblyWriter::printTypeIdSummary(
const TypeIdSummary &TIS) {
3342 Out <<
", summary: (";
3343 printTypeTestResolution(TIS.
TTRes);
3344 if (!TIS.
WPDRes.empty()) {
3345 Out <<
", wpdResolutions: (";
3347 for (
auto &WPDRes : TIS.
WPDRes) {
3349 Out <<
"(offset: " << WPDRes.first <<
", ";
3350 printWPDRes(WPDRes.second);
3358void AssemblyWriter::printTypeIdCompatibleVtableSummary(
3360 Out <<
", summary: (";
3362 for (
auto &
P : TI) {
3364 Out <<
"(offset: " <<
P.AddressPointOffset <<
", ";
3365 Out <<
"^" <<
Machine.getGUIDSlot(
P.VTableVI.getGUID());
3371void AssemblyWriter::printArgs(ArrayRef<uint64_t> Args) {
3375void AssemblyWriter::printWPDRes(
const WholeProgramDevirtResolution &WPDRes) {
3376 Out <<
"wpdRes: (kind: ";
3383 Out <<
", resByArg: (";
3385 for (
auto &ResByArg : WPDRes.
ResByArg) {
3387 printArgs(ResByArg.first);
3388 Out <<
", byArg: (kind: ";
3390 if (ResByArg.second.TheKind ==
3392 ResByArg.second.TheKind ==
3394 Out <<
", info: " << ResByArg.second.Info;
3398 if (ResByArg.second.Byte || ResByArg.second.Bit)
3399 Out <<
", byte: " << ResByArg.second.Byte
3400 <<
", bit: " << ResByArg.second.Bit;
3421void AssemblyWriter::printAliasSummary(
const AliasSummary *AS) {
3422 Out <<
", aliasee: ";
3432void AssemblyWriter::printGlobalVarSummary(
const GlobalVarSummary *GS) {
3433 auto VTableFuncs =
GS->vTableFuncs();
3434 Out <<
", varFlags: (readonly: " <<
GS->VarFlags.MaybeReadOnly <<
", "
3435 <<
"writeonly: " <<
GS->VarFlags.MaybeWriteOnly <<
", "
3436 <<
"constant: " <<
GS->VarFlags.Constant;
3437 if (!VTableFuncs.empty())
3439 <<
"vcall_visibility: " <<
GS->VarFlags.VCallVisibility;
3442 if (!VTableFuncs.empty()) {
3443 Out <<
", vTableFuncs: (";
3445 for (
auto &
P : VTableFuncs) {
3447 Out <<
"(virtFunc: ^" <<
Machine.getGUIDSlot(
P.FuncVI.getGUID())
3448 <<
", offset: " <<
P.VTableOffset;
3466 return "linkonce_odr";
3476 return "extern_weak";
3478 return "available_externally";
3507 return "definition";
3509 return "declaration";
3514void AssemblyWriter::printFunctionSummary(
const FunctionSummary *FS) {
3515 Out <<
", insts: " <<
FS->instCount();
3516 if (
FS->fflags().anyFlagSet())
3517 Out <<
", " <<
FS->fflags();
3519 if (!
FS->calls().empty()) {
3520 Out <<
", calls: (";
3522 for (
auto &
Call :
FS->calls()) {
3524 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(
Call.first.getGUID());
3525 if (
Call.second.getHotness() != CalleeInfo::HotnessType::Unknown)
3529 if (
Call.second.HasTailCall)
3536 if (
const auto *TIdInfo =
FS->getTypeIdInfo())
3537 printTypeIdInfo(*TIdInfo);
3541 auto AllocTypeName = [](uint8_t
Type) ->
const char * {
3543 case (uint8_t)AllocationType::None:
3545 case (uint8_t)AllocationType::NotCold:
3547 case (uint8_t)AllocationType::Cold:
3549 case (uint8_t)AllocationType::Hot:
3555 if (!
FS->allocs().empty()) {
3556 Out <<
", allocs: (";
3558 for (
auto &AI :
FS->allocs()) {
3560 Out <<
"(versions: (";
3562 for (
auto V : AI.Versions) {
3564 Out << AllocTypeName(V);
3566 Out <<
"), memProf: (";
3567 ListSeparator MIBFS;
3568 for (
auto &MIB : AI.MIBs) {
3570 Out <<
"(type: " << AllocTypeName((uint8_t)MIB.AllocType);
3571 Out <<
", stackIds: (";
3572 ListSeparator SIDFS;
3573 for (
auto Id : MIB.StackIdIndices) {
3584 if (!
FS->callsites().empty()) {
3585 Out <<
", callsites: (";
3587 for (
auto &CI :
FS->callsites()) {
3590 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(CI.Callee.getGUID());
3592 Out <<
"(callee: null";
3593 Out <<
", clones: (";
3595 for (
auto V : CI.Clones) {
3599 Out <<
"), stackIds: (";
3600 ListSeparator SIDFS;
3601 for (
auto Id : CI.StackIdIndices) {
3610 auto PrintRange = [&](
const ConstantRange &
Range) {
3614 if (!
FS->paramAccesses().empty()) {
3615 Out <<
", params: (";
3617 for (
auto &PS :
FS->paramAccesses()) {
3619 Out <<
"(param: " << PS.ParamNo;
3620 Out <<
", offset: ";
3622 if (!PS.Calls.empty()) {
3623 Out <<
", calls: (";
3625 for (
auto &
Call : PS.Calls) {
3627 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(
Call.Callee.getGUID());
3628 Out <<
", param: " <<
Call.ParamNo;
3629 Out <<
", offset: ";
3630 PrintRange(
Call.Offsets);
3641void AssemblyWriter::printTypeIdInfo(
3642 const FunctionSummary::TypeIdInfo &TIDInfo) {
3643 Out <<
", typeIdInfo: (";
3644 ListSeparator TIDFS;
3647 Out <<
"typeTests: (";
3650 auto TidIter = TheIndex->
typeIds().equal_range(GUID);
3651 if (TidIter.first == TidIter.second) {
3657 for (
const auto &[GUID, TypeIdPair] :
make_range(TidIter)) {
3659 auto Slot =
Machine.getTypeIdSlot(TypeIdPair.first);
3677 "typeTestAssumeConstVCalls");
3682 "typeCheckedLoadConstVCalls");
3687void AssemblyWriter::printVFuncId(
const FunctionSummary::VFuncId VFId) {
3688 auto TidIter = TheIndex->
typeIds().equal_range(VFId.
GUID);
3689 if (TidIter.first == TidIter.second) {
3690 Out <<
"vFuncId: (";
3691 Out <<
"guid: " << VFId.
GUID;
3692 Out <<
", offset: " << VFId.
Offset;
3698 for (
const auto &[GUID, TypeIdPair] :
make_range(TidIter)) {
3700 Out <<
"vFuncId: (";
3701 auto Slot =
Machine.getTypeIdSlot(TypeIdPair.first);
3704 Out <<
", offset: " << VFId.
Offset;
3709void AssemblyWriter::printNonConstVCalls(
3711 Out <<
Tag <<
": (";
3713 for (
auto &VFuncId : VCallList) {
3715 printVFuncId(VFuncId);
3720void AssemblyWriter::printConstVCalls(
3722 Out <<
Tag <<
": (";
3724 for (
auto &ConstVCall : VCallList) {
3727 printVFuncId(ConstVCall.VFunc);
3728 if (!ConstVCall.Args.empty()) {
3730 printArgs(ConstVCall.Args);
3737void AssemblyWriter::printSummary(
const GlobalValueSummary &Summary) {
3738 GlobalValueSummary::GVFlags GVFlags =
Summary.flags();
3741 Out <<
"(module: ^" <<
Machine.getModulePathSlot(
Summary.modulePath())
3744 Out <<
", visibility: "
3747 Out <<
", live: " << GVFlags.
Live;
3748 Out <<
", dsoLocal: " << GVFlags.
DSOLocal;
3750 Out <<
", importType: "
3762 auto RefList =
Summary.refs();
3763 if (!RefList.empty()) {
3766 for (
auto &
Ref : RefList) {
3768 if (
Ref.isReadOnly())
3770 else if (
Ref.isWriteOnly())
3771 Out <<
"writeonly ";
3772 Out <<
"^" <<
Machine.getGUIDSlot(
Ref.getGUID());
3780void AssemblyWriter::printSummaryInfo(
unsigned Slot,
const ValueInfo &VI) {
3781 Out <<
"^" <<
Slot <<
" = gv: (";
3782 if (
VI.hasName() && !
VI.name().empty())
3783 Out <<
"name: \"" <<
VI.name() <<
"\"";
3785 Out <<
"guid: " <<
VI.getGUID();
3786 if (!
VI.getSummaryList().empty()) {
3787 Out <<
", summaries: (";
3789 for (
auto &Summary :
VI.getSummaryList()) {
3791 printSummary(*Summary);
3796 if (
VI.hasName() && !
VI.name().empty())
3797 Out <<
" ; guid = " <<
VI.getGUID();
3804 Out <<
"<empty name> ";
3806 unsigned char FirstC =
static_cast<unsigned char>(Name[0]);
3807 if (isalpha(FirstC) || FirstC ==
'-' || FirstC ==
'$' || FirstC ==
'.' ||
3812 for (
unsigned i = 1, e = Name.size(); i != e; ++i) {
3813 unsigned char C = Name[i];
3814 if (isalnum(
C) ||
C ==
'-' ||
C ==
'$' ||
C ==
'.' ||
C ==
'_')
3822void AssemblyWriter::printNamedMDNode(
const NamedMDNode *NMD) {
3857 Out <<
"dso_local ";
3875 Out <<
"thread_local ";
3878 Out <<
"thread_local(localdynamic) ";
3881 Out <<
"thread_local(initialexec) ";
3884 Out <<
"thread_local(localexec) ";
3894 return "local_unnamed_addr";
3896 return "unnamed_addr";
3919void AssemblyWriter::printGlobal(
const GlobalVariable *GV) {
3921 Out <<
"; Materializable\n";
3942 Out << (GV->
isConstant() ?
"constant " :
"global ");
3951 Out <<
", section \"";
3956 Out <<
", partition \"";
3961 Out <<
", code_model \"";
3986 Out <<
", no_sanitize_address";
3988 Out <<
", no_sanitize_hwaddress";
3990 Out <<
", sanitize_memtag";
3992 Out <<
", sanitize_address_dyninit";
3997 Out <<
", align " <<
A->value();
4001 printMetadataAttachments(MDs,
", ");
4004 if (
Attrs.hasAttributes())
4005 Out <<
" #" <<
Machine.getAttributeGroupSlot(Attrs);
4010void AssemblyWriter::printAlias(
const GlobalAlias *GA) {
4012 Out <<
"; Materializable\n";
4032 if (
const Constant *Aliasee = GA->
getAliasee()) {
4035 TypePrinter.print(GA->
getType(), Out);
4036 Out <<
" <<NULL ALIASEE>>";
4040 Out <<
", partition \"";
4049void AssemblyWriter::printIFunc(
const GlobalIFunc *GI) {
4051 Out <<
"; Materializable\n";
4066 if (
const Constant *Resolver = GI->
getResolver()) {
4069 TypePrinter.print(GI->
getType(), Out);
4070 Out <<
" <<NULL RESOLVER>>";
4074 Out <<
", partition \"";
4081 printMetadataAttachments(MDs,
", ");
4088void AssemblyWriter::printComdat(
const Comdat *
C) {
4092void AssemblyWriter::printTypeIdentities() {
4093 if (TypePrinter.empty())
4099 auto &NumberedTypes = TypePrinter.getNumberedTypes();
4100 for (
unsigned I = 0,
E = NumberedTypes.size();
I !=
E; ++
I) {
4101 Out <<
'%' <<
I <<
" = type ";
4105 TypePrinter.printStructBody(NumberedTypes[
I], Out);
4109 auto &NamedTypes = TypePrinter.getNamedTypes();
4110 for (StructType *NamedType : NamedTypes) {
4116 TypePrinter.printStructBody(NamedType, Out);
4122void AssemblyWriter::printFunction(
const Function *
F) {
4123 if (
F->isMaterializable())
4124 Out <<
"; Materializable\n";
4125 else if (AnnotationWriter)
4128 const AttributeList &
Attrs =
F->getAttributes();
4129 if (
Attrs.hasFnAttrs()) {
4130 AttributeSet AS =
Attrs.getFnAttrs();
4131 std::string AttrStr;
4134 if (!Attr.isStringAttribute()) {
4135 if (!AttrStr.empty()) AttrStr +=
' ';
4136 AttrStr += Attr.getAsString();
4140 if (!AttrStr.empty())
4141 Out <<
"; Function Attrs: " << AttrStr <<
'\n';
4145 Out <<
"; Unknown intrinsic\n";
4149 if (
F->isDeclaration()) {
4152 F->getAllMetadata(MDs);
4153 printMetadataAttachments(MDs,
" ");
4164 if (
F->getCallingConv() != CallingConv::C) {
4169 FunctionType *FT =
F->getFunctionType();
4170 if (
Attrs.hasRetAttrs())
4171 Out <<
Attrs.getAsString(AttributeList::ReturnIndex) <<
' ';
4172 TypePrinter.print(
F->getReturnType(), Out);
4179 if (
F->isDeclaration() && !IsForDebug) {
4182 for (
unsigned I = 0,
E = FT->getNumParams();
I !=
E; ++
I) {
4185 TypePrinter.print(FT->getParamType(
I), Out);
4187 AttributeSet ArgAttrs =
Attrs.getParamAttrs(
I);
4190 writeAttributeSet(ArgAttrs);
4196 for (
const Argument &Arg :
F->args()) {
4198 printArgument(&Arg,
Attrs.getParamAttrs(Arg.getArgNo()));
4203 if (FT->isVarArg()) {
4204 if (FT->getNumParams()) Out <<
", ";
4215 bool ForcePrintAddressSpace =
4216 !
Mod ||
Mod->getDataLayout().getProgramAddressSpace() != 0;
4218 "", ForcePrintAddressSpace);
4219 if (
Attrs.hasFnAttrs())
4220 Out <<
" #" <<
Machine.getAttributeGroupSlot(
Attrs.getFnAttrs());
4221 if (
F->hasSection()) {
4222 Out <<
" section \"";
4226 if (
F->hasPartition()) {
4227 Out <<
" partition \"";
4232 if (MaybeAlign
A =
F->getAlign())
4233 Out <<
" align " <<
A->value();
4234 if (MaybeAlign
A =
F->getPreferredAlignment())
4235 Out <<
" prefalign(" <<
A->value() <<
')';
4237 Out <<
" gc \"" <<
F->getGC() <<
'"';
4238 if (
F->hasPrefixData()) {
4240 writeOperand(
F->getPrefixData(),
true);
4242 if (
F->hasPrologueData()) {
4243 Out <<
" prologue ";
4244 writeOperand(
F->getPrologueData(),
true);
4246 if (
F->hasPersonalityFn()) {
4247 Out <<
" personality ";
4248 writeOperand(
F->getPersonalityFn(),
true);
4252 if (
auto *MDProf =
F->getMetadata(LLVMContext::MD_prof)) {
4254 MDProf->print(Out, TheModule,
true);
4258 if (
F->isDeclaration()) {
4262 F->getAllMetadata(MDs);
4263 printMetadataAttachments(MDs,
" ");
4267 for (
const BasicBlock &BB : *
F)
4281void AssemblyWriter::printArgument(
const Argument *Arg, AttributeSet Attrs) {
4283 TypePrinter.print(Arg->
getType(), Out);
4286 if (
Attrs.hasAttributes()) {
4288 writeAttributeSet(Attrs);
4297 assert(Slot != -1 &&
"expect argument in function here");
4298 Out <<
" %" <<
Slot;
4309 }
else if (!IsEntryBlock) {
4311 int Slot =
Machine.getLocalSlot(BB);
4318 if (!IsEntryBlock) {
4323 Out <<
" No predecessors!";
4329 writeOperand(Pred,
false);
4340 for (
const DbgRecord &DR :
I.getDbgRecordRange())
4341 printDbgRecordLine(DR);
4342 printInstructionLine(
I);
4349void AssemblyWriter::printInstructionLine(
const Instruction &
I) {
4350 printInstruction(
I);
4356void AssemblyWriter::printGCRelocateComment(
const GCRelocateInst &Relocate) {
4366void AssemblyWriter::printInfoComment(
const Value &V,
bool isMaterializable) {
4368 printGCRelocateComment(*Relocate);
4370 if (AnnotationWriter && !isMaterializable)
4375 if (
I->getDebugLoc()) {
4377 I->getDebugLoc().print(Out);
4383 if (
auto *MD =
I->getMetadata(LLVMContext::MD_prof)) {
4385 MD->print(Out, TheModule,
true);
4396 if (Operand ==
nullptr) {
4397 Out <<
" <cannot get addrspace!>";
4407 bool ForcePrintAddrSpace =
4408 !
Mod ||
Mod->getDataLayout().getProgramAddressSpace() != 0;
4410 ForcePrintAddrSpace);
4414void AssemblyWriter::printInstruction(
const Instruction &
I) {
4424 }
else if (!
I.getType()->isVoidTy()) {
4426 int SlotNum =
Machine.getLocalSlot(&
I);
4428 Out <<
"<badref> = ";
4430 Out <<
'%' << SlotNum <<
" = ";
4434 if (CI->isMustTailCall())
4436 else if (CI->isTailCall())
4438 else if (CI->isNoTailCall())
4443 Out <<
I.getOpcodeName();
4465 Out <<
' ' << CI->getPredicate();
4469 if (RMWI->isElementwise())
4470 Out <<
" elementwise";
4475 const Value *Operand =
I.getNumOperands() ?
I.getOperand(0) :
nullptr;
4480 writeOperand(BI->getCondition(),
true);
4482 writeOperand(BI->getSuccessor(0),
true);
4484 writeOperand(BI->getSuccessor(1),
true);
4489 writeOperand(
SI.getCondition(),
true);
4491 writeOperand(
SI.getDefaultDest(),
true);
4493 for (
auto Case :
SI.cases()) {
4495 writeOperand(Case.getCaseValue(),
true);
4497 writeOperand(Case.getCaseSuccessor(),
true);
4503 writeOperand(Operand,
true);
4507 for (
unsigned i = 1, e =
I.getNumOperands(); i != e; ++i) {
4509 writeOperand(
I.getOperand(i),
true);
4514 TypePrinter.print(
I.getType(), Out);
4518 for (
const auto &[V,
Block] :
4519 zip_equal(PN->incoming_values(), PN->blocks())) {
4521 writeOperand(V,
false);
4523 writeOperand(
Block,
false);
4528 writeOperand(
I.getOperand(0),
true);
4533 writeOperand(
I.getOperand(0),
true); Out <<
", ";
4534 writeOperand(
I.getOperand(1),
true);
4539 TypePrinter.print(
I.getType(), Out);
4540 if (LPI->isCleanup() || LPI->getNumClauses() != 0)
4543 if (LPI->isCleanup())
4546 for (
unsigned i = 0, e = LPI->getNumClauses(); i != e; ++i) {
4547 if (i != 0 || LPI->isCleanup()) Out <<
"\n";
4548 if (LPI->isCatch(i))
4553 writeOperand(LPI->getClause(i),
true);
4557 writeOperand(CatchSwitch->getParentPad(),
false);
4560 for (
const BasicBlock *PadBB : CatchSwitch->handlers()) {
4562 writeOperand(PadBB,
true);
4565 if (
const BasicBlock *UnwindDest = CatchSwitch->getUnwindDest())
4566 writeOperand(UnwindDest,
true);
4571 writeOperand(FPI->getParentPad(),
false);
4574 for (
const Value *
Op : FPI->arg_operands()) {
4576 writeOperand(
Op,
true);
4583 writeOperand(CRI->getOperand(0),
false);
4586 writeOperand(CRI->getOperand(1),
true);
4589 writeOperand(CRI->getOperand(0),
false);
4592 if (CRI->hasUnwindDest())
4593 writeOperand(CRI->getOperand(1),
true);
4598 if (CI->getCallingConv() != CallingConv::C) {
4603 Operand = CI->getCalledOperand();
4604 FunctionType *FTy = CI->getFunctionType();
4605 Type *RetTy = FTy->getReturnType();
4606 const AttributeList &PAL = CI->getAttributes();
4608 if (PAL.hasRetAttrs())
4609 Out <<
' ' << PAL.getAsString(AttributeList::ReturnIndex);
4618 TypePrinter.print(FTy->isVarArg() ? FTy : RetTy, Out);
4620 writeOperand(Operand,
false);
4622 bool HasPrettyPrintedArgs =
4627 Function *CalledFunc = CI->getCalledFunction();
4628 auto PrintArgComment = [&](
unsigned ArgNo) {
4632 std::string ArgComment;
4633 raw_string_ostream ArgCommentStream(ArgComment);
4636 if (ArgComment.empty())
4638 Out <<
"/* " << ArgComment <<
" */ ";
4640 if (HasPrettyPrintedArgs) {
4641 for (
unsigned ArgNo = 0, NumArgs = CI->arg_size(); ArgNo < NumArgs;
4644 PrintArgComment(ArgNo);
4645 writeParamOperand(CI->getArgOperand(ArgNo), PAL.getParamAttrs(ArgNo));
4648 for (
unsigned ArgNo = 0, NumArgs = CI->arg_size(); ArgNo < NumArgs;
4651 writeParamOperand(CI->getArgOperand(ArgNo), PAL.getParamAttrs(ArgNo));
4656 if (CI->isMustTailCall() && CI->getParent() &&
4657 CI->getParent()->getParent() &&
4658 CI->getParent()->getParent()->isVarArg()) {
4659 if (CI->arg_size() > 0)
4665 if (PAL.hasFnAttrs())
4666 Out <<
" #" <<
Machine.getAttributeGroupSlot(PAL.getFnAttrs());
4668 writeOperandBundles(CI);
4670 Operand =
II->getCalledOperand();
4671 FunctionType *FTy =
II->getFunctionType();
4672 Type *RetTy = FTy->getReturnType();
4673 const AttributeList &PAL =
II->getAttributes();
4676 if (
II->getCallingConv() != CallingConv::C) {
4681 if (PAL.hasRetAttrs())
4682 Out <<
' ' << PAL.getAsString(AttributeList::ReturnIndex);
4692 TypePrinter.print(FTy->isVarArg() ? FTy : RetTy, Out);
4694 writeOperand(Operand,
false);
4697 for (
unsigned op = 0, Eop =
II->arg_size();
op < Eop; ++
op) {
4699 writeParamOperand(
II->getArgOperand(
op), PAL.getParamAttrs(
op));
4703 if (PAL.hasFnAttrs())
4704 Out <<
" #" <<
Machine.getAttributeGroupSlot(PAL.getFnAttrs());
4706 writeOperandBundles(
II);
4709 writeOperand(
II->getNormalDest(),
true);
4711 writeOperand(
II->getUnwindDest(),
true);
4713 Operand = CBI->getCalledOperand();
4714 FunctionType *FTy = CBI->getFunctionType();
4715 Type *RetTy = FTy->getReturnType();
4716 const AttributeList &PAL = CBI->getAttributes();
4719 if (CBI->getCallingConv() != CallingConv::C) {
4724 if (PAL.hasRetAttrs())
4725 Out <<
' ' << PAL.getAsString(AttributeList::ReturnIndex);
4732 TypePrinter.print(FTy->isVarArg() ? FTy : RetTy, Out);
4734 writeOperand(Operand,
false);
4736 ListSeparator ArgLS;
4737 for (
unsigned op = 0, Eop = CBI->arg_size();
op < Eop; ++
op) {
4739 writeParamOperand(CBI->getArgOperand(
op), PAL.getParamAttrs(
op));
4743 if (PAL.hasFnAttrs())
4744 Out <<
" #" <<
Machine.getAttributeGroupSlot(PAL.getFnAttrs());
4746 writeOperandBundles(CBI);
4749 writeOperand(CBI->getDefaultDest(),
true);
4751 ListSeparator DestLS;
4752 for (
const BasicBlock *Dest : CBI->getIndirectDests()) {
4754 writeOperand(Dest,
true);
4759 if (AI->isUsedWithInAlloca())
4761 if (AI->isSwiftError())
4762 Out <<
"swifterror ";
4763 TypePrinter.print(AI->getAllocatedType(), Out);
4769 if (!AI->getArraySize() || AI->isArrayAllocation() ||
4770 !AI->getArraySize()->getType()->isIntegerTy(32)) {
4772 writeOperand(AI->getArraySize(),
true);
4774 if (MaybeAlign
A = AI->getAlign()) {
4775 Out <<
", align " <<
A->value();
4783 writeOperand(Operand,
true);
4786 TypePrinter.print(
I.getType(), Out);
4790 writeOperand(Operand,
true);
4793 TypePrinter.print(
I.getType(), Out);
4794 }
else if (Operand) {
4797 TypePrinter.print(
GEP->getSourceElementType(), Out);
4801 TypePrinter.print(LI->getType(), Out);
4808 bool PrintAllTypes =
false;
4816 PrintAllTypes =
true;
4818 for (
unsigned i = 1,
E =
I.getNumOperands(); i !=
E; ++i) {
4819 Operand =
I.getOperand(i);
4822 if (Operand && Operand->
getType() != TheType) {
4823 PrintAllTypes =
true;
4829 if (!PrintAllTypes) {
4831 TypePrinter.print(TheType, Out);
4836 for (
const Value *
Op :
I.operands()) {
4838 writeOperand(
Op, PrintAllTypes);
4845 writeAtomic(LI->getContext(), LI->getOrdering(), LI->getSyncScopeID());
4846 if (MaybeAlign
A = LI->getAlign())
4847 Out <<
", align " <<
A->value();
4850 writeAtomic(
SI->getContext(),
SI->getOrdering(),
SI->getSyncScopeID());
4851 if (MaybeAlign
A =
SI->getAlign())
4852 Out <<
", align " <<
A->value();
4854 writeAtomicCmpXchg(CXI->getContext(), CXI->getSuccessOrdering(),
4855 CXI->getFailureOrdering(), CXI->getSyncScopeID());
4856 Out <<
", align " << CXI->getAlign().value();
4858 writeAtomic(RMWI->getContext(), RMWI->getOrdering(),
4859 RMWI->getSyncScopeID());
4860 Out <<
", align " << RMWI->getAlign().value();
4862 writeAtomic(FI->getContext(), FI->getOrdering(), FI->getSyncScopeID());
4870 printMetadataAttachments(InstMD,
", ");
4873 printInfoComment(
I);
4876void AssemblyWriter::printDbgMarker(
const DbgMarker &Marker) {
4880 printDbgRecord(DPR);
4884 Out <<
" DbgMarker -> { ";
4889void AssemblyWriter::printDbgRecord(
const DbgRecord &DR) {
4891 printDbgVariableRecord(*DVR);
4893 printDbgLabelRecord(*DLR);
4898void AssemblyWriter::printDbgVariableRecord(
const DbgVariableRecord &DVR) {
4902 case DbgVariableRecord::LocationType::Value:
4905 case DbgVariableRecord::LocationType::Declare:
4908 case DbgVariableRecord::LocationType::DeclareValue:
4909 Out <<
"declare_value";
4911 case DbgVariableRecord::LocationType::Assign:
4916 "Tried to print a DbgVariableRecord with an invalid LocationType!");
4947void AssemblyWriter::printDbgRecordLine(
const DbgRecord &DR) {
4954void AssemblyWriter::printDbgLabelRecord(
const DbgLabelRecord &Label) {
4956 Out <<
"#dbg_label(";
4963void AssemblyWriter::printMetadataAttachments(
4964 const SmallVectorImpl<std::pair<unsigned, MDNode *>> &MDs,
4965 StringRef Separator) {
4969 if (MDNames.empty())
4970 MDs[0].second->getContext().getMDKindNames(MDNames);
4973 for (
const auto &
I : MDs) {
4974 unsigned Kind =
I.first;
4976 if (Kind < MDNames.size()) {
4980 Out <<
"!<unknown kind #" <<
Kind <<
">";
4986void AssemblyWriter::writeMDNode(
unsigned Slot,
const MDNode *Node) {
4987 if (AnnotationWriter)
4990 Out <<
'!' <<
Slot <<
" = ";
4991 printMDNodeBody(Node);
4995void AssemblyWriter::writeAllMDNodes() {
5001 for (
unsigned i = 0, e = Nodes.
size(); i != e; ++i) {
5002 writeMDNode(i, Nodes[i]);
5006void AssemblyWriter::printMDNodeBody(
const MDNode *Node) {
5011void AssemblyWriter::writeAttribute(
const Attribute &Attr,
bool InAttrGroup) {
5017 Out << Attribute::getNameFromAttrKind(Attr.
getKindAsEnum());
5020 TypePrinter.print(Ty, Out);
5025void AssemblyWriter::writeAttributeSet(
const AttributeSet &AttrSet,
5027 ListSeparator
LS(
" ");
5028 for (
const auto &Attr : AttrSet) {
5030 writeAttribute(Attr, InAttrGroup);
5034void AssemblyWriter::writeAllAttributeGroups() {
5035 std::vector<std::pair<AttributeSet, unsigned>> asVec;
5036 asVec.resize(
Machine.as_size());
5039 asVec[
I.second] =
I;
5041 for (
const auto &
I : asVec)
5042 Out <<
"attributes #" <<
I.second <<
" = { "
5043 <<
I.first.getAsString(
true) <<
" }\n";
5046void AssemblyWriter::printUseListOrder(
const Value *V,
5047 ArrayRef<unsigned> Shuffle) {
5051 Out <<
"uselistorder ";
5052 writeOperand(V,
true);
5054 assert(Shuffle.
size() >= 2 &&
"Shuffle too small");
5058void AssemblyWriter::printUseLists(
const Function *
F) {
5059 auto It = UseListOrders.find(
F);
5060 if (It == UseListOrders.end())
5063 Out <<
"\n; uselistorder directives\n";
5064 for (
const auto &Pair : It->second)
5065 printUseListOrder(Pair.first, Pair.second);
5073 bool ShouldPreserveUseListOrder,
bool IsForDebug)
const {
5076 AssemblyWriter W(OS, SlotTable, this->
getParent(), AAW, IsForDebug,
5077 ShouldPreserveUseListOrder);
5078 W.printFunction(
this);
5082 bool ShouldPreserveUseListOrder,
5083 bool IsForDebug)
const {
5086 AssemblyWriter W(OS, SlotTable, this->
getModule(), AAW,
5088 ShouldPreserveUseListOrder);
5089 W.printBasicBlock(
this);
5093 bool ShouldPreserveUseListOrder,
bool IsForDebug)
const {
5096 AssemblyWriter W(OS, SlotTable,
this, AAW, IsForDebug,
5097 ShouldPreserveUseListOrder);
5098 W.printModule(
this);
5104 AssemblyWriter W(OS, SlotTable,
getParent(),
nullptr, IsForDebug);
5105 W.printNamedMDNode(
this);
5109 bool IsForDebug)
const {
5110 std::optional<SlotTracker> LocalST;
5116 SlotTable = &*LocalST;
5120 AssemblyWriter W(OS, *SlotTable,
getParent(),
nullptr, IsForDebug);
5121 W.printNamedMDNode(
this);
5126 ROS <<
" = comdat ";
5133 ROS <<
"exactmatch";
5139 ROS <<
"nodeduplicate";
5151 TP.print(
const_cast<Type*
>(
this), OS);
5160 TP.printStructBody(STy, OS);
5166 if (
Function *
F = CI->getCalledFunction())
5167 if (
F->isIntrinsic())
5168 for (
auto &
Op :
I.operands())
5178 print(ROS, MST, IsForDebug);
5184 print(ROS, MST, IsForDebug);
5188 bool IsForDebug)
const {
5196 AssemblyWriter W(OS, SlotTable,
getModuleFromDPI(
this),
nullptr, IsForDebug);
5197 W.printDbgMarker(*
this);
5203 print(ROS, MST, IsForDebug);
5207 bool IsForDebug)
const {
5213 ?
Marker->getParent()->getParent()
5217 AssemblyWriter W(OS, SlotTable,
getModuleFromDPI(
this),
nullptr, IsForDebug);
5218 W.printDbgVariableRecord(*
this);
5222 bool IsForDebug)
const {
5228 Marker->getParent() ?
Marker->getParent()->getParent() :
nullptr;
5232 AssemblyWriter W(OS, SlotTable,
getModuleFromDPI(
this),
nullptr, IsForDebug);
5233 W.printDbgLabelRecord(*
this);
5237 bool ShouldInitializeAllMetadata =
false;
5241 ShouldInitializeAllMetadata =
true;
5244 print(ROS, MST, IsForDebug);
5248 bool IsForDebug)
const {
5253 auto IncorporateFunction = [&](
const Function *
F) {
5259 IncorporateFunction(
I->getParent() ?
I->getParent()->getParent() :
nullptr);
5261 W.printInstruction(*
I);
5263 IncorporateFunction(BB->getParent());
5264 AssemblyWriter W(OS, SlotTable,
getModuleFromVal(BB),
nullptr, IsForDebug);
5265 W.printBasicBlock(BB);
5267 AssemblyWriter W(OS, SlotTable, GV->
getParent(),
nullptr, IsForDebug);
5281 TypePrinting TypePrinter;
5282 TypePrinter.print(
C->getType(), OS);
5284 AsmWriterContext WriterCtx(&TypePrinter, MST.
getMachine());
5300 AsmWriterContext WriterCtx(
nullptr,
Machine, M);
5309 TypePrinting TypePrinter(MST.
getModule());
5340 AsmWriterContext &WriterCtx) {
5353struct MDTreeAsmWriterContext :
public AsmWriterContext {
5356 using EntryTy = std::pair<unsigned, std::string>;
5360 SmallPtrSet<const Metadata *, 4> Visited;
5362 raw_ostream &MainOS;
5364 MDTreeAsmWriterContext(TypePrinting *TP, SlotTracker *ST,
const Module *M,
5365 raw_ostream &OS,
const Metadata *InitMD)
5366 : AsmWriterContext(TP,
ST,
M),
Level(0
U), Visited({InitMD}), MainOS(OS) {}
5368 void onWriteMetadataAsOperand(
const Metadata *MD)
override {
5369 if (!Visited.
insert(MD).second)
5373 raw_string_ostream
SS(Str);
5378 unsigned InsertIdx = Buffer.
size() - 1;
5381 Buffer[InsertIdx].second = std::move(
SS.str());
5385 ~MDTreeAsmWriterContext()
override {
5386 for (
const auto &Entry : Buffer) {
5388 unsigned NumIndent =
Entry.first * 2U;
5397 bool OnlyAsOperand,
bool PrintAsTree =
false) {
5400 TypePrinting TypePrinter(M);
5402 std::unique_ptr<AsmWriterContext> WriterCtx;
5403 if (PrintAsTree && !OnlyAsOperand)
5404 WriterCtx = std::make_unique<MDTreeAsmWriterContext>(
5408 std::make_unique<AsmWriterContext>(&TypePrinter, MST.
getMachine(), M);
5437 const Module *M,
bool )
const {
5456 AssemblyWriter W(OS, SlotTable,
this, IsForDebug);
5457 W.printModuleSummaryIndex();
5461 unsigned UB)
const {
5467 if (
I.second >= LB &&
I.second < UB)
5468 L.push_back(std::make_pair(
I.second,
I.first));
5471#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
amdgpu next use AMDGPU Next Use Analysis Printer
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)
static void writeDICompositeType(raw_ostream &Out, const DICompositeType *N, AsmWriterContext &WriterCtx)
static void writeDIFixedPointType(raw_ostream &Out, const DIFixedPointType *N, AsmWriterContext &WriterCtx)
static void printDSOLocation(const GlobalValue &GV, formatted_raw_ostream &Out)
static const char * getWholeProgDevirtResKindName(WholeProgramDevirtResolution::Kind K)
static void writeDISubrangeType(raw_ostream &Out, const DISubrangeType *N, AsmWriterContext &WriterCtx)
static void WriteFullHexAPInt(raw_ostream &Out, const APInt &Val)
static void writeAPFloatInternal(raw_ostream &Out, const APFloat &APF)
static void printMetadataImpl(raw_ostream &ROS, const Metadata &MD, ModuleSlotTracker &MST, const Module *M, bool OnlyAsOperand, bool PrintAsTree=false)
static void writeDIStringType(raw_ostream &Out, const DIStringType *N, AsmWriterContext &WriterCtx)
static std::string getLinkageNameWithSpace(GlobalValue::LinkageTypes LT)
static cl::opt< bool > PreserveAssemblyUseListOrder("preserve-ll-uselistorder", cl::Hidden, cl::init(false), cl::desc("Preserve use-list order when writing LLVM assembly."))
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 writeDIBasicType(raw_ostream &Out, const DIBasicType *N, AsmWriterContext &WriterCtx)
static StringRef getUnnamedAddrEncoding(GlobalVariable::UnnamedAddr UA)
static const char * getWholeProgDevirtResByArgKindName(WholeProgramDevirtResolution::ByArg::Kind K)
static void writeMDNodeBodyInternal(raw_ostream &Out, const MDNode *Node, AsmWriterContext &Ctx)
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 cl::opt< bool > PrintAddrspaceName("print-addrspace-name", cl::Hidden, cl::init(false), cl::desc("Print address space names"))
static void writeOptimizationInfo(raw_ostream &Out, const User *U)
static bool isReferencingMDNode(const Instruction &I)
#define CC_VLS_CASE(ABI_VLEN)
static void writeDILabel(raw_ostream &Out, const DILabel *N, AsmWriterContext &WriterCtx)
static void writeDIDerivedType(raw_ostream &Out, const DIDerivedType *N, AsmWriterContext &WriterCtx)
static void printMetadataIdentifier(StringRef Name, formatted_raw_ostream &Out)
static void printShuffleMask(raw_ostream &Out, Type *Ty, ArrayRef< int > Mask)
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 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 printCallingConv(unsigned cc, raw_ostream &Out)
static void printAddressSpace(const Module *M, unsigned AS, raw_ostream &OS, StringRef Prefix=" ", StringRef Suffix="", bool ForcePrint=false)
static cl::opt< bool > PrintInstDebugLocs("print-inst-debug-locs", cl::Hidden, cl::desc("Pretty print debug locations of instructions when dumping"))
static void printMetadataImplRec(raw_ostream &ROS, const Metadata &MD, AsmWriterContext &WriterCtx)
Recursive version of printMetadataImpl.
static SlotTracker * createSlotTracker(const Value *V)
static void writeDILocation(raw_ostream &Out, const DILocation *DL, AsmWriterContext &WriterCtx)
static void writeDINamespace(raw_ostream &Out, const DINamespace *N, AsmWriterContext &WriterCtx)
DenseMap< const Function *, MapVector< const Value *, std::vector< unsigned > > > UseListOrderMap
static void writeDICommonBlock(raw_ostream &Out, const DICommonBlock *N, AsmWriterContext &WriterCtx)
static UseListOrderMap predictUseListOrder(const Module *M)
static void printThreadLocalModel(GlobalVariable::ThreadLocalMode TLM, formatted_raw_ostream &Out)
static std::string getLinkageName(GlobalValue::LinkageTypes LT)
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 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 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 writeDILexicalBlockFile(raw_ostream &Out, const DILexicalBlockFile *N, AsmWriterContext &WriterCtx)
static void writeConstantInternal(raw_ostream &Out, const Constant *CV, AsmWriterContext &WriterCtx)
static void writeDIEnumerator(raw_ostream &Out, const DIEnumerator *N, AsmWriterContext &)
static void writeAsOperandInternal(raw_ostream &Out, const Value *V, AsmWriterContext &WriterCtx, bool PrintType=false)
static void printVisibility(GlobalValue::VisibilityTypes Vis, formatted_raw_ostream &Out)
static cl::opt< bool > PrintProfData("print-prof-data", cl::Hidden, cl::desc("Pretty print perf data (branch weights, etc) when dumping"))
static void writeMDTuple(raw_ostream &Out, const MDTuple *Node, AsmWriterContext &WriterCtx)
static void writeDIExpression(raw_ostream &Out, const DIExpression *N, AsmWriterContext &WriterCtx)
static cl::opt< bool > PrintInstAddrs("print-inst-addrs", cl::Hidden, cl::desc("Print addresses of instructions when dumping"))
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 void printDLLStorageClass(GlobalValue::DLLStorageClassTypes SCT, formatted_raw_ostream &Out)
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")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#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...
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...
GlobalValue::SanitizerMetadata SanitizerMetadata
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.
Module.h This file contains the declarations for the Module class.
This defines the Use class.
Machine Check Debug Module
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...
static bool processModule(Module &M, NVPTXTargetMachine &TM)
static bool processFunction(Function &F, NVPTXTargetMachine &TM)
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
uint64_t IntrinsicInst * II
Function const char TargetMachine * Machine
if(auto Err=PB.parsePassPipeline(MPM, Passes)) return wrap(std MPM run * Mod
static StringRef getName(Value *V)
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.
LocallyHashedType DenseMapInfo< LocallyHashedType >::Empty
static UseListOrderStack predictUseListOrder(const Module &M)
static const fltSemantics & PPCDoubleDouble()
void toString(SmallVectorImpl< char > &Str, unsigned FormatPrecision=0, unsigned FormatMaxPadding=3, bool TruncateZero=true) const
const fltSemantics & getSemantics() const
APInt bitcastToAPInt() const
APInt getNaNPayload() const
If the value is a NaN value, return an integer containing the payload of this value.
Class for arbitrary precision integers.
void clearBit(unsigned BitPosition)
Set a given bit to 0.
unsigned getActiveBits() const
Compute the number of active bits in the value.
LLVM_ABI APInt trunc(unsigned width) const
Truncate to new width.
void toStringUnsigned(SmallVectorImpl< char > &Str, unsigned Radix=10) const
Considers the APInt to be unsigned and converts it into a string in the radix given.
bool isZero() const
Determine if this value is zero, i.e. all bits are clear.
bool isSignMask() const
Check if the APInt's value is returned by getSignMask.
unsigned getBitWidth() const
Return the number of bits in the APInt.
APInt lshr(unsigned shiftAmt) const
Logical right-shift function.
Abstract interface of slot tracker storage.
virtual ~AbstractSlotTrackerStorage()
const GlobalValueSummary & getAliasee() const
This class represents an incoming formal argument to a Function.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
Get the array size.
virtual void emitMDNodeAnnot(const MDNode *, formatted_raw_ostream &)
emitMDNodeAnnot - This may be implemented to emit a string right before a metadata node is emitted.
virtual void emitBasicBlockStartAnnot(const BasicBlock *, formatted_raw_ostream &)
emitBasicBlockStartAnnot - This may be implemented to emit a string right after the basic block label...
virtual void emitBasicBlockEndAnnot(const BasicBlock *, formatted_raw_ostream &)
emitBasicBlockEndAnnot - This may be implemented to emit a string right after the basic block.
virtual void emitFunctionAnnot(const Function *, formatted_raw_ostream &)
emitFunctionAnnot - This may be implemented to emit a string right before the start of a function.
virtual void emitInstructionAnnot(const Instruction *, formatted_raw_ostream &)
emitInstructionAnnot - This may be implemented to emit a string right before an instruction is emitte...
virtual void printInfoComment(const Value &, formatted_raw_ostream &)
printInfoComment - This may be implemented to emit a comment to the right of an instruction or global...
virtual ~AssemblyAnnotationWriter()
static LLVM_ABI StringRef getOperationName(BinOp Op)
This class holds the attributes for a particular argument, parameter, function, or return value.
bool hasAttributes() const
Return true if attributes exists in this set.
LLVM_ABI std::string getAsString(bool InAttrGrp=false) const
The Attribute is converted to a string of equivalent mnemonic.
LLVM_ABI Attribute::AttrKind getKindAsEnum() const
Return the attribute's kind as an enum (Attribute::AttrKind).
LLVM_ABI bool isTypeAttribute() const
Return true if the attribute is a type attribute.
LLVM_ABI Type * getValueAsType() const
Return the attribute's value as a Type.
LLVM Basic Block Representation.
const Function * getParent() const
Return the enclosing method, or null if none.
LLVM_ABI 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.
LLVM_ABI bool isEntryBlock() const
Return true if this is the entry block of the containing function.
LLVM_ABI const Module * getModule() const
Return the module owning the function this basic block belongs to, or nullptr if the function does no...
OperandBundleUse getOperandBundleAt(unsigned Index) const
Return the operand bundle at a specific index.
unsigned getNumOperandBundles() const
Return the number of operand bundles associated with this User.
AttributeList getAttributes() const
Return the attributes for this call.
bool hasOperandBundles() const
Return true if this User has any operand bundles.
LLVM_ABI void print(raw_ostream &OS, bool IsForDebug=false) const
LLVM_ABI void dump() 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
static LLVM_ABI ConstantPointerNull * get(PointerType *T)
Static factory methods - Return objects of the specified value.
LLVM_ABI APInt getSignedMin() const
Return the smallest signed value contained in the ConstantRange.
LLVM_ABI APInt getSignedMax() const
Return the largest signed value contained in the ConstantRange.
This is an important base class in LLVM.
LLVM_ABI Constant * getSplatValue(bool AllowPoison=false) const
If all elements of the vector constant have the same value, return that value.
LLVM_ABI 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 LLVM_ABI const char * nameTableKindString(DebugNameTableKind PK)
static LLVM_ABI const char * emissionKindString(DebugEmissionKind EK)
A lightweight wrapper around an expression operand.
static LLVM_ABI const char * fixedPointKindString(FixedPointKind)
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 LLVM_ABI DIFlags splitFlags(DIFlags Flags, SmallVectorImpl< DIFlags > &SplitFlags)
Split up a flags bitfield.
static LLVM_ABI StringRef getFlagString(DIFlags Flag)
Wrapper structure that holds source language identity metadata that includes language name,...
uint32_t getVersion() const
Returns language version. Only valid for versioned language names.
uint16_t getDialect() const
bool hasVersionedName() const
uint16_t getName() const
Returns a versioned or unversioned language name.
String type, Fortran CHARACTER(n)
Subprogram description. Uses SubclassData1.
static LLVM_ABI DISPFlags splitFlags(DISPFlags Flags, SmallVectorImpl< DISPFlags > &SplitFlags)
Split up a flags bitfield for easier printing.
static LLVM_ABI StringRef getFlagString(DISPFlags Flag)
DISPFlags
Debug info subprogram flags.
Type array for a subprogram.
LLVM_ABI void print(raw_ostream &O, bool IsForDebug=false) const
Per-instruction record of debug-info.
LLVM_ABI void dump() const
Instruction * MarkedInstr
Link back to the Instruction that owns this marker.
LLVM_ABI void print(raw_ostream &O, bool IsForDebug=false) const
Implement operator<< on DbgMarker.
LLVM_ABI 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.
DebugLoc getDebugLoc() const
LLVM_ABI void dump() 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
LLVM_ABI 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
LLVM_ABI MDNode * getAsMDNode() const
Return this as a bar MDNode.
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT > iterator
Intrinsic::ID getIntrinsicID() const LLVM_READONLY
getIntrinsicID - This method returns the ID number of the specified function, or Intrinsic::not_intri...
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.
const Function & getFunction() const
const Argument * const_arg_iterator
LLVM_ABI Value * getBasePtr() const
LLVM_ABI 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.
LLVM_ABI 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.
SummaryKind
Sububclass discriminator (for dyn_cast<> et al.)
bool hasPartition() const
static LLVM_ABI GUID getGUIDAssumingExternalLinkage(StringRef GlobalName)
Return a 64-bit global unique ID constructed from the name of a global symbol.
LLVM_ABI const SanitizerMetadata & getSanitizerMetadata() const
bool hasExternalLinkage() const
VisibilityTypes getVisibility() const
bool isImplicitDSOLocal() const
LinkageTypes getLinkage() const
uint64_t GUID
Declare a type to represent a global unique identifier for a global value.
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
LLVM_ABI 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.
LLVM_ABI 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
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.
MaybeAlign getAlign() const
Returns the alignment of the given variable.
bool isConstant() const
If the value is a global constant, its value is immutable throughout the runtime execution of the pro...
A helper class to return the specified delimiter string after the first invocation of operator String...
LLVM_ABI void printTree(raw_ostream &OS, const Module *M=nullptr) const
Print in tree shape.
LLVM_ABI 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.
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.
std::vector< std::pair< unsigned, const MDNode * > > MachineMDNodeListType
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...
uint64_t getBlockCount() const
const TypeIdSummaryMapTy & typeIds() const
ValueInfo getValueInfo(const GlobalValueSummaryMapTy::value_type &R) const
Return a ValueInfo for the index value_type (convenient when iterating index).
static constexpr const char * getRegularLTOModuleName()
const auto & typeIdCompatibleVtableMap() const
const StringMap< ModuleHash > & modulePaths() const
Table of modules, containing module hash and id.
LLVM_ABI void dump() const
Dump to stderr (for debugging).
GlobalValueSummaryMapTy::SortedEntriesRange sortedGlobalValueSummariesRange() const
uint64_t getStackIdAtIndex(unsigned Index) const
LLVM_ABI void print(raw_ostream &OS, bool IsForDebug=false) const
Print to an output stream.
LLVM_ABI uint64_t getFlags() const
A Module instance is used to store all the information related to an LLVM module.
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).
LLVM_ABI void dump() const
LLVM_ABI StringRef getName() const
LLVM_ABI void print(raw_ostream &ROS, bool IsForDebug=false) const
iterator_range< op_iterator > operands()
unsigned getAddressSpace() const
Return the address space of the Pointer type.
This class provides computation of slot numbers for LLVM Assembly writing.
DenseMap< const Value *, unsigned > ValueMap
ValueMap - A mapping of Values to slot numbers.
int getMetadataSlot(const MDNode *N) override
getMetadataSlot - Get the slot number of a MDNode.
~SlotTracker() override=default
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
DenseMap< GlobalValue::GUID, unsigned >::iterator guid_iterator
GUID map iterators.
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)>)
DenseMap< const MDNode *, unsigned >::iterator mdn_iterator
MDNode map iterators.
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.
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
reference emplace_back(ArgTypes &&... Args)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringMap - This is an unconventional map that is specialized for handling keys that are "strings",...
Represent a constant reference to a string, i.e.
constexpr bool empty() const
Check if the string is empty.
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.
LLVM_ABI StringRef getName() const
Return the name for this struct type if it has an identity.
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.
LLVM_ABI void run(const Module &M, bool onlyNamed)
The instances of the Type class are immutable: once they are created, they are never changed.
LLVM_ABI unsigned getPointerAddressSpace() const
Get the address space of this pointer or pointer vector type.
LLVM_ABI StringRef getTargetExtName() const
Type(LLVMContext &C, TypeID tid)
LLVM_ABI void dump() const
LLVM_ABI void print(raw_ostream &O, bool IsForDebug=false, bool NoDetails=false) const
Print the current type.
LLVM_ABI unsigned getByteBitWidth() const
TypeID getTypeID() const
Return the type id for the type.
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.
LLVM_ABI void print(raw_ostream &O, bool IsForDebug=false) const
Implement operator<< on Value.
LLVM_ABI void getAllMetadata(SmallVectorImpl< std::pair< unsigned, MDNode * > > &MDs) const
Appends all metadata attached to this value to MDs, sorting by KindID.
iterator_range< user_iterator > users()
LLVM_ABI 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.
iterator_range< use_iterator > uses()
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
LLVM_ABI 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.
LLVM_ABI StringRef LanguageDialectString(unsigned LanguageDialect)
LLVM_ABI StringRef SourceLanguageNameString(SourceLanguageName Lang)
LLVM_ABI StringRef EnumKindString(unsigned EnumKind)
LLVM_ABI StringRef LanguageString(unsigned Language)
LLVM_ABI StringRef AttributeEncodingString(unsigned Encoding)
LLVM_ABI StringRef ConventionString(unsigned Convention)
LLVM_ABI StringRef MacinfoString(unsigned Encoding)
LLVM_ABI StringRef OperationEncodingString(unsigned Encoding)
LLVM_ABI 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).
@ CHERIoT_CompartmentCall
Calling convention used for CHERIoT when crossing a protection boundary.
@ 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).
@ CHERIoT_CompartmentCallee
Calling convention used for the callee of CHERIoT_CompartmentCall.
@ 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.
@ CHERIoT_LibraryCall
Calling convention used for CHERIoT for cross-library calls to a stateless compartment.
@ 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.
LLVM_ABI void printImmArg(ID IID, unsigned ArgIdx, raw_ostream &OS, const Constant *ImmArgVal)
Print the argument info for the arguments with ArgInfo.
LLVM_ABI bool hasPrettyPrintedArgs(ID id)
Returns true if the intrinsic has pretty printed immediate arguments.
constexpr bool isAtomic(const T &...O)
@ System
Synchronized with respect to all concurrently executing threads.
initializer< Ty > init(const Ty &Val)
@ DW_OP_LLVM_convert
Only used in LLVM metadata.
Context & getContext() const
This is an optimization pass for GlobalISel generic memory operations.
void dump(const SparseBitVector< ElementSize > &LHS, raw_ostream &out)
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
Printable print(const GCNRegPressure &RP, const GCNSubtarget *ST=nullptr, unsigned DynamicVGPRBlockSize=0)
detail::zippy< detail::zip_first, T, U, Args... > zip_equal(T &&t, U &&u, Args &&...args)
zip iterator that assumes that all iteratees have the same length.
InterleavedRange< Range > interleaved(const Range &R, StringRef Separator=", ", StringRef Prefix="", StringRef Suffix="")
Output range R as a sequence of interleaved elements.
const char * getHotnessName(CalleeInfo::HotnessType HT)
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
auto dyn_cast_if_present(const Y &Val)
dyn_cast_if_present<X> - Functionally identical to dyn_cast, except that a null (or none in the case ...
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
LLVM_ABI void printEscapedString(StringRef Name, raw_ostream &Out)
Print each character of the specified string, escaping it if it is not printable or if it is an escap...
constexpr auto equal_to(T &&Arg)
Functor variant of std::equal_to that can be used as a UnaryPredicate in functional algorithms like a...
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr Value
const char * toIRString(AtomicOrdering ao)
String used by LLVM IR to represent atomic ordering.
auto dyn_cast_or_null(const Y &Val)
void sort(IteratorTy Start, IteratorTy End)
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
char hexdigit(unsigned X, bool LowerCase=false)
hexdigit - Return the hexadecimal character for the given number X (which should be less than 16).
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...
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
constexpr int PoisonMaskElem
AtomicOrdering
Atomic ordering for LLVM's memory model.
@ Ref
The access may reference the value stored in memory.
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr Count
DWARFExpression::Operation Op
ArrayRef(const T &OneElt) -> ArrayRef< T >
std::string toString(const APInt &I, unsigned Radix, bool Signed, bool formatAsCLiteral=false, bool UpperCase=true, bool InsertSeparators=false)
LLVM_ABI Printable printBasicBlock(const BasicBlock *BB)
Print BasicBlock BB as an operand or print "<nullptr>" if BB is a nullptr.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
auto predecessors(const MachineBasicBlock *BB)
bool pred_empty(const BasicBlock *BB)
std::vector< TypeIdOffsetVtableInfo > TypeIdCompatibleVtableInfo
List of vtable definitions decorated by a particular type identifier, and their corresponding offsets...
static auto filterDbgVars(iterator_range< simple_ilist< DbgRecord >::iterator > R)
Filter the DbgRecord range to DbgVariableRecord types only and downcast.
LLVM_ABI void printLLVMNameWithoutPrefix(raw_ostream &OS, StringRef Name)
Print out a name of an LLVM value without any prefixes.
@ Default
The result value is uniform if and only if all operands are uniform.
A single checksum, represented by a Kind and a Value (a string).
T Value
The string value of the checksum.
StringRef getKindAsString() const
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....
unsigned NoRenameOnPromotion
This field is written by the ThinLTO prelink stage to decide whether a particular static global value...
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 ...
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
@ 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:...