xref: /aosp_15_r20/external/scudo/standalone/tests/combined_test.cpp (revision 76559068c068bd27e82aff38fac3bfc865233bca)
1*76559068SAndroid Build Coastguard Worker //===-- combined_test.cpp ---------------------------------------*- C++ -*-===//
2*76559068SAndroid Build Coastguard Worker //
3*76559068SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*76559068SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information.
5*76559068SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*76559068SAndroid Build Coastguard Worker //
7*76559068SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
8*76559068SAndroid Build Coastguard Worker 
9*76559068SAndroid Build Coastguard Worker #include "memtag.h"
10*76559068SAndroid Build Coastguard Worker #include "stack_depot.h"
11*76559068SAndroid Build Coastguard Worker #include "tests/scudo_unit_test.h"
12*76559068SAndroid Build Coastguard Worker 
13*76559068SAndroid Build Coastguard Worker #include "allocator_config.h"
14*76559068SAndroid Build Coastguard Worker #include "chunk.h"
15*76559068SAndroid Build Coastguard Worker #include "combined.h"
16*76559068SAndroid Build Coastguard Worker #include "condition_variable.h"
17*76559068SAndroid Build Coastguard Worker #include "mem_map.h"
18*76559068SAndroid Build Coastguard Worker #include "size_class_map.h"
19*76559068SAndroid Build Coastguard Worker 
20*76559068SAndroid Build Coastguard Worker #include <algorithm>
21*76559068SAndroid Build Coastguard Worker #include <condition_variable>
22*76559068SAndroid Build Coastguard Worker #include <memory>
23*76559068SAndroid Build Coastguard Worker #include <mutex>
24*76559068SAndroid Build Coastguard Worker #include <set>
25*76559068SAndroid Build Coastguard Worker #include <stdlib.h>
26*76559068SAndroid Build Coastguard Worker #include <thread>
27*76559068SAndroid Build Coastguard Worker #include <vector>
28*76559068SAndroid Build Coastguard Worker 
29*76559068SAndroid Build Coastguard Worker static constexpr scudo::Chunk::Origin Origin = scudo::Chunk::Origin::Malloc;
30*76559068SAndroid Build Coastguard Worker static constexpr scudo::uptr MinAlignLog = FIRST_32_SECOND_64(3U, 4U);
31*76559068SAndroid Build Coastguard Worker 
32*76559068SAndroid Build Coastguard Worker // Fuchsia complains that the function is not used.
disableDebuggerdMaybe()33*76559068SAndroid Build Coastguard Worker UNUSED static void disableDebuggerdMaybe() {
34*76559068SAndroid Build Coastguard Worker #if SCUDO_ANDROID
35*76559068SAndroid Build Coastguard Worker   // Disable the debuggerd signal handler on Android, without this we can end
36*76559068SAndroid Build Coastguard Worker   // up spending a significant amount of time creating tombstones.
37*76559068SAndroid Build Coastguard Worker   signal(SIGSEGV, SIG_DFL);
38*76559068SAndroid Build Coastguard Worker #endif
39*76559068SAndroid Build Coastguard Worker }
40*76559068SAndroid Build Coastguard Worker 
41*76559068SAndroid Build Coastguard Worker template <class AllocatorT>
isPrimaryAllocation(scudo::uptr Size,scudo::uptr Alignment)42*76559068SAndroid Build Coastguard Worker bool isPrimaryAllocation(scudo::uptr Size, scudo::uptr Alignment) {
43*76559068SAndroid Build Coastguard Worker   const scudo::uptr MinAlignment = 1UL << SCUDO_MIN_ALIGNMENT_LOG;
44*76559068SAndroid Build Coastguard Worker   if (Alignment < MinAlignment)
45*76559068SAndroid Build Coastguard Worker     Alignment = MinAlignment;
46*76559068SAndroid Build Coastguard Worker   const scudo::uptr NeededSize =
47*76559068SAndroid Build Coastguard Worker       scudo::roundUp(Size, MinAlignment) +
48*76559068SAndroid Build Coastguard Worker       ((Alignment > MinAlignment) ? Alignment : scudo::Chunk::getHeaderSize());
49*76559068SAndroid Build Coastguard Worker   return AllocatorT::PrimaryT::canAllocate(NeededSize);
50*76559068SAndroid Build Coastguard Worker }
51*76559068SAndroid Build Coastguard Worker 
52*76559068SAndroid Build Coastguard Worker template <class AllocatorT>
checkMemoryTaggingMaybe(AllocatorT * Allocator,void * P,scudo::uptr Size,scudo::uptr Alignment)53*76559068SAndroid Build Coastguard Worker void checkMemoryTaggingMaybe(AllocatorT *Allocator, void *P, scudo::uptr Size,
54*76559068SAndroid Build Coastguard Worker                              scudo::uptr Alignment) {
55*76559068SAndroid Build Coastguard Worker   const scudo::uptr MinAlignment = 1UL << SCUDO_MIN_ALIGNMENT_LOG;
56*76559068SAndroid Build Coastguard Worker   Size = scudo::roundUp(Size, MinAlignment);
57*76559068SAndroid Build Coastguard Worker   if (Allocator->useMemoryTaggingTestOnly())
58*76559068SAndroid Build Coastguard Worker     EXPECT_DEATH(
59*76559068SAndroid Build Coastguard Worker         {
60*76559068SAndroid Build Coastguard Worker           disableDebuggerdMaybe();
61*76559068SAndroid Build Coastguard Worker           reinterpret_cast<char *>(P)[-1] = 'A';
62*76559068SAndroid Build Coastguard Worker         },
63*76559068SAndroid Build Coastguard Worker         "");
64*76559068SAndroid Build Coastguard Worker   if (isPrimaryAllocation<AllocatorT>(Size, Alignment)
65*76559068SAndroid Build Coastguard Worker           ? Allocator->useMemoryTaggingTestOnly()
66*76559068SAndroid Build Coastguard Worker           : Alignment == MinAlignment) {
67*76559068SAndroid Build Coastguard Worker     EXPECT_DEATH(
68*76559068SAndroid Build Coastguard Worker         {
69*76559068SAndroid Build Coastguard Worker           disableDebuggerdMaybe();
70*76559068SAndroid Build Coastguard Worker           reinterpret_cast<char *>(P)[Size] = 'A';
71*76559068SAndroid Build Coastguard Worker         },
72*76559068SAndroid Build Coastguard Worker         "");
73*76559068SAndroid Build Coastguard Worker   }
74*76559068SAndroid Build Coastguard Worker }
75*76559068SAndroid Build Coastguard Worker 
76*76559068SAndroid Build Coastguard Worker template <typename Config> struct TestAllocator : scudo::Allocator<Config> {
TestAllocatorTestAllocator77*76559068SAndroid Build Coastguard Worker   TestAllocator() {
78*76559068SAndroid Build Coastguard Worker     this->initThreadMaybe();
79*76559068SAndroid Build Coastguard Worker     if (scudo::archSupportsMemoryTagging() &&
80*76559068SAndroid Build Coastguard Worker         !scudo::systemDetectsMemoryTagFaultsTestOnly())
81*76559068SAndroid Build Coastguard Worker       this->disableMemoryTagging();
82*76559068SAndroid Build Coastguard Worker   }
~TestAllocatorTestAllocator83*76559068SAndroid Build Coastguard Worker   ~TestAllocator() { this->unmapTestOnly(); }
84*76559068SAndroid Build Coastguard Worker 
85*76559068SAndroid Build Coastguard Worker   void *operator new(size_t size);
86*76559068SAndroid Build Coastguard Worker   void operator delete(void *ptr);
87*76559068SAndroid Build Coastguard Worker };
88*76559068SAndroid Build Coastguard Worker 
89*76559068SAndroid Build Coastguard Worker constexpr size_t kMaxAlign = std::max({
90*76559068SAndroid Build Coastguard Worker   alignof(scudo::Allocator<scudo::DefaultConfig>),
91*76559068SAndroid Build Coastguard Worker #if SCUDO_CAN_USE_PRIMARY64
92*76559068SAndroid Build Coastguard Worker       alignof(scudo::Allocator<scudo::FuchsiaConfig>),
93*76559068SAndroid Build Coastguard Worker #endif
94*76559068SAndroid Build Coastguard Worker       alignof(scudo::Allocator<scudo::AndroidConfig>)
95*76559068SAndroid Build Coastguard Worker });
96*76559068SAndroid Build Coastguard Worker 
97*76559068SAndroid Build Coastguard Worker #if SCUDO_RISCV64
98*76559068SAndroid Build Coastguard Worker // The allocator is over 4MB large. Rather than creating an instance of this on
99*76559068SAndroid Build Coastguard Worker // the heap, keep it in a global storage to reduce fragmentation from having to
100*76559068SAndroid Build Coastguard Worker // mmap this at the start of every test.
101*76559068SAndroid Build Coastguard Worker struct TestAllocatorStorage {
102*76559068SAndroid Build Coastguard Worker   static constexpr size_t kMaxSize = std::max({
103*76559068SAndroid Build Coastguard Worker     sizeof(scudo::Allocator<scudo::DefaultConfig>),
104*76559068SAndroid Build Coastguard Worker #if SCUDO_CAN_USE_PRIMARY64
105*76559068SAndroid Build Coastguard Worker         sizeof(scudo::Allocator<scudo::FuchsiaConfig>),
106*76559068SAndroid Build Coastguard Worker #endif
107*76559068SAndroid Build Coastguard Worker         sizeof(scudo::Allocator<scudo::AndroidConfig>)
108*76559068SAndroid Build Coastguard Worker   });
109*76559068SAndroid Build Coastguard Worker 
110*76559068SAndroid Build Coastguard Worker   // To alleviate some problem, let's skip the thread safety analysis here.
getTestAllocatorStorage111*76559068SAndroid Build Coastguard Worker   static void *get(size_t size) NO_THREAD_SAFETY_ANALYSIS {
112*76559068SAndroid Build Coastguard Worker     CHECK(size <= kMaxSize &&
113*76559068SAndroid Build Coastguard Worker           "Allocation size doesn't fit in the allocator storage");
114*76559068SAndroid Build Coastguard Worker     M.lock();
115*76559068SAndroid Build Coastguard Worker     return AllocatorStorage;
116*76559068SAndroid Build Coastguard Worker   }
117*76559068SAndroid Build Coastguard Worker 
releaseTestAllocatorStorage118*76559068SAndroid Build Coastguard Worker   static void release(void *ptr) NO_THREAD_SAFETY_ANALYSIS {
119*76559068SAndroid Build Coastguard Worker     M.assertHeld();
120*76559068SAndroid Build Coastguard Worker     M.unlock();
121*76559068SAndroid Build Coastguard Worker     ASSERT_EQ(ptr, AllocatorStorage);
122*76559068SAndroid Build Coastguard Worker   }
123*76559068SAndroid Build Coastguard Worker 
124*76559068SAndroid Build Coastguard Worker   static scudo::HybridMutex M;
125*76559068SAndroid Build Coastguard Worker   static uint8_t AllocatorStorage[kMaxSize];
126*76559068SAndroid Build Coastguard Worker };
127*76559068SAndroid Build Coastguard Worker scudo::HybridMutex TestAllocatorStorage::M;
128*76559068SAndroid Build Coastguard Worker alignas(kMaxAlign) uint8_t TestAllocatorStorage::AllocatorStorage[kMaxSize];
129*76559068SAndroid Build Coastguard Worker #else
130*76559068SAndroid Build Coastguard Worker struct TestAllocatorStorage {
getTestAllocatorStorage131*76559068SAndroid Build Coastguard Worker   static void *get(size_t size) NO_THREAD_SAFETY_ANALYSIS {
132*76559068SAndroid Build Coastguard Worker     void *p = nullptr;
133*76559068SAndroid Build Coastguard Worker     EXPECT_EQ(0, posix_memalign(&p, kMaxAlign, size));
134*76559068SAndroid Build Coastguard Worker     return p;
135*76559068SAndroid Build Coastguard Worker   }
releaseTestAllocatorStorage136*76559068SAndroid Build Coastguard Worker   static void release(void *ptr) NO_THREAD_SAFETY_ANALYSIS { free(ptr); }
137*76559068SAndroid Build Coastguard Worker };
138*76559068SAndroid Build Coastguard Worker #endif
139*76559068SAndroid Build Coastguard Worker 
140*76559068SAndroid Build Coastguard Worker template <typename Config>
operator new(size_t size)141*76559068SAndroid Build Coastguard Worker void *TestAllocator<Config>::operator new(size_t size) {
142*76559068SAndroid Build Coastguard Worker   return TestAllocatorStorage::get(size);
143*76559068SAndroid Build Coastguard Worker }
144*76559068SAndroid Build Coastguard Worker 
145*76559068SAndroid Build Coastguard Worker template <typename Config>
operator delete(void * ptr)146*76559068SAndroid Build Coastguard Worker void TestAllocator<Config>::operator delete(void *ptr) {
147*76559068SAndroid Build Coastguard Worker   TestAllocatorStorage::release(ptr);
148*76559068SAndroid Build Coastguard Worker }
149*76559068SAndroid Build Coastguard Worker 
150*76559068SAndroid Build Coastguard Worker template <class TypeParam> struct ScudoCombinedTest : public Test {
ScudoCombinedTestScudoCombinedTest151*76559068SAndroid Build Coastguard Worker   ScudoCombinedTest() {
152*76559068SAndroid Build Coastguard Worker     UseQuarantine = std::is_same<TypeParam, scudo::AndroidConfig>::value;
153*76559068SAndroid Build Coastguard Worker     Allocator = std::make_unique<AllocatorT>();
154*76559068SAndroid Build Coastguard Worker   }
~ScudoCombinedTestScudoCombinedTest155*76559068SAndroid Build Coastguard Worker   ~ScudoCombinedTest() {
156*76559068SAndroid Build Coastguard Worker     Allocator->releaseToOS(scudo::ReleaseToOS::Force);
157*76559068SAndroid Build Coastguard Worker     UseQuarantine = true;
158*76559068SAndroid Build Coastguard Worker   }
159*76559068SAndroid Build Coastguard Worker 
160*76559068SAndroid Build Coastguard Worker   void RunTest();
161*76559068SAndroid Build Coastguard Worker 
162*76559068SAndroid Build Coastguard Worker   void BasicTest(scudo::uptr SizeLog);
163*76559068SAndroid Build Coastguard Worker 
164*76559068SAndroid Build Coastguard Worker   using AllocatorT = TestAllocator<TypeParam>;
165*76559068SAndroid Build Coastguard Worker   std::unique_ptr<AllocatorT> Allocator;
166*76559068SAndroid Build Coastguard Worker };
167*76559068SAndroid Build Coastguard Worker 
168*76559068SAndroid Build Coastguard Worker template <typename T> using ScudoCombinedDeathTest = ScudoCombinedTest<T>;
169*76559068SAndroid Build Coastguard Worker 
170*76559068SAndroid Build Coastguard Worker namespace scudo {
171*76559068SAndroid Build Coastguard Worker struct TestConditionVariableConfig {
172*76559068SAndroid Build Coastguard Worker   static const bool MaySupportMemoryTagging = true;
173*76559068SAndroid Build Coastguard Worker   template <class A>
174*76559068SAndroid Build Coastguard Worker   using TSDRegistryT =
175*76559068SAndroid Build Coastguard Worker       scudo::TSDRegistrySharedT<A, 8U, 4U>; // Shared, max 8 TSDs.
176*76559068SAndroid Build Coastguard Worker 
177*76559068SAndroid Build Coastguard Worker   struct Primary {
178*76559068SAndroid Build Coastguard Worker     using SizeClassMap = scudo::AndroidSizeClassMap;
179*76559068SAndroid Build Coastguard Worker #if SCUDO_CAN_USE_PRIMARY64
180*76559068SAndroid Build Coastguard Worker     static const scudo::uptr RegionSizeLog = 28U;
181*76559068SAndroid Build Coastguard Worker     typedef scudo::u32 CompactPtrT;
182*76559068SAndroid Build Coastguard Worker     static const scudo::uptr CompactPtrScale = SCUDO_MIN_ALIGNMENT_LOG;
183*76559068SAndroid Build Coastguard Worker     static const scudo::uptr GroupSizeLog = 20U;
184*76559068SAndroid Build Coastguard Worker     static const bool EnableRandomOffset = true;
185*76559068SAndroid Build Coastguard Worker     static const scudo::uptr MapSizeIncrement = 1UL << 18;
186*76559068SAndroid Build Coastguard Worker #else
187*76559068SAndroid Build Coastguard Worker     static const scudo::uptr RegionSizeLog = 18U;
188*76559068SAndroid Build Coastguard Worker     static const scudo::uptr GroupSizeLog = 18U;
189*76559068SAndroid Build Coastguard Worker     typedef scudo::uptr CompactPtrT;
190*76559068SAndroid Build Coastguard Worker #endif
191*76559068SAndroid Build Coastguard Worker     static const scudo::s32 MinReleaseToOsIntervalMs = 1000;
192*76559068SAndroid Build Coastguard Worker     static const scudo::s32 MaxReleaseToOsIntervalMs = 1000;
193*76559068SAndroid Build Coastguard Worker #if SCUDO_LINUX
194*76559068SAndroid Build Coastguard Worker     using ConditionVariableT = scudo::ConditionVariableLinux;
195*76559068SAndroid Build Coastguard Worker #else
196*76559068SAndroid Build Coastguard Worker     using ConditionVariableT = scudo::ConditionVariableDummy;
197*76559068SAndroid Build Coastguard Worker #endif
198*76559068SAndroid Build Coastguard Worker   };
199*76559068SAndroid Build Coastguard Worker #if SCUDO_CAN_USE_PRIMARY64
200*76559068SAndroid Build Coastguard Worker   template <typename Config>
201*76559068SAndroid Build Coastguard Worker   using PrimaryT = scudo::SizeClassAllocator64<Config>;
202*76559068SAndroid Build Coastguard Worker #else
203*76559068SAndroid Build Coastguard Worker   template <typename Config>
204*76559068SAndroid Build Coastguard Worker   using PrimaryT = scudo::SizeClassAllocator32<Config>;
205*76559068SAndroid Build Coastguard Worker #endif
206*76559068SAndroid Build Coastguard Worker 
207*76559068SAndroid Build Coastguard Worker   struct Secondary {
208*76559068SAndroid Build Coastguard Worker     template <typename Config>
209*76559068SAndroid Build Coastguard Worker     using CacheT = scudo::MapAllocatorNoCache<Config>;
210*76559068SAndroid Build Coastguard Worker   };
211*76559068SAndroid Build Coastguard Worker   template <typename Config> using SecondaryT = scudo::MapAllocator<Config>;
212*76559068SAndroid Build Coastguard Worker };
213*76559068SAndroid Build Coastguard Worker } // namespace scudo
214*76559068SAndroid Build Coastguard Worker 
215*76559068SAndroid Build Coastguard Worker #if SCUDO_FUCHSIA
216*76559068SAndroid Build Coastguard Worker #define SCUDO_TYPED_TEST_ALL_TYPES(FIXTURE, NAME)                              \
217*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST_TYPE(FIXTURE, NAME, FuchsiaConfig)
218*76559068SAndroid Build Coastguard Worker #else
219*76559068SAndroid Build Coastguard Worker #define SCUDO_TYPED_TEST_ALL_TYPES(FIXTURE, NAME)                              \
220*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST_TYPE(FIXTURE, NAME, DefaultConfig)                          \
221*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST_TYPE(FIXTURE, NAME, AndroidConfig)                          \
222*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST_TYPE(FIXTURE, NAME, TestConditionVariableConfig)
223*76559068SAndroid Build Coastguard Worker #endif
224*76559068SAndroid Build Coastguard Worker 
225*76559068SAndroid Build Coastguard Worker #define SCUDO_TYPED_TEST_TYPE(FIXTURE, NAME, TYPE)                             \
226*76559068SAndroid Build Coastguard Worker   using FIXTURE##NAME##_##TYPE = FIXTURE##NAME<scudo::TYPE>;                   \
227*76559068SAndroid Build Coastguard Worker   TEST_F(FIXTURE##NAME##_##TYPE, NAME) { FIXTURE##NAME<scudo::TYPE>::Run(); }
228*76559068SAndroid Build Coastguard Worker 
229*76559068SAndroid Build Coastguard Worker #define SCUDO_TYPED_TEST(FIXTURE, NAME)                                        \
230*76559068SAndroid Build Coastguard Worker   template <class TypeParam>                                                   \
231*76559068SAndroid Build Coastguard Worker   struct FIXTURE##NAME : public FIXTURE<TypeParam> {                           \
232*76559068SAndroid Build Coastguard Worker     using BaseT = FIXTURE<TypeParam>;                                          \
233*76559068SAndroid Build Coastguard Worker     void Run();                                                                \
234*76559068SAndroid Build Coastguard Worker   };                                                                           \
235*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST_ALL_TYPES(FIXTURE, NAME)                                    \
236*76559068SAndroid Build Coastguard Worker   template <class TypeParam> void FIXTURE##NAME<TypeParam>::Run()
237*76559068SAndroid Build Coastguard Worker 
238*76559068SAndroid Build Coastguard Worker // Accessing `TSD->getCache()` requires `TSD::Mutex` which isn't easy to test
239*76559068SAndroid Build Coastguard Worker // using thread-safety analysis. Alternatively, we verify the thread safety
240*76559068SAndroid Build Coastguard Worker // through a runtime check in ScopedTSD and mark the test body with
241*76559068SAndroid Build Coastguard Worker // NO_THREAD_SAFETY_ANALYSIS.
242*76559068SAndroid Build Coastguard Worker #define SCUDO_TYPED_TEST_SKIP_THREAD_SAFETY(FIXTURE, NAME)                     \
243*76559068SAndroid Build Coastguard Worker   template <class TypeParam>                                                   \
244*76559068SAndroid Build Coastguard Worker   struct FIXTURE##NAME : public FIXTURE<TypeParam> {                           \
245*76559068SAndroid Build Coastguard Worker     using BaseT = FIXTURE<TypeParam>;                                          \
246*76559068SAndroid Build Coastguard Worker     void Run() NO_THREAD_SAFETY_ANALYSIS;                                      \
247*76559068SAndroid Build Coastguard Worker   };                                                                           \
248*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST_ALL_TYPES(FIXTURE, NAME)                                    \
249*76559068SAndroid Build Coastguard Worker   template <class TypeParam> void FIXTURE##NAME<TypeParam>::Run()
250*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,IsOwned)251*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, IsOwned) {
252*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
253*76559068SAndroid Build Coastguard Worker   static scudo::u8 StaticBuffer[scudo::Chunk::getHeaderSize() + 1];
254*76559068SAndroid Build Coastguard Worker   EXPECT_FALSE(
255*76559068SAndroid Build Coastguard Worker       Allocator->isOwned(&StaticBuffer[scudo::Chunk::getHeaderSize()]));
256*76559068SAndroid Build Coastguard Worker 
257*76559068SAndroid Build Coastguard Worker   scudo::u8 StackBuffer[scudo::Chunk::getHeaderSize() + 1];
258*76559068SAndroid Build Coastguard Worker   for (scudo::uptr I = 0; I < sizeof(StackBuffer); I++)
259*76559068SAndroid Build Coastguard Worker     StackBuffer[I] = 0x42U;
260*76559068SAndroid Build Coastguard Worker   EXPECT_FALSE(Allocator->isOwned(&StackBuffer[scudo::Chunk::getHeaderSize()]));
261*76559068SAndroid Build Coastguard Worker   for (scudo::uptr I = 0; I < sizeof(StackBuffer); I++)
262*76559068SAndroid Build Coastguard Worker     EXPECT_EQ(StackBuffer[I], 0x42U);
263*76559068SAndroid Build Coastguard Worker }
264*76559068SAndroid Build Coastguard Worker 
265*76559068SAndroid Build Coastguard Worker template <class Config>
BasicTest(scudo::uptr SizeLog)266*76559068SAndroid Build Coastguard Worker void ScudoCombinedTest<Config>::BasicTest(scudo::uptr SizeLog) {
267*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
268*76559068SAndroid Build Coastguard Worker 
269*76559068SAndroid Build Coastguard Worker   // This allocates and deallocates a bunch of chunks, with a wide range of
270*76559068SAndroid Build Coastguard Worker   // sizes and alignments, with a focus on sizes that could trigger weird
271*76559068SAndroid Build Coastguard Worker   // behaviors (plus or minus a small delta of a power of two for example).
272*76559068SAndroid Build Coastguard Worker   for (scudo::uptr AlignLog = MinAlignLog; AlignLog <= 16U; AlignLog++) {
273*76559068SAndroid Build Coastguard Worker     const scudo::uptr Align = 1U << AlignLog;
274*76559068SAndroid Build Coastguard Worker     for (scudo::sptr Delta = -32; Delta <= 32; Delta++) {
275*76559068SAndroid Build Coastguard Worker       if ((1LL << SizeLog) + Delta < 0)
276*76559068SAndroid Build Coastguard Worker         continue;
277*76559068SAndroid Build Coastguard Worker       const scudo::uptr Size =
278*76559068SAndroid Build Coastguard Worker           static_cast<scudo::uptr>((1LL << SizeLog) + Delta);
279*76559068SAndroid Build Coastguard Worker       void *P = Allocator->allocate(Size, Origin, Align);
280*76559068SAndroid Build Coastguard Worker       EXPECT_NE(P, nullptr);
281*76559068SAndroid Build Coastguard Worker       EXPECT_TRUE(Allocator->isOwned(P));
282*76559068SAndroid Build Coastguard Worker       EXPECT_TRUE(scudo::isAligned(reinterpret_cast<scudo::uptr>(P), Align));
283*76559068SAndroid Build Coastguard Worker       EXPECT_LE(Size, Allocator->getUsableSize(P));
284*76559068SAndroid Build Coastguard Worker       memset(P, 0xaa, Size);
285*76559068SAndroid Build Coastguard Worker       checkMemoryTaggingMaybe(Allocator, P, Size, Align);
286*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(P, Origin, Size);
287*76559068SAndroid Build Coastguard Worker     }
288*76559068SAndroid Build Coastguard Worker   }
289*76559068SAndroid Build Coastguard Worker 
290*76559068SAndroid Build Coastguard Worker   Allocator->printStats();
291*76559068SAndroid Build Coastguard Worker   Allocator->printFragmentationInfo();
292*76559068SAndroid Build Coastguard Worker }
293*76559068SAndroid Build Coastguard Worker 
294*76559068SAndroid Build Coastguard Worker #define SCUDO_MAKE_BASIC_TEST(SizeLog)                                         \
295*76559068SAndroid Build Coastguard Worker   SCUDO_TYPED_TEST(ScudoCombinedDeathTest, BasicCombined##SizeLog) {           \
296*76559068SAndroid Build Coastguard Worker     this->BasicTest(SizeLog);                                                  \
297*76559068SAndroid Build Coastguard Worker   }
298*76559068SAndroid Build Coastguard Worker 
299*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(0)
300*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(1)
301*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(2)
302*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(3)
303*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(4)
304*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(5)
305*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(6)
306*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(7)
307*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(8)
308*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(9)
309*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(10)
310*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(11)
311*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(12)
312*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(13)
313*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(14)
314*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(15)
315*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(16)
316*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(17)
317*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(18)
318*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(19)
319*76559068SAndroid Build Coastguard Worker SCUDO_MAKE_BASIC_TEST(20)
320*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,ZeroContents)321*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ZeroContents) {
322*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
323*76559068SAndroid Build Coastguard Worker 
324*76559068SAndroid Build Coastguard Worker   // Ensure that specifying ZeroContents returns a zero'd out block.
325*76559068SAndroid Build Coastguard Worker   for (scudo::uptr SizeLog = 0U; SizeLog <= 20U; SizeLog++) {
326*76559068SAndroid Build Coastguard Worker     for (scudo::uptr Delta = 0U; Delta <= 4U; Delta++) {
327*76559068SAndroid Build Coastguard Worker       const scudo::uptr Size = (1U << SizeLog) + Delta * 128U;
328*76559068SAndroid Build Coastguard Worker       void *P = Allocator->allocate(Size, Origin, 1U << MinAlignLog, true);
329*76559068SAndroid Build Coastguard Worker       EXPECT_NE(P, nullptr);
330*76559068SAndroid Build Coastguard Worker       for (scudo::uptr I = 0; I < Size; I++)
331*76559068SAndroid Build Coastguard Worker         ASSERT_EQ((reinterpret_cast<char *>(P))[I], '\0');
332*76559068SAndroid Build Coastguard Worker       memset(P, 0xaa, Size);
333*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(P, Origin, Size);
334*76559068SAndroid Build Coastguard Worker     }
335*76559068SAndroid Build Coastguard Worker   }
336*76559068SAndroid Build Coastguard Worker }
337*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,ZeroFill)338*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ZeroFill) {
339*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
340*76559068SAndroid Build Coastguard Worker 
341*76559068SAndroid Build Coastguard Worker   // Ensure that specifying ZeroFill returns a zero'd out block.
342*76559068SAndroid Build Coastguard Worker   Allocator->setFillContents(scudo::ZeroFill);
343*76559068SAndroid Build Coastguard Worker   for (scudo::uptr SizeLog = 0U; SizeLog <= 20U; SizeLog++) {
344*76559068SAndroid Build Coastguard Worker     for (scudo::uptr Delta = 0U; Delta <= 4U; Delta++) {
345*76559068SAndroid Build Coastguard Worker       const scudo::uptr Size = (1U << SizeLog) + Delta * 128U;
346*76559068SAndroid Build Coastguard Worker       void *P = Allocator->allocate(Size, Origin, 1U << MinAlignLog, false);
347*76559068SAndroid Build Coastguard Worker       EXPECT_NE(P, nullptr);
348*76559068SAndroid Build Coastguard Worker       for (scudo::uptr I = 0; I < Size; I++)
349*76559068SAndroid Build Coastguard Worker         ASSERT_EQ((reinterpret_cast<char *>(P))[I], '\0');
350*76559068SAndroid Build Coastguard Worker       memset(P, 0xaa, Size);
351*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(P, Origin, Size);
352*76559068SAndroid Build Coastguard Worker     }
353*76559068SAndroid Build Coastguard Worker   }
354*76559068SAndroid Build Coastguard Worker }
355*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,PatternOrZeroFill)356*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, PatternOrZeroFill) {
357*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
358*76559068SAndroid Build Coastguard Worker 
359*76559068SAndroid Build Coastguard Worker   // Ensure that specifying PatternOrZeroFill returns a pattern or zero filled
360*76559068SAndroid Build Coastguard Worker   // block. The primary allocator only produces pattern filled blocks if MTE
361*76559068SAndroid Build Coastguard Worker   // is disabled, so we only require pattern filled blocks in that case.
362*76559068SAndroid Build Coastguard Worker   Allocator->setFillContents(scudo::PatternOrZeroFill);
363*76559068SAndroid Build Coastguard Worker   for (scudo::uptr SizeLog = 0U; SizeLog <= 20U; SizeLog++) {
364*76559068SAndroid Build Coastguard Worker     for (scudo::uptr Delta = 0U; Delta <= 4U; Delta++) {
365*76559068SAndroid Build Coastguard Worker       const scudo::uptr Size = (1U << SizeLog) + Delta * 128U;
366*76559068SAndroid Build Coastguard Worker       void *P = Allocator->allocate(Size, Origin, 1U << MinAlignLog, false);
367*76559068SAndroid Build Coastguard Worker       EXPECT_NE(P, nullptr);
368*76559068SAndroid Build Coastguard Worker       for (scudo::uptr I = 0; I < Size; I++) {
369*76559068SAndroid Build Coastguard Worker         unsigned char V = (reinterpret_cast<unsigned char *>(P))[I];
370*76559068SAndroid Build Coastguard Worker         if (isPrimaryAllocation<TestAllocator<TypeParam>>(Size,
371*76559068SAndroid Build Coastguard Worker                                                           1U << MinAlignLog) &&
372*76559068SAndroid Build Coastguard Worker             !Allocator->useMemoryTaggingTestOnly())
373*76559068SAndroid Build Coastguard Worker           ASSERT_EQ(V, scudo::PatternFillByte);
374*76559068SAndroid Build Coastguard Worker         else
375*76559068SAndroid Build Coastguard Worker           ASSERT_TRUE(V == scudo::PatternFillByte || V == 0);
376*76559068SAndroid Build Coastguard Worker       }
377*76559068SAndroid Build Coastguard Worker       memset(P, 0xaa, Size);
378*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(P, Origin, Size);
379*76559068SAndroid Build Coastguard Worker     }
380*76559068SAndroid Build Coastguard Worker   }
381*76559068SAndroid Build Coastguard Worker }
382*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,BlockReuse)383*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, BlockReuse) {
384*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
385*76559068SAndroid Build Coastguard Worker 
386*76559068SAndroid Build Coastguard Worker   // Verify that a chunk will end up being reused, at some point.
387*76559068SAndroid Build Coastguard Worker   const scudo::uptr NeedleSize = 1024U;
388*76559068SAndroid Build Coastguard Worker   void *NeedleP = Allocator->allocate(NeedleSize, Origin);
389*76559068SAndroid Build Coastguard Worker   Allocator->deallocate(NeedleP, Origin);
390*76559068SAndroid Build Coastguard Worker   bool Found = false;
391*76559068SAndroid Build Coastguard Worker   for (scudo::uptr I = 0; I < 1024U && !Found; I++) {
392*76559068SAndroid Build Coastguard Worker     void *P = Allocator->allocate(NeedleSize, Origin);
393*76559068SAndroid Build Coastguard Worker     if (Allocator->getHeaderTaggedPointer(P) ==
394*76559068SAndroid Build Coastguard Worker         Allocator->getHeaderTaggedPointer(NeedleP))
395*76559068SAndroid Build Coastguard Worker       Found = true;
396*76559068SAndroid Build Coastguard Worker     Allocator->deallocate(P, Origin);
397*76559068SAndroid Build Coastguard Worker   }
398*76559068SAndroid Build Coastguard Worker   EXPECT_TRUE(Found);
399*76559068SAndroid Build Coastguard Worker }
400*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,ReallocateLargeIncreasing)401*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ReallocateLargeIncreasing) {
402*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
403*76559068SAndroid Build Coastguard Worker 
404*76559068SAndroid Build Coastguard Worker   // Reallocate a chunk all the way up to a secondary allocation, verifying that
405*76559068SAndroid Build Coastguard Worker   // we preserve the data in the process.
406*76559068SAndroid Build Coastguard Worker   scudo::uptr Size = 16;
407*76559068SAndroid Build Coastguard Worker   void *P = Allocator->allocate(Size, Origin);
408*76559068SAndroid Build Coastguard Worker   const char Marker = 'A';
409*76559068SAndroid Build Coastguard Worker   memset(P, Marker, Size);
410*76559068SAndroid Build Coastguard Worker   while (Size < TypeParam::Primary::SizeClassMap::MaxSize * 4) {
411*76559068SAndroid Build Coastguard Worker     void *NewP = Allocator->reallocate(P, Size * 2);
412*76559068SAndroid Build Coastguard Worker     EXPECT_NE(NewP, nullptr);
413*76559068SAndroid Build Coastguard Worker     for (scudo::uptr J = 0; J < Size; J++)
414*76559068SAndroid Build Coastguard Worker       EXPECT_EQ((reinterpret_cast<char *>(NewP))[J], Marker);
415*76559068SAndroid Build Coastguard Worker     memset(reinterpret_cast<char *>(NewP) + Size, Marker, Size);
416*76559068SAndroid Build Coastguard Worker     Size *= 2U;
417*76559068SAndroid Build Coastguard Worker     P = NewP;
418*76559068SAndroid Build Coastguard Worker   }
419*76559068SAndroid Build Coastguard Worker   Allocator->deallocate(P, Origin);
420*76559068SAndroid Build Coastguard Worker }
421*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,ReallocateLargeDecreasing)422*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ReallocateLargeDecreasing) {
423*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
424*76559068SAndroid Build Coastguard Worker 
425*76559068SAndroid Build Coastguard Worker   // Reallocate a large chunk all the way down to a byte, verifying that we
426*76559068SAndroid Build Coastguard Worker   // preserve the data in the process.
427*76559068SAndroid Build Coastguard Worker   scudo::uptr Size = TypeParam::Primary::SizeClassMap::MaxSize * 2;
428*76559068SAndroid Build Coastguard Worker   const scudo::uptr DataSize = 2048U;
429*76559068SAndroid Build Coastguard Worker   void *P = Allocator->allocate(Size, Origin);
430*76559068SAndroid Build Coastguard Worker   const char Marker = 'A';
431*76559068SAndroid Build Coastguard Worker   memset(P, Marker, scudo::Min(Size, DataSize));
432*76559068SAndroid Build Coastguard Worker   while (Size > 1U) {
433*76559068SAndroid Build Coastguard Worker     Size /= 2U;
434*76559068SAndroid Build Coastguard Worker     void *NewP = Allocator->reallocate(P, Size);
435*76559068SAndroid Build Coastguard Worker     EXPECT_NE(NewP, nullptr);
436*76559068SAndroid Build Coastguard Worker     for (scudo::uptr J = 0; J < scudo::Min(Size, DataSize); J++)
437*76559068SAndroid Build Coastguard Worker       EXPECT_EQ((reinterpret_cast<char *>(NewP))[J], Marker);
438*76559068SAndroid Build Coastguard Worker     P = NewP;
439*76559068SAndroid Build Coastguard Worker   }
440*76559068SAndroid Build Coastguard Worker   Allocator->deallocate(P, Origin);
441*76559068SAndroid Build Coastguard Worker }
442*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedDeathTest,ReallocateSame)443*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedDeathTest, ReallocateSame) {
444*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
445*76559068SAndroid Build Coastguard Worker 
446*76559068SAndroid Build Coastguard Worker   // Check that reallocating a chunk to a slightly smaller or larger size
447*76559068SAndroid Build Coastguard Worker   // returns the same chunk. This requires that all the sizes we iterate on use
448*76559068SAndroid Build Coastguard Worker   // the same block size, but that should be the case for MaxSize - 64 with our
449*76559068SAndroid Build Coastguard Worker   // default class size maps.
450*76559068SAndroid Build Coastguard Worker   constexpr scudo::uptr InitialSize =
451*76559068SAndroid Build Coastguard Worker       TypeParam::Primary::SizeClassMap::MaxSize - 64;
452*76559068SAndroid Build Coastguard Worker   const char Marker = 'A';
453*76559068SAndroid Build Coastguard Worker   Allocator->setFillContents(scudo::PatternOrZeroFill);
454*76559068SAndroid Build Coastguard Worker 
455*76559068SAndroid Build Coastguard Worker   void *P = Allocator->allocate(InitialSize, Origin);
456*76559068SAndroid Build Coastguard Worker   scudo::uptr CurrentSize = InitialSize;
457*76559068SAndroid Build Coastguard Worker   for (scudo::sptr Delta = -32; Delta < 32; Delta += 8) {
458*76559068SAndroid Build Coastguard Worker     memset(P, Marker, CurrentSize);
459*76559068SAndroid Build Coastguard Worker     const scudo::uptr NewSize =
460*76559068SAndroid Build Coastguard Worker         static_cast<scudo::uptr>(static_cast<scudo::sptr>(InitialSize) + Delta);
461*76559068SAndroid Build Coastguard Worker     void *NewP = Allocator->reallocate(P, NewSize);
462*76559068SAndroid Build Coastguard Worker     EXPECT_EQ(NewP, P);
463*76559068SAndroid Build Coastguard Worker 
464*76559068SAndroid Build Coastguard Worker     // Verify that existing contents have been preserved.
465*76559068SAndroid Build Coastguard Worker     for (scudo::uptr I = 0; I < scudo::Min(CurrentSize, NewSize); I++)
466*76559068SAndroid Build Coastguard Worker       EXPECT_EQ((reinterpret_cast<char *>(NewP))[I], Marker);
467*76559068SAndroid Build Coastguard Worker 
468*76559068SAndroid Build Coastguard Worker     // Verify that new bytes are set according to FillContentsMode.
469*76559068SAndroid Build Coastguard Worker     for (scudo::uptr I = CurrentSize; I < NewSize; I++) {
470*76559068SAndroid Build Coastguard Worker       unsigned char V = (reinterpret_cast<unsigned char *>(NewP))[I];
471*76559068SAndroid Build Coastguard Worker       EXPECT_TRUE(V == scudo::PatternFillByte || V == 0);
472*76559068SAndroid Build Coastguard Worker     }
473*76559068SAndroid Build Coastguard Worker 
474*76559068SAndroid Build Coastguard Worker     checkMemoryTaggingMaybe(Allocator, NewP, NewSize, 0);
475*76559068SAndroid Build Coastguard Worker     CurrentSize = NewSize;
476*76559068SAndroid Build Coastguard Worker   }
477*76559068SAndroid Build Coastguard Worker   Allocator->deallocate(P, Origin);
478*76559068SAndroid Build Coastguard Worker }
479*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,IterateOverChunks)480*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, IterateOverChunks) {
481*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
482*76559068SAndroid Build Coastguard Worker   // Allocates a bunch of chunks, then iterate over all the chunks, ensuring
483*76559068SAndroid Build Coastguard Worker   // they are the ones we allocated. This requires the allocator to not have any
484*76559068SAndroid Build Coastguard Worker   // other allocated chunk at this point (eg: won't work with the Quarantine).
485*76559068SAndroid Build Coastguard Worker   // FIXME: Make it work with UseQuarantine and tagging enabled. Internals of
486*76559068SAndroid Build Coastguard Worker   // iterateOverChunks reads header by tagged and non-tagger pointers so one of
487*76559068SAndroid Build Coastguard Worker   // them will fail.
488*76559068SAndroid Build Coastguard Worker   if (!UseQuarantine) {
489*76559068SAndroid Build Coastguard Worker     std::vector<void *> V;
490*76559068SAndroid Build Coastguard Worker     for (scudo::uptr I = 0; I < 64U; I++)
491*76559068SAndroid Build Coastguard Worker       V.push_back(Allocator->allocate(
492*76559068SAndroid Build Coastguard Worker           static_cast<scudo::uptr>(std::rand()) %
493*76559068SAndroid Build Coastguard Worker               (TypeParam::Primary::SizeClassMap::MaxSize / 2U),
494*76559068SAndroid Build Coastguard Worker           Origin));
495*76559068SAndroid Build Coastguard Worker     Allocator->disable();
496*76559068SAndroid Build Coastguard Worker     Allocator->iterateOverChunks(
497*76559068SAndroid Build Coastguard Worker         0U, static_cast<scudo::uptr>(SCUDO_MMAP_RANGE_SIZE - 1),
498*76559068SAndroid Build Coastguard Worker         [](uintptr_t Base, UNUSED size_t Size, void *Arg) {
499*76559068SAndroid Build Coastguard Worker           std::vector<void *> *V = reinterpret_cast<std::vector<void *> *>(Arg);
500*76559068SAndroid Build Coastguard Worker           void *P = reinterpret_cast<void *>(Base);
501*76559068SAndroid Build Coastguard Worker           EXPECT_NE(std::find(V->begin(), V->end(), P), V->end());
502*76559068SAndroid Build Coastguard Worker         },
503*76559068SAndroid Build Coastguard Worker         reinterpret_cast<void *>(&V));
504*76559068SAndroid Build Coastguard Worker     Allocator->enable();
505*76559068SAndroid Build Coastguard Worker     for (auto P : V)
506*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(P, Origin);
507*76559068SAndroid Build Coastguard Worker   }
508*76559068SAndroid Build Coastguard Worker }
509*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedDeathTest,UseAfterFree)510*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedDeathTest, UseAfterFree) {
511*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
512*76559068SAndroid Build Coastguard Worker 
513*76559068SAndroid Build Coastguard Worker   // Check that use-after-free is detected.
514*76559068SAndroid Build Coastguard Worker   for (scudo::uptr SizeLog = 0U; SizeLog <= 20U; SizeLog++) {
515*76559068SAndroid Build Coastguard Worker     const scudo::uptr Size = 1U << SizeLog;
516*76559068SAndroid Build Coastguard Worker     if (!Allocator->useMemoryTaggingTestOnly())
517*76559068SAndroid Build Coastguard Worker       continue;
518*76559068SAndroid Build Coastguard Worker     EXPECT_DEATH(
519*76559068SAndroid Build Coastguard Worker         {
520*76559068SAndroid Build Coastguard Worker           disableDebuggerdMaybe();
521*76559068SAndroid Build Coastguard Worker           void *P = Allocator->allocate(Size, Origin);
522*76559068SAndroid Build Coastguard Worker           Allocator->deallocate(P, Origin);
523*76559068SAndroid Build Coastguard Worker           reinterpret_cast<char *>(P)[0] = 'A';
524*76559068SAndroid Build Coastguard Worker         },
525*76559068SAndroid Build Coastguard Worker         "");
526*76559068SAndroid Build Coastguard Worker     EXPECT_DEATH(
527*76559068SAndroid Build Coastguard Worker         {
528*76559068SAndroid Build Coastguard Worker           disableDebuggerdMaybe();
529*76559068SAndroid Build Coastguard Worker           void *P = Allocator->allocate(Size, Origin);
530*76559068SAndroid Build Coastguard Worker           Allocator->deallocate(P, Origin);
531*76559068SAndroid Build Coastguard Worker           reinterpret_cast<char *>(P)[Size - 1] = 'A';
532*76559068SAndroid Build Coastguard Worker         },
533*76559068SAndroid Build Coastguard Worker         "");
534*76559068SAndroid Build Coastguard Worker   }
535*76559068SAndroid Build Coastguard Worker }
536*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedDeathTest,DoubleFreeFromPrimary)537*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedDeathTest, DoubleFreeFromPrimary) {
538*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
539*76559068SAndroid Build Coastguard Worker 
540*76559068SAndroid Build Coastguard Worker   for (scudo::uptr SizeLog = 0U; SizeLog <= 20U; SizeLog++) {
541*76559068SAndroid Build Coastguard Worker     const scudo::uptr Size = 1U << SizeLog;
542*76559068SAndroid Build Coastguard Worker     if (!isPrimaryAllocation<TestAllocator<TypeParam>>(Size, 0))
543*76559068SAndroid Build Coastguard Worker       break;
544*76559068SAndroid Build Coastguard Worker 
545*76559068SAndroid Build Coastguard Worker     // Verify that a double free results in a chunk state error.
546*76559068SAndroid Build Coastguard Worker     EXPECT_DEATH(
547*76559068SAndroid Build Coastguard Worker         {
548*76559068SAndroid Build Coastguard Worker           // Allocate from primary
549*76559068SAndroid Build Coastguard Worker           void *P = Allocator->allocate(Size, Origin);
550*76559068SAndroid Build Coastguard Worker           ASSERT_TRUE(P != nullptr);
551*76559068SAndroid Build Coastguard Worker           Allocator->deallocate(P, Origin);
552*76559068SAndroid Build Coastguard Worker           Allocator->deallocate(P, Origin);
553*76559068SAndroid Build Coastguard Worker         },
554*76559068SAndroid Build Coastguard Worker         "invalid chunk state");
555*76559068SAndroid Build Coastguard Worker   }
556*76559068SAndroid Build Coastguard Worker }
557*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedDeathTest,DisableMemoryTagging)558*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedDeathTest, DisableMemoryTagging) {
559*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
560*76559068SAndroid Build Coastguard Worker 
561*76559068SAndroid Build Coastguard Worker   if (Allocator->useMemoryTaggingTestOnly()) {
562*76559068SAndroid Build Coastguard Worker     // Check that disabling memory tagging works correctly.
563*76559068SAndroid Build Coastguard Worker     void *P = Allocator->allocate(2048, Origin);
564*76559068SAndroid Build Coastguard Worker     EXPECT_DEATH(reinterpret_cast<char *>(P)[2048] = 'A', "");
565*76559068SAndroid Build Coastguard Worker     scudo::ScopedDisableMemoryTagChecks NoTagChecks;
566*76559068SAndroid Build Coastguard Worker     Allocator->disableMemoryTagging();
567*76559068SAndroid Build Coastguard Worker     reinterpret_cast<char *>(P)[2048] = 'A';
568*76559068SAndroid Build Coastguard Worker     Allocator->deallocate(P, Origin);
569*76559068SAndroid Build Coastguard Worker 
570*76559068SAndroid Build Coastguard Worker     P = Allocator->allocate(2048, Origin);
571*76559068SAndroid Build Coastguard Worker     EXPECT_EQ(scudo::untagPointer(P), P);
572*76559068SAndroid Build Coastguard Worker     reinterpret_cast<char *>(P)[2048] = 'A';
573*76559068SAndroid Build Coastguard Worker     Allocator->deallocate(P, Origin);
574*76559068SAndroid Build Coastguard Worker 
575*76559068SAndroid Build Coastguard Worker     Allocator->releaseToOS(scudo::ReleaseToOS::Force);
576*76559068SAndroid Build Coastguard Worker   }
577*76559068SAndroid Build Coastguard Worker }
578*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,Stats)579*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, Stats) {
580*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
581*76559068SAndroid Build Coastguard Worker 
582*76559068SAndroid Build Coastguard Worker   scudo::uptr BufferSize = 8192;
583*76559068SAndroid Build Coastguard Worker   std::vector<char> Buffer(BufferSize);
584*76559068SAndroid Build Coastguard Worker   scudo::uptr ActualSize = Allocator->getStats(Buffer.data(), BufferSize);
585*76559068SAndroid Build Coastguard Worker   while (ActualSize > BufferSize) {
586*76559068SAndroid Build Coastguard Worker     BufferSize = ActualSize + 1024;
587*76559068SAndroid Build Coastguard Worker     Buffer.resize(BufferSize);
588*76559068SAndroid Build Coastguard Worker     ActualSize = Allocator->getStats(Buffer.data(), BufferSize);
589*76559068SAndroid Build Coastguard Worker   }
590*76559068SAndroid Build Coastguard Worker   std::string Stats(Buffer.begin(), Buffer.end());
591*76559068SAndroid Build Coastguard Worker   // Basic checks on the contents of the statistics output, which also allows us
592*76559068SAndroid Build Coastguard Worker   // to verify that we got it all.
593*76559068SAndroid Build Coastguard Worker   EXPECT_NE(Stats.find("Stats: SizeClassAllocator"), std::string::npos);
594*76559068SAndroid Build Coastguard Worker   EXPECT_NE(Stats.find("Stats: MapAllocator"), std::string::npos);
595*76559068SAndroid Build Coastguard Worker   EXPECT_NE(Stats.find("Stats: Quarantine"), std::string::npos);
596*76559068SAndroid Build Coastguard Worker }
597*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST_SKIP_THREAD_SAFETY(ScudoCombinedTest,CacheDrain)598*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST_SKIP_THREAD_SAFETY(ScudoCombinedTest, CacheDrain) {
599*76559068SAndroid Build Coastguard Worker   using AllocatorT = typename BaseT::AllocatorT;
600*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
601*76559068SAndroid Build Coastguard Worker 
602*76559068SAndroid Build Coastguard Worker   std::vector<void *> V;
603*76559068SAndroid Build Coastguard Worker   for (scudo::uptr I = 0; I < 64U; I++)
604*76559068SAndroid Build Coastguard Worker     V.push_back(Allocator->allocate(
605*76559068SAndroid Build Coastguard Worker         static_cast<scudo::uptr>(std::rand()) %
606*76559068SAndroid Build Coastguard Worker             (TypeParam::Primary::SizeClassMap::MaxSize / 2U),
607*76559068SAndroid Build Coastguard Worker         Origin));
608*76559068SAndroid Build Coastguard Worker   for (auto P : V)
609*76559068SAndroid Build Coastguard Worker     Allocator->deallocate(P, Origin);
610*76559068SAndroid Build Coastguard Worker 
611*76559068SAndroid Build Coastguard Worker   typename AllocatorT::TSDRegistryT::ScopedTSD TSD(
612*76559068SAndroid Build Coastguard Worker       *Allocator->getTSDRegistry());
613*76559068SAndroid Build Coastguard Worker   EXPECT_TRUE(!TSD->getCache().isEmpty());
614*76559068SAndroid Build Coastguard Worker   TSD->getCache().drain();
615*76559068SAndroid Build Coastguard Worker   EXPECT_TRUE(TSD->getCache().isEmpty());
616*76559068SAndroid Build Coastguard Worker }
617*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST_SKIP_THREAD_SAFETY(ScudoCombinedTest,ForceCacheDrain)618*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST_SKIP_THREAD_SAFETY(ScudoCombinedTest, ForceCacheDrain) {
619*76559068SAndroid Build Coastguard Worker   using AllocatorT = typename BaseT::AllocatorT;
620*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
621*76559068SAndroid Build Coastguard Worker 
622*76559068SAndroid Build Coastguard Worker   std::vector<void *> V;
623*76559068SAndroid Build Coastguard Worker   for (scudo::uptr I = 0; I < 64U; I++)
624*76559068SAndroid Build Coastguard Worker     V.push_back(Allocator->allocate(
625*76559068SAndroid Build Coastguard Worker         static_cast<scudo::uptr>(std::rand()) %
626*76559068SAndroid Build Coastguard Worker             (TypeParam::Primary::SizeClassMap::MaxSize / 2U),
627*76559068SAndroid Build Coastguard Worker         Origin));
628*76559068SAndroid Build Coastguard Worker   for (auto P : V)
629*76559068SAndroid Build Coastguard Worker     Allocator->deallocate(P, Origin);
630*76559068SAndroid Build Coastguard Worker 
631*76559068SAndroid Build Coastguard Worker   // `ForceAll` will also drain the caches.
632*76559068SAndroid Build Coastguard Worker   Allocator->releaseToOS(scudo::ReleaseToOS::ForceAll);
633*76559068SAndroid Build Coastguard Worker 
634*76559068SAndroid Build Coastguard Worker   typename AllocatorT::TSDRegistryT::ScopedTSD TSD(
635*76559068SAndroid Build Coastguard Worker       *Allocator->getTSDRegistry());
636*76559068SAndroid Build Coastguard Worker   EXPECT_TRUE(TSD->getCache().isEmpty());
637*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(TSD->getQuarantineCache().getSize(), 0U);
638*76559068SAndroid Build Coastguard Worker   EXPECT_TRUE(Allocator->getQuarantine()->isEmpty());
639*76559068SAndroid Build Coastguard Worker }
640*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,ThreadedCombined)641*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ThreadedCombined) {
642*76559068SAndroid Build Coastguard Worker   std::mutex Mutex;
643*76559068SAndroid Build Coastguard Worker   std::condition_variable Cv;
644*76559068SAndroid Build Coastguard Worker   bool Ready = false;
645*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
646*76559068SAndroid Build Coastguard Worker   std::thread Threads[32];
647*76559068SAndroid Build Coastguard Worker   for (scudo::uptr I = 0; I < ARRAY_SIZE(Threads); I++)
648*76559068SAndroid Build Coastguard Worker     Threads[I] = std::thread([&]() {
649*76559068SAndroid Build Coastguard Worker       {
650*76559068SAndroid Build Coastguard Worker         std::unique_lock<std::mutex> Lock(Mutex);
651*76559068SAndroid Build Coastguard Worker         while (!Ready)
652*76559068SAndroid Build Coastguard Worker           Cv.wait(Lock);
653*76559068SAndroid Build Coastguard Worker       }
654*76559068SAndroid Build Coastguard Worker       std::vector<std::pair<void *, scudo::uptr>> V;
655*76559068SAndroid Build Coastguard Worker       for (scudo::uptr I = 0; I < 256U; I++) {
656*76559068SAndroid Build Coastguard Worker         const scudo::uptr Size = static_cast<scudo::uptr>(std::rand()) % 4096U;
657*76559068SAndroid Build Coastguard Worker         void *P = Allocator->allocate(Size, Origin);
658*76559068SAndroid Build Coastguard Worker         // A region could have ran out of memory, resulting in a null P.
659*76559068SAndroid Build Coastguard Worker         if (P)
660*76559068SAndroid Build Coastguard Worker           V.push_back(std::make_pair(P, Size));
661*76559068SAndroid Build Coastguard Worker       }
662*76559068SAndroid Build Coastguard Worker 
663*76559068SAndroid Build Coastguard Worker       // Try to interleave pushBlocks(), popBatch() and releaseToOS().
664*76559068SAndroid Build Coastguard Worker       Allocator->releaseToOS(scudo::ReleaseToOS::Force);
665*76559068SAndroid Build Coastguard Worker 
666*76559068SAndroid Build Coastguard Worker       while (!V.empty()) {
667*76559068SAndroid Build Coastguard Worker         auto Pair = V.back();
668*76559068SAndroid Build Coastguard Worker         Allocator->deallocate(Pair.first, Origin, Pair.second);
669*76559068SAndroid Build Coastguard Worker         V.pop_back();
670*76559068SAndroid Build Coastguard Worker       }
671*76559068SAndroid Build Coastguard Worker     });
672*76559068SAndroid Build Coastguard Worker   {
673*76559068SAndroid Build Coastguard Worker     std::unique_lock<std::mutex> Lock(Mutex);
674*76559068SAndroid Build Coastguard Worker     Ready = true;
675*76559068SAndroid Build Coastguard Worker     Cv.notify_all();
676*76559068SAndroid Build Coastguard Worker   }
677*76559068SAndroid Build Coastguard Worker   for (auto &T : Threads)
678*76559068SAndroid Build Coastguard Worker     T.join();
679*76559068SAndroid Build Coastguard Worker   Allocator->releaseToOS(scudo::ReleaseToOS::Force);
680*76559068SAndroid Build Coastguard Worker }
681*76559068SAndroid Build Coastguard Worker 
682*76559068SAndroid Build Coastguard Worker // Test that multiple instantiations of the allocator have not messed up the
683*76559068SAndroid Build Coastguard Worker // process's signal handlers (GWP-ASan used to do this).
TEST(ScudoCombinedDeathTest,SKIP_ON_FUCHSIA (testSEGV))684*76559068SAndroid Build Coastguard Worker TEST(ScudoCombinedDeathTest, SKIP_ON_FUCHSIA(testSEGV)) {
685*76559068SAndroid Build Coastguard Worker   const scudo::uptr Size = 4 * scudo::getPageSizeCached();
686*76559068SAndroid Build Coastguard Worker   scudo::ReservedMemoryT ReservedMemory;
687*76559068SAndroid Build Coastguard Worker   ASSERT_TRUE(ReservedMemory.create(/*Addr=*/0U, Size, "testSEGV"));
688*76559068SAndroid Build Coastguard Worker   void *P = reinterpret_cast<void *>(ReservedMemory.getBase());
689*76559068SAndroid Build Coastguard Worker   ASSERT_NE(P, nullptr);
690*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(memset(P, 0xaa, Size), "");
691*76559068SAndroid Build Coastguard Worker   ReservedMemory.release();
692*76559068SAndroid Build Coastguard Worker }
693*76559068SAndroid Build Coastguard Worker 
694*76559068SAndroid Build Coastguard Worker struct DeathSizeClassConfig {
695*76559068SAndroid Build Coastguard Worker   static const scudo::uptr NumBits = 1;
696*76559068SAndroid Build Coastguard Worker   static const scudo::uptr MinSizeLog = 10;
697*76559068SAndroid Build Coastguard Worker   static const scudo::uptr MidSizeLog = 10;
698*76559068SAndroid Build Coastguard Worker   static const scudo::uptr MaxSizeLog = 13;
699*76559068SAndroid Build Coastguard Worker   static const scudo::u16 MaxNumCachedHint = 8;
700*76559068SAndroid Build Coastguard Worker   static const scudo::uptr MaxBytesCachedLog = 12;
701*76559068SAndroid Build Coastguard Worker   static const scudo::uptr SizeDelta = 0;
702*76559068SAndroid Build Coastguard Worker };
703*76559068SAndroid Build Coastguard Worker 
704*76559068SAndroid Build Coastguard Worker static const scudo::uptr DeathRegionSizeLog = 21U;
705*76559068SAndroid Build Coastguard Worker struct DeathConfig {
706*76559068SAndroid Build Coastguard Worker   static const bool MaySupportMemoryTagging = false;
707*76559068SAndroid Build Coastguard Worker   template <class A> using TSDRegistryT = scudo::TSDRegistrySharedT<A, 1U, 1U>;
708*76559068SAndroid Build Coastguard Worker 
709*76559068SAndroid Build Coastguard Worker   struct Primary {
710*76559068SAndroid Build Coastguard Worker     // Tiny allocator, its Primary only serves chunks of four sizes.
711*76559068SAndroid Build Coastguard Worker     using SizeClassMap = scudo::FixedSizeClassMap<DeathSizeClassConfig>;
712*76559068SAndroid Build Coastguard Worker     static const scudo::uptr RegionSizeLog = DeathRegionSizeLog;
713*76559068SAndroid Build Coastguard Worker     static const scudo::s32 MinReleaseToOsIntervalMs = INT32_MIN;
714*76559068SAndroid Build Coastguard Worker     static const scudo::s32 MaxReleaseToOsIntervalMs = INT32_MAX;
715*76559068SAndroid Build Coastguard Worker     typedef scudo::uptr CompactPtrT;
716*76559068SAndroid Build Coastguard Worker     static const scudo::uptr CompactPtrScale = 0;
717*76559068SAndroid Build Coastguard Worker     static const bool EnableRandomOffset = true;
718*76559068SAndroid Build Coastguard Worker     static const scudo::uptr MapSizeIncrement = 1UL << 18;
719*76559068SAndroid Build Coastguard Worker     static const scudo::uptr GroupSizeLog = 18;
720*76559068SAndroid Build Coastguard Worker   };
721*76559068SAndroid Build Coastguard Worker   template <typename Config>
722*76559068SAndroid Build Coastguard Worker   using PrimaryT = scudo::SizeClassAllocator64<Config>;
723*76559068SAndroid Build Coastguard Worker 
724*76559068SAndroid Build Coastguard Worker   struct Secondary {
725*76559068SAndroid Build Coastguard Worker     template <typename Config>
726*76559068SAndroid Build Coastguard Worker     using CacheT = scudo::MapAllocatorNoCache<Config>;
727*76559068SAndroid Build Coastguard Worker   };
728*76559068SAndroid Build Coastguard Worker 
729*76559068SAndroid Build Coastguard Worker   template <typename Config> using SecondaryT = scudo::MapAllocator<Config>;
730*76559068SAndroid Build Coastguard Worker };
731*76559068SAndroid Build Coastguard Worker 
TEST(ScudoCombinedDeathTest,DeathCombined)732*76559068SAndroid Build Coastguard Worker TEST(ScudoCombinedDeathTest, DeathCombined) {
733*76559068SAndroid Build Coastguard Worker   using AllocatorT = TestAllocator<DeathConfig>;
734*76559068SAndroid Build Coastguard Worker   auto Allocator = std::unique_ptr<AllocatorT>(new AllocatorT());
735*76559068SAndroid Build Coastguard Worker 
736*76559068SAndroid Build Coastguard Worker   const scudo::uptr Size = 1000U;
737*76559068SAndroid Build Coastguard Worker   void *P = Allocator->allocate(Size, Origin);
738*76559068SAndroid Build Coastguard Worker   EXPECT_NE(P, nullptr);
739*76559068SAndroid Build Coastguard Worker 
740*76559068SAndroid Build Coastguard Worker   // Invalid sized deallocation.
741*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->deallocate(P, Origin, Size + 8U), "");
742*76559068SAndroid Build Coastguard Worker 
743*76559068SAndroid Build Coastguard Worker   // Misaligned pointer. Potentially unused if EXPECT_DEATH isn't available.
744*76559068SAndroid Build Coastguard Worker   UNUSED void *MisalignedP =
745*76559068SAndroid Build Coastguard Worker       reinterpret_cast<void *>(reinterpret_cast<scudo::uptr>(P) | 1U);
746*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->deallocate(MisalignedP, Origin, Size), "");
747*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->reallocate(MisalignedP, Size * 2U), "");
748*76559068SAndroid Build Coastguard Worker 
749*76559068SAndroid Build Coastguard Worker   // Header corruption.
750*76559068SAndroid Build Coastguard Worker   scudo::u64 *H =
751*76559068SAndroid Build Coastguard Worker       reinterpret_cast<scudo::u64 *>(scudo::Chunk::getAtomicHeader(P));
752*76559068SAndroid Build Coastguard Worker   *H ^= 0x42U;
753*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->deallocate(P, Origin, Size), "");
754*76559068SAndroid Build Coastguard Worker   *H ^= 0x420042U;
755*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->deallocate(P, Origin, Size), "");
756*76559068SAndroid Build Coastguard Worker   *H ^= 0x420000U;
757*76559068SAndroid Build Coastguard Worker 
758*76559068SAndroid Build Coastguard Worker   // Invalid chunk state.
759*76559068SAndroid Build Coastguard Worker   Allocator->deallocate(P, Origin, Size);
760*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->deallocate(P, Origin, Size), "");
761*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->reallocate(P, Size * 2U), "");
762*76559068SAndroid Build Coastguard Worker   EXPECT_DEATH(Allocator->getUsableSize(P), "");
763*76559068SAndroid Build Coastguard Worker }
764*76559068SAndroid Build Coastguard Worker 
765*76559068SAndroid Build Coastguard Worker // Verify that when a region gets full, the allocator will still manage to
766*76559068SAndroid Build Coastguard Worker // fulfill the allocation through a larger size class.
TEST(ScudoCombinedTest,FullRegion)767*76559068SAndroid Build Coastguard Worker TEST(ScudoCombinedTest, FullRegion) {
768*76559068SAndroid Build Coastguard Worker   using AllocatorT = TestAllocator<DeathConfig>;
769*76559068SAndroid Build Coastguard Worker   auto Allocator = std::unique_ptr<AllocatorT>(new AllocatorT());
770*76559068SAndroid Build Coastguard Worker 
771*76559068SAndroid Build Coastguard Worker   std::vector<void *> V;
772*76559068SAndroid Build Coastguard Worker   scudo::uptr FailedAllocationsCount = 0;
773*76559068SAndroid Build Coastguard Worker   for (scudo::uptr ClassId = 1U;
774*76559068SAndroid Build Coastguard Worker        ClassId <= DeathConfig::Primary::SizeClassMap::LargestClassId;
775*76559068SAndroid Build Coastguard Worker        ClassId++) {
776*76559068SAndroid Build Coastguard Worker     const scudo::uptr Size =
777*76559068SAndroid Build Coastguard Worker         DeathConfig::Primary::SizeClassMap::getSizeByClassId(ClassId);
778*76559068SAndroid Build Coastguard Worker     // Allocate enough to fill all of the regions above this one.
779*76559068SAndroid Build Coastguard Worker     const scudo::uptr MaxNumberOfChunks =
780*76559068SAndroid Build Coastguard Worker         ((1U << DeathRegionSizeLog) / Size) *
781*76559068SAndroid Build Coastguard Worker         (DeathConfig::Primary::SizeClassMap::LargestClassId - ClassId + 1);
782*76559068SAndroid Build Coastguard Worker     void *P;
783*76559068SAndroid Build Coastguard Worker     for (scudo::uptr I = 0; I <= MaxNumberOfChunks; I++) {
784*76559068SAndroid Build Coastguard Worker       P = Allocator->allocate(Size - 64U, Origin);
785*76559068SAndroid Build Coastguard Worker       if (!P)
786*76559068SAndroid Build Coastguard Worker         FailedAllocationsCount++;
787*76559068SAndroid Build Coastguard Worker       else
788*76559068SAndroid Build Coastguard Worker         V.push_back(P);
789*76559068SAndroid Build Coastguard Worker     }
790*76559068SAndroid Build Coastguard Worker     while (!V.empty()) {
791*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(V.back(), Origin);
792*76559068SAndroid Build Coastguard Worker       V.pop_back();
793*76559068SAndroid Build Coastguard Worker     }
794*76559068SAndroid Build Coastguard Worker   }
795*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(FailedAllocationsCount, 0U);
796*76559068SAndroid Build Coastguard Worker }
797*76559068SAndroid Build Coastguard Worker 
798*76559068SAndroid Build Coastguard Worker // Ensure that releaseToOS can be called prior to any other allocator
799*76559068SAndroid Build Coastguard Worker // operation without issue.
SCUDO_TYPED_TEST(ScudoCombinedTest,ReleaseToOS)800*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ReleaseToOS) {
801*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
802*76559068SAndroid Build Coastguard Worker   Allocator->releaseToOS(scudo::ReleaseToOS::Force);
803*76559068SAndroid Build Coastguard Worker }
804*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,OddEven)805*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, OddEven) {
806*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
807*76559068SAndroid Build Coastguard Worker   Allocator->setOption(scudo::Option::MemtagTuning, M_MEMTAG_TUNING_BUFFER_OVERFLOW);
808*76559068SAndroid Build Coastguard Worker 
809*76559068SAndroid Build Coastguard Worker   if (!Allocator->useMemoryTaggingTestOnly())
810*76559068SAndroid Build Coastguard Worker     return;
811*76559068SAndroid Build Coastguard Worker 
812*76559068SAndroid Build Coastguard Worker   auto CheckOddEven = [](scudo::uptr P1, scudo::uptr P2) {
813*76559068SAndroid Build Coastguard Worker     scudo::uptr Tag1 = scudo::extractTag(scudo::loadTag(P1));
814*76559068SAndroid Build Coastguard Worker     scudo::uptr Tag2 = scudo::extractTag(scudo::loadTag(P2));
815*76559068SAndroid Build Coastguard Worker     EXPECT_NE(Tag1 % 2, Tag2 % 2);
816*76559068SAndroid Build Coastguard Worker   };
817*76559068SAndroid Build Coastguard Worker 
818*76559068SAndroid Build Coastguard Worker   using SizeClassMap = typename TypeParam::Primary::SizeClassMap;
819*76559068SAndroid Build Coastguard Worker   for (scudo::uptr ClassId = 1U; ClassId <= SizeClassMap::LargestClassId;
820*76559068SAndroid Build Coastguard Worker        ClassId++) {
821*76559068SAndroid Build Coastguard Worker     const scudo::uptr Size = SizeClassMap::getSizeByClassId(ClassId);
822*76559068SAndroid Build Coastguard Worker 
823*76559068SAndroid Build Coastguard Worker     std::set<scudo::uptr> Ptrs;
824*76559068SAndroid Build Coastguard Worker     bool Found = false;
825*76559068SAndroid Build Coastguard Worker     for (unsigned I = 0; I != 65536; ++I) {
826*76559068SAndroid Build Coastguard Worker       scudo::uptr P = scudo::untagPointer(reinterpret_cast<scudo::uptr>(
827*76559068SAndroid Build Coastguard Worker           Allocator->allocate(Size - scudo::Chunk::getHeaderSize(), Origin)));
828*76559068SAndroid Build Coastguard Worker       if (Ptrs.count(P - Size)) {
829*76559068SAndroid Build Coastguard Worker         Found = true;
830*76559068SAndroid Build Coastguard Worker         CheckOddEven(P, P - Size);
831*76559068SAndroid Build Coastguard Worker         break;
832*76559068SAndroid Build Coastguard Worker       }
833*76559068SAndroid Build Coastguard Worker       if (Ptrs.count(P + Size)) {
834*76559068SAndroid Build Coastguard Worker         Found = true;
835*76559068SAndroid Build Coastguard Worker         CheckOddEven(P, P + Size);
836*76559068SAndroid Build Coastguard Worker         break;
837*76559068SAndroid Build Coastguard Worker       }
838*76559068SAndroid Build Coastguard Worker       Ptrs.insert(P);
839*76559068SAndroid Build Coastguard Worker     }
840*76559068SAndroid Build Coastguard Worker     EXPECT_TRUE(Found);
841*76559068SAndroid Build Coastguard Worker   }
842*76559068SAndroid Build Coastguard Worker }
843*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,DisableMemInit)844*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, DisableMemInit) {
845*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
846*76559068SAndroid Build Coastguard Worker 
847*76559068SAndroid Build Coastguard Worker   std::vector<void *> Ptrs(65536);
848*76559068SAndroid Build Coastguard Worker 
849*76559068SAndroid Build Coastguard Worker   Allocator->setOption(scudo::Option::ThreadDisableMemInit, 1);
850*76559068SAndroid Build Coastguard Worker 
851*76559068SAndroid Build Coastguard Worker   constexpr scudo::uptr MinAlignLog = FIRST_32_SECOND_64(3U, 4U);
852*76559068SAndroid Build Coastguard Worker 
853*76559068SAndroid Build Coastguard Worker   // Test that if mem-init is disabled on a thread, calloc should still work as
854*76559068SAndroid Build Coastguard Worker   // expected. This is tricky to ensure when MTE is enabled, so this test tries
855*76559068SAndroid Build Coastguard Worker   // to exercise the relevant code on our MTE path.
856*76559068SAndroid Build Coastguard Worker   for (scudo::uptr ClassId = 1U; ClassId <= 8; ClassId++) {
857*76559068SAndroid Build Coastguard Worker     using SizeClassMap = typename TypeParam::Primary::SizeClassMap;
858*76559068SAndroid Build Coastguard Worker     const scudo::uptr Size =
859*76559068SAndroid Build Coastguard Worker         SizeClassMap::getSizeByClassId(ClassId) - scudo::Chunk::getHeaderSize();
860*76559068SAndroid Build Coastguard Worker     if (Size < 8)
861*76559068SAndroid Build Coastguard Worker       continue;
862*76559068SAndroid Build Coastguard Worker     for (unsigned I = 0; I != Ptrs.size(); ++I) {
863*76559068SAndroid Build Coastguard Worker       Ptrs[I] = Allocator->allocate(Size, Origin);
864*76559068SAndroid Build Coastguard Worker       memset(Ptrs[I], 0xaa, Size);
865*76559068SAndroid Build Coastguard Worker     }
866*76559068SAndroid Build Coastguard Worker     for (unsigned I = 0; I != Ptrs.size(); ++I)
867*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(Ptrs[I], Origin, Size);
868*76559068SAndroid Build Coastguard Worker     for (unsigned I = 0; I != Ptrs.size(); ++I) {
869*76559068SAndroid Build Coastguard Worker       Ptrs[I] = Allocator->allocate(Size - 8, Origin);
870*76559068SAndroid Build Coastguard Worker       memset(Ptrs[I], 0xbb, Size - 8);
871*76559068SAndroid Build Coastguard Worker     }
872*76559068SAndroid Build Coastguard Worker     for (unsigned I = 0; I != Ptrs.size(); ++I)
873*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(Ptrs[I], Origin, Size - 8);
874*76559068SAndroid Build Coastguard Worker     for (unsigned I = 0; I != Ptrs.size(); ++I) {
875*76559068SAndroid Build Coastguard Worker       Ptrs[I] = Allocator->allocate(Size, Origin, 1U << MinAlignLog, true);
876*76559068SAndroid Build Coastguard Worker       for (scudo::uptr J = 0; J < Size; ++J)
877*76559068SAndroid Build Coastguard Worker         ASSERT_EQ((reinterpret_cast<char *>(Ptrs[I]))[J], '\0');
878*76559068SAndroid Build Coastguard Worker     }
879*76559068SAndroid Build Coastguard Worker   }
880*76559068SAndroid Build Coastguard Worker 
881*76559068SAndroid Build Coastguard Worker   Allocator->setOption(scudo::Option::ThreadDisableMemInit, 0);
882*76559068SAndroid Build Coastguard Worker }
883*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,ReallocateInPlaceStress)884*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, ReallocateInPlaceStress) {
885*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
886*76559068SAndroid Build Coastguard Worker 
887*76559068SAndroid Build Coastguard Worker   // Regression test: make realloc-in-place happen at the very right end of a
888*76559068SAndroid Build Coastguard Worker   // mapped region.
889*76559068SAndroid Build Coastguard Worker   constexpr size_t nPtrs = 10000;
890*76559068SAndroid Build Coastguard Worker   for (scudo::uptr i = 1; i < 32; ++i) {
891*76559068SAndroid Build Coastguard Worker     scudo::uptr Size = 16 * i - 1;
892*76559068SAndroid Build Coastguard Worker     std::vector<void *> Ptrs;
893*76559068SAndroid Build Coastguard Worker     for (size_t i = 0; i < nPtrs; ++i) {
894*76559068SAndroid Build Coastguard Worker       void *P = Allocator->allocate(Size, Origin);
895*76559068SAndroid Build Coastguard Worker       P = Allocator->reallocate(P, Size + 1);
896*76559068SAndroid Build Coastguard Worker       Ptrs.push_back(P);
897*76559068SAndroid Build Coastguard Worker     }
898*76559068SAndroid Build Coastguard Worker 
899*76559068SAndroid Build Coastguard Worker     for (size_t i = 0; i < nPtrs; ++i)
900*76559068SAndroid Build Coastguard Worker       Allocator->deallocate(Ptrs[i], Origin);
901*76559068SAndroid Build Coastguard Worker   }
902*76559068SAndroid Build Coastguard Worker }
903*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,RingBufferDefaultDisabled)904*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, RingBufferDefaultDisabled) {
905*76559068SAndroid Build Coastguard Worker   // The RingBuffer is not initialized until tracking is enabled for the
906*76559068SAndroid Build Coastguard Worker   // first time.
907*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
908*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(0u, Allocator->getRingBufferSize());
909*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(nullptr, Allocator->getRingBufferAddress());
910*76559068SAndroid Build Coastguard Worker }
911*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,RingBufferInitOnce)912*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, RingBufferInitOnce) {
913*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
914*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
915*76559068SAndroid Build Coastguard Worker 
916*76559068SAndroid Build Coastguard Worker   auto RingBufferSize = Allocator->getRingBufferSize();
917*76559068SAndroid Build Coastguard Worker   ASSERT_GT(RingBufferSize, 0u);
918*76559068SAndroid Build Coastguard Worker   auto *RingBufferAddress = Allocator->getRingBufferAddress();
919*76559068SAndroid Build Coastguard Worker   EXPECT_NE(nullptr, RingBufferAddress);
920*76559068SAndroid Build Coastguard Worker 
921*76559068SAndroid Build Coastguard Worker   // Enable tracking again to verify that the initialization only happens once.
922*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
923*76559068SAndroid Build Coastguard Worker   ASSERT_EQ(RingBufferSize, Allocator->getRingBufferSize());
924*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(RingBufferAddress, Allocator->getRingBufferAddress());
925*76559068SAndroid Build Coastguard Worker }
926*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,RingBufferSize)927*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, RingBufferSize) {
928*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
929*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
930*76559068SAndroid Build Coastguard Worker 
931*76559068SAndroid Build Coastguard Worker   auto RingBufferSize = Allocator->getRingBufferSize();
932*76559068SAndroid Build Coastguard Worker   ASSERT_GT(RingBufferSize, 0u);
933*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(Allocator->getRingBufferAddress()[RingBufferSize - 1], '\0');
934*76559068SAndroid Build Coastguard Worker }
935*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,RingBufferAddress)936*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, RingBufferAddress) {
937*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
938*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
939*76559068SAndroid Build Coastguard Worker 
940*76559068SAndroid Build Coastguard Worker   auto *RingBufferAddress = Allocator->getRingBufferAddress();
941*76559068SAndroid Build Coastguard Worker   EXPECT_NE(RingBufferAddress, nullptr);
942*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(RingBufferAddress, Allocator->getRingBufferAddress());
943*76559068SAndroid Build Coastguard Worker }
944*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,StackDepotDefaultDisabled)945*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, StackDepotDefaultDisabled) {
946*76559068SAndroid Build Coastguard Worker   // The StackDepot is not initialized until tracking is enabled for the
947*76559068SAndroid Build Coastguard Worker   // first time.
948*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
949*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(0u, Allocator->getStackDepotSize());
950*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(nullptr, Allocator->getStackDepotAddress());
951*76559068SAndroid Build Coastguard Worker }
952*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,StackDepotInitOnce)953*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, StackDepotInitOnce) {
954*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
955*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
956*76559068SAndroid Build Coastguard Worker 
957*76559068SAndroid Build Coastguard Worker   auto StackDepotSize = Allocator->getStackDepotSize();
958*76559068SAndroid Build Coastguard Worker   EXPECT_GT(StackDepotSize, 0u);
959*76559068SAndroid Build Coastguard Worker   auto *StackDepotAddress = Allocator->getStackDepotAddress();
960*76559068SAndroid Build Coastguard Worker   EXPECT_NE(nullptr, StackDepotAddress);
961*76559068SAndroid Build Coastguard Worker 
962*76559068SAndroid Build Coastguard Worker   // Enable tracking again to verify that the initialization only happens once.
963*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
964*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(StackDepotSize, Allocator->getStackDepotSize());
965*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(StackDepotAddress, Allocator->getStackDepotAddress());
966*76559068SAndroid Build Coastguard Worker }
967*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,StackDepotSize)968*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, StackDepotSize) {
969*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
970*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
971*76559068SAndroid Build Coastguard Worker 
972*76559068SAndroid Build Coastguard Worker   auto StackDepotSize = Allocator->getStackDepotSize();
973*76559068SAndroid Build Coastguard Worker   EXPECT_GT(StackDepotSize, 0u);
974*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(Allocator->getStackDepotAddress()[StackDepotSize - 1], '\0');
975*76559068SAndroid Build Coastguard Worker }
976*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,StackDepotAddress)977*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, StackDepotAddress) {
978*76559068SAndroid Build Coastguard Worker   auto *Allocator = this->Allocator.get();
979*76559068SAndroid Build Coastguard Worker   Allocator->setTrackAllocationStacks(true);
980*76559068SAndroid Build Coastguard Worker 
981*76559068SAndroid Build Coastguard Worker   auto *StackDepotAddress = Allocator->getStackDepotAddress();
982*76559068SAndroid Build Coastguard Worker   EXPECT_NE(StackDepotAddress, nullptr);
983*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(StackDepotAddress, Allocator->getStackDepotAddress());
984*76559068SAndroid Build Coastguard Worker }
985*76559068SAndroid Build Coastguard Worker 
SCUDO_TYPED_TEST(ScudoCombinedTest,StackDepot)986*76559068SAndroid Build Coastguard Worker SCUDO_TYPED_TEST(ScudoCombinedTest, StackDepot) {
987*76559068SAndroid Build Coastguard Worker   alignas(scudo::StackDepot) char Buf[sizeof(scudo::StackDepot) +
988*76559068SAndroid Build Coastguard Worker                                       1024 * sizeof(scudo::atomic_u64) +
989*76559068SAndroid Build Coastguard Worker                                       1024 * sizeof(scudo::atomic_u32)] = {};
990*76559068SAndroid Build Coastguard Worker   auto *Depot = reinterpret_cast<scudo::StackDepot *>(Buf);
991*76559068SAndroid Build Coastguard Worker   Depot->init(1024, 1024);
992*76559068SAndroid Build Coastguard Worker   ASSERT_TRUE(Depot->isValid(sizeof(Buf)));
993*76559068SAndroid Build Coastguard Worker   ASSERT_FALSE(Depot->isValid(sizeof(Buf) - 1));
994*76559068SAndroid Build Coastguard Worker   scudo::uptr Stack[] = {1, 2, 3};
995*76559068SAndroid Build Coastguard Worker   scudo::u32 Elem = Depot->insert(&Stack[0], &Stack[3]);
996*76559068SAndroid Build Coastguard Worker   scudo::uptr RingPosPtr = 0;
997*76559068SAndroid Build Coastguard Worker   scudo::uptr SizePtr = 0;
998*76559068SAndroid Build Coastguard Worker   ASSERT_TRUE(Depot->find(Elem, &RingPosPtr, &SizePtr));
999*76559068SAndroid Build Coastguard Worker   ASSERT_EQ(SizePtr, 3u);
1000*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(Depot->at(RingPosPtr), 1u);
1001*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(Depot->at(RingPosPtr + 1), 2u);
1002*76559068SAndroid Build Coastguard Worker   EXPECT_EQ(Depot->at(RingPosPtr + 2), 3u);
1003*76559068SAndroid Build Coastguard Worker }
1004*76559068SAndroid Build Coastguard Worker 
1005*76559068SAndroid Build Coastguard Worker #if SCUDO_CAN_USE_PRIMARY64
1006*76559068SAndroid Build Coastguard Worker #if SCUDO_TRUSTY
1007*76559068SAndroid Build Coastguard Worker 
1008*76559068SAndroid Build Coastguard Worker // TrustyConfig is designed for a domain-specific allocator. Add a basic test
1009*76559068SAndroid Build Coastguard Worker // which covers only simple operations and ensure the configuration is able to
1010*76559068SAndroid Build Coastguard Worker // compile.
TEST(ScudoCombinedTest,BasicTrustyConfig)1011*76559068SAndroid Build Coastguard Worker TEST(ScudoCombinedTest, BasicTrustyConfig) {
1012*76559068SAndroid Build Coastguard Worker   using AllocatorT = scudo::Allocator<scudo::TrustyConfig>;
1013*76559068SAndroid Build Coastguard Worker   auto Allocator = std::unique_ptr<AllocatorT>(new AllocatorT());
1014*76559068SAndroid Build Coastguard Worker 
1015*76559068SAndroid Build Coastguard Worker   for (scudo::uptr ClassId = 1U;
1016*76559068SAndroid Build Coastguard Worker        ClassId <= scudo::TrustyConfig::SizeClassMap::LargestClassId;
1017*76559068SAndroid Build Coastguard Worker        ClassId++) {
1018*76559068SAndroid Build Coastguard Worker     const scudo::uptr Size =
1019*76559068SAndroid Build Coastguard Worker         scudo::TrustyConfig::SizeClassMap::getSizeByClassId(ClassId);
1020*76559068SAndroid Build Coastguard Worker     void *p = Allocator->allocate(Size - scudo::Chunk::getHeaderSize(), Origin);
1021*76559068SAndroid Build Coastguard Worker     ASSERT_NE(p, nullptr);
1022*76559068SAndroid Build Coastguard Worker     free(p);
1023*76559068SAndroid Build Coastguard Worker   }
1024*76559068SAndroid Build Coastguard Worker 
1025*76559068SAndroid Build Coastguard Worker   bool UnlockRequired;
1026*76559068SAndroid Build Coastguard Worker   typename AllocatorT::TSDRegistryT::ScopedTSD TSD(
1027*76559068SAndroid Build Coastguard Worker       *Allocator->getTSDRegistry());
1028*76559068SAndroid Build Coastguard Worker   TSD->getCache().drain();
1029*76559068SAndroid Build Coastguard Worker 
1030*76559068SAndroid Build Coastguard Worker   Allocator->releaseToOS(scudo::ReleaseToOS::Force);
1031*76559068SAndroid Build Coastguard Worker }
1032*76559068SAndroid Build Coastguard Worker 
1033*76559068SAndroid Build Coastguard Worker #endif
1034*76559068SAndroid Build Coastguard Worker #endif
1035