| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * GTSAM Copyright 2010, Georgia Tech Research Corporation, | 
					
						
							|  |  |  |  * Atlanta, Georgia 30332-0415 | 
					
						
							|  |  |  |  * All Rights Reserved | 
					
						
							|  |  |  |  * Authors: Frank Dellaert, et al. (see THANKS for the full author list) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * See LICENSE for the license information | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * -------------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * @file Module.ccp | 
					
						
							|  |  |  |  * @author Frank Dellaert | 
					
						
							|  |  |  |  * @author Alex Cunningham | 
					
						
							|  |  |  |  * @author Andrew Melim | 
					
						
							| 
									
										
										
										
											2012-07-18 23:47:06 +08:00
										 |  |  |  * @author Richard Roberts | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |  **/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Module.h"
 | 
					
						
							|  |  |  | #include "FileWriter.h"
 | 
					
						
							|  |  |  | #include "TypeAttributesTable.h"
 | 
					
						
							| 
									
										
										
										
											2013-10-27 06:22:31 +08:00
										 |  |  | #include "utilities.h"
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | #include <boost/filesystem.hpp>
 | 
					
						
							|  |  |  | #include <boost/lexical_cast.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace std; | 
					
						
							|  |  |  | using namespace wrap; | 
					
						
							|  |  |  | using namespace BOOST_SPIRIT_CLASSIC_NS; | 
					
						
							|  |  |  | namespace bl = boost::lambda; | 
					
						
							|  |  |  | namespace fs = boost::filesystem; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | // We parse an interface file into a Module object.
 | 
					
						
							|  |  |  | // The grammar is defined using the boost/spirit combinatorial parser.
 | 
					
						
							|  |  |  | // For example, str_p("const") parses the string "const", and the >>
 | 
					
						
							|  |  |  | // operator creates a sequence parser. The grammar below, composed of rules
 | 
					
						
							|  |  |  | // and with start rule [class_p], doubles as the specs for our interface files.
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2014-11-12 05:38:50 +08:00
										 |  |  | // If a number of template arguments were given, generate a number of expanded
 | 
					
						
							|  |  |  | // class names, e.g., PriorFactor -> PriorFactorPose2, and add those classes
 | 
					
						
							| 
									
										
										
										
											2016-09-09 19:28:13 +08:00
										 |  |  | static void handle_possible_template(vector<Class>& classes, | 
					
						
							|  |  |  |                                      vector<Class>& uninstantiatedClasses, | 
					
						
							|  |  |  |                                      const Class& cls, const Template& t) { | 
					
						
							|  |  |  |   uninstantiatedClasses.push_back(cls); | 
					
						
							| 
									
										
										
										
											2016-02-08 12:33:48 +08:00
										 |  |  |   if (cls.templateArgs.empty() || t.empty()) { | 
					
						
							| 
									
										
										
										
											2014-11-12 05:38:50 +08:00
										 |  |  |     classes.push_back(cls); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2014-11-13 02:09:30 +08:00
										 |  |  |     if (cls.templateArgs.size() != 1) | 
					
						
							|  |  |  |       throw std::runtime_error( | 
					
						
							|  |  |  |           "In-line template instantiations only handle a single template argument"); | 
					
						
							| 
									
										
										
										
											2016-02-08 12:33:48 +08:00
										 |  |  |     string arg = cls.templateArgs.front(); | 
					
						
							|  |  |  |     vector<Class> classInstantiations = | 
					
						
							|  |  |  |         (t.nrValues() > 0) ? cls.expandTemplate(arg, t.argValues()) : | 
					
						
							|  |  |  |             cls.expandTemplate(arg, t.intList()); | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |     for(const Class& c: classInstantiations) | 
					
						
							| 
									
										
										
										
											2014-11-12 05:38:50 +08:00
										 |  |  |       classes.push_back(c); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-13 06:17:47 +08:00
										 |  |  | static void push_typedef_pair(vector<TypedefPair>& typedefs, | 
					
						
							|  |  |  |                               const Qualified& oldType, | 
					
						
							|  |  |  |                               const Qualified& newType, | 
					
						
							|  |  |  |                               const string& includeFile) { | 
					
						
							|  |  |  |     typedefs.push_back(TypedefPair(oldType, newType, includeFile)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:19 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | Module::Module(const std::string& moduleName, bool enable_verbose) | 
					
						
							|  |  |  | : name(moduleName), verbose(enable_verbose) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | Module::Module(const string& interfacePath, | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:19 +08:00
										 |  |  |          const string& moduleName, bool enable_verbose) | 
					
						
							|  |  |  | : name(moduleName), verbose(enable_verbose) | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:19 +08:00
										 |  |  |   // read interface file
 | 
					
						
							|  |  |  |   string interfaceFile = interfacePath + "/" + moduleName + ".h"; | 
					
						
							|  |  |  |   string contents = file_contents(interfaceFile); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // execute parsing
 | 
					
						
							|  |  |  |   parseMarkup(contents); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | void Module::parseMarkup(const std::string& data) { | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  |   // The parse imperatively :-( updates variables gradually during parse
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |   // The one with postfix 0 are used to reset the variables after parse.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |   // Grammar with actions that build the Class object. Actions are
 | 
					
						
							|  |  |  |   // defined within the square brackets [] and are executed whenever a
 | 
					
						
							|  |  |  |   // rule is successfully parsed. Define BOOST_SPIRIT_DEBUG to debug.
 | 
					
						
							|  |  |  |   // The grammar is allows a very restricted C++ header
 | 
					
						
							|  |  |  |   // lexeme_d turns off white space skipping
 | 
					
						
							|  |  |  |   // http://www.boost.org/doc/libs/1_37_0/libs/spirit/classic/doc/directives.html
 | 
					
						
							|  |  |  |   // ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-01 03:12:03 +08:00
										 |  |  |   // Define Rule and instantiate basic rules
 | 
					
						
							|  |  |  |   typedef rule<phrase_scanner_t> Rule; | 
					
						
							| 
									
										
										
										
											2014-12-02 03:03:26 +08:00
										 |  |  |   BasicRules<phrase_scanner_t> basic; | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-02 20:12:21 +08:00
										 |  |  |   vector<string> namespaces; // current namespace tag
 | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  |   string currentInclude; | 
					
						
							| 
									
										
										
										
											2014-12-02 20:12:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // parse a full class
 | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  |   Class cls0(verbose),cls(verbose); | 
					
						
							| 
									
										
										
										
											2014-12-02 20:12:21 +08:00
										 |  |  |   Template classTemplate; | 
					
						
							|  |  |  |   ClassGrammar class_g(cls,classTemplate); | 
					
						
							| 
									
										
										
										
											2016-09-09 19:28:13 +08:00
										 |  |  |   Rule class_p = class_g  //
 | 
					
						
							| 
									
										
										
										
											2014-12-02 20:12:21 +08:00
										 |  |  |       [assign_a(cls.namespaces_, namespaces)] | 
					
						
							| 
									
										
										
										
											2016-09-09 19:28:13 +08:00
										 |  |  |       [assign_a(cls.includeFile, currentInclude)][bl::bind( | 
					
						
							|  |  |  |           &handle_possible_template, bl::var(classes), | 
					
						
							|  |  |  |           bl::var(uninstantiatedClasses), bl::var(cls), | 
					
						
							|  |  |  |           bl::var(classTemplate))][clear_a(classTemplate)]  //
 | 
					
						
							|  |  |  |       [assign_a(cls, cls0)]; | 
					
						
							| 
									
										
										
										
											2014-12-02 20:12:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 00:56:22 +08:00
										 |  |  |   // parse "gtsam::Pose2" and add to singleInstantiation.typeList
 | 
					
						
							| 
									
										
										
										
											2014-12-01 03:12:03 +08:00
										 |  |  |   TemplateInstantiationTypedef singleInstantiation, singleInstantiation0; | 
					
						
							| 
									
										
										
										
											2014-12-01 19:14:08 +08:00
										 |  |  |   TypeListGrammar<'<','>'> typelist_g(singleInstantiation.typeList); | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-01 00:19:46 +08:00
										 |  |  |   // typedef gtsam::RangeFactor<gtsam::Pose2, gtsam::Point2> RangeFactor2D;
 | 
					
						
							| 
									
										
										
										
											2014-12-01 07:33:54 +08:00
										 |  |  |   TypeGrammar instantiationClass_g(singleInstantiation.class_); | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |   Rule templateSingleInstantiation_p = | 
					
						
							| 
									
										
										
										
											2014-12-01 07:33:54 +08:00
										 |  |  |     (str_p("typedef") >> instantiationClass_g >> | 
					
						
							| 
									
										
										
										
											2014-12-01 19:14:08 +08:00
										 |  |  |     typelist_g >> | 
					
						
							| 
									
										
										
										
											2014-12-01 07:33:54 +08:00
										 |  |  |     basic.className_p[assign_a(singleInstantiation.name_)] >> | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     ';') | 
					
						
							| 
									
										
										
										
											2014-12-01 07:33:54 +08:00
										 |  |  |     [assign_a(singleInstantiation.namespaces_, namespaces)] | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     [push_back_a(templateInstantiationTypedefs, singleInstantiation)] | 
					
						
							|  |  |  |     [assign_a(singleInstantiation, singleInstantiation0)]; | 
					
						
							| 
									
										
										
										
											2016-09-13 06:17:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Qualified oldType, newType; | 
					
						
							|  |  |  |   TypeGrammar typedefOldClass_g(oldType), typedefNewClass_g(newType); | 
					
						
							|  |  |  |   Rule typedef_p = | 
					
						
							|  |  |  |       (str_p("typedef") >> typedefOldClass_g >> typedefNewClass_g >> | 
					
						
							|  |  |  |        ';') | 
					
						
							|  |  |  |       [assign_a(oldType.namespaces_, namespaces)] | 
					
						
							|  |  |  |       [assign_a(newType.namespaces_, namespaces)] | 
					
						
							|  |  |  |       [bl::bind(&push_typedef_pair, bl::var(typedefs), bl::var(oldType), | 
					
						
							|  |  |  |                      bl::var(newType), bl::var(currentInclude))]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-02 20:49:25 +08:00
										 |  |  |   // Create grammar for global functions
 | 
					
						
							| 
									
										
										
										
											2016-11-23 06:09:35 +08:00
										 |  |  |   GlobalFunctionGrammar global_function_g(global_functions, namespaces, | 
					
						
							|  |  |  |                                           currentInclude); | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Rule include_p = str_p("#include") >> ch_p('<') >> | 
					
						
							|  |  |  |                    (*(anychar_p - '>'))[push_back_a(includes)] | 
					
						
							|  |  |  |                    [assign_a(currentInclude)] >> | 
					
						
							|  |  |  |                    ch_p('>'); | 
					
						
							| 
									
										
										
										
											2014-02-24 07:53:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef __clang__
 | 
					
						
							|  |  |  | #pragma clang diagnostic push
 | 
					
						
							|  |  |  | #pragma clang diagnostic ignored "-Wuninitialized"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Rule namespace_def_p = | 
					
						
							|  |  |  |       (str_p("namespace") | 
					
						
							| 
									
										
										
										
											2014-12-01 07:33:54 +08:00
										 |  |  |       >> basic.namespace_p[push_back_a(namespaces)] | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  |       >> ch_p('{') | 
					
						
							| 
									
										
										
										
											2016-09-13 06:17:47 +08:00
										 |  |  |       >> *(include_p | class_p | templateSingleInstantiation_p | typedef_p | global_function_g | namespace_def_p | basic.comments_p) | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  |       >> ch_p('}')) | 
					
						
							|  |  |  |       [pop_a(namespaces)]; | 
					
						
							| 
									
										
										
										
											2014-02-24 07:53:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef __clang__
 | 
					
						
							|  |  |  | #pragma clang diagnostic pop
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 01:04:38 +08:00
										 |  |  |   // parse forward declaration
 | 
					
						
							|  |  |  |   ForwardDeclaration fwDec0, fwDec; | 
					
						
							| 
									
										
										
										
											2016-12-20 06:47:30 +08:00
										 |  |  |   Class fwParentClass; | 
					
						
							|  |  |  |   TypeGrammar className_g(fwDec.cls); | 
					
						
							|  |  |  |   TypeGrammar classParent_g(fwParentClass); | 
					
						
							|  |  |  |   Rule classParent_p = (':' >> classParent_g >> ';') //
 | 
					
						
							|  |  |  |          [bl::bind(&Class::assignParent, bl::var(fwDec.cls), | 
					
						
							|  |  |  |               bl::var(fwParentClass))][clear_a(fwParentClass)]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-24 07:53:48 +08:00
										 |  |  |   Rule forward_declaration_p = | 
					
						
							| 
									
										
										
										
											2016-08-11 10:23:26 +08:00
										 |  |  |       !(str_p("virtual")[assign_a(fwDec.isVirtual, T)]) | 
					
						
							| 
									
										
										
										
											2016-12-20 06:47:30 +08:00
										 |  |  |       >> str_p("class") >> className_g | 
					
						
							|  |  |  |       >> (classParent_p | ';') | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |       [push_back_a(forward_declarations, fwDec)] | 
					
						
							| 
									
										
										
										
											2014-12-19 23:19:02 +08:00
										 |  |  |       [assign_a(cls,cls0)] // also clear class to avoid partial parse
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |       [assign_a(fwDec, fwDec0)]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-11 10:18:53 +08:00
										 |  |  |   Rule module_content_p = basic.comments_p | include_p | class_p | 
					
						
							| 
									
										
										
										
											2014-12-01 03:12:03 +08:00
										 |  |  |       | templateSingleInstantiation_p | forward_declaration_p | 
					
						
							| 
									
										
										
										
											2014-12-02 20:49:25 +08:00
										 |  |  |       | global_function_g | namespace_def_p; | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Rule module_p = *module_content_p >> !end_p; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:19 +08:00
										 |  |  |   // and parse contents
 | 
					
						
							|  |  |  |   parse_info<const char*> info = parse(data.c_str(), module_p, space_p); | 
					
						
							|  |  |  |   if(!info.full) { | 
					
						
							|  |  |  |     printf("parsing stopped at \n%.20s\n",info.stop); | 
					
						
							| 
									
										
										
										
											2014-12-02 20:12:21 +08:00
										 |  |  |     cout << "Stopped in:\n" | 
					
						
							|  |  |  |       "class '" << cls.name_ << "'" << endl; | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:19 +08:00
										 |  |  |     throw ParseFailed((int)info.length); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-09-09 03:51:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:55 +08:00
										 |  |  |   // Post-process classes for serialization markers
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(Class& cls: classes) | 
					
						
							| 
									
										
										
										
											2014-11-13 06:22:59 +08:00
										 |  |  |     cls.erase_serialization(); | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:55 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 19:28:13 +08:00
										 |  |  |   for(Class& cls: uninstantiatedClasses) | 
					
						
							|  |  |  |     cls.erase_serialization(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:55 +08:00
										 |  |  |   // Explicitly add methods to the classes from parents so it shows in documentation
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(Class& cls: classes) | 
					
						
							| 
									
										
										
										
											2014-11-13 06:22:59 +08:00
										 |  |  |     cls.appendInheritedMethods(cls, classes); | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-13 00:38:04 +08:00
										 |  |  |   // - Remove inherited methods for Cython classes in the pxd, otherwise Cython can't decide which one to call.
 | 
					
						
							|  |  |  |   // - Only inherited nontemplateMethods_ in uninstantiatedClasses need to be removed
 | 
					
						
							|  |  |  |   // because that what we serialized to the pxd.
 | 
					
						
							|  |  |  |   // - However, we check against the class parent's *methods_* to avoid looking into
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |   // its grand parent and grand-grand parent, etc., because all those are already
 | 
					
						
							| 
									
										
										
										
											2016-09-13 00:38:04 +08:00
										 |  |  |   // added in its direct parent.
 | 
					
						
							|  |  |  |   // - So this must be called *after* the above code appendInheritedMethods!!
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |   for(Class& cls: uninstantiatedClasses) | 
					
						
							| 
									
										
										
										
											2016-09-13 00:38:04 +08:00
										 |  |  |       cls.removeInheritedNontemplateMethods(uninstantiatedClasses); | 
					
						
							| 
									
										
										
										
											2016-09-12 23:05:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |   // Expand templates - This is done first so that template instantiations are
 | 
					
						
							|  |  |  |   // counted in the list of valid types, have their attributes and dependencies
 | 
					
						
							|  |  |  |   // checked, etc.
 | 
					
						
							| 
									
										
										
										
											2014-11-15 00:04:45 +08:00
										 |  |  |   expandedClasses = ExpandTypedefInstantiations(classes, | 
					
						
							|  |  |  |       templateInstantiationTypedefs); | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Dependency check list
 | 
					
						
							| 
									
										
										
										
											2014-11-15 00:04:45 +08:00
										 |  |  |   vector<string> validTypes = GenerateValidTypes(expandedClasses, | 
					
						
							| 
									
										
										
										
											2016-11-14 12:59:56 +08:00
										 |  |  |       forward_declarations, typedefs); | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Check that all classes have been defined somewhere
 | 
					
						
							|  |  |  |   verifyArguments<GlobalFunction>(validTypes, global_functions); | 
					
						
							|  |  |  |   verifyReturnTypes<GlobalFunction>(validTypes, global_functions); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-15 00:04:45 +08:00
										 |  |  |   hasSerialiable = false; | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: expandedClasses) | 
					
						
							| 
									
										
										
										
											2014-11-13 06:22:59 +08:00
										 |  |  |     cls.verifyAll(validTypes,hasSerialiable); | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Create type attributes table and check validity
 | 
					
						
							|  |  |  |   typeAttributes.addClasses(expandedClasses); | 
					
						
							|  |  |  |   typeAttributes.addForwardDeclarations(forward_declarations); | 
					
						
							| 
									
										
										
										
											2016-11-25 08:24:25 +08:00
										 |  |  |   for (const TypedefPair p: typedefs) | 
					
						
							|  |  |  |     typeAttributes.addType(p.newType); | 
					
						
							| 
									
										
										
										
											2014-11-30 03:01:48 +08:00
										 |  |  |   // add Eigen types as template arguments are also checked ?
 | 
					
						
							|  |  |  |   vector<ForwardDeclaration> eigen; | 
					
						
							|  |  |  |   eigen.push_back(ForwardDeclaration("Vector")); | 
					
						
							|  |  |  |   eigen.push_back(ForwardDeclaration("Matrix")); | 
					
						
							|  |  |  |   typeAttributes.addForwardDeclarations(eigen); | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |   typeAttributes.checkValidity(expandedClasses); | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2017-03-11 12:27:29 +08:00
										 |  |  | void Module::generate_matlab_wrapper(const string& toolboxPath) const { | 
					
						
							| 
									
										
										
										
											2014-11-15 00:04:45 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   fs::create_directories(toolboxPath); | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:57 +08:00
										 |  |  |   // create the unified .cpp switch file
 | 
					
						
							| 
									
										
										
										
											2017-03-19 10:01:24 +08:00
										 |  |  |   const string wrapperName = name + "_wrapper"; | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:57 +08:00
										 |  |  |   string wrapperFileName = toolboxPath + "/" + wrapperName + ".cpp"; | 
					
						
							|  |  |  |   FileWriter wrapperFile(wrapperFileName, verbose, "//"); | 
					
						
							|  |  |  |   wrapperFile.oss << "#include <wrap/matlab.h>\n"; | 
					
						
							|  |  |  |   wrapperFile.oss << "#include <map>\n"; | 
					
						
							|  |  |  |   wrapperFile.oss << "\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:50:03 +08:00
										 |  |  |   // Include boost.serialization archive headers before other class headers
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:57 +08:00
										 |  |  |   if (hasSerialiable) { | 
					
						
							|  |  |  |     wrapperFile.oss << "#include <boost/archive/text_iarchive.hpp>\n"; | 
					
						
							| 
									
										
										
										
											2018-12-31 05:24:19 +08:00
										 |  |  |     wrapperFile.oss << "#include <boost/archive/text_oarchive.hpp>\n"; | 
					
						
							|  |  |  |     wrapperFile.oss << "#include <boost/serialization/export.hpp>\n\n"; | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:57 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-06-20 01:50:03 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Generate includes while avoiding redundant includes
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |   generateIncludes(wrapperFile); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-15 00:04:45 +08:00
										 |  |  |   // create typedef classes - we put this at the top of the wrap file so that
 | 
					
						
							|  |  |  |   // collectors and method arguments can use these typedefs
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: expandedClasses) | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |     if(!cls.typedefName.empty()) | 
					
						
							|  |  |  |       wrapperFile.oss << cls.getTypedef() << "\n"; | 
					
						
							|  |  |  |   wrapperFile.oss << "\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:50:03 +08:00
										 |  |  |   // Generate boost.serialization export flags (needs typedefs from above)
 | 
					
						
							|  |  |  |   if (hasSerialiable) { | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |     for(const Class& cls: expandedClasses) | 
					
						
							| 
									
										
										
										
											2013-06-20 01:50:03 +08:00
										 |  |  |       if(cls.isSerializable) | 
					
						
							|  |  |  |         wrapperFile.oss << cls.getSerializationExport() << "\n"; | 
					
						
							|  |  |  |     wrapperFile.oss << "\n"; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |   // Generate collectors and cleanup function to be called from mexAtExit
 | 
					
						
							|  |  |  |   WriteCollectorsAndCleanupFcn(wrapperFile, name, expandedClasses); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // generate RTTI registry (for returning derived-most types)
 | 
					
						
							|  |  |  |   WriteRTTIRegistry(wrapperFile, name, expandedClasses); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 01:49:57 +08:00
										 |  |  |   vector<string> functionNames; // Function names stored by index for switch
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |   // create proxy class and wrapper code
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: expandedClasses) | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |     cls.matlab_proxy(toolboxPath, wrapperName, typeAttributes, wrapperFile, functionNames); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // create matlab files and wrapper code for global functions
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const GlobalFunctions::value_type& p: global_functions) | 
					
						
							| 
									
										
										
										
											2012-11-28 03:03:23 +08:00
										 |  |  |     p.second.matlab_proxy(toolboxPath, wrapperName, typeAttributes, wrapperFile, functionNames); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // finish wrapper file
 | 
					
						
							|  |  |  |   wrapperFile.oss << "\n"; | 
					
						
							|  |  |  |   finish_wrapper(wrapperFile, functionNames); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wrapperFile.emit(true); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-09-09 03:51:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2017-03-11 12:27:29 +08:00
										 |  |  | void Module::generate_cython_wrapper(const string& toolboxPath, const std::string& pxdImports) const { | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  |   fs::create_directories(toolboxPath); | 
					
						
							| 
									
										
										
										
											2016-12-16 13:23:45 +08:00
										 |  |  |   string pxdFileName = toolboxPath + "/" + name + ".pxd"; | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  |   FileWriter pxdFile(pxdFileName, verbose, "#"); | 
					
						
							| 
									
										
										
										
											2016-12-16 13:23:45 +08:00
										 |  |  |   pxdFile.oss << pxdImports << "\n"; | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  |   emit_cython_pxd(pxdFile); | 
					
						
							|  |  |  |   string pyxFileName = toolboxPath + "/" + name + ".pyx"; | 
					
						
							|  |  |  |   FileWriter pyxFile(pyxFileName, verbose, "#"); | 
					
						
							|  |  |  |   emit_cython_pyx(pyxFile); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  | void Module::emit_cython_pxd(FileWriter& pxdFile) const { | 
					
						
							|  |  |  |   // headers
 | 
					
						
							| 
									
										
										
										
											2017-07-29 03:26:19 +08:00
										 |  |  |   pxdFile.oss << "from gtsam_eigency.core cimport *\n" | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  |                  "from libcpp.string cimport string\n" | 
					
						
							|  |  |  |                  "from libcpp.vector cimport vector\n" | 
					
						
							|  |  |  |                  "from libcpp.pair cimport pair\n" | 
					
						
							|  |  |  |                  "from libcpp.set cimport set\n" | 
					
						
							|  |  |  |                  "from libcpp.map cimport map\n" | 
					
						
							|  |  |  |                  "from libcpp cimport bool\n\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // boost shared_ptr
 | 
					
						
							|  |  |  |   pxdFile.oss << "cdef extern from \"boost/shared_ptr.hpp\" namespace \"boost\":\n" | 
					
						
							| 
									
										
										
										
											2016-11-30 18:56:07 +08:00
										 |  |  |                  "    cppclass shared_ptr[T]:\n" | 
					
						
							|  |  |  |                  "        shared_ptr()\n" | 
					
						
							|  |  |  |                  "        shared_ptr(T*)\n" | 
					
						
							|  |  |  |                  "        T* get()\n" | 
					
						
							| 
									
										
										
										
											2017-08-07 05:58:23 +08:00
										 |  |  |                  "        long use_count() const\n" | 
					
						
							| 
									
										
										
										
											2016-11-30 18:56:07 +08:00
										 |  |  |                  "        T& operator*()\n\n" | 
					
						
							| 
									
										
										
										
											2020-06-21 10:21:17 +08:00
										 |  |  |                  "    cdef shared_ptr[T] dynamic_pointer_cast[T,U](const shared_ptr[U]& r)\n\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // gtsam alignment-friendly shared_ptr
 | 
					
						
							|  |  |  |   pxdFile.oss << "cdef extern from \"gtsam/base/make_shared.h\" namespace \"gtsam\":\n" | 
					
						
							| 
									
										
										
										
											2017-08-07 05:58:23 +08:00
										 |  |  |                  "    cdef shared_ptr[T] make_shared[T](const T& r)\n\n"; | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-13 06:17:47 +08:00
										 |  |  |   for(const TypedefPair& types: typedefs) | 
					
						
							|  |  |  |     types.emit_cython_pxd(pxdFile); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 09:50:55 +08:00
										 |  |  |   //... wrap all classes
 | 
					
						
							| 
									
										
										
										
											2016-09-13 06:36:45 +08:00
										 |  |  |   for (const Class& cls : uninstantiatedClasses) { | 
					
						
							| 
									
										
										
										
											2017-08-07 03:21:26 +08:00
										 |  |  |     cls.emit_cython_pxd(pxdFile); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 08:42:20 +08:00
										 |  |  |     for (const Class& expCls : expandedClasses) { | 
					
						
							|  |  |  |       bool matchingNonTemplated = !expCls.templateClass | 
					
						
							|  |  |  |           && expCls.pxdClassName() == cls.pxdClassName(); | 
					
						
							|  |  |  |       bool isTemplatedFromCls = expCls.templateClass | 
					
						
							|  |  |  |           && expCls.templateClass->pxdClassName() == cls.pxdClassName(); | 
					
						
							| 
									
										
										
										
											2017-08-07 03:21:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       // ctypedef for template instantiations
 | 
					
						
							|  |  |  |       if (isTemplatedFromCls) { | 
					
						
							|  |  |  |         pxdFile.oss << "\n"; | 
					
						
							| 
									
										
										
										
											2017-08-07 08:42:20 +08:00
										 |  |  |         pxdFile.oss << "ctypedef " << expCls.templateClass->pxdClassName() | 
					
						
							| 
									
										
										
										
											2017-08-07 03:21:26 +08:00
										 |  |  |             << "["; | 
					
						
							| 
									
										
										
										
											2017-08-07 08:42:20 +08:00
										 |  |  |         for (size_t i = 0; i < expCls.templateInstTypeList.size(); ++i) | 
					
						
							|  |  |  |           pxdFile.oss << expCls.templateInstTypeList[i].pxdClassName() | 
					
						
							|  |  |  |               << ((i == expCls.templateInstTypeList.size() - 1) ? "" : ", "); | 
					
						
							|  |  |  |         pxdFile.oss << "] " << expCls.pxdClassName() << "\n"; | 
					
						
							| 
									
										
										
										
											2016-09-13 06:36:45 +08:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2017-08-07 03:21:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       // Python wrapper class
 | 
					
						
							|  |  |  |       if (isTemplatedFromCls || matchingNonTemplated) { | 
					
						
							| 
									
										
										
										
											2017-08-07 08:42:20 +08:00
										 |  |  |         expCls.emit_cython_wrapper_pxd(pxdFile); | 
					
						
							| 
									
										
										
										
											2017-08-07 03:21:26 +08:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pxdFile.oss << "\n\n"; | 
					
						
							| 
									
										
										
										
											2016-09-10 09:50:55 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-11-23 06:09:35 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   //... wrap global functions
 | 
					
						
							|  |  |  |   for(const GlobalFunctions::value_type& p: global_functions) | 
					
						
							|  |  |  |     p.second.emit_cython_pxd(pxdFile); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  |   pxdFile.emit(true); | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-10 00:01:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:00:15 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  | void Module::emit_cython_pyx(FileWriter& pyxFile) const { | 
					
						
							| 
									
										
										
										
											2019-01-08 18:28:19 +08:00
										 |  |  |   // directives...
 | 
					
						
							|  |  |  |   // allow str to automatically coerce to std::string and back (for python3)
 | 
					
						
							|  |  |  |   pyxFile.oss << "# cython: c_string_type=str, c_string_encoding=ascii\n\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  |   // headers...
 | 
					
						
							| 
									
										
										
										
											2016-12-16 13:23:45 +08:00
										 |  |  |   string pxdHeader = name; | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  |   pyxFile.oss << "cimport numpy as np\n" | 
					
						
							| 
									
										
										
										
											2016-11-30 18:57:12 +08:00
										 |  |  |                  "import numpy as npp\n" | 
					
						
							| 
									
										
										
										
											2016-12-16 13:23:45 +08:00
										 |  |  |                  "cimport " << pxdHeader << "\n" | 
					
						
							| 
									
										
										
										
											2019-01-30 17:49:38 +08:00
										 |  |  |                  "from ."<< pxdHeader << " cimport shared_ptr\n" | 
					
						
							|  |  |  |                  "from ."<< pxdHeader << " cimport dynamic_pointer_cast\n" | 
					
						
							|  |  |  |                  "from ."<< pxdHeader << " cimport make_shared\n"; | 
					
						
							| 
									
										
										
										
											2017-05-21 03:23:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 02:07:13 +08:00
										 |  |  |   pyxFile.oss << "# C helper function that copies all arguments into a positional list.\n" | 
					
						
							|  |  |  |                  "cdef list process_args(list keywords, tuple args, dict kwargs):\n" | 
					
						
							|  |  |  |                  "   cdef str keyword\n" | 
					
						
							|  |  |  |                  "   cdef int n = len(args), m = len(keywords)\n" | 
					
						
							|  |  |  |                  "   cdef list params = list(args)\n" | 
					
						
							|  |  |  |                  "   assert len(args)+len(kwargs) == m, 'Expected {} arguments'.format(m)\n" | 
					
						
							|  |  |  |                  "   try:\n" | 
					
						
							|  |  |  |                  "       return params + [kwargs[keyword] for keyword in keywords[n:]]\n" | 
					
						
							|  |  |  |                  "   except:\n" | 
					
						
							|  |  |  |                  "       raise ValueError('Epected arguments ' + str(keywords))\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-23 01:13:33 +08:00
										 |  |  |   // import all typedefs, e.g. from gtsam_wrapper cimport Key, so we don't need to say gtsam.Key
 | 
					
						
							| 
									
										
										
										
											2016-11-14 13:08:42 +08:00
										 |  |  |   for(const Qualified& q: Qualified::BasicTypedefs) { | 
					
						
							| 
									
										
										
										
											2016-11-23 01:13:33 +08:00
										 |  |  |     pyxFile.oss << "from " << pxdHeader << " cimport " << q.pxdClassName() << "\n"; | 
					
						
							| 
									
										
										
										
											2016-11-14 13:08:42 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-07-29 03:26:19 +08:00
										 |  |  |   pyxFile.oss << "from gtsam_eigency.core cimport *\n" | 
					
						
							| 
									
										
										
										
											2016-09-11 07:50:12 +08:00
										 |  |  |                  "from libcpp cimport bool\n\n" | 
					
						
							|  |  |  |                  "from libcpp.pair cimport pair\n" | 
					
						
							| 
									
										
										
										
											2016-09-10 10:28:15 +08:00
										 |  |  |                  "from libcpp.string cimport string\n" | 
					
						
							|  |  |  |                  "from cython.operator cimport dereference as deref\n\n\n"; | 
					
						
							| 
									
										
										
										
											2016-11-30 18:57:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-20 06:47:30 +08:00
										 |  |  |   // all classes include all forward declarations
 | 
					
						
							|  |  |  |   std::vector<Class> allClasses = expandedClasses; | 
					
						
							|  |  |  |   for(const ForwardDeclaration& fd: forward_declarations) | 
					
						
							|  |  |  |     allClasses.push_back(fd.cls); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 03:52:44 +08:00
										 |  |  |   for(const Class& cls: expandedClasses) | 
					
						
							| 
									
										
										
										
											2016-12-20 06:47:30 +08:00
										 |  |  |     cls.emit_cython_pyx(pyxFile, allClasses); | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  |   pyxFile.oss << "\n"; | 
					
						
							| 
									
										
										
										
											2017-05-21 03:23:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-23 06:09:35 +08:00
										 |  |  |   //... wrap global functions
 | 
					
						
							|  |  |  |   for(const GlobalFunctions::value_type& p: global_functions) | 
					
						
							|  |  |  |     p.second.emit_cython_pyx(pyxFile); | 
					
						
							| 
									
										
										
										
											2016-09-09 01:33:32 +08:00
										 |  |  |   pyxFile.emit(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2014-11-15 00:04:45 +08:00
										 |  |  | void Module::generateIncludes(FileWriter& file) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // collect includes
 | 
					
						
							|  |  |  |   vector<string> all_includes(includes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // sort and remove duplicates
 | 
					
						
							|  |  |  |   sort(all_includes.begin(), all_includes.end()); | 
					
						
							|  |  |  |   vector<string>::const_iterator last_include = unique(all_includes.begin(), all_includes.end()); | 
					
						
							|  |  |  |   vector<string>::const_iterator it = all_includes.begin(); | 
					
						
							|  |  |  |   // add includes to file
 | 
					
						
							|  |  |  |   for (; it != last_include; ++it) | 
					
						
							|  |  |  |     file.oss << "#include <" << *it << ">" << endl; | 
					
						
							|  |  |  |   file.oss << "\n"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |   void Module::finish_wrapper(FileWriter& file, const std::vector<std::string>& functionNames) const { | 
					
						
							|  |  |  |     file.oss << "void mexFunction(int nargout, mxArray *out[], int nargin, const mxArray *in[])\n"; | 
					
						
							|  |  |  |     file.oss << "{\n"; | 
					
						
							|  |  |  |     file.oss << "  mstream mout;\n"; // Send stdout to MATLAB console
 | 
					
						
							|  |  |  |     file.oss << "  std::streambuf *outbuf = std::cout.rdbuf(&mout);\n\n"; | 
					
						
							|  |  |  |     file.oss << "  _" << name << "_RTTIRegister();\n\n"; | 
					
						
							|  |  |  |     file.oss << "  int id = unwrap<int>(in[0]);\n\n"; | 
					
						
							|  |  |  |     file.oss << "  try {\n"; | 
					
						
							|  |  |  |     file.oss << "    switch(id) {\n"; | 
					
						
							|  |  |  |     for(size_t id = 0; id < functionNames.size(); ++id) { | 
					
						
							|  |  |  |       file.oss << "    case " << id << ":\n"; | 
					
						
							|  |  |  |       file.oss << "      " << functionNames[id] << "(nargout, out, nargin-1, in+1);\n"; | 
					
						
							|  |  |  |       file.oss << "      break;\n"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     file.oss << "    }\n"; | 
					
						
							|  |  |  |     file.oss << "  } catch(const std::exception& e) {\n"; | 
					
						
							|  |  |  |     file.oss << "    mexErrMsgTxt((\"Exception from gtsam:\\n\" + std::string(e.what()) + \"\\n\").c_str());\n"; | 
					
						
							|  |  |  |     file.oss << "  }\n"; | 
					
						
							|  |  |  |     file.oss << "\n"; | 
					
						
							|  |  |  |     file.oss << "  std::cout.rdbuf(outbuf);\n"; // Restore cout
 | 
					
						
							|  |  |  |     file.oss << "}\n"; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | vector<Class> Module::ExpandTypedefInstantiations(const vector<Class>& classes, const vector<TemplateInstantiationTypedef> instantiations) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   vector<Class> expandedClasses = classes; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const TemplateInstantiationTypedef& inst: instantiations) { | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     // Add the new class to the list
 | 
					
						
							|  |  |  |     expandedClasses.push_back(inst.findAndExpand(classes)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Remove all template classes
 | 
					
						
							|  |  |  |   for(size_t i = 0; i < expandedClasses.size(); ++i) | 
					
						
							|  |  |  |     if(!expandedClasses[i].templateArgs.empty()) { | 
					
						
							|  |  |  |       expandedClasses.erase(expandedClasses.begin() + size_t(i)); | 
					
						
							|  |  |  |       -- i; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return expandedClasses; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | vector<string> Module::GenerateValidTypes(const vector<Class>& classes, const vector<ForwardDeclaration>& forwardDeclarations, const vector<TypedefPair>& typedefs) { | 
					
						
							|  |  |  |   vector<string> validTypes; | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const ForwardDeclaration& fwDec: forwardDeclarations) { | 
					
						
							| 
									
										
										
										
											2016-12-20 06:47:30 +08:00
										 |  |  |     validTypes.push_back(fwDec.name()); | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |   } | 
					
						
							|  |  |  |   validTypes.push_back("void"); | 
					
						
							|  |  |  |   validTypes.push_back("string"); | 
					
						
							|  |  |  |   validTypes.push_back("int"); | 
					
						
							|  |  |  |   validTypes.push_back("bool"); | 
					
						
							|  |  |  |   validTypes.push_back("char"); | 
					
						
							|  |  |  |   validTypes.push_back("unsigned char"); | 
					
						
							|  |  |  |   validTypes.push_back("size_t"); | 
					
						
							|  |  |  |   validTypes.push_back("double"); | 
					
						
							|  |  |  |   validTypes.push_back("Vector"); | 
					
						
							|  |  |  |   validTypes.push_back("Matrix"); | 
					
						
							|  |  |  |   //Create a list of parsed classes for dependency checking
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: classes) { | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     validTypes.push_back(cls.qualifiedName("::")); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-11-14 12:59:56 +08:00
										 |  |  |   for(const TypedefPair& p: typedefs) { | 
					
						
							|  |  |  |     validTypes.push_back(p.newType.qualifiedName("::")); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return validTypes; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | void Module::WriteCollectorsAndCleanupFcn(FileWriter& wrapperFile, const std::string& moduleName, const std::vector<Class>& classes) { | 
					
						
							|  |  |  |   // Generate all collectors
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: classes) { | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     const string matlabUniqueName = cls.qualifiedName(), | 
					
						
							|  |  |  |       cppName = cls.qualifiedName("::"); | 
					
						
							|  |  |  |     wrapperFile.oss << "typedef std::set<boost::shared_ptr<" << cppName << ">*> " | 
					
						
							|  |  |  |       << "Collector_" << matlabUniqueName << ";\n"; | 
					
						
							|  |  |  |     wrapperFile.oss << "static Collector_" << matlabUniqueName << | 
					
						
							|  |  |  |       " collector_" << matlabUniqueName << ";\n"; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // generate mexAtExit cleanup function
 | 
					
						
							|  |  |  |   wrapperFile.oss << | 
					
						
							|  |  |  |     "\nvoid _deleteAllObjects()\n" | 
					
						
							|  |  |  |     "{\n" | 
					
						
							|  |  |  |     "  mstream mout;\n" // Send stdout to MATLAB console
 | 
					
						
							|  |  |  |     "  std::streambuf *outbuf = std::cout.rdbuf(&mout);\n\n" | 
					
						
							|  |  |  |     "  bool anyDeleted = false;\n"; | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: classes) { | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     const string matlabUniqueName = cls.qualifiedName(); | 
					
						
							|  |  |  |     const string cppName = cls.qualifiedName("::"); | 
					
						
							|  |  |  |     const string collectorType = "Collector_" + matlabUniqueName; | 
					
						
							|  |  |  |     const string collectorName = "collector_" + matlabUniqueName; | 
					
						
							|  |  |  |     // The extra curly-braces around the for loops work around a limitation in MSVC (existing
 | 
					
						
							|  |  |  |     // since 2005!) preventing more than 248 blocks.
 | 
					
						
							|  |  |  |     wrapperFile.oss << | 
					
						
							|  |  |  |       "  { for(" << collectorType << "::iterator iter = " << collectorName << ".begin();\n" | 
					
						
							|  |  |  |       "      iter != " << collectorName << ".end(); ) {\n" | 
					
						
							|  |  |  |       "    delete *iter;\n" | 
					
						
							|  |  |  |       "    " << collectorName << ".erase(iter++);\n" | 
					
						
							|  |  |  |       "    anyDeleted = true;\n" | 
					
						
							|  |  |  |       "  } }\n"; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   wrapperFile.oss << | 
					
						
							|  |  |  |     "  if(anyDeleted)\n" | 
					
						
							|  |  |  |     "    cout <<\n" | 
					
						
							|  |  |  |     "      \"WARNING:  Wrap modules with variables in the workspace have been reloaded due to\\n\"\n" | 
					
						
							|  |  |  |     "      \"calling destructors, call 'clear all' again if you plan to now recompile a wrap\\n\"\n" | 
					
						
							|  |  |  |     "      \"module, so that your recompiled module is used instead of the old one.\" << endl;\n" | 
					
						
							|  |  |  |     "  std::cout.rdbuf(outbuf);\n" // Restore cout
 | 
					
						
							|  |  |  |     "}\n\n"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | void Module::WriteRTTIRegistry(FileWriter& wrapperFile, const std::string& moduleName, const std::vector<Class>& classes) { | 
					
						
							|  |  |  |   wrapperFile.oss << | 
					
						
							|  |  |  |     "void _" << moduleName << "_RTTIRegister() {\n" | 
					
						
							|  |  |  |     "  const mxArray *alreadyCreated = mexGetVariablePtr(\"global\", \"gtsam_" + moduleName + "_rttiRegistry_created\");\n" | 
					
						
							|  |  |  |     "  if(!alreadyCreated) {\n" | 
					
						
							|  |  |  |     "    std::map<std::string, std::string> types;\n"; | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const Class& cls: classes) { | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     if(cls.isVirtual) | 
					
						
							|  |  |  |       wrapperFile.oss << | 
					
						
							|  |  |  |       "    types.insert(std::make_pair(typeid(" << cls.qualifiedName("::") << ").name(), \"" << cls.qualifiedName(".") << "\"));\n"; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   wrapperFile.oss << "\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wrapperFile.oss << | 
					
						
							|  |  |  |     "    mxArray *registry = mexGetVariable(\"global\", \"gtsamwrap_rttiRegistry\");\n" | 
					
						
							|  |  |  |     "    if(!registry)\n" | 
					
						
							|  |  |  |     "      registry = mxCreateStructMatrix(1, 1, 0, NULL);\n" | 
					
						
							|  |  |  |     "    typedef std::pair<std::string, std::string> StringPair;\n" | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |     "    for(const StringPair& rtti_matlab: types) {\n" | 
					
						
							| 
									
										
										
										
											2016-11-19 00:01:02 +08:00
										 |  |  |     "      int fieldId = mxAddField(registry, rtti_matlab.first.c_str());\n" | 
					
						
							|  |  |  |     "      if(fieldId < 0)\n" | 
					
						
							|  |  |  |     "        mexErrMsgTxt(\"gtsam wrap:  Error indexing RTTI types, inheritance will not work correctly\");\n" | 
					
						
							|  |  |  |     "      mxArray *matlabName = mxCreateString(rtti_matlab.second.c_str());\n" | 
					
						
							|  |  |  |     "      mxSetFieldByNumber(registry, 0, fieldId, matlabName);\n" | 
					
						
							|  |  |  |     "    }\n" | 
					
						
							|  |  |  |     "    if(mexPutVariable(\"global\", \"gtsamwrap_rttiRegistry\", registry) != 0)\n" | 
					
						
							|  |  |  |     "      mexErrMsgTxt(\"gtsam wrap:  Error indexing RTTI types, inheritance will not work correctly\");\n" | 
					
						
							|  |  |  |     "    mxDestroyArray(registry);\n" | 
					
						
							|  |  |  |     "    \n" | 
					
						
							|  |  |  |     "    mxArray *newAlreadyCreated = mxCreateNumericMatrix(0, 0, mxINT8_CLASS, mxREAL);\n" | 
					
						
							|  |  |  |     "    if(mexPutVariable(\"global\", \"gtsam_" + moduleName + "_rttiRegistry_created\", newAlreadyCreated) != 0)\n" | 
					
						
							|  |  |  |     "      mexErrMsgTxt(\"gtsam wrap:  Error indexing RTTI types, inheritance will not work correctly\");\n" | 
					
						
							|  |  |  |     "    mxDestroyArray(newAlreadyCreated);\n" | 
					
						
							|  |  |  |     "  }\n" | 
					
						
							|  |  |  |     "}\n" | 
					
						
							|  |  |  |     "\n"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2017-03-11 12:27:29 +08:00
										 |  |  | void Module::generate_python_wrapper(const string& toolboxPath) const { | 
					
						
							| 
									
										
										
										
											2014-11-15 00:47:25 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   fs::create_directories(toolboxPath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // create the unified .cpp switch file
 | 
					
						
							|  |  |  |   const string wrapperName = name + "_python"; | 
					
						
							|  |  |  |   string wrapperFileName = toolboxPath + "/" + wrapperName + ".cpp"; | 
					
						
							|  |  |  |   FileWriter wrapperFile(wrapperFileName, verbose, "//"); | 
					
						
							|  |  |  |   wrapperFile.oss << "#include <boost/python.hpp>\n\n"; | 
					
						
							|  |  |  |   wrapperFile.oss << "using namespace boost::python;\n"; | 
					
						
							|  |  |  |   wrapperFile.oss << "BOOST_PYTHON_MODULE(" + name + ")\n"; | 
					
						
							|  |  |  |   wrapperFile.oss << "{\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // write out classes
 | 
					
						
							| 
									
										
										
										
											2016-09-11 10:18:53 +08:00
										 |  |  |   for(const Class& cls: expandedClasses) { | 
					
						
							| 
									
										
										
										
											2014-11-15 00:47:25 +08:00
										 |  |  |     cls.python_wrapper(wrapperFile); | 
					
						
							| 
									
										
										
										
											2016-09-09 19:28:13 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-11-15 00:47:25 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // write out global functions
 | 
					
						
							| 
									
										
										
										
											2016-05-21 09:41:18 +08:00
										 |  |  |   for(const GlobalFunctions::value_type& p: global_functions) | 
					
						
							| 
									
										
										
										
											2014-11-15 00:47:25 +08:00
										 |  |  |     p.second.python_wrapper(wrapperFile); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // finish wrapper file
 | 
					
						
							|  |  |  |   wrapperFile.oss << "}\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   wrapperFile.emit(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ |