| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234 | // (C) Copyright 2013 Ruslan Baratov// Copyright (C) 2014 Vicente J. Botet Escriba//// 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 www.boost.org/libs/thread for documentation.#ifndef BOOST_THREAD_WITH_LOCK_GUARD_HPP#define BOOST_THREAD_WITH_LOCK_GUARD_HPP#include <boost/thread/lock_guard.hpp>#include <boost/utility/result_of.hpp>//#include <boost/thread/detail/invoke.hpp>namespace boost {#if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && \    !defined(BOOST_NO_CXX11_DECLTYPE) && \    !defined(BOOST_NO_CXX11_TRAILING_RESULT_TYPES)/** * Utility to run functions in scope protected by mutex. * * Examples: * *     int func(int, int&); *     boost::mutex m; *     int a; *     int result = boost::with_lock_guard(m, func, 1, boost::ref(a)); * *     // using boost::bind *     int result = boost::with_lock_guard( *         m, boost::bind(func, 2, boost::ref(a)) *     ); * *     // using lambda *     int a; *     int result = boost::with_lock_guard( *         m, *         [&a](int x) { *           a = 3; *           return x + 4; *         }, *         5 *     ); */template <class Lockable, class Function, class... Args>typename boost::result_of<Function(Args...)>::type with_lock_guard(    Lockable& m,    BOOST_FWD_REF(Function) func,    BOOST_FWD_REF(Args)... args) //-> decltype(func(boost::forward<Args>(args)...)){  boost::lock_guard<Lockable> lock(m);  return func(boost::forward<Args>(args)...);}#else// Workaround versions for compilers without c++11 variadic templates support.// (function arguments limit: 4)// (for lambda support define BOOST_RESULT_OF_USE_DECLTYPE may be needed)template <class Lockable, class Func>typename boost::result_of<Func()>::type with_lock_guard(    Lockable& m,    BOOST_FWD_REF(Func) func) {  boost::lock_guard<Lockable> lock(m);  return func();}template <class Lockable, class Func, class Arg>typename boost::result_of<Func(Arg)>::type with_lock_guard(    Lockable& m,    BOOST_FWD_REF(Func) func,    BOOST_FWD_REF(Arg) arg) {  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg>(arg)  );}template <class Lockable, class Func, class Arg1, class Arg2>typename boost::result_of<Func(Arg1, Arg2)>::type with_lock_guard(    Lockable& m,    BOOST_FWD_REF(Func) func,    BOOST_FWD_REF(Arg1) arg1,    BOOST_FWD_REF(Arg2) arg2) {  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg1>(arg1),      boost::forward<Arg2>(arg2)  );}template <class Lockable, class Func, class Arg1, class Arg2, class Arg3>typename boost::result_of<Func(Arg1, Arg2, Arg3)>::type with_lock_guard(    Lockable& m,    BOOST_FWD_REF(Func) func,    BOOST_FWD_REF(Arg1) arg1,    BOOST_FWD_REF(Arg2) arg2,    BOOST_FWD_REF(Arg3) arg3) {  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg1>(arg1),      boost::forward<Arg2>(arg2),      boost::forward<Arg3>(arg3)  );}template <    class Lockable, class Func, class Arg1, class Arg2, class Arg3, class Arg4>typename boost::result_of<Func(Arg1, Arg2, Arg3, Arg4)>::type with_lock_guard(    Lockable& m,    BOOST_FWD_REF(Func) func,    BOOST_FWD_REF(Arg1) arg1,    BOOST_FWD_REF(Arg2) arg2,    BOOST_FWD_REF(Arg3) arg3,    BOOST_FWD_REF(Arg4) arg4) {  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg1>(arg1),      boost::forward<Arg2>(arg2),      boost::forward<Arg3>(arg3),      boost::forward<Arg4>(arg4)  );}// overloads for function pointer// (if argument is not function pointer, static assert will trigger)template <class Lockable, class Func>typename boost::result_of<    typename boost::add_pointer<Func>::type()>::type with_lock_guard(    Lockable& m,    Func* func) {  BOOST_STATIC_ASSERT(boost::is_function<Func>::value);  boost::lock_guard<Lockable> lock(m);  return func();}template <class Lockable, class Func, class Arg>typename boost::result_of<    typename boost::add_pointer<Func>::type(Arg)>::type with_lock_guard(    Lockable& m,    Func* func,    BOOST_FWD_REF(Arg) arg) {  BOOST_STATIC_ASSERT(boost::is_function<Func>::value);  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg>(arg)  );}template <class Lockable, class Func, class Arg1, class Arg2>typename boost::result_of<    typename boost::add_pointer<Func>::type(Arg1, Arg2)>::type with_lock_guard(    Lockable& m,    Func* func,    BOOST_FWD_REF(Arg1) arg1,    BOOST_FWD_REF(Arg2) arg2) {  BOOST_STATIC_ASSERT(boost::is_function<Func>::value);  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg1>(arg1),      boost::forward<Arg2>(arg2)  );}template <class Lockable, class Func, class Arg1, class Arg2, class Arg3>typename boost::result_of<    typename boost::add_pointer<Func>::type(Arg1, Arg2, Arg3)>::type with_lock_guard(    Lockable& m,    Func* func,    BOOST_FWD_REF(Arg1) arg1,    BOOST_FWD_REF(Arg2) arg2,    BOOST_FWD_REF(Arg3) arg3) {  BOOST_STATIC_ASSERT(boost::is_function<Func>::value);  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg1>(arg1),      boost::forward<Arg2>(arg2),      boost::forward<Arg3>(arg3)  );}template <    class Lockable, class Func, class Arg1, class Arg2, class Arg3, class Arg4>typename boost::result_of<    typename boost::add_pointer<Func>::type(Arg1, Arg2, Arg3, Arg4)>::type with_lock_guard(    Lockable& m,    Func* func,    BOOST_FWD_REF(Arg1) arg1,    BOOST_FWD_REF(Arg2) arg2,    BOOST_FWD_REF(Arg3) arg3,    BOOST_FWD_REF(Arg4) arg4) {  BOOST_STATIC_ASSERT(boost::is_function<Func>::value);  boost::lock_guard<Lockable> lock(m);  return func(      boost::forward<Arg1>(arg1),      boost::forward<Arg2>(arg2),      boost::forward<Arg3>(arg3),      boost::forward<Arg4>(arg4)  );}#endif} // namespace boost#endif // BOOST_THREAD_WITH_LOCK_GUARD_HPP
 |