HELLO·Android
系统源代码
IT资讯
技术文章
我的收藏
注册
登录
-
我收藏的文章
创建代码块
我的代码块
我的账号
Pie
|
9.0.0_r8
下载
查看原文件
收藏
根目录
external
libcxxabi
src
cxa_demangle.cpp
//===-------------------------- cxa_demangle.cpp --------------------------===// // // The LLVM Compiler Infrastructure // // This file is dual licensed under the MIT and the University of Illinois Open // Source Licenses. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // FIXME: (possibly) incomplete list of features that clang mangles that this // file does not yet support: // - enable_if attribute // - C++ modules TS // - All C++14 and C++17 features #define _LIBCPP_NO_EXCEPTIONS #include "__cxxabi_config.h" #include
#include
#include
#include
#include
#include
#include
#include
#ifdef _MSC_VER // snprintf is implemented in VS 2015 #if _MSC_VER < 1900 #define snprintf _snprintf_s #endif #endif #ifndef NDEBUG #if __has_attribute(noinline) && __has_attribute(used) #define DUMP_METHOD __attribute__((noinline,used)) #else #define DUMP_METHOD #endif #endif namespace { class StringView { const char *First; const char *Last; public: template
StringView(const char (&Str)[N]) : First(Str), Last(Str + N - 1) {} StringView(const char *First_, const char *Last_) : First(First_), Last(Last_) {} StringView() : First(nullptr), Last(nullptr) {} StringView substr(size_t From, size_t To) { if (To >= size()) To = size() - 1; if (From >= size()) From = size() - 1; return StringView(First + From, First + To); } StringView dropFront(size_t N) const { if (N >= size()) N = size() - 1; return StringView(First + N, Last); } bool startsWith(StringView Str) const { if (Str.size() > size()) return false; return std::equal(Str.begin(), Str.end(), begin()); } const char &operator[](size_t Idx) const { return *(begin() + Idx); } const char *begin() const { return First; } const char *end() const { return Last; } size_t size() const { return static_cast
(Last - First); } bool empty() const { return First == Last; } }; bool operator==(const StringView &LHS, const StringView &RHS) { return LHS.size() == RHS.size() && std::equal(LHS.begin(), LHS.end(), RHS.begin()); } // Stream that AST nodes write their string representation into after the AST // has been parsed. class OutputStream { char *Buffer; size_t CurrentPosition; size_t BufferCapacity; // Ensure there is at least n more positions in buffer. void grow(size_t N) { if (N + CurrentPosition >= BufferCapacity) { BufferCapacity *= 2; if (BufferCapacity < N + CurrentPosition) BufferCapacity = N + CurrentPosition; Buffer = static_cast
(std::realloc(Buffer, BufferCapacity)); } } public: OutputStream(char *StartBuf, size_t Size) : Buffer(StartBuf), CurrentPosition(0), BufferCapacity(Size) {} /// If a ParameterPackExpansion (or similar type) is encountered, the offset /// into the pack that we're currently printing. unsigned CurrentPackIndex = std::numeric_limits
::max(); OutputStream &operator+=(StringView R) { size_t Size = R.size(); if (Size == 0) return *this; grow(Size); memmove(Buffer + CurrentPosition, R.begin(), Size); CurrentPosition += Size; return *this; } OutputStream &operator+=(char C) { grow(1); Buffer[CurrentPosition++] = C; return *this; } size_t getCurrentPosition() const { return CurrentPosition; }; char back() const { return CurrentPosition ? Buffer[CurrentPosition - 1] : '\0'; } bool empty() const { return CurrentPosition == 0; } char *getBuffer() { return Buffer; } char *getBufferEnd() { return Buffer + CurrentPosition - 1; } size_t getBufferCapacity() { return BufferCapacity; } }; template
class SwapAndRestore { T &Restore; T OriginalValue; public: SwapAndRestore(T& Restore_, T NewVal) : Restore(Restore_), OriginalValue(Restore) { Restore = std::move(NewVal); } ~SwapAndRestore() { Restore = std::move(OriginalValue); } SwapAndRestore(const SwapAndRestore &) = delete; SwapAndRestore &operator=(const SwapAndRestore &) = delete; }; // Base class of all AST nodes. The AST is built by the parser, then is // traversed by the printLeft/Right functions to produce a demangled string. class Node { public: enum Kind : unsigned char { KDotSuffix, KVendorExtQualType, KQualType, KConversionOperatorType, KPostfixQualifiedType, KNameType, KAbiTagAttr, KObjCProtoName, KPointerType, KLValueReferenceType, KRValueReferenceType, KPointerToMemberType, KArrayType, KFunctionType, KFunctionEncoding, KFunctionQualType, KFunctionRefQualType, KLiteralOperator, KSpecialName, KCtorVtableSpecialName, KQualifiedName, KEmptyName, KVectorType, KParameterPack, KTemplateArgumentPack, KParameterPackExpansion, KTemplateArgs, KNameWithTemplateArgs, KGlobalQualifiedName, KStdQualifiedName, KExpandedSpecialSubstitution, KSpecialSubstitution, KCtorDtorName, KDtorName, KUnnamedTypeName, KLambdaTypeName, KExpr, }; static constexpr unsigned NoParameterPack = std::numeric_limits
::max(); unsigned ParameterPackSize = NoParameterPack; Kind K; /// Three-way bool to track a cached value. Unknown is possible if this node /// has an unexpanded parameter pack below it that may affect this cache. enum class Cache : unsigned char { Yes, No, Unknown, }; /// Tracks if this node has a component on its right side, in which case we /// need to call printRight. Cache RHSComponentCache; /// Track if this node is a (possibly qualified) array type. This can affect /// how we format the output string. Cache ArrayCache; /// Track if this node is a (possibly qualified) function type. This can /// affect how we format the output string. Cache FunctionCache; Node(Kind K_, unsigned ParameterPackSize_ = NoParameterPack, Cache RHSComponentCache_ = Cache::No, Cache ArrayCache_ = Cache::No, Cache FunctionCache_ = Cache::No) : ParameterPackSize(ParameterPackSize_), K(K_), RHSComponentCache(RHSComponentCache_), ArrayCache(ArrayCache_), FunctionCache(FunctionCache_) {} bool containsUnexpandedParameterPack() const { return ParameterPackSize != NoParameterPack; } bool hasRHSComponent(OutputStream &S) const { if (RHSComponentCache != Cache::Unknown) return RHSComponentCache == Cache::Yes; return hasRHSComponentSlow(S); } bool hasArray(OutputStream &S) const { if (ArrayCache != Cache::Unknown) return ArrayCache == Cache::Yes; return hasArraySlow(S); } bool hasFunction(OutputStream &S) const { if (FunctionCache != Cache::Unknown) return FunctionCache == Cache::Yes; return hasFunctionSlow(S); } Kind getKind() const { return K; } virtual bool hasRHSComponentSlow(OutputStream &) const { return false; } virtual bool hasArraySlow(OutputStream &) const { return false; } virtual bool hasFunctionSlow(OutputStream &) const { return false; } /// If this node is a pack expansion that expands to 0 elements. This can have /// an effect on how we should format the output. bool isEmptyPackExpansion() const; void print(OutputStream &S) const { printLeft(S); if (RHSComponentCache != Cache::No) printRight(S); } // Print the "left" side of this Node into OutputStream. virtual void printLeft(OutputStream &) const = 0; // Print the "right". This distinction is necessary to represent C++ types // that appear on the RHS of their subtype, such as arrays or functions. // Since most types don't have such a component, provide a default // implemenation. virtual void printRight(OutputStream &) const {} virtual StringView getBaseName() const { return StringView(); } // Silence compiler warnings, this dtor will never be called. virtual ~Node() = default; #ifndef NDEBUG DUMP_METHOD void dump() const { char *Buffer = static_cast
(std::malloc(1024)); OutputStream S(Buffer, 1024); print(S); S += '\0'; printf("Symbol dump for %p: %s\n", (const void*)this, S.getBuffer()); std::free(S.getBuffer()); } #endif }; class NodeArray { Node **Elements; size_t NumElements; public: NodeArray() : Elements(nullptr), NumElements(0) {} NodeArray(Node **Elements_, size_t NumElements_) : Elements(Elements_), NumElements(NumElements_) {} bool empty() const { return NumElements == 0; } size_t size() const { return NumElements; } Node **begin() const { return Elements; } Node **end() const { return Elements + NumElements; } Node *operator[](size_t Idx) const { return Elements[Idx]; } void printWithComma(OutputStream &S) const { bool FirstElement = true; for (size_t Idx = 0; Idx != NumElements; ++Idx) { if (Elements[Idx]->isEmptyPackExpansion()) continue; if (!FirstElement) S += ", "; FirstElement = false; Elements[Idx]->print(S); } } }; class DotSuffix final : public Node { const Node *Prefix; const StringView Suffix; public: DotSuffix(Node *Prefix_, StringView Suffix_) : Node(KDotSuffix), Prefix(Prefix_), Suffix(Suffix_) {} void printLeft(OutputStream &s) const override { Prefix->print(s); s += " ("; s += Suffix; s += ")"; } }; class VendorExtQualType final : public Node { const Node *Ty; StringView Ext; public: VendorExtQualType(Node *Ty_, StringView Ext_) : Node(KVendorExtQualType, Ty_->ParameterPackSize), Ty(Ty_), Ext(Ext_) {} void printLeft(OutputStream &S) const override { Ty->print(S); S += " "; S += Ext; } }; enum Qualifiers { QualNone = 0, QualConst = 0x1, QualVolatile = 0x2, QualRestrict = 0x4, }; void addQualifiers(Qualifiers &Q1, Qualifiers Q2) { Q1 = static_cast
(Q1 | Q2); } class QualType : public Node { protected: const Qualifiers Quals; const Node *Child; void printQuals(OutputStream &S) const { if (Quals & QualConst) S += " const"; if (Quals & QualVolatile) S += " volatile"; if (Quals & QualRestrict) S += " restrict"; } public: QualType(Node *Child_, Qualifiers Quals_) : Node(KQualType, Child_->ParameterPackSize, Child_->RHSComponentCache, Child_->ArrayCache, Child_->FunctionCache), Quals(Quals_), Child(Child_) {} bool hasRHSComponentSlow(OutputStream &S) const override { return Child->hasRHSComponent(S); } bool hasArraySlow(OutputStream &S) const override { return Child->hasArray(S); } bool hasFunctionSlow(OutputStream &S) const override { return Child->hasFunction(S); } void printLeft(OutputStream &S) const override { Child->printLeft(S); printQuals(S); } void printRight(OutputStream &S) const override { Child->printRight(S); } }; class ConversionOperatorType final : public Node { const Node *Ty; public: ConversionOperatorType(Node *Ty_) : Node(KConversionOperatorType, Ty_->ParameterPackSize), Ty(Ty_) {} void printLeft(OutputStream &S) const override { S += "operator "; Ty->print(S); } }; class PostfixQualifiedType final : public Node { const Node *Ty; const StringView Postfix; public: PostfixQualifiedType(Node *Ty_, StringView Postfix_) : Node(KPostfixQualifiedType, Ty_->ParameterPackSize), Ty(Ty_), Postfix(Postfix_) {} void printLeft(OutputStream &s) const override { Ty->printLeft(s); s += Postfix; } }; class NameType final : public Node { const StringView Name; public: NameType(StringView Name_) : Node(KNameType), Name(Name_) {} StringView getName() const { return Name; } StringView getBaseName() const override { return Name; } void printLeft(OutputStream &s) const override { s += Name; } }; class AbiTagAttr final : public Node { const Node* Base; StringView Tag; public: AbiTagAttr(const Node* Base_, StringView Tag_) : Node(KAbiTagAttr, Base_->ParameterPackSize, Base_->RHSComponentCache, Base_->ArrayCache, Base_->FunctionCache), Base(Base_), Tag(Tag_) {} void printLeft(OutputStream &S) const override { Base->printLeft(S); S += "[abi:"; S += Tag; S += "]"; } }; class ObjCProtoName : public Node { Node *Ty; StringView Protocol; friend class PointerType; public: ObjCProtoName(Node *Ty_, StringView Protocol_) : Node(KObjCProtoName), Ty(Ty_), Protocol(Protocol_) {} bool isObjCObject() const { return Ty->getKind() == KNameType && static_cast
(Ty)->getName() == "objc_object"; } void printLeft(OutputStream &S) const override { Ty->print(S); S += "<"; S += Protocol; S += ">"; } }; class PointerType final : public Node { const Node *Pointee; public: PointerType(Node *Pointee_) : Node(KPointerType, Pointee_->ParameterPackSize, Pointee_->RHSComponentCache), Pointee(Pointee_) {} bool hasRHSComponentSlow(OutputStream &S) const override { return Pointee->hasRHSComponent(S); } void printLeft(OutputStream &s) const override { // We rewrite objc_object
* into id
. if (Pointee->getKind() != KObjCProtoName || !static_cast
(Pointee)->isObjCObject()) { Pointee->printLeft(s); if (Pointee->hasArray(s)) s += " "; if (Pointee->hasArray(s) || Pointee->hasFunction(s)) s += "("; s += "*"; } else { const auto *objcProto = static_cast
(Pointee); s += "id<"; s += objcProto->Protocol; s += ">"; } } void printRight(OutputStream &s) const override { if (Pointee->getKind() != KObjCProtoName || !static_cast
(Pointee)->isObjCObject()) { if (Pointee->hasArray(s) || Pointee->hasFunction(s)) s += ")"; Pointee->printRight(s); } } }; class LValueReferenceType final : public Node { const Node *Pointee; public: LValueReferenceType(Node *Pointee_) : Node(KLValueReferenceType, Pointee_->ParameterPackSize, Pointee_->RHSComponentCache), Pointee(Pointee_) {} bool hasRHSComponentSlow(OutputStream &S) const override { return Pointee->hasRHSComponent(S); } void printLeft(OutputStream &s) const override { Pointee->printLeft(s); if (Pointee->hasArray(s)) s += " "; if (Pointee->hasArray(s) || Pointee->hasFunction(s)) s += "(&"; else s += "&"; } void printRight(OutputStream &s) const override { if (Pointee->hasArray(s) || Pointee->hasFunction(s)) s += ")"; Pointee->printRight(s); } }; class RValueReferenceType final : public Node { const Node *Pointee; public: RValueReferenceType(Node *Pointee_) : Node(KRValueReferenceType, Pointee_->ParameterPackSize, Pointee_->RHSComponentCache), Pointee(Pointee_) {} bool hasRHSComponentSlow(OutputStream &S) const override { return Pointee->hasRHSComponent(S); } void printLeft(OutputStream &s) const override { Pointee->printLeft(s); if (Pointee->hasArray(s)) s += " "; if (Pointee->hasArray(s) || Pointee->hasFunction(s)) s += "(&&"; else s += "&&"; } void printRight(OutputStream &s) const override { if (Pointee->hasArray(s) || Pointee->hasFunction(s)) s += ")"; Pointee->printRight(s); } }; class PointerToMemberType final : public Node { const Node *ClassType; const Node *MemberType; public: PointerToMemberType(Node *ClassType_, Node *MemberType_) : Node(KPointerToMemberType, std::min(MemberType_->ParameterPackSize, ClassType_->ParameterPackSize), MemberType_->RHSComponentCache), ClassType(ClassType_), MemberType(MemberType_) {} bool hasRHSComponentSlow(OutputStream &S) const override { return MemberType->hasRHSComponent(S); } void printLeft(OutputStream &s) const override { MemberType->printLeft(s); if (MemberType->hasArray(s) || MemberType->hasFunction(s)) s += "("; else s += " "; ClassType->print(s); s += "::*"; } void printRight(OutputStream &s) const override { if (MemberType->hasArray(s) || MemberType->hasFunction(s)) s += ")"; MemberType->printRight(s); } }; class NodeOrString { const void *First; const void *Second; public: /* implicit */ NodeOrString(StringView Str) { const char *FirstChar = Str.begin(); const char *SecondChar = Str.end(); if (SecondChar == nullptr) { assert(FirstChar == SecondChar); ++FirstChar, ++SecondChar; } First = static_cast
(FirstChar); Second = static_cast
(SecondChar); } /* implicit */ NodeOrString(Node *N) : First(static_cast
(N)), Second(nullptr) {} NodeOrString() : First(nullptr), Second(nullptr) {} bool isString() const { return Second && First; } bool isNode() const { return First && !Second; } bool isEmpty() const { return !First && !Second; } StringView asString() const { assert(isString()); return StringView(static_cast
(First), static_cast
(Second)); } const Node *asNode() const { assert(isNode()); return static_cast
(First); } }; class ArrayType final : public Node { Node *Base; NodeOrString Dimension; public: ArrayType(Node *Base_, NodeOrString Dimension_) : Node(KArrayType, Base_->ParameterPackSize, /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::Yes), Base(Base_), Dimension(Dimension_) { if (Dimension.isNode()) ParameterPackSize = std::min(ParameterPackSize, Dimension.asNode()->ParameterPackSize); } // Incomplete array type. ArrayType(Node *Base_) : Node(KArrayType, Base_->ParameterPackSize, /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::Yes), Base(Base_) {} bool hasRHSComponentSlow(OutputStream &) const override { return true; } bool hasArraySlow(OutputStream &) const override { return true; } void printLeft(OutputStream &S) const override { Base->printLeft(S); } void printRight(OutputStream &S) const override { if (S.back() != ']') S += " "; S += "["; if (Dimension.isString()) S += Dimension.asString(); else if (Dimension.isNode()) Dimension.asNode()->print(S); S += "]"; Base->printRight(S); } }; class FunctionType final : public Node { Node *Ret; NodeArray Params; public: FunctionType(Node *Ret_, NodeArray Params_) : Node(KFunctionType, Ret_->ParameterPackSize, /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::No, /*FunctionCache=*/Cache::Yes), Ret(Ret_), Params(Params_) { for (Node *P : Params) ParameterPackSize = std::min(ParameterPackSize, P->ParameterPackSize); } bool hasRHSComponentSlow(OutputStream &) const override { return true; } bool hasFunctionSlow(OutputStream &) const override { return true; } // Handle C++'s ... quirky decl grammer by using the left & right // distinction. Consider: // int (*f(float))(char) {} // f is a function that takes a float and returns a pointer to a function // that takes a char and returns an int. If we're trying to print f, start // by printing out the return types's left, then print our parameters, then // finally print right of the return type. void printLeft(OutputStream &S) const override { Ret->printLeft(S); S += " "; } void printRight(OutputStream &S) const override { S += "("; Params.printWithComma(S); S += ")"; Ret->printRight(S); } }; class FunctionEncoding final : public Node { const Node *Ret; const Node *Name; NodeArray Params; public: FunctionEncoding(Node *Ret_, Node *Name_, NodeArray Params_) : Node(KFunctionEncoding, NoParameterPack, /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::No, /*FunctionCache=*/Cache::Yes), Ret(Ret_), Name(Name_), Params(Params_) { for (Node *P : Params) ParameterPackSize = std::min(ParameterPackSize, P->ParameterPackSize); if (Ret) ParameterPackSize = std::min(ParameterPackSize, Ret->ParameterPackSize); } bool hasRHSComponentSlow(OutputStream &) const override { return true; } bool hasFunctionSlow(OutputStream &) const override { return true; } Node *getName() { return const_cast
(Name); } void printLeft(OutputStream &S) const override { if (Ret) { Ret->printLeft(S); if (!Ret->hasRHSComponent(S)) S += " "; } Name->print(S); } void printRight(OutputStream &S) const override { S += "("; Params.printWithComma(S); S += ")"; if (Ret) Ret->printRight(S); } }; enum FunctionRefQual : unsigned char { FrefQualNone, FrefQualLValue, FrefQualRValue, }; class FunctionRefQualType : public Node { Node *Fn; FunctionRefQual Quals; friend class FunctionQualType; public: FunctionRefQualType(Node *Fn_, FunctionRefQual Quals_) : Node(KFunctionRefQualType, Fn_->ParameterPackSize, /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::No, /*FunctionCache=*/Cache::Yes), Fn(Fn_), Quals(Quals_) {} bool hasFunctionSlow(OutputStream &) const override { return true; } bool hasRHSComponentSlow(OutputStream &) const override { return true; } void printQuals(OutputStream &S) const { if (Quals == FrefQualLValue) S += " &"; else S += " &&"; } void printLeft(OutputStream &S) const override { Fn->printLeft(S); } void printRight(OutputStream &S) const override { Fn->printRight(S); printQuals(S); } }; class FunctionQualType final : public QualType { public: FunctionQualType(Node *Child_, Qualifiers Quals_) : QualType(Child_, Quals_) { K = KFunctionQualType; } void printLeft(OutputStream &S) const override { Child->printLeft(S); } void printRight(OutputStream &S) const override { if (Child->getKind() == KFunctionRefQualType) { auto *RefQuals = static_cast
(Child); RefQuals->Fn->printRight(S); printQuals(S); RefQuals->printQuals(S); } else { Child->printRight(S); printQuals(S); } } }; class LiteralOperator : public Node { const Node *OpName; public: LiteralOperator(Node *OpName_) : Node(KLiteralOperator, OpName_->ParameterPackSize), OpName(OpName_) {} void printLeft(OutputStream &S) const override { S += "operator\"\" "; OpName->print(S); } }; class SpecialName final : public Node { const StringView Special; const Node *Child; public: SpecialName(StringView Special_, Node* Child_) : Node(KSpecialName, Child_->ParameterPackSize), Special(Special_), Child(Child_) {} void printLeft(OutputStream &S) const override { S += Special; Child->print(S); } }; class CtorVtableSpecialName final : public Node { const Node *FirstType; const Node *SecondType; public: CtorVtableSpecialName(Node *FirstType_, Node *SecondType_) : Node(KCtorVtableSpecialName, std::min(FirstType_->ParameterPackSize, SecondType_->ParameterPackSize)), FirstType(FirstType_), SecondType(SecondType_) {} void printLeft(OutputStream &S) const override { S += "construction vtable for "; FirstType->print(S); S += "-in-"; SecondType->print(S); } }; class QualifiedName final : public Node { // qualifier::name const Node *Qualifier; const Node *Name; public: QualifiedName(Node* Qualifier_, Node* Name_) : Node(KQualifiedName, std::min(Qualifier_->ParameterPackSize, Name_->ParameterPackSize)), Qualifier(Qualifier_), Name(Name_) {} StringView getBaseName() const override { return Name->getBaseName(); } void printLeft(OutputStream &S) const override { Qualifier->print(S); S += "::"; Name->print(S); } }; class EmptyName : public Node { public: EmptyName() : Node(KEmptyName) {} void printLeft(OutputStream &) const override {} }; class VectorType final : public Node { const Node *BaseType; const NodeOrString Dimension; const bool IsPixel; public: VectorType(NodeOrString Dimension_) : Node(KVectorType), BaseType(nullptr), Dimension(Dimension_), IsPixel(true) { if (Dimension.isNode()) ParameterPackSize = Dimension.asNode()->ParameterPackSize; } VectorType(Node *BaseType_, NodeOrString Dimension_) : Node(KVectorType, BaseType_->ParameterPackSize), BaseType(BaseType_), Dimension(Dimension_), IsPixel(false) { if (Dimension.isNode()) ParameterPackSize = std::min(ParameterPackSize, Dimension.asNode()->ParameterPackSize); } void printLeft(OutputStream &S) const override { if (IsPixel) { S += "pixel vector["; S += Dimension.asString(); S += "]"; } else { BaseType->print(S); S += " vector["; if (Dimension.isNode()) Dimension.asNode()->print(S); else if (Dimension.isString()) S += Dimension.asString(); S += "]"; } } }; /// An unexpanded parameter pack (either in the expression or type context). If /// this AST is correct, this node will have a ParameterPackExpansion node above /// it. /// /// This node is created when some
are found that apply to an ///
, and is stored in the TemplateParams table. In order for this to /// appear in the final AST, it has to referenced via a
(ie, /// T_). class ParameterPack final : public Node { NodeArray Data; public: ParameterPack(NodeArray Data_) : Node(KParameterPack, static_cast
(Data_.size())), Data(Data_) { ArrayCache = FunctionCache = RHSComponentCache = Cache::Unknown; if (std::all_of(Data.begin(), Data.end(), [](Node* P) { return P->ArrayCache == Cache::No; })) ArrayCache = Cache::No; if (std::all_of(Data.begin(), Data.end(), [](Node* P) { return P->FunctionCache == Cache::No; })) FunctionCache = Cache::No; if (std::all_of(Data.begin(), Data.end(), [](Node* P) { return P->RHSComponentCache == Cache::No; })) RHSComponentCache = Cache::No; } bool hasRHSComponentSlow(OutputStream &S) const override { size_t Idx = S.CurrentPackIndex; return Idx < Data.size() && Data[Idx]->hasRHSComponent(S); } bool hasArraySlow(OutputStream &S) const override { size_t Idx = S.CurrentPackIndex; return Idx < Data.size() && Data[Idx]->hasArray(S); } bool hasFunctionSlow(OutputStream &S) const override { size_t Idx = S.CurrentPackIndex; return Idx < Data.size() && Data[Idx]->hasFunction(S); } void printLeft(OutputStream &S) const override { size_t Idx = S.CurrentPackIndex; if (Idx < Data.size()) Data[Idx]->printLeft(S); } void printRight(OutputStream &S) const override { size_t Idx = S.CurrentPackIndex; if (Idx < Data.size()) Data[Idx]->printRight(S); } }; /// A variadic template argument. This node represents an occurance of /// J
E in some
. It isn't itself unexpanded, unless /// one of it's Elements is. The parser inserts a ParameterPack into the /// TemplateParams table if the
this pack belongs to apply to an ///
. class TemplateArgumentPack final : public Node { NodeArray Elements; public: TemplateArgumentPack(NodeArray Elements_) : Node(KTemplateArgumentPack), Elements(Elements_) { for (Node *E : Elements) ParameterPackSize = std::min(E->ParameterPackSize, ParameterPackSize); } NodeArray getElements() const { return Elements; } void printLeft(OutputStream &S) const override { Elements.printWithComma(S); } }; /// A pack expansion. Below this node, there are some unexpanded ParameterPacks /// which each have Child->ParameterPackSize elements. class ParameterPackExpansion final : public Node { const Node *Child; public: ParameterPackExpansion(Node* Child_) : Node(KParameterPackExpansion), Child(Child_) {} const Node *getChild() const { return Child; } void printLeft(OutputStream &S) const override { unsigned PackSize = Child->ParameterPackSize; if (PackSize == NoParameterPack) { Child->print(S); S += "..."; return; } SwapAndRestore
SavePackIndex(S.CurrentPackIndex, 0); for (unsigned I = 0; I != PackSize; ++I) { if (I != 0) S += ", "; S.CurrentPackIndex = I; Child->print(S); } } }; inline bool Node::isEmptyPackExpansion() const { if (getKind() == KParameterPackExpansion) { auto *AsPack = static_cast
(this); return AsPack->getChild()->isEmptyPackExpansion(); } if (getKind() == KTemplateArgumentPack) { auto *AsTemplateArg = static_cast
(this); for (Node *E : AsTemplateArg->getElements()) if (!E->isEmptyPackExpansion()) return false; return true; } return ParameterPackSize == 0; } class TemplateArgs final : public Node { NodeArray Params; public: TemplateArgs(NodeArray Params_) : Node(KTemplateArgs), Params(Params_) { for (Node *P : Params) ParameterPackSize = std::min(ParameterPackSize, P->ParameterPackSize); } NodeArray getParams() { return Params; } void printLeft(OutputStream &S) const override { S += "<"; bool FirstElement = true; for (size_t Idx = 0, E = Params.size(); Idx != E; ++Idx) { if (Params[Idx]->isEmptyPackExpansion()) continue; if (!FirstElement) S += ", "; FirstElement = false; Params[Idx]->print(S); } if (S.back() == '>') S += " "; S += ">"; } }; class NameWithTemplateArgs final : public Node { // name
Node *Name; Node *TemplateArgs; public: NameWithTemplateArgs(Node *Name_, Node *TemplateArgs_) : Node(KNameWithTemplateArgs, std::min(Name_->ParameterPackSize, TemplateArgs_->ParameterPackSize)), Name(Name_), TemplateArgs(TemplateArgs_) {} StringView getBaseName() const override { return Name->getBaseName(); } void printLeft(OutputStream &S) const override { Name->print(S); TemplateArgs->print(S); } }; class GlobalQualifiedName final : public Node { Node *Child; public: GlobalQualifiedName(Node* Child_) : Node(KGlobalQualifiedName, Child_->ParameterPackSize), Child(Child_) {} StringView getBaseName() const override { return Child->getBaseName(); } void printLeft(OutputStream &S) const override { S += "::"; Child->print(S); } }; class StdQualifiedName final : public Node { Node *Child; public: StdQualifiedName(Node *Child_) : Node(KStdQualifiedName, Child_->ParameterPackSize), Child(Child_) {} StringView getBaseName() const override { return Child->getBaseName(); } void printLeft(OutputStream &S) const override { S += "std::"; Child->print(S); } }; enum class SpecialSubKind { allocator, basic_string, string, istream, ostream, iostream, }; class ExpandedSpecialSubstitution final : public Node { SpecialSubKind SSK; public: ExpandedSpecialSubstitution(SpecialSubKind SSK_) : Node(KExpandedSpecialSubstitution), SSK(SSK_) {} StringView getBaseName() const override { switch (SSK) { case SpecialSubKind::allocator: return StringView("allocator"); case SpecialSubKind::basic_string: return StringView("basic_string"); case SpecialSubKind::string: return StringView("basic_string"); case SpecialSubKind::istream: return StringView("basic_istream"); case SpecialSubKind::ostream: return StringView("basic_ostream"); case SpecialSubKind::iostream: return StringView("basic_iostream"); } _LIBCPP_UNREACHABLE(); } void printLeft(OutputStream &S) const override { switch (SSK) { case SpecialSubKind::allocator: S += "std::basic_string
, " "std::allocator
>"; break; case SpecialSubKind::basic_string: case SpecialSubKind::string: S += "std::basic_string
, " "std::allocator
>"; break; case SpecialSubKind::istream: S += "std::basic_istream
>"; break; case SpecialSubKind::ostream: S += "std::basic_ostream
>"; break; case SpecialSubKind::iostream: S += "std::basic_iostream
>"; break; } } }; class SpecialSubstitution final : public Node { public: SpecialSubKind SSK; SpecialSubstitution(SpecialSubKind SSK_) : Node(KSpecialSubstitution), SSK(SSK_) {} StringView getBaseName() const override { switch (SSK) { case SpecialSubKind::allocator: return StringView("allocator"); case SpecialSubKind::basic_string: return StringView("basic_string"); case SpecialSubKind::string: return StringView("string"); case SpecialSubKind::istream: return StringView("istream"); case SpecialSubKind::ostream: return StringView("ostream"); case SpecialSubKind::iostream: return StringView("iostream"); } _LIBCPP_UNREACHABLE(); } void printLeft(OutputStream &S) const override { switch (SSK) { case SpecialSubKind::allocator: S += "std::allocator"; break; case SpecialSubKind::basic_string: S += "std::basic_string"; break; case SpecialSubKind::string: S += "std::string"; break; case SpecialSubKind::istream: S += "std::istream"; break; case SpecialSubKind::ostream: S += "std::ostream"; break; case SpecialSubKind::iostream: S += "std::iostream"; break; } } }; class CtorDtorName final : public Node { const Node *Basename; const bool IsDtor; public: CtorDtorName(Node *Basename_, bool IsDtor_) : Node(KCtorDtorName, Basename_->ParameterPackSize), Basename(Basename_), IsDtor(IsDtor_) {} void printLeft(OutputStream &S) const override { if (IsDtor) S += "~"; S += Basename->getBaseName(); } }; class DtorName : public Node { const Node *Base; public: DtorName(Node *Base_) : Node(KDtorName), Base(Base_) { ParameterPackSize = Base->ParameterPackSize; } void printLeft(OutputStream &S) const override { S += "~"; Base->printLeft(S); } }; class UnnamedTypeName : public Node { const StringView Count; public: UnnamedTypeName(StringView Count_) : Node(KUnnamedTypeName), Count(Count_) {} void printLeft(OutputStream &S) const override { S += "'unnamed"; S += Count; S += "\'"; } }; class LambdaTypeName : public Node { NodeArray Params; StringView Count; public: LambdaTypeName(NodeArray Params_, StringView Count_) : Node(KLambdaTypeName), Params(Params_), Count(Count_) { for (Node *P : Params) ParameterPackSize = std::min(ParameterPackSize, P->ParameterPackSize); } void printLeft(OutputStream &S) const override { S += "\'lambda"; S += Count; S += "\'("; Params.printWithComma(S); S += ")"; } }; // -- Expression Nodes -- struct Expr : public Node { Expr() : Node(KExpr) {} }; class BinaryExpr : public Expr { const Node *LHS; const StringView InfixOperator; const Node *RHS; public: BinaryExpr(Node *LHS_, StringView InfixOperator_, Node *RHS_) : LHS(LHS_), InfixOperator(InfixOperator_), RHS(RHS_) { ParameterPackSize = std::min(LHS->ParameterPackSize, RHS->ParameterPackSize); } void printLeft(OutputStream &S) const override { // might be a template argument expression, then we need to disambiguate // with parens. if (InfixOperator == ">") S += "("; S += "("; LHS->print(S); S += ") "; S += InfixOperator; S += " ("; RHS->print(S); S += ")"; if (InfixOperator == ">") S += ")"; } }; class ArraySubscriptExpr : public Expr { const Node *Op1; const Node *Op2; public: ArraySubscriptExpr(Node *Op1_, Node *Op2_) : Op1(Op1_), Op2(Op2_) { ParameterPackSize = std::min(Op1->ParameterPackSize, Op2->ParameterPackSize); } void printLeft(OutputStream &S) const override { S += "("; Op1->print(S); S += ")["; Op2->print(S); S += "]"; } }; class PostfixExpr : public Expr { const Node *Child; const StringView Operand; public: PostfixExpr(Node *Child_, StringView Operand_) : Child(Child_), Operand(Operand_) { ParameterPackSize = Child->ParameterPackSize; } void printLeft(OutputStream &S) const override { S += "("; Child->print(S); S += ")"; S += Operand; } }; class ConditionalExpr : public Expr { const Node *Cond; const Node *Then; const Node *Else; public: ConditionalExpr(Node *Cond_, Node *Then_, Node *Else_) : Cond(Cond_), Then(Then_), Else(Else_) { ParameterPackSize = std::min(Cond->ParameterPackSize, std::min(Then->ParameterPackSize, Else->ParameterPackSize)); } void printLeft(OutputStream &S) const override { S += "("; Cond->print(S); S += ") ? ("; Then->print(S); S += ") : ("; Else->print(S); S += ")"; } }; class MemberExpr : public Expr { const Node *LHS; const StringView Kind; const Node *RHS; public: MemberExpr(Node *LHS_, StringView Kind_, Node *RHS_) : LHS(LHS_), Kind(Kind_), RHS(RHS_) { ParameterPackSize = std::min(LHS->ParameterPackSize, RHS->ParameterPackSize); } void printLeft(OutputStream &S) const override { LHS->print(S); S += Kind; RHS->print(S); } }; class EnclosingExpr : public Expr { const StringView Prefix; const Node *Infix; const StringView Postfix; public: EnclosingExpr(StringView Prefix_, Node *Infix_, StringView Postfix_) : Prefix(Prefix_), Infix(Infix_), Postfix(Postfix_) { ParameterPackSize = Infix->ParameterPackSize; } void printLeft(OutputStream &S) const override { S += Prefix; Infix->print(S); S += Postfix; } }; class CastExpr : public Expr { // cast_kind
(from) const StringView CastKind; const Node *To; const Node *From; public: CastExpr(StringView CastKind_, Node *To_, Node *From_) : CastKind(CastKind_), To(To_), From(From_) { ParameterPackSize = std::min(To->ParameterPackSize, From->ParameterPackSize); } void printLeft(OutputStream &S) const override { S += CastKind; S += "<"; To->printLeft(S); S += ">("; From->printLeft(S); S += ")"; } }; class SizeofParamPackExpr : public Expr { Node *Pack; public: SizeofParamPackExpr(Node *Pack_) : Pack(Pack_) {} void printLeft(OutputStream &S) const override { S += "sizeof...("; ParameterPackExpansion PPE(Pack); PPE.printLeft(S); S += ")"; } }; class CallExpr : public Expr { const Node *Callee; NodeArray Args; public: CallExpr(Node *Callee_, NodeArray Args_) : Callee(Callee_), Args(Args_) { for (Node *P : Args) ParameterPackSize = std::min(ParameterPackSize, P->ParameterPackSize); ParameterPackSize = std::min(ParameterPackSize, Callee->ParameterPackSize); } void printLeft(OutputStream &S) const override { Callee->print(S); S += "("; Args.printWithComma(S); S += ")"; } }; class NewExpr : public Expr { // new (expr_list) type(init_list) NodeArray ExprList; Node *Type; NodeArray InitList; bool IsGlobal; // ::operator new ? bool IsArray; // new[] ? public: NewExpr(NodeArray ExprList_, Node *Type_, NodeArray InitList_, bool IsGlobal_, bool IsArray_) : ExprList(ExprList_), Type(Type_), InitList(InitList_), IsGlobal(IsGlobal_), IsArray(IsArray_) { for (Node *E : ExprList) ParameterPackSize = std::min(ParameterPackSize, E->ParameterPackSize); for (Node *I : InitList) ParameterPackSize = std::min(ParameterPackSize, I->ParameterPackSize); if (Type) ParameterPackSize = std::min(ParameterPackSize, Type->ParameterPackSize); } void printLeft(OutputStream &S) const override { if (IsGlobal) S += "::operator "; S += "new"; if (IsArray) S += "[]"; S += ' '; if (!ExprList.empty()) { S += "("; ExprList.printWithComma(S); S += ")"; } Type->print(S); if (!InitList.empty()) { S += "("; InitList.printWithComma(S); S += ")"; } } }; class DeleteExpr : public Expr { Node *Op; bool IsGlobal; bool IsArray; public: DeleteExpr(Node *Op_, bool IsGlobal_, bool IsArray_) : Op(Op_), IsGlobal(IsGlobal_), IsArray(IsArray_) { ParameterPackSize = Op->ParameterPackSize; } void printLeft(OutputStream &S) const override { if (IsGlobal) S += "::"; S += "delete"; if (IsArray) S += "[] "; Op->print(S); } }; class PrefixExpr : public Expr { StringView Prefix; Node *Child; public: PrefixExpr(StringView Prefix_, Node *Child_) : Prefix(Prefix_), Child(Child_) { ParameterPackSize = Child->ParameterPackSize; } void printLeft(OutputStream &S) const override { S += Prefix; S += "("; Child->print(S); S += ")"; } }; class FunctionParam : public Expr { StringView Number; public: FunctionParam(StringView Number_) : Number(Number_) {} void printLeft(OutputStream &S) const override { S += "fp"; S += Number; } }; class ConversionExpr : public Expr { const Node *Type; NodeArray Expressions; public: ConversionExpr(const Node *Type_, NodeArray Expressions_) : Type(Type_), Expressions(Expressions_) { for (Node *E : Expressions) ParameterPackSize = std::min(ParameterPackSize, E->ParameterPackSize); ParameterPackSize = std::min(ParameterPackSize, Type->ParameterPackSize); } void printLeft(OutputStream &S) const override { S += "("; Type->print(S); S += ")("; Expressions.printWithComma(S); S += ")"; } }; class ThrowExpr : public Expr { const Node *Op; public: ThrowExpr(Node *Op_) : Op(Op_) { ParameterPackSize = Op->ParameterPackSize; } void printLeft(OutputStream &S) const override { S += "throw "; Op->print(S); } }; class BoolExpr : public Expr { bool Value; public: BoolExpr(bool Value_) : Value(Value_) {} void printLeft(OutputStream &S) const override { S += Value ? StringView("true") : StringView("false"); } }; class IntegerCastExpr : public Expr { // ty(integer) Node *Ty; StringView Integer; public: IntegerCastExpr(Node *Ty_, StringView Integer_) : Ty(Ty_), Integer(Integer_) { ParameterPackSize = Ty->ParameterPackSize; } void printLeft(OutputStream &S) const override { S += "("; Ty->print(S); S += ")"; S += Integer; } }; class IntegerExpr : public Expr { StringView Type; StringView Value; public: IntegerExpr(StringView Type_, StringView Value_) : Type(Type_), Value(Value_) {} void printLeft(OutputStream &S) const override { if (Type.size() > 3) { S += "("; S += Type; S += ")"; } if (Value[0] == 'n') { S += "-"; S += Value.dropFront(1); } else S += Value; if (Type.size() <= 3) S += Type; } }; template
struct FloatData; template
class FloatExpr : public Expr { const StringView Contents; public: FloatExpr(StringView Contents_) : Contents(Contents_) {} void printLeft(OutputStream &s) const override { const char *first = Contents.begin(); const char *last = Contents.end() + 1; const size_t N = FloatData
::mangled_size; if (static_cast
(last - first) > N) { last = first + N; union { Float value; char buf[sizeof(Float)]; }; const char *t = first; char *e = buf; for (; t != last; ++t, ++e) { unsigned d1 = isdigit(*t) ? static_cast
(*t - '0') : static_cast
(*t - 'a' + 10); ++t; unsigned d0 = isdigit(*t) ? static_cast
(*t - '0') : static_cast
(*t - 'a' + 10); *e = static_cast
((d1 << 4) + d0); } #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ std::reverse(buf, e); #endif char num[FloatData
::max_demangled_size] = {0}; int n = snprintf(num, sizeof(num), FloatData
::spec, value); s += StringView(num, num + n); } } }; class BumpPointerAllocator { struct BlockMeta { BlockMeta* Next; size_t Current; }; static constexpr size_t AllocSize = 4096; static constexpr size_t UsableAllocSize = AllocSize - sizeof(BlockMeta); alignas(16) char InitialBuffer[AllocSize]; BlockMeta* BlockList = nullptr; void grow() { char* NewMeta = new char[AllocSize]; BlockList = new (NewMeta) BlockMeta{BlockList, 0}; } void* allocateMassive(size_t NBytes) { NBytes += sizeof(BlockMeta); BlockMeta* NewMeta = reinterpret_cast
(new char[NBytes]); BlockList->Next = new (NewMeta) BlockMeta{BlockList->Next, 0}; return static_cast
(NewMeta + 1); } public: BumpPointerAllocator() : BlockList(new (InitialBuffer) BlockMeta{nullptr, 0}) {} void* allocate(size_t N) { N = (N + 15u) & ~15u; if (N + BlockList->Current >= UsableAllocSize) { if (N > UsableAllocSize) return allocateMassive(N); grow(); } BlockList->Current += N; return static_cast
(reinterpret_cast
(BlockList + 1) + BlockList->Current - N); } ~BumpPointerAllocator() { while (BlockList) { BlockMeta* Tmp = BlockList; BlockList = BlockList->Next; if (reinterpret_cast
(Tmp) != InitialBuffer) delete[] reinterpret_cast
(Tmp); } } }; template
class PODSmallVector { static_assert(std::is_pod
::value, "T is required to be a plain old data type"); T* First; T* Last; T* Cap; T Inline[N]; bool isInline() const { return First == Inline; } void clearInline() { First = Inline; Last = Inline; Cap = Inline + N; } void reserve(size_t NewCap) { size_t S = size(); if (isInline()) { auto* Tmp = static_cast
(std::malloc(NewCap * sizeof(T))); std::copy(First, Last, Tmp); First = Tmp; } else First = static_cast
(std::realloc(First, NewCap * sizeof(T))); Last = First + S; Cap = First + NewCap; } public: PODSmallVector() : First(Inline), Last(First), Cap(Inline + N) {} PODSmallVector(const PODSmallVector&) = delete; PODSmallVector& operator=(const PODSmallVector&) = delete; PODSmallVector(PODSmallVector&& Other) : PODSmallVector() { if (Other.isInline()) { std::copy(Other.begin(), Other.end(), First); Last = First + Other.size(); Other.clear(); return; } First = Other.First; Last = Other.Last; Cap = Other.Cap; Other.clearInline(); } PODSmallVector& operator=(PODSmallVector&& Other) { if (Other.isInline()) { if (!isInline()) { std::free(First); clearInline(); } std::copy(Other.begin(), Other.end(), First); Last = First + Other.size(); Other.clear(); return *this; } if (isInline()) { First = Other.First; Last = Other.Last; Cap = Other.Cap; Other.clearInline(); return *this; } std::swap(First, Other.First); std::swap(Last, Other.Last); std::swap(Cap, Other.Cap); Other.clear(); return *this; } void push_back(const T& Elem) { if (Last == Cap) reserve(size() * 2); *Last++ = Elem; } void pop_back() { assert(Last != First && "Popping empty vector!"); --Last; } void dropBack(size_t Index) { assert(Index <= size() && "dropBack() can't expand!"); Last = First + Index; } T* begin() { return First; } T* end() { return Last; } bool empty() const { return First == Last; } size_t size() const { return static_cast
(Last - First); } T& back() { assert(Last != First && "Calling back() on empty vector!"); return *(Last - 1); } T& operator[](size_t Index) { assert(Index < size() && "Invalid access!"); return *(begin() + Index); } void clear() { Last = First; } ~PODSmallVector() { if (!isInline()) std::free(First); } }; struct Db { const char *First; const char *Last; // Name stack, this is used by the parser to hold temporary names that were // parsed. The parser colapses multiple names into new nodes to construct // the AST. Once the parser is finished, names.size() == 1. PODSmallVector
Names; // Substitution table. Itanium supports name substitutions as a means of // compression. The string "S42_" refers to the 44nd entry (base-36) in this // table. PODSmallVector
Subs; // Template parameter table. Like the above, but referenced like "T42_". // This has a smaller size compared to Subs and Names because it can be // stored on the stack. PODSmallVector
TemplateParams; Qualifiers CV = QualNone; FunctionRefQual RefQuals = FrefQualNone; unsigned EncodingDepth = 0; bool ParsedCtorDtorCV = false; bool TagTemplates = true; bool FixForwardReferences = false; bool TryToParseTemplateArgs = true; BumpPointerAllocator ASTAllocator; template
T *make(Args &&... args) { return new (ASTAllocator.allocate(sizeof(T))) T(std::forward
(args)...); } template
NodeArray makeNodeArray(It begin, It end) { size_t sz = static_cast
(end - begin); void *mem = ASTAllocator.allocate(sizeof(Node *) * sz); Node **data = new (mem) Node *[sz]; std::copy(begin, end, data); return NodeArray(data, sz); } NodeArray popTrailingNodeArray(size_t FromPosition) { assert(FromPosition <= Names.size()); NodeArray res = makeNodeArray(Names.begin() + (long)FromPosition, Names.end()); Names.dropBack(FromPosition); return res; } bool consumeIf(StringView S) { if (StringView(First, Last).startsWith(S)) { First += S.size(); return true; } return false; } bool consumeIf(char C) { if (First != Last && *First == C) { ++First; return true; } return false; } char consume() { return First != Last ? *First++ : '\0'; } char look(unsigned Lookahead = 0) { if (static_cast
(Last - First) <= Lookahead) return '\0'; return First[Lookahead]; } size_t numLeft() const { return static_cast
(Last - First); } StringView parseNumber(bool AllowNegative = false); Qualifiers parseCVQualifiers(); bool parsePositiveInteger(size_t *Out); StringView parseBareSourceName(); /// Parse the
production. Node *parseExpr(); Node *parsePrefixExpr(StringView Kind); Node *parseBinaryExpr(StringView Kind); Node *parseIntegerLiteral(StringView Lit); Node *parseExprPrimary(); template
Node *parseFloatingLiteral(); Node *parseFunctionParam(); Node *parseNewExpr(); Node *parseConversionExpr(); /// Parse the
production. Node *parseType(); Node *parseFunctionType(); Node *parseVectorType(); Node *parseDecltype(); Node *parseArrayType(); Node *parsePointerToMemberType(); Node *parseClassEnumType(); // FIXME: remove this when all the parse_* functions have been rewritten. template
Node *legacyParse() { size_t BeforeType = Names.size(); const char *OrigFirst = First; const char *T = parse_fn(First, Last, *this); if (T == OrigFirst || BeforeType + 1 != Names.size()) return nullptr; First = T; Node *R = Names.back(); Names.pop_back(); return R; } template
Node *legacyParse() { size_t BeforeType = Names.size(); const char *OrigFirst = First; const char *T = parse_fn(First, Last, *this, nullptr); if (T == OrigFirst || BeforeType + 1 != Names.size()) return nullptr; First = T; Node *R = Names.back(); Names.pop_back(); return R; } }; const char *parse_expression(const char *first, const char *last, Db &db) { db.First = first; db.Last = last; Node *R = db.parseExpr(); if (R == nullptr) return first; db.Names.push_back(R); return db.First; } const char *parse_expr_primary(const char *first, const char *last, Db &db) { db.First = first; db.Last = last; Node *R = db.parseExprPrimary(); if (R == nullptr) return first; db.Names.push_back(R); return db.First; } const char *parse_type(const char *first, const char *last, Db &db) { db.First = first; db.Last = last; Node *R = db.parseType(); if (R == nullptr) return first; db.Names.push_back(R); return db.First; } const char *parse_decltype(const char *first, const char *last, Db &db) { db.First = first; db.Last = last; Node *R = db.parseDecltype(); if (R == nullptr) return first; db.Names.push_back(R); return db.First; } const char *parse_type(const char *first, const char *last, Db &db); const char *parse_encoding(const char *first, const char *last, Db &db); const char *parse_name(const char *first, const char *last, Db &db, bool *ends_with_template_args = 0); const char *parse_template_args(const char *first, const char *last, Db &db); const char *parse_template_param(const char *, const char *, Db &); const char *parse_operator_name(const char *first, const char *last, Db &db); const char *parse_unqualified_name(const char *first, const char *last, Db &db); const char *parse_decltype(const char *first, const char *last, Db &db); const char *parse_unresolved_name(const char *, const char *, Db &); const char *parse_substitution(const char *, const char *, Db &); //
::= [n]
StringView Db::parseNumber(bool AllowNegative) { const char *Tmp = First; if (AllowNegative) consumeIf('n'); if (numLeft() == 0 || !std::isdigit(*First)) return StringView(); while (numLeft() != 0 && std::isdigit(*First)) ++First; return StringView(Tmp, First); } //
::= [0-9]* bool Db::parsePositiveInteger(size_t *Out) { *Out = 0; if (look() < '0' || look() > '9') return true; while (look() >= '0' && look() <= '9') { *Out *= 10; *Out += static_cast
(consume() - '0'); } return false; } StringView Db::parseBareSourceName() { size_t Int = 0; if (parsePositiveInteger(&Int) || numLeft() < Int) return StringView(); StringView R(First, First + Int); First += Int; return R; } //
::= F [Y]
[
] E // //
::= R # & ref-qualifier //
::= O # && ref-qualifier Node *Db::parseFunctionType() { if (!consumeIf('F')) return nullptr; consumeIf('Y'); // extern "C" Node *ReturnType = parseType(); if (ReturnType == nullptr) return nullptr; FunctionRefQual ReferenceQualifier = FrefQualNone; size_t ParamsBegin = Names.size(); while (true) { if (consumeIf('E')) break; if (consumeIf('v')) continue; if (consumeIf("RE")) { ReferenceQualifier = FrefQualLValue; break; } if (consumeIf("OE")) { ReferenceQualifier = FrefQualRValue; break; } Node *T = parseType(); if (T == nullptr) return nullptr; Names.push_back(T); } NodeArray Params = popTrailingNodeArray(ParamsBegin); Node *Fn = make
(ReturnType, Params); if (ReferenceQualifier != FrefQualNone) Fn = make
(Fn, ReferenceQualifier); return Fn; } // extension: //
::= Dv
_
// ::= Dv [
] _
//
::=
// ::= p # AltiVec vector pixel Node *Db::parseVectorType() { if (!consumeIf("Dv")) return nullptr; if (look() >= '1' && look() <= '9') { StringView DimensionNumber = parseNumber(); if (!consumeIf('_')) return nullptr; if (consumeIf('p')) return make
(DimensionNumber); Node *ElemType = parseType(); if (ElemType == nullptr) return nullptr; return make
(ElemType, DimensionNumber); } if (!consumeIf('_')) { Node *DimExpr = parseExpr(); if (!DimExpr) return nullptr; if (!consumeIf('_')) return nullptr; Node *ElemType = parseType(); if (!ElemType) return nullptr; return make
(ElemType, DimExpr); } Node *ElemType = parseType(); if (!ElemType) return nullptr; return make
(ElemType, StringView()); } //
::= Dt
E # decltype of an id-expression or class member access (C++0x) // ::= DT
E # decltype of an expression (C++0x) Node *Db::parseDecltype() { if (!consumeIf('D')) return nullptr; if (!consumeIf('t') && !consumeIf('T')) return nullptr; Node *E = parseExpr(); if (E == nullptr) return nullptr; if (!consumeIf('E')) return nullptr; return make
("decltype(", E, ")"); } //
::= A
_
// ::= A [
] _
Node *Db::parseArrayType() { if (!consumeIf('A')) return nullptr; if (std::isdigit(look())) { StringView Dimension = parseNumber(); if (!consumeIf('_')) return nullptr; Node *Ty = parseType(); if (Ty == nullptr) return nullptr; return make
(Ty, Dimension); } if (!consumeIf('_')) { Node *DimExpr = parseExpr(); if (DimExpr == nullptr) return nullptr; if (!consumeIf('_')) return nullptr; Node *ElementType = parseType(); if (ElementType == nullptr) return nullptr; return make
(ElementType, DimExpr); } Node *Ty = parseType(); if (Ty == nullptr) return nullptr; return make
(Ty); } //
::= M
Node *Db::parsePointerToMemberType() { if (!consumeIf('M')) return nullptr; Node *ClassType = parseType(); if (ClassType == nullptr) return nullptr; Node *MemberType = parseType(); if (MemberType == nullptr) return nullptr; return make
(ClassType, MemberType); } //
::=
# non-dependent type name, dependent type name, or dependent typename-specifier // ::= Ts
# dependent elaborated type specifier using 'struct' or 'class' // ::= Tu
# dependent elaborated type specifier using 'union' // ::= Te
# dependent elaborated type specifier using 'enum' Node *Db::parseClassEnumType() { // FIXME: try to parse the elaborated type specifiers here! return legacyParse
(); } //
::=
// ::=
// ::=
// ::=
// ::=
// ::=
// ::=
// ::=
// ::=
// ::= P
# pointer // ::= R
# l-value reference // ::= O
# r-value reference (C++11) // ::= C
# complex pair (C99) // ::= G
# imaginary (C99) // ::=
# See Compression below // extension ::= U
# objc-type
// extension ::=
#
starts with Dv // //
::=
objcproto
# k0 = 9 +
+ k1 //
::=
# PU<11+>objcproto 11objc_object
11objc_object -> id
Node *Db::parseType() { Node *Result = nullptr; switch (look()) { // ::=
case 'r': case 'V': case 'K': { Qualifiers Q = parseCVQualifiers(); bool AppliesToFunction = look() == 'F'; Node *Child = parseType(); if (Child == nullptr) return nullptr; if (AppliesToFunction) Result = make
(Child, Q); else Result = make
(Child, Q); // Itanium C++ ABI 5.1.5.3: // For the purposes of substitution, the CV-qualifiers and ref-qualifier // of a function type are an indivisible part of the type. if (AppliesToFunction) return Result; break; } //
::= U
[
] # vendor extended type qualifier case 'U': { // FIXME: We should fold this into the cvr qualifier parsing above. This // currently adds too many entries into the substitution table if multiple // qualifiers are present on the same type, as all the qualifiers on a type // should just get one entry in the substitution table. ++First; StringView Qual = parseBareSourceName(); if (Qual.empty()) return nullptr; // FIXME parse the optional
here! Result = parseType(); if (Result == nullptr) return nullptr; // extension ::= U
# objc-type
if (Qual.startsWith("objcproto")) { StringView ProtoSourceName = Qual.dropFront(std::strlen("objcproto")); StringView Proto; { SwapAndRestore
SaveFirst(First, ProtoSourceName.begin()), SaveLast(Last, ProtoSourceName.end()); Proto = parseBareSourceName(); } if (Proto.empty()) return nullptr; Result = make
(Result, Proto); } else Result = make
(Result, Qual); break; } //
::= v # void case 'v': ++First; return make
("void"); // ::= w # wchar_t case 'w': ++First; return make
("wchar_t"); // ::= b # bool case 'b': ++First; return make
("bool"); // ::= c # char case 'c': ++First; return make
("char"); // ::= a # signed char case 'a': ++First; return make
("signed char"); // ::= h # unsigned char case 'h': ++First; return make
("unsigned char"); // ::= s # short case 's': ++First; return make
("short"); // ::= t # unsigned short case 't': ++First; return make
("unsigned short"); // ::= i # int case 'i': ++First; return make
("int"); // ::= j # unsigned int case 'j': ++First; return make
("unsigned int"); // ::= l # long case 'l': ++First; return make
("long"); // ::= m # unsigned long case 'm': ++First; return make
("unsigned long"); // ::= x # long long, __int64 case 'x': ++First; return make
("long long"); // ::= y # unsigned long long, __int64 case 'y': ++First; return make
("unsigned long long"); // ::= n # __int128 case 'n': ++First; return make
("__int128"); // ::= o # unsigned __int128 case 'o': ++First; return make
("unsigned __int128"); // ::= f # float case 'f': ++First; return make
("float"); // ::= d # double case 'd': ++First; return make
("double"); // ::= e # long double, __float80 case 'e': ++First; return make
("long double"); // ::= g # __float128 case 'g': ++First; return make
("__float128"); // ::= z # ellipsis case 'z': ++First; return make
("..."); //
::= u
# vendor extended type case 'u': { ++First; StringView Res = parseBareSourceName(); if (Res.empty()) return nullptr; return make
(Res); } case 'D': switch (look(1)) { // ::= Dd # IEEE 754r decimal floating point (64 bits) case 'd': First += 2; return make
("decimal64"); // ::= De # IEEE 754r decimal floating point (128 bits) case 'e': First += 2; return make
("decimal128"); // ::= Df # IEEE 754r decimal floating point (32 bits) case 'f': First += 2; return make
("decimal32"); // ::= Dh # IEEE 754r half-precision floating point (16 bits) case 'h': First += 2; return make
("decimal16"); // ::= Di # char32_t case 'i': First += 2; return make
("char32_t"); // ::= Ds # char16_t case 's': First += 2; return make
("char16_t"); // ::= Da # auto (in dependent new-expressions) case 'a': First += 2; return make
("auto"); // ::= Dc # decltype(auto) case 'c': First += 2; return make
("decltype(auto)"); // ::= Dn # std::nullptr_t (i.e., decltype(nullptr)) case 'n': First += 2; return make
("std::nullptr_t"); // ::=
case 't': case 'T': { Result = parseDecltype(); break; } // extension ::=
#
starts with Dv case 'v': { Result = parseVectorType(); break; } // ::= Dp
# pack expansion (C++0x) case 'p': { First += 2; Node *Child = parseType(); if (!Child) return nullptr; Result = make
(Child); break; } } break; // ::=
case 'F': { Result = parseFunctionType(); break; } // ::=
case 'A': { Result = parseArrayType(); break; } // ::=
case 'M': { Result = parsePointerToMemberType(); break; } // ::=
case 'T': { Result = legacyParse
(); if (Result == nullptr) return nullptr; // Result could be either of: //
::=
//
::=
// //
::=
// ::=
// // If this is followed by some
, and we're permitted to // parse them, take the second production. if (TryToParseTemplateArgs && look() == 'I') { Node *TA = legacyParse
(); if (TA == nullptr) return nullptr; Result = make
(Result, TA); } break; } // ::= P
# pointer case 'P': { ++First; Node *Ptr = parseType(); if (Ptr == nullptr) return nullptr; Result = make
(Ptr); break; } // ::= R
# l-value reference case 'R': { ++First; Node *Ref = parseType(); if (Ref == nullptr) return nullptr; Result = make
(Ref); break; } // ::= O
# r-value reference (C++11) case 'O': { ++First; Node *Ref = parseType(); if (Ref == nullptr) return nullptr; Result = make
(Ref); break; } // ::= C
# complex pair (C99) case 'C': { ++First; Node *P = parseType(); if (P == nullptr) return nullptr; Result = make
(P, " complex"); break; } // ::= G
# imaginary (C99) case 'G': { ++First; Node *P = parseType(); if (P == nullptr) return P; Result = make
(P, " imaginary"); break; } // ::=
# See Compression below case 'S': { if (look(1) && look(1) != 't') { Node *Sub = legacyParse
(); if (Sub == nullptr) return nullptr; // Sub could be either of: //
::=
//
::=
// //
::=
// ::=
// // If this is followed by some
, and we're permitted to // parse them, take the second production. if (TryToParseTemplateArgs && look() == 'I') { Node *TA = legacyParse
(); if (TA == nullptr) return nullptr; Result = make
(Sub, TA); break; } // If all we parsed was a substitution, don't re-insert into the // substitution table. return Sub; } _LIBCPP_FALLTHROUGH(); } // ::=
default: { Result = parseClassEnumType(); break; } } // If we parsed a type, insert it into the substitution table. Note that all //