00001 
00002 
00003 
00004 
00005 
00006 
00007 
00008 
00009 
00010 
00011 
00012 
00013 
00014 
00015 
00016 
00017 
00018 
00019 
00020 
00021 
00022 
00023 
00024 
00025 
00026 
00027 
00028 
00029 #ifndef ROSMATLAB_OBJECT_H
00030 #define ROSMATLAB_OBJECT_H
00031 
00032 #include <rosmatlab/exception.h>
00033 #include <rosmatlab/options.h>
00034 
00035 #include <boost/shared_ptr.hpp>
00036 #include <boost/function.hpp>
00037 #include <boost/type_traits.hpp>
00038 
00039 #include "mex.h"
00040 
00041 #include <stdint.h>
00042 
00043 namespace rosmatlab {
00044 
00045 namespace {
00046   struct null_deleter {
00047     void operator()(void const *) const {}
00048   };
00049 }
00050 
00051 template <class Type>
00052 class Object
00053 {
00054 public:
00055   typedef boost::shared_ptr<Type> Ptr;
00056 
00057   Object() { construct(); }
00058   Object(Type *instance) { *this = instance; construct(); }
00059   Object(const Ptr &instance) { *this = instance; construct(); }
00060   Object(const Object &other) { *this = other; construct(); }
00061   virtual ~Object() { if (handle_) mxDestroyArray(handle_); }
00062 
00063   const Ptr &instance() const { return instance_; }
00064   mxArray *handle() const { return handle_; }
00065 
00066   Type* get() { return instance_.get(); }
00067   const Type* get() const { return instance_.get(); }
00068   Type &operator*() { return *instance_; }
00069   const Type &operator*() const { return *instance_; }
00070 
00071   Object<Type> &operator=(const Object &other) {
00072     return *this = other.instance_;
00073   }
00074 
00075   Object<Type> &operator=(Type *instance) {
00076     return *this = Ptr(instance, null_deleter());
00077   }
00078 
00079   Object<Type> &operator=(const Ptr &instance) {
00080     instance_ = instance;
00081     return *this;
00082   }
00083 
00084   static Object<Type> *byHandle(const mxArray *handle) {
00085     const mxArray *ptr = 0;
00086     if (!handle) return 0;
00087     
00088     if (mxIsClass(handle, class_name_)) {
00089       
00090       ptr = mxGetProperty(handle, 0, "handle");
00091     } else if (mxIsStruct(handle)) {
00092       
00093       ptr = mxGetField(handle, 0, "handle");
00094     } else if (mxIsDouble(handle)) {
00095       
00096       ptr = handle;
00097     }
00098     if (!ptr || !mxIsDouble(ptr) || !(mxGetNumberOfElements(ptr) > 0) || !mxGetPr(ptr)) throw Exception("invalid handle");
00099 
00100     Object<Type> *object = reinterpret_cast<Object<Type> *>(static_cast<uint64_t>(*mxGetPr(ptr)));
00101     return object;
00102   }
00103 
00104   static const char *getClassName() { return class_name_; }
00105 
00106 private:
00107   boost::shared_ptr<Type> instance_;
00108   mxArray *handle_;
00109   static const char *class_name_;
00110 
00111   void construct() {
00112     handle_ = mxCreateDoubleScalar(reinterpret_cast<uint64_t>(this));
00113     mexMakeArrayPersistent(handle_);
00114     assert(byHandle(handle()) == this);
00115   }
00116 };
00117 
00118 template <class Type>
00119 Type *getObject(const mxArray *handle) {
00120   Object<Type> *object = Object<Type>::byHandle(handle);
00121   if (!object) return 0;
00122   return object->get();
00123 }
00124 
00125 namespace internal {
00126 
00127   template <typename T> static inline mxArray *mx_cast(T result) {
00128     return mxCreateDoubleScalar(result);
00129   }
00130 
00131   template<> mxArray *mx_cast(mxArray *result) {
00132     return result;
00133   }
00134 
00135   template<> mxArray *mx_cast(bool result) {
00136     return mxCreateLogicalScalar(result);
00137   }
00138 
00139   template<> mxArray *mx_cast(const std::string& result) {
00140     return mxCreateString(result.c_str());
00141   }
00142 
00143   template <class Type, class Result>
00144   static inline void callMex(const boost::function<Result(Type *, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])> &func, Type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
00145   {
00146     plhs[0] = mx_cast(func(obj, nlhs, plhs, nrhs, prhs));
00147   }
00148 
00149   template <class Type, class Result>
00150   static inline void callMex(const boost::function<Result(Type *, int nrhs, const mxArray *prhs[])> &func, Type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
00151   {
00152     plhs[0] = mx_cast(func(obj, nrhs, prhs));
00153   }
00154 
00155   template <class Type, class Result>
00156   static inline void callMex(const boost::function<Result(Type *)> &func, Type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
00157   {
00158     plhs[0] = mx_cast(func(obj));
00159   }
00160 
00161   template <class Type>
00162   static inline void callMex(const boost::function<void(Type *, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])> &func, Type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
00163   {
00164     func(obj, nlhs, plhs, nrhs, prhs);
00165   }
00166 
00167   template <class Type>
00168   static inline void callMex(const boost::function<void(Type *, int nrhs, const mxArray *prhs[])> &func, Type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
00169   {
00170     func(obj, nrhs, prhs);
00171   }
00172 
00173   template <class Type>
00174   static inline void callMex(const boost::function<void(Type *)> &func, Type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
00175   {
00176     func(obj);
00177   }
00178 
00179   template <class Signature>
00180   class MexFunctor {
00181   public:
00182     typedef Signature* function_pointer;
00183     typedef boost::function<Signature> function_type;
00184     typedef typename boost::remove_pointer<typename boost::function_traits<Signature>::arg1_type>::type object_type;
00185 
00186     MexFunctor(const function_type& func) : target_(func) {}
00187     MexFunctor(function_pointer func) : target_(func) {}
00188 
00189     void operator()(object_type *obj, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[]) {
00190       callMex(target_, obj, nlhs, plhs, nrhs, prhs);
00191     }
00192 
00193   private:
00194     boost::function<Signature> target_;
00195   };
00196 }
00197 
00198 template <class Type>
00199 class MexMethodMap {
00200 private:
00201 
00202 
00203 
00204 
00205 
00206 
00207 
00208 
00209 
00210 
00211 
00212 
00213 
00214 
00215   std::map<std::string,boost::function<void(Type *, int, mxArray *[], int, const mxArray *[])> > methods_;
00216 
00217   bool initialized_;
00218   bool throw_on_unknown_;
00219 
00220 public:
00221   MexMethodMap() : initialized_(false), throw_on_unknown_(false) {}
00222 
00223   bool initialize() {
00224     if (!initialized_) {
00225       initialized_ = true;
00226       return false;
00227     }
00228     return true;
00229   }
00230 
00231   operator void *() const {
00232     return initialized_;
00233   }
00234 
00235   MexMethodMap &throwOnUnknown(bool value = true) {
00236     throw_on_unknown_ = value;
00237     return *this;
00238   }
00239 
00240   template <typename Result>
00241   MexMethodMap &add(const std::string& name, Result (Type::*function)(int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])) {
00242     methods_[name] = internal::MexFunctor<Result (Type *, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])>(function);
00243     return *this;
00244   }
00245 
00246   template <typename Result>
00247   MexMethodMap &add(const std::string& name, Result (Type::*function)(int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[]) const) {
00248     methods_[name] = internal::MexFunctor<Result (const Type *, int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])>(function);
00249     return *this;
00250   }
00251 
00252   template <typename Result>
00253   MexMethodMap &add(const std::string& name, Result (Type::*function)(int nrhs, const mxArray *prhs[])) {
00254     methods_[name] = internal::MexFunctor<Result (Type *, int nrhs, const mxArray *prhs[])>(function);
00255     return *this;
00256   }
00257 
00258   template <typename Result>
00259   MexMethodMap &add(const std::string& name, Result (Type::*function)(int nrhs, const mxArray *prhs[]) const) {
00260     methods_[name] = internal::MexFunctor<Result (const Type *, int nrhs, const mxArray *prhs[])>(function);
00261     return *this;
00262   }
00263 
00264   template <typename Result>
00265   MexMethodMap &add(const std::string& name, Result (Type::*function)()) {
00266     methods_[name] = internal::MexFunctor<Result (Type *)>(function);
00267     return *this;
00268   }
00269 
00270   template <typename Result>
00271   MexMethodMap &add(const std::string& name, Result (Type::*function)() const) {
00272     methods_[name] = internal::MexFunctor<Result (const Type *)>(function);
00273     return *this;
00274   }
00275 
00276 
00277 
00278 
00279 
00280 
00281 
00282 
00283 
00284 
00285 
00286 
00287 
00288 
00289 
00290 
00291 
00292 
00293 
00294 
00295 
00296 
00297 
00298 
00299 
00300 
00301 
00302 
00303 
00304 
00305 
00306   bool has(const std::string& name) const {
00307     return methods_.count(name);
00308   }
00309 
00310   bool call(Type *object, const std::string& name, int &nlhs, mxArray **&plhs, int &nrhs, const mxArray **&prhs) const {
00311     if (methods_.count(name)) {
00312       methods_.at(name)(object, nlhs, plhs, nrhs, prhs);
00313       return true;
00314     }
00315 
00316     if (throw_on_unknown_)
00317       throw Exception(std::string() + "unknown method '" + name + "' for objects of class " + Object<Type>::getClassName());
00318 
00319     if (has("default")) {
00320       return call(object, "default", nlhs, plhs, nrhs, prhs);
00321     }
00322 
00323     return false;
00324   }
00325 };
00326 
00327 template <class Type>
00328 Type *mexClassHelper(int &nlhs, mxArray **&plhs, int &nrhs, const mxArray **&prhs, std::string& method, const MexMethodMap<Type>& methods = MexMethodMap<Type>()) {
00329   if (nrhs < 1) {
00330     throw ArgumentException(1);
00331   }
00332 
00333   Type *object = getObject<Type>(*prhs++); nrhs--;
00334   method.clear();
00335   if (nrhs) { method = Options::getString(*prhs++); nrhs--; }
00336 
00337   
00338   if (method == "create") {
00339     delete object;
00340     
00341     object = new Type(nrhs, prhs);
00342     plhs[0] = object->handle();
00343     method.clear();
00344     return object;
00345   }
00346 
00347   
00348   if (method == "delete") {
00349     
00350     delete object;
00351     return 0;
00352   }
00353 
00354   
00355   if (!object) {
00356     throw Exception("Instance not found");
00357   }
00358 
00359   
00360   if (methods.call(object, method, nlhs, plhs, nrhs, prhs)) {
00361     method.clear();
00362   }
00363 
00364   return object;
00365 }
00366 
00367 template <class Type>
00368 Type *mexClassHelper(int &nlhs, mxArray **&plhs, int &nrhs, const mxArray **&prhs, const MexMethodMap<Type>& methods = MexMethodMap<Type>()) {
00369   std::string method;
00370   return mexClassHelper(nlhs, plhs, nrhs, prhs, method, methods);
00371 }
00372 
00373 } 
00374 
00375 #endif // ROSMATLAB_OBJECT_H