From 68ef6943821861614e446fdcc5cf150369419834 Mon Sep 17 00:00:00 2001 From: Chris Lattner Date: Wed, 13 Jul 2011 04:22:39 +0000 Subject: [PATCH] stop leaking all named struct types with an empty name. Thanks to Benjamin Kramer for steering me in the right direction here. llvm-svn: 135031 --- llvm/lib/VMCore/LLVMContextImpl.cpp | 13 +++++++------ llvm/lib/VMCore/LLVMContextImpl.h | 1 + llvm/lib/VMCore/Type.cpp | 12 ++++++++++-- 3 files changed, 18 insertions(+), 8 deletions(-) diff --git a/llvm/lib/VMCore/LLVMContextImpl.cpp b/llvm/lib/VMCore/LLVMContextImpl.cpp index 12cb2464b53b..2b6bb39167af 100644 --- a/llvm/lib/VMCore/LLVMContextImpl.cpp +++ b/llvm/lib/VMCore/LLVMContextImpl.cpp @@ -81,14 +81,12 @@ LLVMContextImpl::~LLVMContextImpl() { SmallVector MDNodes; MDNodes.reserve(MDNodeSet.size() + NonUniquedMDNodes.size()); for (FoldingSetIterator I = MDNodeSet.begin(), E = MDNodeSet.end(); - I != E; ++I) { + I != E; ++I) MDNodes.push_back(&*I); - } MDNodes.append(NonUniquedMDNodes.begin(), NonUniquedMDNodes.end()); for (SmallVectorImpl::iterator I = MDNodes.begin(), - E = MDNodes.end(); I != E; ++I) { + E = MDNodes.end(); I != E; ++I) (*I)->destroy(); - } assert(MDNodeSet.empty() && NonUniquedMDNodes.empty() && "Destroying all MDNodes didn't empty the Context's sets."); // Destroy MDStrings. @@ -103,7 +101,10 @@ LLVMContextImpl::~LLVMContextImpl() { DeleteContainerSeconds(PointerTypes); DeleteContainerSeconds(ASPointerTypes); - for (StringMap::iterator I = NamedStructTypes.begin(), E = NamedStructTypes.end(); I != E; ++I) { + for (StringMap::iterator I = NamedStructTypes.begin(), + E = NamedStructTypes.end(); I != E; ++I) delete I->getValue(); - } + for (SmallPtrSet::iterator I = EmptyNamedStructTypes.begin(), + E = EmptyNamedStructTypes.end(); I != E; ++I) + delete *I; } diff --git a/llvm/lib/VMCore/LLVMContextImpl.h b/llvm/lib/VMCore/LLVMContextImpl.h index e36864b27b5d..b26068d85f70 100644 --- a/llvm/lib/VMCore/LLVMContextImpl.h +++ b/llvm/lib/VMCore/LLVMContextImpl.h @@ -179,6 +179,7 @@ public: std::map, FunctionType*> FunctionTypes; std::map, StructType*> AnonStructTypes; StringMap NamedStructTypes; + SmallPtrSet EmptyNamedStructTypes; unsigned NamedStructTypesUniqueID; DenseMap, ArrayType*> ArrayTypes; diff --git a/llvm/lib/VMCore/Type.cpp b/llvm/lib/VMCore/Type.cpp index 10467a8d9008..dc5053acc24c 100644 --- a/llvm/lib/VMCore/Type.cpp +++ b/llvm/lib/VMCore/Type.cpp @@ -412,7 +412,10 @@ void StructType::setBody(ArrayRef Elements, bool isPacked) { StructType *StructType::createNamed(LLVMContext &Context, StringRef Name) { StructType *ST = new StructType(Context); - ST->setName(Name); + if (!Name.empty()) + ST->setName(Name); + else + Context.pImpl->EmptyNamedStructTypes.insert(ST); return ST; } @@ -423,11 +426,16 @@ void StructType::setName(StringRef Name) { if (SymbolTableEntry) { getContext().pImpl->NamedStructTypes.erase(getName()); SymbolTableEntry = 0; + } else { + getContext().pImpl->EmptyNamedStructTypes.erase(this); } // If this is just removing the name, we're done. - if (Name.empty()) + if (Name.empty()) { + // Keep track of types with no names so we can free them. + getContext().pImpl->EmptyNamedStructTypes.insert(this); return; + } // Look up the entry for the name. StringMapEntry *Entry = -- GitLab