5#ifndef INCLUDE_CPPGC_INTERNAL_MEMBER_STORAGE_H_
6#define INCLUDE_CPPGC_INTERNAL_MEMBER_STORAGE_H_
26#if defined(CPPGC_POINTER_COMPRESSION)
31#define CPPGC_CONST __attribute__((const))
32#define CPPGC_REQUIRE_CONSTANT_INIT \
33 __attribute__((require_constant_initialization))
36#define CPPGC_REQUIRE_CONSTANT_INIT
41 V8_INLINE CPPGC_CONST
static uintptr_t Get() {
46 V8_INLINE CPPGC_CONST
static bool IsSet() {
48 return (g_base_.base & ~kLowerHalfWordMask) != 0;
53 static constexpr uintptr_t kLowerHalfWordMask =
54 (api_constants::kCagedHeapReservationAlignment - 1);
56 static union alignas(api_constants::kCachelineSize) Base {
58 char cache_line[api_constants::kCachelineSize];
59 } g_base_ CPPGC_REQUIRE_CONSTANT_INIT;
61 CageBaseGlobal() =
delete;
63 V8_INLINE static bool IsBaseConsistent() {
64 return kLowerHalfWordMask == (g_base_.base & kLowerHalfWordMask);
67 friend class CageBaseGlobalUpdater;
70#undef CPPGC_REQUIRE_CONSTANT_INIT
75 struct AtomicInitializerTag {};
77 using IntegralType = uint32_t;
78 static constexpr auto kWriteBarrierSlotType =
79 WriteBarrierSlotType::kCompressed;
81 V8_INLINE CompressedPointer() : value_(0u) {}
82 V8_INLINE explicit CompressedPointer(
const void* value,
83 AtomicInitializerTag) {
86 V8_INLINE explicit CompressedPointer(
const void* ptr)
87 : value_(Compress(ptr)) {}
88 V8_INLINE explicit CompressedPointer(std::nullptr_t) : value_(0u) {}
89 V8_INLINE explicit CompressedPointer(SentinelPointer)
90 : value_(kCompressedSentinel) {}
92 V8_INLINE const void* Load()
const {
return Decompress(value_); }
93 V8_INLINE const void* LoadAtomic()
const {
95 reinterpret_cast<const std::atomic<IntegralType>&
>(value_).load(
96 std::memory_order_relaxed));
99 V8_INLINE void Store(
const void* ptr) { value_ = Compress(ptr); }
100 V8_INLINE void StoreAtomic(
const void* value) {
101 reinterpret_cast<std::atomic<IntegralType>&
>(value_).store(
102 Compress(value), std::memory_order_relaxed);
106 V8_INLINE bool IsCleared()
const {
return !value_; }
108 V8_INLINE bool IsSentinel()
const {
return value_ == kCompressedSentinel; }
110 V8_INLINE uint32_t GetAsInteger()
const {
return value_; }
113 return a.value_ == b.value_;
116 return a.value_ != b.value_;
119 return a.value_ < b.value_;
122 return a.value_ <= b.value_;
125 return a.value_ > b.value_;
128 return a.value_ >= b.value_;
131 static V8_INLINE IntegralType Compress(
const void* ptr) {
132 static_assert(SentinelPointer::kSentinelValue ==
133 1 << api_constants::kPointerCompressionShift,
134 "The compression scheme relies on the sentinel encoded as 1 "
135 "<< kPointerCompressionShift");
136 static constexpr size_t kGigaCageMask =
137 ~(api_constants::kCagedHeapReservationAlignment - 1);
138 static constexpr size_t kPointerCompressionShiftMask =
139 (1 << api_constants::kPointerCompressionShift) - 1;
142 const uintptr_t base = CageBaseGlobal::Get();
144 (base & kGigaCageMask) ==
145 (
reinterpret_cast<uintptr_t
>(ptr) & kGigaCageMask));
147 (
reinterpret_cast<uintptr_t
>(ptr) & kPointerCompressionShiftMask) == 0);
149 const auto uptr =
reinterpret_cast<uintptr_t
>(ptr);
151 auto compressed =
static_cast<IntegralType
>(
152 uptr >> api_constants::kPointerCompressionShift);
155 CPPGC_DCHECK((!compressed || compressed == kCompressedSentinel) ||
156 (compressed & (1 << 31)));
161 [[assume(((
static_cast<uint64_t
>(
static_cast<int32_t
>(compressed))
162 << api_constants::kPointerCompressionShift) &
168 static V8_INLINE void* Decompress(IntegralType ptr) {
170 const uintptr_t base = CageBaseGlobal::Get();
171 return Decompress(ptr, base);
174 static V8_INLINE void* Decompress(IntegralType ptr, uintptr_t base) {
181 const uint64_t mask =
static_cast<uint64_t
>(
static_cast<int32_t
>(ptr))
182 << api_constants::kPointerCompressionShift;
186 return reinterpret_cast<void*
>(mask & base);
192 template <
typename Callback>
193 static V8_INLINE void VisitPossiblePointers(
const void* address,
197 static constexpr IntegralType kCompressedSentinel =
198 SentinelPointer::kSentinelValue >>
199 api_constants::kPointerCompressionShift;
206template <
typename Callback>
208void CompressedPointer::VisitPossiblePointers(
const void* address,
210 const uintptr_t base = CageBaseGlobal::Get();
214 const uint32_t compressed_low =
215 static_cast<uint32_t
>(
reinterpret_cast<uintptr_t
>(address));
216 callback(CompressedPointer::Decompress(compressed_low, base));
217 const uint32_t compressed_high =
static_cast<uint32_t
>(
218 reinterpret_cast<uintptr_t
>(address) >> (
sizeof(uint32_t) * CHAR_BIT));
219 callback(CompressedPointer::Decompress(compressed_high, base));
225 static constexpr uintptr_t kBitForIntermediateValue =
226 (
sizeof(uint32_t) * CHAR_BIT) + api_constants::kPointerCompressionShift;
227 static constexpr uintptr_t kSignExtensionMask =
228 ~((uintptr_t{1} << kBitForIntermediateValue) - 1);
229 const uintptr_t intermediate_sign_extended =
230 reinterpret_cast<uintptr_t
>(address) | kSignExtensionMask;
231 callback(
reinterpret_cast<void*
>(intermediate_sign_extended & base));
241 static constexpr auto kWriteBarrierSlotType =
242 WriteBarrierSlotType::kUncompressed;
252 return reinterpret_cast<const std::atomic<const void*>&
>(ptr_).load(
253 std::memory_order_relaxed);
258 reinterpret_cast<std::atomic<const void*>&
>(ptr_).store(
259 ptr, std::memory_order_relaxed);
268 return reinterpret_cast<uintptr_t
>(ptr_);
272 return a.ptr_ == b.ptr_;
275 return a.ptr_ != b.ptr_;
278 return a.ptr_ < b.ptr_;
281 return a.ptr_ <= b.ptr_;
284 return a.ptr_ > b.ptr_;
287 return a.ptr_ >= b.ptr_;
290 template <
typename Callback>
294 return callback(
const_cast<void*
>(address));
304#if defined(CPPGC_POINTER_COMPRESSION)
Definition: member-storage.h:236
const void * Load() const
Definition: member-storage.h:250
const void * LoadAtomic() const
Definition: member-storage.h:251
RawPointer(const void *ptr)
Definition: member-storage.h:248
static void VisitPossiblePointers(const void *address, Callback callback)
Definition: member-storage.h:291
friend bool operator==(RawPointer a, RawPointer b)
Definition: member-storage.h:271
uintptr_t GetAsInteger() const
Definition: member-storage.h:267
friend bool operator>=(RawPointer a, RawPointer b)
Definition: member-storage.h:286
void Clear()
Definition: member-storage.h:262
friend bool operator<(RawPointer a, RawPointer b)
Definition: member-storage.h:277
RawPointer()
Definition: member-storage.h:244
RawPointer(const void *ptr, AtomicInitializerTag)
Definition: member-storage.h:245
friend bool operator!=(RawPointer a, RawPointer b)
Definition: member-storage.h:274
bool IsSentinel() const
Definition: member-storage.h:265
void Store(const void *ptr)
Definition: member-storage.h:256
uintptr_t IntegralType
Definition: member-storage.h:240
friend bool operator>(RawPointer a, RawPointer b)
Definition: member-storage.h:283
friend bool operator<=(RawPointer a, RawPointer b)
Definition: member-storage.h:280
bool IsCleared() const
Definition: member-storage.h:263
void StoreAtomic(const void *ptr)
Definition: member-storage.h:257
#define CPPGC_DCHECK(condition)
Definition: logging.h:36
bool operator<=(const BasicMember< T1, WeaknessTag1, WriteBarrierPolicy1, CheckingPolicy1, StorageType > &member1, const BasicMember< T2, WeaknessTag2, WriteBarrierPolicy2, CheckingPolicy2, StorageType > &member2)
Definition: member.h:516
bool operator!=(const BasicMember< T1, WeaknessTag1, WriteBarrierPolicy1, CheckingPolicy1, StorageType > &member1, const BasicMember< T2, WeaknessTag2, WriteBarrierPolicy2, CheckingPolicy2, StorageType > &member2)
Definition: member.h:369
bool operator>(const BasicMember< T1, WeaknessTag1, WriteBarrierPolicy1, CheckingPolicy1, StorageType > &member1, const BasicMember< T2, WeaknessTag2, WriteBarrierPolicy2, CheckingPolicy2, StorageType > &member2)
Definition: member.h:531
bool operator==(const BasicMember< T1, WeaknessTag1, WriteBarrierPolicy1, CheckingPolicy1, StorageType > &member1, const BasicMember< T2, WeaknessTag2, WriteBarrierPolicy2, CheckingPolicy2, StorageType > &member2)
Definition: member.h:350
WriteBarrierSlotType
Definition: member-storage.h:21
bool operator>=(const BasicMember< T1, WeaknessTag1, WriteBarrierPolicy1, CheckingPolicy1, StorageType > &member1, const BasicMember< T2, WeaknessTag2, WriteBarrierPolicy2, CheckingPolicy2, StorageType > &member2)
Definition: member.h:546
bool operator<(const BasicMember< T1, WeaknessTag1, WriteBarrierPolicy1, CheckingPolicy1, StorageType > &member1, const BasicMember< T2, WeaknessTag2, WriteBarrierPolicy2, CheckingPolicy2, StorageType > &member2)
Definition: member.h:501
RawPointer DefaultMemberStorage
Definition: member-storage.h:307
Definition: allocation.h:38
constexpr internal::SentinelPointer kSentinelPointer
Definition: sentinel-pointer.h:35
Definition: member-storage.h:238
#define V8_EXPORT
Definition: v8config.h:855
#define V8_INLINE
Definition: v8config.h:508
#define V8_TRIVIAL_ABI
Definition: v8config.h:808