// -*- mode: cpp; mode: fold -*- // Description /*{{{*/ // $Id: generic.h,v 1.4 2002/03/10 05:45:34 mdz Exp $ /* ###################################################################### generic - Some handy functions to make integration a tad simpler Python needs this little _HEAD tacked onto the front of the object.. This complicates the integration with C++. We use some templates to make that quite transparent to us. It would have been nice if Python internally used a page from the C++ ref counting book to hide its little header from the world, but it doesn't. The CppPyObject has the target object and the Python header, this is needed to ensure proper alignment. GetCpp returns the C++ object from a PyObject. CppPyObject_NEW creates the Python object and then uses placement new to init the C++ class.. This is good for simple situations and as an example on how to do it in other more specific cases. CppPyObject_Dealloc should be used in the Type as the destructor function. HandleErrors converts errors from the internal _error stack into Python exceptions and makes sure the _error stack is empty. ##################################################################### */ /*}}}*/ #ifndef GENERIC_H #define GENERIC_H #include #include #include #if PYTHON_API_VERSION < 1013 typedef int Py_ssize_t; #endif template struct CppPyObject : public PyObject { // We are only using CppPyObject and friends as dumb structs only, ie the // c'tor is never called. // However if T doesn't have a default c'tor C++ doesn't generate one for // CppPyObject (since it can't know how it should initialize Object). // // This causes problems then in CppOwnedPyObject, for which C++ can't create // a c'tor that calls the base class c'tor (which causes a compilation // error). // So basically having the c'tor here removes the need for T to have a // default c'tor, which is not always desireable. CppPyObject() { }; T Object; }; template struct CppOwnedPyObject : public CppPyObject { PyObject *Owner; }; template inline T &GetCpp(PyObject *Obj) { return ((CppPyObject *)Obj)->Object; } template inline PyObject *GetOwner(PyObject *Obj) { return ((CppOwnedPyObject *)Obj)->Owner; } // Generic 'new' functions template inline CppPyObject *CppPyObject_NEW(PyTypeObject *Type) { CppPyObject *New = PyObject_NEW(CppPyObject,Type); new (&New->Object) T; return New; } template inline CppPyObject *CppPyObject_NEW(PyTypeObject *Type,A const &Arg) { CppPyObject *New = PyObject_NEW(CppPyObject,Type); new (&New->Object) T(Arg); return New; } template inline CppOwnedPyObject *CppOwnedPyObject_NEW(PyObject *Owner, PyTypeObject *Type) { CppOwnedPyObject *New = PyObject_NEW(CppOwnedPyObject,Type); new (&New->Object) T; New->Owner = Owner; Py_INCREF(Owner); return New; } template inline CppOwnedPyObject *CppOwnedPyObject_NEW(PyObject *Owner, PyTypeObject *Type,A const &Arg) { CppOwnedPyObject *New = PyObject_NEW(CppOwnedPyObject,Type); new (&New->Object) T(Arg); New->Owner = Owner; if (Owner != 0) Py_INCREF(Owner); return New; } // Generic Dealloc type functions template void CppDealloc(PyObject *Obj) { GetCpp(Obj).~T(); PyObject_DEL(Obj); } template void CppOwnedDealloc(PyObject *iObj) { CppOwnedPyObject *Obj = (CppOwnedPyObject *)iObj; Obj->Object.~T(); if (Obj->Owner != 0) Py_DECREF(Obj->Owner); PyObject_DEL(Obj); } inline PyObject *CppPyString(std::string Str) { return PyString_FromStringAndSize(Str.c_str(),Str.length()); } inline PyObject *Safe_FromString(const char *Str) { if (Str == 0) return PyString_FromString(""); return PyString_FromString(Str); } // Convert _error into Python exceptions PyObject *HandleErrors(PyObject *Res = 0); // Convert a list of strings to a char ** const char **ListToCharChar(PyObject *List,bool NullTerm = false); PyObject *CharCharToList(const char **List,unsigned long Size = 0); #endif