mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 18:02:05 -05:00
28da5b002f
This is just like OwnPtr (also single-owner), except it cannot be null. NonnullOwnPtr is perfect as the return type of functions that never need to return nullptr. It's also useful as an argument type to encode the fact that the argument must not be nullptr. The make<Foo>() helper is changed to return NonnullOwnPtr<Foo>. Note: You can move() out of a NonnullOwnPtr, and after that the object is in an invalid state. Internally it will be a nullptr at this point, so we'll still catch misuse, but the only thing that should be done in this state is running the destructor. I've used consumable annotations to generate some warnings when using a NonnullOwnPtr after moving from it, but these only work when compiling with clang, so be aware of that.
152 lines
3.3 KiB
C++
152 lines
3.3 KiB
C++
#pragma once
|
|
|
|
#include <AK/NonnullOwnPtr.h>
|
|
|
|
namespace AK {
|
|
|
|
template<typename T>
|
|
class OwnPtr {
|
|
public:
|
|
OwnPtr() {}
|
|
explicit OwnPtr(T* ptr)
|
|
: m_ptr(ptr)
|
|
{
|
|
}
|
|
OwnPtr(OwnPtr&& other)
|
|
: m_ptr(other.leak_ptr())
|
|
{
|
|
}
|
|
|
|
template<typename U>
|
|
OwnPtr(NonnullOwnPtr<U>&& other)
|
|
: m_ptr(static_cast<T*>(other.leak_ptr()))
|
|
{
|
|
}
|
|
template<typename U>
|
|
OwnPtr(OwnPtr<U>&& other)
|
|
: m_ptr(static_cast<T*>(other.leak_ptr()))
|
|
{
|
|
}
|
|
OwnPtr(std::nullptr_t) {};
|
|
~OwnPtr()
|
|
{
|
|
clear();
|
|
#ifdef SANITIZE_PTRS
|
|
if constexpr (sizeof(T*) == 8)
|
|
m_ptr = (T*)(0xe1e1e1e1e1e1e1e1);
|
|
else
|
|
m_ptr = (T*)(0xe1e1e1e1);
|
|
#endif
|
|
}
|
|
|
|
OwnPtr(const OwnPtr&) = delete;
|
|
template<typename U>
|
|
OwnPtr(const OwnPtr<U>&) = delete;
|
|
OwnPtr& operator=(const OwnPtr&) = delete;
|
|
template<typename U>
|
|
OwnPtr& operator=(const OwnPtr<U>&) = delete;
|
|
|
|
template<typename U>
|
|
OwnPtr(const RefPtr<U>&) = delete;
|
|
template<typename U>
|
|
OwnPtr(const NonnullRefPtr<U>&) = delete;
|
|
template<typename U>
|
|
OwnPtr(const WeakPtr<U>&) = delete;
|
|
template<typename U>
|
|
OwnPtr& operator=(const RefPtr<U>&) = delete;
|
|
template<typename U>
|
|
OwnPtr& operator=(const NonnullRefPtr<U>&) = delete;
|
|
template<typename U>
|
|
OwnPtr& operator=(const WeakPtr<U>&) = delete;
|
|
|
|
OwnPtr& operator=(OwnPtr&& other)
|
|
{
|
|
if (this != &other) {
|
|
delete m_ptr;
|
|
m_ptr = other.leak_ptr();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template<typename U>
|
|
OwnPtr& operator=(OwnPtr<U>&& other)
|
|
{
|
|
if (this != static_cast<void*>(&other)) {
|
|
delete m_ptr;
|
|
m_ptr = other.leak_ptr();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template<typename U>
|
|
OwnPtr& operator=(NonnullOwnPtr<U>&& other)
|
|
{
|
|
ASSERT(m_ptr != other.ptr());
|
|
delete m_ptr;
|
|
m_ptr = other.leak_ptr();
|
|
return *this;
|
|
}
|
|
|
|
OwnPtr& operator=(T* ptr)
|
|
{
|
|
if (m_ptr != ptr)
|
|
delete m_ptr;
|
|
m_ptr = ptr;
|
|
return *this;
|
|
}
|
|
|
|
OwnPtr& operator=(std::nullptr_t)
|
|
{
|
|
clear();
|
|
return *this;
|
|
}
|
|
|
|
void clear()
|
|
{
|
|
delete m_ptr;
|
|
m_ptr = nullptr;
|
|
}
|
|
|
|
bool operator!() const { return !m_ptr; }
|
|
|
|
T* leak_ptr()
|
|
{
|
|
T* leaked_ptr = m_ptr;
|
|
m_ptr = nullptr;
|
|
return leaked_ptr;
|
|
}
|
|
|
|
T* ptr() { return m_ptr; }
|
|
const T* ptr() const { return m_ptr; }
|
|
|
|
T* operator->() { return m_ptr; }
|
|
const T* operator->() const { return m_ptr; }
|
|
|
|
T& operator*() { return *m_ptr; }
|
|
const T& operator*() const { return *m_ptr; }
|
|
|
|
operator const T*() const { return m_ptr; }
|
|
operator T*() { return m_ptr; }
|
|
|
|
operator bool() { return !!m_ptr; }
|
|
|
|
private:
|
|
T* m_ptr = nullptr;
|
|
};
|
|
|
|
template<typename T>
|
|
struct Traits<OwnPtr<T>> : public GenericTraits<OwnPtr<T>> {
|
|
static unsigned hash(const OwnPtr<T>& p) { return (unsigned)p.ptr(); }
|
|
static void dump(const OwnPtr<T>& p) { kprintf("%p", p.ptr()); }
|
|
static bool equals(const OwnPtr<T>& a, const OwnPtr<T>& b) { return a.ptr() == b.ptr(); }
|
|
};
|
|
|
|
template<typename T>
|
|
inline const LogStream& operator<<(const LogStream& stream, const OwnPtr<T>& value)
|
|
{
|
|
return stream << value.ptr();
|
|
}
|
|
|
|
}
|
|
|
|
using AK::OwnPtr;
|