[cfe-commits] r65103 - /cfe/trunk/lib/AST/DeclBase.cpp
Daniel Dunbar
daniel at zuster.org
Thu Feb 19 18:03:22 PST 2009
Totally awesome. My hero x 2.
- Daniel
On Thu, Feb 19, 2009 at 5:44 PM, Chris Lattner <sabre at nondot.org> wrote:
> Author: lattner
> Date: Thu Feb 19 19:44:05 2009
> New Revision: 65103
>
> URL: http://llvm.org/viewvc/llvm-project?rev=65103&view=rev
> Log:
> optimize the 'StoredDeclsMap' for the common case where there is
> exactly one decl with a specific name in a specific context. This
> avoids a bunch of malloc traffic and shrinks StoredDeclsMap to hold
> one pointer instead of 3 words (for a std::vector).
>
> This speeds up -fsyntax-only on cocoa.h with PTH by ~7.3%.
>
> Modified:
> cfe/trunk/lib/AST/DeclBase.cpp
>
> Modified: cfe/trunk/lib/AST/DeclBase.cpp
> URL: http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/AST/DeclBase.cpp?rev=65103&r1=65102&r2=65103&view=diff
>
> ==============================================================================
> --- cfe/trunk/lib/AST/DeclBase.cpp (original)
> +++ cfe/trunk/lib/AST/DeclBase.cpp Thu Feb 19 19:44:05 2009
> @@ -268,15 +268,119 @@
> }
> }
>
> -// FIXME: We really want to use a DenseSet here to eliminate the
> -// redundant storage of the declaration names, but (1) it doesn't give
> -// us the ability to search based on DeclarationName, (2) we really
> -// need something more like a DenseMultiSet, and (3) it's
> -// implemented in terms of DenseMap anyway. However, this data
> -// structure is really space-inefficient, so we'll have to do
> -// something.
> -typedef llvm::DenseMap<DeclarationName, std::vector<NamedDecl*> >
> - StoredDeclsMap;
> +/// StoredDeclsList - This is an array of decls optimized a common case of only
> +/// containing one entry.
> +struct StoredDeclsList {
> + /// Data - If the integer is 0, then the pointer is a NamedDecl*. If the
> + /// integer is 1, then it is a VectorTy;
> + llvm::PointerIntPair<void*, 1, bool> Data;
> +
> + /// VectorTy - When in vector form, this is what the Data pointer points to.
> + typedef llvm::SmallVector<NamedDecl*, 4> VectorTy;
> +public:
> + StoredDeclsList() {}
> + StoredDeclsList(const StoredDeclsList &RHS) : Data(RHS.Data) {
> + if (isVector())
> + Data.setPointer(new VectorTy(getVector()));
> + }
> +
> + ~StoredDeclsList() {
> + // If this is a vector-form, free the vector.
> + if (isVector())
> + delete &getVector();
> + }
> +
> + bool isVector() const { return Data.getInt() != 0; }
> + bool isInline() const { return Data.getInt() == 0; }
> + bool isNull() const { return Data.getPointer() == 0; }
> +
> + void setOnlyValue(NamedDecl *ND) {
> + assert(isInline() && "Not inline");
> + Data.setPointer(ND);
> + }
> +
> + /// getLookupResult - Return an array of all the decls that this list
> + /// represents.
> + DeclContext::lookup_result getLookupResult() {
> + // If we have a single inline unit, return it.
> + if (isInline()) {
> + assert(!isNull() && "Empty list isn't allowed");
> +
> + // Data is a raw pointer to a NamedDecl*, return it.
> + void *Ptr = &Data;
> + return DeclContext::lookup_result((NamedDecl**)Ptr, (NamedDecl**)Ptr+1);
> + }
> +
> + // Otherwise, we have a range result.
> + VectorTy &V = getVector();
> + return DeclContext::lookup_result(&V[0], &V[0]+V.size());
> + }
> +
> + /// HandleRedeclaration - If this is a redeclaration of an existing decl,
> + /// replace the old one with D and return true. Otherwise return false.
> + bool HandleRedeclaration(NamedDecl *D) {
> + // Most decls only have one entry in their list, special case it.
> + if (isInline()) {
> + if (!D->declarationReplaces(getInlineValue()))
> + return false;
> + setOnlyValue(D);
> + return true;
> + }
> +
> + // Determine if this declaration is actually a redeclaration.
> + VectorTy &Vec = getVector();
> + VectorTy::iterator RDI
> + = std::find_if(Vec.begin(), Vec.end(),
> + std::bind1st(std::mem_fun(&NamedDecl::declarationReplaces),
> + D));
> + if (RDI == Vec.end())
> + return false;
> + *RDI = D;
> + return true;
> + }
> +
> + /// AddSubsequentDecl - This is called on the second and later decl when it is
> + /// not a redeclaration to merge it into the appropriate place in our list.
> + ///
> + void AddSubsequentDecl(NamedDecl *D) {
> + // If this is the second decl added to the list, convert this to vector
> + // form.
> + if (isInline()) {
> + NamedDecl *OldD = getInlineValue();
> + Data.setInt(1);
> + VectorTy *VT = new VectorTy();
> + VT->push_back(OldD);
> + Data.setPointer(VT);
> + }
> +
> + VectorTy &Vec = getVector();
> + if (isa<UsingDirectiveDecl>(D) ||
> + D->getIdentifierNamespace() == Decl::IDNS_Tag)
> + Vec.push_back(D);
> + else if (Vec.back()->getIdentifierNamespace() == Decl::IDNS_Tag) {
> + NamedDecl *TagD = Vec.back();
> + Vec.back() = D;
> + Vec.push_back(TagD);
> + } else
> + Vec.push_back(D);
> + }
> +
> +
> +private:
> + VectorTy &getVector() const {
> + assert(isVector() && "Not in vector form");
> + return *static_cast<VectorTy*>(Data.getPointer());
> + }
> +
> + NamedDecl *getInlineValue() const {
> + assert(isInline() && "Not in inline form");
> + return (NamedDecl*)Data.getPointer();
> + }
> +};
> +
> +
> +
> +typedef llvm::DenseMap<DeclarationName, StoredDeclsList> StoredDeclsMap;
>
> DeclContext::~DeclContext() {
> unsigned Size = LookupPtr.getInt();
> @@ -412,9 +516,7 @@
> StoredDeclsMap::iterator Pos = Map->find(Name);
> if (Pos == Map->end())
> return lookup_result(0, 0);
> -
> - return lookup_result(&Pos->second.front(),
> - &Pos->second.front() + Pos->second.size());
> + return Pos->second.getLookupResult();
> }
>
> // We have a small array. Look into it.
> @@ -559,46 +661,21 @@
> }
>
> // Insert this declaration into the map.
> - StoredDeclsMap *Map = static_cast<StoredDeclsMap*>(LookupPtr.getPointer());
> - std::vector<NamedDecl *> &DeclNameEntries = (*Map)[D->getDeclName()];
> - if (DeclNameEntries.empty()) {
> - DeclNameEntries.push_back(D);
> + StoredDeclsMap &Map = *static_cast<StoredDeclsMap*>(LookupPtr.getPointer());
> + StoredDeclsList &DeclNameEntries = Map[D->getDeclName()];
> + if (DeclNameEntries.isNull()) {
> + DeclNameEntries.setOnlyValue(D);
> return;
> }
>
> // If it is possible that this is a redeclaration, check to see if there is
> // already a decl for which declarationReplaces returns true. If there is
> // one, just replace it and return.
> - if (MayBeRedeclaration) {
> - // Most decls only have one entry in their list, special case it.
> - if (DeclNameEntries.size() == 1) {
> - if (D->declarationReplaces(DeclNameEntries[0])) {
> - DeclNameEntries[0] = D;
> - return;
> - }
> - } else {
> - // Determine if this declaration is actually a redeclaration.
> - std::vector<NamedDecl *>::iterator Redecl
> - = std::find_if(DeclNameEntries.begin(), DeclNameEntries.end(),
> - std::bind1st(std::mem_fun(&NamedDecl::declarationReplaces),
> - D));
> - if (Redecl != DeclNameEntries.end()) {
> - *Redecl = D;
> - return;
> - }
> - }
> - }
> + if (MayBeRedeclaration && DeclNameEntries.HandleRedeclaration(D))
> + return;
>
> // Put this declaration into the appropriate slot.
> - if (isa<UsingDirectiveDecl>(D) ||
> - D->getIdentifierNamespace() == Decl::IDNS_Tag)
> - DeclNameEntries.push_back(D);
> - else if (DeclNameEntries.back()->getIdentifierNamespace() == Decl::IDNS_Tag) {
> - NamedDecl *TagD = DeclNameEntries.back();
> - DeclNameEntries.back() = D;
> - DeclNameEntries.push_back(TagD);
> - } else
> - DeclNameEntries.push_back(D);
> + DeclNameEntries.AddSubsequentDecl(D);
> }
>
> /// Returns iterator range [First, Last) of UsingDirectiveDecls stored within
>
>
> _______________________________________________
> cfe-commits mailing list
> cfe-commits at cs.uiuc.edu
> http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits
>
More information about the cfe-commits
mailing list