/*============================================================================= Copyright (c) 2007 Tobias Schwinger Use modification and distribution are subject to 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 #ifdef BOOST_MSVC # pragma warning(disable: 4244) // no conversion warnings, please #endif #include #include #include #include #include #include template class test_func : public Base { int val; public: test_func(int v) : val(v) { } template test_func(test_func const & that) : val(that.val) { } template friend class test_func; int operator()(int & l, int const & r) const { return l=r+val; } long operator()(int & l, int const & r) { return -(l=r+val); } char operator()(int& l, int& r) { return l=r+val; } template struct result { typedef void type; }; // ensure result_of argument types are what's expected // note: this is *not* how client code should look like template struct result< Self const(int&,int const&) > { typedef int type; }; template struct result< Self(int&,int const&) > { typedef long type; }; template struct result< Self(int&,int&) > { typedef char type; }; }; enum { int_, long_, char_ }; int type_of(int) { return int_; } int type_of(long) { return long_; } int type_of(char) { return char_; } int main() { { using boost::is_same; using boost::result_of; typedef boost::forward_adapter< test_func<> > f; // lvalue,rvalue BOOST_TEST(( is_same< result_of< f(int&, int) >::type, long >::value )); BOOST_TEST(( is_same< result_of< f const (int&, int) >::type, int >::value )); // lvalue,const lvalue BOOST_TEST(( is_same< result_of< f(int&, int const &) >::type, long >::value )); BOOST_TEST(( is_same< result_of< f const (int&, int const &) >::type, int >::value )); // lvalue,lvalue BOOST_TEST(( is_same< result_of< f(int&, int&) >::type, char >::value )); // result_of works differently for C++11 here, so compare // with using it against test_func. BOOST_TEST(( is_same< result_of< f const (int&, int&) >::type, result_of< test_func<> const (int&, int&)>::type >::value )); } { using boost::noncopyable; using boost::forward_adapter; int x = 0; test_func f(7); forward_adapter< test_func<> > func(f); forward_adapter< test_func & > func_ref(f); forward_adapter< test_func & > const func_ref_c(f); forward_adapter< test_func<> const > func_c(f); forward_adapter< test_func<> > const func_c2(f); forward_adapter< test_func const & > func_c_ref(f); BOOST_TEST( type_of( func(x,1) ) == long_ ); BOOST_TEST( type_of( func_ref(x,1) ) == long_ ); BOOST_TEST( type_of( func_ref_c(x,1) ) == long_ ); BOOST_TEST( type_of( func_c(x,1) ) == int_ ); BOOST_TEST( type_of( func_c2(x,1) ) == int_ ); BOOST_TEST( type_of( func_c_ref(x,1) ) == int_ ); BOOST_TEST( type_of( func(x,x) ) == char_ ); BOOST_TEST( func(x,1) == -8 ); BOOST_TEST( func_ref(x,1) == -8 ); BOOST_TEST( func_ref_c(x,1) == -8 ); BOOST_TEST( func_c(x,1) == 8 ); BOOST_TEST( func_c2(x,1) == 8 ); BOOST_TEST( func_c_ref(x,1) == 8 ); } return boost::report_errors(); }