From 5350794be2d2eb931ad75d3948944f0d30494c7a Mon Sep 17 00:00:00 2001 From: klzgrad Date: Sat, 8 Feb 2025 19:57:01 +0800 Subject: [PATCH] musl: allocator: Avoid deadlock in pthread_atfork Musl 1.2.3 and before call malloc() in pthread_atfork(), which may result in a deadlock: PartitionRoot::EnableThreadCacheIfSupported() ::partition_alloc::internal::ScopedGuard guard{lock_}; ThreadCache::Create(this); ThreadCache::ThreadCache() PlatformThread::CurrentId() InitAtFork::InitAtFork() pthread_atfork() malloc() ShimMalloc() PartitionAllocFunctionsInternal::Malloc() PartitionRoot::AllocInternal() PartitionRoot::AllocInternalNoHooks() PartitionRoot::RawAlloc() ::partition_alloc::internal::ScopedGuard guard{internal::PartitionRootLock(this)}; --- .../threading/platform_thread_posix.cc | 19 ++++++++++++++++++ .../src/partition_alloc/partition_root.cc | 20 +++++++++++++++++++ 2 files changed, 39 insertions(+) diff --git a/src/base/allocator/partition_allocator/src/partition_alloc/partition_alloc_base/threading/platform_thread_posix.cc b/src/base/allocator/partition_allocator/src/partition_alloc/partition_alloc_base/threading/platform_thread_posix.cc index 47e4368908..c85a0610fc 100644 --- a/src/base/allocator/partition_allocator/src/partition_alloc/partition_alloc_base/threading/platform_thread_posix.cc +++ b/src/base/allocator/partition_allocator/src/partition_alloc/partition_alloc_base/threading/platform_thread_posix.cc @@ -28,6 +28,10 @@ #include #endif +#if defined(__MUSL__) +#include "partition_alloc/shim/allocator_shim.h" +#endif + namespace partition_alloc::internal::base { #if PA_BUILDFLAG(IS_LINUX) || PA_BUILDFLAG(IS_CHROMEOS) @@ -59,7 +63,22 @@ thread_local bool g_is_main_thread = true; class InitAtFork { public: InitAtFork() { +#if defined(__MUSL__) + allocator_shim::AllocatorDispatch d = + *allocator_shim::GetAllocatorDispatchChainHeadForTesting(); + d.alloc_function = +[](size_t size, void*) -> void* { + // The size of the scratch fits struct atfork_funcs in Musl pthread_atfork.c. + static char scratch[5 * sizeof(void*)]; + return size != sizeof(scratch) ? nullptr : scratch; + }; + allocator_shim::InsertAllocatorDispatch(&d); +#endif + pthread_atfork(nullptr, nullptr, internal::InvalidateTidCache); + +#if defined(__MUSL__) + allocator_shim::RemoveAllocatorDispatchForTesting(&d); +#endif } }; diff --git a/src/base/allocator/partition_allocator/src/partition_alloc/partition_root.cc b/src/base/allocator/partition_allocator/src/partition_alloc/partition_root.cc index a025fd2dc4..6955c82cd4 100644 --- a/src/base/allocator/partition_allocator/src/partition_alloc/partition_root.cc +++ b/src/base/allocator/partition_allocator/src/partition_alloc/partition_root.cc @@ -51,6 +51,10 @@ #endif // PA_CONFIG(ENABLE_SHADOW_METADATA) #endif // PA_BUILDFLAG(IS_LINUX) || PA_BUILDFLAG(IS_CHROMEOS) +#if defined(__MUSL__) +#include "partition_alloc/shim/allocator_shim.h" +#endif + namespace partition_alloc::internal { #if PA_BUILDFLAG(RECORD_ALLOC_INFO) @@ -319,9 +323,25 @@ void PartitionAllocMallocInitOnce() { // However, no perfect solution really exists to make threads + fork() // cooperate, but deadlocks are real (and fork() is used in DEATH_TEST()s), // and other malloc() implementations use the same techniques. + +#if defined(__MUSL__) + allocator_shim::AllocatorDispatch d = + *allocator_shim::GetAllocatorDispatchChainHeadForTesting(); + d.alloc_function = +[](size_t size, void*) -> void* { + // The size of the scratch fits struct atfork_funcs in Musl pthread_atfork.c. + static char scratch[5 * sizeof(void*)]; + return size != sizeof(scratch) ? nullptr : scratch; + }; + allocator_shim::InsertAllocatorDispatch(&d); +#endif + int err = pthread_atfork(BeforeForkInParent, AfterForkInParent, AfterForkInChild); PA_CHECK(err == 0); + +#if defined(__MUSL__) + allocator_shim::RemoveAllocatorDispatchForTesting(&d); +#endif #endif // PA_BUILDFLAG(IS_LINUX) || PA_BUILDFLAG(IS_CHROMEOS) }