| // Copyright 2018 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #ifndef BASE_NO_DESTRUCTOR_H_ | 
 | #define BASE_NO_DESTRUCTOR_H_ | 
 |  | 
 | #include <new> | 
 | #include <utility> | 
 |  | 
 | namespace base { | 
 |  | 
 | // A wrapper that makes it easy to create an object of type T with static | 
 | // storage duration that: | 
 | // - is only constructed on first access | 
 | // - never invokes the destructor | 
 | // in order to satisfy the styleguide ban on global constructors and | 
 | // destructors. | 
 | // | 
 | // Runtime constant example: | 
 | // const std::string& GetLineSeparator() { | 
 | //  // Forwards to std::string(size_t, char, const Allocator&) constructor. | 
 | //   static const base::NoDestructor<std::string> s(5, '-'); | 
 | //   return *s; | 
 | // } | 
 | // | 
 | // More complex initialization with a lambda: | 
 | // const std::string& GetSessionNonce() { | 
 | //   static const base::NoDestructor<std::string> nonce([] { | 
 | //     std::string s(16); | 
 | //     crypto::RandString(s.data(), s.size()); | 
 | //     return s; | 
 | //   }()); | 
 | //   return *nonce; | 
 | // } | 
 | // | 
 | // NoDestructor<T> stores the object inline, so it also avoids a pointer | 
 | // indirection and a malloc. Also note that since C++11 static local variable | 
 | // initialization is thread-safe and so is this pattern. Code should prefer to | 
 | // use NoDestructor<T> over: | 
 | // - The CR_DEFINE_STATIC_LOCAL() helper macro. | 
 | // - A function scoped static T* or T& that is dynamically initialized. | 
 | // - A global base::LazyInstance<T>. | 
 | // | 
 | // Note that since the destructor is never run, this *will* leak memory if used | 
 | // as a stack or member variable. Furthermore, a NoDestructor<T> should never | 
 | // have global scope as that may require a static initializer. | 
 | template <typename T> | 
 | class NoDestructor { | 
 |  public: | 
 |   // Not constexpr; just write static constexpr T x = ...; if the value should | 
 |   // be a constexpr. | 
 |   template <typename... Args> | 
 |   explicit NoDestructor(Args&&... args) { | 
 |     new (storage_) T(std::forward<Args>(args)...); | 
 |   } | 
 |  | 
 |   // Allows copy and move construction of the contained type, to allow | 
 |   // construction from an initializer list, e.g. for std::vector. | 
 |   explicit NoDestructor(const T& x) { new (storage_) T(x); } | 
 |   explicit NoDestructor(T&& x) { new (storage_) T(std::move(x)); } | 
 | #if defined(STARBOARD) | 
 |   NoDestructor() { new (storage_) T(); } | 
 | #endif | 
 |  | 
 |   NoDestructor(const NoDestructor&) = delete; | 
 |   NoDestructor& operator=(const NoDestructor&) = delete; | 
 |  | 
 |   ~NoDestructor() = default; | 
 |  | 
 |   const T& operator*() const { return *get(); } | 
 |   T& operator*() { return *get(); } | 
 |  | 
 |   const T* operator->() const { return get(); } | 
 |   T* operator->() { return get(); } | 
 |  | 
 |   const T* get() const { return reinterpret_cast<const T*>(storage_); } | 
 |   T* get() { return reinterpret_cast<T*>(storage_); } | 
 |  | 
 |  private: | 
 |   alignas(T) char storage_[sizeof(T)]; | 
 |  | 
 | #if defined(LEAK_SANITIZER) | 
 |   // TODO(https://crbug.com/812277): This is a hack to work around the fact | 
 |   // that LSan doesn't seem to treat NoDestructor as a root for reachability | 
 |   // analysis. This means that code like this: | 
 |   //   static base::NoDestructor<std::vector<int>> v({1, 2, 3}); | 
 |   // is considered a leak. Using the standard leak sanitizer annotations to | 
 |   // suppress leaks doesn't work: std::vector is implicitly constructed before | 
 |   // calling the base::NoDestructor constructor. | 
 |   // | 
 |   // Unfortunately, I haven't been able to demonstrate this issue in simpler | 
 |   // reproductions: until that's resolved, hold an explicit pointer to the | 
 |   // placement-new'd object in leak sanitizer mode to help LSan realize that | 
 |   // objects allocated by the contained type are still reachable. | 
 |   T* storage_ptr_ = reinterpret_cast<T*>(storage_); | 
 | #endif  // defined(LEAK_SANITIZER) | 
 | }; | 
 |  | 
 | }  // namespace base | 
 |  | 
 | #endif  // BASE_NO_DESTRUCTOR_H_ |