1 //===-- sanitizer_allocator.cpp -------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file is shared between AddressSanitizer and ThreadSanitizer
10 // run-time libraries.
11 // This allocator is used inside run-times.
12 //===----------------------------------------------------------------------===//
14 #include "sanitizer_allocator.h"
16 #include "sanitizer_allocator_checks.h"
17 #include "sanitizer_allocator_internal.h"
18 #include "sanitizer_atomic.h"
19 #include "sanitizer_common.h"
20 #include "sanitizer_platform.h"
22 namespace __sanitizer
{
24 // Default allocator names.
25 const char *PrimaryAllocatorName
= "SizeClassAllocator";
26 const char *SecondaryAllocatorName
= "LargeMmapAllocator";
28 alignas(64) static char internal_alloc_placeholder
[sizeof(InternalAllocator
)];
29 static atomic_uint8_t internal_allocator_initialized
;
30 static StaticSpinMutex internal_alloc_init_mu
;
32 static InternalAllocatorCache internal_allocator_cache
;
33 static StaticSpinMutex internal_allocator_cache_mu
;
35 InternalAllocator
*internal_allocator() {
36 InternalAllocator
*internal_allocator_instance
=
37 reinterpret_cast<InternalAllocator
*>(&internal_alloc_placeholder
);
38 if (atomic_load(&internal_allocator_initialized
, memory_order_acquire
) == 0) {
39 SpinMutexLock
l(&internal_alloc_init_mu
);
40 if (atomic_load(&internal_allocator_initialized
, memory_order_relaxed
) ==
42 internal_allocator_instance
->Init(kReleaseToOSIntervalNever
);
43 atomic_store(&internal_allocator_initialized
, 1, memory_order_release
);
46 return internal_allocator_instance
;
49 static void *RawInternalAlloc(uptr size
, InternalAllocatorCache
*cache
,
51 if (alignment
== 0) alignment
= 8;
53 SpinMutexLock
l(&internal_allocator_cache_mu
);
54 return internal_allocator()->Allocate(&internal_allocator_cache
, size
,
57 return internal_allocator()->Allocate(cache
, size
, alignment
);
60 static void *RawInternalRealloc(void *ptr
, uptr size
,
61 InternalAllocatorCache
*cache
) {
62 constexpr usize alignment
= Max
<usize
>(8, sizeof(void *));
64 SpinMutexLock
l(&internal_allocator_cache_mu
);
65 return internal_allocator()->Reallocate(&internal_allocator_cache
, ptr
,
68 return internal_allocator()->Reallocate(cache
, ptr
, size
, alignment
);
71 static void RawInternalFree(void *ptr
, InternalAllocatorCache
*cache
) {
73 SpinMutexLock
l(&internal_allocator_cache_mu
);
74 return internal_allocator()->Deallocate(&internal_allocator_cache
, ptr
);
76 internal_allocator()->Deallocate(cache
, ptr
);
79 static void NORETURN
ReportInternalAllocatorOutOfMemory(uptr requested_size
) {
80 SetAllocatorOutOfMemory();
81 Report("FATAL: %s: internal allocator is out of memory trying to allocate "
82 "0x%zx bytes\n", SanitizerToolName
, requested_size
);
86 void *InternalAlloc(uptr size
, InternalAllocatorCache
*cache
, uptr alignment
) {
87 void *p
= RawInternalAlloc(size
, cache
, alignment
);
89 ReportInternalAllocatorOutOfMemory(size
);
93 void *InternalRealloc(void *addr
, uptr size
, InternalAllocatorCache
*cache
) {
94 void *p
= RawInternalRealloc(addr
, size
, cache
);
96 ReportInternalAllocatorOutOfMemory(size
);
100 void *InternalReallocArray(void *addr
, uptr count
, uptr size
,
101 InternalAllocatorCache
*cache
) {
102 if (UNLIKELY(CheckForCallocOverflow(count
, size
))) {
104 "FATAL: %s: reallocarray parameters overflow: count * size (%zd * %zd) "
105 "cannot be represented in type size_t\n",
106 SanitizerToolName
, count
, size
);
109 return InternalRealloc(addr
, count
* size
, cache
);
112 void *InternalCalloc(uptr count
, uptr size
, InternalAllocatorCache
*cache
) {
113 if (UNLIKELY(CheckForCallocOverflow(count
, size
))) {
114 Report("FATAL: %s: calloc parameters overflow: count * size (%zd * %zd) "
115 "cannot be represented in type size_t\n", SanitizerToolName
, count
,
119 void *p
= InternalAlloc(count
* size
, cache
);
121 internal_memset(p
, 0, count
* size
);
125 void InternalFree(void *addr
, InternalAllocatorCache
*cache
) {
126 RawInternalFree(addr
, cache
);
129 void InternalAllocatorLock() SANITIZER_NO_THREAD_SAFETY_ANALYSIS
{
130 internal_allocator_cache_mu
.Lock();
131 internal_allocator()->ForceLock();
134 void InternalAllocatorUnlock() SANITIZER_NO_THREAD_SAFETY_ANALYSIS
{
135 internal_allocator()->ForceUnlock();
136 internal_allocator_cache_mu
.Unlock();
140 constexpr usize kLowLevelAllocatorDefaultAlignment
=
141 Max
<usize
>(8, sizeof(void *));
142 constexpr uptr kMinNumPagesRounded
= 16;
143 constexpr uptr kMinRoundedSize
= 65536;
144 static uptr low_level_alloc_min_alignment
= kLowLevelAllocatorDefaultAlignment
;
145 static LowLevelAllocateCallback low_level_alloc_callback
;
147 static LowLevelAllocator Alloc
;
148 LowLevelAllocator
&GetGlobalLowLevelAllocator() { return Alloc
; }
150 void *LowLevelAllocator::Allocate(uptr size
) {
151 // Align allocation size.
152 size
= RoundUpTo(size
, low_level_alloc_min_alignment
);
153 if (allocated_end_
- allocated_current_
< (sptr
)size
) {
154 uptr size_to_allocate
= RoundUpTo(
155 size
, Min(GetPageSizeCached() * kMinNumPagesRounded
, kMinRoundedSize
));
156 allocated_current_
= (char *)MmapOrDie(size_to_allocate
, __func__
);
157 allocated_end_
= allocated_current_
+ size_to_allocate
;
158 if (low_level_alloc_callback
) {
159 low_level_alloc_callback((uptr
)allocated_current_
, size_to_allocate
);
162 CHECK(allocated_end_
- allocated_current_
>= (sptr
)size
);
163 void *res
= allocated_current_
;
164 allocated_current_
+= size
;
168 void SetLowLevelAllocateMinAlignment(uptr alignment
) {
169 CHECK(IsPowerOfTwo(alignment
));
170 low_level_alloc_min_alignment
= Max(alignment
, low_level_alloc_min_alignment
);
173 void SetLowLevelAllocateCallback(LowLevelAllocateCallback callback
) {
174 low_level_alloc_callback
= callback
;
177 // Allocator's OOM and other errors handling support.
179 static atomic_uint8_t allocator_out_of_memory
= {0};
180 static atomic_uint8_t allocator_may_return_null
= {0};
182 bool IsAllocatorOutOfMemory() {
183 return atomic_load_relaxed(&allocator_out_of_memory
);
186 void SetAllocatorOutOfMemory() {
187 atomic_store_relaxed(&allocator_out_of_memory
, 1);
190 bool AllocatorMayReturnNull() {
191 return atomic_load(&allocator_may_return_null
, memory_order_relaxed
);
194 void SetAllocatorMayReturnNull(bool may_return_null
) {
195 atomic_store(&allocator_may_return_null
, may_return_null
,
196 memory_order_relaxed
);
199 void PrintHintAllocatorCannotReturnNull() {
200 Report("HINT: if you don't care about these errors you may set "
201 "allocator_may_return_null=1\n");
204 static atomic_uint8_t rss_limit_exceeded
;
206 bool IsRssLimitExceeded() {
207 return atomic_load(&rss_limit_exceeded
, memory_order_relaxed
);
210 void SetRssLimitExceeded(bool limit_exceeded
) {
211 atomic_store(&rss_limit_exceeded
, limit_exceeded
, memory_order_relaxed
);
214 } // namespace __sanitizer