123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- /*
- * Copyright Andrey Semashev 2007 - 2015.
- * Distributed under 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)
- */
- /*!
- * \file once_block.hpp
- * \author Andrey Semashev
- * \date 23.06.2010
- *
- * \brief The header defines classes and macros for once-blocks.
- */
- #ifndef BOOST_LOG_UTILITY_ONCE_BLOCK_HPP_INCLUDED_
- #define BOOST_LOG_UTILITY_ONCE_BLOCK_HPP_INCLUDED_
- #include <boost/log/detail/config.hpp>
- #include <boost/log/utility/unique_identifier_name.hpp>
- #include <boost/log/detail/header.hpp>
- #ifdef BOOST_HAS_PRAGMA_ONCE
- #pragma once
- #endif
- #ifndef BOOST_LOG_NO_THREADS
- namespace boost {
- BOOST_LOG_OPEN_NAMESPACE
- /*!
- * \brief A flag to detect if a code block has already been executed.
- *
- * This structure should be used in conjunction with the \c BOOST_LOG_ONCE_BLOCK_FLAG
- * macro. Usage example:
- *
- * <code>
- * once_block_flag flag = BOOST_LOG_ONCE_BLOCK_INIT;
- *
- * void foo()
- * {
- * BOOST_LOG_ONCE_BLOCK_FLAG(flag)
- * {
- * puts("Hello, world once!");
- * }
- * }
- * </code>
- */
- struct once_block_flag
- {
- #ifndef BOOST_LOG_DOXYGEN_PASS
- // Do not use, implementation detail
- enum
- {
- uninitialized = 0, // this must be zero, so that zero-initialized once_block_flag is equivalent to the one initialized with uninitialized
- being_initialized,
- initialized
- };
- unsigned char status;
- #endif // BOOST_LOG_DOXYGEN_PASS
- };
- /*!
- * \def BOOST_LOG_ONCE_BLOCK_INIT
- *
- * The static initializer for \c once_block_flag.
- */
- #define BOOST_LOG_ONCE_BLOCK_INIT { boost::log::once_block_flag::uninitialized }
- namespace aux {
- class once_block_sentry
- {
- private:
- once_block_flag& m_flag;
- public:
- explicit once_block_sentry(once_block_flag& f) BOOST_NOEXCEPT : m_flag(f)
- {
- }
- ~once_block_sentry() BOOST_NOEXCEPT
- {
- if (BOOST_UNLIKELY(m_flag.status != once_block_flag::initialized))
- rollback();
- }
- bool executed() const BOOST_NOEXCEPT
- {
- return (m_flag.status == once_block_flag::initialized || enter_once_block());
- }
- BOOST_LOG_API void commit() BOOST_NOEXCEPT;
- private:
- BOOST_LOG_API bool enter_once_block() const BOOST_NOEXCEPT;
- BOOST_LOG_API void rollback() BOOST_NOEXCEPT;
- // Non-copyable, non-assignable
- BOOST_DELETED_FUNCTION(once_block_sentry(once_block_sentry const&))
- BOOST_DELETED_FUNCTION(once_block_sentry& operator= (once_block_sentry const&))
- };
- } // namespace aux
- BOOST_LOG_CLOSE_NAMESPACE // namespace log
- } // namespace boost
- #else // BOOST_LOG_NO_THREADS
- namespace boost {
- BOOST_LOG_OPEN_NAMESPACE
- struct once_block_flag
- {
- bool status;
- };
- #define BOOST_LOG_ONCE_BLOCK_INIT { false }
- namespace aux {
- class once_block_sentry
- {
- private:
- once_block_flag& m_flag;
- public:
- explicit once_block_sentry(once_block_flag& f) BOOST_NOEXCEPT : m_flag(f)
- {
- }
- bool executed() const BOOST_NOEXCEPT
- {
- return m_flag.status;
- }
- void commit() BOOST_NOEXCEPT
- {
- m_flag.status = true;
- }
- // Non-copyable, non-assignable
- BOOST_DELETED_FUNCTION(once_block_sentry(once_block_sentry const&))
- BOOST_DELETED_FUNCTION(once_block_sentry& operator= (once_block_sentry const&))
- };
- } // namespace aux
- BOOST_LOG_CLOSE_NAMESPACE // namespace log
- } // namespace boost
- #endif // BOOST_LOG_NO_THREADS
- #ifndef BOOST_LOG_DOXYGEN_PASS
- #define BOOST_LOG_ONCE_BLOCK_FLAG_INTERNAL(flag_var, sentry_var)\
- for (boost::log::aux::once_block_sentry sentry_var((flag_var));\
- BOOST_UNLIKELY(!sentry_var.executed()); sentry_var.commit())
- // NOTE: flag_var deliberately doesn't have an initializer so that it is zero-initialized at the static initialization stage
- #define BOOST_LOG_ONCE_BLOCK_INTERNAL(flag_var, sentry_var)\
- static boost::log::once_block_flag flag_var;\
- BOOST_LOG_ONCE_BLOCK_FLAG_INTERNAL(flag_var, sentry_var)
- #endif // BOOST_LOG_DOXYGEN_PASS
- /*!
- * \def BOOST_LOG_ONCE_BLOCK_FLAG(flag_var)
- *
- * Begins a code block to be executed only once, with protection against thread concurrency.
- * User has to provide the flag variable that controls whether the block has already
- * been executed.
- */
- #define BOOST_LOG_ONCE_BLOCK_FLAG(flag_var)\
- BOOST_LOG_ONCE_BLOCK_FLAG_INTERNAL(\
- flag_var,\
- BOOST_LOG_UNIQUE_IDENTIFIER_NAME(_boost_log_once_block_sentry_))
- /*!
- * \def BOOST_LOG_ONCE_BLOCK()
- *
- * Begins a code block to be executed only once, with protection against thread concurrency.
- */
- #define BOOST_LOG_ONCE_BLOCK()\
- BOOST_LOG_ONCE_BLOCK_INTERNAL(\
- BOOST_LOG_UNIQUE_IDENTIFIER_NAME(_boost_log_once_block_flag_),\
- BOOST_LOG_UNIQUE_IDENTIFIER_NAME(_boost_log_once_block_sentry_))
- #include <boost/log/detail/footer.hpp>
- #endif // BOOST_LOG_UTILITY_ONCE_BLOCK_HPP_INCLUDED_
|