243 lines
7.1 KiB
C++
243 lines
7.1 KiB
C++
/* ----------------------------------------------------------------------------
|
|
|
|
* 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 Argument.h
|
|
* @brief arguments to constructors and methods
|
|
* @author Frank Dellaert
|
|
* @author Andrew Melim
|
|
* @author Richard Roberts
|
|
**/
|
|
|
|
#pragma once
|
|
|
|
#include "TemplateSubstitution.h"
|
|
#include "FileWriter.h"
|
|
#include "ReturnValue.h"
|
|
|
|
namespace wrap {
|
|
|
|
/// Argument class
|
|
struct Argument {
|
|
Qualified type;
|
|
std::string name;
|
|
bool is_const, is_ref, is_ptr;
|
|
|
|
Argument() :
|
|
is_const(false), is_ref(false), is_ptr(false) {
|
|
}
|
|
|
|
Argument(const Qualified& t, const std::string& n) :
|
|
type(t), name(n), is_const(false), is_ref(false), is_ptr(false) {
|
|
}
|
|
|
|
bool isSameSignature(const Argument& other) const {
|
|
return type == other.type
|
|
&& is_const == other.is_const && is_ref == other.is_ref
|
|
&& is_ptr == other.is_ptr;
|
|
}
|
|
|
|
bool operator==(const Argument& other) const {
|
|
return type == other.type && name == other.name
|
|
&& is_const == other.is_const && is_ref == other.is_ref
|
|
&& is_ptr == other.is_ptr;
|
|
}
|
|
|
|
Argument expandTemplate(const TemplateSubstitution& ts) const;
|
|
|
|
/// return MATLAB class for use in isa(x,class)
|
|
std::string matlabClass(const std::string& delim = "") const;
|
|
|
|
/// MATLAB code generation, MATLAB to C++
|
|
void matlab_unwrap(FileWriter& file, const std::string& matlabName) const;
|
|
|
|
/**
|
|
* emit checking argument to MATLAB proxy
|
|
* @param proxyFile output stream
|
|
*/
|
|
void proxy_check(FileWriter& proxyFile, const std::string& s) const;
|
|
|
|
/**
|
|
* emit arguments for cython pxd
|
|
* @param file output stream
|
|
*/
|
|
void emit_cython_pxd(FileWriter& file, const std::string& className,
|
|
const std::vector<std::string>& templateArgs) const;
|
|
void emit_cython_pyx(FileWriter& file) const;
|
|
std::string pyx_asParam() const;
|
|
std::string pyx_convertEigenTypeAndStorageOrder() const;
|
|
|
|
friend std::ostream& operator<<(std::ostream& os, const Argument& arg) {
|
|
os << (arg.is_const ? "const " : "") << arg.type << (arg.is_ptr ? "*" : "")
|
|
<< (arg.is_ref ? "&" : "");
|
|
return os;
|
|
}
|
|
|
|
};
|
|
|
|
/// Argument list is just a container with Arguments
|
|
struct ArgumentList: public std::vector<Argument> {
|
|
|
|
/// create a comma-separated string listing all argument types (not used)
|
|
std::string types() const;
|
|
|
|
/// create a short "signature" string
|
|
std::string signature() const;
|
|
|
|
/// create a comma-separated string listing all argument names, used in m-files
|
|
std::string names() const;
|
|
|
|
/// Check if all arguments scalar
|
|
bool allScalar() const;
|
|
|
|
ArgumentList expandTemplate(const TemplateSubstitution& ts) const;
|
|
|
|
bool isSameSignature(const ArgumentList& other) const {
|
|
for(size_t i = 0; i<size(); ++i)
|
|
if (!at(i).isSameSignature(other[i])) return false;
|
|
return true;
|
|
}
|
|
|
|
// MATLAB code generation:
|
|
|
|
/**
|
|
* emit code to unwrap arguments
|
|
* @param file output stream
|
|
* @param start initial index for input array, set to 1 for method
|
|
*/
|
|
void matlab_unwrap(FileWriter& file, int start = 0) const; // MATLAB to C++
|
|
|
|
/**
|
|
* emit MATLAB prototype
|
|
* @param file output stream
|
|
* @param name of method or function
|
|
*/
|
|
void emit_prototype(FileWriter& file, const std::string& name) const;
|
|
|
|
/**
|
|
* emit arguments for cython pxd
|
|
* @param file output stream
|
|
*/
|
|
void emit_cython_pxd(FileWriter& file, const std::string& className,
|
|
const std::vector<std::string>& templateArgs) const;
|
|
void emit_cython_pyx(FileWriter& file) const;
|
|
std::string pyx_asParams() const;
|
|
std::string pyx_paramsList() const;
|
|
std::string pyx_castParamsToPythonType(const std::string& indent) const;
|
|
std::string pyx_convertEigenTypeAndStorageOrder(const std::string& indent) const;
|
|
|
|
/**
|
|
* emit checking arguments to MATLAB proxy
|
|
* @param proxyFile output stream
|
|
*/
|
|
void proxy_check(FileWriter& proxyFile) const;
|
|
|
|
/// Output stream operator
|
|
friend std::ostream& operator<<(std::ostream& os,
|
|
const ArgumentList& argList) {
|
|
os << "(";
|
|
if (argList.size() > 0)
|
|
os << argList.front();
|
|
if (argList.size() > 1)
|
|
for (size_t i = 1; i < argList.size(); i++)
|
|
os << ", " << argList[i];
|
|
os << ")";
|
|
return os;
|
|
}
|
|
|
|
};
|
|
|
|
/* ************************************************************************* */
|
|
// http://boost-spirit.com/distrib/spirit_1_8_2/libs/spirit/doc/grammar.html
|
|
struct ArgumentGrammar: public classic::grammar<ArgumentGrammar> {
|
|
|
|
wrap::Argument& result_; ///< successful parse will be placed in here
|
|
TypeGrammar argument_type_g; ///< Type parser for Argument::type
|
|
|
|
/// Construct type grammar and specify where result is placed
|
|
ArgumentGrammar(wrap::Argument& result) :
|
|
result_(result), argument_type_g(result.type) {
|
|
}
|
|
|
|
/// Definition of type grammar
|
|
template<typename ScannerT>
|
|
struct definition: BasicRules<ScannerT> {
|
|
|
|
typedef classic::rule<ScannerT> Rule;
|
|
|
|
Rule argument_p;
|
|
|
|
definition(ArgumentGrammar const& self) {
|
|
using namespace classic;
|
|
|
|
// NOTE: allows for pointers to all types
|
|
// Slightly more permissive than before on basis/eigen type qualification
|
|
// Also, currently parses Point2*&, can't make it work otherwise :-(
|
|
argument_p = !str_p("const")[assign_a(self.result_.is_const, T)] //
|
|
>> self.argument_type_g //
|
|
>> !ch_p('*')[assign_a(self.result_.is_ptr, T)]
|
|
>> !ch_p('&')[assign_a(self.result_.is_ref, T)]
|
|
>> BasicRules<ScannerT>::name_p[assign_a(self.result_.name)];
|
|
}
|
|
|
|
Rule const& start() const {
|
|
return argument_p;
|
|
}
|
|
|
|
};
|
|
};
|
|
// ArgumentGrammar
|
|
|
|
/* ************************************************************************* */
|
|
// http://boost-spirit.com/distrib/spirit_1_8_2/libs/spirit/doc/grammar.html
|
|
struct ArgumentListGrammar: public classic::grammar<ArgumentListGrammar> {
|
|
|
|
wrap::ArgumentList& result_; ///< successful parse will be placed in here
|
|
|
|
/// Construct type grammar and specify where result is placed
|
|
ArgumentListGrammar(wrap::ArgumentList& result) :
|
|
result_(result) {
|
|
}
|
|
|
|
/// Definition of type grammar
|
|
template<typename ScannerT>
|
|
struct definition {
|
|
|
|
const Argument arg0; ///< used to reset arg
|
|
Argument arg; ///< temporary argument for use during parsing
|
|
ArgumentGrammar argument_g; ///< single Argument parser
|
|
|
|
classic::rule<ScannerT> argument_p, argumentList_p;
|
|
|
|
definition(ArgumentListGrammar const& self) :
|
|
argument_g(arg) {
|
|
using namespace classic;
|
|
|
|
argument_p = argument_g //
|
|
[classic::push_back_a(self.result_, arg)] //
|
|
[assign_a(arg, arg0)];
|
|
|
|
argumentList_p = '(' >> !argument_p >> *(',' >> argument_p) >> ')';
|
|
}
|
|
|
|
classic::rule<ScannerT> const& start() const {
|
|
return argumentList_p;
|
|
}
|
|
|
|
};
|
|
};
|
|
// ArgumentListGrammar
|
|
|
|
/* ************************************************************************* */
|
|
|
|
}// \namespace wrap
|
|
|