// // Copyright (c) 2000,2001,2002 Brian McNamara and Yannis Smaragdakis // // Permission to use, copy, modify, distribute and sell this software // and its documentation for any purpose is granted without fee, // provided that the above copyright notice and this permission notice // appear in all source code copies and supporting documentation. The // software is provided "as is" without any express or implied // warranty. #ifndef FCPP_REUSE_DOT_H #define FCPP_REUSE_DOT_H #include "function.h" namespace fcpp { // This is really a "list.h" thing, but Reusers borrow NIL to do some of // their dirty work. struct AUniqueTypeForNil { bool operator==( AUniqueTypeForNil ) const { return true; } bool operator< ( AUniqueTypeForNil ) const { return false; } }; FCPP_MAYBE_NAMESPACE_OPEN FCPP_MAYBE_EXTERN AUniqueTypeForNil NIL; FCPP_MAYBE_NAMESPACE_CLOSE ////////////////////////////////////////////////////////////////////// // "Reuser"s are effectively special-purpose versions of curry() that // enable recursive list functoids to reuse the thunk of the curried // recursive call. See // http://www.cc.gatech.edu/~yannis/fc++/New/reusers.html // for a more detailed description. ////////////////////////////////////////////////////////////////////// // For efficiency, we mark parameters as either "Var"iant or "Inv"ariant. struct Inv {}; struct Var {}; template struct Maybe_Var_Inv; template struct Maybe_Var_Inv { static void remake( X& x, const X& val ) { x.~X(); new (&x) X(val); } static X clone( const X& x ) { return X(x); } }; template struct Maybe_Var_Inv { static void remake( X&, const X& ) {} static const X& clone( const X& x ) { return x; } }; ////////////////////////////////////////////////////////////////////// template struct Reuser0; template struct Thunk0 : public Fun0Impl { mutable F f; Thunk0( const F& ff ) : f(ff) {} void init( const F& ff ) const { Maybe_Var_Inv::remake( f, ff ); } R operator()() const { return Maybe_Var_Inv::clone(f)( Reuser0(this) ); } }; template struct Reuser0 { typedef typename RT::ResultType R; typedef Thunk0 M; IRef ref; Reuser0(AUniqueTypeForNil) {} Reuser0(const M* m) : ref(m) {} Fun0 operator()( const F& f ) { if( !ref ) ref = IRef( new M(f) ); else ref->init(f); return Fun0( 1, ref ); } void iter( const F& f ) { if( ref ) ref->init(f); } }; ////////////////////////////////////////////////////////////////////// template struct Reuser1; template struct Thunk1 : public Fun0Impl { mutable F f; mutable X x; Thunk1( const F& ff, const X& xx ) : f(ff), x(xx) {} void init( const F& ff, const X& xx ) const { Maybe_Var_Inv::remake( f, ff ); Maybe_Var_Inv::remake( x, xx ); } R operator()() const { return Maybe_Var_Inv::clone(f)( Maybe_Var_Inv::clone(x), Reuser1(this) ); } }; template struct Reuser1 { typedef typename RT::ResultType R; typedef Thunk1 M; IRef ref; Reuser1(AUniqueTypeForNil) {} Reuser1(const M* m) : ref(m) {} Fun0 operator()( const F& f, const X& x ) { if( !ref ) ref = IRef( new M(f,x) ); else ref->init(f,x); return Fun0( 1, ref ); } void iter( const F& f, const X& x ) { if( ref ) ref->init(f,x); } }; ////////////////////////////////////////////////////////////////////// template struct Reuser2; template struct Thunk2 : public Fun0Impl { mutable F f; mutable X x; mutable Y y; Thunk2( const F& ff, const X& xx, const Y& yy ) : f(ff), x(xx), y(yy) {} void init( const F& ff, const X& xx, const Y& yy ) const { Maybe_Var_Inv::remake( f, ff ); Maybe_Var_Inv::remake( x, xx ); Maybe_Var_Inv::remake( y, yy ); } R operator()() const { return Maybe_Var_Inv::clone(f)( Maybe_Var_Inv::clone(x), Maybe_Var_Inv::clone(y), Reuser2(this) ); } }; template struct Reuser2 { typedef typename RT::ResultType R; typedef Thunk2 M; IRef ref; Reuser2(AUniqueTypeForNil) {} Reuser2(const M* m) : ref(m) {} Fun0 operator()( const F& f, const X& x, const Y& y ) { if( !ref ) ref = IRef( new M(f,x,y) ); else ref->init(f,x,y); return Fun0( 1, ref ); } void iter( const F& f, const X& x, const Y& y ) { if( ref ) ref->init(f,x,y); } }; ////////////////////////////////////////////////////////////////////// template struct Reuser3; template struct Thunk3 : public Fun0Impl { mutable F f; mutable X x; mutable Y y; mutable Z z; Thunk3( const F& ff, const X& xx, const Y& yy, const Z& zz ) : f(ff), x(xx), y(yy), z(zz) {} void init( const F& ff, const X& xx, const Y& yy, const Z& zz ) const { Maybe_Var_Inv::remake( f, ff ); Maybe_Var_Inv::remake( x, xx ); Maybe_Var_Inv::remake( y, yy ); Maybe_Var_Inv::remake( z, zz ); } R operator()() const { return Maybe_Var_Inv::clone(f)( Maybe_Var_Inv::clone(x), Maybe_Var_Inv::clone(y), Maybe_Var_Inv::clone(z), Reuser3(this) ); } }; template struct Reuser3 { typedef typename RT::ResultType R; typedef Thunk3 M; IRef ref; Reuser3(AUniqueTypeForNil) {} Reuser3(const M* m) : ref(m) {} Fun0 operator()( const F& f, const X& x, const Y& y, const Z& z ) { if( !ref ) ref = IRef( new M(f,x,y,z) ); else ref->init(f,x,y,z); return Fun0( 1, ref ); } void iter( const F& f, const X& x, const Y& y, const Z& z ) { if( ref ) ref->init(f,x,y,z); } }; } // namespace fcpp #endif