xref: /aosp_15_r20/external/skia/src/base/SkTDArray.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2018 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTDArray.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkMalloc.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTFitsIn.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTo.h"
13*c8dee2aaSAndroid Build Coastguard Worker 
14*c8dee2aaSAndroid Build Coastguard Worker #include <climits>
15*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
16*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
17*c8dee2aaSAndroid Build Coastguard Worker #include <cstring>
18*c8dee2aaSAndroid Build Coastguard Worker #include <new>
19*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
20*c8dee2aaSAndroid Build Coastguard Worker 
SkTDStorage(int sizeOfT)21*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage::SkTDStorage(int sizeOfT) : fSizeOfT{sizeOfT} {}
22*c8dee2aaSAndroid Build Coastguard Worker 
SkTDStorage(const void * src,int size,int sizeOfT)23*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage::SkTDStorage(const void* src, int size, int sizeOfT)
24*c8dee2aaSAndroid Build Coastguard Worker         : fSizeOfT{sizeOfT}
25*c8dee2aaSAndroid Build Coastguard Worker         , fCapacity{size}
26*c8dee2aaSAndroid Build Coastguard Worker         , fSize{size} {
27*c8dee2aaSAndroid Build Coastguard Worker     if (size > 0) {
28*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(src != nullptr);
29*c8dee2aaSAndroid Build Coastguard Worker         size_t storageSize = this->bytes(size);
30*c8dee2aaSAndroid Build Coastguard Worker         fStorage = static_cast<std::byte*>(sk_malloc_throw(storageSize));
31*c8dee2aaSAndroid Build Coastguard Worker         memcpy(fStorage, src, storageSize);
32*c8dee2aaSAndroid Build Coastguard Worker     }
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker 
SkTDStorage(const SkTDStorage & that)35*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage::SkTDStorage(const SkTDStorage& that)
36*c8dee2aaSAndroid Build Coastguard Worker         : SkTDStorage{that.fStorage, that.fSize, that.fSizeOfT} {}
37*c8dee2aaSAndroid Build Coastguard Worker 
operator =(const SkTDStorage & that)38*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage& SkTDStorage::operator=(const SkTDStorage& that) {
39*c8dee2aaSAndroid Build Coastguard Worker     if (this != &that) {
40*c8dee2aaSAndroid Build Coastguard Worker         if (that.fSize <= fCapacity) {
41*c8dee2aaSAndroid Build Coastguard Worker             fSize = that.fSize;
42*c8dee2aaSAndroid Build Coastguard Worker             if (fSize > 0) {
43*c8dee2aaSAndroid Build Coastguard Worker                 memcpy(fStorage, that.data(), that.size_bytes());
44*c8dee2aaSAndroid Build Coastguard Worker             }
45*c8dee2aaSAndroid Build Coastguard Worker         } else {
46*c8dee2aaSAndroid Build Coastguard Worker             *this = SkTDStorage{that.data(), that.size(), that.fSizeOfT};
47*c8dee2aaSAndroid Build Coastguard Worker         }
48*c8dee2aaSAndroid Build Coastguard Worker     }
49*c8dee2aaSAndroid Build Coastguard Worker     return *this;
50*c8dee2aaSAndroid Build Coastguard Worker }
51*c8dee2aaSAndroid Build Coastguard Worker 
SkTDStorage(SkTDStorage && that)52*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage::SkTDStorage(SkTDStorage&& that)
53*c8dee2aaSAndroid Build Coastguard Worker         : fSizeOfT{that.fSizeOfT}
54*c8dee2aaSAndroid Build Coastguard Worker         , fStorage(std::exchange(that.fStorage, nullptr))
55*c8dee2aaSAndroid Build Coastguard Worker         , fCapacity{that.fCapacity}
56*c8dee2aaSAndroid Build Coastguard Worker         , fSize{that.fSize} {}
57*c8dee2aaSAndroid Build Coastguard Worker 
operator =(SkTDStorage && that)58*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage& SkTDStorage::operator=(SkTDStorage&& that) {
59*c8dee2aaSAndroid Build Coastguard Worker     if (this != &that) {
60*c8dee2aaSAndroid Build Coastguard Worker         this->~SkTDStorage();
61*c8dee2aaSAndroid Build Coastguard Worker         new (this) SkTDStorage{std::move(that)};
62*c8dee2aaSAndroid Build Coastguard Worker     }
63*c8dee2aaSAndroid Build Coastguard Worker     return *this;
64*c8dee2aaSAndroid Build Coastguard Worker }
65*c8dee2aaSAndroid Build Coastguard Worker 
~SkTDStorage()66*c8dee2aaSAndroid Build Coastguard Worker SkTDStorage::~SkTDStorage() {
67*c8dee2aaSAndroid Build Coastguard Worker     sk_free(fStorage);
68*c8dee2aaSAndroid Build Coastguard Worker }
69*c8dee2aaSAndroid Build Coastguard Worker 
reset()70*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::reset() {
71*c8dee2aaSAndroid Build Coastguard Worker     const int sizeOfT = fSizeOfT;
72*c8dee2aaSAndroid Build Coastguard Worker     this->~SkTDStorage();
73*c8dee2aaSAndroid Build Coastguard Worker     new (this) SkTDStorage{sizeOfT};
74*c8dee2aaSAndroid Build Coastguard Worker }
75*c8dee2aaSAndroid Build Coastguard Worker 
swap(SkTDStorage & that)76*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::swap(SkTDStorage& that) {
77*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fSizeOfT == that.fSizeOfT);
78*c8dee2aaSAndroid Build Coastguard Worker     using std::swap;
79*c8dee2aaSAndroid Build Coastguard Worker     swap(fStorage, that.fStorage);
80*c8dee2aaSAndroid Build Coastguard Worker     swap(fCapacity, that.fCapacity);
81*c8dee2aaSAndroid Build Coastguard Worker     swap(fSize, that.fSize);
82*c8dee2aaSAndroid Build Coastguard Worker }
83*c8dee2aaSAndroid Build Coastguard Worker 
resize(int newSize)84*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::resize(int newSize) {
85*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(newSize >= 0);
86*c8dee2aaSAndroid Build Coastguard Worker     if (newSize > fCapacity) {
87*c8dee2aaSAndroid Build Coastguard Worker         this->reserve(newSize);
88*c8dee2aaSAndroid Build Coastguard Worker     }
89*c8dee2aaSAndroid Build Coastguard Worker     fSize = newSize;
90*c8dee2aaSAndroid Build Coastguard Worker }
91*c8dee2aaSAndroid Build Coastguard Worker 
reserve(int newCapacity)92*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::reserve(int newCapacity) {
93*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(newCapacity >= 0);
94*c8dee2aaSAndroid Build Coastguard Worker     if (newCapacity > fCapacity) {
95*c8dee2aaSAndroid Build Coastguard Worker         // Establish the maximum number of elements that includes a valid count for end. In the
96*c8dee2aaSAndroid Build Coastguard Worker         // largest case end() = &fArray[INT_MAX] which is 1 after the last indexable element.
97*c8dee2aaSAndroid Build Coastguard Worker         static constexpr int kMaxCount = INT_MAX;
98*c8dee2aaSAndroid Build Coastguard Worker 
99*c8dee2aaSAndroid Build Coastguard Worker         // Assume that the array will max out.
100*c8dee2aaSAndroid Build Coastguard Worker         int expandedReserve = kMaxCount;
101*c8dee2aaSAndroid Build Coastguard Worker         if (kMaxCount - newCapacity > 4) {
102*c8dee2aaSAndroid Build Coastguard Worker             // Add 1/4 more than we need. Add 4 to ensure this grows by at least 1. Pin to
103*c8dee2aaSAndroid Build Coastguard Worker             // kMaxCount if no room for 1/4 growth.
104*c8dee2aaSAndroid Build Coastguard Worker             int growth = 4 + ((newCapacity + 4) >> 2);
105*c8dee2aaSAndroid Build Coastguard Worker             // Read this line as: if (count + growth < kMaxCount) { ... }
106*c8dee2aaSAndroid Build Coastguard Worker             // It's rewritten to avoid signed integer overflow.
107*c8dee2aaSAndroid Build Coastguard Worker             if (kMaxCount - newCapacity > growth) {
108*c8dee2aaSAndroid Build Coastguard Worker                 expandedReserve = newCapacity + growth;
109*c8dee2aaSAndroid Build Coastguard Worker             }
110*c8dee2aaSAndroid Build Coastguard Worker         }
111*c8dee2aaSAndroid Build Coastguard Worker 
112*c8dee2aaSAndroid Build Coastguard Worker 
113*c8dee2aaSAndroid Build Coastguard Worker         // With a T size of 1, the above allocator produces the progression of 7, 15, ... Since,
114*c8dee2aaSAndroid Build Coastguard Worker         // the sizeof max_align_t is often 16, there is no reason to allocate anything less than
115*c8dee2aaSAndroid Build Coastguard Worker         // 16 bytes. This eliminates a realloc when pushing back bytes to an SkTDArray.
116*c8dee2aaSAndroid Build Coastguard Worker         if (fSizeOfT == 1) {
117*c8dee2aaSAndroid Build Coastguard Worker             // Round up to the multiple of 16.
118*c8dee2aaSAndroid Build Coastguard Worker             expandedReserve = (expandedReserve + 15) & ~15;
119*c8dee2aaSAndroid Build Coastguard Worker         }
120*c8dee2aaSAndroid Build Coastguard Worker 
121*c8dee2aaSAndroid Build Coastguard Worker         fCapacity = expandedReserve;
122*c8dee2aaSAndroid Build Coastguard Worker         size_t newStorageSize = this->bytes(fCapacity);
123*c8dee2aaSAndroid Build Coastguard Worker         fStorage = static_cast<std::byte*>(sk_realloc_throw(fStorage, newStorageSize));
124*c8dee2aaSAndroid Build Coastguard Worker     }
125*c8dee2aaSAndroid Build Coastguard Worker }
126*c8dee2aaSAndroid Build Coastguard Worker 
shrink_to_fit()127*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::shrink_to_fit() {
128*c8dee2aaSAndroid Build Coastguard Worker     if (fCapacity != fSize) {
129*c8dee2aaSAndroid Build Coastguard Worker         fCapacity = fSize;
130*c8dee2aaSAndroid Build Coastguard Worker         // Because calling realloc with size of 0 is implementation defined, force to a good state
131*c8dee2aaSAndroid Build Coastguard Worker         // by freeing fStorage.
132*c8dee2aaSAndroid Build Coastguard Worker         if (fCapacity > 0) {
133*c8dee2aaSAndroid Build Coastguard Worker             fStorage = static_cast<std::byte*>(sk_realloc_throw(fStorage, this->bytes(fCapacity)));
134*c8dee2aaSAndroid Build Coastguard Worker         } else {
135*c8dee2aaSAndroid Build Coastguard Worker             sk_free(fStorage);
136*c8dee2aaSAndroid Build Coastguard Worker             fStorage = nullptr;
137*c8dee2aaSAndroid Build Coastguard Worker         }
138*c8dee2aaSAndroid Build Coastguard Worker     }
139*c8dee2aaSAndroid Build Coastguard Worker }
140*c8dee2aaSAndroid Build Coastguard Worker 
erase(int index,int count)141*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::erase(int index, int count) {
142*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(count >= 0);
143*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fSize >= count);
144*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(0 <= index && index <= fSize);
145*c8dee2aaSAndroid Build Coastguard Worker 
146*c8dee2aaSAndroid Build Coastguard Worker     if (count > 0) {
147*c8dee2aaSAndroid Build Coastguard Worker         // Check that the resulting size fits in an int. This will abort if not.
148*c8dee2aaSAndroid Build Coastguard Worker         const int newCount = this->calculateSizeOrDie(-count);
149*c8dee2aaSAndroid Build Coastguard Worker         this->moveTail(index, index + count, fSize);
150*c8dee2aaSAndroid Build Coastguard Worker         this->resize(newCount);
151*c8dee2aaSAndroid Build Coastguard Worker     }
152*c8dee2aaSAndroid Build Coastguard Worker }
153*c8dee2aaSAndroid Build Coastguard Worker 
removeShuffle(int index)154*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::removeShuffle(int index) {
155*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fSize > 0);
156*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(0 <= index && index < fSize);
157*c8dee2aaSAndroid Build Coastguard Worker     // Check that the new count is valid.
158*c8dee2aaSAndroid Build Coastguard Worker     const int newCount = this->calculateSizeOrDie(-1);
159*c8dee2aaSAndroid Build Coastguard Worker     this->moveTail(index, fSize - 1, fSize);
160*c8dee2aaSAndroid Build Coastguard Worker     this->resize(newCount);
161*c8dee2aaSAndroid Build Coastguard Worker }
162*c8dee2aaSAndroid Build Coastguard Worker 
prepend()163*c8dee2aaSAndroid Build Coastguard Worker void* SkTDStorage::prepend() {
164*c8dee2aaSAndroid Build Coastguard Worker     return this->insert(/*index=*/0);
165*c8dee2aaSAndroid Build Coastguard Worker }
166*c8dee2aaSAndroid Build Coastguard Worker 
append()167*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::append() {
168*c8dee2aaSAndroid Build Coastguard Worker     if (fSize < fCapacity) {
169*c8dee2aaSAndroid Build Coastguard Worker         fSize++;
170*c8dee2aaSAndroid Build Coastguard Worker     } else {
171*c8dee2aaSAndroid Build Coastguard Worker         this->insert(fSize);
172*c8dee2aaSAndroid Build Coastguard Worker     }
173*c8dee2aaSAndroid Build Coastguard Worker }
174*c8dee2aaSAndroid Build Coastguard Worker 
append(int count)175*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::append(int count) {
176*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(count >= 0);
177*c8dee2aaSAndroid Build Coastguard Worker     // Read as: if (fSize + count <= fCapacity) {...}. This is a UB safe way to avoid the add.
178*c8dee2aaSAndroid Build Coastguard Worker     if (fCapacity - fSize >= count) {
179*c8dee2aaSAndroid Build Coastguard Worker         fSize += count;
180*c8dee2aaSAndroid Build Coastguard Worker     } else {
181*c8dee2aaSAndroid Build Coastguard Worker         this->insert(fSize, count, nullptr);
182*c8dee2aaSAndroid Build Coastguard Worker     }
183*c8dee2aaSAndroid Build Coastguard Worker }
184*c8dee2aaSAndroid Build Coastguard Worker 
append(const void * src,int count)185*c8dee2aaSAndroid Build Coastguard Worker void* SkTDStorage::append(const void* src, int count) {
186*c8dee2aaSAndroid Build Coastguard Worker     return this->insert(fSize, count, src);
187*c8dee2aaSAndroid Build Coastguard Worker }
188*c8dee2aaSAndroid Build Coastguard Worker 
insert(int index)189*c8dee2aaSAndroid Build Coastguard Worker void* SkTDStorage::insert(int index) {
190*c8dee2aaSAndroid Build Coastguard Worker     return this->insert(index, /*count=*/1, nullptr);
191*c8dee2aaSAndroid Build Coastguard Worker }
192*c8dee2aaSAndroid Build Coastguard Worker 
insert(int index,int count,const void * src)193*c8dee2aaSAndroid Build Coastguard Worker void* SkTDStorage::insert(int index, int count, const void* src) {
194*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(0 <= index && index <= fSize);
195*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(count >= 0);
196*c8dee2aaSAndroid Build Coastguard Worker 
197*c8dee2aaSAndroid Build Coastguard Worker     if (count > 0) {
198*c8dee2aaSAndroid Build Coastguard Worker         const int oldCount = fSize;
199*c8dee2aaSAndroid Build Coastguard Worker         const int newCount = this->calculateSizeOrDie(count);
200*c8dee2aaSAndroid Build Coastguard Worker         this->resize(newCount);
201*c8dee2aaSAndroid Build Coastguard Worker         this->moveTail(index + count, index, oldCount);
202*c8dee2aaSAndroid Build Coastguard Worker 
203*c8dee2aaSAndroid Build Coastguard Worker         if (src != nullptr) {
204*c8dee2aaSAndroid Build Coastguard Worker             this->copySrc(index, src, count);
205*c8dee2aaSAndroid Build Coastguard Worker         }
206*c8dee2aaSAndroid Build Coastguard Worker     }
207*c8dee2aaSAndroid Build Coastguard Worker 
208*c8dee2aaSAndroid Build Coastguard Worker     return this->address(index);
209*c8dee2aaSAndroid Build Coastguard Worker }
210*c8dee2aaSAndroid Build Coastguard Worker 
operator ==(const SkTDStorage & a,const SkTDStorage & b)211*c8dee2aaSAndroid Build Coastguard Worker bool operator==(const SkTDStorage& a, const SkTDStorage& b) {
212*c8dee2aaSAndroid Build Coastguard Worker     return a.size() == b.size() && (a.empty() || !memcmp(a.data(), b.data(), a.bytes(a.size())));
213*c8dee2aaSAndroid Build Coastguard Worker }
214*c8dee2aaSAndroid Build Coastguard Worker 
calculateSizeOrDie(int delta)215*c8dee2aaSAndroid Build Coastguard Worker int SkTDStorage::calculateSizeOrDie(int delta) {
216*c8dee2aaSAndroid Build Coastguard Worker     // Check that count will not go negative.
217*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT_RELEASE(-fSize <= delta);
218*c8dee2aaSAndroid Build Coastguard Worker 
219*c8dee2aaSAndroid Build Coastguard Worker     // We take care to avoid overflow here.
220*c8dee2aaSAndroid Build Coastguard Worker     // Because count and delta are both signed 32-bit ints, the sum of count and delta is at
221*c8dee2aaSAndroid Build Coastguard Worker     // most 4294967294, which fits fine in uint32_t. Proof follows in assert.
222*c8dee2aaSAndroid Build Coastguard Worker     static_assert(UINT32_MAX >= (uint32_t)INT_MAX + (uint32_t)INT_MAX);
223*c8dee2aaSAndroid Build Coastguard Worker     uint32_t testCount = (uint32_t)fSize + (uint32_t)delta;
224*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT_RELEASE(SkTFitsIn<int>(testCount));
225*c8dee2aaSAndroid Build Coastguard Worker     return SkToInt(testCount);
226*c8dee2aaSAndroid Build Coastguard Worker }
227*c8dee2aaSAndroid Build Coastguard Worker 
moveTail(int to,int tailStart,int tailEnd)228*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::moveTail(int to, int tailStart, int tailEnd) {
229*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(0 <= to && to <= fSize);
230*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(0 <= tailStart && tailStart <= tailEnd && tailEnd <= fSize);
231*c8dee2aaSAndroid Build Coastguard Worker     if (to != tailStart && tailStart != tailEnd) {
232*c8dee2aaSAndroid Build Coastguard Worker         this->copySrc(to, this->address(tailStart), tailEnd - tailStart);
233*c8dee2aaSAndroid Build Coastguard Worker     }
234*c8dee2aaSAndroid Build Coastguard Worker }
235*c8dee2aaSAndroid Build Coastguard Worker 
copySrc(int dstIndex,const void * src,int count)236*c8dee2aaSAndroid Build Coastguard Worker void SkTDStorage::copySrc(int dstIndex, const void* src, int count) {
237*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(count > 0);
238*c8dee2aaSAndroid Build Coastguard Worker     memmove(this->address(dstIndex), src, this->bytes(count));
239*c8dee2aaSAndroid Build Coastguard Worker }
240