mirror of
https://github.com/slint-ui/slint.git
synced 2025-10-24 01:02:38 +00:00
270 lines
7.5 KiB
C++
270 lines
7.5 KiB
C++
// Copyright © SixtyFPS GmbH <info@slint.dev>
|
|
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
|
|
|
|
#pragma once
|
|
|
|
#include <cstddef>
|
|
#include <new>
|
|
#include <algorithm>
|
|
#include <optional>
|
|
#include <atomic>
|
|
|
|
#ifdef __APPLE__
|
|
# include <AvailabilityMacros.h>
|
|
#endif
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
// In C++17, it is conditionally supported, but still valid for all compiler we care
|
|
# pragma GCC diagnostic ignored "-Winvalid-offsetof"
|
|
#endif
|
|
|
|
namespace vtable {
|
|
|
|
template<typename T>
|
|
struct VRefMut
|
|
{
|
|
const T *vtable;
|
|
void *instance;
|
|
};
|
|
|
|
struct Layout
|
|
{
|
|
std::size_t size;
|
|
std::size_t align;
|
|
};
|
|
|
|
// For the C++'s purpose, they are all the same
|
|
template<typename T>
|
|
using VRef = VRefMut<T>;
|
|
|
|
template<typename T>
|
|
using Pin = T;
|
|
|
|
template<typename T>
|
|
struct VBox
|
|
{
|
|
const T *vtable = nullptr;
|
|
void *instance = nullptr;
|
|
explicit VBox(const T *vtable, void *instance) : vtable(vtable), instance(instance) { }
|
|
VBox(const VBox &) = delete;
|
|
VBox() = default;
|
|
VBox &operator=(const VBox &) = delete;
|
|
~VBox()
|
|
{
|
|
if (vtable && instance) {
|
|
vtable->drop({ vtable, instance });
|
|
}
|
|
}
|
|
};
|
|
|
|
struct AllowPin;
|
|
|
|
template<typename Base, typename T, typename Flag = void>
|
|
struct VOffset
|
|
{
|
|
const T *vtable;
|
|
std::uintptr_t offset;
|
|
};
|
|
|
|
template<typename VTable, typename X>
|
|
struct VRcInner
|
|
{
|
|
template<typename VTable_, typename X_>
|
|
friend class VRc;
|
|
template<typename VTable_, typename X_>
|
|
friend class VWeak;
|
|
|
|
private:
|
|
VRcInner() : layout {} { }
|
|
const VTable *vtable = &X::static_vtable;
|
|
std::atomic<int> strong_ref = 1;
|
|
std::atomic<int> weak_ref = 1;
|
|
std::uint16_t data_offset = offsetof(VRcInner, data);
|
|
union {
|
|
X data;
|
|
Layout layout;
|
|
};
|
|
|
|
void *data_ptr() { return reinterpret_cast<char *>(this) + data_offset; }
|
|
~VRcInner() = delete;
|
|
};
|
|
|
|
struct Dyn
|
|
{
|
|
};
|
|
|
|
template<typename VTable, typename X = Dyn>
|
|
class VRc
|
|
{
|
|
VRcInner<VTable, X> *inner;
|
|
VRc(VRcInner<VTable, X> *inner) : inner(inner) { }
|
|
template<typename VTable_, typename X_>
|
|
friend class VWeak;
|
|
|
|
public:
|
|
~VRc()
|
|
{
|
|
if (!--inner->strong_ref) {
|
|
Layout layout = inner->vtable->drop_in_place({ inner->vtable, &inner->data });
|
|
layout.size = std::max<size_t>(layout.size, sizeof(Layout)); // because of the union
|
|
layout.size += inner->data_offset;
|
|
layout.align = std::max<size_t>(layout.align, alignof(VRcInner<VTable, Dyn>));
|
|
inner->layout = layout;
|
|
if (!--inner->weak_ref) {
|
|
inner->vtable->dealloc(inner->vtable, reinterpret_cast<uint8_t *>(inner), layout);
|
|
}
|
|
}
|
|
}
|
|
VRc(const VRc &other) : inner(other.inner) { inner->strong_ref++; }
|
|
VRc &operator=(const VRc &other)
|
|
{
|
|
if (inner == other.inner)
|
|
return *this;
|
|
this->~VRc();
|
|
new (this) VRc(other);
|
|
return *this;
|
|
}
|
|
/// Construct a new VRc holding an X.
|
|
///
|
|
/// The type X must have a static member `static_vtable` of type VTable
|
|
template<typename... Args>
|
|
static VRc make(Args... args)
|
|
{
|
|
#if !defined(__APPLE__) || MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_14
|
|
auto mem = ::operator new(sizeof(VRcInner<VTable, X>),
|
|
static_cast<std::align_val_t>(alignof(VRcInner<VTable, X>)));
|
|
#else
|
|
auto mem = ::operator new(sizeof(VRcInner<VTable, X>));
|
|
#endif
|
|
auto inner = new (mem) VRcInner<VTable, X>;
|
|
new (&inner->data) X(args...);
|
|
return VRc(inner);
|
|
}
|
|
|
|
const X *operator->() const { return &inner->data; }
|
|
const X &operator*() const { return inner->data; }
|
|
X *operator->() { return &inner->data; }
|
|
X &operator*() { return inner->data; }
|
|
|
|
const VRc<VTable, Dyn> &into_dyn() const
|
|
{
|
|
return *reinterpret_cast<const VRc<VTable, Dyn> *>(this);
|
|
}
|
|
|
|
VRef<VTable> borrow() const { return { inner->vtable, inner->data_ptr() }; }
|
|
|
|
friend bool operator==(const VRc &a, const VRc &b) { return a.inner == b.inner; }
|
|
friend bool operator!=(const VRc &a, const VRc &b) { return a.inner != b.inner; }
|
|
const VTable *vtable() const { return inner->vtable; }
|
|
};
|
|
|
|
template<typename VTable, typename X = Dyn>
|
|
class VWeak
|
|
{
|
|
VRcInner<VTable, X> *inner = nullptr;
|
|
|
|
public:
|
|
VWeak() = default;
|
|
~VWeak()
|
|
{
|
|
if (inner && !--inner->weak_ref) {
|
|
inner->vtable->dealloc(inner->vtable, reinterpret_cast<uint8_t *>(inner),
|
|
inner->layout);
|
|
}
|
|
}
|
|
VWeak(const VWeak &other) : inner(other.inner) { inner && inner->weak_ref++; }
|
|
VWeak(const VRc<VTable, X> &other) : inner(other.inner) { inner && inner->weak_ref++; }
|
|
VWeak &operator=(const VWeak &other)
|
|
{
|
|
if (inner == other.inner)
|
|
return *this;
|
|
this->~VWeak();
|
|
new (this) VWeak(other);
|
|
return *this;
|
|
}
|
|
|
|
std::optional<VRc<VTable, X>> lock() const
|
|
{
|
|
if (!inner || inner->strong_ref == 0)
|
|
return {};
|
|
inner->strong_ref++;
|
|
return { VRc<VTable, X>(inner) };
|
|
}
|
|
|
|
const VWeak<VTable, Dyn> &into_dyn() const
|
|
{
|
|
return *reinterpret_cast<const VWeak<VTable, Dyn> *>(this);
|
|
}
|
|
|
|
friend bool operator==(const VWeak &a, const VWeak &b) { return a.inner == b.inner; }
|
|
friend bool operator!=(const VWeak &a, const VWeak &b) { return a.inner != b.inner; }
|
|
const VTable *vtable() const { return inner ? inner->vtable : nullptr; }
|
|
};
|
|
|
|
template<typename VTable, typename MappedType>
|
|
class VRcMapped
|
|
{
|
|
VRc<VTable, Dyn> parent_strong;
|
|
MappedType *object;
|
|
|
|
template<typename VTable_, typename MappedType_>
|
|
friend class VWeakMapped;
|
|
|
|
public:
|
|
/// Constructs a pointer to MappedType that shares ownership with parent_strong.
|
|
template<typename X>
|
|
explicit VRcMapped(VRc<VTable, X> parent_strong, MappedType *object)
|
|
: parent_strong(parent_strong.into_dyn()), object(object)
|
|
{
|
|
}
|
|
|
|
const MappedType *operator->() const { return object; }
|
|
const MappedType &operator*() const { return *object; }
|
|
MappedType *operator->() { return object; }
|
|
MappedType &operator*() { return *object; }
|
|
};
|
|
|
|
template<typename VTable, typename MappedType>
|
|
class VWeakMapped
|
|
{
|
|
VWeak<VTable, Dyn> parent_weak;
|
|
MappedType *object = nullptr;
|
|
|
|
public:
|
|
VWeakMapped(const VRcMapped<VTable, MappedType> &strong)
|
|
: parent_weak(strong.parent_strong), object(strong.object)
|
|
{
|
|
}
|
|
VWeakMapped() = default;
|
|
|
|
std::optional<VRcMapped<VTable, MappedType>> lock() const
|
|
{
|
|
if (auto parent = parent_weak.lock()) {
|
|
return VRcMapped<VTable, MappedType>(std::move(*parent), object);
|
|
} else {
|
|
return {};
|
|
}
|
|
}
|
|
};
|
|
|
|
template<typename VTable>
|
|
inline void dealloc(const VTable *, uint8_t *ptr, [[maybe_unused]] Layout layout)
|
|
{
|
|
#ifdef __cpp_sized_deallocation
|
|
::operator delete(reinterpret_cast<void *>(ptr), layout.size,
|
|
static_cast<std::align_val_t>(layout.align));
|
|
#elif !defined(__APPLE__) || MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_14
|
|
::operator delete(reinterpret_cast<void *>(ptr), static_cast<std::align_val_t>(layout.align));
|
|
#else
|
|
::operator delete(reinterpret_cast<void *>(ptr));
|
|
#endif
|
|
}
|
|
|
|
template<typename VTable, typename T>
|
|
inline Layout drop_in_place(VRefMut<VTable> item_tree)
|
|
{
|
|
reinterpret_cast<T *>(item_tree.instance)->~T();
|
|
return vtable::Layout { sizeof(T), alignof(T) };
|
|
}
|
|
|
|
} // namespace vtable
|