summaryrefslogtreecommitdiffstats
path: root/gi/pygi-struct.c
diff options
context:
space:
mode:
Diffstat (limited to 'gi/pygi-struct.c')
-rw-r--r--gi/pygi-struct.c175
1 files changed, 175 insertions, 0 deletions
diff --git a/gi/pygi-struct.c b/gi/pygi-struct.c
new file mode 100644
index 0000000..8b672c7
--- /dev/null
+++ b/gi/pygi-struct.c
@@ -0,0 +1,175 @@
+/* -*- Mode: C; c-basic-offset: 4 -*-
+ * vim: tabstop=4 shiftwidth=4 expandtab
+ *
+ * Copyright (C) 2009 Simon van der Linden <svdlinden@src.gnome.org>
+ *
+ * pygi-struct.c: wrapper to handle non-registered structures.
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
+ * USA
+ */
+
+#include "pygi-private.h"
+
+#include <pygobject.h>
+#include <girepository.h>
+
+static void
+_struct_dealloc (PyGIStruct *self)
+{
+ PyObject_GC_UnTrack((PyObject *)self);
+
+ PyObject_ClearWeakRefs((PyObject *)self);
+
+ if (self->free_on_dealloc) {
+ g_free(((PyGPointer *)self)->pointer);
+ }
+
+ ((PyGPointer *)self)->ob_type->tp_free((PyObject *)self);
+}
+
+static PyObject *
+_struct_new (PyTypeObject *type,
+ PyObject *args,
+ PyObject *kwargs)
+{
+ static char *kwlist[] = { NULL };
+
+ GIBaseInfo *info;
+ gboolean is_simple;
+ gsize size;
+ gpointer pointer;
+ PyObject *self = NULL;
+
+ if (!PyArg_ParseTupleAndKeywords(args, kwargs, "", kwlist)) {
+ return NULL;
+ }
+
+ info = _pygi_object_get_gi_info((PyObject *)type, &PyGIStructInfo_Type);
+ if (info == NULL) {
+ if (PyErr_ExceptionMatches(PyExc_AttributeError)) {
+ PyErr_Format(PyExc_TypeError, "missing introspection information");
+ }
+ return NULL;
+ }
+
+ is_simple = pygi_g_struct_info_is_simple((GIStructInfo *)info);
+ if (!is_simple) {
+ PyErr_Format(PyExc_TypeError, "cannot create '%s' instances", type->tp_name);
+ goto out;
+ }
+
+ size = g_struct_info_get_size((GIStructInfo *)info);
+ pointer = g_try_malloc(size);
+ if (pointer == NULL) {
+ PyErr_NoMemory();
+ goto out;
+ }
+
+ self = _pygi_struct_new(type, pointer, TRUE);
+ if (self == NULL) {
+ g_free(pointer);
+ }
+
+out:
+ g_base_info_unref(info);
+
+ return (PyObject *)self;
+}
+
+static int
+_struct_init (PyObject *self,
+ PyObject *args,
+ PyObject *kwargs)
+{
+ /* Don't call PyGPointer's init, which raises an exception. */
+ return 0;
+}
+
+
+PyTypeObject PyGIStruct_Type = {
+ PyObject_HEAD_INIT(NULL)
+ 0,
+ "gi.Struct", /* tp_name */
+ sizeof(PyGIStruct), /* tp_basicsize */
+ 0, /* tp_itemsize */
+ (destructor)_struct_dealloc, /* tp_dealloc */
+ (printfunc)NULL, /* tp_print */
+ (getattrfunc)NULL, /* tp_getattr */
+ (setattrfunc)NULL, /* tp_setattr */
+ (cmpfunc)NULL, /* tp_compare */
+ (reprfunc)NULL, /* tp_repr */
+ NULL, /* tp_as_number */
+ NULL, /* tp_as_sequence */
+ NULL, /* tp_as_mapping */
+ (hashfunc)NULL, /* tp_hash */
+ (ternaryfunc)NULL, /* tp_call */
+ (reprfunc)NULL, /* tp_str */
+ (getattrofunc)NULL, /* tp_getattro */
+ (setattrofunc)NULL, /* tp_setattro */
+ NULL, /* tp_as_buffer */
+ Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /* tp_flags */
+ NULL, /* tp_doc */
+ (traverseproc)NULL, /* tp_traverse */
+ (inquiry)NULL, /* tp_clear */
+ (richcmpfunc)NULL, /* tp_richcompare */
+ 0, /* tp_weaklistoffset */
+ (getiterfunc)NULL, /* tp_iter */
+ (iternextfunc)NULL, /* tp_iternext */
+ NULL, /* tp_methods */
+ NULL, /* tp_members */
+ NULL, /* tp_getset */
+ (PyTypeObject *)NULL, /* tp_base */
+};
+
+PyObject *
+_pygi_struct_new (PyTypeObject *type,
+ gpointer pointer,
+ gboolean free_on_dealloc)
+{
+ PyGIStruct *self;
+ GType g_type;
+
+ if (!PyType_IsSubtype(type, &PyGIStruct_Type)) {
+ PyErr_SetString(PyExc_TypeError, "must be a subtype of gi.Struct");
+ return NULL;
+ }
+
+ self = (PyGIStruct *)type->tp_alloc(type, 0);
+ if (self == NULL) {
+ return NULL;
+ }
+
+ g_type = pyg_type_from_object((PyObject *)type);
+
+ ((PyGPointer *)self)->gtype = g_type;
+ ((PyGPointer *)self)->pointer = pointer;
+ self->free_on_dealloc = free_on_dealloc;
+
+ return (PyObject *)self;
+}
+
+void
+_pygi_struct_register_types (PyObject *m)
+{
+ PyGIStruct_Type.ob_type = &PyType_Type;
+ PyGIStruct_Type.tp_base = &PyGPointer_Type;
+ PyGIStruct_Type.tp_new = (newfunc)_struct_new;
+ PyGIStruct_Type.tp_init = (initproc)_struct_init;
+ if (PyType_Ready(&PyGIStruct_Type))
+ return;
+ if (PyModule_AddObject(m, "Struct", (PyObject *)&PyGIStruct_Type))
+ return;
+}