- Embedding (a Tcl interpreter into a C app)
- Extending (a Tcl interpreter statically, or with dynamic libs)
- allows to define a C "proc" (in fact, the central part of a main program),
- writes a C source file to disk ("outsourcing"),
- compiles that to an executable in the same directory, and
- creates a Tcl wrapper proc that in turn calls the executable "C helper".
- Tcl arguments are passed in via argv; each is mapped to a char pointer with the same name. You may convert other representations from that (e.g. atoi in the example below)
- The helper C code is spliced into a int main(int argc, char *argv[]) frame, stdio.h and stlib.h are included already. Other #includes can be done early in the helper code.
- Additional C functions can be defined completely in the -with argument. Other arguments allow to override the defaults for compiler name, flags, or generation directory.
- The helper puts its regular output to stdout. This is where exec and open read from, and you don't have to bother with allocating memory for the result.
- In error cases just put a message to stderr, so it is seen from inside Tcl (I hate empty strings as error messages ;-) This will cause exec and the wrapper proc to error out. A FATAL macro which takes a constant string is pre-defined (see example).
- Errors in your C code will cause an error in cproc, where the compiler's error message (at least with gcc) is reported to Tcl.
For more powerful code generators, see Pipe servers in C from Tcl - Extending Tcl in C from Tcl
proc cproc {name argl cbody args} {
if [llength [info command $name]] {error "$name exists"}
array set a [list -cc gcc -ccflags {-s -Wall -W -ansi -pedantic}\
-dir $::env(TEMP) -with {}]
array set a $args
set cargs ""
set narg 0
foreach i $argl {
append cargs "\n\t\t char *[lindex $i 0] = argv\[[incr narg]\];"
}
set nname [file nativename [file join $a(-dir) $name]]
set fp [open $nname.c w]
puts $fp "/* $name.c - Generated by cproc */
#include <stdio.h>
#include <stdlib.h>
#define MAXLINE 256
#define FATAL(_s) {fprintf(stderr,\"error: %s\",_s); return -1;}
$a(-with)
int main(int argc, char *argv\[\]) { $cargs
if(argc!=[incr narg]) FATAL(\"usage: $name $argl\");
{$cbody
}
return 0;
}"
close $fp
eval exec $a(-cc) $a(-ccflags) [list $nname.c -o $nname]
set body "exec [list $nname]"
foreach i $argl {append body " \$[lindex $i 0]"}
proc $name $argl $body
}
# That's all, now for some usage examples:
cproc foo {s {count 2}} {
/* repeat a string s n times, where count holds the string rep of n */
int n = atoi(count);
int i;
if(n<0) FATAL("count must be non-negative");
for(i=0; i<n; i++) { results(s); }
} -with {
void results(char *x) {printf("%s", x); /* just to show a function */ }
}
foo grill 5
#foo bar -1 => should raise an errorThis is what was generated - file foo.c:
/* foo.c - Generated by cproc */
#include <stdio.h>
#include <stdlib.h>
#define MAXLINE 256
#define FATAL(_s) {fprintf(stderr,"error: %s",_s); return -1;}
void results(char *x) {printf("%s", x); /* just to show a function */ }
int main(int argc, char *argv[]) {
char *s = argv[1];
char *count = argv[2];
if(argc!=3) FATAL("usage: foo s {count 2}");
{
/* repeat a string s n times, where count holds the string rep of n */
int n = atoi(count);
int i;
if(n<0) FATAL("count must be non-negative");
for(i=0; i<n; i++) { results(s); }
}
return 0;
}And on the Tcl side: proc foo {s {count 2}} {exec {C:\WINDOWS\TEMP\foo} $s $count}
#--------------------------------------------------------- Another example:
cproc strrev s {
/* Revert a string */
char *cp = s+strlen(s);
while(cp > s) putchar(*--cp);
}
strrev "A man, a plan, a canal: Panama" ;#=> amanaP :lanac a ,nalp a ,nam A
# The same in Tcl, for comparison - 410(Tcl) vs. 86000(cproc) microseconds:
proc strrevert s {
set res ""
foreach i [split $s ""] {set res $i$res}
set res
}
