123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218 |
- /*
- * Copyright Andrey Semashev 2007 - 2013.
- * 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 counter.hpp
- * \author Andrey Semashev
- * \date 01.05.2007
- *
- * The header contains implementation of the counter attribute.
- */
- #ifndef BOOST_LOG_ATTRIBUTES_COUNTER_HPP_INCLUDED_
- #define BOOST_LOG_ATTRIBUTES_COUNTER_HPP_INCLUDED_
- #include <boost/static_assert.hpp>
- #include <boost/type_traits/is_integral.hpp>
- #include <boost/log/detail/config.hpp>
- #include <boost/log/attributes/attribute.hpp>
- #include <boost/log/attributes/attribute_cast.hpp>
- #include <boost/log/attributes/attribute_value_impl.hpp>
- #ifndef BOOST_LOG_NO_THREADS
- #include <boost/detail/atomic_count.hpp>
- #endif // BOOST_LOG_NO_THREADS
- #include <boost/log/detail/header.hpp>
- #ifdef BOOST_HAS_PRAGMA_ONCE
- #pragma once
- #endif
- namespace boost {
- BOOST_LOG_OPEN_NAMESPACE
- namespace attributes {
- /*!
- * \brief A class of an attribute that counts an integral value
- *
- * This type of attribute acts as a counter, that is, it returns a monotonously
- * changing value each time requested. The attribute value type can be specified
- * as a template parameter. However, the type must be an integral type of size no
- * more than <tt>sizeof(long)</tt>.
- */
- template< typename T >
- class counter :
- public attribute
- {
- // For now only integral types up to long are supported
- BOOST_STATIC_ASSERT_MSG(is_integral< T >::value && sizeof(T) <= sizeof(long), "Boost.Log: Only integral types up to long are supported by counter attribute");
- public:
- //! A counter value type
- typedef T value_type;
- protected:
- //! Base class for factory implementation
- class BOOST_LOG_NO_VTABLE BOOST_SYMBOL_VISIBLE impl :
- public attribute::impl
- {
- };
- //! Generic factory implementation
- class impl_generic;
- #ifndef BOOST_LOG_NO_THREADS
- //! Increment-by-one factory implementation
- class impl_inc;
- //! Decrement-by-one factory implementation
- class impl_dec;
- #endif
- public:
- /*!
- * Constructor
- *
- * \param initial Initial value of the counter
- * \param step Changing step of the counter. Each value acquired from the attribute
- * will be greater than the previous one to this amount.
- */
- explicit counter(value_type initial = (value_type)0, long step = 1) :
- #ifndef BOOST_LOG_NO_THREADS
- attribute()
- {
- if (step == 1)
- this->set_impl(new impl_inc(initial));
- else if (step == -1)
- this->set_impl(new impl_dec(initial));
- else
- this->set_impl(new impl_generic(initial, step));
- }
- #else
- attribute(new impl_generic(initial, step))
- {
- }
- #endif
- /*!
- * Constructor for casting support
- */
- explicit counter(cast_source const& source) :
- attribute(source.as< impl >())
- {
- }
- };
- #ifndef BOOST_LOG_NO_THREADS
- template< typename T >
- class counter< T >::impl_generic :
- public impl
- {
- private:
- //! Initial value
- const value_type m_Initial;
- //! Step value
- const long m_Step;
- //! The counter
- boost::detail::atomic_count m_Counter;
- public:
- /*!
- * Initializing constructor
- */
- impl_generic(value_type initial, long step) : m_Initial(initial), m_Step(step), m_Counter(-1)
- {
- }
- attribute_value get_value()
- {
- register unsigned long next_counter = static_cast< unsigned long >(++m_Counter);
- register value_type next = static_cast< value_type >(m_Initial + (next_counter * m_Step));
- return make_attribute_value(next);
- }
- };
- template< typename T >
- class counter< T >::impl_inc :
- public impl
- {
- private:
- //! The counter
- boost::detail::atomic_count m_Counter;
- public:
- /*!
- * Initializing constructor
- */
- explicit impl_inc(value_type initial) : m_Counter(initial - 1)
- {
- }
- attribute_value get_value()
- {
- return make_attribute_value(static_cast< value_type >(++m_Counter));
- }
- };
- template< typename T >
- class counter< T >::impl_dec :
- public impl
- {
- private:
- //! The counter
- boost::detail::atomic_count m_Counter;
- public:
- /*!
- * Initializing constructor
- */
- explicit impl_dec(value_type initial) : m_Counter(initial + 1)
- {
- }
- attribute_value get_value()
- {
- return make_attribute_value(static_cast< value_type >(--m_Counter));
- }
- };
- #else // BOOST_LOG_NO_THREADS
- template< typename T >
- class counter< T >::impl_generic :
- public impl
- {
- private:
- //! Step value
- const long m_Step;
- //! The counter
- value_type m_Counter;
- public:
- /*!
- * Initializing constructor
- */
- impl_generic(value_type initial, long step) : m_Step(step), m_Counter(initial - step)
- {
- }
- attribute_value get_value()
- {
- m_Counter += m_Step;
- return make_attribute_value(m_Counter);
- }
- };
- #endif // BOOST_LOG_NO_THREADS
- } // namespace attributes
- BOOST_LOG_CLOSE_NAMESPACE // namespace log
- } // namespace boost
- #include <boost/log/detail/footer.hpp>
- #endif // BOOST_LOG_ATTRIBUTES_COUNTER_HPP_INCLUDED_
|