152 lines
5.6 KiB
C++
152 lines
5.6 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 StaticMethod.ccp
|
|
* @author Frank Dellaert
|
|
* @author Andrew Melim
|
|
* @author Richard Roberts
|
|
**/
|
|
|
|
#include "StaticMethod.h"
|
|
#include "utilities.h"
|
|
#include "Class.h"
|
|
|
|
#include <boost/lexical_cast.hpp>
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <iostream>
|
|
#include <fstream>
|
|
|
|
using namespace std;
|
|
using namespace wrap;
|
|
|
|
/* ************************************************************************* */
|
|
void StaticMethod::proxy_header(FileWriter& proxyFile) const {
|
|
string upperName = matlabName();
|
|
upperName[0] = toupper(upperName[0], locale());
|
|
proxyFile.oss << " function varargout = " << upperName << "(varargin)\n";
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
string StaticMethod::wrapper_call(FileWriter& wrapperFile, Str cppClassName,
|
|
Str matlabUniqueName, const ArgumentList& args) const {
|
|
// check arguments
|
|
// NOTE: for static functions, there is no object passed
|
|
wrapperFile.oss << " checkArguments(\"" << matlabUniqueName << "." << name_
|
|
<< "\",nargout,nargin," << args.size() << ");\n";
|
|
|
|
// unwrap arguments, see Argument.cpp
|
|
args.matlab_unwrap(wrapperFile, 0); // We start at 0 because there is no self object
|
|
|
|
// call method and wrap result
|
|
// example: out[0]=wrap<bool>(staticMethod(t));
|
|
string expanded = cppClassName + "::" + name_;
|
|
if (templateArgValue_)
|
|
expanded += ("<" + templateArgValue_->qualifiedName("::") + ">");
|
|
|
|
return expanded;
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
void StaticMethod::emit_cython_pxd(FileWriter& file, const Class& cls) const {
|
|
for(size_t i = 0; i < nrOverloads(); ++i) {
|
|
file.oss << " @staticmethod\n";
|
|
file.oss << " ";
|
|
returnVals_[i].emit_cython_pxd(file, cls.pxdClassName(), cls.templateArgs);
|
|
file.oss << name_ + ((i>0)?"_" + to_string(i):"") << " \"" << name_ << "\"" << "(";
|
|
argumentList(i).emit_cython_pxd(file, cls.pxdClassName(), cls.templateArgs);
|
|
file.oss << ") except +\n";
|
|
}
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
void StaticMethod::emit_cython_wrapper_pxd(FileWriter& file,
|
|
const Class& cls) const {
|
|
if (nrOverloads() > 1) {
|
|
for (size_t i = 0; i < nrOverloads(); ++i) {
|
|
string funcName = name_ + "_" + to_string(i);
|
|
file.oss << " @staticmethod\n";
|
|
file.oss << " cdef tuple " + funcName + "(tuple args, dict kwargs)\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
void StaticMethod::emit_cython_pyx_no_overload(FileWriter& file,
|
|
const Class& cls) const {
|
|
assert(nrOverloads() == 1);
|
|
file.oss << " @staticmethod\n";
|
|
file.oss << " def " << name_ << "(";
|
|
argumentList(0).emit_cython_pyx(file);
|
|
file.oss << "):\n";
|
|
|
|
/// Call cython corresponding function and return
|
|
file.oss << argumentList(0).pyx_convertEigenTypeAndStorageOrder(" ");
|
|
string call = pyx_functionCall(cls.pxd_class_in_pyx(), name_, 0);
|
|
file.oss << " ";
|
|
if (!returnVals_[0].isVoid()) {
|
|
file.oss << "return " << returnVals_[0].pyx_casting(call) << "\n";
|
|
} else
|
|
file.oss << call << "\n";
|
|
file.oss << "\n";
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
void StaticMethod::emit_cython_pyx(FileWriter& file, const Class& cls) const {
|
|
size_t N = nrOverloads();
|
|
if (N == 1) {
|
|
emit_cython_pyx_no_overload(file, cls);
|
|
return;
|
|
}
|
|
|
|
// Dealing with overloads..
|
|
file.oss << " @staticmethod # overloaded\n";
|
|
file.oss << " def " << name_ << "(*args, **kwargs):\n";
|
|
for (size_t i = 0; i < N; ++i) {
|
|
string funcName = name_ + "_" + to_string(i);
|
|
file.oss << " success, results = " << cls.pyxClassName() << "."
|
|
<< funcName << "(args, kwargs)\n";
|
|
file.oss << " if success:\n return results\n";
|
|
}
|
|
file.oss << " raise TypeError('Could not find the correct overload')\n\n";
|
|
|
|
// Create cdef methods for all overloaded methods
|
|
for(size_t i = 0; i < N; ++i) {
|
|
string funcName = name_ + "_" + to_string(i);
|
|
file.oss << " @staticmethod\n";
|
|
file.oss << " cdef tuple " + funcName + "(tuple args, dict kwargs):\n";
|
|
file.oss << " cdef list __params\n";
|
|
if (!returnVals_[i].isVoid()) {
|
|
file.oss << " cdef " << returnVals_[i].pyx_returnType() << " return_value\n";
|
|
}
|
|
file.oss << " try:\n";
|
|
ArgumentList args = argumentList(i);
|
|
file.oss << pyx_resolveOverloadParams(args, false, 3);
|
|
|
|
/// Call cython corresponding function and return
|
|
file.oss << args.pyx_convertEigenTypeAndStorageOrder(" ");
|
|
string pxdFuncName = name_ + ((i>0)?"_" + to_string(i):"");
|
|
string call = pyx_functionCall(cls.pxd_class_in_pyx(), pxdFuncName, i);
|
|
if (!returnVals_[i].isVoid()) {
|
|
file.oss << " return_value = " << call << "\n";
|
|
file.oss << " return True, " << returnVals_[i].pyx_casting("return_value") << "\n";
|
|
} else {
|
|
file.oss << " " << call << "\n";
|
|
file.oss << " return True, None\n";
|
|
}
|
|
file.oss << " except:\n";
|
|
file.oss << " return False, None\n\n";
|
|
}
|
|
}
|
|
|
|
/* ************************************************************************* */
|