mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-11-15 07:09:43 +03:00
3483407ddc
Even if the pointer value is const, the value they point to is not necessarily const, so these functions should not add the qualifier. This also removes the redundant non-const implementations of these operators.
193 lines
4.6 KiB
C++
193 lines
4.6 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/Assertions.h>
|
|
#include <AK/Format.h>
|
|
#include <AK/RefCounted.h>
|
|
#include <AK/StdLibExtras.h>
|
|
#include <AK/Traits.h>
|
|
#include <AK/Types.h>
|
|
|
|
#define NONNULLOWNPTR_SCRUB_BYTE 0xf1
|
|
|
|
namespace AK {
|
|
|
|
template<typename T>
|
|
class WeakPtr;
|
|
|
|
template<typename T>
|
|
class [[nodiscard]] NonnullOwnPtr {
|
|
public:
|
|
using ElementType = T;
|
|
|
|
enum AdoptTag { Adopt };
|
|
|
|
NonnullOwnPtr(AdoptTag, T& ptr)
|
|
: m_ptr(&ptr)
|
|
{
|
|
static_assert(
|
|
requires { requires typename T::AllowOwnPtr()(); } || !requires { requires !typename T::AllowOwnPtr()(); declval<T>().ref(); declval<T>().unref(); },
|
|
"Use NonnullRefPtr<> for RefCounted types");
|
|
}
|
|
NonnullOwnPtr(NonnullOwnPtr&& other)
|
|
: m_ptr(other.leak_ptr())
|
|
{
|
|
VERIFY(m_ptr);
|
|
}
|
|
template<typename U>
|
|
NonnullOwnPtr(NonnullOwnPtr<U>&& other)
|
|
: m_ptr(other.leak_ptr())
|
|
{
|
|
VERIFY(m_ptr);
|
|
}
|
|
~NonnullOwnPtr()
|
|
{
|
|
clear();
|
|
#ifdef SANITIZE_PTRS
|
|
m_ptr = (T*)(explode_byte(NONNULLOWNPTR_SCRUB_BYTE));
|
|
#endif
|
|
}
|
|
|
|
NonnullOwnPtr(NonnullOwnPtr const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr(NonnullOwnPtr<U> const&) = delete;
|
|
NonnullOwnPtr& operator=(NonnullOwnPtr const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr& operator=(NonnullOwnPtr<U> const&) = delete;
|
|
|
|
template<typename U>
|
|
NonnullOwnPtr(RefPtr<U> const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr(NonnullRefPtr<U> const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr(WeakPtr<U> const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr& operator=(RefPtr<U> const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr& operator=(NonnullRefPtr<U> const&) = delete;
|
|
template<typename U>
|
|
NonnullOwnPtr& operator=(WeakPtr<U> const&) = delete;
|
|
|
|
NonnullOwnPtr& operator=(NonnullOwnPtr&& other)
|
|
{
|
|
NonnullOwnPtr ptr(move(other));
|
|
swap(ptr);
|
|
return *this;
|
|
}
|
|
|
|
template<typename U>
|
|
NonnullOwnPtr& operator=(NonnullOwnPtr<U>&& other)
|
|
{
|
|
NonnullOwnPtr ptr(move(other));
|
|
swap(ptr);
|
|
return *this;
|
|
}
|
|
|
|
[[nodiscard]] T* leak_ptr()
|
|
{
|
|
return exchange(m_ptr, nullptr);
|
|
}
|
|
|
|
ALWAYS_INLINE RETURNS_NONNULL T* ptr() const
|
|
{
|
|
VERIFY(m_ptr);
|
|
return m_ptr;
|
|
}
|
|
|
|
ALWAYS_INLINE RETURNS_NONNULL T* operator->() const { return ptr(); }
|
|
|
|
ALWAYS_INLINE T& operator*() const { return *ptr(); }
|
|
|
|
ALWAYS_INLINE RETURNS_NONNULL operator T*() const { return ptr(); }
|
|
|
|
operator bool() const = delete;
|
|
bool operator!() const = delete;
|
|
|
|
void swap(NonnullOwnPtr& other)
|
|
{
|
|
::swap(m_ptr, other.m_ptr);
|
|
}
|
|
|
|
template<typename U>
|
|
void swap(NonnullOwnPtr<U>& other)
|
|
{
|
|
::swap(m_ptr, other.m_ptr);
|
|
}
|
|
|
|
template<typename U>
|
|
NonnullOwnPtr<U> release_nonnull()
|
|
{
|
|
VERIFY(m_ptr);
|
|
return NonnullOwnPtr<U>(NonnullOwnPtr<U>::Adopt, static_cast<U&>(*leak_ptr()));
|
|
}
|
|
|
|
private:
|
|
void clear()
|
|
{
|
|
if (!m_ptr)
|
|
return;
|
|
delete m_ptr;
|
|
m_ptr = nullptr;
|
|
}
|
|
|
|
T* m_ptr = nullptr;
|
|
};
|
|
|
|
#if !defined(KERNEL)
|
|
|
|
template<typename T>
|
|
inline NonnullOwnPtr<T> adopt_own(T& object)
|
|
{
|
|
return NonnullOwnPtr<T>(NonnullOwnPtr<T>::Adopt, object);
|
|
}
|
|
|
|
template<class T, class... Args>
|
|
requires(IsConstructible<T, Args...>) inline NonnullOwnPtr<T> make(Args&&... args)
|
|
{
|
|
return NonnullOwnPtr<T>(NonnullOwnPtr<T>::Adopt, *new T(forward<Args>(args)...));
|
|
}
|
|
|
|
// FIXME: Remove once P0960R3 is available in Clang.
|
|
template<class T, class... Args>
|
|
inline NonnullOwnPtr<T> make(Args&&... args)
|
|
{
|
|
return NonnullOwnPtr<T>(NonnullOwnPtr<T>::Adopt, *new T { forward<Args>(args)... });
|
|
}
|
|
|
|
#endif
|
|
|
|
template<typename T>
|
|
struct Traits<NonnullOwnPtr<T>> : public GenericTraits<NonnullOwnPtr<T>> {
|
|
using PeekType = T*;
|
|
using ConstPeekType = const T*;
|
|
static unsigned hash(NonnullOwnPtr<T> const& p) { return ptr_hash((FlatPtr)p.ptr()); }
|
|
static bool equals(NonnullOwnPtr<T> const& a, NonnullOwnPtr<T> const& b) { return a.ptr() == b.ptr(); }
|
|
};
|
|
|
|
template<typename T, typename U>
|
|
inline void swap(NonnullOwnPtr<T>& a, NonnullOwnPtr<U>& b)
|
|
{
|
|
a.swap(b);
|
|
}
|
|
|
|
template<typename T>
|
|
struct Formatter<NonnullOwnPtr<T>> : Formatter<const T*> {
|
|
ErrorOr<void> format(FormatBuilder& builder, NonnullOwnPtr<T> const& value)
|
|
{
|
|
return Formatter<const T*>::format(builder, value.ptr());
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
#if !defined(KERNEL)
|
|
using AK::adopt_own;
|
|
using AK::make;
|
|
#endif
|
|
using AK::NonnullOwnPtr;
|