34 return !S.empty() && std::isdigit(S.front());
43 if (!llvm::itanium_demangle::starts_with(S,
C))
50 if (!llvm::itanium_demangle::starts_with(S,
C))
52 S.remove_prefix(
C.size());
56static bool consumeFront(std::string_view &S, std::string_view PrefixA,
57 std::string_view PrefixB,
bool A) {
58 const std::string_view &Prefix =
A ? PrefixA : PrefixB;
62static bool startsWith(std::string_view S, std::string_view PrefixA,
63 std::string_view PrefixB,
bool A) {
64 const std::string_view &Prefix =
A ? PrefixA : PrefixB;
65 return llvm::itanium_demangle::starts_with(S, Prefix);
68bool Demangler::isMemberPointer(std::string_view MangledName,
bool &
Error) {
70 const char F = MangledName.front();
71 MangledName.remove_prefix(1);
97 if (MangledName[0] !=
'6' && MangledName[0] !=
'8') {
101 return (MangledName[0] ==
'8');
109 demanglePointerAuthQualifier(MangledName);
111 if (MangledName.empty()) {
117 switch (MangledName.front()) {
175 size_t End = S.find(
'?');
176 if (End == std::string_view::npos)
178 std::string_view Candidate = S.substr(0, End);
179 if (Candidate.empty())
184 if (Candidate.size() == 1)
185 return Candidate[0] ==
'@' || (Candidate[0] >=
'0' && Candidate[0] <=
'9');
188 if (Candidate.back() !=
'@')
190 Candidate.remove_suffix(1);
198 if (Candidate[0] <
'B' || Candidate[0] >
'P')
200 Candidate.remove_prefix(1);
201 while (!Candidate.empty()) {
202 if (Candidate[0] <
'A' || Candidate[0] >
'P')
204 Candidate.remove_prefix(1);
224 if (llvm::itanium_demangle::starts_with(S,
"$$Q"))
238static bool isArrayType(std::string_view S) {
return S[0] ==
'Y'; }
241 return llvm::itanium_demangle::starts_with(S,
"$$A8@@") ||
242 llvm::itanium_demangle::starts_with(S,
"$$A6");
254static std::pair<Qualifiers, PointerAffinity>
259 const char F = MangledName.front();
260 MangledName.remove_prefix(1);
284 for (
size_t I = 0;
I <
Count; ++
I) {
285 N->Nodes[
I] = Head->
N;
291std::string_view Demangler::copyString(std::string_view Borrowed) {
296 std::memcpy(Stable, Borrowed.data(), Borrowed.size());
298 return {Stable, Borrowed.size()};
302Demangler::demangleSpecialTableSymbolNode(std::string_view &MangledName,
304 NamedIdentifierNode *NI = Arena.
alloc<NamedIdentifierNode>();
307 NI->
Name =
"`vftable'";
310 NI->
Name =
"`vbtable'";
313 NI->
Name =
"`local vftable'";
316 NI->
Name =
"`RTTI Complete Object Locator'";
321 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, NI);
322 SpecialTableSymbolNode *STSN = Arena.
alloc<SpecialTableSymbolNode>();
325 if (MangledName.empty()) {
329 char Front = MangledName.front();
330 MangledName.remove_prefix(1);
331 if (Front !=
'6' && Front !=
'7') {
336 std::tie(STSN->
Quals, IsMember) = demangleQualifiers(MangledName);
350 TargetCurrent =
Next;
351 QualifiedNameNode *QN = demangleFullyQualifiedTypeName(MangledName);
355 TargetCurrent->
N = QN;
365Demangler::demangleLocalStaticGuard(std::string_view &MangledName,
367 LocalStaticGuardIdentifierNode *LSGI =
368 Arena.
alloc<LocalStaticGuardIdentifierNode>();
370 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, LSGI);
371 LocalStaticGuardVariableNode *LSGVN =
372 Arena.
alloc<LocalStaticGuardVariableNode>();
384 if (!MangledName.empty())
385 LSGI->
ScopeIndex = demangleUnsigned(MangledName);
390 std::string_view Name) {
407 std::string_view Name) {
414 std::string_view VariableName) {
423 std::string_view &MangledName,
424 std::string_view VariableName) {
426 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, NI);
427 VariableSymbolNode *VSN = Arena.
alloc<VariableSymbolNode>();
437Demangler::demangleRttiBaseClassDescriptorNode(
ArenaAllocator &Arena,
438 std::string_view &MangledName) {
439 RttiBaseClassDescriptorNode *RBCDN =
440 Arena.
alloc<RttiBaseClassDescriptorNode>();
441 RBCDN->
NVOffset = demangleUnsigned(MangledName);
444 RBCDN->
Flags = demangleUnsigned(MangledName);
448 VariableSymbolNode *VSN = Arena.
alloc<VariableSymbolNode>();
449 VSN->
Name = demangleNameScopeChain(MangledName, RBCDN);
455Demangler::demangleInitFiniStub(std::string_view &MangledName,
457 DynamicStructorIdentifierNode *DSIN =
458 Arena.
alloc<DynamicStructorIdentifierNode>();
461 bool IsKnownStaticDataMember =
false;
463 IsKnownStaticDataMember =
true;
465 SymbolNode *
Symbol = demangleDeclarator(MangledName);
469 FunctionSymbolNode *FSN =
nullptr;
478 int AtCount = IsKnownStaticDataMember ? 2 : 1;
479 for (
int I = 0;
I < AtCount; ++
I) {
486 FSN = demangleFunctionEncoding(MangledName);
490 if (IsKnownStaticDataMember) {
496 FSN =
static_cast<FunctionSymbolNode *
>(
Symbol);
504SymbolNode *Demangler::demangleSpecialIntrinsic(std::string_view &MangledName) {
511 return demangleStringLiteral(MangledName);
516 return demangleSpecialTableSymbolNode(MangledName, SIK);
518 return demangleVcallThunkNode(MangledName);
520 return demangleLocalStaticGuard(MangledName,
false);
522 return demangleLocalStaticGuard(MangledName,
true);
529 if (!MangledName.empty())
534 return demangleUntypedVariable(Arena, MangledName,
535 "`RTTI Base Class Array'");
537 return demangleUntypedVariable(Arena, MangledName,
538 "`RTTI Class Hierarchy Descriptor'");
540 return demangleRttiBaseClassDescriptorNode(Arena, MangledName);
542 return demangleInitFiniStub(MangledName,
false);
544 return demangleInitFiniStub(MangledName,
true);
558Demangler::demangleFunctionIdentifierCode(std::string_view &MangledName) {
559 assert(llvm::itanium_demangle::starts_with(MangledName,
'?'));
560 MangledName.remove_prefix(1);
561 if (MangledName.empty()) {
567 return demangleFunctionIdentifierCode(
570 return demangleFunctionIdentifierCode(MangledName,
572 return demangleFunctionIdentifierCode(MangledName,
577Demangler::demangleStructorIdentifier(std::string_view &MangledName,
579 StructorIdentifierNode *
N = Arena.
alloc<StructorIdentifierNode>();
580 N->IsDestructor = IsDestructor;
585Demangler::demangleConversionOperatorIdentifier(std::string_view &MangledName) {
586 ConversionOperatorIdentifierNode *
N =
587 Arena.
alloc<ConversionOperatorIdentifierNode>();
592Demangler::demangleLiteralOperatorIdentifier(std::string_view &MangledName) {
593 LiteralOperatorIdentifierNode *
N =
594 Arena.
alloc<LiteralOperatorIdentifierNode>();
595 N->Name = demangleSimpleString(MangledName,
false);
600Demangler::translateIntrinsicFunctionCode(
char CH,
603 if (!(
CH >=
'0' &&
CH <=
'9') && !(
CH >=
'A' &&
CH <=
'Z')) {
611 static IFK
Basic[36] = {
637 IFK::GreaterThanEqual,
649 static IFK
Under[36] = {
654 IFK::BitwiseAndEqual,
656 IFK::BitwiseXorEqual,
665 IFK::DefaultCtorClosure,
669 IFK::VecVbaseCtorIter,
673 IFK::EHVecVbaseCtorIter,
674 IFK::CopyCtorClosure,
679 IFK::LocalVftableCtorClosure,
698 IFK::ManVectorCtorIter,
699 IFK::ManVectorDtorIter,
700 IFK::EHVectorCopyCtorIter,
701 IFK::EHVectorVbaseCopyCtorIter,
704 IFK::VectorCopyCtorIter,
705 IFK::VectorVbaseCopyCtorIter,
706 IFK::ManVectorVbaseCopyCtorIter,
727 int Index = (
CH >=
'0' &&
CH <=
'9') ? (
CH -
'0') : (
CH -
'A' + 10);
740Demangler::demangleFunctionIdentifierCode(std::string_view &MangledName,
742 if (MangledName.empty()) {
746 const char CH = MangledName.front();
749 MangledName.remove_prefix(1);
753 return demangleStructorIdentifier(MangledName,
CH ==
'1');
755 return demangleConversionOperatorIdentifier(MangledName);
757 return Arena.
alloc<IntrinsicFunctionIdentifierNode>(
758 translateIntrinsicFunctionCode(
CH, Group));
761 MangledName.remove_prefix(1);
762 return Arena.
alloc<IntrinsicFunctionIdentifierNode>(
763 translateIntrinsicFunctionCode(
CH, Group));
765 MangledName.remove_prefix(1);
768 return demangleLiteralOperatorIdentifier(MangledName);
770 return Arena.
alloc<IntrinsicFunctionIdentifierNode>(
771 translateIntrinsicFunctionCode(
CH, Group));
778SymbolNode *Demangler::demangleEncodedSymbol(std::string_view &MangledName,
780 if (MangledName.empty()) {
786 switch (MangledName.front()) {
792 StorageClass SC = demangleVariableStorageClass(MangledName);
793 return demangleVariableEncoding(MangledName, SC);
796 FunctionSymbolNode *FSN = demangleFunctionEncoding(MangledName);
798 IdentifierNode *UQN =
Name->getUnqualifiedIdentifier();
800 ConversionOperatorIdentifierNode *COIN =
801 static_cast<ConversionOperatorIdentifierNode *
>(UQN);
808SymbolNode *Demangler::demangleDeclarator(std::string_view &MangledName) {
811 QualifiedNameNode *QN = demangleFullyQualifiedSymbolName(MangledName);
815 SymbolNode *
Symbol = demangleEncodedSymbol(MangledName, QN);
822 ConversionOperatorIdentifierNode *COIN =
823 static_cast<ConversionOperatorIdentifierNode *
>(UQN);
832SymbolNode *Demangler::demangleMD5Name(std::string_view &MangledName) {
833 assert(llvm::itanium_demangle::starts_with(MangledName,
"??@"));
837 size_t MD5Last = MangledName.find(
'@', strlen(
"??@"));
838 if (MD5Last == std::string_view::npos) {
842 const char *
Start = MangledName.data();
843 const size_t StartSize = MangledName.size();
844 MangledName.remove_prefix(MD5Last + 1);
858 assert(MangledName.size() < StartSize);
859 const size_t Count = StartSize - MangledName.size();
867SymbolNode *Demangler::demangleTypeinfoName(std::string_view &MangledName) {
868 assert(llvm::itanium_demangle::starts_with(MangledName,
'.'));
872 if (
Error || !MangledName.empty()) {
884 if (llvm::itanium_demangle::starts_with(MangledName,
'.'))
885 return demangleTypeinfoName(MangledName);
887 if (llvm::itanium_demangle::starts_with(MangledName,
"??@"))
888 return demangleMD5Name(MangledName);
891 if (!llvm::itanium_demangle::starts_with(MangledName,
'?')) {
900 if (
SymbolNode *
SI = demangleSpecialIntrinsic(MangledName))
903 return demangleDeclarator(MangledName);
906TagTypeNode *Demangler::parseTagUniqueName(std::string_view &MangledName) {
912 if (MangledName.empty()) {
917 return demangleClassType(MangledName);
928Demangler::demangleVariableEncoding(std::string_view &MangledName,
946 demanglePointerExtQualifiers(MangledName));
948 bool IsMember =
false;
949 std::tie(ExtraChildQuals, IsMember) = demangleQualifiers(MangledName);
953 demangleFullyQualifiedTypeName(MangledName);
961 VSN->
Type->
Quals = demangleQualifiers(MangledName).first;
979std::pair<uint64_t, bool>
980Demangler::demangleNumber(std::string_view &MangledName) {
984 uint64_t
Ret = MangledName[0] -
'0' + 1;
985 MangledName.remove_prefix(1);
986 return {
Ret, IsNegative};
990 for (
size_t i = 0; i < MangledName.size(); ++i) {
991 char C = MangledName[i];
993 MangledName.remove_prefix(i + 1);
994 return {
Ret, IsNegative};
996 if (
'A' <=
C &&
C <=
'P') {
1004 return {0ULL,
false};
1007uint64_t Demangler::demangleUnsigned(std::string_view &MangledName) {
1008 bool IsNegative =
false;
1010 std::tie(
Number, IsNegative) = demangleNumber(MangledName);
1016int64_t Demangler::demangleSigned(std::string_view &MangledName) {
1017 bool IsNegative =
false;
1019 std::tie(
Number, IsNegative) = demangleNumber(MangledName);
1022 int64_t
I =
static_cast<int64_t
>(
Number);
1023 return IsNegative ? -
I :
I;
1028void Demangler::memorizeString(std::string_view S) {
1031 for (
size_t i = 0; i < Backrefs.
NamesCount; ++i)
1034 NamedIdentifierNode *
N = Arena.
alloc<NamedIdentifierNode>();
1040Demangler::demangleBackRefName(std::string_view &MangledName) {
1043 size_t I = MangledName[0] -
'0';
1049 MangledName.remove_prefix(1);
1050 return Backrefs.
Names[
I];
1058 std::string_view Owned = copyString(OB);
1059 memorizeString(Owned);
1060 std::free(
OB.getBuffer());
1064Demangler::demangleTemplateInstantiationName(std::string_view &MangledName,
1066 assert(llvm::itanium_demangle::starts_with(MangledName,
"?$"));
1069 BackrefContext OuterContext;
1073 demangleUnqualifiedSymbolName(MangledName,
NBB_Simple);
1075 Identifier->TemplateParams = demangleTemplateParameterList(MangledName);
1091 memorizeIdentifier(Identifier);
1098Demangler::demangleSimpleName(std::string_view &MangledName,
bool Memorize) {
1099 std::string_view S = demangleSimpleString(MangledName, Memorize);
1103 NamedIdentifierNode *
Name = Arena.
alloc<NamedIdentifierNode>();
1112 return (
C <=
'J') ? (
C -
'A') : (10 +
C -
'K');
1115uint8_t Demangler::demangleCharLiteral(std::string_view &MangledName) {
1116 assert(!MangledName.empty());
1117 if (!llvm::itanium_demangle::starts_with(MangledName,
'?')) {
1118 const uint8_t
F = MangledName.front();
1119 MangledName.remove_prefix(1);
1123 MangledName.remove_prefix(1);
1124 if (MangledName.empty())
1125 goto CharLiteralError;
1129 if (MangledName.size() < 2)
1130 goto CharLiteralError;
1131 std::string_view Nibbles = MangledName.substr(0, 2);
1133 goto CharLiteralError;
1137 MangledName.remove_prefix(2);
1138 return (C1 << 4) | C2;
1142 const char *
Lookup =
",/\\:. \n\t'-";
1143 char C =
Lookup[MangledName[0] -
'0'];
1144 MangledName.remove_prefix(1);
1148 if (MangledName[0] >=
'a' && MangledName[0] <=
'z') {
1149 char Lookup[26] = {
'\xE1',
'\xE2',
'\xE3',
'\xE4',
'\xE5',
'\xE6',
'\xE7',
1150 '\xE8',
'\xE9',
'\xEA',
'\xEB',
'\xEC',
'\xED',
'\xEE',
1151 '\xEF',
'\xF0',
'\xF1',
'\xF2',
'\xF3',
'\xF4',
'\xF5',
1152 '\xF6',
'\xF7',
'\xF8',
'\xF9',
'\xFA'};
1153 char C =
Lookup[MangledName[0] -
'a'];
1154 MangledName.remove_prefix(1);
1158 if (MangledName[0] >=
'A' && MangledName[0] <=
'Z') {
1159 char Lookup[26] = {
'\xC1',
'\xC2',
'\xC3',
'\xC4',
'\xC5',
'\xC6',
'\xC7',
1160 '\xC8',
'\xC9',
'\xCA',
'\xCB',
'\xCC',
'\xCD',
'\xCE',
1161 '\xCF',
'\xD0',
'\xD1',
'\xD2',
'\xD3',
'\xD4',
'\xD5',
1162 '\xD6',
'\xD7',
'\xD8',
'\xD9',
'\xDA'};
1163 char C =
Lookup[MangledName[0] -
'A'];
1164 MangledName.remove_prefix(1);
1173wchar_t Demangler::demangleWcharLiteral(std::string_view &MangledName) {
1176 C1 = demangleCharLiteral(MangledName);
1177 if (
Error || MangledName.empty())
1178 goto WCharLiteralError;
1179 C2 = demangleCharLiteral(MangledName);
1181 goto WCharLiteralError;
1183 return ((
wchar_t)C1 << 8) | (wchar_t)C2;
1192 *Buffer = (Digit < 10) ? (
'0' + Digit) : (
'A' + Digit - 10);
1203 char TempBuffer[17];
1205 ::memset(TempBuffer, 0,
sizeof(TempBuffer));
1206 constexpr int MaxPos =
sizeof(TempBuffer) - 1;
1208 int Pos = MaxPos - 1;
1210 for (
int I = 0;
I < 2; ++
I) {
1215 TempBuffer[Pos--] =
'x';
1217 TempBuffer[Pos--] =
'\\';
1218 OB << std::string_view(&TempBuffer[Pos + 1]);
1260 if (
C > 0x1F &&
C < 0x7F) {
1272 while (
Length > 0 && *End == 0) {
1284 if (*StringBytes++ == 0)
1298 if (NumBytes % 2 == 1)
1304 if (NumBytes < 32) {
1306 if (TrailingNulls >= 4 && NumBytes % 4 == 0)
1308 if (TrailingNulls >= 2)
1320 if (Nulls >= 2 * NumChars / 3 && NumBytes % 4 == 0)
1322 if (Nulls >= NumChars / 3)
1328 unsigned CharIndex,
unsigned CharBytes) {
1329 assert(CharBytes == 1 || CharBytes == 2 || CharBytes == 4);
1330 unsigned Offset = CharIndex * CharBytes;
1332 StringBytes = StringBytes +
Offset;
1333 for (
unsigned I = 0;
I < CharBytes; ++
I) {
1334 unsigned C =
static_cast<unsigned>(StringBytes[
I]);
1341Demangler::demangleVcallThunkNode(std::string_view &MangledName) {
1342 FunctionSymbolNode *FSN = Arena.
alloc<FunctionSymbolNode>();
1343 VcallThunkIdentifierNode *VTIN = Arena.
alloc<VcallThunkIdentifierNode>();
1347 FSN->
Name = demangleNameScopeChain(MangledName, VTIN);
1351 VTIN->OffsetInVTable = demangleUnsigned(MangledName);
1356 return (
Error) ? nullptr : FSN;
1360Demangler::demangleStringLiteral(std::string_view &MangledName) {
1363 std::string_view CRC;
1364 uint64_t StringByteSize;
1365 bool IsWcharT =
false;
1366 bool IsNegative =
false;
1367 size_t CrcEndPos = 0;
1370 EncodedStringLiteralNode *
Result = Arena.
alloc<EncodedStringLiteralNode>();
1374 goto StringLiteralError;
1375 if (MangledName.empty())
1376 goto StringLiteralError;
1379 F = MangledName.front();
1380 MangledName.remove_prefix(1);
1388 goto StringLiteralError;
1392 std::tie(StringByteSize, IsNegative) = demangleNumber(MangledName);
1393 if (
Error || IsNegative || StringByteSize < (IsWcharT ? 2 : 1))
1394 goto StringLiteralError;
1397 CrcEndPos = MangledName.find(
'@');
1398 if (CrcEndPos == std::string_view::npos)
1399 goto StringLiteralError;
1400 CRC = MangledName.substr(0, CrcEndPos);
1401 MangledName.remove_prefix(CrcEndPos + 1);
1402 if (MangledName.empty())
1403 goto StringLiteralError;
1407 if (StringByteSize > 64)
1408 Result->IsTruncated =
true;
1412 if (StringByteSize % 2 != 0)
1413 goto StringLiteralError;
1414 if (StringByteSize == 0)
1415 goto StringLiteralError;
1416 if (MangledName.size() < 2)
1417 goto StringLiteralError;
1418 wchar_t W = demangleWcharLiteral(MangledName);
1419 if (StringByteSize != 2 ||
Result->IsTruncated)
1421 StringByteSize -= 2;
1423 goto StringLiteralError;
1428 constexpr unsigned MaxStringByteLength = 32 * 4;
1429 uint8_t StringBytes[MaxStringByteLength];
1431 unsigned BytesDecoded = 0;
1433 if (MangledName.size() < 1 || BytesDecoded >= MaxStringByteLength)
1434 goto StringLiteralError;
1435 StringBytes[BytesDecoded++] = demangleCharLiteral(MangledName);
1438 if (StringByteSize > BytesDecoded)
1439 Result->IsTruncated =
true;
1441 unsigned CharBytes =
1443 assert(StringByteSize % CharBytes == 0);
1444 switch (CharBytes) {
1457 const unsigned NumChars = BytesDecoded / CharBytes;
1458 for (
unsigned CharIndex = 0; CharIndex < NumChars; ++CharIndex) {
1461 if (CharIndex + 1 < NumChars ||
Result->IsTruncated)
1466 Result->DecodedString = copyString(OB);
1467 std::free(
OB.getBuffer());
1472 std::free(
OB.getBuffer());
1478std::string_view Demangler::demangleSimpleString(std::string_view &MangledName,
1481 for (
size_t i = 0; i < MangledName.size(); ++i) {
1482 if (MangledName[i] !=
'@')
1486 S = MangledName.substr(0, i);
1487 MangledName.remove_prefix(i + 1);
1499Demangler::demangleAnonymousNamespaceName(std::string_view &MangledName) {
1500 assert(llvm::itanium_demangle::starts_with(MangledName,
"?A"));
1503 NamedIdentifierNode *
Node = Arena.
alloc<NamedIdentifierNode>();
1504 Node->Name =
"`anonymous namespace'";
1505 size_t EndPos = MangledName.find(
'@');
1506 if (EndPos == std::string_view::npos) {
1510 std::string_view NamespaceKey = MangledName.substr(0, EndPos);
1511 memorizeString(NamespaceKey);
1512 MangledName = MangledName.substr(EndPos + 1);
1517Demangler::demangleLocallyScopedNamePiece(std::string_view &MangledName) {
1523 bool IsNegative =
false;
1524 std::tie(
Number, IsNegative) = demangleNumber(MangledName);
1543 std::free(
OB.getBuffer());
1549Demangler::demangleFullyQualifiedTypeName(std::string_view &MangledName) {
1551 demangleUnqualifiedTypeName(MangledName,
true);
1556 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, Identifier);
1567Demangler::demangleFullyQualifiedSymbolName(std::string_view &MangledName) {
1574 demangleUnqualifiedSymbolName(MangledName,
NBB_Simple);
1578 QualifiedNameNode *QN = demangleNameScopeChain(MangledName, Identifier);
1587 StructorIdentifierNode *SIN =
1588 static_cast<StructorIdentifierNode *
>(
Identifier);
1590 SIN->
Class =
static_cast<IdentifierNode *
>(ClassNode);
1597Demangler::demangleUnqualifiedTypeName(std::string_view &MangledName,
1604 return demangleBackRefName(MangledName);
1606 if (llvm::itanium_demangle::starts_with(MangledName,
"?$"))
1607 return demangleTemplateInstantiationName(MangledName,
NBB_Template);
1609 return demangleSimpleName(MangledName, Memorize);
1613Demangler::demangleUnqualifiedSymbolName(std::string_view &MangledName,
1616 return demangleBackRefName(MangledName);
1617 if (llvm::itanium_demangle::starts_with(MangledName,
"?$"))
1618 return demangleTemplateInstantiationName(MangledName, NBB);
1619 if (llvm::itanium_demangle::starts_with(MangledName,
'?'))
1620 return demangleFunctionIdentifierCode(MangledName);
1621 return demangleSimpleName(MangledName, (NBB &
NBB_Simple) != 0);
1625Demangler::demangleNameScopePiece(std::string_view &MangledName) {
1627 return demangleBackRefName(MangledName);
1629 if (llvm::itanium_demangle::starts_with(MangledName,
"?$"))
1630 return demangleTemplateInstantiationName(MangledName,
NBB_Template);
1632 if (llvm::itanium_demangle::starts_with(MangledName,
"?A"))
1633 return demangleAnonymousNamespaceName(MangledName);
1636 return demangleLocallyScopedNamePiece(MangledName);
1638 return demangleSimpleName(MangledName,
true);
1642Demangler::demangleNameScopeChain(std::string_view &MangledName,
1646 Head->
N = UnqualifiedName;
1652 NewHead->
Next = Head;
1655 if (MangledName.empty()) {
1661 IdentifierNode *Elem = demangleNameScopePiece(MangledName);
1668 QualifiedNameNode *QN = Arena.
alloc<QualifiedNameNode>();
1673FuncClass Demangler::demangleFunctionClass(std::string_view &MangledName) {
1674 const char F = MangledName.front();
1675 MangledName.remove_prefix(1);
1735 if (MangledName.empty())
1737 const char F = MangledName.front();
1738 MangledName.remove_prefix(1);
1761Demangler::demangleCallingConvention(std::string_view &MangledName) {
1762 if (MangledName.empty()) {
1767 const char F = MangledName.front();
1768 MangledName.remove_prefix(1);
1803Demangler::demangleVariableStorageClass(std::string_view &MangledName) {
1804 assert(MangledName.front() >=
'0' && MangledName.front() <=
'4');
1806 const char F = MangledName.front();
1807 MangledName.remove_prefix(1);
1823std::pair<Qualifiers, bool>
1824Demangler::demangleQualifiers(std::string_view &MangledName) {
1825 if (MangledName.empty()) {
1827 return std::make_pair(
Q_None,
false);
1830 const char F = MangledName.front();
1831 MangledName.remove_prefix(1);
1835 return std::make_pair(
Q_None,
true);
1837 return std::make_pair(
Q_Const,
true);
1844 return std::make_pair(
Q_None,
false);
1846 return std::make_pair(
Q_Const,
false);
1853 return std::make_pair(
Q_None,
false);
1858TypeNode *Demangler::demangleType(std::string_view &MangledName,
1863 std::tie(Quals, IsMember) = demangleQualifiers(MangledName);
1866 std::tie(Quals, IsMember) = demangleQualifiers(MangledName);
1869 if (MangledName.empty()) {
1874 TypeNode *Ty =
nullptr;
1876 Ty = demangleClassType(MangledName);
1878 if (isMemberPointer(MangledName,
Error))
1879 Ty = demangleMemberPointerType(MangledName);
1881 Ty = demanglePointerType(MangledName);
1885 Ty = demangleArrayType(MangledName);
1888 Ty = demangleFunctionType(MangledName,
true);
1890 assert(llvm::itanium_demangle::starts_with(MangledName,
"$$A6"));
1892 Ty = demangleFunctionType(MangledName,
false);
1895 Ty = demangleCustomType(MangledName);
1897 Ty = demanglePrimitiveType(MangledName);
1906bool Demangler::demangleThrowSpecification(std::string_view &MangledName) {
1917Demangler::demangleFunctionType(std::string_view &MangledName,
1918 bool HasThisQuals) {
1919 FunctionSignatureNode *FTy = Arena.
alloc<FunctionSignatureNode>();
1922 FTy->
Quals = demanglePointerExtQualifiers(MangledName);
1938 FTy->
IsNoexcept = demangleThrowSpecification(MangledName);
1944Demangler::demangleFunctionEncoding(std::string_view &MangledName) {
1949 if (MangledName.empty()) {
1954 FuncClass FC = demangleFunctionClass(MangledName);
1957 FunctionSignatureNode *FSN =
nullptr;
1958 ThunkSignatureNode *TTN =
nullptr;
1960 TTN = Arena.
alloc<ThunkSignatureNode>();
1963 TTN = Arena.
alloc<ThunkSignatureNode>();
1976 FSN = Arena.
alloc<FunctionSignatureNode>();
1979 FSN = demangleFunctionType(MangledName, HasThisQuals);
1986 *
static_cast<FunctionSignatureNode *
>(TTN) = *FSN;
1991 FunctionSymbolNode *
Symbol = Arena.
alloc<FunctionSymbolNode>();
1996CustomTypeNode *Demangler::demangleCustomType(std::string_view &MangledName) {
1997 assert(llvm::itanium_demangle::starts_with(MangledName,
'?'));
1998 MangledName.remove_prefix(1);
2000 CustomTypeNode *CTN = Arena.
alloc<CustomTypeNode>();
2001 CTN->
Identifier = demangleUnqualifiedTypeName(MangledName,
true);
2011Demangler::demanglePrimitiveType(std::string_view &MangledName) {
2015 const char F = MangledName.front();
2016 MangledName.remove_prefix(1);
2045 if (MangledName.empty()) {
2049 const char F = MangledName.front();
2050 MangledName.remove_prefix(1);
2078TagTypeNode *Demangler::demangleClassType(std::string_view &MangledName) {
2079 TagTypeNode *
TT =
nullptr;
2081 const char F = MangledName.front();
2082 MangledName.remove_prefix(1);
2104 TT->QualifiedName = demangleFullyQualifiedTypeName(MangledName);
2110PointerTypeNode *Demangler::demanglePointerType(std::string_view &MangledName) {
2111 PointerTypeNode *
Pointer = Arena.
alloc<PointerTypeNode>();
2117 Pointer->Pointee = demangleFunctionType(MangledName,
false);
2121 Qualifiers ExtQuals = demanglePointerExtQualifiers(MangledName);
2124 Pointer->PointerAuthQualifier = createPointerAuthQualifier(MangledName);
2131Demangler::demangleMemberPointerType(std::string_view &MangledName) {
2132 PointerTypeNode *
Pointer = Arena.
alloc<PointerTypeNode>();
2138 Qualifiers ExtQuals = demanglePointerExtQualifiers(MangledName);
2144 Pointer->ClassParent = demangleFullyQualifiedTypeName(MangledName);
2145 Pointer->Pointee = demangleFunctionType(MangledName,
true);
2149 std::tie(PointeeQuals, IsMember) = demangleQualifiers(MangledName);
2151 Pointer->ClassParent = demangleFullyQualifiedTypeName(MangledName);
2155 Pointer->Pointee->Quals = PointeeQuals;
2162Demangler::demanglePointerExtQualifiers(std::string_view &MangledName) {
2174std::optional<PointerAuthQualifierNode::ArgArray>
2175Demangler::demanglePointerAuthQualifier(std::string_view &MangledName) {
2177 return std::nullopt;
2182 for (
unsigned I = 0;
I < NumArgs; ++
I) {
2183 bool IsNegative =
false;
2185 std::tie(
Value, IsNegative) = demangleNumber(MangledName);
2187 return std::nullopt;
2196Demangler::createPointerAuthQualifier(std::string_view &MangledName) {
2198 std::optional<PointerAuthQualifierNode::ArgArray> Vals =
2199 demanglePointerAuthQualifier(MangledName);
2204 PointerAuthQualifierNode *PtrAuthQual =
2205 Arena.
alloc<PointerAuthQualifierNode>();
2206 NodeArrayNode *
Array = Arena.
alloc<NodeArrayNode>();
2208 Array->Count = NumArgs;
2211 for (
unsigned I = 0;
I < NumArgs; ++
I)
2212 Array->Nodes[
I] = Arena.
alloc<IntegerLiteralNode>((*Vals)[
I],
false);
2217ArrayTypeNode *Demangler::demangleArrayType(std::string_view &MangledName) {
2218 assert(MangledName.front() ==
'Y');
2219 MangledName.remove_prefix(1);
2222 bool IsNegative =
false;
2223 std::tie(Rank, IsNegative) = demangleNumber(MangledName);
2224 if (IsNegative || Rank == 0) {
2229 ArrayTypeNode *ATy = Arena.
alloc<ArrayTypeNode>();
2233 for (uint64_t
I = 0;
I < Rank; ++
I) {
2235 std::tie(
D, IsNegative) = demangleNumber(MangledName);
2236 if (
Error || IsNegative) {
2240 Tail->N = Arena.
alloc<IntegerLiteralNode>(
D, IsNegative);
2250 std::tie(ATy->
Quals, IsMember) = demangleQualifiers(MangledName);
2263Demangler::demangleFunctionParameterList(std::string_view &MangledName,
2272 while (!
Error && !llvm::itanium_demangle::starts_with(MangledName,
'@') &&
2273 !llvm::itanium_demangle::starts_with(MangledName,
'Z')) {
2277 size_t N = MangledName[0] -
'0';
2282 MangledName.remove_prefix(1);
2286 Current = &(*Current)->Next;
2290 size_t OldSize = MangledName.size();
2299 size_t CharsConsumed = OldSize - MangledName.size();
2300 assert(CharsConsumed != 0);
2307 Current = &(*Current)->Next;
2329Demangler::demangleTemplateParameterList(std::string_view &MangledName) {
2334 while (!llvm::itanium_demangle::starts_with(MangledName,
'@')) {
2349 const bool IsAutoNTTP =
consumeFront(MangledName,
"$M");
2360 TemplateParameterReferenceNode *TPRN =
nullptr;
2363 TP.N = demangleFullyQualifiedTypeName(MangledName);
2370 }
else if (
startsWith(MangledName,
"$1",
"1", !IsAutoNTTP) ||
2371 startsWith(MangledName,
"$H",
"H", !IsAutoNTTP) ||
2372 startsWith(MangledName,
"$I",
"I", !IsAutoNTTP) ||
2373 startsWith(MangledName,
"$J",
"J", !IsAutoNTTP)) {
2375 TP.N = TPRN = Arena.
alloc<TemplateParameterReferenceNode>();
2379 MangledName.remove_prefix(1);
2385 char InheritanceSpecifier = MangledName.front();
2386 MangledName.remove_prefix(1);
2387 SymbolNode *S =
nullptr;
2388 if (llvm::itanium_demangle::starts_with(MangledName,
'?')) {
2389 S =
parse(MangledName);
2397 switch (InheritanceSpecifier) {
2400 demangleSigned(MangledName);
2404 demangleSigned(MangledName);
2408 demangleSigned(MangledName);
2417 }
else if (llvm::itanium_demangle::starts_with(MangledName,
"$E?")) {
2420 TP.N = TPRN = Arena.
alloc<TemplateParameterReferenceNode>();
2423 }
else if (
startsWith(MangledName,
"$F",
"F", !IsAutoNTTP) ||
2424 startsWith(MangledName,
"$G",
"G", !IsAutoNTTP)) {
2425 TP.N = TPRN = Arena.
alloc<TemplateParameterReferenceNode>();
2429 MangledName.remove_prefix(1);
2430 char InheritanceSpecifier = MangledName.front();
2431 MangledName.remove_prefix(1);
2433 switch (InheritanceSpecifier) {
2436 demangleSigned(MangledName);
2440 demangleSigned(MangledName);
2442 demangleSigned(MangledName);
2449 }
else if (
consumeFront(MangledName,
"$0",
"0", !IsAutoNTTP)) {
2451 bool IsNegative =
false;
2453 std::tie(
Value, IsNegative) = demangleNumber(MangledName);
2455 TP.N = Arena.
alloc<IntegerLiteralNode>(
Value, IsNegative);
2470 assert(llvm::itanium_demangle::starts_with(
2476void Demangler::dumpBackReferences() {
2477 std::printf(
"%d function parameter backreferences\n",
2478 (
int)Backrefs.FunctionParamCount);
2482 for (
size_t I = 0;
I < Backrefs.FunctionParamCount; ++
I) {
2483 OB.setCurrentPosition(0);
2488 std::string_view
B = OB;
2489 std::printf(
" [%d] - %.*s\n", (
int)
I, (
int)
B.size(),
B.data());
2491 std::free(OB.getBuffer());
2493 if (Backrefs.FunctionParamCount > 0)
2495 std::printf(
"%d name backreferences\n", (
int)Backrefs.NamesCount);
2496 for (
size_t I = 0;
I < Backrefs.NamesCount; ++
I) {
2497 std::printf(
" [%d] - %.*s\n", (
int)
I, (
int)Backrefs.Names[
I]->Name.size(),
2498 Backrefs.Names[
I]->Name.data());
2500 if (Backrefs.NamesCount > 0)
2504std::optional<size_t>
2506 std::string_view ProcessedName{MangledName};
2510 return std::nullopt;
2515 D.demangleFullyQualifiedSymbolName(ProcessedName);
2517 return std::nullopt;
2519 return MangledName.length() - ProcessedName.length();
2526 std::string_view Name{MangledName};
2528 if (!
D.Error && NMangled)
2529 *NMangled = MangledName.size() - Name.size();
2532 D.dumpBackReferences();
2554 Buf = OB.getBuffer();
2558 *
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