35 return !S.empty() && std::isdigit(S.front());
44 if (!llvm::itanium_demangle::starts_with(S,
C))
51 if (!llvm::itanium_demangle::starts_with(S,
C))
53 S.remove_prefix(
C.size());
57static bool consumeFront(std::string_view &S, std::string_view PrefixA,
58 std::string_view PrefixB,
bool A) {
59 const std::string_view &Prefix =
A ? PrefixA : PrefixB;
63static bool startsWith(std::string_view S, std::string_view PrefixA,
64 std::string_view PrefixB,
bool A) {
65 const std::string_view &Prefix =
A ? PrefixA : PrefixB;
66 return llvm::itanium_demangle::starts_with(S, Prefix);
69bool Demangler::isMemberPointer(std::string_view MangledName,
bool &
Error) {
71 const char F = MangledName.front();
72 MangledName.remove_prefix(1);
98 if (MangledName[0] !=
'6' && MangledName[0] !=
'8') {
102 return (MangledName[0] ==
'8');
110 demanglePointerAuthQualifier(MangledName);
112 if (MangledName.empty()) {
118 switch (MangledName.front()) {
176 size_t End = S.find(
'?');
177 if (End == std::string_view::npos)
179 std::string_view Candidate = S.substr(0, End);
180 if (Candidate.empty())
185 if (Candidate.size() == 1)
186 return Candidate[0] ==
'@' || (Candidate[0] >=
'0' && Candidate[0] <=
'9');
189 if (Candidate.back() !=
'@')
191 Candidate.remove_suffix(1);
199 if (Candidate[0] <
'B' || Candidate[0] >
'P')
201 Candidate.remove_prefix(1);
202 while (!Candidate.empty()) {
203 if (Candidate[0] <
'A' || Candidate[0] >
'P')
205 Candidate.remove_prefix(1);
225 if (llvm::itanium_demangle::starts_with(S,
"$$Q"))
239static bool isArrayType(std::string_view S) {
return S[0] ==
'Y'; }
242 return llvm::itanium_demangle::starts_with(S,
"$$A8@@") ||
243 llvm::itanium_demangle::starts_with(S,
"$$A6");
255static std::pair<Qualifiers, PointerAffinity>
260 const char F = MangledName.front();
261 MangledName.remove_prefix(1);
285 for (
size_t I = 0;
I <
Count; ++
I) {
286 N->Nodes[
I] = Head->
N;
292std::string_view Demangler::copyString(std::string_view Borrowed) {
297 std::memcpy(Stable, Borrowed.data(), Borrowed.size());
299 return {Stable, Borrowed.size()};
303Demangler::demangleSpecialTableSymbolNode(std::string_view &MangledName,
305 NamedIdentifierNode *NI = Arena.
alloc<NamedIdentifierNode>();
308 NI->
Name =
"`vftable'";
311 NI->
Name =
"`vbtable'";
314 NI->
Name =
"`local vftable'";
317 NI->
Name =
"`RTTI Complete Object Locator'";
322 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, NI);
323 SpecialTableSymbolNode *STSN = Arena.
alloc<SpecialTableSymbolNode>();
326 if (MangledName.empty()) {
330 char Front = MangledName.front();
331 MangledName.remove_prefix(1);
332 if (Front !=
'6' && Front !=
'7') {
337 std::tie(STSN->
Quals, IsMember) = demangleQualifiers(MangledName);
351 TargetCurrent =
Next;
352 QualifiedNameNode *QN = demangleFullyQualifiedTypeName(MangledName);
356 TargetCurrent->
N = QN;
366Demangler::demangleLocalStaticGuard(std::string_view &MangledName,
368 LocalStaticGuardIdentifierNode *LSGI =
369 Arena.
alloc<LocalStaticGuardIdentifierNode>();
371 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, LSGI);
372 LocalStaticGuardVariableNode *LSGVN =
373 Arena.
alloc<LocalStaticGuardVariableNode>();
385 if (!MangledName.empty())
386 LSGI->
ScopeIndex = demangleUnsigned(MangledName);
391 std::string_view Name) {
408 std::string_view Name) {
415 std::string_view VariableName) {
424 std::string_view &MangledName,
425 std::string_view VariableName) {
427 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, NI);
428 VariableSymbolNode *VSN = Arena.
alloc<VariableSymbolNode>();
438Demangler::demangleRttiBaseClassDescriptorNode(
ArenaAllocator &Arena,
439 std::string_view &MangledName) {
440 RttiBaseClassDescriptorNode *RBCDN =
441 Arena.
alloc<RttiBaseClassDescriptorNode>();
442 RBCDN->
NVOffset = demangleUnsigned(MangledName);
445 RBCDN->
Flags = demangleUnsigned(MangledName);
449 VariableSymbolNode *VSN = Arena.
alloc<VariableSymbolNode>();
450 VSN->
Name = demangleNameScopeChain(MangledName, RBCDN);
456Demangler::demangleInitFiniStub(std::string_view &MangledName,
458 DynamicStructorIdentifierNode *DSIN =
459 Arena.
alloc<DynamicStructorIdentifierNode>();
462 bool IsKnownStaticDataMember =
false;
464 IsKnownStaticDataMember =
true;
466 SymbolNode *
Symbol = demangleDeclarator(MangledName);
470 FunctionSymbolNode *FSN =
nullptr;
479 int AtCount = IsKnownStaticDataMember ? 2 : 1;
480 for (
int I = 0;
I < AtCount; ++
I) {
487 FSN = demangleFunctionEncoding(MangledName);
491 if (IsKnownStaticDataMember) {
497 FSN =
static_cast<FunctionSymbolNode *
>(
Symbol);
505SymbolNode *Demangler::demangleSpecialIntrinsic(std::string_view &MangledName) {
512 return demangleStringLiteral(MangledName);
517 return demangleSpecialTableSymbolNode(MangledName, SIK);
519 return demangleVcallThunkNode(MangledName);
521 return demangleLocalStaticGuard(MangledName,
false);
523 return demangleLocalStaticGuard(MangledName,
true);
530 if (!MangledName.empty())
535 return demangleUntypedVariable(Arena, MangledName,
536 "`RTTI Base Class Array'");
538 return demangleUntypedVariable(Arena, MangledName,
539 "`RTTI Class Hierarchy Descriptor'");
541 return demangleRttiBaseClassDescriptorNode(Arena, MangledName);
543 return demangleInitFiniStub(MangledName,
false);
545 return demangleInitFiniStub(MangledName,
true);
559Demangler::demangleFunctionIdentifierCode(std::string_view &MangledName) {
560 assert(llvm::itanium_demangle::starts_with(MangledName,
'?'));
561 MangledName.remove_prefix(1);
562 if (MangledName.empty()) {
568 return demangleFunctionIdentifierCode(
571 return demangleFunctionIdentifierCode(MangledName,
573 return demangleFunctionIdentifierCode(MangledName,
578Demangler::demangleStructorIdentifier(std::string_view &MangledName,
580 StructorIdentifierNode *
N = Arena.
alloc<StructorIdentifierNode>();
581 N->IsDestructor = IsDestructor;
586Demangler::demangleConversionOperatorIdentifier(std::string_view &MangledName) {
587 ConversionOperatorIdentifierNode *
N =
588 Arena.
alloc<ConversionOperatorIdentifierNode>();
593Demangler::demangleLiteralOperatorIdentifier(std::string_view &MangledName) {
594 LiteralOperatorIdentifierNode *
N =
595 Arena.
alloc<LiteralOperatorIdentifierNode>();
596 N->Name = demangleSimpleString(MangledName,
false);
601Demangler::translateIntrinsicFunctionCode(
char CH,
604 if (!(
CH >=
'0' &&
CH <=
'9') && !(
CH >=
'A' &&
CH <=
'Z')) {
612 static IFK
Basic[36] = {
638 IFK::GreaterThanEqual,
650 static IFK
Under[36] = {
655 IFK::BitwiseAndEqual,
657 IFK::BitwiseXorEqual,
666 IFK::DefaultCtorClosure,
670 IFK::VecVbaseCtorIter,
674 IFK::EHVecVbaseCtorIter,
675 IFK::CopyCtorClosure,
680 IFK::LocalVftableCtorClosure,
699 IFK::ManVectorCtorIter,
700 IFK::ManVectorDtorIter,
701 IFK::EHVectorCopyCtorIter,
702 IFK::EHVectorVbaseCopyCtorIter,
705 IFK::VectorCopyCtorIter,
706 IFK::VectorVbaseCopyCtorIter,
707 IFK::ManVectorVbaseCopyCtorIter,
728 int Index = (
CH >=
'0' &&
CH <=
'9') ? (
CH -
'0') : (
CH -
'A' + 10);
741Demangler::demangleFunctionIdentifierCode(std::string_view &MangledName,
743 if (MangledName.empty()) {
747 const char CH = MangledName.front();
750 MangledName.remove_prefix(1);
754 return demangleStructorIdentifier(MangledName,
CH ==
'1');
756 return demangleConversionOperatorIdentifier(MangledName);
758 return Arena.
alloc<IntrinsicFunctionIdentifierNode>(
759 translateIntrinsicFunctionCode(
CH, Group));
762 MangledName.remove_prefix(1);
763 return Arena.
alloc<IntrinsicFunctionIdentifierNode>(
764 translateIntrinsicFunctionCode(
CH, Group));
766 MangledName.remove_prefix(1);
769 return demangleLiteralOperatorIdentifier(MangledName);
771 return Arena.
alloc<IntrinsicFunctionIdentifierNode>(
772 translateIntrinsicFunctionCode(
CH, Group));
779SymbolNode *Demangler::demangleEncodedSymbol(std::string_view &MangledName,
781 if (MangledName.empty()) {
787 switch (MangledName.front()) {
793 StorageClass SC = demangleVariableStorageClass(MangledName);
794 return demangleVariableEncoding(MangledName, SC);
797 FunctionSymbolNode *FSN = demangleFunctionEncoding(MangledName);
799 IdentifierNode *UQN =
Name->getUnqualifiedIdentifier();
801 ConversionOperatorIdentifierNode *COIN =
802 static_cast<ConversionOperatorIdentifierNode *
>(UQN);
809SymbolNode *Demangler::demangleDeclarator(std::string_view &MangledName) {
812 QualifiedNameNode *QN = demangleFullyQualifiedSymbolName(MangledName);
816 SymbolNode *
Symbol = demangleEncodedSymbol(MangledName, QN);
823 ConversionOperatorIdentifierNode *COIN =
824 static_cast<ConversionOperatorIdentifierNode *
>(UQN);
833SymbolNode *Demangler::demangleMD5Name(std::string_view &MangledName) {
834 assert(llvm::itanium_demangle::starts_with(MangledName,
"??@"));
838 size_t MD5Last = MangledName.find(
'@', strlen(
"??@"));
839 if (MD5Last == std::string_view::npos) {
843 const char *
Start = MangledName.data();
844 const size_t StartSize = MangledName.size();
845 MangledName.remove_prefix(MD5Last + 1);
859 assert(MangledName.size() < StartSize);
860 const size_t Count = StartSize - MangledName.size();
868SymbolNode *Demangler::demangleTypeinfoName(std::string_view &MangledName) {
869 assert(llvm::itanium_demangle::starts_with(MangledName,
'.'));
873 if (
Error || !MangledName.empty()) {
885 if (llvm::itanium_demangle::starts_with(MangledName,
'.'))
886 return demangleTypeinfoName(MangledName);
888 if (llvm::itanium_demangle::starts_with(MangledName,
"??@"))
889 return demangleMD5Name(MangledName);
892 if (!llvm::itanium_demangle::starts_with(MangledName,
'?')) {
901 if (
SymbolNode *
SI = demangleSpecialIntrinsic(MangledName))
904 return demangleDeclarator(MangledName);
907TagTypeNode *Demangler::parseTagUniqueName(std::string_view &MangledName) {
913 if (MangledName.empty()) {
918 return demangleClassType(MangledName);
929Demangler::demangleVariableEncoding(std::string_view &MangledName,
947 demanglePointerExtQualifiers(MangledName));
949 bool IsMember =
false;
950 std::tie(ExtraChildQuals, IsMember) = demangleQualifiers(MangledName);
954 demangleFullyQualifiedTypeName(MangledName);
962 VSN->
Type->
Quals = demangleQualifiers(MangledName).first;
980std::pair<uint64_t, bool>
981Demangler::demangleNumber(std::string_view &MangledName) {
985 uint64_t
Ret = MangledName[0] -
'0' + 1;
986 MangledName.remove_prefix(1);
987 return {
Ret, IsNegative};
991 for (
size_t i = 0; i < MangledName.size(); ++i) {
992 char C = MangledName[i];
994 MangledName.remove_prefix(i + 1);
995 return {
Ret, IsNegative};
997 if (
'A' <=
C &&
C <=
'P') {
1005 return {0ULL,
false};
1008uint64_t Demangler::demangleUnsigned(std::string_view &MangledName) {
1009 bool IsNegative =
false;
1011 std::tie(
Number, IsNegative) = demangleNumber(MangledName);
1017int64_t Demangler::demangleSigned(std::string_view &MangledName) {
1018 bool IsNegative =
false;
1020 std::tie(
Number, IsNegative) = demangleNumber(MangledName);
1023 int64_t
I =
static_cast<int64_t
>(
Number);
1024 return IsNegative ? -
I :
I;
1029void Demangler::memorizeString(std::string_view S) {
1032 for (
size_t i = 0; i < Backrefs.
NamesCount; ++i)
1035 NamedIdentifierNode *
N = Arena.
alloc<NamedIdentifierNode>();
1041Demangler::demangleBackRefName(std::string_view &MangledName) {
1044 size_t I = MangledName[0] -
'0';
1050 MangledName.remove_prefix(1);
1051 return Backrefs.
Names[
I];
1059 std::string_view Owned = copyString(OB);
1060 memorizeString(Owned);
1061 std::free(
OB.getBuffer());
1065Demangler::demangleTemplateInstantiationName(std::string_view &MangledName,
1067 assert(llvm::itanium_demangle::starts_with(MangledName,
"?$"));
1070 BackrefContext OuterContext;
1074 demangleUnqualifiedSymbolName(MangledName,
NBB_Simple);
1076 Identifier->TemplateParams = demangleTemplateParameterList(MangledName);
1092 memorizeIdentifier(Identifier);
1099Demangler::demangleSimpleName(std::string_view &MangledName,
bool Memorize) {
1100 std::string_view S = demangleSimpleString(MangledName, Memorize);
1104 NamedIdentifierNode *
Name = Arena.
alloc<NamedIdentifierNode>();
1113 return (
C <=
'J') ? (
C -
'A') : (10 +
C -
'K');
1116uint8_t Demangler::demangleCharLiteral(std::string_view &MangledName) {
1117 assert(!MangledName.empty());
1118 if (!llvm::itanium_demangle::starts_with(MangledName,
'?')) {
1119 const uint8_t
F = MangledName.front();
1120 MangledName.remove_prefix(1);
1124 MangledName.remove_prefix(1);
1125 if (MangledName.empty())
1126 goto CharLiteralError;
1130 if (MangledName.size() < 2)
1131 goto CharLiteralError;
1132 std::string_view Nibbles = MangledName.substr(0, 2);
1134 goto CharLiteralError;
1138 MangledName.remove_prefix(2);
1139 return (C1 << 4) | C2;
1143 const char *
Lookup =
",/\\:. \n\t'-";
1144 char C =
Lookup[MangledName[0] -
'0'];
1145 MangledName.remove_prefix(1);
1149 if (MangledName[0] >=
'a' && MangledName[0] <=
'z') {
1150 char Lookup[26] = {
'\xE1',
'\xE2',
'\xE3',
'\xE4',
'\xE5',
'\xE6',
'\xE7',
1151 '\xE8',
'\xE9',
'\xEA',
'\xEB',
'\xEC',
'\xED',
'\xEE',
1152 '\xEF',
'\xF0',
'\xF1',
'\xF2',
'\xF3',
'\xF4',
'\xF5',
1153 '\xF6',
'\xF7',
'\xF8',
'\xF9',
'\xFA'};
1154 char C =
Lookup[MangledName[0] -
'a'];
1155 MangledName.remove_prefix(1);
1159 if (MangledName[0] >=
'A' && MangledName[0] <=
'Z') {
1160 char Lookup[26] = {
'\xC1',
'\xC2',
'\xC3',
'\xC4',
'\xC5',
'\xC6',
'\xC7',
1161 '\xC8',
'\xC9',
'\xCA',
'\xCB',
'\xCC',
'\xCD',
'\xCE',
1162 '\xCF',
'\xD0',
'\xD1',
'\xD2',
'\xD3',
'\xD4',
'\xD5',
1163 '\xD6',
'\xD7',
'\xD8',
'\xD9',
'\xDA'};
1164 char C =
Lookup[MangledName[0] -
'A'];
1165 MangledName.remove_prefix(1);
1174wchar_t Demangler::demangleWcharLiteral(std::string_view &MangledName) {
1177 C1 = demangleCharLiteral(MangledName);
1178 if (
Error || MangledName.empty())
1179 goto WCharLiteralError;
1180 C2 = demangleCharLiteral(MangledName);
1182 goto WCharLiteralError;
1184 return ((
wchar_t)C1 << 8) | (wchar_t)C2;
1193 *Buffer = (Digit < 10) ? (
'0' + Digit) : (
'A' + Digit - 10);
1204 char TempBuffer[17];
1206 ::memset(TempBuffer, 0,
sizeof(TempBuffer));
1207 constexpr int MaxPos =
sizeof(TempBuffer) - 1;
1209 int Pos = MaxPos - 1;
1211 for (
int I = 0;
I < 2; ++
I) {
1216 TempBuffer[Pos--] =
'x';
1218 TempBuffer[Pos--] =
'\\';
1219 OB << std::string_view(&TempBuffer[Pos + 1]);
1261 if (
C > 0x1F &&
C < 0x7F) {
1273 while (
Length > 0 && *End == 0) {
1285 if (*StringBytes++ == 0)
1299 if (NumBytes % 2 == 1)
1305 if (NumBytes < 32) {
1307 if (TrailingNulls >= 4 && NumBytes % 4 == 0)
1309 if (TrailingNulls >= 2)
1321 if (Nulls >= 2 * NumChars / 3 && NumBytes % 4 == 0)
1323 if (Nulls >= NumChars / 3)
1329 unsigned CharIndex,
unsigned CharBytes) {
1330 assert(CharBytes == 1 || CharBytes == 2 || CharBytes == 4);
1331 unsigned Offset = CharIndex * CharBytes;
1333 StringBytes = StringBytes +
Offset;
1334 for (
unsigned I = 0;
I < CharBytes; ++
I) {
1335 unsigned C =
static_cast<unsigned>(StringBytes[
I]);
1342Demangler::demangleVcallThunkNode(std::string_view &MangledName) {
1343 FunctionSymbolNode *FSN = Arena.
alloc<FunctionSymbolNode>();
1344 VcallThunkIdentifierNode *VTIN = Arena.
alloc<VcallThunkIdentifierNode>();
1348 FSN->
Name = demangleNameScopeChain(MangledName, VTIN);
1352 VTIN->OffsetInVTable = demangleUnsigned(MangledName);
1357 return (
Error) ? nullptr : FSN;
1361Demangler::demangleStringLiteral(std::string_view &MangledName) {
1364 std::string_view CRC;
1365 uint64_t StringByteSize;
1366 bool IsWcharT =
false;
1367 bool IsNegative =
false;
1368 size_t CrcEndPos = 0;
1371 EncodedStringLiteralNode *
Result = Arena.
alloc<EncodedStringLiteralNode>();
1375 goto StringLiteralError;
1376 if (MangledName.empty())
1377 goto StringLiteralError;
1380 F = MangledName.front();
1381 MangledName.remove_prefix(1);
1389 goto StringLiteralError;
1393 std::tie(StringByteSize, IsNegative) = demangleNumber(MangledName);
1394 if (
Error || IsNegative || StringByteSize < (IsWcharT ? 2 : 1))
1395 goto StringLiteralError;
1398 CrcEndPos = MangledName.find(
'@');
1399 if (CrcEndPos == std::string_view::npos)
1400 goto StringLiteralError;
1401 CRC = MangledName.substr(0, CrcEndPos);
1402 MangledName.remove_prefix(CrcEndPos + 1);
1403 if (MangledName.empty())
1404 goto StringLiteralError;
1408 if (StringByteSize > 64)
1409 Result->IsTruncated =
true;
1413 if (StringByteSize % 2 != 0)
1414 goto StringLiteralError;
1415 if (StringByteSize == 0)
1416 goto StringLiteralError;
1417 if (MangledName.size() < 2)
1418 goto StringLiteralError;
1419 wchar_t W = demangleWcharLiteral(MangledName);
1420 if (StringByteSize != 2 ||
Result->IsTruncated)
1422 StringByteSize -= 2;
1424 goto StringLiteralError;
1429 constexpr unsigned MaxStringByteLength = 32 * 4;
1430 uint8_t StringBytes[MaxStringByteLength];
1432 unsigned BytesDecoded = 0;
1434 if (MangledName.size() < 1 || BytesDecoded >= MaxStringByteLength)
1435 goto StringLiteralError;
1436 StringBytes[BytesDecoded++] = demangleCharLiteral(MangledName);
1439 if (StringByteSize > BytesDecoded)
1440 Result->IsTruncated =
true;
1442 unsigned CharBytes =
1444 assert(StringByteSize % CharBytes == 0);
1445 switch (CharBytes) {
1458 const unsigned NumChars = BytesDecoded / CharBytes;
1459 for (
unsigned CharIndex = 0; CharIndex < NumChars; ++CharIndex) {
1462 if (CharIndex + 1 < NumChars ||
Result->IsTruncated)
1467 Result->DecodedString = copyString(OB);
1468 std::free(
OB.getBuffer());
1473 std::free(
OB.getBuffer());
1479std::string_view Demangler::demangleSimpleString(std::string_view &MangledName,
1482 for (
size_t i = 0; i < MangledName.size(); ++i) {
1483 if (MangledName[i] !=
'@')
1487 S = MangledName.substr(0, i);
1488 MangledName.remove_prefix(i + 1);
1500Demangler::demangleAnonymousNamespaceName(std::string_view &MangledName) {
1501 assert(llvm::itanium_demangle::starts_with(MangledName,
"?A"));
1504 NamedIdentifierNode *
Node = Arena.
alloc<NamedIdentifierNode>();
1505 Node->Name =
"`anonymous namespace'";
1506 size_t EndPos = MangledName.find(
'@');
1507 if (EndPos == std::string_view::npos) {
1511 std::string_view NamespaceKey = MangledName.substr(0, EndPos);
1512 memorizeString(NamespaceKey);
1513 MangledName = MangledName.substr(EndPos + 1);
1518Demangler::demangleLocallyScopedNamePiece(std::string_view &MangledName) {
1524 bool IsNegative =
false;
1525 std::tie(
Number, IsNegative) = demangleNumber(MangledName);
1544 std::free(
OB.getBuffer());
1550Demangler::demangleFullyQualifiedTypeName(std::string_view &MangledName) {
1552 demangleUnqualifiedTypeName(MangledName,
true);
1557 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, Identifier);
1568Demangler::demangleFullyQualifiedSymbolName(std::string_view &MangledName) {
1575 demangleUnqualifiedSymbolName(MangledName,
NBB_Simple);
1579 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, Identifier);
1588 StructorIdentifierNode *SIN =
1589 static_cast<StructorIdentifierNode *
>(
Identifier);
1591 SIN->
Class =
static_cast<IdentifierNode *
>(ClassNode);
1598Demangler::demangleUnqualifiedTypeName(std::string_view &MangledName,
1605 return demangleBackRefName(MangledName);
1607 if (llvm::itanium_demangle::starts_with(MangledName,
"?$"))
1608 return demangleTemplateInstantiationName(MangledName,
NBB_Template);
1610 return demangleSimpleName(MangledName, Memorize);
1614Demangler::demangleUnqualifiedSymbolName(std::string_view &MangledName,
1617 return demangleBackRefName(MangledName);
1618 if (llvm::itanium_demangle::starts_with(MangledName,
"?$"))
1619 return demangleTemplateInstantiationName(MangledName, NBB);
1620 if (llvm::itanium_demangle::starts_with(MangledName,
'?'))
1621 return demangleFunctionIdentifierCode(MangledName);
1622 return demangleSimpleName(MangledName, (NBB &
NBB_Simple) != 0);
1626Demangler::demangleNameScopePiece(std::string_view &MangledName) {
1628 return demangleBackRefName(MangledName);
1630 if (llvm::itanium_demangle::starts_with(MangledName,
"?$"))
1631 return demangleTemplateInstantiationName(MangledName,
NBB_Template);
1633 if (llvm::itanium_demangle::starts_with(MangledName,
"?A"))
1634 return demangleAnonymousNamespaceName(MangledName);
1637 return demangleLocallyScopedNamePiece(MangledName);
1639 return demangleSimpleName(MangledName,
true);
1643Demangler::demangleNameScopeChain(std::string_view &MangledName,
1647 Head->
N = UnqualifiedName;
1653 NewHead->
Next = Head;
1656 if (MangledName.empty()) {
1662 IdentifierNode *Elem = demangleNameScopePiece(MangledName);
1669 QualifiedNameNode *QN = Arena.
alloc<QualifiedNameNode>();
1674FuncClass Demangler::demangleFunctionClass(std::string_view &MangledName) {
1675 const char F = MangledName.front();
1676 MangledName.remove_prefix(1);
1736 if (MangledName.empty())
1738 const char F = MangledName.front();
1739 MangledName.remove_prefix(1);
1762Demangler::demangleCallingConvention(std::string_view &MangledName) {
1763 if (MangledName.empty()) {
1768 const char F = MangledName.front();
1769 MangledName.remove_prefix(1);
1804Demangler::demangleVariableStorageClass(std::string_view &MangledName) {
1805 assert(MangledName.front() >=
'0' && MangledName.front() <=
'4');
1807 const char F = MangledName.front();
1808 MangledName.remove_prefix(1);
1824std::pair<Qualifiers, bool>
1825Demangler::demangleQualifiers(std::string_view &MangledName) {
1826 if (MangledName.empty()) {
1828 return std::make_pair(
Q_None,
false);
1831 const char F = MangledName.front();
1832 MangledName.remove_prefix(1);
1836 return std::make_pair(
Q_None,
true);
1838 return std::make_pair(
Q_Const,
true);
1845 return std::make_pair(
Q_None,
false);
1847 return std::make_pair(
Q_Const,
false);
1854 return std::make_pair(
Q_None,
false);
1859TypeNode *Demangler::demangleType(std::string_view &MangledName,
1864 std::tie(Quals, IsMember) = demangleQualifiers(MangledName);
1867 std::tie(Quals, IsMember) = demangleQualifiers(MangledName);
1870 if (MangledName.empty()) {
1875 TypeNode *Ty =
nullptr;
1877 Ty = demangleClassType(MangledName);
1879 if (isMemberPointer(MangledName,
Error))
1880 Ty = demangleMemberPointerType(MangledName);
1882 Ty = demanglePointerType(MangledName);
1886 Ty = demangleArrayType(MangledName);
1889 Ty = demangleFunctionType(MangledName,
true);
1891 assert(llvm::itanium_demangle::starts_with(MangledName,
"$$A6"));
1893 Ty = demangleFunctionType(MangledName,
false);
1896 Ty = demangleCustomType(MangledName);
1898 Ty = demanglePrimitiveType(MangledName);
1907bool Demangler::demangleThrowSpecification(std::string_view &MangledName) {
1918Demangler::demangleFunctionType(std::string_view &MangledName,
1919 bool HasThisQuals) {
1920 FunctionSignatureNode *FTy = Arena.
alloc<FunctionSignatureNode>();
1923 FTy->
Quals = demanglePointerExtQualifiers(MangledName);
1939 FTy->
IsNoexcept = demangleThrowSpecification(MangledName);
1945Demangler::demangleFunctionEncoding(std::string_view &MangledName) {
1950 if (MangledName.empty()) {
1955 FuncClass FC = demangleFunctionClass(MangledName);
1958 FunctionSignatureNode *FSN =
nullptr;
1959 ThunkSignatureNode *TTN =
nullptr;
1961 TTN = Arena.
alloc<ThunkSignatureNode>();
1964 TTN = Arena.
alloc<ThunkSignatureNode>();
1977 FSN = Arena.
alloc<FunctionSignatureNode>();
1980 FSN = demangleFunctionType(MangledName, HasThisQuals);
1987 *
static_cast<FunctionSignatureNode *
>(TTN) = *FSN;
1992 FunctionSymbolNode *
Symbol = Arena.
alloc<FunctionSymbolNode>();
1997CustomTypeNode *Demangler::demangleCustomType(std::string_view &MangledName) {
1998 assert(llvm::itanium_demangle::starts_with(MangledName,
'?'));
1999 MangledName.remove_prefix(1);
2001 CustomTypeNode *CTN = Arena.
alloc<CustomTypeNode>();
2002 CTN->
Identifier = demangleUnqualifiedTypeName(MangledName,
true);
2012Demangler::demanglePrimitiveType(std::string_view &MangledName) {
2016 const char F = MangledName.front();
2017 MangledName.remove_prefix(1);
2046 if (MangledName.empty()) {
2050 const char F = MangledName.front();
2051 MangledName.remove_prefix(1);
2079TagTypeNode *Demangler::demangleClassType(std::string_view &MangledName) {
2080 TagTypeNode *
TT =
nullptr;
2082 const char F = MangledName.front();
2083 MangledName.remove_prefix(1);
2105 TT->QualifiedName = demangleFullyQualifiedTypeName(MangledName);
2111PointerTypeNode *Demangler::demanglePointerType(std::string_view &MangledName) {
2112 PointerTypeNode *
Pointer = Arena.
alloc<PointerTypeNode>();
2118 Pointer->Pointee = demangleFunctionType(MangledName,
false);
2122 Qualifiers ExtQuals = demanglePointerExtQualifiers(MangledName);
2125 Pointer->PointerAuthQualifier = createPointerAuthQualifier(MangledName);
2132Demangler::demangleMemberPointerType(std::string_view &MangledName) {
2133 PointerTypeNode *
Pointer = Arena.
alloc<PointerTypeNode>();
2139 Qualifiers ExtQuals = demanglePointerExtQualifiers(MangledName);
2145 Pointer->ClassParent = demangleFullyQualifiedTypeName(MangledName);
2146 Pointer->Pointee = demangleFunctionType(MangledName,
true);
2150 std::tie(PointeeQuals, IsMember) = demangleQualifiers(MangledName);
2152 Pointer->ClassParent = demangleFullyQualifiedTypeName(MangledName);
2156 Pointer->Pointee->Quals = PointeeQuals;
2163Demangler::demanglePointerExtQualifiers(std::string_view &MangledName) {
2175std::optional<PointerAuthQualifierNode::ArgArray>
2176Demangler::demanglePointerAuthQualifier(std::string_view &MangledName) {
2178 return std::nullopt;
2183 for (
unsigned I = 0;
I < NumArgs; ++
I) {
2184 bool IsNegative =
false;
2186 std::tie(
Value, IsNegative) = demangleNumber(MangledName);
2188 return std::nullopt;
2197Demangler::createPointerAuthQualifier(std::string_view &MangledName) {
2199 std::optional<PointerAuthQualifierNode::ArgArray> Vals =
2200 demanglePointerAuthQualifier(MangledName);
2205 PointerAuthQualifierNode *PtrAuthQual =
2206 Arena.
alloc<PointerAuthQualifierNode>();
2207 NodeArrayNode *
Array = Arena.
alloc<NodeArrayNode>();
2209 Array->Count = NumArgs;
2212 for (
unsigned I = 0;
I < NumArgs; ++
I)
2213 Array->Nodes[
I] = Arena.
alloc<IntegerLiteralNode>((*Vals)[
I],
false);
2218ArrayTypeNode *Demangler::demangleArrayType(std::string_view &MangledName) {
2219 assert(MangledName.front() ==
'Y');
2220 MangledName.remove_prefix(1);
2223 bool IsNegative =
false;
2224 std::tie(Rank, IsNegative) = demangleNumber(MangledName);
2225 if (IsNegative || Rank == 0) {
2230 ArrayTypeNode *ATy = Arena.
alloc<ArrayTypeNode>();
2234 for (uint64_t
I = 0;
I < Rank; ++
I) {
2236 std::tie(
D, IsNegative) = demangleNumber(MangledName);
2237 if (
Error || IsNegative) {
2241 Tail->N = Arena.
alloc<IntegerLiteralNode>(
D, IsNegative);
2251 std::tie(ATy->
Quals, IsMember) = demangleQualifiers(MangledName);
2264Demangler::demangleFunctionParameterList(std::string_view &MangledName,
2273 while (!
Error && !llvm::itanium_demangle::starts_with(MangledName,
'@') &&
2274 !llvm::itanium_demangle::starts_with(MangledName,
'Z')) {
2278 size_t N = MangledName[0] -
'0';
2283 MangledName.remove_prefix(1);
2287 Current = &(*Current)->Next;
2291 size_t OldSize = MangledName.size();
2300 size_t CharsConsumed = OldSize - MangledName.size();
2301 assert(CharsConsumed != 0);
2308 Current = &(*Current)->Next;
2330Demangler::demangleTemplateParameterList(std::string_view &MangledName) {
2335 while (!llvm::itanium_demangle::starts_with(MangledName,
'@')) {
2350 const bool IsAutoNTTP =
consumeFront(MangledName,
"$M");
2361 TemplateParameterReferenceNode *TPRN =
nullptr;
2364 TP.N = demangleFullyQualifiedTypeName(MangledName);
2371 }
else if (
startsWith(MangledName,
"$1",
"1", !IsAutoNTTP) ||
2372 startsWith(MangledName,
"$H",
"H", !IsAutoNTTP) ||
2373 startsWith(MangledName,
"$I",
"I", !IsAutoNTTP) ||
2374 startsWith(MangledName,
"$J",
"J", !IsAutoNTTP)) {
2376 TP.N = TPRN = Arena.
alloc<TemplateParameterReferenceNode>();
2380 MangledName.remove_prefix(1);
2386 char InheritanceSpecifier = MangledName.front();
2387 MangledName.remove_prefix(1);
2388 SymbolNode *S =
nullptr;
2389 if (llvm::itanium_demangle::starts_with(MangledName,
'?')) {
2390 S =
parse(MangledName);
2398 switch (InheritanceSpecifier) {
2401 demangleSigned(MangledName);
2405 demangleSigned(MangledName);
2409 demangleSigned(MangledName);
2418 }
else if (llvm::itanium_demangle::starts_with(MangledName,
"$E?")) {
2421 TP.N = TPRN = Arena.
alloc<TemplateParameterReferenceNode>();
2424 }
else if (
startsWith(MangledName,
"$F",
"F", !IsAutoNTTP) ||
2425 startsWith(MangledName,
"$G",
"G", !IsAutoNTTP)) {
2426 TP.N = TPRN = Arena.
alloc<TemplateParameterReferenceNode>();
2430 MangledName.remove_prefix(1);
2431 char InheritanceSpecifier = MangledName.front();
2432 MangledName.remove_prefix(1);
2434 switch (InheritanceSpecifier) {
2437 demangleSigned(MangledName);
2441 demangleSigned(MangledName);
2443 demangleSigned(MangledName);
2450 }
else if (
consumeFront(MangledName,
"$0",
"0", !IsAutoNTTP)) {
2452 bool IsNegative =
false;
2454 std::tie(
Value, IsNegative) = demangleNumber(MangledName);
2456 TP.N = Arena.
alloc<IntegerLiteralNode>(
Value, IsNegative);
2471 assert(llvm::itanium_demangle::starts_with(
2477void Demangler::dumpBackReferences() {
2478 std::printf(
"%d function parameter backreferences\n",
2479 (
int)Backrefs.FunctionParamCount);
2483 for (
size_t I = 0;
I < Backrefs.FunctionParamCount; ++
I) {
2484 OB.setCurrentPosition(0);
2489 std::string_view
B = OB;
2490 std::printf(
" [%d] - %.*s\n", (
int)
I, (
int)
B.size(),
B.data());
2492 std::free(OB.getBuffer());
2494 if (Backrefs.FunctionParamCount > 0)
2496 std::printf(
"%d name backreferences\n", (
int)Backrefs.NamesCount);
2497 for (
size_t I = 0;
I < Backrefs.NamesCount; ++
I) {
2498 std::printf(
" [%d] - %.*s\n", (
int)
I, (
int)Backrefs.Names[
I]->Name.size(),
2499 Backrefs.Names[
I]->Name.data());
2501 if (Backrefs.NamesCount > 0)
2505std::optional<size_t>
2507 std::string_view ProcessedName{MangledName};
2511 return std::nullopt;
2516 D.demangleFullyQualifiedSymbolName(ProcessedName);
2518 return std::nullopt;
2520 return MangledName.length() - ProcessedName.length();
2527 std::string_view Name{MangledName};
2529 if (!
D.Error && NMangled)
2530 *NMangled = MangledName.size() - Name.size();
2533 D.dumpBackReferences();
2555 Buf = OB.getBuffer();
2559 *
Status = InternalStatus;
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
#define DEMANGLE_FALLTHROUGH
#define DEMANGLE_UNREACHABLE
static bool startsWithLocalScopePattern(std::string_view S)
static bool isArrayType(std::string_view S)
static unsigned countEmbeddedNulls(const uint8_t *StringBytes, unsigned Length)
static bool startsWithDigit(std::string_view S)
static QualifiedNameNode * synthesizeQualifiedName(ArenaAllocator &Arena, IdentifierNode *Identifier)
static void outputEscapedChar(OutputBuffer &OB, unsigned C)
static bool isCustomType(std::string_view S)
static void outputHex(OutputBuffer &OB, unsigned C)
static std::pair< Qualifiers, PointerAffinity > demanglePointerCVQualifiers(std::string_view &MangledName)
static VariableSymbolNode * synthesizeVariable(ArenaAllocator &Arena, TypeNode *Type, std::string_view VariableName)
static unsigned decodeMultiByteChar(const uint8_t *StringBytes, unsigned CharIndex, unsigned CharBytes)
static void writeHexDigit(char *Buffer, uint8_t Digit)
static FunctionRefQualifier demangleFunctionRefQualifier(std::string_view &MangledName)
static bool isRebasedHexDigit(char C)
static NodeArrayNode * nodeListToNodeArray(ArenaAllocator &Arena, NodeList *Head, size_t Count)
static uint8_t rebasedHexDigitToNumber(char C)
static unsigned countTrailingNullBytes(const uint8_t *StringBytes, int Length)
static NamedIdentifierNode * synthesizeNamedIdentifier(ArenaAllocator &Arena, std::string_view Name)
static bool startsWith(std::string_view S, std::string_view PrefixA, std::string_view PrefixB, bool A)
static bool consumeFront(std::string_view &S, char C)
static bool isFunctionType(std::string_view S)
static bool isPointerType(std::string_view S)
static unsigned guessCharByteSize(const uint8_t *StringBytes, unsigned NumChars, uint64_t NumBytes)
static SpecialIntrinsicKind consumeSpecialIntrinsicKind(std::string_view &MangledName)
static bool isTagType(std::string_view S)
static int Lookup(ArrayRef< TableEntry > Table, unsigned Opcode)
Lightweight error class with error context and mandatory checking.
The instances of the Type class are immutable: once they are created, they are never changed.
T * alloc(Args &&... ConstructorArgs)
char * allocUnalignedBuffer(size_t Size)
T * allocArray(size_t Count)
DEMANGLE_ABI SymbolNode * parse(std::string_view &MangledName)
@ Tail
Attemps to make calls as fast as possible while guaranteeing that tail call optimization can always b...
@ C
The default llvm calling convention, compatible with C.
@ OB
OB - OneByte - Set if this instruction has a one byte opcode.
Scope
Defines the scope in which this symbol should be visible: Default – Visible in the public interface o...
FunctionIdentifierCodeGroup
@ ConversionOperatorIdentifier
@ RttiBaseClassDescriptor
@ DynamicAtexitDestructor
@ RttiClassHierarchyDescriptor
NodeAddr< NodeBase * > Node
SmallVector< Node, 4 > NodeList
This is an optimization pass for GlobalISel generic memory operations.
FunctionAddr VTableAddr Value
@ demangle_invalid_mangled_name
DEMANGLE_ABI std::optional< size_t > getArm64ECInsertionPointInMangledName(std::string_view MangledName)
FunctionAddr VTableAddr Count
DEMANGLE_ABI char * microsoftDemangle(std::string_view mangled_name, size_t *n_read, int *status, MSDemangleFlags Flags=MSDF_None)
Demangles the Microsoft symbol pointed at by mangled_name and returns it.
@ MSDF_NoCallingConvention
FunctionAddr VTableAddr Next
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.
NodeArrayNode * Dimensions
static constexpr size_t Max
NamedIdentifierNode * Names[Max]
TypeNode * FunctionParams[Max]
size_t FunctionParamCount
IdentifierNode * Identifier
VariableSymbolNode * Variable
FunctionRefQualifier RefQualifier
CallingConv CallConvention
FunctionSignatureNode * Signature
std::array< uint64_t, NumArgs > ArgArray
static constexpr unsigned NumArgs
NodeArrayNode * Components
QualifiedNameNode * ClassParent
IdentifierNode * getUnqualifiedIdentifier()
NodeArrayNode * Components
NodeArrayNode * TargetNames
void output(OutputBuffer &OB, OutputFlags Flags) const override
std::array< int64_t, 3 > ThunkOffsets