diff options
author | Douglas Gregor <dgregor@apple.com> | 2011-02-01 23:59:42 +0000 |
---|---|---|
committer | Douglas Gregor <dgregor@apple.com> | 2011-02-01 23:59:42 +0000 |
commit | e349d2c61256f11180b7112ab592b4ae700f3dcf (patch) | |
tree | df4b87bd98ec095506f73a06801d22eb3f63f6d6 | |
parent | 40249e7487c3314f185c63302aaad9edde6dfd53 (diff) |
Unique code-completion strings. On Cocoa.h, this costs us about 4% in
speed but saves us about 25% of the memory usage for strings.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@124704 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | include/clang/Sema/CodeCompleteConsumer.h | 38 | ||||
-rw-r--r-- | lib/Sema/CodeCompleteConsumer.cpp | 31 |
2 files changed, 69 insertions, 0 deletions
diff --git a/include/clang/Sema/CodeCompleteConsumer.h b/include/clang/Sema/CodeCompleteConsumer.h index 8cb8e75393..bd9a868101 100644 --- a/include/clang/Sema/CodeCompleteConsumer.h +++ b/include/clang/Sema/CodeCompleteConsumer.h @@ -15,7 +15,9 @@ #include "clang/AST/Type.h" #include "clang/AST/CanonicalType.h" +#include "llvm/ADT/DenseSet.h" #include "llvm/ADT/SmallVector.h" +#include "llvm/ADT/StringExtras.h" #include "llvm/ADT/StringRef.h" #include "llvm/Support/Allocator.h" #include "clang-c/Index.h" @@ -421,11 +423,47 @@ public: std::string getAsString() const; }; +/// \brief \c DenseMap information object for StringRefs. +struct DenseMapStringRefInfo { + static inline llvm::StringRef getEmptyKey() { + return llvm::StringRef(reinterpret_cast<const char*>((intptr_t)-1), 0); + } + static inline llvm::StringRef getTombstoneKey() { + return llvm::StringRef(reinterpret_cast<const char *>((intptr_t)-2), 0); + } + static unsigned getHashValue(llvm::StringRef Str) { + return llvm::HashString(Str); + } + static bool isEqual(llvm::StringRef LHS, llvm::StringRef RHS) { + if (LHS.size() == 0 && RHS.size() == 0) { + intptr_t LHSVal = reinterpret_cast<intptr_t>(LHS.data()); + intptr_t RHSVal = reinterpret_cast<intptr_t>(RHS.data()); + if (LHSVal == -1 || LHSVal == -2 || RHSVal == -1 || RHSVal == -2) + return LHSVal == RHSVal; + + return true; + } + + return LHS == RHS; + } +}; + /// \brief An allocator used specifically for the purpose of code completion. class CodeCompletionAllocator : public llvm::BumpPtrAllocator { + llvm::DenseSet<llvm::StringRef, DenseMapStringRefInfo> UniqueStrings; + unsigned StringBytesAllocated; + unsigned StringBytesUniqued; + unsigned StringsAllocated; + unsigned StringsUniqued; + public: + CodeCompletionAllocator(); + ~CodeCompletionAllocator(); + /// \brief Copy the given string into this allocator. const char *CopyString(llvm::StringRef String); + + void PrintStats(); }; /// \brief A builder class used to construct new code-completion strings. diff --git a/lib/Sema/CodeCompleteConsumer.cpp b/lib/Sema/CodeCompleteConsumer.cpp index cb2dd234b4..572c8dcbd0 100644 --- a/lib/Sema/CodeCompleteConsumer.cpp +++ b/lib/Sema/CodeCompleteConsumer.cpp @@ -220,10 +220,41 @@ const char *CodeCompletionString::getTypedText() const { return 0; } +CodeCompletionAllocator::CodeCompletionAllocator() + : llvm::BumpPtrAllocator(), StringBytesAllocated(0), StringBytesUniqued(0), + StringsAllocated(0), StringsUniqued(0) +{ +} + +CodeCompletionAllocator::~CodeCompletionAllocator() { } + +void CodeCompletionAllocator::PrintStats() { + llvm::errs() << "---Code completion memory allocation---\n" + << "String bytes uniqued: " << StringBytesUniqued << "/" + << StringBytesAllocated << " (" + << ((float)StringBytesUniqued*100/StringBytesAllocated) + << ")\nStrings uniqued: " << StringsUniqued << "/" << StringsAllocated + << " (" << ((float)StringsUniqued*100/StringsAllocated) + << ")\n"; + + llvm::BumpPtrAllocator::PrintStats(); +} + const char *CodeCompletionAllocator::CopyString(llvm::StringRef String) { + llvm::DenseSet<llvm::StringRef, DenseMapStringRefInfo>::iterator Uniqued + = UniqueStrings.find(String); + ++StringsAllocated; + StringBytesAllocated += String.size() + 1; + if (Uniqued != UniqueStrings.end()) { + StringBytesUniqued += String.size() + 1; + ++StringsUniqued; + return Uniqued->data(); + } + char *Mem = (char *)Allocate(String.size() + 1, 1); std::copy(String.begin(), String.end(), Mem); Mem[String.size()] = 0; + UniqueStrings.insert(llvm::StringRef(Mem, String.size())); return Mem; } |