X-Git-Url: https://git.saurik.com/apple/javascriptcore.git/blobdiff_plain/b80e619319b1def83d1e8b4f84042b661be1be7f..14957cd040308e3eeec43d26bae5d76da13fcd85:/wtf/OwnArrayPtr.h diff --git a/wtf/OwnArrayPtr.h b/wtf/OwnArrayPtr.h index 61375c7..2828698 100644 --- a/wtf/OwnArrayPtr.h +++ b/wtf/OwnArrayPtr.h @@ -1,5 +1,5 @@ /* - * Copyright (C) 2006 Apple Computer, Inc. + * Copyright (C) 2006, 2010 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 @@ -21,52 +21,153 @@ #ifndef WTF_OwnArrayPtr_h #define WTF_OwnArrayPtr_h +#include "Assertions.h" +#include "Noncopyable.h" +#include "NullPtr.h" +#include "PassOwnArrayPtr.h" #include -#include -#include namespace WTF { - template class OwnArrayPtr : public Noncopyable { - public: - explicit OwnArrayPtr(T* ptr = 0) : m_ptr(ptr) { } - ~OwnArrayPtr() { safeDelete(); } +template class PassOwnArrayPtr; +template PassOwnArrayPtr adoptArrayPtr(T*); - T* get() const { return m_ptr; } - T* release() { T* ptr = m_ptr; m_ptr = 0; return ptr; } +template class OwnArrayPtr { +public: + typedef T* PtrType; - void set(T* ptr) { ASSERT(m_ptr != ptr); safeDelete(); m_ptr = ptr; } - void clear() { safeDelete(); m_ptr = 0; } + OwnArrayPtr() : m_ptr(0) { } - T& operator*() const { ASSERT(m_ptr); return *m_ptr; } - T* operator->() const { ASSERT(m_ptr); return m_ptr; } + // See comment in PassOwnArrayPtr.h for why this takes a const reference. + template OwnArrayPtr(const PassOwnArrayPtr& o); - T& operator[](std::ptrdiff_t i) const { ASSERT(m_ptr); ASSERT(i >= 0); return m_ptr[i]; } + // This copy constructor is used implicitly by gcc when it generates + // transients for assigning a PassOwnArrayPtr object to a stack-allocated + // OwnArrayPtr object. It should never be called explicitly and gcc + // should optimize away the constructor when generating code. + OwnArrayPtr(const OwnArrayPtr&); - bool operator!() const { return !m_ptr; } + ~OwnArrayPtr() { deleteOwnedArrayPtr(m_ptr); } - // This conversion operator allows implicit conversion to bool but not to other integer types. + PtrType get() const { return m_ptr; } + + void clear(); + PassOwnArrayPtr release(); + PtrType leakPtr() WARN_UNUSED_RETURN; + + T& operator*() const { ASSERT(m_ptr); return *m_ptr; } + PtrType operator->() const { ASSERT(m_ptr); return m_ptr; } + + T& operator[](std::ptrdiff_t i) const { ASSERT(m_ptr); ASSERT(i >= 0); return m_ptr[i]; } + + 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; } + operator bool() const { return m_ptr; } #else - typedef T* OwnArrayPtr::*UnspecifiedBoolType; - operator UnspecifiedBoolType() const { return m_ptr ? &OwnArrayPtr::m_ptr : 0; } + typedef T* OwnArrayPtr::*UnspecifiedBoolType; + operator UnspecifiedBoolType() const { return m_ptr ? &OwnArrayPtr::m_ptr : 0; } #endif - void swap(OwnArrayPtr& o) { std::swap(m_ptr, o.m_ptr); } + OwnArrayPtr& operator=(const PassOwnArrayPtr&); + OwnArrayPtr& operator=(std::nullptr_t) { clear(); return *this; } + template OwnArrayPtr& operator=(const PassOwnArrayPtr&); + + void swap(OwnArrayPtr& o) { std::swap(m_ptr, o.m_ptr); } - private: - void safeDelete() { typedef char known[sizeof(T) ? 1 : -1]; if (sizeof(known)) delete [] m_ptr; } +#ifdef LOOSE_OWN_ARRAY_PTR + explicit OwnArrayPtr(PtrType ptr) : m_ptr(ptr) { } + void set(PtrType); +#endif - T* m_ptr; - }; - - template inline void swap(OwnArrayPtr& a, OwnArrayPtr& b) { a.swap(b); } +private: + PtrType m_ptr; +}; + +template template inline OwnArrayPtr::OwnArrayPtr(const PassOwnArrayPtr& o) + : m_ptr(o.leakPtr()) +{ +} + +template inline void OwnArrayPtr::clear() +{ + PtrType ptr = m_ptr; + m_ptr = 0; + deleteOwnedArrayPtr(ptr); +} + +template inline PassOwnArrayPtr OwnArrayPtr::release() +{ + PtrType ptr = m_ptr; + m_ptr = 0; + return adoptArrayPtr(ptr); +} + +template inline typename OwnArrayPtr::PtrType OwnArrayPtr::leakPtr() +{ + PtrType ptr = m_ptr; + m_ptr = 0; + return ptr; +} + +#ifdef LOOSE_OWN_ARRAY_PTR +template inline void OwnArrayPtr::set(PtrType ptr) +{ + ASSERT(!ptr || m_ptr != ptr); + PtrType oldPtr = m_ptr; + m_ptr = ptr; + deleteOwnedArrayPtr(oldPtr); +} +#endif - template inline T* getPtr(const OwnArrayPtr& p) - { - return p.get(); - } +template inline OwnArrayPtr& OwnArrayPtr::operator=(const PassOwnArrayPtr& o) +{ + PtrType ptr = m_ptr; + m_ptr = o.leakPtr(); + ASSERT(!ptr || m_ptr != ptr); + deleteOwnedArrayPtr(ptr); + return *this; +} + +template template inline OwnArrayPtr& OwnArrayPtr::operator=(const PassOwnArrayPtr& o) +{ + PtrType ptr = m_ptr; + m_ptr = o.leakPtr(); + ASSERT(!ptr || m_ptr != ptr); + deleteOwnedArrayPtr(ptr); + return *this; +} + +template inline void swap(OwnArrayPtr& a, OwnArrayPtr& b) +{ + a.swap(b); +} + +template inline bool operator==(const OwnArrayPtr& a, U* b) +{ + return a.get() == b; +} + +template inline bool operator==(T* a, const OwnArrayPtr& b) +{ + return a == b.get(); +} + +template inline bool operator!=(const OwnArrayPtr& a, U* b) +{ + return a.get() != b; +} + +template inline bool operator!=(T* a, const OwnArrayPtr& b) +{ + return a != b.get(); +} + +template inline T* getPtr(const OwnArrayPtr& p) +{ + return p.get(); +} } // namespace WTF