/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include <array> #include <atomic> #include <memory> #include <folly/CppAttributes.h> #include <folly/Portability.h> #include <folly/Unit.h> #include <folly/concurrency/CacheLocality.h> #include <folly/synchronization/Hazptr.h> namespace folly { // On mobile we do not expect high concurrency, and memory is more important, so // use more conservative caching. constexpr size_t kCoreCachedSharedPtrDefaultMaxSlots = …; namespace core_cached_shared_ptr_detail { template <size_t kMaxSlots> class SlotsConfig { … }; // Initialize with a valid num so that get() always returns a valid stripe, even // if initialize() has not been called yet. template <size_t kMaxSlots> std::atomic<size_t> SlotsConfig<kMaxSlots>::num_{ … }; template <size_t kMaxSlots, class T> void makeSlots(std::shared_ptr<T> p, folly::Range<std::shared_ptr<T>*> slots) { … } // Check whether a shared_ptr is equivalent to default-constructed. Because of // aliasing constructors, there can be both nullptr with a managed object, and // non-nullptr with no managed object, so we need to check both. template <class T> bool isDefault(const std::shared_ptr<T>& p) { … } } // namespace core_cached_shared_ptr_detail /** * This class creates core-local caches for a given shared_ptr, to * mitigate contention when acquiring/releasing it. * * It has the same thread-safety guarantees as shared_ptr: it is safe * to concurrently call get(), but reset()s must be synchronized with * reads and other reset()s. */ template <class T, size_t kMaxSlots = kCoreCachedSharedPtrDefaultMaxSlots> class CoreCachedSharedPtr { … }; template <class T, size_t kMaxSlots = kCoreCachedSharedPtrDefaultMaxSlots> class CoreCachedWeakPtr { … }; /** * This class creates core-local caches for a given shared_ptr, to * mitigate contention when acquiring/releasing it. * * All methods are threadsafe. Hazard pointers are used to avoid * use-after-free for concurrent reset() and get() operations. * * Concurrent reset()s are sequenced with respect to each other: the * sharded shared_ptrs will always all be set to the same value. * get()s will never see a newer pointer on one core, and an older * pointer on another after a subsequent thread migration. */ template <class T, size_t kMaxSlots = kCoreCachedSharedPtrDefaultMaxSlots> class AtomicCoreCachedSharedPtr { … }; } // namespace folly