aboutsummaryrefslogtreecommitdiffstats
path: root/libshiboken
diff options
context:
space:
mode:
authorHugo Parente Lima <hugo.pl@gmail.com>2010-11-08 17:04:58 -0200
committerHugo Parente Lima <hugo.pl@gmail.com>2012-03-08 16:08:56 -0300
commit6d69fbb13907e80ab87f86bc9407fc4ab0a102d7 (patch)
tree0d96b5d81b1f70d6a51e4a22419745ede68cded8 /libshiboken
parent95d5b63552f804e7fc5b80428cabb34f57443c0b (diff)
Create enum python types on the fly, avoiding generation of structs and duplication of many C functions.
Diffstat (limited to 'libshiboken')
-rw-r--r--libshiboken/conversions.h6
-rw-r--r--libshiboken/sbkenum.cpp149
-rw-r--r--libshiboken/sbkenum.h27
3 files changed, 123 insertions, 59 deletions
diff --git a/libshiboken/conversions.h b/libshiboken/conversions.h
index d4059b565..67e627926 100644
--- a/libshiboken/conversions.h
+++ b/libshiboken/conversions.h
@@ -491,11 +491,11 @@ struct EnumConverter
static inline PyObject* toPython(void* cppobj) { return toPython(*reinterpret_cast<CppEnum*>(cppobj)); }
static inline PyObject* toPython(CppEnum cppenum)
{
- return SbkEnumObject_New(SbkType<CppEnum>(), (long) cppenum);
+ return Shiboken::Enum::newItem(Shiboken::SbkType<CppEnum>(), (long) cppenum);
}
- static inline CppEnum toCpp(PyObject* pyobj)
+ static inline CppEnum toCpp(PyObject* pyObj)
{
- return (CppEnum) reinterpret_cast<SbkEnumObject*>(pyobj)->ob_ival;
+ return (CppEnum) Shiboken::Enum::getValue(pyObj);;
}
};
diff --git a/libshiboken/sbkenum.cpp b/libshiboken/sbkenum.cpp
index f71a74769..34a4b7729 100644
--- a/libshiboken/sbkenum.cpp
+++ b/libshiboken/sbkenum.cpp
@@ -21,12 +21,52 @@
*/
#include "sbkenum.h"
+#include <cstring>
+#include <list>
+#include "sbkdbg.h"
+#include "autodecref.h"
-namespace Shiboken
+extern "C"
{
-extern "C"
+struct SbkEnumObject
+{
+ PyObject_HEAD
+ long ob_ival;
+ PyObject* ob_name;
+};
+
+static PyObject* SbkEnumObject_repr(PyObject* self)
+{
+ return PyString_FromFormat("<enum-item %s.%s (%ld)>",
+ self->ob_type->tp_name,
+ PyString_AS_STRING(((SbkEnumObject*)self)->ob_name),
+ ((SbkEnumObject*)self)->ob_ival);
+}
+
+static PyObject* SbkEnumObject_name(PyObject* self, void*)
{
+ Py_INCREF(((SbkEnumObject*)self)->ob_name);
+ return ((SbkEnumObject*)self)->ob_name;
+}
+
+static PyObject* SbkEnum_tp_new(PyTypeObject* type, PyObject* args, PyObject* kwds)
+{
+ int itemValue = 0;
+ if (!PyArg_ParseTuple(args, "|i:__new__", &itemValue))
+ return 0;
+
+ SbkEnumObject* self = PyObject_New(SbkEnumObject, type);
+ if (!self)
+ return 0;
+ self->ob_ival = itemValue;
+ return reinterpret_cast<PyObject*>(self);
+}
+
+static PyGetSetDef SbkEnumGetSetList[] = {
+ {const_cast<char*>("name"), &SbkEnumObject_name},
+ {0} // Sentinel
+};
PyTypeObject SbkEnumType_Type = {
PyObject_HEAD_INIT(0)
@@ -77,65 +117,98 @@ PyTypeObject SbkEnumType_Type = {
/*tp_weaklist*/ 0
};
-}
+} // extern "C"
+
+namespace Shiboken {
-PyObject*
-SbkEnumObject_New(PyTypeObject *type, long item_value, PyObject* item_name)
+class DeclaredEnumTypes
{
- if (!item_name)
- item_name = PyString_FromString("");
- SbkEnumObject* enum_obj = PyObject_New(SbkEnumObject, type);
- enum_obj->ob_name = item_name;
- enum_obj->ob_ival = item_value;
- return (PyObject*) enum_obj;
-}
+public:
+ DeclaredEnumTypes();
+ ~DeclaredEnumTypes();
+ static DeclaredEnumTypes& instance();
+ void addEnumType(PyTypeObject* type);
+private:
+ DeclaredEnumTypes(const DeclaredEnumTypes&);
+ DeclaredEnumTypes& operator=(const DeclaredEnumTypes&);
+ std::list<PyTypeObject*> m_enumTypes;
+};
+
+namespace Enum {
-PyObject*
-SbkEnumObject_New(PyTypeObject *type, long item_value, const char* item_name)
+PyObject* newItem(PyTypeObject* enumType, long itemValue, const char* itemName)
{
- PyObject* py_item_name = 0;
- if (item_name)
- py_item_name = PyString_FromString(item_name);
+ if (!itemName)
+ itemName = "";
+ PyObject* pyItemName = PyString_FromString(itemName);
- PyObject* enum_obj = SbkEnumObject_New(type, item_value, py_item_name);
- if (!enum_obj) {
- Py_XDECREF(py_item_name);
+ SbkEnumObject* enumObj = PyObject_New(SbkEnumObject, enumType);
+ if (!enumObj) {
+ Py_XDECREF(pyItemName);
return 0;
}
- if (item_name) {
- PyObject* values = PyDict_GetItemString(type->tp_dict, const_cast<char*>("values"));
+ enumObj->ob_name = pyItemName;
+ enumObj->ob_ival = itemValue;
+ if (itemName) {
+ PyObject* values = PyDict_GetItemString(enumType->tp_dict, const_cast<char*>("values"));
if (!values) {
values = PyDict_New();
- PyDict_SetItemString(type->tp_dict, const_cast<char*>("values"), values);
+ PyDict_SetItemString(enumType->tp_dict, const_cast<char*>("values"), values);
Py_DECREF(values); // ^ values still alive, because setitemstring incref it
}
- PyDict_SetItemString(values, item_name, enum_obj);
+ PyDict_SetItemString(values, itemName, reinterpret_cast<PyObject*>(enumObj));
}
- return enum_obj;
+ return reinterpret_cast<PyObject*>(enumObj);
}
-extern "C"
+PyTypeObject* newType(const char* name)
{
+ PyTypeObject* type = new PyTypeObject;
+ ::memset(type, 0, sizeof(PyTypeObject));
+ type->ob_type = &SbkEnumType_Type;
+ type->tp_basicsize = sizeof(SbkEnumObject);
+ type->tp_repr = &SbkEnumObject_repr;
+ type->tp_str = &SbkEnumObject_repr;
+ type->tp_flags = Py_TPFLAGS_DEFAULT;
+ type->tp_base = &PyInt_Type;
+ type->tp_name = name;
+ type->tp_getset = SbkEnumGetSetList;
+ type->tp_new = SbkEnum_tp_new;
+
+ DeclaredEnumTypes::instance().addEnumType(type);
+ return type;
+}
-PyObject*
-SbkEnumObject_repr(PyObject* self)
+long int getValue(PyObject* enumItem)
{
- return PyString_FromFormat("<enum-item %s.%s (%ld)>",
- self->ob_type->tp_name,
- PyString_AS_STRING(((SbkEnumObject*)self)->ob_name),
- ((SbkEnumObject*)self)->ob_ival);
+ return reinterpret_cast<SbkEnumObject*>(enumItem)->ob_ival;
}
-PyObject*
-SbkEnumObject_name(PyObject* self)
+} // namespace Enum
+
+DeclaredEnumTypes& DeclaredEnumTypes::instance()
{
- Py_INCREF(((SbkEnumObject*)self)->ob_name);
- return ((SbkEnumObject*)self)->ob_name;
+ static DeclaredEnumTypes me;
+ return me;
}
-} // extern "C"
+DeclaredEnumTypes::DeclaredEnumTypes()
+{
+}
-} // namespace Shiboken
+DeclaredEnumTypes::~DeclaredEnumTypes()
+{
+ std::list<PyTypeObject*>::const_iterator it = m_enumTypes.begin();
+ for (; it != m_enumTypes.end(); ++it)
+ delete *it;
+ m_enumTypes.clear();
+}
+void DeclaredEnumTypes::addEnumType(PyTypeObject* type)
+{
+ m_enumTypes.push_back(type);
+}
+
+}
diff --git a/libshiboken/sbkenum.h b/libshiboken/sbkenum.h
index 447997e75..50ac9220a 100644
--- a/libshiboken/sbkenum.h
+++ b/libshiboken/sbkenum.h
@@ -26,36 +26,27 @@
#include <Python.h>
#include "shibokenmacros.h"
-namespace Shiboken
-{
-
extern "C"
{
-typedef struct {
- PyObject_HEAD
- long ob_ival;
- PyObject* ob_name;
-} SbkEnumObject;
-
extern LIBSHIBOKEN_API PyTypeObject SbkEnumType_Type;
-LIBSHIBOKEN_API PyObject* SbkEnumObject_repr(PyObject* self);
-LIBSHIBOKEN_API PyObject* SbkEnumObject_name(PyObject* self);
-
} // extern "C"
+namespace Shiboken
+{
+
inline bool isShibokenEnum(PyObject* pyObj)
{
return pyObj->ob_type->ob_type == &SbkEnumType_Type;
}
-LIBSHIBOKEN_API PyObject* SbkEnumObject_New(PyTypeObject *instanceType,
- long item_value,
- const char* item_name);
-LIBSHIBOKEN_API PyObject* SbkEnumObject_New(PyTypeObject *instanceType,
- long item_value,
- PyObject* item_name = 0);
+namespace Enum
+{
+ LIBSHIBOKEN_API PyObject* newItem(PyTypeObject* enumType, long itemValue, const char* itemName = 0);
+ LIBSHIBOKEN_API PyTypeObject* newType(const char* name);
+ LIBSHIBOKEN_API long getValue(PyObject* enumItem);
+}
} // namespace Shiboken