#!/usr/bin/perl use strict; print < This file is part of GVPE. GVPE is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with gvpe; if not, write to the Free Software Foundation, Inc. 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #ifndef CALLBACK_H__ #define CALLBACK_H__ #define CALLBACK_H_VERSION 3 template struct callback_funtype_trait; template struct callback_get_impl; EOF for my $a (0..10) { my $CLASS = join "", map ", class A$_", 1..$a; my $TYPE = join ", ", map "A$_", 1..$a; my $ARG = join ", ", map "a$_", 1..$a; my $TYPEARG = join ", ", map "A$_ a$_", 1..$a; my $TYPEDEFS = join " ", map "typedef A$_ arg$_\_type;", 1..$a; my $TYPEvoid = $TYPE ? $TYPE : "void"; my $_TYPE = $TYPE ? ", $TYPE" : ""; my $_TTYPE = $a ? join "", map ", typename T::arg$_\_type", 1..$a : ""; print < class callback$a { struct klass; // it is vital that this is never defined typedef R (klass::*ptr_type)($TYPE); klass *o; R (klass::*m)($TYPE); public: template explicit callback$a (O1 *object, R (O2::*method)($TYPE)) { o = reinterpret_cast(object); m = reinterpret_cast(method); } // this works because a standards-compliant C++ compiler // basically can't help it: it doesn't have the knowledge // required to miscompile (klass is not defined anywhere // and nothing is known about the constructor arguments) :) R call($TYPEARG) const { return (o->*m) ($ARG); } R operator ()($TYPEARG) const { return call ($ARG); } }; template struct callback_funtype_trait$a { static const int arity = $a; typedef R type ($TYPEvoid); typedef R result_type; $TYPEDEFS }; template struct callback_funtype_trait : callback_funtype_trait$a { }; template struct callback_get_impl<$a, signature> { typedef callback_funtype_trait T; typedef callback$a type; }; EOF } print < struct callback : callback_get_impl::arity, signature>::type { typedef typename callback_get_impl::arity, signature>::type base_type; template explicit callback (O object, M method) : base_type (object, method) { } }; #endif EOF