/*============================================================================== Copyright (c) 2004 Peter Dimov Copyright (c) 2005-2010 Joel de Guzman Copyright (c) 2010 Thomas Heller 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 #if defined(BOOST_MSVC) #pragma warning(disable: 4786) // identifier truncated in debug info #pragma warning(disable: 4710) // function not inlined #pragma warning(disable: 4711) // function selected for automatic inline expansion #pragma warning(disable: 4514) // unreferenced inline removed #endif #include #include #if defined(BOOST_MSVC) && (BOOST_MSVC < 1300) #pragma warning(push, 3) #endif #include #if defined(BOOST_MSVC) && (BOOST_MSVC < 1300) #pragma warning(pop) #endif #include class X { private: mutable int state_; public: X(): state_(0) { } int state() const { return state_; } typedef int result_type; int operator()() const { return state_ += 17041; } int operator()(int x1) const { return state_ += x1; } int operator()(int x1, int x2) const { return state_ += x1+x2; } int operator()(int x1, int x2, int x3) const { return state_ += x1+x2+x3; } int operator()(int x1, int x2, int x3, int x4) const { return state_ += x1+x2+x3+x4; } int operator()(int x1, int x2, int x3, int x4, int x5) const { return state_ += x1+x2+x3+x4+x5; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6) const { return state_ += x1+x2+x3+x4+x5+x6; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6, int x7) const { return state_ += x1+x2+x3+x4+x5+x6+x7; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6, int x7, int x8) const { return state_ += x1+x2+x3+x4+x5+x6+x7+x8; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6, int x7, int x8, int x9) const { return state_ += x1+x2+x3+x4+x5+x6+x7+x8+x9; } }; class Y { private: int state_; public: Y(): state_(0) { } int state() const { return state_; } typedef int result_type; int operator()() { return state_ += 17041; } int operator()(int x1) { return state_ += x1; } int operator()(int x1, int x2) { return state_ += x1+x2; } int operator()(int x1, int x2, int x3) { return state_ += x1+x2+x3; } int operator()(int x1, int x2, int x3, int x4) { return state_ += x1+x2+x3+x4; } int operator()(int x1, int x2, int x3, int x4, int x5) { return state_ += x1+x2+x3+x4+x5; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6) { return state_ += x1+x2+x3+x4+x5+x6; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6, int x7) { return state_ += x1+x2+x3+x4+x5+x6+x7; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6, int x7, int x8) { return state_ += x1+x2+x3+x4+x5+x6+x7+x8; } int operator()(int x1, int x2, int x3, int x4, int x5, int x6, int x7, int x8, int x9) { return state_ += x1+x2+x3+x4+x5+x6+x7+x8+x9; } }; int f0(int & state_) { return state_ += 17041; } int f1(int & state_, int x1) { return state_ += x1; } int f2(int & state_, int x1, int x2) { return state_ += x1+x2; } int f3(int & state_, int x1, int x2, int x3) { return state_ += x1+x2+x3; } int f4(int & state_, int x1, int x2, int x3, int x4) { return state_ += x1+x2+x3+x4; } int f5(int & state_, int x1, int x2, int x3, int x4, int x5) { return state_ += x1+x2+x3+x4+x5; } int f6(int & state_, int x1, int x2, int x3, int x4, int x5, int x6) { return state_ += x1+x2+x3+x4+x5+x6; } int f7(int & state_, int x1, int x2, int x3, int x4, int x5, int x6, int x7) { return state_ += x1+x2+x3+x4+x5+x6+x7; } int f8(int & state_, int x1, int x2, int x3, int x4, int x5, int x6, int x7, int x8) { return state_ += x1+x2+x3+x4+x5+x6+x7+x8; } template struct wrap {}; template void test(F f, int a, int b) { BOOST_TEST( f() == a + b ); BOOST_TEST( f() == a + 2*b ); BOOST_TEST( f() == a + 3*b ); } using boost::phoenix::bind; using boost::phoenix::ref; void stateful_function_object_test() { ::test( bind( X() ), 0, 17041 ); ::test( bind( X(), 1 ), 0, 1 ); ::test( bind( X(), 1, 2 ), 0, 1+2 ); ::test( bind( X(), 1, 2, 3 ), 0, 1+2+3 ); ::test( bind( X(), 1, 2, 3, 4 ), 0, 1+2+3+4 ); ::test( bind( X(), 1, 2, 3, 4, 5 ), 0, 1+2+3+4+5 ); ::test( bind( X(), 1, 2, 3, 4, 5, 6 ), 0, 1+2+3+4+5+6 ); ::test( bind( X(), 1, 2, 3, 4, 5, 6, 7 ), 0, 1+2+3+4+5+6+7 ); ::test( bind( X(), 1, 2, 3, 4, 5, 6, 7, 8 ), 0, 1+2+3+4+5+6+7+8 ); ::test( bind( X(), 1, 2, 3, 4, 5, 6, 7, 8, 9 ), 0, 1+2+3+4+5+6+7+8+9 ); Y y; int n = y.state(); ::test( bind( ref(y) ), n, 17041 ); n += 3 * 17041; ::test( bind( ref(y), 1 ), n, 1 ); n += 3*1; ::test( bind( ref(y), 1, 2 ), n, 1+2 ); n += 3*(1+2); ::test( bind( ref(y), 1, 2, 3 ), n, 1+2+3 ); n += 3*(1+2+3); ::test( bind( ref(y), 1, 2, 3, 4 ), n, 1+2+3+4 ); n += 3*(1+2+3+4); ::test( bind( ref(y), 1, 2, 3, 4, 5 ), n, 1+2+3+4+5 ); n += 3*(1+2+3+4+5); ::test( bind( ref(y), 1, 2, 3, 4, 5, 6 ), n, 1+2+3+4+5+6 ); n += 3*(1+2+3+4+5+6); ::test( bind( ref(y), 1, 2, 3, 4, 5, 6, 7 ), n, 1+2+3+4+5+6+7 ); n += 3*(1+2+3+4+5+6+7); ::test( bind( ref(y), 1, 2, 3, 4, 5, 6, 7, 8 ), n, 1+2+3+4+5+6+7+8 ); n += 3*(1+2+3+4+5+6+7+8); ::test( bind( ref(y), 1, 2, 3, 4, 5, 6, 7, 8, 9 ), n, 1+2+3+4+5+6+7+8+9 ); n += 3*(1+2+3+4+5+6+7+8+9); BOOST_TEST( y.state() == n ); } void stateful_function_test() { using boost::phoenix::ref; ::test( bind( f0, 0), 0, 17041 ); ::test( bind( f1, 0, 1 ), 0, 1 ); ::test( bind( f2, 0, 1, 2 ), 0, 1+2 ); ::test( bind( f3, 0, 1, 2, 3 ), 0, 1+2+3 ); ::test( bind( f4, 0, 1, 2, 3, 4 ), 0, 1+2+3+4 ); ::test( bind( f5, 0, 1, 2, 3, 4, 5 ), 0, 1+2+3+4+5 ); ::test( bind( f6, 0, 1, 2, 3, 4, 5, 6 ), 0, 1+2+3+4+5+6 ); ::test( bind( f7, 0, 1, 2, 3, 4, 5, 6, 7 ), 0, 1+2+3+4+5+6+7 ); ::test( bind( f8, 0, 1, 2, 3, 4, 5, 6, 7, 8 ), 0, 1+2+3+4+5+6+7+8 ); } int main() { stateful_function_object_test(); stateful_function_test(); return boost::report_errors(); }