10b57cec5SDimitry Andric //===-- asan_allocator.h ----------------------------------------*- C++ -*-===// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric // 90b57cec5SDimitry Andric // This file is a part of AddressSanitizer, an address sanity checker. 100b57cec5SDimitry Andric // 1168d75effSDimitry Andric // ASan-private header for asan_allocator.cpp. 120b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 130b57cec5SDimitry Andric 140b57cec5SDimitry Andric #ifndef ASAN_ALLOCATOR_H 150b57cec5SDimitry Andric #define ASAN_ALLOCATOR_H 160b57cec5SDimitry Andric 170b57cec5SDimitry Andric #include "asan_flags.h" 180b57cec5SDimitry Andric #include "asan_interceptors.h" 19e8d8bef9SDimitry Andric #include "asan_internal.h" 200b57cec5SDimitry Andric #include "sanitizer_common/sanitizer_allocator.h" 210b57cec5SDimitry Andric #include "sanitizer_common/sanitizer_list.h" 22e8d8bef9SDimitry Andric #include "sanitizer_common/sanitizer_platform.h" 230b57cec5SDimitry Andric 240b57cec5SDimitry Andric namespace __asan { 250b57cec5SDimitry Andric 260b57cec5SDimitry Andric enum AllocType { 270b57cec5SDimitry Andric FROM_MALLOC = 1, // Memory block came from malloc, calloc, realloc, etc. 280b57cec5SDimitry Andric FROM_NEW = 2, // Memory block came from operator new. 290b57cec5SDimitry Andric FROM_NEW_BR = 3 // Memory block came from operator new [ ] 300b57cec5SDimitry Andric }; 310b57cec5SDimitry Andric 32e8d8bef9SDimitry Andric class AsanChunk; 330b57cec5SDimitry Andric 340b57cec5SDimitry Andric struct AllocatorOptions { 350b57cec5SDimitry Andric u32 quarantine_size_mb; 360b57cec5SDimitry Andric u32 thread_local_quarantine_size_kb; 370b57cec5SDimitry Andric u16 min_redzone; 380b57cec5SDimitry Andric u16 max_redzone; 390b57cec5SDimitry Andric u8 may_return_null; 400b57cec5SDimitry Andric u8 alloc_dealloc_mismatch; 410b57cec5SDimitry Andric s32 release_to_os_interval_ms; 420b57cec5SDimitry Andric 430b57cec5SDimitry Andric void SetFrom(const Flags *f, const CommonFlags *cf); 440b57cec5SDimitry Andric void CopyTo(Flags *f, CommonFlags *cf); 450b57cec5SDimitry Andric }; 460b57cec5SDimitry Andric 470b57cec5SDimitry Andric void InitializeAllocator(const AllocatorOptions &options); 480b57cec5SDimitry Andric void ReInitializeAllocator(const AllocatorOptions &options); 490b57cec5SDimitry Andric void GetAllocatorOptions(AllocatorOptions *options); 500b57cec5SDimitry Andric 510b57cec5SDimitry Andric class AsanChunkView { 520b57cec5SDimitry Andric public: 530b57cec5SDimitry Andric explicit AsanChunkView(AsanChunk *chunk) : chunk_(chunk) {} 540b57cec5SDimitry Andric bool IsValid() const; // Checks if AsanChunkView points to a valid 550b57cec5SDimitry Andric // allocated or quarantined chunk. 560b57cec5SDimitry Andric bool IsAllocated() const; // Checks if the memory is currently allocated. 570b57cec5SDimitry Andric bool IsQuarantined() const; // Checks if the memory is currently quarantined. 580b57cec5SDimitry Andric uptr Beg() const; // First byte of user memory. 590b57cec5SDimitry Andric uptr End() const; // Last byte of user memory. 600b57cec5SDimitry Andric uptr UsedSize() const; // Size requested by the user. 610b57cec5SDimitry Andric u32 UserRequestedAlignment() const; // Originally requested alignment. 620b57cec5SDimitry Andric uptr AllocTid() const; 630b57cec5SDimitry Andric uptr FreeTid() const; 640b57cec5SDimitry Andric bool Eq(const AsanChunkView &c) const { return chunk_ == c.chunk_; } 650b57cec5SDimitry Andric u32 GetAllocStackId() const; 660b57cec5SDimitry Andric u32 GetFreeStackId() const; 670b57cec5SDimitry Andric AllocType GetAllocType() const; 680b57cec5SDimitry Andric bool AddrIsInside(uptr addr, uptr access_size, sptr *offset) const { 690b57cec5SDimitry Andric if (addr >= Beg() && (addr + access_size) <= End()) { 700b57cec5SDimitry Andric *offset = addr - Beg(); 710b57cec5SDimitry Andric return true; 720b57cec5SDimitry Andric } 730b57cec5SDimitry Andric return false; 740b57cec5SDimitry Andric } 750b57cec5SDimitry Andric bool AddrIsAtLeft(uptr addr, uptr access_size, sptr *offset) const { 760b57cec5SDimitry Andric (void)access_size; 770b57cec5SDimitry Andric if (addr < Beg()) { 780b57cec5SDimitry Andric *offset = Beg() - addr; 790b57cec5SDimitry Andric return true; 800b57cec5SDimitry Andric } 810b57cec5SDimitry Andric return false; 820b57cec5SDimitry Andric } 830b57cec5SDimitry Andric bool AddrIsAtRight(uptr addr, uptr access_size, sptr *offset) const { 840b57cec5SDimitry Andric if (addr + access_size > End()) { 850b57cec5SDimitry Andric *offset = addr - End(); 860b57cec5SDimitry Andric return true; 870b57cec5SDimitry Andric } 880b57cec5SDimitry Andric return false; 890b57cec5SDimitry Andric } 900b57cec5SDimitry Andric 910b57cec5SDimitry Andric private: 920b57cec5SDimitry Andric AsanChunk *const chunk_; 930b57cec5SDimitry Andric }; 940b57cec5SDimitry Andric 950b57cec5SDimitry Andric AsanChunkView FindHeapChunkByAddress(uptr address); 960b57cec5SDimitry Andric AsanChunkView FindHeapChunkByAllocBeg(uptr address); 970b57cec5SDimitry Andric 980b57cec5SDimitry Andric // List of AsanChunks with total size. 990b57cec5SDimitry Andric class AsanChunkFifoList: public IntrusiveList<AsanChunk> { 1000b57cec5SDimitry Andric public: 1010b57cec5SDimitry Andric explicit AsanChunkFifoList(LinkerInitialized) { } 1020b57cec5SDimitry Andric AsanChunkFifoList() { clear(); } 1030b57cec5SDimitry Andric void Push(AsanChunk *n); 1040b57cec5SDimitry Andric void PushList(AsanChunkFifoList *q); 1050b57cec5SDimitry Andric AsanChunk *Pop(); 1060b57cec5SDimitry Andric uptr size() { return size_; } 1070b57cec5SDimitry Andric void clear() { 1080b57cec5SDimitry Andric IntrusiveList<AsanChunk>::clear(); 1090b57cec5SDimitry Andric size_ = 0; 1100b57cec5SDimitry Andric } 1110b57cec5SDimitry Andric private: 1120b57cec5SDimitry Andric uptr size_; 1130b57cec5SDimitry Andric }; 1140b57cec5SDimitry Andric 1150b57cec5SDimitry Andric struct AsanMapUnmapCallback { 1160b57cec5SDimitry Andric void OnMap(uptr p, uptr size) const; 11706c3fb27SDimitry Andric void OnMapSecondary(uptr p, uptr size, uptr user_begin, uptr user_size) const; 1180b57cec5SDimitry Andric void OnUnmap(uptr p, uptr size) const; 1190b57cec5SDimitry Andric }; 1200b57cec5SDimitry Andric 1210b57cec5SDimitry Andric #if SANITIZER_CAN_USE_ALLOCATOR64 1220b57cec5SDimitry Andric # if SANITIZER_FUCHSIA 123*5f757f3fSDimitry Andric // This is a sentinel indicating we do not want the primary allocator arena to 124*5f757f3fSDimitry Andric // be placed at a fixed address. It will be anonymously mmap'd. 1250b57cec5SDimitry Andric const uptr kAllocatorSpace = ~(uptr)0; 126*5f757f3fSDimitry Andric # if SANITIZER_RISCV64 127*5f757f3fSDimitry Andric 128*5f757f3fSDimitry Andric // These are sanitizer tunings that allow all bringup tests for RISCV-64 Sv39 + 129*5f757f3fSDimitry Andric // Fuchsia to run with asan-instrumented. That is, we can run bringup, e2e, 130*5f757f3fSDimitry Andric // libc, and scudo tests with this configuration. 131*5f757f3fSDimitry Andric // 132*5f757f3fSDimitry Andric // TODO: This is specifically tuned for Sv39. 48/57 will likely require other 133*5f757f3fSDimitry Andric // tunings, or possibly use the same tunings Fuchsia uses for other archs. The 134*5f757f3fSDimitry Andric // VMA size isn't technically tied to the Fuchsia System ABI, so once 48/57 is 135*5f757f3fSDimitry Andric // supported, we'd need a way of dynamically checking what the VMA size is and 136*5f757f3fSDimitry Andric // determining optimal configuration. 137*5f757f3fSDimitry Andric 138*5f757f3fSDimitry Andric // This indicates the total amount of space dedicated for the primary allocator 139*5f757f3fSDimitry Andric // during initialization. This is roughly proportional to the size set by the 140*5f757f3fSDimitry Andric // FuchsiaConfig for scudo (~11.25GB == ~2^33.49). Requesting any more could 141*5f757f3fSDimitry Andric // lead to some failures in sanitized bringup tests where we can't allocate new 142*5f757f3fSDimitry Andric // vmars because there wouldn't be enough contiguous space. We could try 2^34 if 143*5f757f3fSDimitry Andric // we re-evaluate the SizeClassMap settings. 144*5f757f3fSDimitry Andric const uptr kAllocatorSize = UINT64_C(1) << 33; // 8GB 145*5f757f3fSDimitry Andric 146*5f757f3fSDimitry Andric // This is roughly equivalent to the configuration for the VeryDenseSizeClassMap 147*5f757f3fSDimitry Andric // but has fewer size classes (ideally at most 32). Fewer class sizes means the 148*5f757f3fSDimitry Andric // region size for each class is larger, thus less chances of running out of 149*5f757f3fSDimitry Andric // space for each region. The main differences are the MidSizeLog (which is 150*5f757f3fSDimitry Andric // smaller) and the MaxSizeLog (which is larger). 151*5f757f3fSDimitry Andric // 152*5f757f3fSDimitry Andric // - The MaxSizeLog is higher to allow some of the largest allocations I've 153*5f757f3fSDimitry Andric // observed to be placed in the primary allocator's arena as opposed to being 154*5f757f3fSDimitry Andric // mmap'd by the secondary allocator. This helps reduce fragmentation from 155*5f757f3fSDimitry Andric // large classes. A huge example of this the scudo allocator tests (and its 156*5f757f3fSDimitry Andric // testing infrastructure) which malloc's/new's objects on the order of 157*5f757f3fSDimitry Andric // hundreds of kilobytes which normally would not be in the primary allocator 158*5f757f3fSDimitry Andric // arena with the default VeryDenseSizeClassMap. 159*5f757f3fSDimitry Andric // - The MidSizeLog is reduced to help shrink the number of size classes and 160*5f757f3fSDimitry Andric // increase region size. Without this, we'd see ASan complain many times about 161*5f757f3fSDimitry Andric // a region running out of available space. 162*5f757f3fSDimitry Andric // 163*5f757f3fSDimitry Andric // This differs a bit from the fuchsia config in scudo, mainly from the NumBits, 164*5f757f3fSDimitry Andric // MaxSizeLog, and NumCachedHintT. This should place the number of size classes 165*5f757f3fSDimitry Andric // for scudo at 45 and some large objects allocated by this config would be 166*5f757f3fSDimitry Andric // placed in the arena whereas scudo would mmap them. The asan allocator needs 167*5f757f3fSDimitry Andric // to have a number of classes that are a power of 2 for various internal things 168*5f757f3fSDimitry Andric // to work, so we can't match the scudo settings to a tee. The sanitizer 169*5f757f3fSDimitry Andric // allocator is slightly slower than scudo's but this is enough to get 170*5f757f3fSDimitry Andric // memory-intensive scudo tests to run with asan instrumentation. 171*5f757f3fSDimitry Andric typedef SizeClassMap</*kNumBits=*/2, 172*5f757f3fSDimitry Andric /*kMinSizeLog=*/5, 173*5f757f3fSDimitry Andric /*kMidSizeLog=*/8, 174*5f757f3fSDimitry Andric /*kMaxSizeLog=*/18, 175*5f757f3fSDimitry Andric /*kNumCachedHintT=*/8, 176*5f757f3fSDimitry Andric /*kMaxBytesCachedLog=*/10> 177*5f757f3fSDimitry Andric SizeClassMap; 178*5f757f3fSDimitry Andric static_assert(SizeClassMap::kNumClassesRounded <= 32, 179*5f757f3fSDimitry Andric "The above tunings were specifically selected to ensure there " 180*5f757f3fSDimitry Andric "would be at most 32 size classes. This restriction could be " 181*5f757f3fSDimitry Andric "loosened to 64 size classes if we can find a configuration of " 182*5f757f3fSDimitry Andric "allocator size and SizeClassMap tunings that allows us to " 183*5f757f3fSDimitry Andric "reliably run all bringup tests in a sanitized environment."); 184*5f757f3fSDimitry Andric 185*5f757f3fSDimitry Andric # else 186*5f757f3fSDimitry Andric // These are the default allocator tunings for non-RISCV environments where the 187*5f757f3fSDimitry Andric // VMA is usually 48 bits and we have lots of space. 1880b57cec5SDimitry Andric const uptr kAllocatorSize = 0x40000000000ULL; // 4T. 1890b57cec5SDimitry Andric typedef DefaultSizeClassMap SizeClassMap; 190*5f757f3fSDimitry Andric # endif 1910b57cec5SDimitry Andric # elif defined(__powerpc64__) 1920b57cec5SDimitry Andric const uptr kAllocatorSpace = ~(uptr)0; 1930b57cec5SDimitry Andric const uptr kAllocatorSize = 0x20000000000ULL; // 2T. 1940b57cec5SDimitry Andric typedef DefaultSizeClassMap SizeClassMap; 1950b57cec5SDimitry Andric # elif defined(__aarch64__) && SANITIZER_ANDROID 1960b57cec5SDimitry Andric // Android needs to support 39, 42 and 48 bit VMA. 1970b57cec5SDimitry Andric const uptr kAllocatorSpace = ~(uptr)0; 1980b57cec5SDimitry Andric const uptr kAllocatorSize = 0x2000000000ULL; // 128G. 1990b57cec5SDimitry Andric typedef VeryCompactSizeClassMap SizeClassMap; 200e8d8bef9SDimitry Andric # elif SANITIZER_RISCV64 201e8d8bef9SDimitry Andric const uptr kAllocatorSpace = ~(uptr)0; 202e8d8bef9SDimitry Andric const uptr kAllocatorSize = 0x2000000000ULL; // 128G. 203e8d8bef9SDimitry Andric typedef VeryDenseSizeClassMap SizeClassMap; 2040b57cec5SDimitry Andric # elif defined(__sparc__) 2050b57cec5SDimitry Andric const uptr kAllocatorSpace = ~(uptr)0; 2060b57cec5SDimitry Andric const uptr kAllocatorSize = 0x20000000000ULL; // 2T. 2070b57cec5SDimitry Andric typedef DefaultSizeClassMap SizeClassMap; 2080b57cec5SDimitry Andric # elif SANITIZER_WINDOWS 2090b57cec5SDimitry Andric const uptr kAllocatorSpace = ~(uptr)0; 2100b57cec5SDimitry Andric const uptr kAllocatorSize = 0x8000000000ULL; // 500G 2110b57cec5SDimitry Andric typedef DefaultSizeClassMap SizeClassMap; 21206c3fb27SDimitry Andric # elif SANITIZER_APPLE 2130b57cec5SDimitry Andric const uptr kAllocatorSpace = 0x600000000000ULL; 2140b57cec5SDimitry Andric const uptr kAllocatorSize = 0x40000000000ULL; // 4T. 2150b57cec5SDimitry Andric typedef DefaultSizeClassMap SizeClassMap; 21606c3fb27SDimitry Andric # else 21706c3fb27SDimitry Andric const uptr kAllocatorSpace = 0x500000000000ULL; 21806c3fb27SDimitry Andric const uptr kAllocatorSize = 0x40000000000ULL; // 4T. 21906c3fb27SDimitry Andric typedef DefaultSizeClassMap SizeClassMap; 2200b57cec5SDimitry Andric # endif 2210b57cec5SDimitry Andric template <typename AddressSpaceViewTy> 2220b57cec5SDimitry Andric struct AP64 { // Allocator64 parameters. Deliberately using a short name. 2230b57cec5SDimitry Andric static const uptr kSpaceBeg = kAllocatorSpace; 2240b57cec5SDimitry Andric static const uptr kSpaceSize = kAllocatorSize; 2250b57cec5SDimitry Andric static const uptr kMetadataSize = 0; 2260b57cec5SDimitry Andric typedef __asan::SizeClassMap SizeClassMap; 2270b57cec5SDimitry Andric typedef AsanMapUnmapCallback MapUnmapCallback; 2280b57cec5SDimitry Andric static const uptr kFlags = 0; 2290b57cec5SDimitry Andric using AddressSpaceView = AddressSpaceViewTy; 2300b57cec5SDimitry Andric }; 2310b57cec5SDimitry Andric 2320b57cec5SDimitry Andric template <typename AddressSpaceView> 2330b57cec5SDimitry Andric using PrimaryAllocatorASVT = SizeClassAllocator64<AP64<AddressSpaceView>>; 2340b57cec5SDimitry Andric using PrimaryAllocator = PrimaryAllocatorASVT<LocalAddressSpaceView>; 2350b57cec5SDimitry Andric #else // Fallback to SizeClassAllocator32. 2360b57cec5SDimitry Andric typedef CompactSizeClassMap SizeClassMap; 2370b57cec5SDimitry Andric template <typename AddressSpaceViewTy> 2380b57cec5SDimitry Andric struct AP32 { 2390b57cec5SDimitry Andric static const uptr kSpaceBeg = 0; 2400b57cec5SDimitry Andric static const u64 kSpaceSize = SANITIZER_MMAP_RANGE_SIZE; 241e8d8bef9SDimitry Andric static const uptr kMetadataSize = 0; 2420b57cec5SDimitry Andric typedef __asan::SizeClassMap SizeClassMap; 2430b57cec5SDimitry Andric static const uptr kRegionSizeLog = 20; 2440b57cec5SDimitry Andric using AddressSpaceView = AddressSpaceViewTy; 2450b57cec5SDimitry Andric typedef AsanMapUnmapCallback MapUnmapCallback; 2460b57cec5SDimitry Andric static const uptr kFlags = 0; 2470b57cec5SDimitry Andric }; 2480b57cec5SDimitry Andric template <typename AddressSpaceView> 2490b57cec5SDimitry Andric using PrimaryAllocatorASVT = SizeClassAllocator32<AP32<AddressSpaceView> >; 2500b57cec5SDimitry Andric using PrimaryAllocator = PrimaryAllocatorASVT<LocalAddressSpaceView>; 2510b57cec5SDimitry Andric #endif // SANITIZER_CAN_USE_ALLOCATOR64 2520b57cec5SDimitry Andric 2530b57cec5SDimitry Andric static const uptr kNumberOfSizeClasses = SizeClassMap::kNumClasses; 2540b57cec5SDimitry Andric 2550b57cec5SDimitry Andric template <typename AddressSpaceView> 2560b57cec5SDimitry Andric using AsanAllocatorASVT = 2570b57cec5SDimitry Andric CombinedAllocator<PrimaryAllocatorASVT<AddressSpaceView>>; 2580b57cec5SDimitry Andric using AsanAllocator = AsanAllocatorASVT<LocalAddressSpaceView>; 2590b57cec5SDimitry Andric using AllocatorCache = AsanAllocator::AllocatorCache; 2600b57cec5SDimitry Andric 2610b57cec5SDimitry Andric struct AsanThreadLocalMallocStorage { 2620b57cec5SDimitry Andric uptr quarantine_cache[16]; 2630b57cec5SDimitry Andric AllocatorCache allocator_cache; 2640b57cec5SDimitry Andric void CommitBack(); 2650b57cec5SDimitry Andric private: 2660b57cec5SDimitry Andric // These objects are allocated via mmap() and are zero-initialized. 2670b57cec5SDimitry Andric AsanThreadLocalMallocStorage() {} 2680b57cec5SDimitry Andric }; 2690b57cec5SDimitry Andric 2700b57cec5SDimitry Andric void *asan_memalign(uptr alignment, uptr size, BufferedStackTrace *stack, 2710b57cec5SDimitry Andric AllocType alloc_type); 2720b57cec5SDimitry Andric void asan_free(void *ptr, BufferedStackTrace *stack, AllocType alloc_type); 2730b57cec5SDimitry Andric void asan_delete(void *ptr, uptr size, uptr alignment, 2740b57cec5SDimitry Andric BufferedStackTrace *stack, AllocType alloc_type); 2750b57cec5SDimitry Andric 2760b57cec5SDimitry Andric void *asan_malloc(uptr size, BufferedStackTrace *stack); 2770b57cec5SDimitry Andric void *asan_calloc(uptr nmemb, uptr size, BufferedStackTrace *stack); 2780b57cec5SDimitry Andric void *asan_realloc(void *p, uptr size, BufferedStackTrace *stack); 2790b57cec5SDimitry Andric void *asan_reallocarray(void *p, uptr nmemb, uptr size, 2800b57cec5SDimitry Andric BufferedStackTrace *stack); 2810b57cec5SDimitry Andric void *asan_valloc(uptr size, BufferedStackTrace *stack); 2820b57cec5SDimitry Andric void *asan_pvalloc(uptr size, BufferedStackTrace *stack); 2830b57cec5SDimitry Andric 2840b57cec5SDimitry Andric void *asan_aligned_alloc(uptr alignment, uptr size, BufferedStackTrace *stack); 2850b57cec5SDimitry Andric int asan_posix_memalign(void **memptr, uptr alignment, uptr size, 2860b57cec5SDimitry Andric BufferedStackTrace *stack); 2870b57cec5SDimitry Andric uptr asan_malloc_usable_size(const void *ptr, uptr pc, uptr bp); 2880b57cec5SDimitry Andric 2890b57cec5SDimitry Andric uptr asan_mz_size(const void *ptr); 2900b57cec5SDimitry Andric void asan_mz_force_lock(); 2910b57cec5SDimitry Andric void asan_mz_force_unlock(); 2920b57cec5SDimitry Andric 2930b57cec5SDimitry Andric void PrintInternalAllocatorStats(); 2940b57cec5SDimitry Andric void AsanSoftRssLimitExceededCallback(bool exceeded); 2950b57cec5SDimitry Andric 2960b57cec5SDimitry Andric } // namespace __asan 2970b57cec5SDimitry Andric #endif // ASAN_ALLOCATOR_H 298