From: Nuno Lopes Date: Sun, 1 Jun 2008 22:53:53 +0000 (+0000) Subject: fix decl attributes cleaning X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=9141bee299a05349924a34a7153d7064c9e0a107;p=clang fix decl attributes cleaning this plugs the leak of attributes and also fixes a crash in the test git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@51862 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/clang/AST/DeclBase.h b/include/clang/AST/DeclBase.h index 210f169150..b4f4c3b046 100644 --- a/include/clang/AST/DeclBase.h +++ b/include/clang/AST/DeclBase.h @@ -136,6 +136,7 @@ public: void addAttr(Attr *attr); const Attr *getAttrs() const; void swapAttrs(Decl *D); + void invalidateAttrs(); template const T *getAttr() const { for (const Attr *attr = getAttrs(); attr; attr = attr->getNext()) diff --git a/lib/AST/Decl.cpp b/lib/AST/Decl.cpp index 8d87d1cb4a..8d9913b1aa 100644 --- a/lib/AST/Decl.cpp +++ b/lib/AST/Decl.cpp @@ -336,14 +336,9 @@ Decl::~Decl() { DeclAttrMapTy::iterator it = DeclAttrs->find(this); assert(it != DeclAttrs->end() && "No attrs found but HasAttrs is true!"); - // FIXME: Properly release attributes. - // delete it->second; - DeclAttrs->erase(it); - - if (DeclAttrs->empty()) { - delete DeclAttrs; - DeclAttrs = 0; - } + // release attributes. + delete it->second; + invalidateAttrs(); } void Decl::addAttr(Attr *NewAttr) { @@ -358,6 +353,19 @@ void Decl::addAttr(Attr *NewAttr) { HasAttrs = true; } +void Decl::invalidateAttrs() { + if (!HasAttrs) return; + + HasAttrs = false; + (*DeclAttrs)[this] = 0; + DeclAttrs->erase(this); + + if (DeclAttrs->empty()) { + delete DeclAttrs; + DeclAttrs = 0; + } +} + const Attr *Decl::getAttrs() const { if (!HasAttrs) return 0; diff --git a/lib/Sema/SemaDecl.cpp b/lib/Sema/SemaDecl.cpp index eb0abafcaa..230ea3cb9d 100644 --- a/lib/Sema/SemaDecl.cpp +++ b/lib/Sema/SemaDecl.cpp @@ -276,7 +276,6 @@ static bool DeclHasAttr(const Decl *decl, const Attr *target) { static void MergeAttributes(Decl *New, Decl *Old) { Attr *attr = const_cast(Old->getAttrs()), *tmp; -// FIXME: fix this code to cleanup the Old attrs correctly while (attr) { tmp = attr; attr = attr->getNext(); @@ -288,6 +287,8 @@ static void MergeAttributes(Decl *New, Decl *Old) { delete(tmp); } } + + Old->invalidateAttrs(); } /// MergeFunctionDecl - We just parsed a function 'New' from diff --git a/test/CodeGen/merge-attrs.c b/test/CodeGen/merge-attrs.c new file mode 100644 index 0000000000..5ac0217e2b --- /dev/null +++ b/test/CodeGen/merge-attrs.c @@ -0,0 +1,13 @@ +// RUN: clang %s -emit-llvm + +void *malloc(int size) __attribute__ ((__nothrow__)); + +inline static void __zend_malloc() { + malloc(1); +} + +void *malloc(int size) __attribute__ ((__nothrow__)); + +void fontFetch() { + __zend_malloc(1); +}