Hello,
I want to build a library to help exporting c++ functions to a
scripting languagge. The scripting language provides a function to
register functions like:
ANY f0()
ANY f1(ANY)
ANY f2(ANY, ANY)
ANY f3(ANY, ANY, ANY)
...
where ANY is an opaque type, with functions available to convert from
and to basic C types.
I would like to avoid writing a wrapper for any function I have to
export.
My first try (restricted to functions of one argument, but can easily
be extended):
//------------------------------------------------
template <class T_return, class T_arg, T_return (*fn)(T_arg)>
struct Translator1 {
static ANY proc(ANY arg)
{
T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type
T_return r = to_any<T_return>((*fn)(a)); // Conversion from return
type to ANY
return r;
}
};
int foo(int a) { return a*10; }
typedef Translator1<int, int, foo> FN;
c_reg_function( FN::proc );
//-----------------------------------------------------
Ok, that works ! But it would be fine to let the compiler guess the
type of the arguments:
//--------------------------------------------------------
template <class T_return, class T_arg>
void export_fn(T_return (*fn)(T_arg))
{
c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg));
}
export_fn(foo);
//-----------------------------------------------------------------
But now, something goes wrong (g++ v3.3) !!!
test.cc: En function `void export_fn(T_return (*)(T_arg)) [with
T_return = int,
T_arg = int]':
test.cc:43: instantiated from here
test.cc:33: error: `fn' is not a valid template argument
test.cc:33: error: it must be the address of a function with external
linkage
test.cc:33: error: `arg' undeclared (first use this function)
test.cc:33: error: (Each undeclared identifier is reported only once
for each
function it appears in.)
test.cc:33: error: `<type error>' is not a class type
I'm afraid that the compiler cannot expand a template class within a
template function with fewer template arguments ! I cannot use
functors because the register function expects C functions !
Any good idea ? I cannot think of any solution that avoids the use of
dirty preprocesor macros.
Thanks !!!
A reduced version of the code is provided so that you can try
yourself:
////////////////////////////////////////////
#include <iostream>
using namespace std;
typedef long ANY;
void c_reg_function(ANY (*fn)(ANY)); // external c library function
void c_reg_function(ANY (*fn)(ANY))
{
cout << "Registro: " << (*fn)(2) << endl;
}
template<class T> ANY to_any(T); // ANY conversion functions
template<class T> T from_any(ANY);
template<> int from_any(ANY a) { return a; }
template<> ANY to_any(int a) { return a; }
template <class T_return, class T_arg, T_return (*fn)(T_arg)>
struct Translator1 {
static ANY proc(ANY arg)
{
T_arg a = from_any<T_arg>(arg);
T_return r = to_any<T_return>((*fn)(a));
return r;
}
};
template <class T_return, class T_arg>
void export_fn(T_return (*fn)(T_arg))
{
c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg));
}
int foo(int a) { return a*10; }
int main(int argc, char** argv)
{
typedef Translator1<int, int, foo> FN;
c_reg_function( FN::proc );
export_fn(foo);
}
/////////////////////////////////////7 8 11210
"vpadial" <vi***********@yahoo.es> wrote in message
news:e9*************************@posting.google.co m... Hello,
I want to build a library to help exporting c++ functions to a scripting languagge. The scripting language provides a function to register functions like:
ANY f0() ANY f1(ANY) ANY f2(ANY, ANY) ANY f3(ANY, ANY, ANY) ...
where ANY is an opaque type, with functions available to convert from and to basic C types.
I would like to avoid writing a wrapper for any function I have to export. My first try (restricted to functions of one argument, but can easily be extended):
//------------------------------------------------
template <class T_return, class T_arg, T_return (*fn)(T_arg)> struct Translator1 { static ANY proc(ANY arg) { T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type T_return r = to_any<T_return>((*fn)(a)); // Conversion from return type to ANY return r; } };
int foo(int a) { return a*10; }
typedef Translator1<int, int, foo> FN; c_reg_function( FN::proc );
//-----------------------------------------------------
Ok, that works ! But it would be fine to let the compiler guess the type of the arguments:
//--------------------------------------------------------
template <class T_return, class T_arg> void export_fn(T_return (*fn)(T_arg)) { c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg)); }
export_fn(foo);
//-----------------------------------------------------------------
But now, something goes wrong (g++ v3.3) !!!
test.cc: En function `void export_fn(T_return (*)(T_arg)) [with T_return = int, T_arg = int]': test.cc:43: instantiated from here test.cc:33: error: `fn' is not a valid template argument test.cc:33: error: it must be the address of a function with external linkage test.cc:33: error: `arg' undeclared (first use this function) test.cc:33: error: (Each undeclared identifier is reported only once for each function it appears in.) test.cc:33: error: `<type error>' is not a class type
I'm afraid that the compiler cannot expand a template class within a template function with fewer template arguments ! I cannot use functors because the register function expects C functions !
It's not complaining about the number of arguments. The value of "fn" won't
be known until runtime. You can't expect the compiler to instantiate a
template based on its value if it can't possibly know the value.
Why is it a template argument in the first place? Couldn't you just write
it like this:
template <class T_return, class T_arg>
struct Translator1 {
static ANY proc(ANY arg, T_return (*fn)(T_arg))
{
T_arg a = from_any<T_arg>(arg);
T_return r = to_any<T_return>((*fn)(a));
return r;
}
};
template <class T_return, class T_arg>
void export_fn(T_return (*fn)(T_arg))
{
c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg,fn));
}
or do you plan on doing some specializations based on its value?
--
David Hilsee
"David Hilsee" <da*************@yahoo.com> wrote in message
news:rv********************@comcast.com...
[...] c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg,fn));
Sorry, I meant
c_reg_function(Translator1<T_return, T_arg>::proc(arg,fn));
--
David Hilsee
"David Hilsee" <da*************@yahoo.com> wrote in message
news:rv********************@comcast.com... "vpadial" <vi***********@yahoo.es> wrote in message news:e9*************************@posting.google.co m... Hello,
I want to build a library to help exporting c++ functions to a scripting languagge. The scripting language provides a function to register functions like:
ANY f0() ANY f1(ANY) ANY f2(ANY, ANY) ANY f3(ANY, ANY, ANY) ...
where ANY is an opaque type, with functions available to convert from and to basic C types.
I would like to avoid writing a wrapper for any function I have to export. My first try (restricted to functions of one argument, but can easily be extended):
//------------------------------------------------
template <class T_return, class T_arg, T_return (*fn)(T_arg)> struct Translator1 { static ANY proc(ANY arg) { T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type T_return r = to_any<T_return>((*fn)(a)); // Conversion from return type to ANY return r; } };
int foo(int a) { return a*10; }
typedef Translator1<int, int, foo> FN; c_reg_function( FN::proc );
//-----------------------------------------------------
Ok, that works ! But it would be fine to let the compiler guess the type of the arguments:
//--------------------------------------------------------
template <class T_return, class T_arg> void export_fn(T_return (*fn)(T_arg)) { c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg)); }
export_fn(foo);
//-----------------------------------------------------------------
But now, something goes wrong (g++ v3.3) !!!
test.cc: En function `void export_fn(T_return (*)(T_arg)) [with T_return = int, T_arg = int]': test.cc:43: instantiated from here test.cc:33: error: `fn' is not a valid template argument test.cc:33: error: it must be the address of a function with external linkage test.cc:33: error: `arg' undeclared (first use this function) test.cc:33: error: (Each undeclared identifier is reported only once for each function it appears in.) test.cc:33: error: `<type error>' is not a class type
I'm afraid that the compiler cannot expand a template class within a template function with fewer template arguments ! I cannot use functors because the register function expects C functions ! It's not complaining about the number of arguments. The value of "fn"
won't be known until runtime. You can't expect the compiler to instantiate a template based on its value if it can't possibly know the value.
Why is it a template argument in the first place? Couldn't you just write it like this:
template <class T_return, class T_arg> struct Translator1 { static ANY proc(ANY arg, T_return (*fn)(T_arg)) { T_arg a = from_any<T_arg>(arg); T_return r = to_any<T_return>((*fn)(a)); return r; } };
template <class T_return, class T_arg> void export_fn(T_return (*fn)(T_arg)) { c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg,fn)); } [...]
After rereading your post several times, I realized that I did not fully
understand what you were trying to do. I was confused by the sample code
that appeared to be an attempt at invoking "proc" (Translator1<T_return,
T_arg, fn>::proc(arg)). I think the line "c_reg_function( FN::proc );"
should have instead been "c_reg_function( &FN::proc );" For some reason, I
thought you had omitted some arguments to the function call on that line.
Sorry for any confusion my responses may have caused.
Even so, I can't think of any way that you could have the compiler easily
and automatically determine the return type and arguments in the way that
you need them to be determined. The function template you wrote won't work
for the reason a mentioned before: the value is not known at compile time.
--
David Hilsee
"David Hilsee" <da*************@yahoo.com> wrote in message news:<Kc********************@comcast.com>... "David Hilsee" <da*************@yahoo.com> wrote in message news:rv********************@comcast.com... [...] c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg,fn));
Sorry, I meant
c_reg_function(Translator1<T_return, T_arg>::proc(arg,fn));
The problem is that c_reg_function expects a pointer to a function (ANY (*fn)(ANY))
On 6 Oct 2004 14:58:45 -0700, vi***********@yahoo.es (vpadial) wrote: Hello,
I want to build a library to help exporting c++ functions to a scripting languagge. The scripting language provides a function to register functions like:
ANY f0() ANY f1(ANY) ANY f2(ANY, ANY) ANY f3(ANY, ANY, ANY) ...
where ANY is an opaque type, with functions available to convert from and to basic C types.
I would like to avoid writing a wrapper for any function I have to export. My first try (restricted to functions of one argument, but can easily be extended):
//------------------------------------------------
template <class T_return, class T_arg, T_return (*fn)(T_arg)> struct Translator1 { static ANY proc(ANY arg) { T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type T_return r = to_any<T_return>((*fn)(a)); // Conversion from return type to ANY return r; } };
int foo(int a) { return a*10; }
typedef Translator1<int, int, foo> FN; c_reg_function( FN::proc );
//-----------------------------------------------------
Ok, that works ! But it would be fine to let the compiler guess the type of the arguments:
//--------------------------------------------------------
template <class T_return, class T_arg> void export_fn(T_return (*fn)(T_arg)) { c_reg_function(Translator1<T_return, T_arg, fn>::proc(arg)); }
export_fn(foo);
//-----------------------------------------------------------------
But now, something goes wrong (g++ v3.3) !!!
test.cc: En function `void export_fn(T_return (*)(T_arg)) [with T_return = int, T_arg = int]': test.cc:43: instantiated from here test.cc:33: error: `fn' is not a valid template argument test.cc:33: error: it must be the address of a function with external linkage test.cc:33: error: `arg' undeclared (first use this function) test.cc:33: error: (Each undeclared identifier is reported only once for each function it appears in.) test.cc:33: error: `<type error>' is not a class type
I'm afraid that the compiler cannot expand a template class within a template function with fewer template arguments ! I cannot use functors because the register function expects C functions !
Any good idea ? I cannot think of any solution that avoids the use of dirty preprocesor macros.
The problem is that your function argument "fn" has a runtime value,
but template arguments are required to be known at compile time.
I can't think of a standard C++ solution, but I can think of a gcc
specific solution - typeof. e.g.
template <typename FuncType, FuncType func>
struct Translator1
{
//need to write FuncArgs, using partial specialization
typedef FuncArgs<FuncType>::first_argument_type T_arg;
typedef FuncArgs<FuncType>::return_type T_return;
static ANY proc(ANY arg)
{
T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type
T_return r = to_any<T_return>((*fn)(a));
return r;
}
};
c_reg_function(&Translator1<typeof(&foo), foo>::proc);
Whether you can put up with the non-portability is up to you.
Tom
Tom Widmer <to********@hotmail.com> wrote in message news:<k7********************************@4ax.com>. .. The problem is that your function argument "fn" has a runtime value, but template arguments are required to be known at compile time.
I can't think of a standard C++ solution, but I can think of a gcc specific solution - typeof. e.g.
template <typename FuncType, FuncType func> struct Translator1 { //need to write FuncArgs, using partial specialization typedef FuncArgs<FuncType>::first_argument_type T_arg; typedef FuncArgs<FuncType>::return_type T_return;
static ANY proc(ANY arg) { T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type T_return r = to_any<T_return>((*fn)(a)); return r; } };
c_reg_function(&Translator1<typeof(&foo), foo>::proc);
Whether you can put up with the non-portability is up to you.
Thanks Tom, but I can't get it working. Could you explain better the
use of FuncArgs ? How can I specialize types of a template class ?
Must I specialize FuncArgs for any new function signature ?
Thanks in advance vi***********@yahoo.es (vpadial) wrote in message news:<e9*************************@posting.google.c om>... Tom Widmer <to********@hotmail.com> wrote in message news:<k7********************************@4ax.com>. .. The problem is that your function argument "fn" has a runtime value, but template arguments are required to be known at compile time.
I can't think of a standard C++ solution, but I can think of a gcc specific solution - typeof. e.g.
template <typename FuncType, FuncType func> struct Translator1 { //need to write FuncArgs, using partial specialization typedef FuncArgs<FuncType>::first_argument_type T_arg; typedef FuncArgs<FuncType>::return_type T_return;
static ANY proc(ANY arg) { T_arg a = from_any<T_arg>(arg); // Conversion from ANY to arg type T_return r = to_any<T_return>((*fn)(a)); return r; } };
c_reg_function(&Translator1<typeof(&foo), foo>::proc);
Whether you can put up with the non-portability is up to you.
Thanks Tom, but I can't get it working. Could you explain better the use of FuncArgs ? How can I specialize types of a template class ? Must I specialize FuncArgs for any new function signature ?
Thanks in advance
Don't bother. I've seen Boost provides function_traits that I guess is
equivalent to your FuncArgs class: http://www.boost.org/libs/type_traits/#function_traits
Thanks for your help This thread has been closed and replies have been disabled. Please start a new discussion. Similar topics
by: Surya Kiran |
last post by:
Hi all,
I've written a function template. say
template <class T>
fn (T var) { ... }
Is there any way, from within the function, can we check what type of
argument we've passed on to the...
|
by: Suzanne Vogel |
last post by:
Is it possible to store a pointer to a template function?
eg,
template<class T> fooFunc() {...}
fptr = fooFunc; // <--
I couldn't find any info here, not even in the forums:...
|
by: John Harrison |
last post by:
Can anyone explain why this code fails to compile?
#include <iostream>
void func()
{
std::cout << "success!\n";
}
struct S
|
by: jimjim |
last post by:
Hello,
My question concerns as to how a pointer is passed by reference as a
function argument. The following is from code taken from the MICO
implementation of the CORBA specification.
in...
|
by: Vijai Kalyan |
last post by:
I was decomposing a task into different policies. Essentially, there is
a general option obtained from a server and user options obtained from
configuration variables. The two options are...
|
by: infogoogle |
last post by:
Hello,
i'm having problems with the type of a template function:
This code:
class A {};
class B : A {};
template<class T B* fnull() { return 0; };
|
by: Jef Driesen |
last post by:
Suppose I have a datastructure (actually it's a graph) with one template
parameter (the property P for each edge and vertex):
struct graph<P>;
struct vertex<P>;
struct edge<P>;
I also have...
|
by: StephQ |
last post by:
This is from a thread that I posted on another forum some days ago.
I didn't get any response, so I'm proposing it in this ng in hope of
better luck :)
The standard explanation is that pointer...
|
by: webinfinite |
last post by:
This is my first post here, please tell me if I did anything wrong.
in the following code snippet:
1. template <class In, class Pred>
2. In find_if(In begin, In end, Pred f) {
3. while (begin...
|
by: Fei Liu |
last post by:
Hello, I just hit a strange problem regarding SFINAE. The following code
causes compile error (void cannot be array element type), I thought
SFINA should match test(...) version instead and not...
|
by: Naresh1 |
last post by:
What is WebLogic Admin Training?
WebLogic Admin Training is a specialized program designed to equip individuals with the skills and knowledge required to effectively administer and manage Oracle...
|
by: antdb |
last post by:
Ⅰ. Advantage of AntDB: hyper-convergence + streaming processing engine
In the overall architecture, a new "hyper-convergence" concept was proposed, which integrated multiple engines and...
|
by: Arjunsri |
last post by:
I have a Redshift database that I need to use as an import data source. I have configured the DSN connection using the server, port, database, and credentials and received a successful connection...
|
by: WisdomUfot |
last post by:
It's an interesting question you've got about how Gmail hides the HTTP referrer when a link in an email is clicked. While I don't have the specific technical details, Gmail likely implements measures...
|
by: Matthew3360 |
last post by:
Hi,
I have been trying to connect to a local host using php curl. But I am finding it hard to do this. I am doing the curl get request from my web server and have made sure to enable curl. I get a...
|
by: BLUEPANDA |
last post by:
At BluePanda Dev, we're passionate about building high-quality software and sharing our knowledge with the community. That's why we've created a SaaS starter kit that's not only easy to use but also...
|
by: Johno34 |
last post by:
I have this click event on my form. It speaks to a Datasheet Subform
Private Sub Command260_Click()
Dim r As DAO.Recordset
Set r = Form_frmABCD.Form.RecordsetClone
r.MoveFirst
Do
If...
|
by: ezappsrUS |
last post by:
Hi,
I wonder if someone knows where I am going wrong below. I have a continuous form and two labels where only one would be visible depending on the checkbox being checked or not. Below is the...
|
by: jack2019x |
last post by:
hello, Is there code or static lib for hook swapchain present?
I wanna hook dxgi swapchain present for dx11 and dx9.
| |