123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128 |
- /*! \file static_object.hpp
- \brief Internal polymorphism static object support
- \ingroup Internal */
- /*
- Copyright (c) 2014, Randolph Voorhies, Shane Grant
- All rights reserved.
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided that the following conditions are met:
- * Redistributions of source code must retain the above copyright
- notice, this list of conditions and the following disclaimer.
- * Redistributions in binary form must reproduce the above copyright
- notice, this list of conditions and the following disclaimer in the
- documentation and/or other materials provided with the distribution.
- * Neither the name of the copyright holder nor the
- names of its contributors may be used to endorse or promote products
- derived from this software without specific prior written permission.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
- ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY
- DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
- (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
- LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
- ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
- SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #ifndef CEREAL_DETAILS_STATIC_OBJECT_HPP_
- #define CEREAL_DETAILS_STATIC_OBJECT_HPP_
- #include "cereal/macros.hpp"
- #if CEREAL_THREAD_SAFE
- #include <mutex>
- #endif
- //! Prevent link optimization from removing non-referenced static objects
- /*! Especially for polymorphic support, we create static objects which
- may not ever be explicitly referenced. Most linkers will detect this
- and remove the code causing various unpleasant runtime errors. These
- macros, adopted from Boost (see force_include.hpp) prevent this
- (C) Copyright 2002 Robert Ramey - http://www.rrsd.com .
- Use, modification and distribution is subject to the Boost Software
- License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
- http://www.boost.org/LICENSE_1_0.txt) */
- #if defined(_MSC_VER) && !defined(__clang__)
- # define CEREAL_DLL_EXPORT __declspec(dllexport)
- # define CEREAL_USED
- #else // clang or gcc
- # define CEREAL_DLL_EXPORT __attribute__ ((visibility("default")))
- # define CEREAL_USED __attribute__ ((__used__))
- #endif
- namespace cereal
- {
- namespace detail
- {
- //! A static, pre-execution object
- /*! This class will create a single copy (singleton) of some
- type and ensures that merely referencing this type will
- cause it to be instantiated and initialized pre-execution.
- For example, this is used heavily in the polymorphic pointer
- serialization mechanisms to bind various archive types with
- different polymorphic classes */
- template <class T>
- class CEREAL_DLL_EXPORT StaticObject
- {
- private:
- static T & create()
- {
- static T t;
- //! Forces instantiation at pre-execution time
- (void)instance;
- return t;
- }
- StaticObject( StaticObject const & /*other*/ ) {}
- public:
- static T & getInstance()
- {
- return create();
- }
- //! A class that acts like std::lock_guard
- class LockGuard
- {
- #if CEREAL_THREAD_SAFE
- public:
- LockGuard(std::mutex & m) : lock(m) {}
- private:
- std::unique_lock<std::mutex> lock;
- #else
- public:
- LockGuard() = default;
- LockGuard(LockGuard const &) = default; // prevents implicit copy ctor warning
- ~LockGuard() CEREAL_NOEXCEPT {} // prevents variable not used
- #endif
- };
- //! Attempts to lock this static object for the current scope
- /*! @note This function is a no-op if cereal is not compiled with
- thread safety enabled (CEREAL_THREAD_SAFE = 1).
- This function returns an object that holds a lock for
- this StaticObject that will release its lock upon destruction. This
- call will block until the lock is available. */
- static LockGuard lock()
- {
- #if CEREAL_THREAD_SAFE
- static std::mutex instanceMutex;
- return LockGuard{instanceMutex};
- #else
- return LockGuard{};
- #endif
- }
- private:
- static T & instance;
- };
- template <class T> T & StaticObject<T>::instance = StaticObject<T>::create();
- } // namespace detail
- } // namespace cereal
- #endif // CEREAL_DETAILS_STATIC_OBJECT_HPP_
|