// (C) Copyright 2013 Ruslan Baratov // // 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 #include namespace boost { #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) /** * 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 auto with_lock_guard( Lockable& m, BOOST_FWD_REF(Function) f, BOOST_FWD_REF(Args)... args ) -> decltype(f(boost::forward(args)...)) { boost::lock_guard lock(m); return f(boost::forward(args)...); } #else // Workaround versions for compilers without c++11 support. // (function arguments limit: 4, no lambda support) template typename boost::result_of::type with_lock_guard( Lockable& m, Func func ) { boost::lock_guard lock(m); return func(); } template typename boost::result_of::type with_lock_guard( Lockable& m, Func func, BOOST_FWD_REF(Arg) arg ) { boost::lock_guard lock(m); return func( boost::forward(arg) ); } template typename boost::result_of::type with_lock_guard( Lockable& m, Func func, BOOST_FWD_REF(Arg1) arg1, BOOST_FWD_REF(Arg2) arg2 ) { boost::lock_guard lock(m); return func( boost::forward(arg1), boost::forward(arg2) ); } template typename boost::result_of::type with_lock_guard( Lockable& m, Func func, BOOST_FWD_REF(Arg1) arg1, BOOST_FWD_REF(Arg2) arg2, BOOST_FWD_REF(Arg3) arg3 ) { boost::lock_guard lock(m); return func( boost::forward(arg1), boost::forward(arg2), boost::forward(arg3) ); } template < class Lockable, class Func, class Arg1, class Arg2, class Arg3, class Arg4 > typename boost::result_of::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::lock_guard lock(m); return func( boost::forward(arg1), boost::forward(arg2), boost::forward(arg3), boost::forward(arg4) ); } #endif // BOOST_NO_CXX11_VARIADIC_TEMPLATES } // namespace boost #endif // BOOST_THREAD_WITH_LOCK_GUARD_HPP