X-Git-Url: https://git.saurik.com/apple/javascriptcore.git/blobdiff_plain/b80e619319b1def83d1e8b4f84042b661be1be7f..14957cd040308e3eeec43d26bae5d76da13fcd85:/wtf/PassOwnArrayPtr.h diff --git a/wtf/PassOwnArrayPtr.h b/wtf/PassOwnArrayPtr.h new file mode 100644 index 0000000..3f30924 --- /dev/null +++ b/wtf/PassOwnArrayPtr.h @@ -0,0 +1,227 @@ +/* + * Copyright (C) 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. + * + * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 INC. OR + * 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 WTF_PassOwnArrayPtr_h +#define WTF_PassOwnArrayPtr_h + +#include "Assertions.h" +#include "NullPtr.h" +#include "TypeTraits.h" + +namespace WTF { + +template class OwnArrayPtr; +template class PassOwnArrayPtr; +template PassOwnArrayPtr adoptArrayPtr(T*); +template void deleteOwnedArrayPtr(T* ptr); + +template class PassOwnArrayPtr { +public: + typedef T* PtrType; + + PassOwnArrayPtr() : m_ptr(0) { } + +#if !defined(LOOSE_PASS_OWN_PTR) || !HAVE(NULLPTR) + PassOwnArrayPtr(std::nullptr_t) : m_ptr(0) { } +#endif + + // It somewhat breaks the type system to allow transfer of ownership out of + // a const PassOwnArrayPtr. However, it makes it much easier to work with PassOwnArrayPtr + // temporaries, and we don't have a need to use real const PassOwnArrayPtrs anyway. + PassOwnArrayPtr(const PassOwnArrayPtr& o) : m_ptr(o.leakPtr()) { } + template PassOwnArrayPtr(const PassOwnArrayPtr& o) : m_ptr(o.leakPtr()) { } + + ~PassOwnArrayPtr() { deleteOwnedArrayPtr(m_ptr); } + + PtrType get() const { return m_ptr; } + + void clear(); + PtrType leakPtr() const WARN_UNUSED_RETURN; + + T& operator*() const { ASSERT(m_ptr); return *m_ptr; } + PtrType operator->() const { ASSERT(m_ptr); return m_ptr; } + + bool operator!() const { return !m_ptr; } + + // This conversion operator allows implicit conversion to bool but not to other integer types. +#if COMPILER(WINSCW) + operator bool() const { return m_ptr; } +#else + typedef PtrType PassOwnArrayPtr::*UnspecifiedBoolType; + operator UnspecifiedBoolType() const { return m_ptr ? &PassOwnArrayPtr::m_ptr : 0; } +#endif + + PassOwnArrayPtr& operator=(const PassOwnArrayPtr&); +#if !defined(LOOSE_PASS_OWN_ARRAY_PTR) || !HAVE(NULLPTR) + PassOwnArrayPtr& operator=(std::nullptr_t) { clear(); return *this; } +#endif + template PassOwnArrayPtr& operator=(const PassOwnArrayPtr&); + + template friend PassOwnArrayPtr adoptArrayPtr(U*); + +#ifdef LOOSE_PASS_OWN_ARRAY_PTR + PassOwnArrayPtr(PtrType ptr) : m_ptr(ptr) { } + PassOwnArrayPtr& operator=(PtrType); +#endif + +private: +#ifndef LOOSE_PASS_OWN_ARRAY_PTR + explicit PassOwnArrayPtr(PtrType ptr) : m_ptr(ptr) { } +#endif + + mutable PtrType m_ptr; +}; + +template inline void PassOwnArrayPtr::clear() +{ + PtrType ptr = m_ptr; + m_ptr = 0; + deleteOwnedArrayPtr(ptr); +} + +template inline typename PassOwnArrayPtr::PtrType PassOwnArrayPtr::leakPtr() const +{ + PtrType ptr = m_ptr; + m_ptr = 0; + return ptr; +} + +#ifdef LOOSE_PASS_OWN_ARRAY_PTR +template inline PassOwnArrayPtr& PassOwnArrayPtr::operator=(PtrType optr) +{ + PtrType ptr = m_ptr; + m_ptr = optr; + ASSERT(!ptr || m_ptr != ptr); + if (ptr) + deleteOwnedArrayPtr(ptr); + return *this; +} +#endif + +template inline PassOwnArrayPtr& PassOwnArrayPtr::operator=(const PassOwnArrayPtr& optr) +{ + PtrType ptr = m_ptr; + m_ptr = optr.leakPtr(); + ASSERT(!ptr || m_ptr != ptr); + if (ptr) + deleteOwnedArrayPtr(ptr); + return *this; +} + +template template inline PassOwnArrayPtr& PassOwnArrayPtr::operator=(const PassOwnArrayPtr& optr) +{ + PtrType ptr = m_ptr; + m_ptr = optr.leakPtr(); + ASSERT(!ptr || m_ptr != ptr); + if (ptr) + deleteOwnedArrayPtr(ptr); + return *this; +} + +template inline bool operator==(const PassOwnArrayPtr& a, const PassOwnArrayPtr& b) +{ + return a.get() == b.get(); +} + +template inline bool operator==(const PassOwnArrayPtr& a, const OwnArrayPtr& b) +{ + return a.get() == b.get(); +} + +template inline bool operator==(const OwnArrayPtr& a, const PassOwnArrayPtr& b) +{ + return a.get() == b.get(); +} + +template inline bool operator==(const PassOwnArrayPtr& a, U* b) +{ + return a.get() == b; +} + +template inline bool operator==(T* a, const PassOwnArrayPtr& b) +{ + return a == b.get(); +} + +template inline bool operator!=(const PassOwnArrayPtr& a, const PassOwnArrayPtr& b) +{ + return a.get() != b.get(); +} + +template inline bool operator!=(const PassOwnArrayPtr& a, const OwnArrayPtr& b) +{ + return a.get() != b.get(); +} + +template inline bool operator!=(const OwnArrayPtr& a, const PassOwnArrayPtr& b) +{ + return a.get() != b.get(); +} + +template inline bool operator!=(const PassOwnArrayPtr& a, U* b) +{ + return a.get() != b; +} + +template inline bool operator!=(T* a, const PassOwnArrayPtr& b) +{ + return a != b.get(); +} + +template inline PassOwnArrayPtr adoptArrayPtr(T* ptr) +{ + return PassOwnArrayPtr(ptr); +} + +template inline void deleteOwnedArrayPtr(T* ptr) +{ + typedef char known[sizeof(T) ? 1 : -1]; + if (sizeof(known)) + delete [] ptr; +} + +template inline PassOwnArrayPtr static_pointer_cast(const PassOwnArrayPtr& p) +{ + return adoptArrayPtr(static_cast(p.leakPtr())); +} + +template inline PassOwnArrayPtr const_pointer_cast(const PassOwnArrayPtr& p) +{ + return adoptArrayPtr(const_cast(p.leakPtr())); +} + +template inline T* getPtr(const PassOwnArrayPtr& p) +{ + return p.get(); +} + +} // namespace WTF + +using WTF::PassOwnArrayPtr; +using WTF::adoptArrayPtr; +using WTF::const_pointer_cast; +using WTF::static_pointer_cast; + +#endif // WTF_PassOwnArrayPtr_h