31#include "llvm/Config/llvm-config.h"
95 cl::desc(
"Print addresses of instructions when dumping"));
99 cl::desc(
"Pretty print debug locations of instructions when dumping"));
103 cl::desc(
"Pretty print perf data (branch weights, etc) when dumping"));
122 return VAM->getValue();
135 for (
const Value *
Op :
C->operands())
142 unsigned ID = OM.size() + 1;
149 auto orderConstantValue = [&OM](
const Value *V) {
154 auto OrderConstantFromMetadata = [&](
Metadata *MD) {
156 orderConstantValue(VAM->getValue());
158 for (
const auto *VAM : AL->getArgs())
159 orderConstantValue(VAM->getValue());
164 if (
G.hasInitializer())
180 for (
const Use &U :
F.operands())
186 if (
F.isDeclaration())
199 OrderConstantFromMetadata(DVR.getRawLocation());
200 if (DVR.isDbgAssign())
201 OrderConstantFromMetadata(DVR.getRawAddress());
204 for (
const Value *
Op :
I.operands()) {
217static std::vector<unsigned>
220 using Entry = std::pair<const Use *, unsigned>;
224 if (OM.lookup(U.getUser()))
225 List.
push_back(std::make_pair(&U, List.size()));
236 ID = OM.lookup(BA->getBasicBlock());
237 llvm::sort(List, [&](
const Entry &L,
const Entry &R) {
238 const Use *LU = L.first;
239 const Use *RU = R.first;
243 auto LID = OM.lookup(LU->getUser());
244 auto RID = OM.lookup(RU->getUser());
264 return LU->getOperandNo() < RU->getOperandNo();
265 return LU->getOperandNo() > RU->getOperandNo();
273 std::vector<unsigned> Shuffle(List.size());
274 for (
size_t I = 0,
E = List.size();
I !=
E; ++
I)
275 Shuffle[
I] = List[
I].second;
282 for (
const auto &Pair : OM) {
283 const Value *V = Pair.first;
284 if (V->use_empty() || std::next(V->use_begin()) == V->use_end())
287 std::vector<unsigned> Shuffle =
294 F =
I->getFunction();
299 ULOM[
F][V] = std::move(Shuffle);
306 return MA->getParent() ? MA->getParent()->getParent() :
nullptr;
309 return BB->getParent() ? BB->getParent()->getParent() :
nullptr;
312 const Function *M =
I->getParent() ?
I->getParent()->getParent() :
nullptr;
313 return M ? M->getParent() :
nullptr;
317 return GV->getParent();
342 default: Out <<
"cc" << cc;
break;
365 Out <<
"aarch64_sve_vector_pcs";
368 Out <<
"aarch64_sme_preservemost_from_x0";
371 Out <<
"aarch64_sme_preservemost_from_x1";
374 Out <<
"aarch64_sme_preservemost_from_x2";
402 Out <<
"amdgpu_cs_chain";
405 Out <<
"amdgpu_cs_chain_preserve";
410 Out <<
"amdgpu_gfx_whole_wave";
414 Out <<
"riscv_vector_cc";
416#define CC_VLS_CASE(ABI_VLEN) \
417 case CallingConv::RISCV_VLSCall_##ABI_VLEN: \
418 Out << "riscv_vls_cc(" #ABI_VLEN ")"; \
434 Out <<
"cheriot_compartmentcallcc";
437 Out <<
"cheriot_compartmentcalleecc";
440 Out <<
"cheriot_librarycallcc";
454 assert(!Name.empty() &&
"Cannot get empty name!");
457 bool NeedsQuotes = isdigit(
static_cast<unsigned char>(Name[0]));
459 for (
unsigned char C : Name) {
464 if (!isalnum(
C) &&
C !=
'-' &&
C !=
'.' &&
C !=
'_') {
518 Out << Mask.size() <<
" x i32> ";
519 bool FirstElt =
true;
520 if (
all_of(Mask, [](
int Elt) {
return Elt == 0; })) {
521 Out <<
"zeroinitializer";
526 for (
int Elt : Mask) {
545 TypePrinting(
const Module *M =
nullptr) : DeferredM(
M) {}
547 TypePrinting(
const TypePrinting &) =
delete;
548 TypePrinting &operator=(
const TypePrinting &) =
delete;
551 TypeFinder &getNamedTypes();
554 std::vector<StructType *> &getNumberedTypes();
560 void printStructBody(StructType *Ty, raw_ostream &OS);
563 void incorporateTypes();
568 TypeFinder NamedTypes;
571 DenseMap<StructType *, unsigned> Type2Number;
573 std::vector<StructType *> NumberedTypes;
583std::vector<StructType *> &TypePrinting::getNumberedTypes() {
589 if (NumberedTypes.size() == Type2Number.size())
590 return NumberedTypes;
592 NumberedTypes.resize(Type2Number.size());
593 for (
const auto &
P : Type2Number) {
594 assert(
P.second < NumberedTypes.size() &&
"Didn't get a dense numbering?");
595 assert(!NumberedTypes[
P.second] &&
"Didn't get a unique numbering?");
596 NumberedTypes[
P.second] =
P.first;
598 return NumberedTypes;
601bool TypePrinting::empty() {
603 return NamedTypes.
empty() && Type2Number.empty();
606void TypePrinting::incorporateTypes() {
610 NamedTypes.
run(*DeferredM,
false);
615 unsigned NextNumber = 0;
617 std::vector<StructType *>::iterator NextToUse = NamedTypes.
begin();
618 for (StructType *STy : NamedTypes) {
620 if (STy->isLiteral())
623 if (STy->getName().empty())
624 Type2Number[STy] = NextNumber++;
629 NamedTypes.erase(NextToUse, NamedTypes.end());
634void TypePrinting::print(
Type *Ty, raw_ostream &OS) {
636 case Type::VoidTyID: OS <<
"void";
return;
637 case Type::HalfTyID: OS <<
"half";
return;
638 case Type::BFloatTyID: OS <<
"bfloat";
return;
639 case Type::FloatTyID: OS <<
"float";
return;
640 case Type::DoubleTyID: OS <<
"double";
return;
641 case Type::X86_FP80TyID: OS <<
"x86_fp80";
return;
642 case Type::FP128TyID: OS <<
"fp128";
return;
643 case Type::PPC_FP128TyID: OS <<
"ppc_fp128";
return;
644 case Type::LabelTyID: OS <<
"label";
return;
645 case Type::MetadataTyID:
648 case Type::X86_AMXTyID: OS <<
"x86_amx";
return;
649 case Type::TokenTyID: OS <<
"token";
return;
650 case Type::IntegerTyID:
651 OS << 'i' << cast<IntegerType>(Ty)->getBitWidth();
654 case Type::FunctionTyID: {
656 print(FTy->getReturnType(), OS);
659 for (
Type *Ty : FTy->params()) {
668 case Type::StructTyID: {
672 return printStructBody(STy, OS);
678 const auto I = Type2Number.find(STy);
679 if (
I != Type2Number.end())
680 OS <<
'%' <<
I->second;
682 OS <<
"%\"type " << STy <<
'\"';
685 case Type::PointerTyID: {
692 case Type::ArrayTyID: {
694 OS <<
'[' << ATy->getNumElements() <<
" x ";
695 print(ATy->getElementType(), OS);
699 case Type::FixedVectorTyID:
700 case Type::ScalableVectorTyID: {
702 ElementCount
EC = PTy->getElementCount();
706 OS <<
EC.getKnownMinValue() <<
" x ";
707 print(PTy->getElementType(), OS);
711 case Type::TypedPointerTyID: {
717 case Type::TargetExtTyID:
724 Inner->print(OS,
false,
true);
727 OS <<
", " << IntParam;
734void TypePrinting::printStructBody(StructType *STy, raw_ostream &OS) {
778 const Function* TheFunction =
nullptr;
779 bool FunctionProcessed =
false;
780 bool ShouldInitializeAllMetadata;
785 ProcessFunctionHookFn;
800 unsigned mdnNext = 0;
808 unsigned ModulePathNext = 0;
812 unsigned GUIDNext = 0;
816 unsigned TypeIdNext = 0;
821 unsigned TypeIdCompatibleVtableNext = 0;
830 bool ShouldInitializeAllMetadata =
false);
838 bool ShouldInitializeAllMetadata =
false);
855 void createMetadataSlot(
const MDNode *
N)
override;
859 int getLocalSlot(
const Value *V);
861 int getMetadataSlot(
const MDNode *
N)
override;
866 int getTypeIdCompatibleVtableSlot(
StringRef Id);
872 FunctionProcessed =
false;
880 void purgeFunction();
887 unsigned mdn_size()
const {
return mdnMap.size(); }
895 unsigned as_size()
const {
return asMap.size(); }
911 void CreateMetadataSlot(
const MDNode *
N);
914 void CreateFunctionSlot(
const Value *V);
919 inline void CreateModulePathSlot(
StringRef Path);
922 void CreateTypeIdCompatibleVtableSlot(
StringRef Id);
926 void processModule();
934 void processGlobalObjectMetadata(
const GlobalObject &GO);
937 void processFunctionMetadata(
const Function &
F);
943 void processDbgRecordMetadata(
const DbgRecord &DVR);
950 : M(M), F(F), Machine(&Machine) {}
953 bool ShouldInitializeAllMetadata)
954 : ShouldCreateStorage(M),
955 ShouldInitializeAllMetadata(ShouldInitializeAllMetadata), M(M) {}
960 if (!ShouldCreateStorage)
963 ShouldCreateStorage =
false;
965 std::make_unique<SlotTracker>(M, ShouldInitializeAllMetadata);
966 Machine = MachineStorage.get();
967 if (ProcessModuleHookFn)
968 Machine->setProcessHook(ProcessModuleHookFn);
969 if (ProcessFunctionHookFn)
970 Machine->setProcessHook(ProcessFunctionHookFn);
983 Machine->purgeFunction();
984 Machine->incorporateFunction(&F);
989 assert(F &&
"No function incorporated");
990 return Machine->getLocalSlot(V);
996 ProcessModuleHookFn = Fn;
1002 ProcessFunctionHookFn = Fn;
1032#define ST_DEBUG(X) dbgs() << X
1040 : TheModule(M), ShouldInitializeAllMetadata(ShouldInitializeAllMetadata) {}
1045 : TheModule(
F ?
F->
getParent() : nullptr), TheFunction(
F),
1046 ShouldInitializeAllMetadata(ShouldInitializeAllMetadata) {}
1049 : TheModule(nullptr), ShouldInitializeAllMetadata(
false), TheIndex(Index) {}
1054 TheModule =
nullptr;
1057 if (TheFunction && !FunctionProcessed)
1064 int NumSlots = processIndex();
1071void SlotTracker::processModule() {
1072 ST_DEBUG(
"begin processModule!\n");
1077 CreateModuleSlot(&Var);
1078 processGlobalObjectMetadata(Var);
1079 auto Attrs = Var.getAttributes();
1080 if (Attrs.hasAttributes())
1081 CreateAttributeSetSlot(Attrs);
1086 CreateModuleSlot(&
A);
1089 for (
const GlobalIFunc &
I : TheModule->ifuncs()) {
1091 CreateModuleSlot(&
I);
1092 processGlobalObjectMetadata(
I);
1096 for (
const NamedMDNode &NMD : TheModule->named_metadata()) {
1097 for (
const MDNode *
N : NMD.operands())
1098 CreateMetadataSlot(
N);
1101 for (
const Function &
F : *TheModule) {
1104 CreateModuleSlot(&
F);
1106 if (ShouldInitializeAllMetadata)
1107 processFunctionMetadata(
F);
1111 AttributeSet FnAttrs =
F.getAttributes().getFnAttrs();
1113 CreateAttributeSetSlot(FnAttrs);
1116 if (ProcessModuleHookFn)
1117 ProcessModuleHookFn(
this, TheModule, ShouldInitializeAllMetadata);
1123void SlotTracker::processFunction() {
1124 ST_DEBUG(
"begin processFunction!\n");
1128 if (!ShouldInitializeAllMetadata)
1129 processFunctionMetadata(*TheFunction);
1133 AE = TheFunction->arg_end(); AI != AE; ++AI)
1135 CreateFunctionSlot(&*AI);
1137 ST_DEBUG(
"Inserting Instructions:\n");
1140 for (
auto &BB : *TheFunction) {
1142 CreateFunctionSlot(&BB);
1144 for (
auto &
I : BB) {
1145 if (!
I.getType()->isVoidTy() && !
I.hasName())
1146 CreateFunctionSlot(&
I);
1153 if (
Attrs.hasAttributes())
1154 CreateAttributeSetSlot(Attrs);
1159 if (ProcessFunctionHookFn)
1160 ProcessFunctionHookFn(
this, TheFunction, ShouldInitializeAllMetadata);
1162 FunctionProcessed =
true;
1164 ST_DEBUG(
"end processFunction!\n");
1168int SlotTracker::processIndex() {
1175 std::vector<StringRef> ModulePaths;
1176 for (
auto &[ModPath,
_] : TheIndex->modulePaths())
1177 ModulePaths.push_back(ModPath);
1179 for (
auto &ModPath : ModulePaths)
1180 CreateModulePathSlot(ModPath);
1183 GUIDNext = ModulePathNext;
1185 for (
auto &GlobalList : *TheIndex)
1186 CreateGUIDSlot(GlobalList.first);
1189 TypeIdCompatibleVtableNext = GUIDNext;
1190 for (
auto &TId : TheIndex->typeIdCompatibleVtableMap())
1191 CreateTypeIdCompatibleVtableSlot(TId.first);
1194 TypeIdNext = TypeIdCompatibleVtableNext;
1195 for (
const auto &TID : TheIndex->typeIds())
1196 CreateTypeIdSlot(TID.second.first);
1202void SlotTracker::processGlobalObjectMetadata(
const GlobalObject &GO) {
1205 for (
auto &MD : MDs)
1206 CreateMetadataSlot(MD.second);
1209void SlotTracker::processFunctionMetadata(
const Function &
F) {
1210 processGlobalObjectMetadata(
F);
1211 for (
auto &BB :
F) {
1212 for (
auto &
I : BB) {
1213 for (
const DbgRecord &DR :
I.getDbgRecordRange())
1214 processDbgRecordMetadata(DR);
1215 processInstructionMetadata(
I);
1220void SlotTracker::processDbgRecordMetadata(
const DbgRecord &DR) {
1231 CreateMetadataSlot(
Empty);
1232 if (DVR->getRawVariable())
1233 CreateMetadataSlot(DVR->getRawVariable());
1234 if (DVR->isDbgAssign()) {
1235 if (
auto *AssignID = DVR->getRawAssignID())
1238 CreateMetadataSlot(
Empty);
1241 CreateMetadataSlot(DLR->getRawLabel());
1249void SlotTracker::processInstructionMetadata(
const Instruction &
I) {
1252 if (Function *
F = CI->getCalledFunction())
1253 if (
F->isIntrinsic())
1254 for (
auto &
Op :
I.operands())
1257 CreateMetadataSlot(
N);
1261 I.getAllMetadata(MDs);
1262 for (
auto &MD : MDs)
1263 CreateMetadataSlot(MD.second);
1270 ST_DEBUG(
"begin purgeFunction!\n");
1272 TheFunction =
nullptr;
1273 FunctionProcessed =
false;
1284 return MI == mMap.end() ? -1 : (int)
MI->second;
1290 ProcessModuleHookFn = Fn;
1296 ProcessFunctionHookFn = Fn;
1309 return MI == mdnMap.end() ? -1 : (int)
MI->second;
1320 return FI == fMap.end() ? -1 : (int)FI->second;
1329 return AI == asMap.end() ? -1 : (int)AI->second;
1337 auto I = ModulePathMap.find(Path);
1338 return I == ModulePathMap.end() ? -1 : (int)
I->second;
1347 return I == GUIDMap.end() ? -1 : (int)
I->second;
1355 auto I = TypeIdMap.find(Id);
1356 return I == TypeIdMap.end() ? -1 : (int)
I->second;
1364 auto I = TypeIdCompatibleVtableMap.find(Id);
1365 return I == TypeIdCompatibleVtableMap.end() ? -1 : (int)
I->second;
1369void SlotTracker::CreateModuleSlot(
const GlobalValue *V) {
1370 assert(V &&
"Can't insert a null Value into SlotTracker!");
1371 assert(!V->getType()->isVoidTy() &&
"Doesn't need a slot!");
1372 assert(!V->hasName() &&
"Doesn't need a slot!");
1374 unsigned DestSlot = mNext++;
1377 ST_DEBUG(
" Inserting value [" << V->getType() <<
"] = " << V <<
" slot=" <<
1387void SlotTracker::CreateFunctionSlot(
const Value *V) {
1388 assert(!V->getType()->isVoidTy() && !V->hasName() &&
"Doesn't need a slot!");
1390 unsigned DestSlot = fNext++;
1394 ST_DEBUG(
" Inserting value [" << V->getType() <<
"] = " << V <<
" slot=" <<
1395 DestSlot <<
" [o]\n");
1399void SlotTracker::CreateMetadataSlot(
const MDNode *
N) {
1400 assert(
N &&
"Can't insert a null Value into SlotTracker!");
1406 unsigned DestSlot = mdnNext;
1407 if (!mdnMap.insert(std::make_pair(
N, DestSlot)).second)
1412 for (
unsigned i = 0, e =
N->getNumOperands(); i != e; ++i)
1414 CreateMetadataSlot(
Op);
1417void SlotTracker::CreateAttributeSetSlot(
AttributeSet AS) {
1420 if (asMap.try_emplace(AS, asNext).second)
1425void SlotTracker::CreateModulePathSlot(
StringRef Path) {
1426 ModulePathMap[
Path] = ModulePathNext++;
1431 GUIDMap[
GUID] = GUIDNext++;
1435void SlotTracker::CreateTypeIdSlot(
StringRef Id) {
1436 TypeIdMap[
Id] = TypeIdNext++;
1440void SlotTracker::CreateTypeIdCompatibleVtableSlot(
StringRef Id) {
1441 TypeIdCompatibleVtableMap[
Id] = TypeIdCompatibleVtableNext++;
1446struct AsmWriterContext {
1447 TypePrinting *TypePrinter =
nullptr;
1448 SlotTracker *
Machine =
nullptr;
1451 AsmWriterContext(TypePrinting *TP, SlotTracker *ST,
const Module *M =
nullptr)
1454 static AsmWriterContext &getEmpty() {
1455 static AsmWriterContext EmptyCtx(
nullptr,
nullptr);
1461 virtual void onWriteMetadataAsOperand(
const Metadata *) {}
1463 virtual ~AsmWriterContext() =
default;
1472 AsmWriterContext &WriterCtx);
1475 AsmWriterContext &WriterCtx,
1476 bool FromValue =
false);
1480 Out << FPO->getFastMathFlags();
1484 if (OBO->hasNoUnsignedWrap())
1486 if (OBO->hasNoSignedWrap())
1494 if (PDI->isDisjoint())
1497 if (
GEP->isInBounds())
1499 else if (
GEP->hasNoUnsignedSignedWrap())
1501 if (
GEP->hasNoUnsignedWrap())
1504 Out <<
" inrange(" <<
InRange->getLower() <<
", " <<
InRange->getUpper()
1508 if (NNI->hasNonNeg())
1511 if (TI->hasNoUnsignedWrap())
1513 if (TI->hasNoSignedWrap())
1516 if (ICmp->hasSameSign())
1532 bool isNaN = APF.
isNaN();
1534 if (!isInf && !isNaN) {
1543 ((StrVal[0] ==
'-' || StrVal[0] ==
'+') &&
isDigit(StrVal[1]))) &&
1544 "[-+]?[0-9] regex does not match!");
1556 static_assert(
sizeof(double) ==
sizeof(
uint64_t),
1557 "assuming that double is 64 bits!");
1615 AsmWriterContext &WriterCtx) {
1617 Type *Ty = CI->getType();
1619 if (Ty->isVectorTy()) {
1621 WriterCtx.TypePrinter->print(Ty->getScalarType(), Out);
1625 if (Ty->getScalarType()->isIntegerTy(1))
1626 Out << (CI->getZExtValue() ?
"true" :
"false");
1628 Out << CI->getValue();
1630 if (Ty->isVectorTy())
1637 Type *Ty = CFP->getType();
1639 if (Ty->isVectorTy()) {
1641 WriterCtx.TypePrinter->print(Ty->getScalarType(), Out);
1647 if (Ty->isVectorTy())
1654 Out <<
"zeroinitializer";
1659 Out <<
"blockaddress(";
1668 Out <<
"dso_local_equivalent ";
1683 unsigned NumOpsToWrite = 2;
1684 if (!CPA->getOperand(2)->isNullValue())
1686 if (!CPA->getOperand(3)->isNullValue())
1690 for (
unsigned i = 0, e = NumOpsToWrite; i != e; ++i) {
1692 WriterCtx.TypePrinter->print(CPA->getOperand(i)->getType(), Out);
1701 Type *ETy = CA->getType()->getElementType();
1703 WriterCtx.TypePrinter->print(ETy, Out);
1706 for (
unsigned i = 1, e = CA->getNumOperands(); i != e; ++i) {
1708 WriterCtx.TypePrinter->print(ETy, Out);
1719 if (CA->isString()) {
1726 Type *ETy = CA->getType()->getElementType();
1728 WriterCtx.TypePrinter->print(ETy, Out);
1731 for (
uint64_t i = 1, e = CA->getNumElements(); i != e; ++i) {
1733 WriterCtx.TypePrinter->print(ETy, Out);
1742 if (CS->getType()->isPacked())
1745 unsigned N = CS->getNumOperands();
1748 WriterCtx.TypePrinter->print(CS->getOperand(0)->getType(), Out);
1753 for (
unsigned i = 1; i <
N; i++) {
1755 WriterCtx.TypePrinter->print(CS->getOperand(i)->getType(), Out);
1764 if (CS->getType()->isPacked())
1771 Type *ETy = CVVTy->getElementType();
1781 WriterCtx.TypePrinter->print(ETy, Out);
1790 WriterCtx.TypePrinter->print(ETy, Out);
1793 for (
unsigned i = 1, e = CVVTy->getNumElements(); i != e; ++i) {
1795 WriterCtx.TypePrinter->print(ETy, Out);
1829 if (CE->getOpcode() == Instruction::ShuffleVector) {
1830 if (
auto *SplatVal = CE->getSplatValue()) {
1833 WriterCtx.TypePrinter->print(SplatVal->getType(), Out);
1842 Out << CE->getOpcodeName();
1847 WriterCtx.TypePrinter->print(
GEP->getSourceElementType(), Out);
1853 WriterCtx.TypePrinter->print((*OI)->getType(), Out);
1856 if (OI+1 != CE->op_end())
1862 WriterCtx.TypePrinter->print(CE->getType(), Out);
1865 if (CE->getOpcode() == Instruction::ShuffleVector)
1872 Out <<
"<placeholder or erroneous Constant>";
1876 AsmWriterContext &WriterCtx) {
1878 for (
unsigned mi = 0, me =
Node->getNumOperands(); mi != me; ++mi) {
1883 Value *V = MDV->getValue();
1884 WriterCtx.TypePrinter->print(V->getType(), Out);
1889 WriterCtx.onWriteMetadataAsOperand(MD);
1900struct FieldSeparator {
1904 FieldSeparator(
const char *Sep =
", ") : Sep(Sep) {}
1912 return OS <<
FS.Sep;
1915struct MDFieldPrinter {
1918 AsmWriterContext &WriterCtx;
1920 explicit MDFieldPrinter(raw_ostream &Out)
1921 : Out(Out), WriterCtx(AsmWriterContext::getEmpty()) {}
1922 MDFieldPrinter(raw_ostream &Out, AsmWriterContext &Ctx)
1923 : Out(Out), WriterCtx(Ctx) {}
1925 void printTag(
const DINode *
N);
1926 void printMacinfoType(
const DIMacroNode *
N);
1927 void printChecksum(
const DIFile::ChecksumInfo<StringRef> &
N);
1928 void printString(StringRef Name, StringRef
Value,
1929 bool ShouldSkipEmpty =
true);
1930 void printMetadata(StringRef Name,
const Metadata *MD,
1931 bool ShouldSkipNull =
true);
1932 void printMetadataOrInt(StringRef Name,
const Metadata *MD,
bool IsUnsigned,
1933 bool ShouldSkipZero =
true);
1934 template <
class IntTy>
1935 void printInt(StringRef Name, IntTy
Int,
bool ShouldSkipZero =
true);
1936 void printAPInt(StringRef Name,
const APInt &
Int,
bool IsUnsigned,
1937 bool ShouldSkipZero);
1938 void printBool(StringRef Name,
bool Value,
1939 std::optional<bool>
Default = std::nullopt);
1942 template <
class IntTy,
class Stringifier>
1943 void printDwarfEnum(StringRef Name, IntTy
Value, Stringifier
toString,
1944 bool ShouldSkipZero =
true);
1946 void printNameTableKind(StringRef Name,
1953void MDFieldPrinter::printTag(
const DINode *
N) {
1954 Out <<
FS <<
"tag: ";
1962void MDFieldPrinter::printMacinfoType(
const DIMacroNode *
N) {
1963 Out <<
FS <<
"type: ";
1968 Out <<
N->getMacinfoType();
1971void MDFieldPrinter::printChecksum(
1974 printString(
"checksum", Checksum.
Value,
false);
1978 bool ShouldSkipEmpty) {
1979 if (ShouldSkipEmpty &&
Value.empty())
1982 Out <<
FS <<
Name <<
": \"";
1988 AsmWriterContext &WriterCtx) {
1994 WriterCtx.onWriteMetadataAsOperand(MD);
1998 bool ShouldSkipNull) {
1999 if (ShouldSkipNull && !MD)
2002 Out <<
FS <<
Name <<
": ";
2007 bool IsUnsigned,
bool ShouldSkipZero) {
2014 printInt(Name, CV->getZExtValue(), ShouldSkipZero);
2016 printInt(Name, CV->getSExtValue(), ShouldSkipZero);
2018 printMetadata(Name, MD);
2021template <
class IntTy>
2022void MDFieldPrinter::printInt(
StringRef Name, IntTy
Int,
bool ShouldSkipZero) {
2023 if (ShouldSkipZero && !
Int)
2030 bool IsUnsigned,
bool ShouldSkipZero) {
2031 if (ShouldSkipZero &&
Int.isZero())
2034 Out <<
FS <<
Name <<
": ";
2035 Int.print(Out, !IsUnsigned);
2039 std::optional<bool>
Default) {
2042 Out <<
FS <<
Name <<
": " << (
Value ?
"true" :
"false");
2049 Out <<
FS <<
Name <<
": ";
2054 FieldSeparator FlagsFS(
" | ");
2055 for (
auto F : SplitFlags) {
2057 assert(!StringF.empty() &&
"Expected valid flag");
2058 Out << FlagsFS << StringF;
2060 if (Extra || SplitFlags.empty())
2061 Out << FlagsFS << Extra;
2064void MDFieldPrinter::printDISPFlags(
StringRef Name,
2068 Out <<
FS <<
Name <<
": ";
2078 FieldSeparator FlagsFS(
" | ");
2079 for (
auto F : SplitFlags) {
2081 assert(!StringF.empty() &&
"Expected valid flag");
2082 Out << FlagsFS << StringF;
2084 if (Extra || SplitFlags.empty())
2085 Out << FlagsFS << Extra;
2088void MDFieldPrinter::printEmissionKind(
StringRef Name,
2093void MDFieldPrinter::printNameTableKind(
StringRef Name,
2100void MDFieldPrinter::printFixedPointKind(
StringRef Name,
2105template <
class IntTy,
class Stringifier>
2107 Stringifier
toString,
bool ShouldSkipZero) {
2108 if (ShouldSkipZero && !
Value)
2111 Out <<
FS <<
Name <<
": ";
2120 AsmWriterContext &WriterCtx) {
2121 Out <<
"!GenericDINode(";
2122 MDFieldPrinter
Printer(Out, WriterCtx);
2124 Printer.printString(
"header",
N->getHeader());
2125 if (
N->getNumDwarfOperands()) {
2126 Out <<
Printer.FS <<
"operands: {";
2128 for (
auto &
I :
N->dwarf_operands()) {
2138 AsmWriterContext &WriterCtx) {
2139 Out <<
"!DILocation(";
2140 MDFieldPrinter
Printer(Out, WriterCtx);
2142 Printer.printInt(
"line",
DL->getLine(),
false);
2143 Printer.printInt(
"column",
DL->getColumn());
2144 Printer.printMetadata(
"scope",
DL->getRawScope(),
false);
2145 Printer.printMetadata(
"inlinedAt",
DL->getRawInlinedAt());
2146 Printer.printBool(
"isImplicitCode",
DL->isImplicitCode(),
2148 Printer.printInt(
"atomGroup",
DL->getAtomGroup());
2149 Printer.printInt<
unsigned>(
"atomRank",
DL->getAtomRank());
2154 AsmWriterContext &WriterCtx) {
2155 Out <<
"!DIAssignID()";
2156 MDFieldPrinter
Printer(Out, WriterCtx);
2160 AsmWriterContext &WriterCtx) {
2161 Out <<
"!DISubrange(";
2162 MDFieldPrinter
Printer(Out, WriterCtx);
2164 Printer.printMetadataOrInt(
"count",
N->getRawCountNode(),
2170 Printer.printMetadataOrInt(
"lowerBound",
N->getRawLowerBound(),
2173 Printer.printMetadataOrInt(
"upperBound",
N->getRawUpperBound(),
2176 Printer.printMetadataOrInt(
"stride",
N->getRawStride(),
2184 AsmWriterContext &WriterCtx) {
2185 Out <<
"!DIGenericSubrange(";
2186 MDFieldPrinter
Printer(Out, WriterCtx);
2188 auto GetConstant = [&](
Metadata *Bound) -> std::optional<int64_t> {
2191 return std::nullopt;
2192 if (BE->isConstant() &&
2194 *BE->isConstant()) {
2195 return static_cast<int64_t
>(BE->getElement(1));
2197 return std::nullopt;
2200 auto *
Count =
N->getRawCountNode();
2201 if (
auto ConstantCount = GetConstant(
Count))
2202 Printer.printInt(
"count", *ConstantCount,
2207 auto *LBound =
N->getRawLowerBound();
2208 if (
auto ConstantLBound = GetConstant(LBound))
2209 Printer.printInt(
"lowerBound", *ConstantLBound,
2212 Printer.printMetadata(
"lowerBound", LBound,
true);
2214 auto *UBound =
N->getRawUpperBound();
2215 if (
auto ConstantUBound = GetConstant(UBound))
2216 Printer.printInt(
"upperBound", *ConstantUBound,
2219 Printer.printMetadata(
"upperBound", UBound,
true);
2221 auto *Stride =
N->getRawStride();
2222 if (
auto ConstantStride = GetConstant(Stride))
2223 Printer.printInt(
"stride", *ConstantStride,
2226 Printer.printMetadata(
"stride", Stride,
true);
2232 AsmWriterContext &) {
2233 Out <<
"!DIEnumerator(";
2235 Printer.printString(
"name",
N->getName(),
false);
2236 Printer.printAPInt(
"value",
N->getValue(),
N->isUnsigned(),
2238 if (
N->isUnsigned())
2239 Printer.printBool(
"isUnsigned",
true);
2244 AsmWriterContext &WriterCtx) {
2245 Out <<
"!DIBasicType(";
2246 MDFieldPrinter
Printer(Out, WriterCtx);
2247 if (
N->getTag() != dwarf::DW_TAG_base_type)
2249 Printer.printString(
"name",
N->getName());
2250 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2251 Printer.printInt(
"align",
N->getAlignInBits());
2252 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2254 Printer.printInt(
"num_extra_inhabitants",
N->getNumExtraInhabitants());
2255 Printer.printDIFlags(
"flags",
N->getFlags());
2260 AsmWriterContext &WriterCtx) {
2261 Out <<
"!DIFixedPointType(";
2262 MDFieldPrinter
Printer(Out, WriterCtx);
2263 if (
N->getTag() != dwarf::DW_TAG_base_type)
2265 Printer.printString(
"name",
N->getName());
2266 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2267 Printer.printInt(
"align",
N->getAlignInBits());
2268 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2270 Printer.printDIFlags(
"flags",
N->getFlags());
2271 Printer.printFixedPointKind(
"kind",
N->getKind());
2272 if (
N->isRational()) {
2273 bool IsUnsigned = !
N->isSigned();
2274 Printer.printAPInt(
"numerator",
N->getNumerator(), IsUnsigned,
false);
2275 Printer.printAPInt(
"denominator",
N->getDenominator(), IsUnsigned,
false);
2277 Printer.printInt(
"factor",
N->getFactor());
2283 AsmWriterContext &WriterCtx) {
2284 Out <<
"!DIStringType(";
2285 MDFieldPrinter
Printer(Out, WriterCtx);
2286 if (
N->getTag() != dwarf::DW_TAG_string_type)
2288 Printer.printString(
"name",
N->getName());
2289 Printer.printMetadata(
"stringLength",
N->getRawStringLength());
2290 Printer.printMetadata(
"stringLengthExpression",
N->getRawStringLengthExp());
2291 Printer.printMetadata(
"stringLocationExpression",
2292 N->getRawStringLocationExp());
2293 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2294 Printer.printInt(
"align",
N->getAlignInBits());
2295 Printer.printDwarfEnum(
"encoding",
N->getEncoding(),
2301 AsmWriterContext &WriterCtx) {
2302 Out <<
"!DIDerivedType(";
2303 MDFieldPrinter
Printer(Out, WriterCtx);
2305 Printer.printString(
"name",
N->getName());
2306 Printer.printMetadata(
"scope",
N->getRawScope());
2307 Printer.printMetadata(
"file",
N->getRawFile());
2308 Printer.printInt(
"line",
N->getLine());
2309 Printer.printMetadata(
"baseType",
N->getRawBaseType(),
2311 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2312 Printer.printInt(
"align",
N->getAlignInBits());
2313 Printer.printMetadataOrInt(
"offset",
N->getRawOffsetInBits(),
true);
2314 Printer.printDIFlags(
"flags",
N->getFlags());
2315 Printer.printMetadata(
"extraData",
N->getRawExtraData());
2316 if (
const auto &DWARFAddressSpace =
N->getDWARFAddressSpace())
2317 Printer.printInt(
"dwarfAddressSpace", *DWARFAddressSpace,
2319 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2320 if (
auto PtrAuthData =
N->getPtrAuthData()) {
2321 Printer.printInt(
"ptrAuthKey", PtrAuthData->key());
2322 Printer.printBool(
"ptrAuthIsAddressDiscriminated",
2323 PtrAuthData->isAddressDiscriminated());
2324 Printer.printInt(
"ptrAuthExtraDiscriminator",
2325 PtrAuthData->extraDiscriminator());
2326 Printer.printBool(
"ptrAuthIsaPointer", PtrAuthData->isaPointer());
2327 Printer.printBool(
"ptrAuthAuthenticatesNullValues",
2328 PtrAuthData->authenticatesNullValues());
2334 AsmWriterContext &WriterCtx) {
2335 Out <<
"!DISubrangeType(";
2336 MDFieldPrinter
Printer(Out, WriterCtx);
2337 Printer.printString(
"name",
N->getName());
2338 Printer.printMetadata(
"scope",
N->getRawScope());
2339 Printer.printMetadata(
"file",
N->getRawFile());
2340 Printer.printInt(
"line",
N->getLine());
2341 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2342 Printer.printInt(
"align",
N->getAlignInBits());
2343 Printer.printDIFlags(
"flags",
N->getFlags());
2344 Printer.printMetadata(
"baseType",
N->getRawBaseType(),
2346 Printer.printMetadata(
"lowerBound",
N->getRawLowerBound());
2347 Printer.printMetadata(
"upperBound",
N->getRawUpperBound());
2348 Printer.printMetadata(
"stride",
N->getRawStride());
2349 Printer.printMetadata(
"bias",
N->getRawBias());
2354 AsmWriterContext &WriterCtx) {
2355 Out <<
"!DICompositeType(";
2356 MDFieldPrinter
Printer(Out, WriterCtx);
2358 Printer.printString(
"name",
N->getName());
2359 Printer.printMetadata(
"scope",
N->getRawScope());
2360 Printer.printMetadata(
"file",
N->getRawFile());
2361 Printer.printInt(
"line",
N->getLine());
2362 Printer.printMetadata(
"baseType",
N->getRawBaseType());
2363 Printer.printMetadataOrInt(
"size",
N->getRawSizeInBits(),
true);
2364 Printer.printInt(
"align",
N->getAlignInBits());
2365 Printer.printMetadataOrInt(
"offset",
N->getRawOffsetInBits(),
true);
2366 Printer.printInt(
"num_extra_inhabitants",
N->getNumExtraInhabitants());
2367 Printer.printDIFlags(
"flags",
N->getFlags());
2368 Printer.printMetadata(
"elements",
N->getRawElements());
2369 Printer.printDwarfEnum(
"runtimeLang",
N->getRuntimeLang(),
2371 Printer.printMetadata(
"vtableHolder",
N->getRawVTableHolder());
2372 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2373 Printer.printString(
"identifier",
N->getIdentifier());
2374 Printer.printMetadata(
"discriminator",
N->getRawDiscriminator());
2375 Printer.printMetadata(
"dataLocation",
N->getRawDataLocation());
2376 Printer.printMetadata(
"associated",
N->getRawAssociated());
2377 Printer.printMetadata(
"allocated",
N->getRawAllocated());
2378 if (
auto *RankConst =
N->getRankConst())
2379 Printer.printInt(
"rank", RankConst->getSExtValue(),
2382 Printer.printMetadata(
"rank",
N->getRawRank(),
true);
2383 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2384 if (
auto *Specification =
N->getRawSpecification())
2385 Printer.printMetadata(
"specification", Specification);
2387 if (
auto EnumKind =
N->getEnumKind())
2391 Printer.printMetadata(
"bitStride",
N->getRawBitStride());
2396 AsmWriterContext &WriterCtx) {
2397 Out <<
"!DISubroutineType(";
2398 MDFieldPrinter
Printer(Out, WriterCtx);
2399 Printer.printDIFlags(
"flags",
N->getFlags());
2401 Printer.printMetadata(
"types",
N->getRawTypeArray(),
2409 Printer.printString(
"filename",
N->getFilename(),
2411 Printer.printString(
"directory",
N->getDirectory(),
2414 if (
N->getChecksum())
2415 Printer.printChecksum(*
N->getChecksum());
2417 Printer.printString(
"source", *
N->getSource(),
2423 AsmWriterContext &WriterCtx) {
2424 Out <<
"!DICompileUnit(";
2425 MDFieldPrinter
Printer(Out, WriterCtx);
2426 Printer.printDwarfEnum(
"language",
N->getSourceLanguage(),
2428 Printer.printMetadata(
"file",
N->getRawFile(),
false);
2429 Printer.printString(
"producer",
N->getProducer());
2430 Printer.printBool(
"isOptimized",
N->isOptimized());
2431 Printer.printString(
"flags",
N->getFlags());
2432 Printer.printInt(
"runtimeVersion",
N->getRuntimeVersion(),
2434 Printer.printString(
"splitDebugFilename",
N->getSplitDebugFilename());
2435 Printer.printEmissionKind(
"emissionKind",
N->getEmissionKind());
2436 Printer.printMetadata(
"enums",
N->getRawEnumTypes());
2437 Printer.printMetadata(
"retainedTypes",
N->getRawRetainedTypes());
2438 Printer.printMetadata(
"globals",
N->getRawGlobalVariables());
2439 Printer.printMetadata(
"imports",
N->getRawImportedEntities());
2440 Printer.printMetadata(
"macros",
N->getRawMacros());
2441 Printer.printInt(
"dwoId",
N->getDWOId());
2442 Printer.printBool(
"splitDebugInlining",
N->getSplitDebugInlining(),
true);
2443 Printer.printBool(
"debugInfoForProfiling",
N->getDebugInfoForProfiling(),
2445 Printer.printNameTableKind(
"nameTableKind",
N->getNameTableKind());
2446 Printer.printBool(
"rangesBaseAddress",
N->getRangesBaseAddress(),
false);
2447 Printer.printString(
"sysroot",
N->getSysRoot());
2448 Printer.printString(
"sdk",
N->getSDK());
2453 AsmWriterContext &WriterCtx) {
2454 Out <<
"!DISubprogram(";
2455 MDFieldPrinter
Printer(Out, WriterCtx);
2456 Printer.printString(
"name",
N->getName());
2457 Printer.printString(
"linkageName",
N->getLinkageName());
2458 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2459 Printer.printMetadata(
"file",
N->getRawFile());
2460 Printer.printInt(
"line",
N->getLine());
2461 Printer.printMetadata(
"type",
N->getRawType());
2462 Printer.printInt(
"scopeLine",
N->getScopeLine());
2463 Printer.printMetadata(
"containingType",
N->getRawContainingType());
2464 if (
N->getVirtuality() != dwarf::DW_VIRTUALITY_none ||
2465 N->getVirtualIndex() != 0)
2466 Printer.printInt(
"virtualIndex",
N->getVirtualIndex(),
false);
2467 Printer.printInt(
"thisAdjustment",
N->getThisAdjustment());
2468 Printer.printDIFlags(
"flags",
N->getFlags());
2469 Printer.printDISPFlags(
"spFlags",
N->getSPFlags());
2470 Printer.printMetadata(
"unit",
N->getRawUnit());
2471 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2472 Printer.printMetadata(
"declaration",
N->getRawDeclaration());
2473 Printer.printMetadata(
"retainedNodes",
N->getRawRetainedNodes());
2474 Printer.printMetadata(
"thrownTypes",
N->getRawThrownTypes());
2475 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2476 Printer.printString(
"targetFuncName",
N->getTargetFuncName());
2477 Printer.printBool(
"keyInstructions",
N->getKeyInstructionsEnabled(),
false);
2482 AsmWriterContext &WriterCtx) {
2483 Out <<
"!DILexicalBlock(";
2484 MDFieldPrinter
Printer(Out, WriterCtx);
2485 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2486 Printer.printMetadata(
"file",
N->getRawFile());
2487 Printer.printInt(
"line",
N->getLine());
2488 Printer.printInt(
"column",
N->getColumn());
2494 AsmWriterContext &WriterCtx) {
2495 Out <<
"!DILexicalBlockFile(";
2496 MDFieldPrinter
Printer(Out, WriterCtx);
2497 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2498 Printer.printMetadata(
"file",
N->getRawFile());
2499 Printer.printInt(
"discriminator",
N->getDiscriminator(),
2505 AsmWriterContext &WriterCtx) {
2506 Out <<
"!DINamespace(";
2507 MDFieldPrinter
Printer(Out, WriterCtx);
2508 Printer.printString(
"name",
N->getName());
2509 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2510 Printer.printBool(
"exportSymbols",
N->getExportSymbols(),
false);
2515 AsmWriterContext &WriterCtx) {
2516 Out <<
"!DICommonBlock(";
2517 MDFieldPrinter
Printer(Out, WriterCtx);
2518 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2519 Printer.printMetadata(
"declaration",
N->getRawDecl(),
false);
2520 Printer.printString(
"name",
N->getName());
2521 Printer.printMetadata(
"file",
N->getRawFile());
2522 Printer.printInt(
"line",
N->getLineNo());
2527 AsmWriterContext &WriterCtx) {
2529 MDFieldPrinter
Printer(Out, WriterCtx);
2531 Printer.printInt(
"line",
N->getLine());
2532 Printer.printString(
"name",
N->getName());
2533 Printer.printString(
"value",
N->getValue());
2538 AsmWriterContext &WriterCtx) {
2539 Out <<
"!DIMacroFile(";
2540 MDFieldPrinter
Printer(Out, WriterCtx);
2541 Printer.printInt(
"line",
N->getLine());
2542 Printer.printMetadata(
"file",
N->getRawFile(),
false);
2543 Printer.printMetadata(
"nodes",
N->getRawElements());
2548 AsmWriterContext &WriterCtx) {
2549 Out <<
"!DIModule(";
2550 MDFieldPrinter
Printer(Out, WriterCtx);
2551 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2552 Printer.printString(
"name",
N->getName());
2553 Printer.printString(
"configMacros",
N->getConfigurationMacros());
2554 Printer.printString(
"includePath",
N->getIncludePath());
2555 Printer.printString(
"apinotes",
N->getAPINotesFile());
2556 Printer.printMetadata(
"file",
N->getRawFile());
2557 Printer.printInt(
"line",
N->getLineNo());
2558 Printer.printBool(
"isDecl",
N->getIsDecl(),
false);
2564 AsmWriterContext &WriterCtx) {
2565 Out <<
"!DITemplateTypeParameter(";
2566 MDFieldPrinter
Printer(Out, WriterCtx);
2567 Printer.printString(
"name",
N->getName());
2568 Printer.printMetadata(
"type",
N->getRawType(),
false);
2569 Printer.printBool(
"defaulted",
N->isDefault(),
false);
2575 AsmWriterContext &WriterCtx) {
2576 Out <<
"!DITemplateValueParameter(";
2577 MDFieldPrinter
Printer(Out, WriterCtx);
2578 if (
N->getTag() != dwarf::DW_TAG_template_value_parameter)
2580 Printer.printString(
"name",
N->getName());
2581 Printer.printMetadata(
"type",
N->getRawType());
2582 Printer.printBool(
"defaulted",
N->isDefault(),
false);
2583 Printer.printMetadata(
"value",
N->getValue(),
false);
2588 AsmWriterContext &WriterCtx) {
2589 Out <<
"!DIGlobalVariable(";
2590 MDFieldPrinter
Printer(Out, WriterCtx);
2591 Printer.printString(
"name",
N->getName());
2592 Printer.printString(
"linkageName",
N->getLinkageName());
2593 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2594 Printer.printMetadata(
"file",
N->getRawFile());
2595 Printer.printInt(
"line",
N->getLine());
2596 Printer.printMetadata(
"type",
N->getRawType());
2597 Printer.printBool(
"isLocal",
N->isLocalToUnit());
2598 Printer.printBool(
"isDefinition",
N->isDefinition());
2599 Printer.printMetadata(
"declaration",
N->getRawStaticDataMemberDeclaration());
2600 Printer.printMetadata(
"templateParams",
N->getRawTemplateParams());
2601 Printer.printInt(
"align",
N->getAlignInBits());
2602 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2607 AsmWriterContext &WriterCtx) {
2608 Out <<
"!DILocalVariable(";
2609 MDFieldPrinter
Printer(Out, WriterCtx);
2610 Printer.printString(
"name",
N->getName());
2611 Printer.printInt(
"arg",
N->getArg());
2612 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2613 Printer.printMetadata(
"file",
N->getRawFile());
2614 Printer.printInt(
"line",
N->getLine());
2615 Printer.printMetadata(
"type",
N->getRawType());
2616 Printer.printDIFlags(
"flags",
N->getFlags());
2617 Printer.printInt(
"align",
N->getAlignInBits());
2618 Printer.printMetadata(
"annotations",
N->getRawAnnotations());
2623 AsmWriterContext &WriterCtx) {
2625 MDFieldPrinter
Printer(Out, WriterCtx);
2626 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2627 Printer.printString(
"name",
N->getName());
2628 Printer.printMetadata(
"file",
N->getRawFile());
2629 Printer.printInt(
"line",
N->getLine());
2630 Printer.printInt(
"column",
N->getColumn());
2631 Printer.printBool(
"isArtificial",
N->isArtificial(),
false);
2632 if (
N->getCoroSuspendIdx())
2633 Printer.printInt(
"coroSuspendIdx", *
N->getCoroSuspendIdx(),
2639 AsmWriterContext &WriterCtx) {
2640 Out <<
"!DIExpression(";
2645 assert(!OpStr.empty() &&
"Expected valid opcode");
2649 Out << FS <<
Op.getArg(0);
2652 for (
unsigned A = 0, AE =
Op.getNumArgs();
A != AE; ++
A)
2653 Out << FS <<
Op.getArg(
A);
2657 for (
const auto &
I :
N->getElements())
2664 AsmWriterContext &WriterCtx,
2665 bool FromValue =
false) {
2667 "Unexpected DIArgList metadata outside of value argument");
2668 Out <<
"!DIArgList(";
2670 MDFieldPrinter
Printer(Out, WriterCtx);
2680 AsmWriterContext &WriterCtx) {
2681 Out <<
"!DIGlobalVariableExpression(";
2682 MDFieldPrinter
Printer(Out, WriterCtx);
2683 Printer.printMetadata(
"var",
N->getVariable());
2684 Printer.printMetadata(
"expr",
N->getExpression());
2689 AsmWriterContext &WriterCtx) {
2690 Out <<
"!DIObjCProperty(";
2691 MDFieldPrinter
Printer(Out, WriterCtx);
2692 Printer.printString(
"name",
N->getName());
2693 Printer.printMetadata(
"file",
N->getRawFile());
2694 Printer.printInt(
"line",
N->getLine());
2695 Printer.printString(
"setter",
N->getSetterName());
2696 Printer.printString(
"getter",
N->getGetterName());
2697 Printer.printInt(
"attributes",
N->getAttributes());
2698 Printer.printMetadata(
"type",
N->getRawType());
2703 AsmWriterContext &WriterCtx) {
2704 Out <<
"!DIImportedEntity(";
2705 MDFieldPrinter
Printer(Out, WriterCtx);
2707 Printer.printString(
"name",
N->getName());
2708 Printer.printMetadata(
"scope",
N->getRawScope(),
false);
2709 Printer.printMetadata(
"entity",
N->getRawEntity());
2710 Printer.printMetadata(
"file",
N->getRawFile());
2711 Printer.printInt(
"line",
N->getLine());
2712 Printer.printMetadata(
"elements",
N->getRawElements());
2717 AsmWriterContext &Ctx) {
2718 if (
Node->isDistinct())
2720 else if (
Node->isTemporary())
2721 Out <<
"<temporary!> ";
2723 switch (
Node->getMetadataID()) {
2726#define HANDLE_MDNODE_LEAF(CLASS) \
2727 case Metadata::CLASS##Kind: \
2728 write##CLASS(Out, cast<CLASS>(Node), Ctx); \
2730#include "llvm/IR/Metadata.def"
2737 AsmWriterContext &WriterCtx) {
2745 assert(WriterCtx.TypePrinter &&
"Constants require TypePrinting!");
2752 if (IA->hasSideEffects())
2753 Out <<
"sideeffect ";
2754 if (IA->isAlignStack())
2755 Out <<
"alignstack ";
2758 Out <<
"inteldialect ";
2777 auto *
Machine = WriterCtx.Machine;
2781 Slot =
Machine->getGlobalSlot(GV);
2784 Slot =
Machine->getLocalSlot(V);
2791 Slot =
Machine->getLocalSlot(V);
2798 Slot =
Machine->getGlobalSlot(GV);
2801 Slot =
Machine->getLocalSlot(V);
2810 Out << Prefix << Slot;
2816 AsmWriterContext &WriterCtx,
2830 std::unique_ptr<SlotTracker> MachineStorage;
2832 if (!WriterCtx.Machine) {
2833 MachineStorage = std::make_unique<SlotTracker>(WriterCtx.Context);
2834 WriterCtx.Machine = MachineStorage.get();
2844 Out <<
"<" <<
N <<
">";
2858 assert(WriterCtx.TypePrinter &&
"TypePrinter required for metadata values");
2860 "Unexpected function-local metadata outside of value argument");
2862 WriterCtx.TypePrinter->print(V->getValue()->getType(), Out);
2869class AssemblyWriter {
2870 formatted_raw_ostream &Out;
2871 const Module *TheModule =
nullptr;
2872 const ModuleSummaryIndex *TheIndex =
nullptr;
2873 std::unique_ptr<SlotTracker> SlotTrackerStorage;
2875 TypePrinting TypePrinter;
2876 AssemblyAnnotationWriter *AnnotationWriter =
nullptr;
2877 SetVector<const Comdat *> Comdats;
2879 bool ShouldPreserveUseListOrder;
2884 DenseMap<const GlobalValueSummary *, GlobalValue::GUID> SummaryToGUIDMap;
2888 AssemblyWriter(formatted_raw_ostream &o, SlotTracker &Mac,
const Module *M,
2889 AssemblyAnnotationWriter *AAW,
bool IsForDebug,
2890 bool ShouldPreserveUseListOrder =
false);
2892 AssemblyWriter(formatted_raw_ostream &o, SlotTracker &Mac,
2893 const ModuleSummaryIndex *Index,
bool IsForDebug);
2896 return AsmWriterContext(&TypePrinter, &
Machine, TheModule);
2899 void printMDNodeBody(
const MDNode *MD);
2900 void printNamedMDNode(
const NamedMDNode *NMD);
2902 void printModule(
const Module *M);
2904 void writeOperand(
const Value *
Op,
bool PrintType);
2905 void writeParamOperand(
const Value *Operand, AttributeSet Attrs);
2906 void writeOperandBundles(
const CallBase *
Call);
2907 void writeSyncScope(
const LLVMContext &
Context,
2909 void writeAtomic(
const LLVMContext &
Context,
2912 void writeAtomicCmpXchg(
const LLVMContext &
Context,
2917 void writeAllMDNodes();
2918 void writeMDNode(
unsigned Slot,
const MDNode *Node);
2919 void writeAttribute(
const Attribute &Attr,
bool InAttrGroup =
false);
2920 void writeAttributeSet(
const AttributeSet &AttrSet,
bool InAttrGroup =
false);
2921 void writeAllAttributeGroups();
2923 void printTypeIdentities();
2924 void printGlobal(
const GlobalVariable *GV);
2925 void printAlias(
const GlobalAlias *GA);
2926 void printIFunc(
const GlobalIFunc *GI);
2927 void printComdat(
const Comdat *
C);
2928 void printFunction(
const Function *
F);
2929 void printArgument(
const Argument *FA, AttributeSet Attrs);
2930 void printBasicBlock(
const BasicBlock *BB);
2931 void printInstructionLine(
const Instruction &
I);
2932 void printInstruction(
const Instruction &
I);
2933 void printDbgMarker(
const DbgMarker &DPI);
2934 void printDbgVariableRecord(
const DbgVariableRecord &DVR);
2935 void printDbgLabelRecord(
const DbgLabelRecord &DLR);
2936 void printDbgRecord(
const DbgRecord &DR);
2937 void printDbgRecordLine(
const DbgRecord &DR);
2939 void printUseListOrder(
const Value *V,
const std::vector<unsigned> &Shuffle);
2940 void printUseLists(
const Function *
F);
2942 void printModuleSummaryIndex();
2943 void printSummaryInfo(
unsigned Slot,
const ValueInfo &VI);
2944 void printSummary(
const GlobalValueSummary &Summary);
2945 void printAliasSummary(
const AliasSummary *AS);
2946 void printGlobalVarSummary(
const GlobalVarSummary *GS);
2947 void printFunctionSummary(
const FunctionSummary *FS);
2948 void printTypeIdSummary(
const TypeIdSummary &TIS);
2950 void printTypeTestResolution(
const TypeTestResolution &TTRes);
2951 void printArgs(
const std::vector<uint64_t> &Args);
2952 void printWPDRes(
const WholeProgramDevirtResolution &WPDRes);
2953 void printTypeIdInfo(
const FunctionSummary::TypeIdInfo &TIDInfo);
2954 void printVFuncId(
const FunctionSummary::VFuncId VFId);
2956 printNonConstVCalls(
const std::vector<FunctionSummary::VFuncId> &VCallList,
2959 printConstVCalls(
const std::vector<FunctionSummary::ConstVCall> &VCallList,
2964 void printMetadataAttachments(
2965 const SmallVectorImpl<std::pair<unsigned, MDNode *>> &MDs,
2966 StringRef Separator);
2970 void printInfoComment(
const Value &V);
2974 void printGCRelocateComment(
const GCRelocateInst &Relocate);
2981 bool IsForDebug,
bool ShouldPreserveUseListOrder)
2982 : Out(
o), TheModule(
M),
Machine(Mac), TypePrinter(
M), AnnotationWriter(AAW),
2983 IsForDebug(IsForDebug),
2984 ShouldPreserveUseListOrder(ShouldPreserveUseListOrder) {
2987 for (
const GlobalObject &GO : TheModule->global_objects())
2994 : Out(
o), TheIndex(Index),
Machine(Mac), TypePrinter(nullptr),
2995 IsForDebug(IsForDebug), ShouldPreserveUseListOrder(
false) {}
2997void AssemblyWriter::writeOperand(
const Value *Operand,
bool PrintType) {
2999 Out <<
"<null operand!>";
3003 TypePrinter.print(Operand->
getType(), Out);
3010void AssemblyWriter::writeSyncScope(
const LLVMContext &
Context,
3018 Context.getSyncScopeNames(SSNs);
3020 Out <<
" syncscope(\"";
3028void AssemblyWriter::writeAtomic(
const LLVMContext &
Context,
3031 if (Ordering == AtomicOrdering::NotAtomic)
3034 writeSyncScope(
Context, SSID);
3038void AssemblyWriter::writeAtomicCmpXchg(
const LLVMContext &
Context,
3042 assert(SuccessOrdering != AtomicOrdering::NotAtomic &&
3043 FailureOrdering != AtomicOrdering::NotAtomic);
3045 writeSyncScope(
Context, SSID);
3050void AssemblyWriter::writeParamOperand(
const Value *Operand,
3051 AttributeSet Attrs) {
3053 Out <<
"<null operand!>";
3058 TypePrinter.print(Operand->
getType(), Out);
3060 if (
Attrs.hasAttributes()) {
3062 writeAttributeSet(Attrs);
3070void AssemblyWriter::writeOperandBundles(
const CallBase *
Call) {
3076 bool FirstBundle =
true;
3082 FirstBundle =
false;
3090 bool FirstInput =
true;
3092 for (
const auto &Input : BU.
Inputs) {
3097 if (Input ==
nullptr)
3098 Out <<
"<null operand bundle!>";
3100 TypePrinter.print(Input->getType(), Out);
3112void AssemblyWriter::printModule(
const Module *M) {
3115 if (ShouldPreserveUseListOrder)
3118 if (!
M->getModuleIdentifier().empty() &&
3121 M->getModuleIdentifier().find(
'\n') == std::string::npos)
3122 Out <<
"; ModuleID = '" <<
M->getModuleIdentifier() <<
"'\n";
3124 if (!
M->getSourceFileName().empty()) {
3125 Out <<
"source_filename = \"";
3130 const std::string &
DL =
M->getDataLayoutStr();
3132 Out <<
"target datalayout = \"" <<
DL <<
"\"\n";
3133 if (!
M->getTargetTriple().empty())
3134 Out <<
"target triple = \"" <<
M->getTargetTriple().str() <<
"\"\n";
3136 if (!
M->getModuleInlineAsm().empty()) {
3140 StringRef
Asm =
M->getModuleInlineAsm();
3143 std::tie(Front, Asm) =
Asm.split(
'\n');
3147 Out <<
"module asm \"";
3150 }
while (!
Asm.empty());
3153 printTypeIdentities();
3156 if (!Comdats.empty())
3158 for (
const Comdat *
C : Comdats) {
3160 if (
C != Comdats.back())
3165 if (!
M->global_empty()) Out <<
'\n';
3166 for (
const GlobalVariable &GV :
M->globals()) {
3167 printGlobal(&GV); Out <<
'\n';
3171 if (!
M->alias_empty()) Out <<
"\n";
3172 for (
const GlobalAlias &GA :
M->aliases())
3176 if (!
M->ifunc_empty()) Out <<
"\n";
3177 for (
const GlobalIFunc &GI :
M->ifuncs())
3181 for (
const Function &
F : *M) {
3187 printUseLists(
nullptr);
3192 writeAllAttributeGroups();
3196 if (!
M->named_metadata_empty()) Out <<
'\n';
3198 for (
const NamedMDNode &Node :
M->named_metadata())
3199 printNamedMDNode(&Node);
3208void AssemblyWriter::printModuleSummaryIndex() {
3210 int NumSlots =
Machine.initializeIndexIfNeeded();
3216 std::vector<std::pair<std::string, ModuleHash>> moduleVec;
3217 std::string RegularLTOModuleName =
3219 moduleVec.resize(TheIndex->modulePaths().size());
3220 for (
auto &[ModPath, ModHash] : TheIndex->modulePaths())
3221 moduleVec[
Machine.getModulePathSlot(ModPath)] = std::make_pair(
3224 ModPath.empty() ? RegularLTOModuleName : std::string(ModPath), ModHash);
3227 for (
auto &ModPair : moduleVec) {
3228 Out <<
"^" << i++ <<
" = module: (";
3231 Out <<
"\", hash: (";
3233 for (
auto Hash : ModPair.second)
3240 for (
auto &GlobalList : *TheIndex) {
3241 auto GUID = GlobalList.first;
3242 for (
auto &Summary : GlobalList.second.SummaryList)
3247 for (
auto &GlobalList : *TheIndex) {
3248 auto GUID = GlobalList.first;
3249 auto VI = TheIndex->getValueInfo(GlobalList);
3250 printSummaryInfo(
Machine.getGUIDSlot(GUID), VI);
3254 for (
const auto &TID : TheIndex->typeIds()) {
3255 Out <<
"^" <<
Machine.getTypeIdSlot(TID.second.first)
3256 <<
" = typeid: (name: \"" << TID.second.first <<
"\"";
3257 printTypeIdSummary(TID.second.second);
3258 Out <<
") ; guid = " << TID.first <<
"\n";
3262 for (
auto &TId : TheIndex->typeIdCompatibleVtableMap()) {
3264 Out <<
"^" <<
Machine.getTypeIdCompatibleVtableSlot(TId.first)
3265 <<
" = typeidCompatibleVTable: (name: \"" << TId.first <<
"\"";
3266 printTypeIdCompatibleVtableSummary(TId.second);
3267 Out <<
") ; guid = " <<
GUID <<
"\n";
3271 if (TheIndex->getFlags()) {
3272 Out <<
"^" << NumSlots <<
" = flags: " << TheIndex->getFlags() <<
"\n";
3276 Out <<
"^" << NumSlots <<
" = blockcount: " << TheIndex->getBlockCount()
3286 return "singleImpl";
3288 return "branchFunnel";
3299 return "uniformRetVal";
3301 return "uniqueRetVal";
3303 return "virtualConstProp";
3326void AssemblyWriter::printTypeTestResolution(
const TypeTestResolution &TTRes) {
3333 Out <<
", alignLog2: " << TTRes.
AlignLog2;
3335 Out <<
", sizeM1: " << TTRes.
SizeM1;
3338 Out <<
", bitMask: " << (unsigned)TTRes.
BitMask;
3345void AssemblyWriter::printTypeIdSummary(
const TypeIdSummary &TIS) {
3346 Out <<
", summary: (";
3347 printTypeTestResolution(TIS.
TTRes);
3348 if (!TIS.
WPDRes.empty()) {
3349 Out <<
", wpdResolutions: (";
3351 for (
auto &WPDRes : TIS.
WPDRes) {
3353 Out <<
"(offset: " << WPDRes.first <<
", ";
3354 printWPDRes(WPDRes.second);
3362void AssemblyWriter::printTypeIdCompatibleVtableSummary(
3364 Out <<
", summary: (";
3366 for (
auto &
P : TI) {
3368 Out <<
"(offset: " <<
P.AddressPointOffset <<
", ";
3369 Out <<
"^" <<
Machine.getGUIDSlot(
P.VTableVI.getGUID());
3375void AssemblyWriter::printArgs(
const std::vector<uint64_t> &Args) {
3378 for (
auto arg : Args) {
3385void AssemblyWriter::printWPDRes(
const WholeProgramDevirtResolution &WPDRes) {
3386 Out <<
"wpdRes: (kind: ";
3393 Out <<
", resByArg: (";
3395 for (
auto &ResByArg : WPDRes.
ResByArg) {
3397 printArgs(ResByArg.first);
3398 Out <<
", byArg: (kind: ";
3400 if (ResByArg.second.TheKind ==
3402 ResByArg.second.TheKind ==
3404 Out <<
", info: " << ResByArg.second.Info;
3408 if (ResByArg.second.Byte || ResByArg.second.Bit)
3409 Out <<
", byte: " << ResByArg.second.Byte
3410 <<
", bit: " << ResByArg.second.Bit;
3431void AssemblyWriter::printAliasSummary(
const AliasSummary *AS) {
3432 Out <<
", aliasee: ";
3442void AssemblyWriter::printGlobalVarSummary(
const GlobalVarSummary *GS) {
3443 auto VTableFuncs =
GS->vTableFuncs();
3444 Out <<
", varFlags: (readonly: " <<
GS->VarFlags.MaybeReadOnly <<
", "
3445 <<
"writeonly: " <<
GS->VarFlags.MaybeWriteOnly <<
", "
3446 <<
"constant: " <<
GS->VarFlags.Constant;
3447 if (!VTableFuncs.empty())
3449 <<
"vcall_visibility: " <<
GS->VarFlags.VCallVisibility;
3452 if (!VTableFuncs.empty()) {
3453 Out <<
", vTableFuncs: (";
3455 for (
auto &
P : VTableFuncs) {
3457 Out <<
"(virtFunc: ^" <<
Machine.getGUIDSlot(
P.FuncVI.getGUID())
3458 <<
", offset: " <<
P.VTableOffset;
3476 return "linkonce_odr";
3486 return "extern_weak";
3488 return "available_externally";
3517 return "definition";
3519 return "declaration";
3524void AssemblyWriter::printFunctionSummary(
const FunctionSummary *FS) {
3525 Out <<
", insts: " <<
FS->instCount();
3526 if (
FS->fflags().anyFlagSet())
3527 Out <<
", " <<
FS->fflags();
3529 if (!
FS->calls().empty()) {
3530 Out <<
", calls: (";
3532 for (
auto &
Call :
FS->calls()) {
3534 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(
Call.first.getGUID());
3535 if (
Call.second.getHotness() != CalleeInfo::HotnessType::Unknown)
3537 else if (
Call.second.RelBlockFreq)
3538 Out <<
", relbf: " <<
Call.second.RelBlockFreq;
3541 if (
Call.second.HasTailCall)
3548 if (
const auto *TIdInfo =
FS->getTypeIdInfo())
3549 printTypeIdInfo(*TIdInfo);
3553 auto AllocTypeName = [](uint8_t
Type) ->
const char * {
3555 case (uint8_t)AllocationType::None:
3557 case (uint8_t)AllocationType::NotCold:
3559 case (uint8_t)AllocationType::Cold:
3561 case (uint8_t)AllocationType::Hot:
3567 if (!
FS->allocs().empty()) {
3568 Out <<
", allocs: (";
3570 for (
auto &AI :
FS->allocs()) {
3572 Out <<
"(versions: (";
3574 for (
auto V : AI.Versions) {
3576 Out << AllocTypeName(V);
3578 Out <<
"), memProf: (";
3579 FieldSeparator MIBFS;
3580 for (
auto &MIB : AI.MIBs) {
3582 Out <<
"(type: " << AllocTypeName((uint8_t)MIB.AllocType);
3583 Out <<
", stackIds: (";
3584 FieldSeparator SIDFS;
3585 for (
auto Id : MIB.StackIdIndices) {
3587 Out << TheIndex->getStackIdAtIndex(Id);
3596 if (!
FS->callsites().empty()) {
3597 Out <<
", callsites: (";
3598 FieldSeparator SNFS;
3599 for (
auto &CI :
FS->callsites()) {
3602 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(CI.Callee.getGUID());
3604 Out <<
"(callee: null";
3605 Out <<
", clones: (";
3607 for (
auto V : CI.Clones) {
3611 Out <<
"), stackIds: (";
3612 FieldSeparator SIDFS;
3613 for (
auto Id : CI.StackIdIndices) {
3615 Out << TheIndex->getStackIdAtIndex(Id);
3622 auto PrintRange = [&](
const ConstantRange &
Range) {
3626 if (!
FS->paramAccesses().empty()) {
3627 Out <<
", params: (";
3629 for (
auto &PS :
FS->paramAccesses()) {
3631 Out <<
"(param: " << PS.ParamNo;
3632 Out <<
", offset: ";
3634 if (!PS.Calls.empty()) {
3635 Out <<
", calls: (";
3637 for (
auto &
Call : PS.Calls) {
3639 Out <<
"(callee: ^" <<
Machine.getGUIDSlot(
Call.Callee.getGUID());
3640 Out <<
", param: " <<
Call.ParamNo;
3641 Out <<
", offset: ";
3642 PrintRange(
Call.Offsets);
3653void AssemblyWriter::printTypeIdInfo(
3654 const FunctionSummary::TypeIdInfo &TIDInfo) {
3655 Out <<
", typeIdInfo: (";
3656 FieldSeparator TIDFS;
3659 Out <<
"typeTests: (";
3662 auto TidIter = TheIndex->typeIds().equal_range(GUID);
3663 if (TidIter.first == TidIter.second) {
3669 for (
const auto &[GUID, TypeIdPair] :
make_range(TidIter)) {
3671 auto Slot =
Machine.getTypeIdSlot(TypeIdPair.first);
3689 "typeTestAssumeConstVCalls");
3694 "typeCheckedLoadConstVCalls");
3699void AssemblyWriter::printVFuncId(
const FunctionSummary::VFuncId VFId) {
3700 auto TidIter = TheIndex->typeIds().equal_range(VFId.
GUID);
3701 if (TidIter.first == TidIter.second) {
3702 Out <<
"vFuncId: (";
3703 Out <<
"guid: " << VFId.
GUID;
3704 Out <<
", offset: " << VFId.
Offset;
3710 for (
const auto &[GUID, TypeIdPair] :
make_range(TidIter)) {
3712 Out <<
"vFuncId: (";
3713 auto Slot =
Machine.getTypeIdSlot(TypeIdPair.first);
3716 Out <<
", offset: " << VFId.
Offset;
3721void AssemblyWriter::printNonConstVCalls(
3722 const std::vector<FunctionSummary::VFuncId> &VCallList,
const char *
Tag) {
3723 Out <<
Tag <<
": (";
3725 for (
auto &VFuncId : VCallList) {
3727 printVFuncId(VFuncId);
3732void AssemblyWriter::printConstVCalls(
3733 const std::vector<FunctionSummary::ConstVCall> &VCallList,
3735 Out <<
Tag <<
": (";
3737 for (
auto &ConstVCall : VCallList) {
3740 printVFuncId(ConstVCall.VFunc);
3741 if (!ConstVCall.Args.empty()) {
3743 printArgs(ConstVCall.Args);
3750void AssemblyWriter::printSummary(
const GlobalValueSummary &Summary) {
3751 GlobalValueSummary::GVFlags GVFlags =
Summary.flags();
3754 Out <<
"(module: ^" <<
Machine.getModulePathSlot(
Summary.modulePath())
3757 Out <<
", visibility: "
3760 Out <<
", live: " << GVFlags.
Live;
3761 Out <<
", dsoLocal: " << GVFlags.
DSOLocal;
3763 Out <<
", importType: "
3774 auto RefList =
Summary.refs();
3775 if (!RefList.empty()) {
3778 for (
auto &
Ref : RefList) {
3780 if (
Ref.isReadOnly())
3782 else if (
Ref.isWriteOnly())
3783 Out <<
"writeonly ";
3784 Out <<
"^" <<
Machine.getGUIDSlot(
Ref.getGUID());
3792void AssemblyWriter::printSummaryInfo(
unsigned Slot,
const ValueInfo &VI) {
3793 Out <<
"^" <<
Slot <<
" = gv: (";
3794 if (
VI.hasName() && !
VI.name().empty())
3795 Out <<
"name: \"" <<
VI.name() <<
"\"";
3797 Out <<
"guid: " <<
VI.getGUID();
3798 if (!
VI.getSummaryList().empty()) {
3799 Out <<
", summaries: (";
3801 for (
auto &Summary :
VI.getSummaryList()) {
3803 printSummary(*Summary);
3808 if (
VI.hasName() && !
VI.name().empty())
3809 Out <<
" ; guid = " <<
VI.getGUID();
3816 Out <<
"<empty name> ";
3818 unsigned char FirstC =
static_cast<unsigned char>(Name[0]);
3819 if (isalpha(FirstC) || FirstC ==
'-' || FirstC ==
'$' || FirstC ==
'.' ||
3824 for (
unsigned i = 1, e = Name.size(); i != e; ++i) {
3825 unsigned char C = Name[i];
3826 if (isalnum(
C) ||
C ==
'-' ||
C ==
'$' ||
C ==
'.' ||
C ==
'_')
3834void AssemblyWriter::printNamedMDNode(
const NamedMDNode *NMD) {
3871 Out <<
"dso_local ";
3889 Out <<
"thread_local ";
3892 Out <<
"thread_local(localdynamic) ";
3895 Out <<
"thread_local(initialexec) ";
3898 Out <<
"thread_local(localexec) ";
3908 return "local_unnamed_addr";
3910 return "unnamed_addr";
3933void AssemblyWriter::printGlobal(
const GlobalVariable *GV) {
3935 Out <<
"; Materializable\n";
3956 Out << (GV->
isConstant() ?
"constant " :
"global ");
3965 Out <<
", section \"";
3970 Out <<
", partition \"";
3975 Out <<
", code_model \"";
4000 Out <<
", no_sanitize_address";
4002 Out <<
", no_sanitize_hwaddress";
4004 Out <<
", sanitize_memtag";
4006 Out <<
", sanitize_address_dyninit";
4011 Out <<
", align " <<
A->value();
4015 printMetadataAttachments(MDs,
", ");
4018 if (
Attrs.hasAttributes())
4019 Out <<
" #" <<
Machine.getAttributeGroupSlot(Attrs);
4021 printInfoComment(*GV);
4024void AssemblyWriter::printAlias(
const GlobalAlias *GA) {
4026 Out <<
"; Materializable\n";
4046 if (
const Constant *Aliasee = GA->
getAliasee()) {
4049 TypePrinter.print(GA->
getType(), Out);
4050 Out <<
" <<NULL ALIASEE>>";
4054 Out <<
", partition \"";
4059 printInfoComment(*GA);
4063void AssemblyWriter::printIFunc(
const GlobalIFunc *GI) {
4065 Out <<
"; Materializable\n";
4080 if (
const Constant *Resolver = GI->
getResolver()) {
4083 TypePrinter.print(GI->
getType(), Out);
4084 Out <<
" <<NULL RESOLVER>>";
4088 Out <<
", partition \"";
4095 printMetadataAttachments(MDs,
", ");
4098 printInfoComment(*GI);
4102void AssemblyWriter::printComdat(
const Comdat *
C) {
4106void AssemblyWriter::printTypeIdentities() {
4107 if (TypePrinter.empty())
4113 auto &NumberedTypes = TypePrinter.getNumberedTypes();
4114 for (
unsigned I = 0,
E = NumberedTypes.size();
I !=
E; ++
I) {
4115 Out <<
'%' <<
I <<
" = type ";
4119 TypePrinter.printStructBody(NumberedTypes[
I], Out);
4123 auto &NamedTypes = TypePrinter.getNamedTypes();
4124 for (StructType *NamedType : NamedTypes) {
4130 TypePrinter.printStructBody(NamedType, Out);
4136void AssemblyWriter::printFunction(
const Function *
F) {
4139 if (
F->isMaterializable())
4140 Out <<
"; Materializable\n";
4142 const AttributeList &
Attrs =
F->getAttributes();
4143 if (
Attrs.hasFnAttrs()) {
4144 AttributeSet AS =
Attrs.getFnAttrs();
4145 std::string AttrStr;
4148 if (!Attr.isStringAttribute()) {
4149 if (!AttrStr.empty()) AttrStr +=
' ';
4150 AttrStr += Attr.getAsString();
4154 if (!AttrStr.empty())
4155 Out <<
"; Function Attrs: " << AttrStr <<
'\n';
4159 Out <<
"; Unknown intrinsic\n";
4163 if (
F->isDeclaration()) {
4166 F->getAllMetadata(MDs);
4167 printMetadataAttachments(MDs,
" ");
4178 if (
F->getCallingConv() != CallingConv::C) {
4183 FunctionType *FT =
F->getFunctionType();
4184 if (
Attrs.hasRetAttrs())
4185 Out <<
Attrs.getAsString(AttributeList::ReturnIndex) <<
' ';
4186 TypePrinter.print(
F->getReturnType(), Out);
4193 if (
F->isDeclaration() && !IsForDebug) {
4195 for (
unsigned I = 0,
E = FT->getNumParams();
I !=
E; ++
I) {
4200 TypePrinter.print(FT->getParamType(
I), Out);
4202 AttributeSet ArgAttrs =
Attrs.getParamAttrs(
I);
4205 writeAttributeSet(ArgAttrs);
4210 for (
const Argument &Arg :
F->args()) {
4212 if (Arg.getArgNo() != 0)
4214 printArgument(&Arg,
Attrs.getParamAttrs(Arg.getArgNo()));
4219 if (FT->isVarArg()) {
4220 if (FT->getNumParams()) Out <<
", ";
4231 if (
F->getAddressSpace() != 0 || !
Mod ||
4232 Mod->getDataLayout().getProgramAddressSpace() != 0)
4233 Out <<
" addrspace(" <<
F->getAddressSpace() <<
")";
4234 if (
Attrs.hasFnAttrs())
4235 Out <<
" #" <<
Machine.getAttributeGroupSlot(
Attrs.getFnAttrs());
4236 if (
F->hasSection()) {
4237 Out <<
" section \"";
4241 if (
F->hasPartition()) {
4242 Out <<
" partition \"";
4247 if (MaybeAlign
A =
F->getAlign())
4248 Out <<
" align " <<
A->value();
4250 Out <<
" gc \"" <<
F->getGC() <<
'"';
4251 if (
F->hasPrefixData()) {
4253 writeOperand(
F->getPrefixData(),
true);
4255 if (
F->hasPrologueData()) {
4256 Out <<
" prologue ";
4257 writeOperand(
F->getPrologueData(),
true);
4259 if (
F->hasPersonalityFn()) {
4260 Out <<
" personality ";
4261 writeOperand(
F->getPersonalityFn(),
true);
4265 if (
auto *MDProf =
F->getMetadata(LLVMContext::MD_prof)) {
4267 MDProf->print(Out, TheModule,
true);
4271 if (
F->isDeclaration()) {
4275 F->getAllMetadata(MDs);
4276 printMetadataAttachments(MDs,
" ");
4280 for (
const BasicBlock &BB : *
F)
4281 printBasicBlock(&BB);
4294void AssemblyWriter::printArgument(
const Argument *Arg, AttributeSet Attrs) {
4296 TypePrinter.print(Arg->
getType(), Out);
4299 if (
Attrs.hasAttributes()) {
4301 writeAttributeSet(Attrs);
4310 assert(Slot != -1 &&
"expect argument in function here");
4311 Out <<
" %" <<
Slot;
4316void AssemblyWriter::printBasicBlock(
const BasicBlock *BB) {
4322 }
else if (!IsEntryBlock) {
4331 if (!IsEntryBlock) {
4338 Out <<
" No predecessors!";
4341 writeOperand(*PI,
false);
4342 for (++PI; PI != PE; ++PI) {
4344 writeOperand(*PI,
false);
4354 for (
const Instruction &
I : *BB) {
4355 for (
const DbgRecord &DR :
I.getDbgRecordRange())
4356 printDbgRecordLine(DR);
4357 printInstructionLine(
I);
4364void AssemblyWriter::printInstructionLine(
const Instruction &
I) {
4365 printInstruction(
I);
4371void AssemblyWriter::printGCRelocateComment(
const GCRelocateInst &Relocate) {
4381void AssemblyWriter::printInfoComment(
const Value &V) {
4383 printGCRelocateComment(*Relocate);
4385 if (AnnotationWriter) {
4391 if (
I->getDebugLoc()) {
4393 I->getDebugLoc().print(Out);
4399 if (
auto *MD =
I->getMetadata(LLVMContext::MD_prof)) {
4401 MD->print(Out, TheModule,
true);
4413 if (Operand ==
nullptr) {
4414 Out <<
" <cannot get addrspace!>";
4418 bool PrintAddrSpace = CallAddrSpace != 0;
4419 if (!PrintAddrSpace) {
4424 if (!
Mod ||
Mod->getDataLayout().getProgramAddressSpace() != 0)
4425 PrintAddrSpace =
true;
4428 Out <<
" addrspace(" << CallAddrSpace <<
")";
4432void AssemblyWriter::printInstruction(
const Instruction &
I) {
4442 }
else if (!
I.getType()->isVoidTy()) {
4444 int SlotNum =
Machine.getLocalSlot(&
I);
4446 Out <<
"<badref> = ";
4448 Out <<
'%' << SlotNum <<
" = ";
4452 if (CI->isMustTailCall())
4454 else if (CI->isTailCall())
4456 else if (CI->isNoTailCall())
4461 Out <<
I.getOpcodeName();
4483 Out <<
' ' << CI->getPredicate();
4490 const Value *Operand =
I.getNumOperands() ?
I.getOperand(0) :
nullptr;
4496 writeOperand(BI.getCondition(),
true);
4498 writeOperand(BI.getSuccessor(0),
true);
4500 writeOperand(BI.getSuccessor(1),
true);
4506 writeOperand(
SI.getCondition(),
true);
4508 writeOperand(
SI.getDefaultDest(),
true);
4510 for (
auto Case :
SI.cases()) {
4512 writeOperand(Case.getCaseValue(),
true);
4514 writeOperand(Case.getCaseSuccessor(),
true);
4520 writeOperand(Operand,
true);
4523 for (
unsigned i = 1, e =
I.getNumOperands(); i != e; ++i) {
4526 writeOperand(
I.getOperand(i),
true);
4531 TypePrinter.print(
I.getType(), Out);
4534 for (
unsigned op = 0, Eop = PN->getNumIncomingValues();
op < Eop; ++
op) {
4535 if (
op) Out <<
", ";
4537 writeOperand(PN->getIncomingValue(
op),
false); Out <<
", ";
4538 writeOperand(PN->getIncomingBlock(
op),
false); Out <<
" ]";
4542 writeOperand(
I.getOperand(0),
true);
4543 for (
unsigned i : EVI->indices())
4547 writeOperand(
I.getOperand(0),
true); Out <<
", ";
4548 writeOperand(
I.getOperand(1),
true);
4549 for (
unsigned i : IVI->indices())
4553 TypePrinter.print(
I.getType(), Out);
4554 if (LPI->isCleanup() || LPI->getNumClauses() != 0)
4557 if (LPI->isCleanup())
4560 for (
unsigned i = 0, e = LPI->getNumClauses(); i != e; ++i) {
4561 if (i != 0 || LPI->isCleanup()) Out <<
"\n";
4562 if (LPI->isCatch(i))
4567 writeOperand(LPI->getClause(i),
true);
4571 writeOperand(CatchSwitch->getParentPad(),
false);
4574 for (
const BasicBlock *PadBB : CatchSwitch->handlers()) {
4577 writeOperand(PadBB,
true);
4581 if (
const BasicBlock *UnwindDest = CatchSwitch->getUnwindDest())
4582 writeOperand(UnwindDest,
true);
4587 writeOperand(FPI->getParentPad(),
false);
4592 writeOperand(FPI->getArgOperand(
Op),
true);
4599 writeOperand(CRI->getOperand(0),
false);
4602 writeOperand(CRI->getOperand(1),
true);
4605 writeOperand(CRI->getOperand(0),
false);
4608 if (CRI->hasUnwindDest())
4609 writeOperand(CRI->getOperand(1),
true);
4614 if (CI->getCallingConv() != CallingConv::C) {
4619 Operand = CI->getCalledOperand();
4620 FunctionType *FTy = CI->getFunctionType();
4621 Type *RetTy = FTy->getReturnType();
4622 const AttributeList &PAL = CI->getAttributes();
4624 if (PAL.hasRetAttrs())
4625 Out <<
' ' << PAL.getAsString(AttributeList::ReturnIndex);
4634 TypePrinter.print(FTy->isVarArg() ? FTy : RetTy, Out);
4636 writeOperand(Operand,
false);
4638 for (
unsigned op = 0, Eop = CI->arg_size();
op < Eop; ++
op) {
4641 writeParamOperand(CI->getArgOperand(
op), PAL.getParamAttrs(
op));
4646 if (CI->isMustTailCall() && CI->getParent() &&
4647 CI->getParent()->getParent() &&
4648 CI->getParent()->getParent()->isVarArg()) {
4649 if (CI->arg_size() > 0)
4655 if (PAL.hasFnAttrs())
4656 Out <<
" #" <<
Machine.getAttributeGroupSlot(PAL.getFnAttrs());
4658 writeOperandBundles(CI);
4660 Operand =
II->getCalledOperand();
4661 FunctionType *FTy =
II->getFunctionType();
4662 Type *RetTy = FTy->getReturnType();
4663 const AttributeList &PAL =
II->getAttributes();
4666 if (
II->getCallingConv() != CallingConv::C) {
4671 if (PAL.hasRetAttrs())
4672 Out <<
' ' << PAL.getAsString(AttributeList::ReturnIndex);
4682 TypePrinter.print(FTy->isVarArg() ? FTy : RetTy, Out);
4684 writeOperand(Operand,
false);
4686 for (
unsigned op = 0, Eop =
II->arg_size();
op < Eop; ++
op) {
4689 writeParamOperand(
II->getArgOperand(
op), PAL.getParamAttrs(
op));
4693 if (PAL.hasFnAttrs())
4694 Out <<
" #" <<
Machine.getAttributeGroupSlot(PAL.getFnAttrs());
4696 writeOperandBundles(
II);
4699 writeOperand(
II->getNormalDest(),
true);
4701 writeOperand(
II->getUnwindDest(),
true);
4703 Operand = CBI->getCalledOperand();
4704 FunctionType *FTy = CBI->getFunctionType();
4705 Type *RetTy = FTy->getReturnType();
4706 const AttributeList &PAL = CBI->getAttributes();
4709 if (CBI->getCallingConv() != CallingConv::C) {
4714 if (PAL.hasRetAttrs())
4715 Out <<
' ' << PAL.getAsString(AttributeList::ReturnIndex);
4722 TypePrinter.print(FTy->isVarArg() ? FTy : RetTy, Out);
4724 writeOperand(Operand,
false);
4726 for (
unsigned op = 0, Eop = CBI->arg_size();
op < Eop; ++
op) {
4729 writeParamOperand(CBI->getArgOperand(
op), PAL.getParamAttrs(
op));
4733 if (PAL.hasFnAttrs())
4734 Out <<
" #" <<
Machine.getAttributeGroupSlot(PAL.getFnAttrs());
4736 writeOperandBundles(CBI);
4739 writeOperand(CBI->getDefaultDest(),
true);
4741 for (
unsigned i = 0, e = CBI->getNumIndirectDests(); i != e; ++i) {
4744 writeOperand(CBI->getIndirectDest(i),
true);
4749 if (AI->isUsedWithInAlloca())
4751 if (AI->isSwiftError())
4752 Out <<
"swifterror ";
4753 TypePrinter.print(AI->getAllocatedType(), Out);
4759 if (!AI->getArraySize() || AI->isArrayAllocation() ||
4760 !AI->getArraySize()->getType()->isIntegerTy(32)) {
4762 writeOperand(AI->getArraySize(),
true);
4764 if (MaybeAlign
A = AI->getAlign()) {
4765 Out <<
", align " <<
A->value();
4768 unsigned AddrSpace = AI->getAddressSpace();
4769 if (AddrSpace != 0) {
4770 Out <<
", addrspace(" << AddrSpace <<
')';
4775 writeOperand(Operand,
true);
4778 TypePrinter.print(
I.getType(), Out);
4782 writeOperand(Operand,
true);
4785 TypePrinter.print(
I.getType(), Out);
4786 }
else if (Operand) {
4789 TypePrinter.print(
GEP->getSourceElementType(), Out);
4793 TypePrinter.print(LI->getType(), Out);
4800 bool PrintAllTypes =
false;
4808 PrintAllTypes =
true;
4810 for (
unsigned i = 1,
E =
I.getNumOperands(); i !=
E; ++i) {
4811 Operand =
I.getOperand(i);
4814 if (Operand && Operand->
getType() != TheType) {
4815 PrintAllTypes =
true;
4821 if (!PrintAllTypes) {
4823 TypePrinter.print(TheType, Out);
4827 for (
unsigned i = 0,
E =
I.getNumOperands(); i !=
E; ++i) {
4829 writeOperand(
I.getOperand(i), PrintAllTypes);
4836 writeAtomic(LI->getContext(), LI->getOrdering(), LI->getSyncScopeID());
4837 if (MaybeAlign
A = LI->getAlign())
4838 Out <<
", align " <<
A->value();
4841 writeAtomic(
SI->getContext(),
SI->getOrdering(),
SI->getSyncScopeID());
4842 if (MaybeAlign
A =
SI->getAlign())
4843 Out <<
", align " <<
A->value();
4845 writeAtomicCmpXchg(CXI->getContext(), CXI->getSuccessOrdering(),
4846 CXI->getFailureOrdering(), CXI->getSyncScopeID());
4847 Out <<
", align " << CXI->getAlign().value();
4849 writeAtomic(RMWI->getContext(), RMWI->getOrdering(),
4850 RMWI->getSyncScopeID());
4851 Out <<
", align " << RMWI->getAlign().value();
4853 writeAtomic(FI->getContext(), FI->getOrdering(), FI->getSyncScopeID());
4861 printMetadataAttachments(InstMD,
", ");
4864 printInfoComment(
I);
4867void AssemblyWriter::printDbgMarker(
const DbgMarker &Marker) {
4871 printDbgRecord(DPR);
4875 Out <<
" DbgMarker -> { ";
4880void AssemblyWriter::printDbgRecord(
const DbgRecord &DR) {
4882 printDbgVariableRecord(*DVR);
4884 printDbgLabelRecord(*DLR);
4889void AssemblyWriter::printDbgVariableRecord(
const DbgVariableRecord &DVR) {
4893 case DbgVariableRecord::LocationType::Value:
4896 case DbgVariableRecord::LocationType::Declare:
4899 case DbgVariableRecord::LocationType::Assign:
4904 "Tried to print a DbgVariableRecord with an invalid LocationType!");
4935void AssemblyWriter::printDbgRecordLine(
const DbgRecord &DR) {
4942void AssemblyWriter::printDbgLabelRecord(
const DbgLabelRecord &Label) {
4944 Out <<
"#dbg_label(";
4951void AssemblyWriter::printMetadataAttachments(
4952 const SmallVectorImpl<std::pair<unsigned, MDNode *>> &MDs,
4953 StringRef Separator) {
4957 if (MDNames.empty())
4958 MDs[0].second->getContext().getMDKindNames(MDNames);
4961 for (
const auto &
I : MDs) {
4962 unsigned Kind =
I.first;
4964 if (Kind < MDNames.size()) {
4968 Out <<
"!<unknown kind #" <<
Kind <<
">";
4974void AssemblyWriter::writeMDNode(
unsigned Slot,
const MDNode *Node) {
4975 Out <<
'!' <<
Slot <<
" = ";
4976 printMDNodeBody(Node);
4980void AssemblyWriter::writeAllMDNodes() {
4986 for (
unsigned i = 0, e = Nodes.
size(); i != e; ++i) {
4987 writeMDNode(i, Nodes[i]);
4991void AssemblyWriter::printMDNodeBody(
const MDNode *Node) {
4996void AssemblyWriter::writeAttribute(
const Attribute &Attr,
bool InAttrGroup) {
5002 Out << Attribute::getNameFromAttrKind(Attr.
getKindAsEnum());
5005 TypePrinter.print(Ty, Out);
5010void AssemblyWriter::writeAttributeSet(
const AttributeSet &AttrSet,
5012 bool FirstAttr =
true;
5013 for (
const auto &Attr : AttrSet) {
5016 writeAttribute(Attr, InAttrGroup);
5021void AssemblyWriter::writeAllAttributeGroups() {
5022 std::vector<std::pair<AttributeSet, unsigned>> asVec;
5023 asVec.resize(
Machine.as_size());
5026 asVec[
I.second] =
I;
5028 for (
const auto &
I : asVec)
5029 Out <<
"attributes #" <<
I.second <<
" = { "
5030 <<
I.first.getAsString(
true) <<
" }\n";
5033void AssemblyWriter::printUseListOrder(
const Value *V,
5034 const std::vector<unsigned> &Shuffle) {
5039 Out <<
"uselistorder";
5042 writeOperand(BB->getParent(),
false);
5044 writeOperand(BB,
false);
5047 writeOperand(V,
true);
5050 assert(Shuffle.size() >= 2 &&
"Shuffle too small");
5054void AssemblyWriter::printUseLists(
const Function *
F) {
5055 auto It = UseListOrders.find(
F);
5056 if (It == UseListOrders.end())
5059 Out <<
"\n; uselistorder directives\n";
5060 for (
const auto &Pair : It->second)
5061 printUseListOrder(Pair.first, Pair.second);
5069 bool ShouldPreserveUseListOrder,
5070 bool IsForDebug)
const {
5073 AssemblyWriter W(OS, SlotTable, this->
getParent(), AAW,
5075 ShouldPreserveUseListOrder);
5076 W.printFunction(
this);
5080 bool ShouldPreserveUseListOrder,
5081 bool IsForDebug)
const {
5084 AssemblyWriter W(OS, SlotTable, this->
getModule(), AAW,
5086 ShouldPreserveUseListOrder);
5087 W.printBasicBlock(
this);
5091 bool ShouldPreserveUseListOrder,
bool IsForDebug)
const {
5094 AssemblyWriter W(OS, SlotTable,
this, AAW, IsForDebug,
5095 ShouldPreserveUseListOrder);
5096 W.printModule(
this);
5102 AssemblyWriter W(OS, SlotTable,
getParent(),
nullptr, IsForDebug);
5103 W.printNamedMDNode(
this);
5107 bool IsForDebug)
const {
5108 std::optional<SlotTracker> LocalST;
5114 SlotTable = &*LocalST;
5118 AssemblyWriter W(OS, *SlotTable,
getParent(),
nullptr, IsForDebug);
5119 W.printNamedMDNode(
this);
5124 ROS <<
" = comdat ";
5131 ROS <<
"exactmatch";
5137 ROS <<
"nodeduplicate";
5149 TP.print(
const_cast<Type*
>(
this), OS);
5158 TP.printStructBody(STy, OS);
5164 if (
Function *
F = CI->getCalledFunction())
5165 if (
F->isIntrinsic())
5166 for (
auto &
Op :
I.operands())
5176 print(ROS, MST, IsForDebug);
5182 print(ROS, MST, IsForDebug);
5186 bool IsForDebug)
const {
5191 auto incorporateFunction = [&](
const Function *
F) {
5196 AssemblyWriter W(OS, SlotTable,
getModuleFromDPI(
this),
nullptr, IsForDebug);
5197 W.printDbgMarker(*
this);
5203 print(ROS, MST, IsForDebug);
5207 bool IsForDebug)
const {
5212 auto incorporateFunction = [&](
const Function *
F) {
5217 ?
Marker->getParent()->getParent()
5219 AssemblyWriter W(OS, SlotTable,
getModuleFromDPI(
this),
nullptr, IsForDebug);
5220 W.printDbgVariableRecord(*
this);
5224 bool IsForDebug)
const {
5229 auto incorporateFunction = [&](
const Function *
F) {
5233 incorporateFunction(
Marker->getParent() ?
Marker->getParent()->getParent()
5235 AssemblyWriter W(OS, SlotTable,
getModuleFromDPI(
this),
nullptr, IsForDebug);
5236 W.printDbgLabelRecord(*
this);
5240 bool ShouldInitializeAllMetadata =
false;
5244 ShouldInitializeAllMetadata =
true;
5247 print(ROS, MST, IsForDebug);
5251 bool IsForDebug)
const {
5256 auto incorporateFunction = [&](
const Function *
F) {
5262 incorporateFunction(
I->getParent() ?
I->getParent()->getParent() :
nullptr);
5264 W.printInstruction(*
I);
5266 incorporateFunction(BB->getParent());
5267 AssemblyWriter W(OS, SlotTable,
getModuleFromVal(BB),
nullptr, IsForDebug);
5268 W.printBasicBlock(BB);
5270 AssemblyWriter W(OS, SlotTable, GV->
getParent(),
nullptr, IsForDebug);
5284 TypePrinting TypePrinter;
5285 TypePrinter.print(
C->getType(), OS);
5287 AsmWriterContext WriterCtx(&TypePrinter, MST.
getMachine());
5303 AsmWriterContext WriterCtx(
nullptr,
Machine, M);
5312 TypePrinting TypePrinter(MST.
getModule());
5314 TypePrinter.print(V.getType(), O);
5348 AsmWriterContext &WriterCtx) {
5361struct MDTreeAsmWriterContext :
public AsmWriterContext {
5364 using EntryTy = std::pair<unsigned, std::string>;
5368 SmallPtrSet<const Metadata *, 4> Visited;
5370 raw_ostream &MainOS;
5372 MDTreeAsmWriterContext(TypePrinting *TP, SlotTracker *ST,
const Module *M,
5373 raw_ostream &OS,
const Metadata *InitMD)
5374 : AsmWriterContext(TP,
ST,
M),
Level(0
U), Visited({InitMD}), MainOS(OS) {}
5376 void onWriteMetadataAsOperand(
const Metadata *MD)
override {
5377 if (!Visited.
insert(MD).second)
5381 raw_string_ostream
SS(Str);
5386 unsigned InsertIdx = Buffer.
size() - 1;
5389 Buffer[InsertIdx].second = std::move(
SS.str());
5393 ~MDTreeAsmWriterContext() {
5394 for (
const auto &Entry : Buffer) {
5396 unsigned NumIndent =
Entry.first * 2U;
5405 bool OnlyAsOperand,
bool PrintAsTree =
false) {
5408 TypePrinting TypePrinter(M);
5410 std::unique_ptr<AsmWriterContext> WriterCtx;
5411 if (PrintAsTree && !OnlyAsOperand)
5412 WriterCtx = std::make_unique<MDTreeAsmWriterContext>(
5416 std::make_unique<AsmWriterContext>(&TypePrinter, MST.
getMachine(), M);
5445 const Module *M,
bool )
const {
5464 AssemblyWriter W(OS, SlotTable,
this, IsForDebug);
5465 W.printModuleSummaryIndex();
5469 unsigned UB)
const {
5475 if (
I.second >= LB &&
I.second < UB)
5476 L.push_back(std::make_pair(
I.second,
I.first));
5479#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
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 PrintCallingConv(unsigned cc, raw_ostream &Out)
static void writeDICompositeType(raw_ostream &Out, const DICompositeType *N, AsmWriterContext &WriterCtx)
static void writeDIFixedPointType(raw_ostream &Out, const DIFixedPointType *N, AsmWriterContext &WriterCtx)
static const char * getWholeProgDevirtResKindName(WholeProgramDevirtResolution::Kind K)
static void writeDISubrangeType(raw_ostream &Out, const DISubrangeType *N, AsmWriterContext &WriterCtx)
static void printMetadataImpl(raw_ostream &ROS, const Metadata &MD, ModuleSlotTracker &MST, const Module *M, bool OnlyAsOperand, bool PrintAsTree=false)
static void WriteOptimizationInfo(raw_ostream &Out, const User *U)
static void writeDIStringType(raw_ostream &Out, const DIStringType *N, AsmWriterContext &WriterCtx)
static std::string getLinkageNameWithSpace(GlobalValue::LinkageTypes LT)
static std::vector< unsigned > predictValueUseListOrder(const Value *V, unsigned ID, const OrderMap &OM)
static void writeDIGlobalVariable(raw_ostream &Out, const DIGlobalVariable *N, AsmWriterContext &WriterCtx)
static void orderValue(const Value *V, OrderMap &OM)
static void PrintThreadLocalModel(GlobalVariable::ThreadLocalMode TLM, formatted_raw_ostream &Out)
static void writeDIBasicType(raw_ostream &Out, const DIBasicType *N, AsmWriterContext &WriterCtx)
static void PrintLLVMName(raw_ostream &OS, StringRef Name, PrefixType Prefix)
Turn the specified name into an 'LLVM name', which is either prefixed with % (if the string only cont...
static StringRef getUnnamedAddrEncoding(GlobalVariable::UnnamedAddr UA)
static const char * getWholeProgDevirtResByArgKindName(WholeProgramDevirtResolution::ByArg::Kind K)
static void PrintShuffleMask(raw_ostream &Out, Type *Ty, ArrayRef< int > Mask)
static void writeDIModule(raw_ostream &Out, const DIModule *N, AsmWriterContext &WriterCtx)
static void writeDIFile(raw_ostream &Out, const DIFile *N, AsmWriterContext &)
static void writeDISubroutineType(raw_ostream &Out, const DISubroutineType *N, AsmWriterContext &WriterCtx)
static bool isReferencingMDNode(const Instruction &I)
#define CC_VLS_CASE(ABI_VLEN)
static void writeDILabel(raw_ostream &Out, const DILabel *N, AsmWriterContext &WriterCtx)
static void WriteMDNodeBodyInternal(raw_ostream &Out, const MDNode *Node, AsmWriterContext &Ctx)
static void writeDIDerivedType(raw_ostream &Out, const DIDerivedType *N, AsmWriterContext &WriterCtx)
static void printMetadataIdentifier(StringRef Name, formatted_raw_ostream &Out)
static void writeDIImportedEntity(raw_ostream &Out, const DIImportedEntity *N, AsmWriterContext &WriterCtx)
static const Module * getModuleFromDPI(const DbgMarker *Marker)
static void printAsOperandImpl(const Value &V, raw_ostream &O, bool PrintType, ModuleSlotTracker &MST)
static void writeDIObjCProperty(raw_ostream &Out, const DIObjCProperty *N, AsmWriterContext &WriterCtx)
static void PrintDLLStorageClass(GlobalValue::DLLStorageClassTypes SCT, formatted_raw_ostream &Out)
static void writeDISubprogram(raw_ostream &Out, const DISubprogram *N, AsmWriterContext &WriterCtx)
static const char * getSummaryKindName(GlobalValueSummary::SummaryKind SK)
static OrderMap orderModule(const Module *M)
static const char * getVisibilityName(GlobalValue::VisibilityTypes Vis)
static 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 WriteAPFloatInternal(raw_ostream &Out, const APFloat &APF)
static void writeDILocation(raw_ostream &Out, const DILocation *DL, AsmWriterContext &WriterCtx)
static void writeDINamespace(raw_ostream &Out, const DINamespace *N, AsmWriterContext &WriterCtx)
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 WriteConstantInternal(raw_ostream &Out, const Constant *CV, AsmWriterContext &WriterCtx)
static void WriteAsOperandInternal(raw_ostream &Out, const Value *V, AsmWriterContext &WriterCtx)
static std::string getLinkageName(GlobalValue::LinkageTypes LT)
static void writeGenericDINode(raw_ostream &Out, const GenericDINode *N, AsmWriterContext &WriterCtx)
static void writeDILocalVariable(raw_ostream &Out, const DILocalVariable *N, AsmWriterContext &WriterCtx)
static const char * getTTResKindName(TypeTestResolution::Kind K)
static void writeDITemplateTypeParameter(raw_ostream &Out, const DITemplateTypeParameter *N, AsmWriterContext &WriterCtx)
static const char * getImportTypeName(GlobalValueSummary::ImportKind IK)
static void writeDICompileUnit(raw_ostream &Out, const DICompileUnit *N, AsmWriterContext &WriterCtx)
static const Module * getModuleFromVal(const Value *V)
static void maybePrintCallAddrSpace(const Value *Operand, const Instruction *I, raw_ostream &Out)
static void writeDIGenericSubrange(raw_ostream &Out, const DIGenericSubrange *N, AsmWriterContext &WriterCtx)
static void writeDISubrange(raw_ostream &Out, const DISubrange *N, AsmWriterContext &WriterCtx)
static void PrintVisibility(GlobalValue::VisibilityTypes Vis, formatted_raw_ostream &Out)
static void writeDILexicalBlockFile(raw_ostream &Out, const DILexicalBlockFile *N, AsmWriterContext &WriterCtx)
static void writeDIEnumerator(raw_ostream &Out, const DIEnumerator *N, AsmWriterContext &)
static 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 void PrintDSOLocation(const GlobalValue &GV, formatted_raw_ostream &Out)
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 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...
dxil pretty DXIL Metadata Pretty Printer
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.
const size_t AbstractManglingParser< Derived, Alloc >::NumOps
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 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 APFloat getSNaN(const fltSemantics &Sem, bool Negative=false, const APInt *payload=nullptr)
Factory for SNaN values.
LLVM_ABI opStatus convert(const fltSemantics &ToSemantics, roundingMode RM, bool *losesInfo)
LLVM_ABI double convertToDouble() const
Converts this APFloat to host double value.
void toString(SmallVectorImpl< char > &Str, unsigned FormatPrecision=0, unsigned FormatMaxPadding=3, bool TruncateZero=true) const
const fltSemantics & getSemantics() const
APInt bitcastToAPInt() const
Class for arbitrary precision integers.
LLVM_ABI APInt getLoBits(unsigned numBits) const
Compute an APInt containing numBits lowbits from this APInt.
uint64_t getZExtValue() const
Get zero extended value.
LLVM_ABI APInt getHiBits(unsigned numBits) const
Compute an APInt containing numBits highbits from this APInt.
Abstract interface of slot tracker storage.
virtual ~AbstractSlotTrackerStorage()
const GlobalValueSummary & getAliasee() const
This class represents an incoming formal argument to a Function.
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
virtual 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...
The address of a basic block.
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
ConstantArray - Constant Array Declarations.
An array constant whose element type is a simple 1/2/4/8-byte integer or float/double,...
A constant value that is initialized with an expression using other constant values.
ConstantFP - Floating Point Values [float, double].
This is the shared class of boolean and integer constants.
A signed pointer, in the ptrauth sense.
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)
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
MDNode * getAsMDNode() const
Return this as a bar MDNode.
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT > iterator
Utility class for floating point operations which can have information about relaxed accuracy require...
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...
static constexpr const char * getRegularLTOModuleName()
LLVM_ABI void dump() const
Dump to stderr (for debugging).
LLVM_ABI void print(raw_ostream &OS, bool IsForDebug=false) const
Print to an output stream.
A Module instance is used to store all the information related to an LLVM module.
iterator_range< 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
LLVM_ABI MDNode * getOperand(unsigned i) const
LLVM_ABI unsigned getNumOperands() const
Utility class for integer operators which may exhibit overflow - Add, Sub, Mul, and Shl.
unsigned getAddressSpace() const
Return the address space of the Pointer type.
An or instruction, which can be marked as "disjoint", indicating that the inputs don't have a 1 in th...
A udiv, sdiv, lshr, or ashr instruction, which can be marked as "exact", indicating that no bits are ...
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.
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",...
StringRef - Represent a constant reference to a string, i.e.
constexpr bool empty() const
empty - Check if the string is empty.
Class to represent struct types.
ArrayRef< Type * > elements() const
unsigned getNumElements() const
Random access to the elements.
bool isLiteral() const
Return true if this type is uniqued by structural equivalence, false if it is a struct definition.
bool isOpaque() const
Return true if this is a type with an identity that has no body specified yet.
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.
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.
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.
const Use * const_op_iterator
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 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.
@ System
Synchronized with respect to all concurrently executing threads.
@ 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)
FunctionAddr VTableAddr Value
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)
auto pred_end(const MachineBasicBlock *BB)
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...
PredIterator< const BasicBlock, Value::const_user_iterator > const_pred_iterator
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 isDigit(char C)
Checks if character C is one of the 10 decimal digits.
FunctionAddr VTableAddr Count
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...
FormattedNumber format_hex(uint64_t N, unsigned Width, bool Upper=false)
format_hex - Output N as a fixed width hexadecimal.
FormattedNumber format_hex_no_prefix(uint64_t N, unsigned Width, bool Upper=false)
format_hex_no_prefix - Output N as a fixed width hexadecimal.
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.
DWARFExpression::Operation Op
raw_ostream & operator<<(raw_ostream &OS, const APFixedPoint &FX)
std::string toString(const APInt &I, unsigned Radix, bool Signed, bool formatAsCLiteral=false, bool UpperCase=true, bool InsertSeparators=false)
auto pred_begin(const MachineBasicBlock *BB)
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
std::vector< TypeIdOffsetVtableInfo > TypeIdCompatibleVtableInfo
List of vtable definitions decorated by a particular type identifier, and their corresponding offsets...
@ Default
The result values are uniform if and only if all operands are uniform.
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.
static LLVM_ABI const fltSemantics & IEEEsingle() LLVM_READNONE
static constexpr roundingMode rmNearestTiesToEven
static LLVM_ABI const fltSemantics & PPCDoubleDouble() LLVM_READNONE
static LLVM_ABI const fltSemantics & x87DoubleExtended() LLVM_READNONE
static LLVM_ABI const fltSemantics & IEEEquad() LLVM_READNONE
static LLVM_ABI const fltSemantics & IEEEdouble() LLVM_READNONE
static LLVM_ABI const fltSemantics & IEEEhalf() LLVM_READNONE
static LLVM_ABI const fltSemantics & BFloat() LLVM_READNONE
A single checksum, represented by a Kind and a Value (a string).
T Value
The string value of the checksum.
StringRef getKindAsString() const
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 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:...