/* * Copyright (C) 2006, 2007, 2009, 2010, 2012 Apple Inc. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */ #ifndef SKY_ENGINE_CORE_DOM_TREESHARED_H_ #define SKY_ENGINE_CORE_DOM_TREESHARED_H_ #include "sky/engine/public/platform/WebPrivatePtr.h" #include "sky/engine/wtf/Assertions.h" #include "sky/engine/wtf/MainThread.h" #include "sky/engine/wtf/Noncopyable.h" namespace blink { #if ENABLE(SECURITY_ASSERT) template class TreeShared; template void adopted(TreeShared*); #endif template class TreeShared { WTF_MAKE_NONCOPYABLE(TreeShared); protected: TreeShared() : m_refCount(1) #if ENABLE(SECURITY_ASSERT) , m_deletionHasBegun(false) #if ENABLE(ASSERT) , m_inRemovedLastRefFunction(false) , m_adoptionIsRequired(true) #endif #endif { ASSERT(isMainThread()); } ~TreeShared() { ASSERT(isMainThread()); ASSERT(!m_refCount); ASSERT_WITH_SECURITY_IMPLICATION(m_deletionHasBegun); ASSERT(!m_adoptionIsRequired); } public: void ref() { ASSERT(isMainThread()); ASSERT_WITH_SECURITY_IMPLICATION(!m_deletionHasBegun); ASSERT(!m_inRemovedLastRefFunction); ASSERT(!m_adoptionIsRequired); ++m_refCount; } void deref() { ASSERT(isMainThread()); ASSERT(m_refCount > 0); ASSERT_WITH_SECURITY_IMPLICATION(!m_deletionHasBegun); ASSERT(!m_inRemovedLastRefFunction); ASSERT(!m_adoptionIsRequired); NodeType* thisNode = static_cast(this); if (!--m_refCount && !thisNode->hasTreeSharedParent()) { #if ENABLE(ASSERT) m_inRemovedLastRefFunction = true; #endif thisNode->removedLastRef(); } } int refCount() const { return m_refCount; } private: int m_refCount; #if ENABLE(SECURITY_ASSERT) public: bool m_deletionHasBegun; #if ENABLE(ASSERT) bool m_inRemovedLastRefFunction; private: friend void adopted<>(TreeShared*); bool m_adoptionIsRequired; #endif #endif }; #if ENABLE(SECURITY_ASSERT) template void adopted(TreeShared* object) { if (!object) return; ASSERT_WITH_SECURITY_IMPLICATION(!object->m_deletionHasBegun); #if ENABLE(ASSERT) ASSERT(!object->m_inRemovedLastRefFunction); object->m_adoptionIsRequired = false; #endif } #endif } #endif // SKY_ENGINE_CORE_DOM_TREESHARED_H_