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
|