/** * @file py_panda.h */ #ifndef PY_PANDA_H_ #define PY_PANDA_H_ #include #include #include #ifdef USE_DEBUG_PYTHON #define Py_DEBUG #endif #include "pnotify.h" #include "vector_uchar.h" #include "register_type.h" #if defined(HAVE_PYTHON) && !defined(CPPPARSER) // py_compat.h includes Python.h. #include "py_compat.h" #include using namespace std; // this is tempory .. untill this is glued better into the panda build system #if defined(_WIN32) && !defined(LINK_ALL_STATIC) #define EXPORT_THIS __declspec(dllexport) #define IMPORT_THIS extern __declspec(dllimport) #else #define EXPORT_THIS #define IMPORT_THIS extern #endif struct Dtool_PyTypedObject; // used to stamp dtool instance.. #define PY_PANDA_SIGNATURE 0xbeaf typedef void *(*UpcastFunction)(PyObject *,Dtool_PyTypedObject *); typedef void *(*DowncastFunction)(void *, Dtool_PyTypedObject *); typedef void *(*CoerceFunction)(PyObject *, void *); typedef void (*ModuleClassInitFunction)(PyObject *module); // INSTANCE CONTAINER FOR ALL panda py objects.... struct Dtool_PyInstDef { PyObject_HEAD // This is a pointer to the Dtool_PyTypedObject type. It's tempting not to // store this and to instead use PY_TYPE(self) and upcast that, but that // breaks when someone inherits from our class in Python. struct Dtool_PyTypedObject *_My_Type; // Pointer to the underlying C++ object. void *_ptr_to_object; // This is always set to PY_PANDA_SIGNATURE, so that we can quickly detect // whether an object is a Panda object. unsigned short _signature; // True if we own the pointer and should delete it or unref it. bool _memory_rules; // True if this is a "const" pointer. bool _is_const; }; // The Class Definition Structor For a Dtool python type. struct Dtool_PyTypedObject { // Standard Python Features.. PyTypeObject _PyType; // May be TypeHandle::none() to indicate a non-TypedObject class. TypeHandle _type; ModuleClassInitFunction _Dtool_ModuleClassInit; UpcastFunction _Dtool_UpcastInterface; // The Upcast Function By Slot DowncastFunction _Dtool_DowncastInterface; // The Downcast Function By Slot CoerceFunction _Dtool_ConstCoerce; CoerceFunction _Dtool_Coerce; }; // This is now simply a forward declaration. The actual definition is created // by the code generator. #define Define_Dtool_Class(MODULE_NAME, CLASS_NAME, PUBLIC_NAME) \ extern Dtool_PyTypedObject Dtool_##CLASS_NAME; // More Macro(s) to Implement class functions.. Usually used if C++ needs type // information #define Define_Dtool_new(CLASS_NAME,CNAME)\ static PyObject *Dtool_new_##CLASS_NAME(PyTypeObject *type, PyObject *args, PyObject *kwds) {\ (void) args; (void) kwds;\ PyObject *self = type->tp_alloc(type, 0);\ ((Dtool_PyInstDef *)self)->_signature = PY_PANDA_SIGNATURE;\ ((Dtool_PyInstDef *)self)->_My_Type = &Dtool_##CLASS_NAME;\ return self;\ } // The following used to be in the above macro, but it doesn't seem to be // necessary as tp_alloc memsets the object to 0. // ((Dtool_PyInstDef *)self)->_ptr_to_object = NULL; // ((Dtool_PyInstDef *)self)->_memory_rules = false; // ((Dtool_PyInstDef *)self)->_is_const = false; // Delete functions.. #ifdef NDEBUG #define Define_Dtool_FreeInstance_Private(CLASS_NAME,CNAME)\ static void Dtool_FreeInstance_##CLASS_NAME(PyObject *self) {\ Py_TYPE(self)->tp_free(self);\ } #else // NDEBUG #define Define_Dtool_FreeInstance_Private(CLASS_NAME,CNAME)\ static void Dtool_FreeInstance_##CLASS_NAME(PyObject *self) {\ if (DtoolInstance_VOID_PTR(self) != nullptr) {\ if (((Dtool_PyInstDef *)self)->_memory_rules) {\ std::cerr << "Detected leak for " << #CLASS_NAME \ << " which interrogate cannot delete.\n"; \ }\ }\ Py_TYPE(self)->tp_free(self);\ } #endif // NDEBUG #define Define_Dtool_FreeInstance(CLASS_NAME,CNAME)\ static void Dtool_FreeInstance_##CLASS_NAME(PyObject *self) {\ if (DtoolInstance_VOID_PTR(self) != nullptr) {\ if (((Dtool_PyInstDef *)self)->_memory_rules) {\ delete (CNAME *)DtoolInstance_VOID_PTR(self);\ }\ }\ Py_TYPE(self)->tp_free(self);\ } #define Define_Dtool_FreeInstanceRef(CLASS_NAME,CNAME)\ static void Dtool_FreeInstance_##CLASS_NAME(PyObject *self) {\ if (DtoolInstance_VOID_PTR(self) != nullptr) {\ if (((Dtool_PyInstDef *)self)->_memory_rules) {\ unref_delete((CNAME *)DtoolInstance_VOID_PTR(self));\ }\ }\ Py_TYPE(self)->tp_free(self);\ } #define Define_Dtool_FreeInstanceRef_Private(CLASS_NAME,CNAME)\ static void Dtool_FreeInstance_##CLASS_NAME(PyObject *self) {\ if (DtoolInstance_VOID_PTR(self) != nullptr) {\ if (((Dtool_PyInstDef *)self)->_memory_rules) {\ unref_delete((ReferenceCount *)(CNAME *)DtoolInstance_VOID_PTR(self));\ }\ }\ Py_TYPE(self)->tp_free(self);\ } #define Define_Dtool_Simple_FreeInstance(CLASS_NAME, CNAME)\ static void Dtool_FreeInstance_##CLASS_NAME(PyObject *self) {\ ((Dtool_InstDef_##CLASS_NAME *)self)->_value.~##CLASS_NAME();\ Py_TYPE(self)->tp_free(self);\ } // Use DtoolInstance_Check to check whether a PyObject* is a DtoolInstance. #define DtoolInstance_Check(obj) \ (Py_TYPE(obj)->tp_basicsize >= (int)sizeof(Dtool_PyInstDef) && \ ((Dtool_PyInstDef *)obj)->_signature == PY_PANDA_SIGNATURE) // These macros access the DtoolInstance without error checking. #define DtoolInstance_TYPE(obj) (((Dtool_PyInstDef *)obj)->_My_Type) #define DtoolInstance_IS_CONST(obj) (((Dtool_PyInstDef *)obj)->_is_const) #define DtoolInstance_VOID_PTR(obj) (((Dtool_PyInstDef *)obj)->_ptr_to_object) #define DtoolInstance_INIT_PTR(obj, ptr) { ((Dtool_PyInstDef *)obj)->_ptr_to_object = (void*)(ptr); } #define DtoolInstance_UPCAST(obj, type) (((Dtool_PyInstDef *)(obj))->_My_Type->_Dtool_UpcastInterface((obj), &(type))) // ** HACK ** allert.. Need to keep a runtime type dictionary ... that is // forward declared of typed object. We rely on the fact that typed objects // are uniquly defined by an integer. typedef std::map Dtool_TypeMap; EXPCL_PYPANDA Dtool_TypeMap *Dtool_GetGlobalTypeMap(); /** */ EXPCL_PYPANDA void DTOOL_Call_ExtractThisPointerForType(PyObject *self, Dtool_PyTypedObject *classdef, void **answer); EXPCL_PYPANDA void *DTOOL_Call_GetPointerThisClass(PyObject *self, Dtool_PyTypedObject *classdef, int param, const std::string &function_name, bool const_ok, bool report_errors); EXPCL_PYPANDA bool Dtool_Call_ExtractThisPointer(PyObject *self, Dtool_PyTypedObject &classdef, void **answer); EXPCL_PYPANDA bool Dtool_Call_ExtractThisPointer_NonConst(PyObject *self, Dtool_PyTypedObject &classdef, void **answer, const char *method_name); template INLINE bool DtoolInstance_GetPointer(PyObject *self, T *&into); template INLINE bool DtoolInstance_GetPointer(PyObject *self, T *&into, Dtool_PyTypedObject &classdef); INLINE Py_hash_t DtoolInstance_HashPointer(PyObject *self); INLINE int DtoolInstance_ComparePointers(PyObject *v1, PyObject *v2); INLINE PyObject *DtoolInstance_RichComparePointers(PyObject *v1, PyObject *v2, int op); // Functions related to error reporting. EXPCL_PYPANDA bool _Dtool_CheckErrorOccurred(); #ifdef NDEBUG #define Dtool_CheckErrorOccurred() (UNLIKELY(_PyErr_OCCURRED() != nullptr)) #else #define Dtool_CheckErrorOccurred() (UNLIKELY(_Dtool_CheckErrorOccurred())) #endif EXPCL_PYPANDA PyObject *Dtool_Raise_AssertionError(); EXPCL_PYPANDA PyObject *Dtool_Raise_TypeError(const char *message); EXPCL_PYPANDA PyObject *Dtool_Raise_ArgTypeError(PyObject *obj, int param, const char *function_name, const char *type_name); EXPCL_PYPANDA PyObject *Dtool_Raise_AttributeError(PyObject *obj, const char *attribute); EXPCL_PYPANDA PyObject *_Dtool_Raise_BadArgumentsError(); #ifdef NDEBUG // Define it to a function that just prints a generic message. #define Dtool_Raise_BadArgumentsError(x) _Dtool_Raise_BadArgumentsError() #else // Expand this to a TypeError listing all of the overloads. #define Dtool_Raise_BadArgumentsError(x) Dtool_Raise_TypeError("Arguments must match:\n" x) #endif // These functions are similar to Dtool_WrapValue, except that they also // contain code for checking assertions and exceptions when compiling with // NDEBUG mode on. EXPCL_PYPANDA PyObject *_Dtool_Return_None(); EXPCL_PYPANDA PyObject *Dtool_Return_Bool(bool value); EXPCL_PYPANDA PyObject *_Dtool_Return(PyObject *value); #ifdef NDEBUG #define Dtool_Return_None() (LIKELY(_PyErr_OCCURRED() == nullptr) ? (Py_INCREF(Py_None), Py_None) : nullptr) #define Dtool_Return(value) (LIKELY(_PyErr_OCCURRED() == nullptr) ? value : nullptr) #else #define Dtool_Return_None() _Dtool_Return_None() #define Dtool_Return(value) _Dtool_Return(value) #endif ALWAYS_INLINE void Dtool_Assign_PyObject(PyObject *&ptr, PyObject *value); /** * Wrapper around Python 3.4's enum library, which does not have a C API. */ EXPCL_PYPANDA PyTypeObject *Dtool_EnumType_Create(const char *name, PyObject *names, const char *module = nullptr); INLINE long Dtool_EnumValue_AsLong(PyObject *value); /** */ EXPCL_PYPANDA PyObject *DTool_CreatePyInstanceTyped(void *local_this_in, Dtool_PyTypedObject &known_class_type, bool memory_rules, bool is_const, int RunTimeType); // DTool_CreatePyInstance .. wrapper function to finalize the existance of a // general dtool py instance.. EXPCL_PYPANDA PyObject *DTool_CreatePyInstance(void *local_this, Dtool_PyTypedObject &in_classdef, bool memory_rules, bool is_const); // These template methods allow use when the Dtool_PyTypedObject is not known. // They require a get_class_type() to be defined for the class. template INLINE PyObject *DTool_CreatePyInstance(const T *obj, bool memory_rules); template INLINE PyObject *DTool_CreatePyInstance(T *obj, bool memory_rules); template INLINE PyObject *DTool_CreatePyInstanceTyped(const T *obj, bool memory_rules); template INLINE PyObject *DTool_CreatePyInstanceTyped(T *obj, bool memory_rules); // Macro(s) class definition .. Used to allocate storage and init some values // for a Dtool Py Type object. // struct Dtool_PyTypedObject Dtool_##CLASS_NAME; #define Define_Module_Class_Internal(MODULE_NAME,CLASS_NAME,CNAME)\ extern struct Dtool_PyTypedObject Dtool_##CLASS_NAME;\ static int Dtool_Init_##CLASS_NAME(PyObject *self, PyObject *args, PyObject *kwds);\ static PyObject *Dtool_new_##CLASS_NAME(PyTypeObject *type, PyObject *args, PyObject *kwds); #define Define_Module_Class(MODULE_NAME,CLASS_NAME,CNAME,PUBLIC_NAME)\ Define_Module_Class_Internal(MODULE_NAME,CLASS_NAME,CNAME)\ Define_Dtool_new(CLASS_NAME,CNAME)\ Define_Dtool_FreeInstance(CLASS_NAME,CNAME)\ Define_Dtool_Class(MODULE_NAME,CLASS_NAME,PUBLIC_NAME) #define Define_Module_Class_Private(MODULE_NAME,CLASS_NAME,CNAME,PUBLIC_NAME)\ Define_Module_Class_Internal(MODULE_NAME,CLASS_NAME,CNAME)\ Define_Dtool_new(CLASS_NAME,CNAME)\ Define_Dtool_FreeInstance_Private(CLASS_NAME,CNAME)\ Define_Dtool_Class(MODULE_NAME,CLASS_NAME,PUBLIC_NAME) #define Define_Module_ClassRef_Private(MODULE_NAME,CLASS_NAME,CNAME,PUBLIC_NAME)\ Define_Module_Class_Internal(MODULE_NAME,CLASS_NAME,CNAME)\ Define_Dtool_new(CLASS_NAME,CNAME)\ Define_Dtool_FreeInstanceRef_Private(CLASS_NAME,CNAME)\ Define_Dtool_Class(MODULE_NAME,CLASS_NAME,PUBLIC_NAME) #define Define_Module_ClassRef(MODULE_NAME,CLASS_NAME,CNAME,PUBLIC_NAME)\ Define_Module_Class_Internal(MODULE_NAME,CLASS_NAME,CNAME)\ Define_Dtool_new(CLASS_NAME,CNAME)\ Define_Dtool_FreeInstanceRef(CLASS_NAME,CNAME)\ Define_Dtool_Class(MODULE_NAME,CLASS_NAME,PUBLIC_NAME) // The finalizer for simple instances. INLINE int DTool_PyInit_Finalize(PyObject *self, void *This, Dtool_PyTypedObject *type, bool memory_rules, bool is_const); // A heler function to glu methed definition together .. that can not be done // at code generation time becouse of multiple generation passes in // interigate.. typedef std::map MethodDefmap; // We need a way to runtime merge compile units into a python "Module" .. this // is done with the fallowing structors and code.. along with the support of // interigate_module struct Dtool_TypeDef { const char *const name; Dtool_PyTypedObject *type; }; struct LibraryDef { PyMethodDef *const _methods; const Dtool_TypeDef *const _types; Dtool_TypeDef *const _external_types; }; #if PY_MAJOR_VERSION >= 3 EXPCL_PYPANDA PyObject *Dtool_PyModuleInitHelper(const LibraryDef *defs[], PyModuleDef *module_def); #else EXPCL_PYPANDA PyObject *Dtool_PyModuleInitHelper(const LibraryDef *defs[], const char *modulename); #endif // HACK.... Be carefull Dtool_BorrowThisReference This function can be used to // grab the "THIS" pointer from an object and use it Required to support fom // historical inharatence in the for of "is this instance of".. EXPCL_PYPANDA PyObject *Dtool_BorrowThisReference(PyObject *self, PyObject *args); #define DTOOL_PyObject_HashPointer DtoolInstance_HashPointer #define DTOOL_PyObject_ComparePointers DtoolInstance_ComparePointers EXPCL_PYPANDA PyObject * copy_from_make_copy(PyObject *self, PyObject *noargs); EXPCL_PYPANDA PyObject * copy_from_copy_constructor(PyObject *self, PyObject *noargs); EXPCL_PYPANDA PyObject * map_deepcopy_to_copy(PyObject *self, PyObject *args); /** * These functions check whether the arguments passed to a function conform to * certain expectations. */ ALWAYS_INLINE bool Dtool_CheckNoArgs(PyObject *args); ALWAYS_INLINE bool Dtool_CheckNoArgs(PyObject *args, PyObject *kwds); EXPCL_PYPANDA bool Dtool_ExtractArg(PyObject **result, PyObject *args, PyObject *kwds, const char *keyword); EXPCL_PYPANDA bool Dtool_ExtractArg(PyObject **result, PyObject *args, PyObject *kwds); EXPCL_PYPANDA bool Dtool_ExtractOptionalArg(PyObject **result, PyObject *args, PyObject *kwds, const char *keyword); EXPCL_PYPANDA bool Dtool_ExtractOptionalArg(PyObject **result, PyObject *args, PyObject *kwds); /** * These functions convert a C++ value into the corresponding Python object. * This used to be generated by the code generator, but it seems more reliable * and maintainable to define these as overloads and have the compiler sort * it out. */ ALWAYS_INLINE PyObject *Dtool_WrapValue(int value); ALWAYS_INLINE PyObject *Dtool_WrapValue(unsigned int value); ALWAYS_INLINE PyObject *Dtool_WrapValue(long value); ALWAYS_INLINE PyObject *Dtool_WrapValue(unsigned long value); ALWAYS_INLINE PyObject *Dtool_WrapValue(long long value); ALWAYS_INLINE PyObject *Dtool_WrapValue(unsigned long long value); ALWAYS_INLINE PyObject *Dtool_WrapValue(bool value); ALWAYS_INLINE PyObject *Dtool_WrapValue(double value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const char *value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const wchar_t *value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const std::string &value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const std::wstring &value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const std::string *value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const std::wstring *value); ALWAYS_INLINE PyObject *Dtool_WrapValue(char value); ALWAYS_INLINE PyObject *Dtool_WrapValue(wchar_t value); ALWAYS_INLINE PyObject *Dtool_WrapValue(std::nullptr_t); ALWAYS_INLINE PyObject *Dtool_WrapValue(PyObject *value); ALWAYS_INLINE PyObject *Dtool_WrapValue(const vector_uchar &value); #if PY_MAJOR_VERSION >= 0x02060000 ALWAYS_INLINE PyObject *Dtool_WrapValue(Py_buffer *value); #endif template ALWAYS_INLINE PyObject *Dtool_WrapValue(const std::pair &value); EXPCL_PYPANDA Dtool_PyTypedObject *Dtool_GetSuperBase(); #include "py_panda.I" #include "py_wrappers.h" #endif // HAVE_PYTHON && !CPPPARSER #endif // PY_PANDA_H_