////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Peter Dimov 2008. // (C) Copyright Ion Gaztanaga 2013-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) // // See http://www.boost.org/libs/interprocess for documentation. // ////////////////////////////////////////////////////////////////////////////// //Parts of this file come from boost/smart_ptr/detail/yield_k.hpp //Many thanks to Peter Dimov. #ifndef BOOST_INTERPROCESS_SYNC_WAIT_HPP_INCLUDED #define BOOST_INTERPROCESS_SYNC_WAIT_HPP_INCLUDED #ifndef BOOST_CONFIG_HPP # include #endif # #if defined(BOOST_HAS_PRAGMA_ONCE) # pragma once #endif #include #include #include //#define BOOST_INTERPROCESS_SPIN_WAIT_DEBUG #ifdef BOOST_INTERPROCESS_SPIN_WAIT_DEBUG #include #endif // BOOST_INTERPROCESS_SMT_PAUSE #if defined(_MSC_VER) && ( defined(_M_IX86) || defined(_M_X64) ) extern "C" void _mm_pause(); #pragma intrinsic( _mm_pause ) #define BOOST_INTERPROCESS_SMT_PAUSE _mm_pause(); #elif defined(__GNUC__) && ( defined(__i386__) || defined(__x86_64__) ) && !defined(_CRAYC) #define BOOST_INTERPROCESS_SMT_PAUSE __asm__ __volatile__( "rep; nop" : : : "memory" ); #endif namespace boost{ namespace interprocess{ namespace ipcdetail { template class num_core_holder { public: static unsigned int get() { if(!num_cores){ return ipcdetail::get_num_cores(); } else{ return num_cores; } } private: static unsigned int num_cores; }; template unsigned int num_core_holder::num_cores = ipcdetail::get_num_cores(); } //namespace ipcdetail { class spin_wait { public: static const unsigned int nop_pause_limit = 32u; spin_wait() : m_count_start(), m_ul_yield_only_counts(), m_k() {} #ifdef BOOST_INTERPROCESS_SPIN_WAIT_DEBUG ~spin_wait() { if(m_k){ std::cout << "final m_k: " << m_k << " system tick(us): " << ipcdetail::get_system_tick_us() << std::endl; } } #endif unsigned int count() const { return m_k; } void yield() { //Lazy initialization of limits if( !m_k){ this->init_limits(); } //Nop tries if( m_k < (nop_pause_limit >> 2) ){ } //Pause tries if the processor supports it #if defined(BOOST_INTERPROCESS_SMT_PAUSE) else if( m_k < nop_pause_limit ){ BOOST_INTERPROCESS_SMT_PAUSE } #endif //Yield/Sleep strategy else{ //Lazy initialization of tick information if(m_k == nop_pause_limit){ this->init_tick_info(); } else if( this->yield_or_sleep() ){ ipcdetail::thread_yield(); } else{ ipcdetail::thread_sleep_tick(); } } ++m_k; } void reset() { m_k = 0u; } private: void init_limits() { unsigned int num_cores = ipcdetail::num_core_holder<0>::get(); m_k = num_cores > 1u ? 0u : nop_pause_limit; } void init_tick_info() { m_ul_yield_only_counts = ipcdetail::get_system_tick_in_highres_counts(); m_count_start = ipcdetail::get_current_system_highres_count(); } //Returns true if yield must be called, false is sleep must be called bool yield_or_sleep() { if(!m_ul_yield_only_counts){ //If yield-only limit was reached then yield one in every two tries return (m_k & 1u) != 0; } else{ //Try to see if we've reched yield-only time limit const ipcdetail::OS_highres_count_t now = ipcdetail::get_current_system_highres_count(); const ipcdetail::OS_highres_count_t elapsed = ipcdetail::system_highres_count_subtract(now, m_count_start); if(!ipcdetail::system_highres_count_less_ul(elapsed, m_ul_yield_only_counts)){ #ifdef BOOST_INTERPROCESS_SPIN_WAIT_DEBUG std::cout << "elapsed!\n" << " m_ul_yield_only_counts: " << m_ul_yield_only_counts << " system tick(us): " << ipcdetail::get_system_tick_us() << '\n' << " m_k: " << m_k << " elapsed counts: "; ipcdetail::ostream_highres_count(std::cout, elapsed) << std::endl; #endif //Yield-only time reached, now it's time to sleep m_ul_yield_only_counts = 0ul; return false; } } return true; //Otherwise yield } ipcdetail::OS_highres_count_t m_count_start; unsigned long m_ul_yield_only_counts; unsigned int m_k; }; } // namespace interprocess } // namespace boost #include #endif // #ifndef BOOST_INTERPROCESS_SYNC_WAIT_HPP_INCLUDED