/* * Copyright (C) 2005, 2006, 2007, 2010 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef JSRetainPtr_h #define JSRetainPtr_h #include #include inline void JSRetain(JSStringRef string) { JSStringRetain(string); } inline void JSRelease(JSStringRef string) { JSStringRelease(string); } enum AdoptTag { Adopt }; template class JSRetainPtr { public: JSRetainPtr() : m_ptr(0) { } JSRetainPtr(T ptr) : m_ptr(ptr) { if (ptr) JSRetain(ptr); } JSRetainPtr(AdoptTag, T ptr) : m_ptr(ptr) { } JSRetainPtr(const JSRetainPtr&); template JSRetainPtr(const JSRetainPtr&); ~JSRetainPtr(); T get() const { return m_ptr; } void clear(); T leakRef(); T operator->() const { return m_ptr; } bool operator!() const { return !m_ptr; } // This conversion operator allows implicit conversion to bool but not to other integer types. typedef T JSRetainPtr::*UnspecifiedBoolType; operator UnspecifiedBoolType() const { return m_ptr ? &JSRetainPtr::m_ptr : 0; } JSRetainPtr& operator=(const JSRetainPtr&); template JSRetainPtr& operator=(const JSRetainPtr&); JSRetainPtr& operator=(T); template JSRetainPtr& operator=(U*); void adopt(T); void swap(JSRetainPtr&); // FIXME: Remove releaseRef once we change all callers to call leakRef instead. T releaseRef() { return leakRef(); } private: T m_ptr; }; template inline JSRetainPtr::JSRetainPtr(const JSRetainPtr& o) : m_ptr(o.m_ptr) { if (m_ptr) JSRetain(m_ptr); } template template inline JSRetainPtr::JSRetainPtr(const JSRetainPtr& o) : m_ptr(o.get()) { if (m_ptr) JSRetain(m_ptr); } template inline JSRetainPtr::~JSRetainPtr() { if (m_ptr) JSRelease(m_ptr); } template inline void JSRetainPtr::clear() { if (T ptr = m_ptr) { m_ptr = 0; JSRelease(ptr); } } template inline T JSRetainPtr::leakRef() { T ptr = m_ptr; m_ptr = 0; return ptr; } template inline JSRetainPtr& JSRetainPtr::operator=(const JSRetainPtr& o) { T optr = o.get(); if (optr) JSRetain(optr); T ptr = m_ptr; m_ptr = optr; if (ptr) JSRelease(ptr); return *this; } template template inline JSRetainPtr& JSRetainPtr::operator=(const JSRetainPtr& o) { T optr = o.get(); if (optr) JSRetain(optr); T ptr = m_ptr; m_ptr = optr; if (ptr) JSRelease(ptr); return *this; } template inline JSRetainPtr& JSRetainPtr::operator=(T optr) { if (optr) JSRetain(optr); T ptr = m_ptr; m_ptr = optr; if (ptr) JSRelease(ptr); return *this; } template inline void JSRetainPtr::adopt(T optr) { T ptr = m_ptr; m_ptr = optr; if (ptr) JSRelease(ptr); } template template inline JSRetainPtr& JSRetainPtr::operator=(U* optr) { if (optr) JSRetain(optr); T ptr = m_ptr; m_ptr = optr; if (ptr) JSRelease(ptr); return *this; } template inline void JSRetainPtr::swap(JSRetainPtr& o) { std::swap(m_ptr, o.m_ptr); } template inline void swap(JSRetainPtr& a, JSRetainPtr& b) { a.swap(b); } template inline bool operator==(const JSRetainPtr& a, const JSRetainPtr& b) { return a.get() == b.get(); } template inline bool operator==(const JSRetainPtr& a, U* b) { return a.get() == b; } template inline bool operator==(T* a, const JSRetainPtr& b) { return a == b.get(); } template inline bool operator!=(const JSRetainPtr& a, const JSRetainPtr& b) { return a.get() != b.get(); } template inline bool operator!=(const JSRetainPtr& a, U* b) { return a.get() != b; } template inline bool operator!=(T* a, const JSRetainPtr& b) { return a != b.get(); } #endif // JSRetainPtr_h