new example
git-svn-id: https://swig.svn.sourceforge.net/svnroot/swig/trunk/SWIG@844 626c5289-ae23-0410-ae9c-e8d60b6d4f22
This commit is contained in:
parent
dcefa8ad30
commit
6f556a1b14
7 changed files with 173 additions and 0 deletions
19
Examples/tcl/funcptr/Makefile
Normal file
19
Examples/tcl/funcptr/Makefile
Normal file
|
|
@ -0,0 +1,19 @@
|
|||
TOP = ../..
|
||||
SWIG = $(TOP)/../swig
|
||||
SRCS = example.c
|
||||
TARGET = my_tclsh
|
||||
DLTARGET = example
|
||||
INTERFACE = example.i
|
||||
|
||||
all::
|
||||
$(MAKE) -f $(TOP)/Makefile SRCS='$(SRCS)' SWIG='$(SWIG)' \
|
||||
TARGET='$(DLTARGET)' INTERFACE='$(INTERFACE)' tcl
|
||||
|
||||
static::
|
||||
$(MAKE) -f $(TOP)/Makefile SRCS='$(SRCS)' SWIG='$(SWIG)' \
|
||||
TARGET='$(TARGET)' INTERFACE='$(INTERFACE)' tclsh
|
||||
|
||||
clean::
|
||||
rm -f *_wrap* *.o my_tclsh *~ .~* core *.so *.sl
|
||||
|
||||
check: all
|
||||
17
Examples/tcl/funcptr/example.c
Normal file
17
Examples/tcl/funcptr/example.c
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
/* File : example.c */
|
||||
|
||||
int do_op(int a, int b, int (*op)(int,int)) {
|
||||
return (*op)(a,b);
|
||||
}
|
||||
|
||||
int add(int a, int b) {
|
||||
return a+b;
|
||||
}
|
||||
|
||||
int sub(int a, int b) {
|
||||
return a-b;
|
||||
}
|
||||
|
||||
int mul(int a, int b) {
|
||||
return a*b;
|
||||
}
|
||||
7
Examples/tcl/funcptr/example.h
Normal file
7
Examples/tcl/funcptr/example.h
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
/* file: example.h */
|
||||
|
||||
extern int do_op(int,int, int (*op)(int,int));
|
||||
extern int add(int,int);
|
||||
extern int sub(int,int);
|
||||
extern int mul(int,int);
|
||||
|
||||
15
Examples/tcl/funcptr/example.i
Normal file
15
Examples/tcl/funcptr/example.i
Normal file
|
|
@ -0,0 +1,15 @@
|
|||
/* File : example.i */
|
||||
%module example
|
||||
%{
|
||||
#include "example.h"
|
||||
%}
|
||||
|
||||
/* Wrap a function taking a pointer to a function */
|
||||
extern int do_op(int a, int b, int (*op)(int, int));
|
||||
|
||||
/* Now install a bunch of "ops" as constants */
|
||||
%constant(int (*)(int,int)) ADD = add;
|
||||
%constant(int (*)(int,int)) SUB = sub;
|
||||
%constant(int (*)(int,int)) MUL = mul;
|
||||
|
||||
|
||||
22
Examples/tcl/funcptr/example.tcl
Normal file
22
Examples/tcl/funcptr/example.tcl
Normal file
|
|
@ -0,0 +1,22 @@
|
|||
# file: example.tcl
|
||||
|
||||
catch { load ./example.so example}
|
||||
catch { load ./example.dll example} ;# Windows
|
||||
|
||||
set a 37
|
||||
set b 42
|
||||
|
||||
# Now call our C function with a bunch of callbacks
|
||||
|
||||
puts "Trying some C callback functions"
|
||||
puts " a = $a"
|
||||
puts " b = $b"
|
||||
puts " ADD(a,b) = [do_op $a $b $ADD]"
|
||||
puts " SUB(a,b) = [do_op $a $b $SUB]"
|
||||
puts " MUL(a,b) = [do_op $a $b $MUL]"
|
||||
|
||||
puts "Here is what the C callback function objects look like in Tcl"
|
||||
puts " ADD = $ADD"
|
||||
puts " SUB = $SUB"
|
||||
puts " MUL = $MUL"
|
||||
|
||||
92
Examples/tcl/funcptr/index.html
Normal file
92
Examples/tcl/funcptr/index.html
Normal file
|
|
@ -0,0 +1,92 @@
|
|||
<html>
|
||||
<head>
|
||||
<title>SWIG:Examples:tcl:funcptr</title>
|
||||
</head>
|
||||
|
||||
<body bgcolor="#ffffff">
|
||||
|
||||
|
||||
<tt>SWIG/Examples/tcl/funcptr/</tt>
|
||||
<hr>
|
||||
|
||||
<H2>Pointers to Functions</H2>
|
||||
|
||||
<tt>$Header$</tt><br>
|
||||
|
||||
<p>
|
||||
Okay, just what in the heck does SWIG do with a declaration like this?
|
||||
|
||||
<blockquote>
|
||||
<pre>
|
||||
int do_op(int a, int b, int (*op)(int, int));
|
||||
</pre>
|
||||
</blockquote>
|
||||
|
||||
Well, it creates a wrapper as usual. Of course, that does raise some
|
||||
questions about the third argument (the pointer to a function).
|
||||
|
||||
<p>
|
||||
In this case, SWIG will wrap the function pointer as it does for all other
|
||||
pointers. However, in order to actually call this function from a script,
|
||||
you will need to pass some kind of C function pointer object. In C,
|
||||
this is easy, you just supply a function name as an argument like this:
|
||||
|
||||
<blockquote>
|
||||
<pre>
|
||||
/* Some callback function */
|
||||
int add(int a, int b) {
|
||||
return a+b;
|
||||
}
|
||||
...
|
||||
int r = do_op(x,y,add);
|
||||
</pre>
|
||||
</blockquote>
|
||||
|
||||
To make this work with SWIG, you will need to do a little extra work. Specifically,
|
||||
you need to create some function pointer objects using the %constant directive like this:
|
||||
|
||||
<blockquote>
|
||||
<pre>
|
||||
%constant(int (*)(int,int)) ADD = add;
|
||||
</pre>
|
||||
</blockquote>
|
||||
|
||||
Now, in a script, you would do this:
|
||||
|
||||
<blockquote>
|
||||
<pre>
|
||||
set r [do_op $x $y $ADD]
|
||||
</pre>
|
||||
</blockquote>
|
||||
|
||||
<h2>An Example</h2>
|
||||
|
||||
Here are some files that illustrate this with a simple example:
|
||||
|
||||
<ul>
|
||||
<li><a href="example.c">example.c</a>
|
||||
<li><a href="example.h">example.h</a>
|
||||
<li><a href="example.i">example.i</a> (SWIG interface)
|
||||
<li><a href="example.tcl">example.tcl</a> (Sample script)
|
||||
</ul>
|
||||
|
||||
<h2>Notes</h2>
|
||||
|
||||
<ul>
|
||||
<li>The value of a function pointer must correspond to a function written in C or C++.
|
||||
It is not possible to pass an arbitrary Tcl function object in as a substitute for a C
|
||||
function pointer.
|
||||
|
||||
<p>
|
||||
<li>A Tcl function can be used as a C/C++ callback if you write some
|
||||
clever typemaps and are very careful about how you create your extension.
|
||||
This is an advanced topic not covered here.
|
||||
</ul>
|
||||
|
||||
<hr>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
|
||||
|
||||
|
||||
|
|
@ -21,6 +21,7 @@ certain C declarations are turned into constants.
|
|||
<li><a href="class/index.html">class</a>. How wrap a simple C++ class.
|
||||
<li><a href="reference/index.html">reference</a>. C++ references.
|
||||
<li><a href="pointer/index.html">pointer</a>. Simple pointer handling.
|
||||
<li><a href="funcptr/index.html">funcptr</a>. Pointers to functions.
|
||||
</ul>
|
||||
|
||||
<h2>Compilation Issues</h2>
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue