]> git.saurik.com Git - wxWidgets.git/blob - wxPython/wxSWIG/Modules/tcl.cxx
added ability to build contrib samples
[wxWidgets.git] / wxPython / wxSWIG / Modules / tcl.cxx
1 /*******************************************************************************
2 * Simplified Wrapper and Interface Generator (SWIG)
3 *
4 * Author : David Beazley
5 *
6 * Department of Computer Science
7 * University of Chicago
8 * 1100 E 58th Street
9 * Chicago, IL 60637
10 * beazley@cs.uchicago.edu
11 *
12 * Please read the file LICENSE for the copyright and terms by which SWIG
13 * can be used and distributed.
14 *******************************************************************************/
15
16 /***********************************************************************
17 * $Header$
18 *
19 * tcl.cxx
20 *
21 * Definitions for creating a simple, stand-alone TCL implementation.
22 *
23 ***********************************************************************/
24
25 #include "swig.h"
26 #include "swigtcl.h"
27 #include <ctype.h>
28
29 static char *Tcl_config="swigtcl.swg";
30 static char *usage = "\
31 Tcl Options (available with -tcl)\n\
32 -module name - Set name of module\n\
33 -prefix name - Set a prefix to be appended to all names\n\
34 -namespace - Build module into a Tcl8 namespace. \n\
35 -noobject - Omit code for object oriented interface.\n\
36 -old - Use old SWIG interface (same as -noobject).\n\n";
37
38
39 static char *ns_name = 0;
40 static String mod_init;
41 static String mod_extern;
42
43 // ---------------------------------------------------------------------
44 // TCL::parse_args(int argc, char *argv[])
45 //
46 // Parse tcl specific command line options
47 // ---------------------------------------------------------------------
48
49 void TCL::parse_args(int argc, char *argv[]) {
50
51 int i = 1;
52 sprintf(LibDir,"%s",tcl_path);
53
54 // Look for certain command line options
55
56 for (i = 1; i < argc; i++) {
57 if (argv[i]) {
58 if (strcmp(argv[i],"-prefix") == 0) {
59 if (argv[i+1]) {
60 prefix = new char[strlen(argv[i+1])+2];
61 strcpy(prefix, argv[i+1]);
62 mark_arg(i);
63 mark_arg(i+1);
64 i++;
65 } else {
66 arg_error();
67 }
68 } else if (strcmp(argv[i],"-module") == 0) {
69 if (argv[i+1]) {
70 set_module(argv[i+1],0);
71 mark_arg(i);
72 mark_arg(i+1);
73 i++;
74 } else {
75 arg_error();
76 }
77 } else if (strcmp(argv[i],"-namespace") == 0) {
78 nspace = 1;
79 mark_arg(i);
80 } else if (strcmp(argv[i],"-old") == 0) {
81 shadow = 0;
82 mark_arg(i);
83 } else if (strcmp(argv[i],"-noobject") == 0) {
84 shadow = 0;
85 mark_arg(i);
86 } else if (strcmp(argv[i],"-help") == 0) {
87 fputs(usage,stderr);
88 }
89 }
90 }
91
92 // If a package has been specified, make sure it ends with a '_'
93
94 if (prefix) {
95 ns_name = copy_string(prefix);
96 if (prefix[strlen(prefix)] != '_') {
97 prefix[strlen(prefix)+1] = 0;
98 prefix[strlen(prefix)] = '_';
99 }
100 } else
101 prefix = "";
102
103 // Create a symbol SWIGTCL
104
105 add_symbol("SWIGTCL",0,0);
106 typemap_lang = "tcl";
107
108 // Attempt to load up the C++ configuration files
109
110 get_file("delcmd.swg",delcmd);
111 get_file("methodcmd.swg",methodcmd);
112 get_file("objcmd.swg",objcmd);
113
114 }
115
116 // ---------------------------------------------------------------------
117 // void TCL::parse()
118 //
119 // Start parsing an interface file for Tcl.
120 // ---------------------------------------------------------------------
121
122 void TCL::parse() {
123
124 fprintf(stderr,"Making wrappers for Tcl\n");
125
126 // Print out TCL specific headers
127
128 headers();
129
130 // Run the parser
131
132 yyparse();
133
134 }
135
136 // ---------------------------------------------------------------------
137 // TCL::set_module(char *mod_name,char **mod_list)
138 //
139 // Sets the module name.
140 // Does nothing if it's already set (so it can be overridden as a command
141 // line option).
142 //
143 // mod_list is a NULL-terminated list of additional modules. This
144 // is really only useful when building static executables and other
145 // things.
146 //----------------------------------------------------------------------
147
148 void TCL::set_module(char *mod_name, char **mod_list) {
149
150 char temp[256], *c;
151 int i;
152
153 if (module) return;
154
155 module = new char[strlen(mod_name)+1];
156 strcpy(module,mod_name);
157
158 // Fix capitalization for Tcl
159
160 c = module;
161 while (*c) {
162 *c = (char) tolower(*c);
163 c++;
164 }
165
166 // Now create an initialization function
167
168 sprintf(temp,"%s_Init", module);
169 init_name = new char[strlen(temp) + 1];
170 strcpy(init_name, temp);
171 *init_name = toupper(*init_name);
172
173 if (!ns_name) ns_name = copy_string(module);
174
175 // If namespaces have been specified, set the prefix to the module name
176
177 if ((nspace) && (strlen(prefix) < 1)) {
178 prefix = new char[strlen(module)+2];
179 strcpy(prefix,module);
180 prefix[strlen(module)] = '_';
181 prefix[strlen(module)+1] = 0;
182 }
183
184 // If additional modules have been specified, create some code for
185 // initializing them.
186
187 if (mod_list) {
188 i = 0;
189 while (mod_list[i]) {
190 c = mod_list[i];
191 while (*c) {
192 *c = (char) tolower(*c);
193 c++;
194 }
195 sprintf(temp,"%s_Init",mod_list[i]);
196 temp[0] = toupper(temp[0]);
197
198 // Dump out some initialization code
199
200 mod_init << tab4 << "if (" << temp << "(" << interp_name << ") == TCL_ERROR) {\n"
201 << tab8 << "return TCL_ERROR;\n"
202 << tab4 << "}\n\n";
203 mod_extern << "extern int " << temp << "(Tcl_Interp *);\n";
204 i++;
205 }
206 }
207 }
208
209
210 // ---------------------------------------------------------------------
211 // TCL::set_init(char *iname)
212 //
213 // Sets the initialization function name.
214 // Does nothing if it's already set
215 //
216 //----------------------------------------------------------------------
217
218 void TCL::set_init(char *iname) {
219
220 if (init_name) return;
221
222 init_name = new char[strlen(iname)+1];
223 strcpy(init_name, iname);
224
225 }
226
227 // ---------------------------------------------------------------------
228 // TCL::headers(void)
229 //
230 // Generate the appropriate header files for TCL interface.
231 // ----------------------------------------------------------------------
232
233 void TCL::headers(void)
234 {
235
236 emit_banner(f_header);
237 fprintf(f_header,"/* Implementation : TCL 7.x */\n\n");
238 fprintf(f_header,"#include <tcl.h>\n");
239 fprintf(f_header,"#include <string.h>\n");
240 fprintf(f_header,"#include <stdlib.h>\n");
241 fprintf(f_header,"#define SWIGTCL\n");
242
243 if (NoInclude) {
244 fprintf(f_header,"#define SWIG_NOINCLUDE\n");
245 }
246
247 if (insert_file("swigtcl.swg", f_header) == -1) {
248 fprintf(stderr,"Unable to find 'swigtcl.swg'. Possible installation problem.\n");
249 SWIG_exit(1);
250 }
251 }
252
253 // --------------------------------------------------------------------
254 // TCL::initialize(void)
255 //
256 // Produces an initialization function. Assumes that the init function
257 // name has already been specified.
258 // ---------------------------------------------------------------------
259
260 void TCL::initialize()
261 {
262
263 if ((!ns_name) && (nspace)) {
264 fprintf(stderr,"Tcl error. Must specify a namespace.\n");
265 SWIG_exit(1);
266 }
267
268 if (!init_name) {
269 init_name = "Swig_Init";
270 fprintf(stderr,"SWIG : *** Warning. No module name specified.\n");
271 }
272
273 fprintf(f_header,"#define SWIG_init %s\n", init_name);
274 if (!module) module = "swig";
275 fprintf(f_header,"#define SWIG_name \"%s\"\n", module);
276 if (nspace) {
277 fprintf(f_header,"#define SWIG_prefix \"%s::\"\n", ns_name);
278 fprintf(f_header,"#define SWIG_namespace \"%s\"\n\n", ns_name);
279 } else {
280 fprintf(f_header,"#define SWIG_prefix \"%s\"\n", prefix);
281 fprintf(f_header,"#define SWIG_namespace \"\"\n\n");
282 }
283 fprintf(f_header,"#ifdef __cplusplus\n");
284 fprintf(f_header,"extern \"C\" {\n");
285 fprintf(f_header,"#endif\n");
286 fprintf(f_header,"#ifdef MAC_TCL\n");
287 fprintf(f_header,"#pragma export on\n");
288 fprintf(f_header,"#endif\n");
289 fprintf(f_header,"SWIGEXPORT(int) %s(Tcl_Interp *);\n", init_name);
290 fprintf(f_header,"#ifdef MAC_TCL\n");
291 fprintf(f_header,"#pragma export off\n");
292 fprintf(f_header,"#endif\n");
293 fprintf(f_header,"#ifdef __cplusplus\n");
294 fprintf(f_header,"}\n");
295 fprintf(f_header,"#endif\n");
296
297 fprintf(f_init,"SWIGEXPORT(int) %s(Tcl_Interp *%s) {\n", init_name, interp_name);
298
299 fprintf(f_init,"\t if (%s == 0) \n", interp_name);
300 fprintf(f_init,"\t\t return TCL_ERROR;\n");
301
302 /* Check to see if other initializations need to be performed */
303
304 if (strlen(mod_extern.get())) {
305 fprintf(f_init,"%s\n",mod_init.get());
306 fprintf(f_header,"#ifdef __cplusplus\n");
307 fprintf(f_header,"extern \"C\" {\n");
308 fprintf(f_header,"#endif\n");
309 fprintf(f_header,"%s\n",mod_extern.get());
310 fprintf(f_header,"#ifdef __cplusplus\n");
311 fprintf(f_header,"}\n");
312 fprintf(f_header,"#endif\n");
313 }
314
315 /* Check to see if we're adding support for Tcl8 nspaces */
316 if (nspace) {
317 fprintf(f_init,"#if (TCL_MAJOR_VERSION >= 8)\n");
318 fprintf(f_init,"\t Tcl_Eval(%s,\"namespace eval %s { }\");\n", interp_name, ns_name);
319 fprintf(f_init,"#endif\n");
320 }
321 }
322
323 // ---------------------------------------------------------------------
324 // TCL::close(void)
325 //
326 // Wrap things up. Close initialization function.
327 // ---------------------------------------------------------------------
328
329 void TCL::close(void)
330 {
331
332 // Dump the pointer equivalency table
333
334 emit_ptr_equivalence(f_init);
335
336 // Close the init file and quit
337 fprintf(f_init,"%s",postinit.get());
338 fprintf(f_init,"\t return TCL_OK;\n");
339 fprintf(f_init,"}\n");
340 }
341
342 // ----------------------------------------------------------------------
343 // TCL::get_pointer(char *iname, char *srcname, char *src, char *dest,
344 // DataType *t, String &f, char *ret)
345 //
346 // iname = name of function or variable
347 // srcname = name of source
348 // src = source variable in wrapper code
349 // dest = destination variable in wrapper code
350 // t = Datatype
351 // f = String where output is going to go
352 // ret = Return action
353 // ----------------------------------------------------------------------
354
355 void TCL::get_pointer(char *iname, char *srcname, char *src, char *dest,
356 DataType *t, String &f, char *ret) {
357
358 // Pointers are read as hex-strings with encoded type information
359
360 f << tab4 << "if (SWIG_GetPtr(" << src << ",(void **) &" << dest << ",";
361
362 if (t->type == T_VOID) f << "(char *) 0)) {\n";
363 else
364 f << "\"" << t->print_mangle() << "\")) {\n";
365
366 // Now emit code according to the level of strictness desired
367
368 switch(TypeStrict) {
369 case 0: // No type checking
370 f << tab4 << "}\n";
371 break;
372 case 1: // Warning message only
373 f << tab8 << "fprintf(stderr,\"Warning : type mismatch in " << srcname
374 << " of " << iname << ". Expected " << t->print_mangle()
375 << ", received %s\\n\"," << src << ");\n"
376 << tab4 << "}\n";
377 case 2: // Super strict mode.
378 f << tab8 << "Tcl_SetResult(interp, \"Type error in " << srcname << " of " << iname
379 << ". Expected " << t->print_mangle() << ", received \", TCL_STATIC);\n"
380 << tab8 << "Tcl_AppendResult(interp, " << src << ", (char *) NULL);\n"
381 << tab8 << ret << ";\n"
382 << tab4 << "}\n";
383 break;
384 default :
385 fprintf(stderr,"Unknown strictness level\n");
386 break;
387 }
388 }
389
390 // ----------------------------------------------------------------------
391 // TCL::create_command(char *cname, char *iname)
392 //
393 // Creates a Tcl command from a C function.
394 // ----------------------------------------------------------------------
395
396 void TCL::create_command(char *cname, char *iname) {
397
398
399 char *wname = name_wrapper(cname,prefix);
400
401 fprintf(f_init,"\t Tcl_CreateCommand(%s,SWIG_prefix \"%s\",%s, (ClientData) NULL, (Tcl_CmdDeleteProc *) NULL);\n", interp_name, iname, wname);
402
403 // Add interpreter name to repeatcmd hash table. This hash is used in C++ code
404 // generation to try and find repeated wrapper functions.
405
406 repeatcmd.add(iname,copy_string(wname));
407 }
408
409 // ----------------------------------------------------------------------
410 // TCL::create_function(char *name, char *iname, DataType *d, ParmList *l)
411 //
412 // Create a function declaration and register it with the interpreter.
413 // ----------------------------------------------------------------------
414
415 void TCL::create_function(char *name, char *iname, DataType *d, ParmList *l)
416 {
417 Parm *p;
418 int pcount,i,j;
419 char *wname;
420 char *usage = 0;
421 char source[256], target[256], argnum[32], *tm;
422 String cleanup, outarg;
423 int numopt;
424 int have_build = 0; // Status indicating build typemap
425 String build;
426
427 WrapperFunction f;
428
429 // Make a wrapper name for this function
430
431 wname = name_wrapper(iname,prefix);
432
433 // Now write the wrapper function itself....this is pretty ugly
434
435 f.def << "static int " << wname << "(ClientData clientData, Tcl_Interp *interp, int argc, char *argv[]) {\n";
436 f.code << tab4 << "clientData = clientData; argv = argv;\n";
437
438 // Print out variables for storing arguments.
439
440 pcount = emit_args(d, l, f);
441
442 // Get number of optional arguments
443
444 numopt = l->numopt();
445
446 // Check the number of arguments
447
448 usage = usage_func(iname,d,l); // Create a usage string
449 f.code << tab4 << "if ((argc < " << (pcount-numopt) +1 << ") || (argc > " << l->numarg()+1 << ")) {\n"
450 << tab8 << "Tcl_SetResult(interp, \"Wrong # args. " << usage << "\",TCL_STATIC);\n"
451 << tab8 << "return TCL_ERROR;\n"
452 << tab4 << "}\n";
453
454 // Extract parameters. This case statement should be used to extract
455 // Function parameters. Add more cases if you want to do more.
456
457 i = 0;
458 j = 0;
459 p = l->get_first();
460 while (p != 0) {
461 // Produce string representations of the source and target arguments
462 sprintf(source,"argv[%d]",j+1);
463 sprintf(target,"_arg%d",i);
464 sprintf(argnum,"%d",j+1);
465
466 // See if this argument is being ignored
467
468 if (!p->ignore) {
469 if (j >= (pcount-numopt))
470 f.code << tab4 << "if (argc >" << j+1 << ") { \n";
471
472 // See if there is a type-mapping
473
474 if ((tm = typemap_lookup("in","tcl",p->t,p->name,source,target,&f))) {
475 // Yep. Use it instead of the default
476 f.code << tm << "\n";
477 f.code.replace("$argnum",argnum);
478 f.code.replace("$arg",source);
479 } else {
480
481 if (!p->t->is_pointer) {
482
483 // Extract a parameter by value.
484
485 switch(p->t->type) {
486
487 // Signed Integers
488
489 case T_BOOL:
490 case T_INT:
491 case T_SINT:
492 case T_SHORT:
493 case T_SSHORT:
494 case T_LONG:
495 case T_SLONG:
496 case T_SCHAR:
497 f.code << tab4 << "_arg" << i << " = " << p->t->print_cast() << " atol(argv[" << j+1 << "]);\n";
498 break;
499
500 // Unsigned Integers
501
502 case T_UINT:
503 case T_USHORT:
504 case T_ULONG:
505 case T_UCHAR:
506 f.code << tab4 << "_arg" << i << " = " << p->t->print_cast() << " strtoul(argv[" << j+1
507 << "],(char **) NULL,0);\n";
508 break;
509
510 // Floating point
511
512 case T_FLOAT:
513 case T_DOUBLE:
514 f.code << tab4 << "_arg" << i << " = " << p->t->print_cast() << " atof(argv[" << j+1 << "]);\n";
515 break;
516
517 // A single character
518
519 case T_CHAR :
520 f.code << tab4 << "_arg" << i << " = *argv[" << j+1 << "];\n";
521 break;
522
523 // Void.. Do nothing.
524
525 case T_VOID :
526 break;
527
528 // User defined. This is an error.
529
530 case T_USER:
531
532 // Unsupported data type
533
534 default :
535 fprintf(stderr,"%s : Line %d: Unable to use type %s as a function argument.\n",
536 input_file, line_number, p->t->print_type());
537 break;
538 }
539 } else { /* Pointer type */
540
541 // Function argument is some sort of pointer
542 // Look for a string. Otherwise, just pull off a pointer.
543
544 if ((p->t->type == T_CHAR) && (p->t->is_pointer == 1)) {
545 f.code << tab4 << "_arg" << i << " = argv[" << j+1 << "];\n";
546 } else {
547
548 // Have a generic pointer type here. Read it in as
549 // a hex-string
550
551 char arg_temp[256];
552 sprintf(arg_temp,"argument %d",j+1);
553 get_pointer(iname,arg_temp,source,target,p->t,f.code,"return TCL_ERROR");
554 }
555 }
556 }
557 if (j >= (pcount-numopt))
558 f.code << tab4 << "} \n";
559 j++;
560 }
561
562 // Check to see if there is any sort of "build" typemap (highly complicated)
563
564 if ((tm = typemap_lookup("build","tcl",p->t,p->name,source,target))) {
565 build << tm << "\n";
566 have_build = 1;
567 }
568
569 // Check to see if there was any sort of a constaint typemap
570 if ((tm = typemap_lookup("check","tcl",p->t,p->name,source,target))) {
571 // Yep. Use it instead of the default
572 f.code << tm << "\n";
573 f.code.replace("$argnum",argnum);
574 f.code.replace("$arg",source);
575 }
576
577 // Check if there was any cleanup code (save it for later)
578 if ((tm = typemap_lookup("freearg","tcl",p->t,p->name,target,"interp->result"))) {
579 // Yep. Use it instead of the default
580 cleanup << tm << "\n";
581 cleanup.replace("$argnum",argnum);
582 cleanup.replace("$arg",source);
583 }
584 if ((tm = typemap_lookup("argout","tcl",p->t,p->name,target,"interp->result"))) {
585 // Yep. Use it instead of the default
586 outarg << tm << "\n";
587 outarg.replace("$argnum",argnum);
588 outarg.replace("$arg",source);
589 }
590 p = l->get_next(); // Get next parameter and continue
591 i++;
592 }
593
594 // If there was a "build" typemap, we need to go in and perform a serious hack
595
596 if (have_build) {
597 char temp1[32];
598 char temp2[256];
599 l->sub_parmnames(build); // Replace all parameter names
600 j = 1;
601 for (i = 0; i < l->nparms; i++) {
602 p = l->get(i);
603 if (strlen(p->name) > 0) {
604 sprintf(temp1,"_in_%s", p->name);
605 } else {
606 sprintf(temp1,"_in_arg%d", i);
607 }
608 sprintf(temp2,"argv[%d]",j);
609 build.replaceid(temp1,temp2);
610 if (!p->ignore)
611 j++;
612 }
613 f.code << build;
614 }
615
616 // Now write code to make the function call
617
618 emit_func_call(name,d,l,f);
619
620 // Return value if necessary
621
622 if ((tm = typemap_lookup("out","tcl",d,name,"_result","interp->result"))) {
623 // Yep. Use it instead of the default
624 f.code << tm << "\n";
625 } else if ((d->type != T_VOID) || (d->is_pointer)) {
626 // Normal SWIG mappings
627
628 if (!d->is_pointer) {
629
630 // Function returns a "value"
631
632 switch(d->type) {
633 // Is a signed integer
634 case T_BOOL:
635 case T_INT:
636 case T_SINT:
637 case T_SHORT:
638 case T_SSHORT:
639 case T_LONG :
640 case T_SLONG:
641 case T_SCHAR:
642 f.code << tab4 << "sprintf(interp->result,\"%ld\", (long) _result);\n";
643 break;
644
645 // Is an unsigned integer
646 case T_UINT:
647 case T_USHORT:
648 case T_ULONG:
649 case T_UCHAR:
650 f.code << tab4 << "sprintf(interp->result,\"%lu\", (unsigned long) _result);\n";
651 break;
652
653 // Is a single character. Assume we return it as a string
654 case T_CHAR :
655 f.code << tab4 << "sprintf(interp->result,\"%c\", _result);\n";
656 break;
657
658 // Floating point number
659 case T_DOUBLE :
660 case T_FLOAT :
661 f.code << tab4 << "Tcl_PrintDouble(interp,(double) _result, interp->result);\n";
662 //sprintf(interp->result,\"%0.17f\",(double) _result);\n";
663 break;
664
665 // User defined type
666 case T_USER :
667
668 // Okay. We're returning malloced memory at this point.
669 // Probably dangerous, but what the hell
670
671 d->is_pointer++;
672 f.code << tab4 << "SWIG_MakePtr(interp->result, (void *) _result,\"" << d->print_mangle()
673 << "\");\n";
674 d->is_pointer--;
675 break;
676
677 // Unknown type
678 default :
679 fprintf(stderr,"%s : Line %d: Unable to use return type %s in function %s.\n",
680 input_file, line_number, d->print_type(), name);
681 break;
682 }
683 } else {
684
685 // Is a pointer return type
686
687 if ((d->type == T_CHAR) && (d->is_pointer == 1)) {
688 // Return a character string
689 f.code << tab4 << "Tcl_SetResult(interp, (char *) _result, " << char_result << ");\n";
690 } else {
691 f.code << tab4 << "SWIG_MakePtr(interp->result, (void *) _result,\"" << d->print_mangle() << "\");\n";
692 }
693 }
694 }
695
696 // Dump argument output code;
697 f.code << outarg;
698
699 // Dump the argument cleanup code
700 f.code << cleanup;
701
702 // Look for any remaining cleanup
703
704 if (NewObject) {
705 if ((tm = typemap_lookup("newfree","tcl",d,iname,"_result",""))) {
706 f.code << tm << "\n";
707 }
708 }
709
710 if ((tm = typemap_lookup("ret","tcl",d,name,"_result",""))) {
711 // Yep. Use it instead of the default
712 f.code << tm << "\n";
713 }
714
715 // Wrap things up (in a manner of speaking)
716
717 f.code << tab4 << "return TCL_OK;\n}";
718
719 // Substitute the cleanup code
720 f.code.replace("$cleanup",cleanup);
721 f.code.replace("$name",iname);
722
723 // Dump out the function
724
725 f.print(f_wrappers);
726
727 // Add interpreter name to repeatcmd hash table. This hash is used in C++ code
728 // generation to try and find repeated wrapper functions.
729
730 fprintf(f_init,"\t Tcl_CreateCommand(%s, SWIG_prefix \"%s\", %s, (ClientData) NULL, (Tcl_CmdDeleteProc *) NULL);\n", interp_name, iname, wname);
731
732 // If there's a documentation entry, produce a usage string
733
734 if (doc_entry) {
735
736 static DocEntry *last_doc_entry = 0;
737
738 // Use usage as description
739 doc_entry->usage << usage;
740
741 // Set the cinfo field to specific a return type
742
743 if (last_doc_entry != doc_entry) {
744 doc_entry->cinfo << "returns " << d->print_type();
745 last_doc_entry = doc_entry;
746 }
747 }
748 }
749
750 // -----------------------------------------------------------------------
751 // TCL::link_variable(char *name, char *iname, DataType *t,
752 // int ex)
753 //
754 // Create a TCL link to a variable.
755 // -----------------------------------------------------------------------
756
757 void TCL::link_variable(char *name, char *iname, DataType *t)
758 {
759
760 String s;
761 char *tm,*tm1;
762 String realname;
763
764
765 // See if there were any typemaps
766
767 tm = typemap_lookup("varin","tcl",t,name,"","");
768 tm1 = typemap_lookup("varout","tcl",t,name,"","");
769 if (tm || tm1) {
770 fprintf(stderr,"%s : Line %d. Warning. varin/varout typemap methods not supported.",
771 input_file, line_number);
772 }
773
774 // Check the datatype. Must be a valid Tcl type (there aren't many)
775
776 if (((t->type == T_INT) && (!t->is_pointer)) ||
777 ((t->type == T_UINT) && (!t->is_pointer)) ||
778 ((t->type == T_SINT) && (!t->is_pointer)) ||
779 ((t->type == T_DOUBLE) && (!t->is_pointer)) ||
780 ((t->type == T_BOOL) && (!t->is_pointer)) ||
781 ((t->type == T_CHAR) && (t->is_pointer == 1))) {
782
783 // This is a valid TCL type.
784
785 if (t->type == T_UINT)
786 fprintf(stderr,"%s : Line %d : ** Warning. Linkage of unsigned type may be unsafe.\n",
787 input_file, line_number);
788
789 // Now add symbol to the TCL interpreter
790
791 switch(t->type) {
792 case T_CHAR :
793 if (t->arraystr) {
794 // Is an array. We have to do something different
795 fprintf(f_wrappers,"static char *tclvar%s = %s;\n",name,name);
796 s << "(char *) &tclvar" << name << ", TCL_LINK_STRING";
797 } else {
798 s << "(char *) &" << name << ", TCL_LINK_STRING";
799 }
800 break;
801 case T_BOOL:
802 case T_INT :
803 case T_UINT:
804 case T_SINT:
805 s << "(char *) &" << name << ", TCL_LINK_INT";
806 break;
807 case T_DOUBLE :
808 s << "(char *) &" << name << ", TCL_LINK_DOUBLE";
809 break;
810 default :
811 fprintf(f_init,"Fatal error. Internal error (Tcl:link_variable)\n");
812 break;
813 }
814
815 if (Status & STAT_READONLY)
816 s << " | TCL_LINK_READ_ONLY);\n";
817 else
818 s << ");\n";
819
820 fprintf(f_init,"\t Tcl_LinkVar(%s, SWIG_prefix \"%s\", %s",interp_name, iname, s.get());
821
822 // Make a usage string for it
823
824 if (doc_entry) {
825 doc_entry->usage << usage_var(iname,t);
826 doc_entry->cinfo = "";
827 doc_entry->cinfo << "Global : " << t->print_type() << " " << name;
828 }
829 } else {
830
831 // Have some sort of "other" type.
832 // We're going to emit some functions to set/get it's value instead
833
834 emit_set_get(name,iname, t);
835 if (doc_entry) {
836 doc_entry->cinfo = "";
837 doc_entry->cinfo << "Global : " << t->print_type() << " " << iname;
838 }
839
840 // If shadow classes are enabled and we have a user-defined type
841 // that we know about, create a command for it.
842
843 if (shadow) {
844 if ((t->type == T_USER) && (t->is_pointer < 1)) {
845 // See if the datatype is in our hash table
846 if (hash.lookup(t->name)) {
847 // Yep. Try to create a command for it
848 // postinit << tab4 << "Tcl_CreateCommand(interp, SWIG_prefix \"" << iname << "\",Tcl" << (char *) hash.lookup(t->name)
849 // << "MethodCmd, (ClientData)";
850
851 postinit << tab4 << "{\n"
852 << tab8 << "char cmd[] = \""
853 << (char *) hash.lookup(t->name) << " " << iname << " -this ["
854 << iname << "_get ]\";\n"
855 << tab8 << "Tcl_GlobalEval(interp,cmd);\n"
856 << tab4 << "}\n";
857
858 // postinit << tab4 << "Tcl_GlobalEval(interp,\"set " << iname << " [" << iname << "_get ];"
859 // << (char *) hash.lookup(t->name) << " -this $" << iname << "\");\n";
860
861 }
862 }
863 }
864 }
865 }
866
867 // -----------------------------------------------------------------------
868 // TCL::declare_const(char *name, char *iname, DataType *type, char *value)
869 //
870 // Makes a constant. Really just creates a variable and links to it.
871 // Tcl variable linkage allows read-only variables so we'll use that
872 // instead of just creating a Tcl variable.
873 // ------------------------------------------------------------------------
874
875 void TCL::declare_const(char *name, char *, DataType *type, char *value) {
876
877 int OldStatus = Status; // Save old status flags
878 DataType *t;
879 char var_name[256];
880 char *tm;
881 String rvalue;
882
883 Status = STAT_READONLY; // Enable readonly mode.
884
885 // Make a static variable;
886
887 sprintf(var_name,"_wrap_const_%s",name);
888
889 // See if there's a typemap
890
891 rvalue = value;
892 if ((type->type == T_CHAR) && (type->is_pointer == 1)) {
893 rvalue << "\"";
894 "\"" >> rvalue;
895 }
896 if ((type->type == T_CHAR) && (type->is_pointer == 0)) {
897 rvalue << "'";
898 "'" >> rvalue;
899 }
900 if ((tm = typemap_lookup("const","tcl",type,name,rvalue.get(),name))) {
901 // Yep. Use it instead of the default
902 fprintf(f_init,"%s\n",tm);
903 } else {
904
905 // Create variable and assign it a value
906
907 if (type->is_pointer == 0) {
908 switch(type->type) {
909 case T_BOOL: case T_INT: case T_SINT: case T_DOUBLE:
910 fprintf(f_header,"static %s %s = %s;\n", type->print_type(), var_name, value);
911 link_variable(var_name,name,type);
912 break;
913 case T_SHORT:
914 case T_LONG:
915 case T_SSHORT:
916 case T_SCHAR:
917 case T_SLONG:
918 fprintf(f_header,"static %s %s = %s;\n", type->print_type(), var_name, value);
919 fprintf(f_header,"static char *%s_char;\n", var_name);
920 if (CPlusPlus)
921 fprintf(f_init,"\t %s_char = new char[32];\n",var_name);
922 else
923 fprintf(f_init,"\t %s_char = (char *) malloc(32);\n",var_name);
924
925 fprintf(f_init,"\t sprintf(%s_char,\"%%ld\", (long) %s);\n", var_name, var_name);
926 sprintf(var_name,"%s_char",var_name);
927 t = new DataType(T_CHAR);
928 t->is_pointer = 1;
929 link_variable(var_name,name,t);
930 delete t;
931 break;
932 case T_UINT:
933 case T_USHORT:
934 case T_ULONG:
935 case T_UCHAR:
936 fprintf(f_header,"static %s %s = %s;\n", type->print_type(), var_name, value);
937 fprintf(f_header,"static char *%s_char;\n", var_name);
938 if (CPlusPlus)
939 fprintf(f_init,"\t %s_char = new char[32];\n",var_name);
940 else
941 fprintf(f_init,"\t %s_char = (char *) malloc(32);\n",var_name);
942
943 fprintf(f_init,"\t sprintf(%s_char,\"%%lu\", (unsigned long) %s);\n", var_name, var_name);
944 sprintf(var_name,"%s_char",var_name);
945 t = new DataType(T_CHAR);
946 t->is_pointer = 1;
947 link_variable(var_name,name,t);
948 delete t;
949 break;
950 case T_FLOAT:
951 type->type = T_DOUBLE;
952 strcpy(type->name,"double");
953 fprintf(f_header,"static %s %s = %s (%s);\n", type->print_type(), var_name, type->print_cast(), value);
954 link_variable(var_name,name,type);
955 break;
956
957 case T_CHAR:
958 type->is_pointer++;
959 fprintf(f_header,"static %s %s = \"%s\";\n", type->print_type(), var_name, value);
960 link_variable(var_name,name,type);
961 type->is_pointer--;
962 break;
963 default:
964 fprintf(stderr,"%s : Line %d. Unsupported constant value.\n", input_file, line_number);
965 break;
966 }
967 } else {
968 // Have some sort of pointer value here
969 if ((type->type == T_CHAR) && (type->is_pointer == 1)) {
970 // Character string
971 fprintf(f_header,"static %s %s = \"%s\";\n", type->print_type(), var_name, value);
972 link_variable(var_name,name,type);
973 } else {
974 // Something else. Some sort of pointer value
975 fprintf(f_header,"static %s %s = %s;\n", type->print_type(), var_name, value);
976 fprintf(f_header,"static char *%s_char;\n", var_name);
977 if (CPlusPlus)
978 fprintf(f_init,"\t %s_char = new char[%d];\n",var_name,(int) strlen(type->print_mangle())+ 20);
979 else
980 fprintf(f_init,"\t %s_char = (char *) malloc(%d);\n",var_name,(int) strlen(type->print_mangle())+ 20);
981
982 t = new DataType(T_CHAR);
983 t->is_pointer = 1;
984 fprintf(f_init,"\t SWIG_MakePtr(%s_char, (void *) %s,\"%s\");\n",
985 var_name, var_name, type->print_mangle());
986 sprintf(var_name,"%s_char",var_name);
987 link_variable(var_name,name,t);
988 delete t;
989 }
990 }
991 }
992
993 // Create a documentation entry for this
994
995 if (doc_entry) {
996 doc_entry->usage = ""; // Destroy any previous information from linking
997 doc_entry->usage << usage_const(name, type, value);
998 doc_entry->cinfo = "";
999 doc_entry->cinfo << "Constant : " << type->print_type();
1000 }
1001 Status = OldStatus;
1002 }
1003
1004 // ----------------------------------------------------------------------
1005 // TCL::usage_var(char *iname, DataType *t, char **s)
1006 //
1007 // Produces a usage string for a tcl variable. Stores it in s
1008 // ----------------------------------------------------------------------
1009
1010 char *TCL::usage_var(char *iname, DataType *t) {
1011
1012 static String temp;
1013
1014 temp = "";
1015 if (!nspace) {
1016 temp << "$" << prefix << iname;
1017 } else {
1018 temp << ns_name << "::" << iname;
1019 }
1020 if (!(((t->type == T_INT) && (!t->is_pointer)) ||
1021 ((t->type == T_UINT) && (!t->is_pointer)) ||
1022 ((t->type == T_DOUBLE) && (!t->is_pointer)) ||
1023 ((t->type == T_BOOL) && (!t->is_pointer)) ||
1024 ((t->type == T_CHAR) && (t->is_pointer)))) {
1025 /* We emitted a pair of set/get functions instead. Doc will be generated for that */
1026 return temp;
1027 }
1028 return temp;
1029 }
1030
1031
1032 // ---------------------------------------------------------------------------
1033 // char *TCL::usage_string(char *iname, DataType *t, ParmList *l),
1034 //
1035 // Generates a generic usage string for a Tcl function.
1036 // ---------------------------------------------------------------------------
1037
1038 char * TCL::usage_string(char *iname, DataType *, ParmList *l) {
1039
1040 static String temp;
1041 Parm *p;
1042 int i, numopt,pcount;
1043
1044 temp = "";
1045 temp << iname << " ";
1046
1047 /* Now go through and print parameters */
1048 i = 0;
1049 pcount = l->nparms;
1050 numopt = l->numopt();
1051 p = l->get_first();
1052 while (p != 0) {
1053
1054 // Only print an argument if not ignored
1055
1056 if (!typemap_check("ignore","tcl",p->t,p->name)) {
1057 if (i >= (pcount-numopt))
1058 temp << "?";
1059
1060 /* If parameter has been named, use that. Otherwise, just print a type */
1061
1062 if ((p->t->type != T_VOID) || (p->t->is_pointer)) {
1063 if (strlen(p->name) > 0) {
1064 temp << p->name;
1065 }
1066 else {
1067 temp << "{ " << p->t->print_type() << " }";
1068 }
1069 }
1070 if (i >= (pcount-numopt))
1071 temp << "?";
1072 temp << " ";
1073 i++;
1074 }
1075 p = l->get_next();
1076 }
1077 return temp;
1078 }
1079
1080 // ---------------------------------------------------------------------------
1081 // char *TCL::usage_func(char *iname, DataType *t, ParmList *l),
1082 //
1083 // Produces a usage string for a function in Tcl
1084 // ---------------------------------------------------------------------------
1085
1086 char * TCL::usage_func(char *iname, DataType *t, ParmList *l) {
1087
1088 String temp;
1089
1090 if (nspace) {
1091 temp << ns_name << "::" << iname;
1092 } else {
1093 temp << prefix << iname;
1094 }
1095 return usage_string(temp,t,l);
1096 }
1097
1098 // -----------------------------------------------------------------
1099 // TCL::usage_const(char *name, DataType *type, char *value)
1100 // char **s)
1101 //
1102 // Makes a usage string and returns it
1103 // -----------------------------------------------------------------
1104
1105 char *TCL::usage_const(char *name, DataType *, char *value) {
1106 static String temp;
1107 temp = "";
1108 if (nspace) {
1109 temp << ns_name << "::" << name << " = " << value;
1110 } else {
1111 temp << "$" << prefix << name << " = " << value;
1112 }
1113 return temp;
1114 }
1115
1116 // -------------------------------------------------------------------
1117 // TCL::add_native(char *name, char *funcname)
1118 //
1119 // This adds an already written Tcl wrapper function to our
1120 // initialization function.
1121 // -------------------------------------------------------------------
1122
1123
1124 void TCL::add_native(char *name, char *funcname) {
1125
1126 fprintf(f_init,"\t Tcl_CreateCommand(%s, SWIG_prefix \"%s\", %s, (ClientData) NULL, (Tcl_CmdDeleteProc *) NULL);\n",interp_name, name, funcname);
1127
1128 if (doc_entry) {
1129 if (nspace)
1130 doc_entry->usage << ns_name << "::" << name << " args";
1131 else
1132 doc_entry->usage << prefix << name << " args";
1133
1134 doc_entry->cinfo << "Native method : " << funcname;
1135 }
1136
1137 }
1138
1139 // -------------------------------------------------------------------
1140 // TCL::pragma(char *lname, char *name, char *value)
1141 //
1142 // This is an experimental %pragma handler. Officially unsupported
1143 // in this release, but requested in e-mail.
1144 // --------------------------------------------------------------------
1145
1146 void TCL::pragma(char *lname, char *name, char *) {
1147
1148 if (strcmp(lname,"tcl") == 0) {
1149 if (strcmp(name,"dynamic") == 0) {
1150 char_result = "TCL_DYNAMIC";
1151 } else if (strcmp(name,"static") == 0) {
1152 char_result = "TCL_STATIC";
1153 } else if (strcmp(name,"volatile") == 0) {
1154 char_result = "TCL_VOLATILE";
1155 }
1156 }
1157 }
1158
1159
1160 // ---------------------------------------------------------------------
1161 // C++ Handling
1162 //
1163 // The following functions provide some support for C++ classes and
1164 // C structs.
1165 // ---------------------------------------------------------------------
1166
1167 void TCL::cpp_open_class(char *classname, char *rename, char *ctype, int strip) {
1168 char temp[256];
1169 this->Language::cpp_open_class(classname,rename,ctype,strip);
1170
1171 if (shadow) {
1172
1173 config = "";
1174 cget = "";
1175 methods = "";
1176 options = "";
1177 config_options = "";
1178 methodnames = "";
1179
1180 have_constructor = 0;
1181 have_destructor = 0;
1182 have_methods = 0;
1183 have_config = 0;
1184 have_cget = 0;
1185
1186 if (rename)
1187 class_name = copy_string(rename);
1188 else
1189 class_name = copy_string(classname);
1190
1191 base_class = (char *) 0;
1192 if (!strip) {
1193 class_type = new char[strlen(ctype)+2];
1194 sprintf(class_type,"%s ", ctype);
1195 } else
1196 class_type = "";
1197
1198 real_classname = copy_string(classname);
1199
1200 // Build up the hash table
1201
1202 hash.add(real_classname,copy_string(class_name));
1203 sprintf(temp,"%s %s", class_type, real_classname);
1204 hash.add(temp,copy_string(class_name));
1205 }
1206 }
1207
1208 void TCL::cpp_close_class() {
1209 String code;
1210 DataType *t;
1211 String temp;
1212
1213 this->Language::cpp_close_class();
1214 if (shadow) {
1215
1216 t = new DataType;
1217 sprintf(t->name,"%s%s", class_type, real_classname);
1218 t->type = T_USER;
1219 t->is_pointer = 1;
1220
1221 // Note : The object oriented interface is defined by three files
1222 // delcmd.swg - Object deletion wrapper
1223 // methodcmd.swg - Method invocation command
1224 // objcmd.swg - Command to create a new object
1225 //
1226 // These files are located in the SWIG library. This module
1227 // grabs the files and performs marker replacements to
1228 // build the wrapper function.
1229
1230 // Generate a Tcl function for object destruction
1231
1232 if (have_destructor) {
1233 code << delcmd;
1234 }
1235
1236 // Dump out method code
1237 code << methodcmd;
1238
1239 // Dump out object creation command
1240 code << objcmd;
1241
1242 // Now perform marker replacements
1243 code.replace("@CLASS@",class_name);
1244 temp = "";
1245 temp << name_destroy(class_name);
1246 code.replace("@DESTRUCTOR@",temp);
1247 code.replace("@CLASSTYPE@",t->print_type());
1248 "configure " >> methodnames;
1249 "cget " >> methodnames;
1250 code.replace("@METHODLIST@", methodnames);
1251 code.replace("@CLASSMANGLE@",t->print_mangle());
1252 code.replace("@METHODS@",methods);
1253 code.replace("@CONFIGMETHODS@",config);
1254 code.replace("@CGETMETHODS@",cget);
1255 if (have_constructor) {
1256 temp = "";
1257 temp << name_wrapper(name_construct(class_name),prefix);
1258 } else {
1259 temp = "0";
1260 }
1261 code.replace("@TCLCONSTRUCTOR@",temp);
1262 code.replace("@CONFIGLIST@",config_options);
1263 code.replace("@CGETLIST@",options);
1264 if (have_destructor) {
1265 temp = "TclDelete";
1266 temp << class_name;
1267 } else {
1268 temp = "0";
1269 }
1270 code.replace("@TCLDESTRUCTOR@",temp);
1271 fprintf(f_wrappers,"%s\n", code.get());
1272 fprintf(f_init,"\t Tcl_CreateCommand(interp, SWIG_prefix \"%s\",Tcl%sCmd, (ClientData) NULL, NULL);\n", class_name, class_name);
1273 }
1274 }
1275
1276 void TCL::cpp_member_func(char *name, char *iname, DataType *t, ParmList *l) {
1277
1278 char *realname;
1279 String temp;
1280 char *rname;
1281
1282 this->Language::cpp_member_func(name,iname,t,l);
1283
1284 if (shadow) {
1285 if (iname)
1286 realname = iname;
1287 else
1288 realname = name;
1289
1290 // Add stubs for this member to our class handler function
1291
1292 if (have_methods)
1293 methods << tab4 << "else ";
1294 else
1295 methods << tab4;
1296
1297 // Try to figure out if there is a wrapper for this function
1298
1299 temp = "";
1300 temp << name_member(realname,class_name);
1301
1302 rname = (char *) repeatcmd.lookup(temp);
1303
1304 if (!rname)
1305 rname = name_wrapper(temp.get(),prefix);
1306
1307 methods << "if (strcmp(argv[1],\"" << realname << "\") == 0) {\n"
1308 << tab4 << tab4 << "cmd = " << rname << ";\n"
1309 << tab4 << "}";
1310
1311 have_methods = 1;
1312 methodnames << realname << " ";
1313
1314 if (doc_entry) {
1315 doc_entry->usage = "";
1316 doc_entry->usage << usage_string(realname,t,l);
1317 }
1318 }
1319 }
1320
1321 void TCL::cpp_variable(char *name, char *iname, DataType *t) {
1322 char *realname;
1323 String temp, temp1;
1324 char *rname;
1325
1326 this->Language::cpp_variable(name, iname, t);
1327
1328 if (shadow) {
1329 if (iname)
1330 realname = iname;
1331 else
1332 realname = name;
1333
1334 // Write config code
1335
1336 char *bc = class_name;
1337
1338 if (!(Status & STAT_READONLY)) {
1339 if (!have_config) {
1340 config << tab8 << tab8;
1341 } else {
1342 config << " else ";
1343 }
1344
1345 temp = "";
1346 temp << name_set(name_member(realname,bc));
1347 rname = (char *) repeatcmd.lookup(temp);
1348 if (!rname)
1349 rname = name_wrapper(temp.get(),prefix);
1350
1351 config << "if (strcmp(argv[i],\"-" << realname << "\") == 0) {\n"
1352 << tab8 << tab8 << tab4 << "cmd = " << rname << ";\n"
1353 << tab8 << tab8 << "} ";
1354
1355 have_config = 1;
1356 }
1357
1358 // Write cget code
1359
1360 if (!have_cget) {
1361 cget << tab8 << tab8;
1362 } else {
1363 cget << " else ";
1364 }
1365
1366 // Try to figure out if there is a wrapper for this function
1367 temp = "";
1368 temp << name_get(name_member(realname,bc));
1369 rname = (char *) repeatcmd.lookup(temp);
1370 if (!rname)
1371 rname = name_wrapper(temp.get(),prefix);
1372
1373 cget << "if (strcmp(argv[2],\"-" << realname << "\") == 0) {\n"
1374 << tab8 << tab8 << tab4 << "cmd = " << rname << ";\n"
1375 << tab8 << tab8 << "} ";
1376 have_cget = 1;
1377
1378 options << "-" << realname << " ";
1379 if (!(Status & STAT_READONLY)) {
1380 config_options << "-" << realname << " ";
1381 }
1382 if (doc_entry){
1383 doc_entry->usage = "";
1384 doc_entry->usage << "-" << realname << "\n";
1385 }
1386 }
1387 }
1388
1389 void TCL::cpp_constructor(char *name, char *iname, ParmList *l) {
1390 this->Language::cpp_constructor(name,iname,l);
1391
1392 if (shadow) {
1393 if ((!have_constructor) && (doc_entry)) {
1394 doc_entry->usage = "";
1395 doc_entry->usage << class_name << usage_string(" name",0,l);
1396 }
1397 have_constructor = 1;
1398 }
1399 }
1400 void TCL::cpp_destructor(char *name, char *newname) {
1401 this->Language::cpp_destructor(name,newname);
1402 if (shadow) {
1403 if (!have_destructor) {
1404 if (doc_entry) {
1405 doc_entry->usage = "rename obj {}";
1406 }
1407 }
1408 have_destructor = 1;
1409 }
1410 }
1411
1412 void TCL::cpp_inherit(char **baseclass, int) {
1413 this->Language::cpp_inherit(baseclass);
1414 }
1415
1416 void TCL::cpp_declare_const(char *name, char *iname, DataType *type, char *value) {
1417 this->Language::cpp_declare_const(name,iname,type,value);
1418 }
1419
1420
1421 // --------------------------------------------------------------------------------
1422 // TCL::add_typedef(DataType *t, char *name)
1423 //
1424 // This is called whenever a typedef is encountered. When shadow classes are
1425 // used, this function lets us discovered hidden uses of a class. For example :
1426 //
1427 // struct FooBar {
1428 // ...
1429 // }
1430 //
1431 // typedef FooBar *FooBarPtr;
1432 //
1433 // --------------------------------------------------------------------------------
1434
1435 void TCL::add_typedef(DataType *t, char *name) {
1436
1437 if (!shadow) return;
1438
1439 // First check to see if there aren't too many pointers
1440
1441 if (t->is_pointer > 1) return;
1442 if (hash.lookup(name)) return; // Already added
1443
1444 // Now look up the datatype in our shadow class hash table
1445
1446 if (hash.lookup(t->name)) {
1447
1448 // Yep. This datatype is in the hash
1449 // Put this types 'new' name into the hash
1450 hash.add(name,copy_string((char *) hash.lookup(t->name)));
1451 }
1452 }
1453
1454 // -----------------------------------------------------------------------
1455 // TCL::cpp_class_decl(char *name, char *rename, char *type)
1456 //
1457 // Treatment of an empty class definition. Used to handle
1458 // shadow classes across modules.
1459 // -----------------------------------------------------------------------
1460
1461 void TCL::cpp_class_decl(char *name, char *rename, char *type) {
1462 char temp[256];
1463 this->Language::cpp_class_decl(name,rename, type);
1464 if (shadow) {
1465 hash.add(name,copy_string(rename));
1466 // Add full name of datatype to the hash table
1467 if (strlen(type) > 0) {
1468 sprintf(temp,"%s %s", type, name);
1469 hash.add(temp,copy_string(rename));
1470 }
1471 }
1472 }