Index: compiler-rt/lib/asan/asan_allocator.h =================================================================== --- compiler-rt/lib/asan/asan_allocator.h +++ compiler-rt/lib/asan/asan_allocator.h @@ -118,39 +118,75 @@ void OnUnmap(uptr p, uptr size) const; }; -#if SANITIZER_CAN_USE_ALLOCATOR64 +#if defined(__aarch64__) +// AArch64 supports 39, 42 and 48-bit VMA. +const uptr kAllocatorSpace = ~(uptr)0; +#if SANITIZER_ANDROID +const uptr kAllocatorSize = 0x2000000000ULL; // 128G. +typedef VeryCompactSizeClassMap SizeClassMap64; +#else +const uptr kAllocatorSize = 0x40000000000ULL; // 4T. +typedef DefaultSizeClassMap SizeClassMap64; +#endif + +template +struct AP64 { // Allocator64 parameters. Deliberately using a short name. + static const uptr kSpaceBeg = kAllocatorSpace; + static const uptr kSpaceSize = kAllocatorSize; + static const uptr kMetadataSize = 0; + typedef __asan::SizeClassMap64 SizeClassMap; + typedef AsanMapUnmapCallback MapUnmapCallback; + static const uptr kFlags = 0; + using AddressSpaceView = AddressSpaceViewTy; +}; +template +using Allocator64ASVT = SizeClassAllocator64>; +using Allocator64 = Allocator64ASVT; + +typedef CompactSizeClassMap SizeClassMap32; +template +struct AP32 { + static const uptr kSpaceBeg = 0; + static const u64 kSpaceSize = SANITIZER_MMAP_RANGE_SIZE; + static const uptr kMetadataSize = 16; + typedef __asan::SizeClassMap32 SizeClassMap; + static const uptr kRegionSizeLog = 20; + using AddressSpaceView = AddressSpaceViewTy; + typedef AsanMapUnmapCallback MapUnmapCallback; + static const uptr kFlags = 0; +}; +template +using Allocator32ASVT = SizeClassAllocator32>; +using Allocator32 = Allocator32ASVT; +using Allocator32or64 = DoubleAllocator; + +static const uptr kMaxNumberOfSizeClasses = + SizeClassMap32::kNumClasses < SizeClassMap64::kNumClasses + ? SizeClassMap64::kNumClasses + : SizeClassMap32::kNumClasses; + +template +using PrimaryAllocatorASVT = DoubleAllocator, + Allocator64ASVT>; +#elif SANITIZER_CAN_USE_ALLOCATOR64 # if SANITIZER_FUCHSIA const uptr kAllocatorSpace = ~(uptr)0; const uptr kAllocatorSize = 0x40000000000ULL; // 4T. -typedef DefaultSizeClassMap SizeClassMap; # elif defined(__powerpc64__) const uptr kAllocatorSpace = ~(uptr)0; const uptr kAllocatorSize = 0x20000000000ULL; // 2T. -typedef DefaultSizeClassMap SizeClassMap; -# elif defined(__aarch64__) && SANITIZER_ANDROID -// Android needs to support 39, 42 and 48 bit VMA. -const uptr kAllocatorSpace = ~(uptr)0; -const uptr kAllocatorSize = 0x2000000000ULL; // 128G. -typedef VeryCompactSizeClassMap SizeClassMap; -# elif defined(__aarch64__) -// AArch64/SANITIZER_CAN_USE_ALLOCATOR64 is only for 42-bit VMA -// so no need to different values for different VMA. -const uptr kAllocatorSpace = 0x10000000000ULL; -const uptr kAllocatorSize = 0x10000000000ULL; // 3T. -typedef DefaultSizeClassMap SizeClassMap; -#elif defined(__sparc__) +# elif defined(__sparc__) const uptr kAllocatorSpace = ~(uptr)0; const uptr kAllocatorSize = 0x20000000000ULL; // 2T. -typedef DefaultSizeClassMap SizeClassMap; # elif SANITIZER_WINDOWS const uptr kAllocatorSpace = ~(uptr)0; const uptr kAllocatorSize = 0x8000000000ULL; // 500G -typedef DefaultSizeClassMap SizeClassMap; # else const uptr kAllocatorSpace = 0x600000000000ULL; const uptr kAllocatorSize = 0x40000000000ULL; // 4T. -typedef DefaultSizeClassMap SizeClassMap; # endif +typedef DefaultSizeClassMap SizeClassMap; +static const uptr kMaxNumberOfSizeClasses = SizeClassMap::kNumClasses; template struct AP64 { // Allocator64 parameters. Deliberately using a short name. static const uptr kSpaceBeg = kAllocatorSpace; @@ -164,9 +200,9 @@ template using PrimaryAllocatorASVT = SizeClassAllocator64>; -using PrimaryAllocator = PrimaryAllocatorASVT; #else // Fallback to SizeClassAllocator32. typedef CompactSizeClassMap SizeClassMap; +static const uptr kMaxNumberOfSizeClasses = SizeClassMap::kNumClasses; template struct AP32 { static const uptr kSpaceBeg = 0; @@ -180,16 +216,14 @@ }; template using PrimaryAllocatorASVT = SizeClassAllocator32 >; -using PrimaryAllocator = PrimaryAllocatorASVT; #endif // SANITIZER_CAN_USE_ALLOCATOR64 -static const uptr kNumberOfSizeClasses = SizeClassMap::kNumClasses; - template using AsanAllocatorASVT = CombinedAllocator>; using AsanAllocator = AsanAllocatorASVT; using AllocatorCache = AsanAllocator::AllocatorCache; +using PrimaryAllocator = PrimaryAllocatorASVT; struct AsanThreadLocalMallocStorage { uptr quarantine_cache[16]; Index: compiler-rt/lib/asan/asan_allocator.cc =================================================================== --- compiler-rt/lib/asan/asan_allocator.cc +++ compiler-rt/lib/asan/asan_allocator.cc @@ -113,7 +113,7 @@ struct AsanChunk: ChunkBase { uptr Beg() { return reinterpret_cast(this) + kChunkHeaderSize; } uptr UsedSize(bool locked_version = false) { - if (user_requested_size != SizeClassMap::kMaxSize) + if (user_requested_size != get_allocator().KMaxSize()) return user_requested_size; return *reinterpret_cast( get_allocator().GetMetaData(AllocBeg(locked_version))); @@ -430,7 +430,7 @@ bool using_primary_allocator = true; // If we are allocating from the secondary allocator, there will be no // automatic right redzone, so add the right redzone manually. - if (!PrimaryAllocator::CanAllocate(needed_size, alignment)) { + if (!get_allocator().CanAllocate(needed_size, alignment)) { needed_size += rz_size; using_primary_allocator = false; } @@ -499,7 +499,7 @@ CHECK(allocator.FromPrimary(allocated)); } else { CHECK(!allocator.FromPrimary(allocated)); - m->user_requested_size = SizeClassMap::kMaxSize; + m->user_requested_size = get_allocator().KMaxSize(); uptr *meta = reinterpret_cast(allocator.GetMetaData(allocated)); meta[0] = size; meta[1] = chunk_beg; @@ -524,10 +524,10 @@ thread_stats.mallocs++; thread_stats.malloced += size; thread_stats.malloced_redzones += needed_size - size; - if (needed_size > SizeClassMap::kMaxSize) + if (needed_size > get_allocator().KMaxSize()) thread_stats.malloc_large++; else - thread_stats.malloced_by_size[SizeClassMap::ClassID(needed_size)]++; + thread_stats.malloced_by_size[get_allocator().ClassID(needed_size)]++; void *res = reinterpret_cast(user_beg); if (can_fill && fl.max_malloc_fill_size) { Index: compiler-rt/lib/asan/asan_stats.h =================================================================== --- compiler-rt/lib/asan/asan_stats.h +++ compiler-rt/lib/asan/asan_stats.h @@ -38,7 +38,7 @@ uptr munmaps; uptr munmaped; uptr malloc_large; - uptr malloced_by_size[kNumberOfSizeClasses]; + uptr malloced_by_size[kMaxNumberOfSizeClasses]; // Ctor for global AsanStats (accumulated stats for dead threads). explicit AsanStats(LinkerInitialized) { } Index: compiler-rt/lib/asan/asan_stats.cc =================================================================== --- compiler-rt/lib/asan/asan_stats.cc +++ compiler-rt/lib/asan/asan_stats.cc @@ -30,9 +30,9 @@ } static void PrintMallocStatsArray(const char *prefix, - uptr (&array)[kNumberOfSizeClasses]) { + uptr *array) { Printf("%s", prefix); - for (uptr i = 0; i < kNumberOfSizeClasses; i++) { + for (uptr i = 0; i < kMaxNumberOfSizeClasses; i++) { if (!array[i]) continue; Printf("%zu:%zu; ", i, array[i]); } Index: compiler-rt/lib/lsan/lsan_allocator.h =================================================================== --- compiler-rt/lib/lsan/lsan_allocator.h +++ compiler-rt/lib/lsan/lsan_allocator.h @@ -49,8 +49,45 @@ u32 stack_trace_id; }; -#if defined(__mips64) || defined(__aarch64__) || defined(__i386__) || \ - defined(__arm__) +#if defined(__aarch64__) +template +struct AP32 { + static const uptr kSpaceBeg = 0; + static const u64 kSpaceSize = SANITIZER_MMAP_RANGE_SIZE; + static const uptr kMetadataSize = sizeof(ChunkMetadata); + typedef __sanitizer::CompactSizeClassMap SizeClassMap; + static const uptr kRegionSizeLog = 20; + using AddressSpaceView = AddressSpaceViewTy; + typedef NoOpMapUnmapCallback MapUnmapCallback; + static const uptr kFlags = 0; +}; + +const uptr kAllocatorSpace = 0x600000000000ULL; +const uptr kAllocatorSize = 0x40000000000ULL; // 4T. + +template +struct AP64 { // Allocator64 parameters. Deliberately using a short name. + static const uptr kSpaceBeg = kAllocatorSpace; + static const uptr kSpaceSize = kAllocatorSize; + static const uptr kMetadataSize = sizeof(ChunkMetadata); + typedef DefaultSizeClassMap SizeClassMap; + typedef NoOpMapUnmapCallback MapUnmapCallback; + static const uptr kFlags = 0; + using AddressSpaceView = AddressSpaceViewTy; +}; + +template +using Allocator32ASVT = SizeClassAllocator32>; +template +using Allocator64ASVT = SizeClassAllocator64>; + +using Allocator32 = Allocator32ASVT; +using Allocator64 = Allocator64ASVT; + +template +using PrimaryAllocatorASVT = DoubleAllocator, + Allocator64ASVT>; +#elif defined(__mips64) || defined(__i386__) || defined(__arm__) template struct AP32 { static const uptr kSpaceBeg = 0; @@ -64,7 +101,6 @@ }; template using PrimaryAllocatorASVT = SizeClassAllocator32>; -using PrimaryAllocator = PrimaryAllocatorASVT; #elif defined(__x86_64__) || defined(__powerpc64__) # if defined(__powerpc64__) const uptr kAllocatorSpace = 0xa0000000000ULL; @@ -86,13 +122,13 @@ template using PrimaryAllocatorASVT = SizeClassAllocator64>; -using PrimaryAllocator = PrimaryAllocatorASVT; #endif template using AllocatorASVT = CombinedAllocator>; using Allocator = AllocatorASVT; using AllocatorCache = Allocator::AllocatorCache; +using PrimaryAllocator = PrimaryAllocatorASVT; Allocator::AllocatorCache *GetAllocatorCache(); Index: compiler-rt/lib/sanitizer_common/sanitizer_allocator.h =================================================================== --- compiler-rt/lib/sanitizer_common/sanitizer_allocator.h +++ compiler-rt/lib/sanitizer_common/sanitizer_allocator.h @@ -75,6 +75,7 @@ #include "sanitizer_allocator_local_cache.h" #include "sanitizer_allocator_secondary.h" #include "sanitizer_allocator_combined.h" +#include "sanitizer_double_allocator.h" } // namespace __sanitizer Index: compiler-rt/lib/sanitizer_common/sanitizer_allocator_combined.h =================================================================== --- compiler-rt/lib/sanitizer_common/sanitizer_allocator_combined.h +++ compiler-rt/lib/sanitizer_common/sanitizer_allocator_combined.h @@ -41,6 +41,10 @@ secondary_.Init(); } + bool CanAllocate(uptr size, uptr alignment) { + return primary_.CanAllocate(size, alignment); + } + void *Allocate(AllocatorCache *cache, uptr size, uptr alignment) { // Returning 0 on malloc(0) may break a lot of code. if (size == 0) @@ -194,6 +198,10 @@ secondary_.ForEachChunk(callback, arg); } + uptr KNumClasses() { return primary_.KNumClasses(); } + uptr KMaxSize() { return primary_.KMaxSize(); } + uptr ClassID(uptr size) { return primary_.ClassID(size); } + private: PrimaryAllocator primary_; SecondaryAllocator secondary_; Index: compiler-rt/lib/sanitizer_common/sanitizer_allocator_primary32.h =================================================================== --- compiler-rt/lib/sanitizer_common/sanitizer_allocator_primary32.h +++ compiler-rt/lib/sanitizer_common/sanitizer_allocator_primary32.h @@ -271,6 +271,9 @@ typedef SizeClassMap SizeClassMapT; static const uptr kNumClasses = SizeClassMap::kNumClasses; + static uptr KNumClasses() { return SizeClassMap::kNumClasses; } + static uptr KMaxSize() { return SizeClassMap::kMaxSize; } + private: static const uptr kRegionSize = 1 << kRegionSizeLog; static const uptr kNumPossibleRegions = kSpaceSize / kRegionSize; Index: compiler-rt/lib/sanitizer_common/sanitizer_allocator_primary64.h =================================================================== --- compiler-rt/lib/sanitizer_common/sanitizer_allocator_primary64.h +++ compiler-rt/lib/sanitizer_common/sanitizer_allocator_primary64.h @@ -319,6 +319,9 @@ static const uptr kNumClasses = SizeClassMap::kNumClasses; static const uptr kNumClassesRounded = SizeClassMap::kNumClassesRounded; + static uptr KNumClasses() { return SizeClassMap::kNumClasses; } + static uptr KMaxSize() { return SizeClassMap::kMaxSize; } + // A packed array of counters. Each counter occupies 2^n bits, enough to store // counter's max_value. Ctor will try to allocate the required buffer via // mapper->MapPackedCounterArrayBuffer and the caller is expected to check Index: compiler-rt/lib/sanitizer_common/sanitizer_double_allocator.h =================================================================== --- /dev/null +++ compiler-rt/lib/sanitizer_common/sanitizer_double_allocator.h @@ -0,0 +1,179 @@ +//===-- sanitizer_double_allocator.h ----------------------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// Select one of the allocators at runtime. +// +//===----------------------------------------------------------------------===// + +#ifndef SANITIZER_DOUBLEALLOCATOR_H +#define SANITIZER_DOUBLEALLOCATOR_H + +template +class DoubleAllocator { + first_ a1; + second_ a2; + + public: + bool use_first_; + + class DoubleAllocatorCache { + typename first_::AllocatorCache a1; + typename second_::AllocatorCache a2; + + public: + void Init(AllocatorGlobalStats *s) { + if (this->use_first_) + a1.Init(s); + else + a2.Init(s); + } + void *Allocate(DoubleAllocator *allocator, uptr class_id) { + if (allocator->use_first_) + return a1.Allocate(&allocator->a1, class_id); + return a2.Allocate(&allocator->a2, class_id); + } + + void Deallocate(DoubleAllocator *allocator, uptr class_id, void *p) { + if (allocator->use_first_) + a1.Deallocate(&allocator->a1, class_id, p); + else + a2.Deallocate(&allocator->a2, class_id, p); + } + + void Drain(DoubleAllocator *allocator) { + if (allocator->use_first_) + a1.Drain(&allocator->a1); + else + a2.Drain(&allocator->a2); + } + + void Destroy(DoubleAllocator *allocator, AllocatorGlobalStats *s) { + if (allocator->use_first_) + a1.Destroy(&allocator->a1, s); + else + a2.Destroy(&allocator->a2, s); + } + }; + + using MapUnmapCallback = typename first_::MapUnmapCallback; + using AddressSpaceView = typename first_::AddressSpaceView; + using AllocatorCache = DoubleAllocatorCache; + + void Init(s32 release_to_os_interval_ms) { + // Use the first allocator when the address + // space is too small for the 64-bit allocator. + use_first_ = GetMaxVirtualAddress() < (((uptr)1ULL << 48) - 1); + if (use_first_) + a1.Init(release_to_os_interval_ms); + else + a2.Init(release_to_os_interval_ms); + } + + bool CanAllocate(uptr size, uptr alignment) { + if (use_first_) + return first_::CanAllocate(size, alignment); + return second_::CanAllocate(size, alignment); + } + + uptr ClassID(uptr size) { + if (use_first_) + return first_::ClassID(size); + return second_::ClassID(size); + } + + uptr KNumClasses() { + if (use_first_) + return first_::KNumClasses(); + return second_::KNumClasses(); + } + + uptr KMaxSize() { + if (use_first_) + return first_::KMaxSize(); + return second_::KMaxSize(); + } + + bool PointerIsMine(const void *p) { + if (use_first_) + return a1.PointerIsMine(p); + return a2.PointerIsMine(p); + } + + void *GetMetaData(const void *p) { + if (use_first_) + return a1.GetMetaData(p); + return a2.GetMetaData(p); + } + + uptr GetSizeClass(const void *p) { + if (use_first_) + return a1.GetSizeClass(p); + return a2.GetSizeClass(p); + } + + void ForEachChunk(ForEachChunkCallback callback, void *arg) { + if (use_first_) + a1.ForEachChunk(callback, arg); + else + a2.ForEachChunk(callback, arg); + } + + void TestOnlyUnmap() { + if (use_first_) + a1.TestOnlyUnmap(); + else + a2.TestOnlyUnmap(); + } + void ForceLock() { + if (use_first_) + a1.ForceLock(); + else + a2.ForceLock(); + } + void ForceUnlock() { + if (use_first_) + a1.ForceUnlock(); + else + a2.ForceUnlock(); + } + void *GetBlockBegin(const void *p) { + if (use_first_) + return a1.GetBlockBegin(p); + return a2.GetBlockBegin(p); + } + uptr GetActuallyAllocatedSize(void *p) { + if (use_first_) + return a1.GetActuallyAllocatedSize(p); + return a2.GetActuallyAllocatedSize(p); + } + void SetReleaseToOSIntervalMs(s32 release_to_os_interval_ms) { + if (use_first_) + a1.SetReleaseToOSIntervalMs(release_to_os_interval_ms); + else + a2.SetReleaseToOSIntervalMs(release_to_os_interval_ms); + } + s32 ReleaseToOSIntervalMs() const { + if (use_first_) + return a1.ReleaseToOSIntervalMs(); + return a2.ReleaseToOSIntervalMs(); + } + void ForceReleaseToOS() { + if (use_first_) + a1.ForceReleaseToOS(); + else + a2.ForceReleaseToOS(); + } + void PrintStats() { + if (use_first_) + a1.PrintStats(); + else + a2.PrintStats(); + } +}; + +#endif // SANITIZER_DOUBLEALLOCATOR_H Index: compiler-rt/lib/sanitizer_common/tests/sanitizer_allocator_test.cc =================================================================== --- compiler-rt/lib/sanitizer_common/tests/sanitizer_allocator_test.cc +++ compiler-rt/lib/sanitizer_common/tests/sanitizer_allocator_test.cc @@ -160,6 +160,9 @@ SizeClassAllocator32>; using Allocator32Compact = Allocator32CompactASVT; +using Allocator32or64Compact = + DoubleAllocator; + template void TestSizeClassMap() { typedef SizeClassMap SCMap; @@ -274,6 +277,13 @@ TestSizeClassAllocator(); } +TEST(SanitizerCommon, SizeClassAllocator32or64Compact) { + Allocator32or64Compact::UseAllocator1 = false; + TestSizeClassAllocator(); + Allocator32or64Compact::UseAllocator1 = true; + TestSizeClassAllocator(); +} + TEST(SanitizerCommon, SizeClassAllocator64Dense) { TestSizeClassAllocator(); } @@ -357,6 +367,12 @@ TEST(SanitizerCommon, SizeClassAllocator64CompactMetadataStress) { SizeClassAllocatorMetadataStress(); } +TEST(SanitizerCommon, SizeClassAllocator32or64CompactMetadataStress) { + Allocator32or64Compact::UseAllocator1 = false; + SizeClassAllocatorMetadataStress(); + Allocator32or64Compact::UseAllocator1 = true; + SizeClassAllocatorMetadataStress(); +} #endif #endif @@ -404,6 +420,12 @@ TEST(SanitizerCommon, SizeClassAllocator64CompactGetBlockBegin) { SizeClassAllocatorGetBlockBeginStress(1ULL << 33); } +TEST(SanitizerCommon, SizeClassAllocator32or64CompactGetBlockBegin) { + Allocator32or64Compact::UseAllocator1 = false; + SizeClassAllocatorGetBlockBeginStress(1ULL << 33); + Allocator32or64Compact::UseAllocator1 = true; + SizeClassAllocatorGetBlockBeginStress(1ULL << 33); +} #endif TEST(SanitizerCommon, SizeClassAllocator64VeryCompactGetBlockBegin) { // Does not have > 4Gb for each class. @@ -694,6 +716,12 @@ TEST(SanitizerCommon, CombinedAllocator64Compact) { TestCombinedAllocator(); } +TEST(SanitizerCommon, CombinedDoubleAllocator) { + Allocator32or64Compact::UseAllocator1 = false; + TestCombinedAllocator(); + Allocator32or64Compact::UseAllocator1 = true; + TestCombinedAllocator(); +} #endif TEST(SanitizerCommon, CombinedAllocator64VeryCompact) { @@ -755,6 +783,12 @@ TEST(SanitizerCommon, SizeClassAllocator64CompactLocalCache) { TestSizeClassAllocatorLocalCache(); } +TEST(SanitizerCommon, SizeClassAllocator32or64CompactLocalCache) { + Allocator32or64Compact::UseAllocator1 = false; + TestSizeClassAllocatorLocalCache(); + Allocator32or64Compact::UseAllocator1 = true; + TestSizeClassAllocatorLocalCache(); +} #endif TEST(SanitizerCommon, SizeClassAllocator64VeryCompactLocalCache) { TestSizeClassAllocatorLocalCache(); @@ -1333,6 +1367,12 @@ TEST(SanitizerCommon, SizeClassAllocator64CompactReleaseFreeMemoryToOS) { TestReleaseFreeMemoryToOS(); } +TEST(SanitizerCommon, SizeClassAllocator32or64CompactReleaseFreeMemoryToOS) { + Allocator32or64Compact::UseAllocator1 = false; + TestReleaseFreeMemoryToOS(); + Allocator32or64Compact::UseAllocator1 = true; + TestReleaseFreeMemoryToOS(); +} TEST(SanitizerCommon, SizeClassAllocator64VeryCompactReleaseFreeMemoryToOS) { TestReleaseFreeMemoryToOS();