https://swig.svn.sourceforge.net/svnroot/swig/trunk ................ r10487 | wsfulton | 2008-05-21 23:15:52 +0100 (Wed, 21 May 2008) | 1 line Apply patch from Petr Viktorin to fix some recent regressions in CFFI/CLOS ................ r10498 | talby | 2008-05-26 21:09:56 +0100 (Mon, 26 May 2008) | 2 lines run test cases in the Perl set by the --with-perl5 configure option. ................ r10499 | talby | 2008-05-26 22:04:06 +0100 (Mon, 26 May 2008) | 3 lines The perl5 minherit runtime test will work better if the classes are actually built under SWIGPERL. ................ r10503 | wsfulton | 2008-05-28 10:44:37 +0100 (Wed, 28 May 2008) | 1 line Fix variable wrappers when using -proxy. Patch from Jan Jezabek ................ r10504 | bhy | 2008-05-28 18:27:48 +0100 (Wed, 28 May 2008) | 2 lines Fixed SF #1971977: typo in pycontainer.swg (related to -extranative option) ................ r10506 | wsfulton | 2008-05-29 01:45:28 +0100 (Thu, 29 May 2008) | 1 line Fix variable wrappers when using -noproxy ................ r10508 | bhy | 2008-05-30 14:53:33 +0100 (Fri, 30 May 2008) | 1 line Fixed SF #1976978, apply the macros for primitive types to std::wstring ................ r10511 | olly | 2008-05-30 17:11:27 +0100 (Fri, 30 May 2008) | 4 lines Fix typo in handling of /*@SWIG[...]*/ comments in the scanner. This just meant we were only actually looking for /*@SWI at the start of the comment, so was pretty harmless in practice. ................ r10515 | wsfulton | 2008-06-02 21:10:40 +0100 (Mon, 02 Jun 2008) | 1 line Fix samename testcase for c# and java ................ r10516 | wsfulton | 2008-06-02 21:15:39 +0100 (Mon, 02 Jun 2008) | 1 line Fix enums when using -noproxy ................ r10518 | bhy | 2008-06-07 12:20:07 +0100 (Sat, 07 Jun 2008) | 4 lines Added a test case for keyword renaming. Now it works for Python in SWIG's -c++ mode, but in C mode it doesn't work! (you can try with make keyword_rename.ctest) ................ r10519 | bhy | 2008-06-07 14:40:51 +0100 (Sat, 07 Jun 2008) | 1 line fixed keyword_rename.ctest tese case, caused by a mistake in Swig/naming.c ................ r10527 | mgossage | 2008-06-17 03:57:15 +0100 (Tue, 17 Jun 2008) | 1 line [lua] bugfix 1938142 (bool& and bool* support) ................ r10530 | wsfulton | 2008-06-19 21:02:13 +0100 (Thu, 19 Jun 2008) | 1 line Add R keyword support. Rename keywords for successful compilation of Java and C# code. More consistent keyword warnings across the different languages. ................ r10531 | wsfulton | 2008-06-19 22:15:48 +0100 (Thu, 19 Jun 2008) | 1 line add complete list of R reserved words ................ r10536 | wsfulton | 2008-06-21 12:35:33 +0100 (Sat, 21 Jun 2008) | 1 line better terminology for static types ................ r10537 | wsfulton | 2008-06-21 12:42:48 +0100 (Sat, 21 Jun 2008) | 1 line remove raise as keyword test- it conflicts with _raise in LIBCMT on windows ................ r10539 | wsfulton | 2008-06-21 16:21:29 +0100 (Sat, 21 Jun 2008) | 1 line Lua example warning removal fixes for vc++ ................ r10540 | wsfulton | 2008-06-21 16:23:02 +0100 (Sat, 21 Jun 2008) | 1 line Remove some vc++ /W4 warnings ................ r10541 | wsfulton | 2008-06-21 17:04:55 +0100 (Sat, 21 Jun 2008) | 1 line minor vc++ /W4 warning fixes ................ r10542 | wsfulton | 2008-06-21 20:07:51 +0100 (Sat, 21 Jun 2008) | 1 line 'byte' is already used in Ruby on windows, so use another keyword ................ r10543 | wsfulton | 2008-06-21 21:45:32 +0100 (Sat, 21 Jun 2008) | 1 line Fix crashing in the Ruby reject method in the STL wrappers ................ r10544 | wsfulton | 2008-06-21 21:48:28 +0100 (Sat, 21 Jun 2008) | 1 line Fix crashing in the Ruby reject method in the STL wrappers ................ r10545 | wsfulton | 2008-06-21 21:49:10 +0100 (Sat, 21 Jun 2008) | 1 line remove unnecessary variable int the char **STRING_ARRAY out typemap ................ r10546 | wsfulton | 2008-06-21 22:07:49 +0100 (Sat, 21 Jun 2008) | 1 line Fix Ruby C++ example dependencies in dsp files ................ r10547 | wsfulton | 2008-06-21 23:25:36 +0100 (Sat, 21 Jun 2008) | 1 line Fix unused parameter warnings in python when using gcc's -W -Wall options ................ r10548 | wsfulton | 2008-06-21 23:26:35 +0100 (Sat, 21 Jun 2008) | 1 line Fix virtual destructor ................ r10549 | wsfulton | 2008-06-22 00:25:20 +0100 (Sun, 22 Jun 2008) | 1 line various warning fixes ................ r10550 | wsfulton | 2008-06-22 01:09:11 +0100 (Sun, 22 Jun 2008) | 1 line Another fix for the JVM hanging on exit problem when using directors ................ r10551 | wsfulton | 2008-06-22 01:09:51 +0100 (Sun, 22 Jun 2008) | 1 line documentation sections update ................ r10552 | wsfulton | 2008-06-22 01:18:10 +0100 (Sun, 22 Jun 2008) | 1 line more docs on defining macros for the thread hanging problem ................ r10558 | wsfulton | 2008-06-22 22:30:20 +0100 (Sun, 22 Jun 2008) | 1 line fix unused parms in last commit for C code ................ r10559 | wsfulton | 2008-06-22 23:12:43 +0100 (Sun, 22 Jun 2008) | 1 line Suppress unused methods warning for VC++ ................ r10560 | wsfulton | 2008-06-23 21:26:07 +0100 (Mon, 23 Jun 2008) | 1 line fix partialcheck-test-suite and parallel make for r, chicken, tcl and php ................ r10561 | wsfulton | 2008-06-23 21:39:41 +0100 (Mon, 23 Jun 2008) | 1 line correct message display when running the partialcheck-test-suite make target ................ r10562 | wsfulton | 2008-06-23 22:14:53 +0100 (Mon, 23 Jun 2008) | 1 line fix typo ................ r10563 | olly | 2008-06-23 22:23:54 +0100 (Mon, 23 Jun 2008) | 3 lines Fix bad use of Python API (untested, since I can't even compile this code on x86-64!) ................ r10564 | olly | 2008-06-23 23:58:03 +0100 (Mon, 23 Jun 2008) | 3 lines [PHP] Fix segfault when wrapping a non-class function marked with %newobject (testcase char_strings). ................ r10565 | olly | 2008-06-24 01:27:34 +0100 (Tue, 24 Jun 2008) | 3 lines [PHP] Fix assertion failure when handling %typemap(in,numinputs=0) (testcase ignore_parameter). ................ r10566 | olly | 2008-06-24 01:33:08 +0100 (Tue, 24 Jun 2008) | 2 lines [PHP] Fix typemap_namespace.i to not try to copy a non-existent typemap. ................ r10567 | olly | 2008-06-24 01:41:07 +0100 (Tue, 24 Jun 2008) | 3 lines Clean up dead and unused code in SwigToPhpType(), and rename to GetShadowReturnType(). ................ r10568 | olly | 2008-06-24 01:42:29 +0100 (Tue, 24 Jun 2008) | 2 lines Fix cosmetic typo in string constant. ................ r10574 | wsfulton | 2008-06-24 21:10:28 +0100 (Tue, 24 Jun 2008) | 1 line zap last entry ................ r10575 | wsfulton | 2008-06-24 21:11:46 +0100 (Tue, 24 Jun 2008) | 1 line variable name changes to remove php keywords ................ r10576 | wsfulton | 2008-06-24 21:12:08 +0100 (Tue, 24 Jun 2008) | 1 line variable name hiding fix ................ r10577 | wsfulton | 2008-06-24 21:12:43 +0100 (Tue, 24 Jun 2008) | 1 line More info about numobjects added ................ r10578 | wsfulton | 2008-06-24 21:13:41 +0100 (Tue, 24 Jun 2008) | 1 line update for 1.3.36 release ................ r10579 | wsfulton | 2008-06-24 22:48:46 +0100 (Tue, 24 Jun 2008) | 1 line remove deprecated -c commandline option (runtime library generation) ................ r10580 | wsfulton | 2008-06-24 22:53:12 +0100 (Tue, 24 Jun 2008) | 1 line correct comment about deprecated option ................ r10582 | wsfulton | 2008-06-25 00:00:27 +0100 (Wed, 25 Jun 2008) | 1 line use rsync and ssh to upload releases to SourceForge as ftp no longer works ................ r10584 | wsfulton | 2008-06-25 00:24:48 +0100 (Wed, 25 Jun 2008) | 1 line correction for 1.3.36 ................ r10588 | wsfulton | 2008-06-25 01:16:04 +0100 (Wed, 25 Jun 2008) | 1 line section update ................ r10589 | wsfulton | 2008-06-25 01:16:40 +0100 (Wed, 25 Jun 2008) | 1 line bump version to 1.3.37 ................ r10594 | wsfulton | 2008-06-26 19:33:06 +0100 (Thu, 26 Jun 2008) | 1 line correct typo in first entry about %fragment ................ r10607 | wsfulton | 2008-06-29 01:19:05 +0100 (Sun, 29 Jun 2008) | 1 line fix some potential null pointer usage as reported by CoveriCoverity Prevent ................ r10608 | wsfulton | 2008-06-29 01:50:27 +0100 (Sun, 29 Jun 2008) | 1 line fix potential null pointer usage as reported by Coverity Prevent ................ r10609 | wsfulton | 2008-06-29 10:57:41 +0100 (Sun, 29 Jun 2008) | 1 line make life easier for svn status updates as everyone has different autotool versions installed - use svn:ignore on config.sub and config.guess ................ r10610 | wsfulton | 2008-06-29 11:08:14 +0100 (Sun, 29 Jun 2008) | 1 line make life easier for svn status updates as everyone has different autotool versions installed - use svn:ignore on config.sub and config.guess ................ r10616 | talby | 2008-07-01 00:41:27 +0100 (Tue, 01 Jul 2008) | 2 lines Hopefully this provides more portable values for Infinity and NaN in Perl. ................ r10621 | wsfulton | 2008-07-01 23:02:14 +0100 (Tue, 01 Jul 2008) | 1 line update old commandline option ................ r10624 | olly | 2008-07-02 04:17:46 +0100 (Wed, 02 Jul 2008) | 5 lines [Python] Import the C extension differently for Python 2.6 and later so that an implicit relative import doesn't produce a deprecation warning for 2.6 and a failure for 2.7 and later. Patch from Richard Boulton in SF#2008229. ................ r10625 | bhy | 2008-07-02 05:56:11 +0100 (Wed, 02 Jul 2008) | 1 line fix the relative import patch by try both relative and absolute import ................ r10626 | olly | 2008-07-02 06:55:18 +0100 (Wed, 02 Jul 2008) | 4 lines [PHP4] Support for PHP4 has been removed. The PHP developers are no longer making new PHP4 releases, and won't even be patching critical security issues after 2008-08-08. ................ r10627 | bhy | 2008-07-02 07:06:56 +0100 (Wed, 02 Jul 2008) | 1 line fix the previous commit -- new approach using imp module ................ r10628 | bhy | 2008-07-02 07:29:15 +0100 (Wed, 02 Jul 2008) | 1 line minor fix on relative import: close fp if exception occured during imp.find_module ................ r10629 | bhy | 2008-07-02 08:08:00 +0100 (Wed, 02 Jul 2008) | 1 line reconstruct the relative import, now the generated code looks better ................ r10633 | olly | 2008-07-03 01:09:56 +0100 (Thu, 03 Jul 2008) | 9 lines WARN_* constants are user visible, so keep existing WARN_PHP4_* for backward compatibility, but add preferred forms WARN_PHP_* and use these ourselves. Rename Lib/php4 to Lib/php, Source/Modules/php4.cxx to Source/Modules/php.cxx. Add typemaps for const reference so Examples/test-suite/apply_signed_char.i works. ................ r10634 | olly | 2008-07-03 01:12:58 +0100 (Thu, 03 Jul 2008) | 2 lines Remove file added in error. ................ r10635 | olly | 2008-07-03 01:37:38 +0100 (Thu, 03 Jul 2008) | 2 lines Rename php4*.swg to php*.swg. ................ r10636 | olly | 2008-07-03 02:13:51 +0100 (Thu, 03 Jul 2008) | 2 lines Update wrt PHP4. ................ r10637 | olly | 2008-07-03 03:32:11 +0100 (Thu, 03 Jul 2008) | 4 lines [PHP] The deprecated command line option "-phpfull" has been removed. We recommend building your extension as a dynamically loadable module. ................ r10638 | olly | 2008-07-03 03:32:41 +0100 (Thu, 03 Jul 2008) | 2 lines Fix version number ................ r10639 | olly | 2008-07-03 07:04:24 +0100 (Thu, 03 Jul 2008) | 2 lines Make t_output_helper() static. ................ r10640 | olly | 2008-07-03 07:05:34 +0100 (Thu, 03 Jul 2008) | 2 lines Make testcase li_typemaps work for PHP. ................ r10641 | olly | 2008-07-03 10:12:16 +0100 (Thu, 03 Jul 2008) | 2 lines Fix check::classname() to compare case-insensitively and not try to interpolate objects into strings. ................ r10643 | olly | 2008-07-03 23:00:01 +0100 (Thu, 03 Jul 2008) | 4 lines The operator& trick in allowexcept.i doesn't work for SWIG/PHP because the generated code takes the address of the variable in the code in the "vinit" section. So comment out the private operator& for PHP. ................ r10647 | olly | 2008-07-04 00:55:33 +0100 (Fri, 04 Jul 2008) | 2 lines [PHP] The SWIG cdata.i library module is now supported. ................ r10649 | olly | 2008-07-04 02:02:22 +0100 (Fri, 04 Jul 2008) | 4 lines [PHP] The deprecated command line option "-make" has been removed. Searches on Google codesearch suggest that nobody is using it now anyway. ................ r10650 | olly | 2008-07-04 02:22:11 +0100 (Fri, 04 Jul 2008) | 4 lines [PHP] For std_vector.i and std_map.i, rename empty() to is_empty() since "empty" is a PHP reserved word. Based of patch from Mark Klein in SF#1943417. ................ r10662 | wsfulton | 2008-07-14 22:09:23 +0100 (Mon, 14 Jul 2008) | 1 line Fix director typemaps for pointers so that NULL pointers are correctly marshalled to C#/Java null in director methods ................ r10672 | olly | 2008-07-17 02:17:08 +0100 (Thu, 17 Jul 2008) | 2 lines Fix a typo; improve wording. ................ r10676 | wsfulton | 2008-07-17 22:05:49 +0100 (Thu, 17 Jul 2008) | 1 line SF #2019156 Configuring with --without-octave or --without-alllang did not disable octave. ................ r10677 | wsfulton | 2008-07-17 22:08:22 +0100 (Thu, 17 Jul 2008) | 1 line tidy output after detecting X11 headers ................ r10682 | wsfulton | 2008-07-19 19:08:06 +0100 (Sat, 19 Jul 2008) | 1 line a bit more on pgcpp ................ r10683 | wsfulton | 2008-07-19 23:45:54 +0100 (Sat, 19 Jul 2008) | 1 line Fix building of Tcl examples/test-suite on Mac OSX ................ r10714 | olly | 2008-07-27 13:51:16 +0100 (Sun, 27 Jul 2008) | 2 lines Fix "can can" typo in docs (SF#2026756) ................ r10726 | wuzzeb | 2008-08-02 09:28:02 +0100 (Sat, 02 Aug 2008) | 1 line Commit patch 2019314 ................ r10727 | wuzzeb | 2008-08-02 09:49:43 +0100 (Sat, 02 Aug 2008) | 1 line add assert.h to fix chicken build of external runtime (ext_test testsuite) ................ r10738 | talby | 2008-08-07 07:28:13 +0100 (Thu, 07 Aug 2008) | 2 lines hoist globals to local scope where trival. ................ r10743 | talby | 2008-08-08 04:10:55 +0100 (Fri, 08 Aug 2008) | 2 lines initial steps to clean up perl5 class methods (primarily constructors). ................ r10747 | talby | 2008-08-09 05:08:26 +0100 (Sat, 09 Aug 2008) | 2 lines moves perl space constructor fixups into wrapper code. ................ r10749 | talby | 2008-08-10 00:57:55 +0100 (Sun, 10 Aug 2008) | 2 lines usage_func() fix + CHANGES.current entry to explain my past few commits. ................ r10794 | talby | 2008-08-19 08:09:29 +0100 (Tue, 19 Aug 2008) | 3 lines rollback 10737:10749, this work is better left on a branch until it is in support a significant feature enhancement. ................ r10801 | drjoe | 2008-08-31 17:07:43 +0100 (Sun, 31 Aug 2008) | 2 lines change Rf_warning to warning. Looks like a typo ................ r10802 | wsfulton | 2008-09-01 20:59:23 +0100 (Mon, 01 Sep 2008) | 1 line Fix ruby detection problem breaking configure ................ r10803 | wsfulton | 2008-09-01 21:00:28 +0100 (Mon, 01 Sep 2008) | 1 line Fix another perl global namespace pollution ................ r10804 | wsfulton | 2008-09-01 21:32:53 +0100 (Mon, 01 Sep 2008) | 1 line Fix array bounds checking in C# std::vector wrappers ................ r10805 | wsfulton | 2008-09-01 21:34:49 +0100 (Mon, 01 Sep 2008) | 1 line fix runtime problem in Ruby std_vector example ................ r10806 | wsfulton | 2008-09-01 21:42:40 +0100 (Mon, 01 Sep 2008) | 1 line Fix portability problem using /dev/null ................ r10807 | wsfulton | 2008-09-01 22:46:41 +0100 (Mon, 01 Sep 2008) | 1 line patch #2079381 submitted by Boris Smilga - constant exprs put into no-eval context in DEFCENUM ................ r10808 | wsfulton | 2008-09-02 21:56:52 +0100 (Tue, 02 Sep 2008) | 1 line Add new terminator attribute for the csin typemap for C# ................ r10809 | wsfulton | 2008-09-03 00:01:37 +0100 (Wed, 03 Sep 2008) | 1 line Fix #1988296 - Multiple module director linking issue ................ r10827 | richardb | 2008-09-10 11:08:21 +0100 (Wed, 10 Sep 2008) | 6 lines [Python] Commit patch #2089149: Director exception handling mangles returned exception. Exceptions raised by Python code in directors are now passed through to the caller without change. Also, remove the ": " prefix which used to be added to other director exceptions (eg, those due to incorrect return types). ................ r10828 | bhy | 2008-09-10 11:45:33 +0100 (Wed, 10 Sep 2008) | 3 lines Initialized merge tracking via "svnmerge" with revisions "1-10827" from https://swig.svn.sourceforge.net/svnroot/swig/branches/gsoc2008-bhy ................ r10829 | bhy | 2008-09-10 12:04:09 +0100 (Wed, 10 Sep 2008) | 4 lines Removed merge tracking for "svnmerge" for https://swig.svn.sourceforge.net/svnroot/swig/branches/gsoc2008-bhy (Undo the init due to user error) ................ r10830 | bhy | 2008-09-10 12:08:16 +0100 (Wed, 10 Sep 2008) | 3 lines Initialized merge tracking via "svnmerge" with revisions "1-10403" from https://swig.svn.sourceforge.net/svnroot/swig/branches/gsoc2008-bhy ................ r10831 | bhy | 2008-09-10 12:51:12 +0100 (Wed, 10 Sep 2008) | 11 lines (Still trying to set right properity to get svnmerge work...) Recorded merge of revisions 10404 via svnmerge from https://swig.svn.sourceforge.net/svnroot/swig/branches/gsoc2008-bhy ........ r10404 | bhy | 2008-05-01 08:35:49 +0800 (Thu, 01 May 2008) | 5 lines Start of gsoc2008-bhy branch. This is a branch for Haoyu Bai's Python 3.0 backend project. Some file already modified since a little of work already done when starting the branch. ........ ................ r10834 | bhy | 2008-09-11 18:18:07 +0100 (Thu, 11 Sep 2008) | 8 lines Merged the Python 3.0 support branch. The merging progress is not so smooth, so hope this commit won't make anything broken. This is the (incomplemete) log produced by svnmerge.py: Merged revisions 10405-10409,10420-10422,10426,10438,10445,10451,10454-10465,10467,10473-10475,10485,10488-10489,10493-10495,10497,10509-10510,10513-10514,10517,10520,10525,10528-10529,10533-10535,10554-10557,10570,10573,10593,10614,10666-10669,10673,10678,10687,10690,10704-10706,10731,10744,10750-10752,10755,10759,10770,10775-10776,10813,10819 via svnmerge from https://swig.svn.sourceforge.net/svnroot/swig/branches/gsoc2008-bhy ................ git-svn-id: https://swig.svn.sourceforge.net/svnroot/swig/branches/gsoc2008-cherylfoil@10837 626c5289-ae23-0410-ae9c-e8d60b6d4f22
1065 lines
32 KiB
C++
1065 lines
32 KiB
C++
/* -----------------------------------------------------------------------------
|
|
* See the LICENSE file for information on copyright, usage and redistribution
|
|
* of SWIG, and the README file for authors - http://www.swig.org/release.html.
|
|
*
|
|
* cffi.cxx
|
|
*
|
|
* cffi language module for SWIG.
|
|
* ----------------------------------------------------------------------------- */
|
|
|
|
char cvsroot_cffi_cxx[] = "$Id$";
|
|
|
|
#include "swigmod.h"
|
|
#include "cparse.h"
|
|
#include <ctype.h>
|
|
|
|
//#define CFFI_DEBUG
|
|
//#define CFFI_WRAP_DEBUG
|
|
|
|
class CFFI:public Language {
|
|
public:
|
|
String *f_cl;
|
|
String *f_clhead;
|
|
String *f_clwrap;
|
|
bool CWrap; // generate wrapper file for C code?
|
|
File *f_cxx;
|
|
File *f_cxx_header;
|
|
File *f_cxx_wrapper;
|
|
File *f_clos;
|
|
|
|
String *module;
|
|
virtual void main(int argc, char *argv[]);
|
|
virtual int top(Node *n);
|
|
virtual int functionWrapper(Node *n);
|
|
virtual int variableWrapper(Node *n);
|
|
virtual int constantWrapper(Node *n);
|
|
// virtual int classDeclaration(Node *n);
|
|
virtual int enumDeclaration(Node *n);
|
|
virtual int typedefHandler(Node *n);
|
|
|
|
//c++ specific code
|
|
virtual int constructorHandler(Node *n);
|
|
virtual int destructorHandler(Node *n);
|
|
virtual int memberfunctionHandler(Node *n);
|
|
virtual int membervariableHandler(Node *n);
|
|
virtual int classHandler(Node *n);
|
|
|
|
private:
|
|
void emit_defun(Node *n, String *name);
|
|
void emit_defmethod(Node *n);
|
|
void emit_initialize_instance(Node *n);
|
|
void emit_getter(Node *n);
|
|
void emit_setter(Node *n);
|
|
void emit_class(Node *n);
|
|
void emit_struct_union(Node *n, bool un);
|
|
void emit_export(Node *n, String *name);
|
|
void emit_inline(Node *n, String *name);
|
|
String *lispy_name(char *name);
|
|
String *lispify_name(Node *n, String *ty, const char *flag, bool kw = false);
|
|
String *convert_literal(String *num_param, String *type, bool try_to_split = true);
|
|
String *infix_to_prefix(String *val, char split_op, const String *op, String *type);
|
|
String *strip_parens(String *string);
|
|
String *trim(String *string);
|
|
int generate_typedef_flag;
|
|
bool no_swig_lisp;
|
|
};
|
|
|
|
void CFFI::main(int argc, char *argv[]) {
|
|
int i;
|
|
|
|
SWIG_library_directory("cffi");
|
|
SWIG_config_file("cffi.swg");
|
|
generate_typedef_flag = 0;
|
|
no_swig_lisp = false;
|
|
CWrap = false;
|
|
for (i = 1; i < argc; i++) {
|
|
if (!Strcmp(argv[i], "-help")) {
|
|
Printf(stdout, "cffi Options (available with -cffi)\n");
|
|
Printf(stdout,
|
|
" -generate-typedef\n"
|
|
"\tIf this option is given then defctype will be used to generate\n"
|
|
"\tshortcuts according to the typedefs in the input.\n"
|
|
" -[no]cwrap\n"
|
|
"\tTurn on or turn off generation of an intermediate C file when\n"
|
|
"\tcreating a C interface. By default this is only done for C++ code.\n"
|
|
" -[no]swig-lisp\n"
|
|
"\tTurns on or off generation of code for helper lisp macro, functions,\n"
|
|
"\tetc. which SWIG uses while generating wrappers. These macros, functions\n" "\tmay still be used by generated wrapper code.\n");
|
|
} else if (!strcmp(argv[i], "-cwrap")) {
|
|
CWrap = true;
|
|
Swig_mark_arg(i);
|
|
} else if ((Strcmp(argv[i], "-generate-typedef") == 0)) {
|
|
generate_typedef_flag = 1;
|
|
Swig_mark_arg(i);
|
|
} else if (!strcmp(argv[i], "-nocwrap")) {
|
|
CWrap = false;
|
|
Swig_mark_arg(i);
|
|
} else if (!strcmp(argv[i], "-swig-lisp")) {
|
|
no_swig_lisp = false;
|
|
Swig_mark_arg(i);
|
|
} else if (!strcmp(argv[i], "-noswig-lisp")) {
|
|
no_swig_lisp = true;
|
|
Swig_mark_arg(i);
|
|
}
|
|
|
|
}
|
|
f_clhead = NewString("");
|
|
f_clwrap = NewString("");
|
|
f_cl = NewString("");
|
|
|
|
allow_overloading();
|
|
}
|
|
|
|
int CFFI::top(Node *n) {
|
|
File *f_null = NewString("");
|
|
module = Getattr(n, "name");
|
|
|
|
String *cxx_filename = Getattr(n, "outfile");
|
|
String *lisp_filename = NewString("");
|
|
|
|
Printf(lisp_filename, "%s%s.lisp", SWIG_output_directory(), module);
|
|
|
|
File *f_lisp = NewFile(lisp_filename, "w");
|
|
NewFile(lisp_filename, "w");
|
|
if (!f_lisp) {
|
|
FileErrorDisplay(lisp_filename);
|
|
SWIG_exit(EXIT_FAILURE);
|
|
}
|
|
|
|
if (CPlusPlus || CWrap) {
|
|
f_cxx = NewFile(cxx_filename, "w");
|
|
if (!f_cxx) {
|
|
Close(f_lisp);
|
|
Delete(f_lisp);
|
|
Printf(stderr, "Unable to open %s for writing\n", cxx_filename);
|
|
SWIG_exit(EXIT_FAILURE);
|
|
}
|
|
|
|
String *clos_filename = NewString("");
|
|
Printf(clos_filename, "%s%s-clos.lisp", SWIG_output_directory(), module);
|
|
f_clos = NewFile(clos_filename, "w");
|
|
if (!f_clos) {
|
|
Close(f_lisp);
|
|
Delete(f_lisp);
|
|
Printf(stderr, "Unable to open %s for writing\n", cxx_filename);
|
|
SWIG_exit(EXIT_FAILURE);
|
|
}
|
|
} else {
|
|
f_cxx = NewString("");
|
|
f_clos = NewString("");
|
|
}
|
|
|
|
f_cxx_header = f_cxx;
|
|
f_cxx_wrapper = NewString("");
|
|
|
|
Swig_register_filebyname("header", f_cxx_header);
|
|
Swig_register_filebyname("wrapper", f_cxx_wrapper);
|
|
Swig_register_filebyname("runtime", f_cxx);
|
|
Swig_register_filebyname("lisphead", f_clhead);
|
|
if (!no_swig_lisp)
|
|
Swig_register_filebyname("swiglisp", f_cl);
|
|
else
|
|
Swig_register_filebyname("swiglisp", f_null);
|
|
|
|
Language::top(n);
|
|
Printf(f_lisp, "%s\n", f_clhead);
|
|
Printf(f_lisp, "%s\n", f_cl);
|
|
Printf(f_lisp, "%s\n", f_clwrap);
|
|
|
|
Close(f_lisp);
|
|
Delete(f_lisp); // Deletes the handle, not the file
|
|
Delete(f_cl);
|
|
Delete(f_clhead);
|
|
Delete(f_clwrap);
|
|
Close(f_cxx);
|
|
Delete(f_cxx);
|
|
Delete(f_cxx_wrapper);
|
|
Delete(f_null);
|
|
|
|
return SWIG_OK;
|
|
}
|
|
|
|
int CFFI::classHandler(Node *n) {
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, "class %s::%s\n", "some namespace", //current_namespace,
|
|
Getattr(n, "sym:name"));
|
|
#endif
|
|
String *name = Getattr(n, "sym:name");
|
|
String *kind = Getattr(n, "kind");
|
|
|
|
// maybe just remove this check and get rid of the else clause below.
|
|
if (Strcmp(kind, "struct") == 0) {
|
|
emit_struct_union(n, false);
|
|
return SWIG_OK;
|
|
} else if (Strcmp(kind, "union") == 0) {
|
|
emit_struct_union(n, true);
|
|
return SWIG_OK;
|
|
} else if (Strcmp(kind, "class") == 0) {
|
|
emit_class(n);
|
|
Language::classHandler(n);
|
|
} else {
|
|
Printf(stderr, "Don't know how to deal with %s kind of class yet.\n", kind);
|
|
Printf(stderr, " (name: %s)\n", name);
|
|
SWIG_exit(EXIT_FAILURE);
|
|
return SWIG_OK;
|
|
}
|
|
|
|
return SWIG_OK;
|
|
}
|
|
|
|
int CFFI::constructorHandler(Node *n) {
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, "constructor %s\n", Getattr(n, "name"));
|
|
Printf(stderr, "constructor %s\n and %s and %s", Getattr(n, "kind"), Getattr(n, "sym:name"), Getattr(n, "allegrocl:old-sym:name"));
|
|
#endif
|
|
Setattr(n, "cffi:constructorfunction", "1");
|
|
// Let SWIG generate a global forwarding function.
|
|
return Language::constructorHandler(n);
|
|
}
|
|
|
|
int CFFI::destructorHandler(Node *n) {
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, "destructor %s\n", Getattr(n, "name"));
|
|
#endif
|
|
|
|
// Let SWIG generate a global forwarding function.
|
|
return Language::destructorHandler(n);
|
|
}
|
|
|
|
void CFFI::emit_defmethod(Node *n) {
|
|
String *args_placeholder = NewStringf("");
|
|
String *args_call = NewStringf("");
|
|
|
|
ParmList *pl = Getattr(n, "parms");
|
|
int argnum = 0;
|
|
Node *parent = parentNode(n);
|
|
bool first = 0;
|
|
|
|
for (Parm *p = pl; p; p = nextSibling(p), argnum++) {
|
|
String *argname = Getattr(p, "name");
|
|
String *ffitype = Swig_typemap_lookup("lispclass", p, "", 0);
|
|
|
|
int tempargname = 0;
|
|
|
|
if(!first)
|
|
first = true;
|
|
else
|
|
Printf(args_placeholder, " ");
|
|
|
|
if (!argname) {
|
|
argname = NewStringf("arg%d", argnum);
|
|
tempargname = 1;
|
|
} else if (Strcmp(argname, "t") == 0 || Strcmp(argname, "T") == 0) {
|
|
argname = NewStringf("t-arg%d", argnum);
|
|
tempargname = 1;
|
|
}
|
|
if (Len(ffitype) > 0)
|
|
Printf(args_placeholder, "(%s %s)", argname, ffitype);
|
|
else
|
|
Printf(args_placeholder, "%s", argname);
|
|
|
|
if (ffitype && Strcmp(ffitype, lispify_name(parent, lispy_name(Char(Getattr(parent, "sym:name"))), "'classname")) == 0)
|
|
Printf(args_call, " (ff-pointer %s)", argname);
|
|
else
|
|
Printf(args_call, " %s", argname);
|
|
|
|
Delete(ffitype);
|
|
|
|
if (tempargname)
|
|
Delete(argname);
|
|
}
|
|
|
|
String *method_name = Getattr(n, "name");
|
|
int x = Replace(method_name, "operator ", "", DOH_REPLACE_FIRST); //
|
|
|
|
if (x == 1)
|
|
Printf(f_clos, "(cl:shadow \"%s\")\n", method_name);
|
|
|
|
Printf(f_clos, "(cl:defmethod %s (%s)\n (%s%s))\n\n",
|
|
lispify_name(n, lispy_name(Char(method_name)), "'method"), args_placeholder,
|
|
lispify_name(n, Getattr(n, "sym:name"), "'function"), args_call);
|
|
|
|
}
|
|
|
|
void CFFI::emit_initialize_instance(Node *n) {
|
|
String *args_placeholder = NewStringf("");
|
|
String *args_call = NewStringf("");
|
|
|
|
ParmList *pl = Getattr(n, "parms");
|
|
int argnum = 0;
|
|
Node *parent = parentNode(n);
|
|
|
|
for (Parm *p = pl; p; p = nextSibling(p), argnum++) {
|
|
String *argname = Getattr(p, "name");
|
|
String *ffitype = Swig_typemap_lookup("lispclass", p, "", 0);
|
|
|
|
int tempargname = 0;
|
|
if (!argname) {
|
|
argname = NewStringf("arg%d", argnum);
|
|
tempargname = 1;
|
|
} else if (Strcmp(argname, "t") == 0 || Strcmp(argname, "T") == 0) {
|
|
argname = NewStringf("t-arg%d", argnum);
|
|
tempargname = 1;
|
|
}
|
|
if (Len(ffitype) > 0)
|
|
Printf(args_placeholder, " (%s %s)", argname, ffitype);
|
|
else
|
|
Printf(args_placeholder, " %s", argname);
|
|
|
|
if (Strcmp(ffitype, lispify_name(parent, lispy_name(Char(Getattr(parent, "sym:name"))), "'classname")) == 0)
|
|
Printf(args_call, " (ff-pointer %s)", argname);
|
|
else
|
|
Printf(args_call, " %s", argname);
|
|
|
|
Delete(ffitype);
|
|
|
|
if (tempargname)
|
|
Delete(argname);
|
|
}
|
|
|
|
Printf(f_clos, "(cl:defmethod initialize-instance :after ((obj %s) &key%s)\n (setf (slot-value obj 'ff-pointer) (%s%s)))\n\n",
|
|
lispify_name(parent, lispy_name(Char(Getattr(parent, "sym:name"))), "'class"), args_placeholder,
|
|
lispify_name(n, Getattr(n, "sym:name"), "'function"), args_call);
|
|
|
|
}
|
|
|
|
void CFFI::emit_setter(Node *n) {
|
|
Node *p = parentNode(n);
|
|
Printf(f_clos, "(cl:defmethod (cl:setf %s) (arg0 (obj %s))\n (%s (ff-pointer obj) arg0))\n\n",
|
|
lispify_name(n, Getattr(n, "name"), "'method"),
|
|
lispify_name(p, lispy_name(Char(Getattr(p, "sym:name"))), "'class"), lispify_name(n, Getattr(n, "sym:name"), "'function"));
|
|
}
|
|
|
|
|
|
void CFFI::emit_getter(Node *n) {
|
|
Node *p = parentNode(n);
|
|
Printf(f_clos, "(cl:defmethod %s ((obj %s))\n (%s (ff-pointer obj)))\n\n",
|
|
lispify_name(n, Getattr(n, "name"), "'method"),
|
|
lispify_name(p, lispy_name(Char(Getattr(p, "sym:name"))), "'class"), lispify_name(n, Getattr(n, "sym:name"), "'function"));
|
|
}
|
|
|
|
int CFFI::memberfunctionHandler(Node *n) {
|
|
// Let SWIG generate a global forwarding function.
|
|
Setattr(n, "cffi:memberfunction", "1");
|
|
return Language::memberfunctionHandler(n);
|
|
}
|
|
|
|
int CFFI::membervariableHandler(Node *n) {
|
|
// Let SWIG generate a get/set function pair.
|
|
Setattr(n, "cffi:membervariable", "1");
|
|
return Language::membervariableHandler(n);
|
|
}
|
|
|
|
int CFFI::functionWrapper(Node *n) {
|
|
|
|
ParmList *parms = Getattr(n, "parms");
|
|
String *iname = Getattr(n, "sym:name");
|
|
Wrapper *f = NewWrapper();
|
|
|
|
String *raw_return_type = Swig_typemap_lookup("ctype", n, "", 0);
|
|
SwigType *return_type = Swig_cparse_type(raw_return_type);
|
|
SwigType *resolved = SwigType_typedef_resolve_all(return_type);
|
|
int is_void_return = (Cmp(resolved, "void") == 0);
|
|
Delete(resolved);
|
|
|
|
if (!is_void_return) {
|
|
String *lresult_init = NewStringf("lresult = (%s)0", raw_return_type);
|
|
Wrapper_add_localv(f, "lresult", raw_return_type, lresult_init, NIL);
|
|
Delete(lresult_init);
|
|
}
|
|
|
|
String *overname = 0;
|
|
if (Getattr(n, "sym:overloaded")) {
|
|
overname = Getattr(n, "sym:overname");
|
|
} else {
|
|
if (!addSymbol(iname, n)) {
|
|
DelWrapper(f);
|
|
return SWIG_ERROR;
|
|
}
|
|
}
|
|
|
|
String *wname = Swig_name_wrapper(iname);
|
|
if (overname) {
|
|
Append(wname, overname);
|
|
}
|
|
Setattr(n, "wrap:name", wname);
|
|
|
|
// Emit all of the local variables for holding arguments.
|
|
emit_parameter_variables(parms, f);
|
|
|
|
// Attach the standard typemaps
|
|
Swig_typemap_attach_parms("ctype", parms, f);
|
|
emit_attach_parmmaps(parms, f);
|
|
|
|
int num_arguments = emit_num_arguments(parms);
|
|
String *name_and_parms = NewStringf("%s (", wname);
|
|
int i;
|
|
Parm *p;
|
|
int gencomma = 0;
|
|
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, "function - %s - %d\n", Getattr(n, "name"), num_arguments);
|
|
#endif
|
|
|
|
for (i = 0, p = parms; i < num_arguments; i++) {
|
|
|
|
while (checkAttribute(p, "tmap:in:numinputs", "0")) {
|
|
p = Getattr(p, "tmap:in:next");
|
|
}
|
|
|
|
SwigType *c_parm_type = Swig_cparse_type(Getattr(p, "tmap:ctype"));
|
|
String *arg = NewStringf("l%s", Getattr(p, "lname"));
|
|
|
|
// Emit parameter declaration
|
|
if (gencomma)
|
|
Printf(name_and_parms, ", ");
|
|
String *parm_decl = SwigType_str(c_parm_type, arg);
|
|
Printf(name_and_parms, "%s", parm_decl);
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, " param: %s\n", parm_decl);
|
|
#endif
|
|
Delete(parm_decl);
|
|
gencomma = 1;
|
|
|
|
// Emit parameter conversion code
|
|
String *parm_code = Getattr(p, "tmap:in");
|
|
{
|
|
Replaceall(parm_code, "$input", arg);
|
|
Setattr(p, "emit:input", arg);
|
|
Printf(f->code, "%s\n", parm_code);
|
|
p = Getattr(p, "tmap:in:next");
|
|
}
|
|
|
|
Delete(arg);
|
|
}
|
|
Printf(name_and_parms, ")");
|
|
|
|
// Emit the function definition
|
|
String *signature = SwigType_str(return_type, name_and_parms);
|
|
Printf(f->def, "EXPORT %s {", signature);
|
|
Printf(f->code, " try {\n");
|
|
|
|
String *actioncode = emit_action(n);
|
|
|
|
String *result_convert = Swig_typemap_lookup_out("out", n, "result", f, actioncode);
|
|
Replaceall(result_convert, "$result", "lresult");
|
|
Printf(f->code, "%s\n", result_convert);
|
|
if(!is_void_return) Printf(f->code, " return lresult;\n");
|
|
Delete(result_convert);
|
|
emit_return_variable(n, Getattr(n, "type"), f);
|
|
|
|
Printf(f->code, " } catch (...) {\n");
|
|
if (!is_void_return)
|
|
Printf(f->code, " return (%s)0;\n", raw_return_type);
|
|
Printf(f->code, " }\n");
|
|
Printf(f->code, "}\n");
|
|
|
|
if (CPlusPlus)
|
|
Wrapper_print(f, f_cxx);
|
|
|
|
if (CPlusPlus) {
|
|
emit_defun(n, wname);
|
|
if (Getattr(n, "cffi:memberfunction"))
|
|
emit_defmethod(n);
|
|
else if (Getattr(n, "cffi:membervariable")) {
|
|
if (Getattr(n, "memberget"))
|
|
emit_getter(n);
|
|
else if (Getattr(n, "memberset"))
|
|
emit_setter(n);
|
|
}
|
|
else if (Getattr(n, "cffi:constructorfunction")) {
|
|
emit_initialize_instance(n);
|
|
}
|
|
} else
|
|
emit_defun(n, iname);
|
|
|
|
// if (!overloaded || !Getattr(n, "sym:nextSibling")) {
|
|
// update_package_if_needed(n);
|
|
// emit_buffered_defuns(n);
|
|
// // this is the last overload.
|
|
// if (overloaded) {
|
|
// emit_dispatch_defun(n);
|
|
// }
|
|
// }
|
|
|
|
Delete(wname);
|
|
DelWrapper(f);
|
|
|
|
return SWIG_OK;
|
|
}
|
|
|
|
|
|
void CFFI::emit_defun(Node *n, String *name) {
|
|
|
|
// String *storage=Getattr(n,"storage");
|
|
// if(!storage || (Strcmp(storage,"extern") && Strcmp(storage,"externc")))
|
|
// return SWIG_OK;
|
|
|
|
String *func_name = Getattr(n, "sym:name");
|
|
|
|
ParmList *pl = Getattr(n, "parms");
|
|
|
|
int argnum = 0;
|
|
|
|
func_name = lispify_name(n, func_name, "'function");
|
|
|
|
emit_inline(n, func_name);
|
|
|
|
Printf(f_cl, "\n(cffi:defcfun (\"%s\" %s)", name, func_name);
|
|
String *ffitype = Swig_typemap_lookup("cout", n, ":pointer", 0);
|
|
|
|
Printf(f_cl, " %s", ffitype);
|
|
Delete(ffitype);
|
|
|
|
for (Parm *p = pl; p; p = nextSibling(p), argnum++) {
|
|
|
|
if (SwigType_isvarargs(Getattr(p, "type"))) {
|
|
Printf(f_cl, "\n %s", NewString("&rest"));
|
|
continue;
|
|
}
|
|
|
|
String *argname = Getattr(p, "name");
|
|
|
|
ffitype = Swig_typemap_lookup("cin", p, "", 0);
|
|
|
|
int tempargname = 0;
|
|
if (!argname) {
|
|
|
|
argname = NewStringf("arg%d", argnum);
|
|
tempargname = 1;
|
|
} else if (Strcmp(argname, "t") == 0 || Strcmp(argname, "T") == 0) {
|
|
argname = NewStringf("t_arg%d", argnum);
|
|
tempargname = 1;
|
|
}
|
|
|
|
Printf(f_cl, "\n (%s %s)", argname, ffitype);
|
|
|
|
Delete(ffitype);
|
|
|
|
if (tempargname)
|
|
Delete(argname);
|
|
}
|
|
Printf(f_cl, ")\n"); /* finish arg list */
|
|
|
|
emit_export(n, func_name);
|
|
}
|
|
|
|
|
|
int CFFI::constantWrapper(Node *n) {
|
|
String *type = Getattr(n, "type");
|
|
String *converted_value = convert_literal(Getattr(n, "value"), type);
|
|
String *name = lispify_name(n, Getattr(n, "sym:name"), "'constant");
|
|
|
|
if (Strcmp(name, "t") == 0 || Strcmp(name, "T") == 0)
|
|
name = NewStringf("t_var");
|
|
|
|
Printf(f_cl, "\n(cl:defconstant %s %s)\n", name, converted_value);
|
|
Delete(converted_value);
|
|
|
|
emit_export(n, name);
|
|
return SWIG_OK;
|
|
}
|
|
|
|
int CFFI::variableWrapper(Node *n) {
|
|
// String *storage=Getattr(n,"storage");
|
|
// Printf(stdout,"\"%s\" %s)\n",storage,Getattr(n, "sym:name"));
|
|
|
|
// if(!storage || (Strcmp(storage,"extern") && Strcmp(storage,"externc")))
|
|
// return SWIG_OK;
|
|
|
|
String *var_name = Getattr(n, "sym:name");
|
|
String *lisp_type = Swig_typemap_lookup("cin", n, "", 0);
|
|
String *lisp_name = lispify_name(n, var_name, "'variable");
|
|
|
|
if (Strcmp(lisp_name, "t") == 0 || Strcmp(lisp_name, "T") == 0)
|
|
lisp_name = NewStringf("t_var");
|
|
|
|
Printf(f_cl, "\n(cffi:defcvar (\"%s\" %s)\n %s)\n", var_name, lisp_name, lisp_type);
|
|
|
|
Delete(lisp_type);
|
|
|
|
emit_export(n, lisp_name);
|
|
return SWIG_OK;
|
|
}
|
|
|
|
int CFFI::typedefHandler(Node *n) {
|
|
if (generate_typedef_flag && strncmp(Char(Getattr(n, "type")), "enum", 4)) {
|
|
String *lisp_name = lispify_name(n, Getattr(n, "name"), "'typename");
|
|
Printf(f_cl, "\n(cffi:defctype %s %s)\n", lisp_name, Swig_typemap_lookup("cin", n, "", 0));
|
|
emit_export(n, lisp_name);
|
|
}
|
|
return Language::typedefHandler(n);
|
|
}
|
|
|
|
int CFFI::enumDeclaration(Node *n) {
|
|
String *name = Getattr(n, "sym:name");
|
|
bool slot_name_keywords;
|
|
String *lisp_name = 0;
|
|
if (name && Len(name) != 0) {
|
|
lisp_name = lispify_name(n, name, "'enumname");
|
|
if (GetFlag(n, "feature:bitfield")) {
|
|
Printf(f_cl, "\n(cffi:defbitfield %s", lisp_name);
|
|
} else {
|
|
Printf(f_cl, "\n(cffi:defcenum %s", lisp_name);
|
|
}
|
|
slot_name_keywords = true;
|
|
|
|
//Registering the enum name to the cin and cout typemaps
|
|
Parm *pattern = NewParm(name, NULL);
|
|
Swig_typemap_register("cin", pattern, lisp_name, NULL, NULL);
|
|
Swig_typemap_register("cout", pattern, lisp_name, NULL, NULL);
|
|
Delete(pattern);
|
|
//Registering with the kind, i.e., enum
|
|
pattern = NewParm(NewStringf("enum %s", name), NULL);
|
|
Swig_typemap_register("cin", pattern, lisp_name, NULL, NULL);
|
|
Swig_typemap_register("cout", pattern, lisp_name, NULL, NULL);
|
|
Delete(pattern);
|
|
|
|
} else {
|
|
Printf(f_cl, "\n(defanonenum %s", name);
|
|
slot_name_keywords = false;
|
|
}
|
|
|
|
for (Node *c = firstChild(n); c; c = nextSibling(c)) {
|
|
|
|
String *slot_name = lispify_name(c, Getattr(c, "name"), "'enumvalue", slot_name_keywords);
|
|
String *value = Getattr(c, "enumvalue");
|
|
|
|
if (!value || GetFlag(n, "feature:bitfield:ignore_values"))
|
|
Printf(f_cl, "\n\t%s", slot_name);
|
|
else {
|
|
String *type = Getattr(c, "type");
|
|
String *converted_value = convert_literal(value, type);
|
|
Printf(f_cl, "\n\t(%s #.%s)", slot_name, converted_value);
|
|
Delete(converted_value);
|
|
}
|
|
Delete(value);
|
|
}
|
|
|
|
Printf(f_cl, ")\n");
|
|
|
|
// No need to export keywords
|
|
if (lisp_name && Len(lisp_name) != 0) {
|
|
emit_export(n, lisp_name);
|
|
} else {
|
|
for (Node *c = firstChild(n); c; c = nextSibling(c))
|
|
emit_export(c, lispify_name(c, Getattr(c, "name"), "'enumvalue"));
|
|
}
|
|
|
|
return SWIG_OK;
|
|
}
|
|
void CFFI::emit_class(Node *n) {
|
|
|
|
#ifdef CFFI_WRAP_DEBUG
|
|
Printf(stderr, "emit_class: ENTER... '%s'(%x)\n", Getattr(n, "sym:name"), n);
|
|
#endif
|
|
|
|
String *name = Getattr(n, "sym:name");
|
|
String *lisp_name = lispify_name(n, lispy_name(Char(name)), "'classname");
|
|
|
|
String *bases = Getattr(n, "bases");
|
|
String *supers = NewString("(");
|
|
if (bases) {
|
|
int first = 1;
|
|
for (Iterator i = First(bases); i.item; i = Next(i)) {
|
|
if (!first)
|
|
Printf(supers, " ");
|
|
String *s = Getattr(i.item, "name");
|
|
Printf(supers, "%s", lispify_name(i.item, s, "'classname"));
|
|
}
|
|
} else {
|
|
// Printf(supers,"ff:foreign-pointer");
|
|
}
|
|
|
|
Printf(supers, ")");
|
|
Printf(f_clos, "\n(cl:defclass %s%s", lisp_name, supers);
|
|
Printf(f_clos, "\n ((ff-pointer :reader ff-pointer)))\n\n");
|
|
|
|
Parm *pattern = NewParm(Getattr(n, "name"), NULL);
|
|
|
|
Swig_typemap_register("lispclass", pattern, lisp_name, NULL, NULL);
|
|
SwigType_add_pointer(Getattr(pattern, "type"));
|
|
Swig_typemap_register("lispclass", pattern, lisp_name, NULL, NULL);
|
|
SwigType_add_qualifier(Getattr(pattern, "type"), "const");
|
|
Swig_typemap_register("lispclass", pattern, lisp_name, NULL, NULL);
|
|
SwigType_del_pointer(Getattr(pattern, "type"));
|
|
SwigType_add_reference(Getattr(pattern, "type"));
|
|
Swig_typemap_register("lispclass", pattern, lisp_name, NULL, NULL);
|
|
|
|
#ifdef CFFI_WRAP_DEBUG
|
|
Printf(stderr, " pattern %s name %s .. ... %s .\n", pattern, lisp_name);
|
|
#endif
|
|
|
|
Delete(pattern);
|
|
|
|
// Walk children to generate type definition.
|
|
String *slotdefs = NewString(" ");
|
|
|
|
#ifdef CFFI_WRAP_DEBUG
|
|
Printf(stderr, " walking children...\n");
|
|
#endif
|
|
|
|
Node *c;
|
|
for (c = firstChild(n); c; c = nextSibling(c)) {
|
|
String *storage_type = Getattr(c, "storage");
|
|
if ((!Strcmp(nodeType(c), "cdecl") && (!storage_type || Strcmp(storage_type, "typedef")))) {
|
|
String *access = Getattr(c, "access");
|
|
|
|
// hack. why would decl have a value of "variableHandler" and now "0"?
|
|
String *childDecl = Getattr(c, "decl");
|
|
// Printf(stderr,"childDecl = '%s' (%s)\n", childDecl, Getattr(c,"view"));
|
|
if (!Strcmp(childDecl, "0"))
|
|
childDecl = NewString("");
|
|
|
|
SwigType *childType = NewStringf("%s%s", childDecl,
|
|
Getattr(c, "type"));
|
|
String *cname = (access && Strcmp(access, "public")) ? NewString("nil") : Copy(Getattr(c, "name"));
|
|
|
|
if (!SwigType_isfunction(childType)) {
|
|
// Printf(slotdefs, ";;; member functions don't appear as slots.\n ");
|
|
// Printf(slotdefs, ";; ");
|
|
// String *ns = listify_namespace(Getattr(n, "cffi:package"));
|
|
String *ns = NewString("");
|
|
#ifdef CFFI_WRAP_DEBUG
|
|
Printf(stderr, "slot name = '%s' ns = '%s' class-of '%s' and type = '%s'\n", cname, ns, name, childType);
|
|
#endif
|
|
Printf(slotdefs, "(#.(swig-insert-id \"%s\" %s :type :slot :class \"%s\") %s)", cname, ns, name, childType); //compose_foreign_type(childType)
|
|
Delete(ns);
|
|
if (access && Strcmp(access, "public"))
|
|
Printf(slotdefs, " ;; %s member", access);
|
|
|
|
Printf(slotdefs, "\n ");
|
|
}
|
|
Delete(childType);
|
|
Delete(cname);
|
|
}
|
|
}
|
|
|
|
|
|
// String *ns_list = listify_namespace(Getattr(n,"cffi:namespace"));
|
|
// update_package_if_needed(n,f_clhead);
|
|
// Printf(f_clos,
|
|
// "(swig-def-foreign-class \"%s\"\n %s\n (:%s\n%s))\n\n",
|
|
// name, supers, kind, slotdefs);
|
|
|
|
Delete(supers);
|
|
// Delete(ns_list);
|
|
|
|
// Parm *pattern = NewParm(name,NULL);
|
|
// Swig_typemap_register("cin",pattern,lisp_name,NULL,NULL);
|
|
//Swig_typemap_register("cout",pattern,lisp_name,NULL,NULL);
|
|
//Delete(pattern);
|
|
|
|
#ifdef CFFI_WRAP_DEBUG
|
|
Printf(stderr, "emit_class: EXIT\n");
|
|
#endif
|
|
}
|
|
|
|
// Includes structs
|
|
void CFFI::emit_struct_union(Node *n, bool un = false) {
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, "struct/union %s\n", Getattr(n, "name"));
|
|
Printf(stderr, "struct/union %s\n and %s", Getattr(n, "kind"), Getattr(n, "sym:name"));
|
|
#endif
|
|
|
|
String *name = Getattr(n, "sym:name");
|
|
String *kind = Getattr(n, "kind");
|
|
|
|
if (Strcmp(kind, "struct") != 0 && Strcmp(kind, "union") != 0) {
|
|
Printf(stderr, "Don't know how to deal with %s kind of class yet.\n", kind);
|
|
Printf(stderr, " (name: %s)\n", name);
|
|
SWIG_exit(EXIT_FAILURE);
|
|
}
|
|
String *lisp_name = lispify_name(n, name, "'classname");
|
|
|
|
//Register the struct/union name to the cin and cout typemaps
|
|
|
|
Parm *pattern = NewParm(name, NULL);
|
|
Swig_typemap_register("cin", pattern, lisp_name, NULL, NULL);
|
|
Swig_typemap_register("cout", pattern, lisp_name, NULL, NULL);
|
|
Delete(pattern);
|
|
//Registering with the kind, i.e., struct or union
|
|
pattern = NewParm(NewStringf("%s %s", kind, name), NULL);
|
|
Swig_typemap_register("cin", pattern, lisp_name, NULL, NULL);
|
|
Swig_typemap_register("cout", pattern, lisp_name, NULL, NULL);
|
|
Delete(pattern);
|
|
|
|
if (un) {
|
|
Printf(f_cl, "\n(cffi:defcunion %s", lisp_name);
|
|
} else
|
|
Printf(f_cl, "\n(cffi:defcstruct %s", lisp_name);
|
|
|
|
|
|
for (Node *c = firstChild(n); c; c = nextSibling(c)) {
|
|
#ifdef CFFI_DEBUG
|
|
Printf(stderr, "struct/union %s\n", Getattr(c, "name"));
|
|
Printf(stderr, "struct/union %s and %s \n", Getattr(c, "kind"), Getattr(c, "sym:name"));
|
|
#endif
|
|
|
|
if (Strcmp(nodeType(c), "cdecl")) {
|
|
//C declaration ignore
|
|
// Printf(stderr, "Structure %s has a slot that we can't deal with.\n",
|
|
// name);
|
|
// Printf(stderr, "nodeType: %s, name: %s, type: %s\n",
|
|
// nodeType(c),
|
|
// Getattr(c, "name"),
|
|
// Getattr(c, "type"));
|
|
// SWIG_exit(EXIT_FAILURE);
|
|
} else {
|
|
SwigType *childType = NewStringf("%s%s", Getattr(c, "decl"),
|
|
Getattr(c, "type"));
|
|
|
|
Hash *typemap = Swig_typemap_search("cin", childType, "", 0);
|
|
String *typespec = NewString("");
|
|
if (typemap) {
|
|
typespec = NewString(Getattr(typemap, "code"));
|
|
}
|
|
|
|
String *slot_name = lispify_name(c, Getattr(c, "sym:name"), "'slotname");
|
|
if (Strcmp(slot_name, "t") == 0 || Strcmp(slot_name, "T") == 0)
|
|
slot_name = NewStringf("t_var");
|
|
|
|
Printf(f_cl, "\n\t(%s %s)", slot_name, typespec);
|
|
|
|
Delete(typespec);
|
|
}
|
|
}
|
|
|
|
Printf(f_cl, ")\n");
|
|
|
|
emit_export(n, lisp_name);
|
|
for (Node *child = firstChild(n); child; child = nextSibling(child)) {
|
|
if (!Strcmp(nodeType(child), "cdecl")) {
|
|
emit_export(child, lispify_name(child, Getattr(child, "sym:name"), "'slotname"));
|
|
}
|
|
}
|
|
|
|
/* Add this structure to the known lisp types */
|
|
//Printf(stdout, "Adding %s foreign type\n", name);
|
|
// add_defined_foreign_type(name);
|
|
|
|
}
|
|
|
|
void CFFI::emit_export(Node *n, String *name) {
|
|
if (GetInt(n, "feature:export"))
|
|
Printf(f_cl, "\n(cl:export '%s)\n", name);
|
|
}
|
|
|
|
void CFFI::emit_inline(Node *n, String *name) {
|
|
if (GetInt(n, "feature:inline"))
|
|
Printf(f_cl, "\n(cl:declaim (cl:inline %s))\n", name);
|
|
}
|
|
|
|
String *CFFI::lispify_name(Node *n, String *ty, const char *flag, bool kw) {
|
|
String *intern_func = Getattr(n, "feature:intern_function");
|
|
if (intern_func) {
|
|
if (Strcmp(intern_func, "1") == 0)
|
|
intern_func = NewStringf("swig-lispify");
|
|
return NewStringf("#.(%s \"%s\" %s%s)", intern_func, ty, flag, kw ? " :keyword" : "");
|
|
} else if (kw)
|
|
return NewStringf(":%s", ty);
|
|
else
|
|
return ty;
|
|
}
|
|
|
|
/* utilities */
|
|
/* returns new string w/ parens stripped */
|
|
String *CFFI::strip_parens(String *string) {
|
|
char *s = Char(string), *p;
|
|
int len = Len(string);
|
|
String *res;
|
|
|
|
if (len == 0 || s[0] != '(' || s[len - 1] != ')') {
|
|
return NewString(string);
|
|
}
|
|
|
|
p = (char *) malloc(len - 2 + 1);
|
|
if (!p) {
|
|
Printf(stderr, "Malloc failed\n");
|
|
SWIG_exit(EXIT_FAILURE);
|
|
}
|
|
|
|
strncpy(p, s + 1, len - 1);
|
|
p[len - 2] = 0; /* null terminate */
|
|
|
|
res = NewString(p);
|
|
free(p);
|
|
|
|
return res;
|
|
}
|
|
|
|
String *CFFI::trim(String *str) {
|
|
char *c = Char(str);
|
|
while (*c != '\0' && isspace((int) *c))
|
|
++c;
|
|
String *result = NewString(c);
|
|
Chop(result);
|
|
return result;
|
|
}
|
|
|
|
String *CFFI::infix_to_prefix(String *val, char split_op, const String *op, String *type) {
|
|
List *ored = Split(val, split_op, -1);
|
|
|
|
// some float hackery
|
|
//i don't understand it, if you do then please explain
|
|
// if ( ((split_op == '+') || (split_op == '-')) && Len(ored) == 2 &&
|
|
// (SwigType_type(type) == T_FLOAT || SwigType_type(type) == T_DOUBLE ||
|
|
// SwigType_type(type) == T_LONGDOUBLE) ) {
|
|
// // check that we're not splitting a float
|
|
// String *possible_result = convert_literal(val, type, false);
|
|
// if (possible_result) return possible_result;
|
|
|
|
// }
|
|
|
|
// try parsing the split results. if any part fails, kick out.
|
|
bool part_failed = false;
|
|
if (Len(ored) > 1) {
|
|
String *result = NewStringf("(%s", op);
|
|
for (Iterator i = First(ored); i.item; i = Next(i)) {
|
|
String *converted = convert_literal(i.item, type);
|
|
if (converted) {
|
|
Printf(result, " %s", converted);
|
|
Delete(converted);
|
|
} else {
|
|
part_failed = true;
|
|
break;
|
|
}
|
|
}
|
|
Printf(result, ")");
|
|
Delete(ored);
|
|
return part_failed ? 0 : result;
|
|
} else {
|
|
Delete(ored);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* To be called by code generating the lisp interface
|
|
Will return a String containing the literal based on type.
|
|
Will return null if there are problems.
|
|
|
|
try_to_split defaults to true (see stub above).
|
|
*/
|
|
String *CFFI::convert_literal(String *literal, String *type, bool try_to_split) {
|
|
String *num_param = Copy(literal);
|
|
String *trimmed = trim(num_param);
|
|
String *num = strip_parens(trimmed), *res = 0;
|
|
Delete(trimmed);
|
|
char *s = Char(num);
|
|
|
|
// very basic parsing of infix expressions.
|
|
if (try_to_split) {
|
|
if ((res = infix_to_prefix(num, '|', "cl:logior", type)))
|
|
return res;
|
|
if ((res = infix_to_prefix(num, '&', "cl:logand", type)))
|
|
return res;
|
|
if ((res = infix_to_prefix(num, '^', "cl:logxor", type)))
|
|
return res;
|
|
if ((res = infix_to_prefix(num, '*', "cl:*", type)))
|
|
return res;
|
|
if ((res = infix_to_prefix(num, '/', "cl:/", type)))
|
|
return res;
|
|
if ((res = infix_to_prefix(num, '+', "cl:+", type)))
|
|
return res;
|
|
if ((res = infix_to_prefix(num, '-', "cl:-", type)))
|
|
return res;
|
|
}
|
|
|
|
if (SwigType_type(type) == T_FLOAT || SwigType_type(type) == T_DOUBLE || SwigType_type(type) == T_LONGDOUBLE) {
|
|
// Use CL syntax for float literals
|
|
|
|
// careful. may be a float identifier or float constant.
|
|
char *num_start = Char(num);
|
|
char *num_end = num_start + strlen(num_start) - 1;
|
|
|
|
bool is_literal = isdigit(*num_start) || (*num_start == '.') || (*num_start == '+') || (*num_start == '-');
|
|
|
|
String *lisp_exp = 0;
|
|
if (is_literal) {
|
|
if (*num_end == 'f' || *num_end == 'F') {
|
|
lisp_exp = NewString("f");
|
|
} else {
|
|
lisp_exp = NewString("d");
|
|
}
|
|
|
|
if (*num_end == 'l' || *num_end == 'L' || *num_end == 'f' || *num_end == 'F') {
|
|
*num_end = '\0';
|
|
num_end--;
|
|
}
|
|
|
|
int exponents = Replaceall(num, "e", lisp_exp) + Replaceall(num, "E", lisp_exp);
|
|
|
|
if (!exponents)
|
|
Printf(num, "%s0", lisp_exp);
|
|
|
|
if (exponents > 1 || (exponents + Replaceall(num, ".", ".") == 0)) {
|
|
Delete(num);
|
|
num = 0;
|
|
}
|
|
}
|
|
return num;
|
|
} else if (SwigType_type(type) == T_CHAR) {
|
|
/* Use CL syntax for character literals */
|
|
String* result = NewStringf("#\\%c", s[2]);
|
|
Delete(num);
|
|
// Printf(stderr, "%s %c %d", s, s[2], s);
|
|
return result;
|
|
} else if (SwigType_type(type) == T_STRING) {
|
|
/* Use CL syntax for string literals */
|
|
String* result = NewStringf("\"%s\"", num_param);
|
|
Delete(num);
|
|
return result;
|
|
} else if (SwigType_type(type) == T_INT || SwigType_type(type) == T_UINT) {
|
|
// Printf(stderr, "Is a T_INT or T_UINT %s, before replaceall\n", s);
|
|
Replaceall(num, "u", "");
|
|
Replaceall(num, "U", "");
|
|
Replaceall(num, "l", "");
|
|
Replaceall(num, "L", "");
|
|
|
|
int i, j;
|
|
if (sscanf(s, "%d >> %d", &i, &j) == 2) {
|
|
String* result = NewStringf("(cl:ash %d -%d)", i, j);
|
|
Delete(num);
|
|
return result;
|
|
} else if (sscanf(s, "%d << %d", &i, &j) == 2) {
|
|
String* result = NewStringf("(cl:ash %d %d)", i, j);
|
|
Delete(num);
|
|
return result;
|
|
}
|
|
}
|
|
|
|
if (Len(num) >= 2 && s[0] == '0') { /* octal or hex */
|
|
if (s[1] == 'x'){
|
|
DohReplace(num,"0","#",DOH_REPLACE_FIRST);
|
|
}
|
|
else{
|
|
DohReplace(num,"0","#o",DOH_REPLACE_FIRST);
|
|
}
|
|
}
|
|
return num;
|
|
}
|
|
|
|
//less flexible as it does the conversion in C, the lispify name does the conversion in lisp
|
|
String *CFFI::lispy_name(char *name) {
|
|
bool helper = false;
|
|
String *new_name = NewString("");
|
|
for (unsigned int i = 0; i < strlen(name); i++) {
|
|
if (name[i] == '_' || name[i] == '-') {
|
|
Printf(new_name, "%c", '-');
|
|
helper = false;
|
|
} else if (name[i] >= 'A' && name[i] <= 'Z') {
|
|
if (helper)
|
|
Printf(new_name, "%c", '-');
|
|
Printf(new_name, "%c", ('a' + (name[i] - 'A')));
|
|
helper = false;
|
|
} else {
|
|
helper = true;
|
|
Printf(new_name, "%c", name[i]);
|
|
}
|
|
}
|
|
return new_name;
|
|
}
|
|
|
|
extern "C" Language *swig_cffi(void) {
|
|
return new CFFI();
|
|
}
|