122 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| #ifndef BOOST_SMART_PTR_DETAIL_SPINLOCK_GCC_ARM_HPP_INCLUDED
 | |
| #define BOOST_SMART_PTR_DETAIL_SPINLOCK_GCC_ARM_HPP_INCLUDED
 | |
| 
 | |
| //
 | |
| //  Copyright (c) 2008, 2011 Peter Dimov
 | |
| //
 | |
| //  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)
 | |
| //
 | |
| 
 | |
| #include <boost/smart_ptr/detail/yield_k.hpp>
 | |
| 
 | |
| #if defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7EM__) || defined(__ARM_ARCH_7S__)
 | |
| 
 | |
| # define BOOST_SP_ARM_BARRIER "dmb"
 | |
| # define BOOST_SP_ARM_HAS_LDREX
 | |
| 
 | |
| #elif defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__)
 | |
| 
 | |
| # define BOOST_SP_ARM_BARRIER "mcr p15, 0, r0, c7, c10, 5"
 | |
| # define BOOST_SP_ARM_HAS_LDREX
 | |
| 
 | |
| #else
 | |
| 
 | |
| # define BOOST_SP_ARM_BARRIER ""
 | |
| 
 | |
| #endif
 | |
| 
 | |
| namespace boost
 | |
| {
 | |
| 
 | |
| namespace detail
 | |
| {
 | |
| 
 | |
| class spinlock
 | |
| {
 | |
| public:
 | |
| 
 | |
|     int v_;
 | |
| 
 | |
| public:
 | |
| 
 | |
|     bool try_lock()
 | |
|     {
 | |
|         int r;
 | |
| 
 | |
| #ifdef BOOST_SP_ARM_HAS_LDREX
 | |
| 
 | |
|         __asm__ __volatile__(
 | |
|             "ldrex %0, [%2]; \n"
 | |
|             "cmp %0, %1; \n"
 | |
|             "strexne %0, %1, [%2]; \n"
 | |
|             BOOST_SP_ARM_BARRIER :
 | |
|             "=&r"( r ): // outputs
 | |
|             "r"( 1 ), "r"( &v_ ): // inputs
 | |
|             "memory", "cc" );
 | |
| 
 | |
| #else
 | |
| 
 | |
|         __asm__ __volatile__(
 | |
|             "swp %0, %1, [%2];\n"
 | |
|             BOOST_SP_ARM_BARRIER :
 | |
|             "=&r"( r ): // outputs
 | |
|             "r"( 1 ), "r"( &v_ ): // inputs
 | |
|             "memory", "cc" );
 | |
| 
 | |
| #endif
 | |
| 
 | |
|         return r == 0;
 | |
|     }
 | |
| 
 | |
|     void lock()
 | |
|     {
 | |
|         for( unsigned k = 0; !try_lock(); ++k )
 | |
|         {
 | |
|             boost::detail::yield( k );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void unlock()
 | |
|     {
 | |
|         __asm__ __volatile__( BOOST_SP_ARM_BARRIER ::: "memory" );
 | |
|         *const_cast< int volatile* >( &v_ ) = 0;
 | |
|         __asm__ __volatile__( BOOST_SP_ARM_BARRIER ::: "memory" );
 | |
|     }
 | |
| 
 | |
| public:
 | |
| 
 | |
|     class scoped_lock
 | |
|     {
 | |
|     private:
 | |
| 
 | |
|         spinlock & sp_;
 | |
| 
 | |
|         scoped_lock( scoped_lock const & );
 | |
|         scoped_lock & operator=( scoped_lock const & );
 | |
| 
 | |
|     public:
 | |
| 
 | |
|         explicit scoped_lock( spinlock & sp ): sp_( sp )
 | |
|         {
 | |
|             sp.lock();
 | |
|         }
 | |
| 
 | |
|         ~scoped_lock()
 | |
|         {
 | |
|             sp_.unlock();
 | |
|         }
 | |
|     };
 | |
| };
 | |
| 
 | |
| } // namespace detail
 | |
| } // namespace boost
 | |
| 
 | |
| #define BOOST_DETAIL_SPINLOCK_INIT {0}
 | |
| 
 | |
| #undef BOOST_SP_ARM_BARRIER
 | |
| #undef BOOST_SP_ARM_HAS_LDREX
 | |
| 
 | |
| #endif // #ifndef BOOST_SMART_PTR_DETAIL_SPINLOCK_GCC_ARM_HPP_INCLUDED
 | 
