/* -*- Mode: C; ; c-file-style: "k&r"; c-basic-offset: 4 -*- */ /* gst-python * Copyright (C) 2002 David I. Lehn * Copyright (C) 2012 Thibault Saunier * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library 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. * * Author: David I. Lehn */ #ifdef HAVE_CONFIG_H #include "config.h" #endif /* include this first, before NO_IMPORT_PYGOBJECT is defined */ #include #include #include #include #define PYGLIB_MODULE_START(symbol, modname) \ static struct PyModuleDef _##symbol##module = { \ PyModuleDef_HEAD_INIT, \ modname, \ NULL, \ -1, \ symbol##_functions, \ NULL, \ NULL, \ NULL, \ NULL \ }; \ PyMODINIT_FUNC PyInit_##symbol(void); \ PyMODINIT_FUNC PyInit_##symbol(void) \ { \ PyObject *module; \ module = PyModule_Create(&_##symbol##module); #define PYGLIB_MODULE_END return module; } GST_DEBUG_CATEGORY_STATIC (python_debug); GST_DEBUG_CATEGORY_STATIC (pygst_debug); #define GST_CAT_DEFAULT pygst_debug static PyObject * gi_gst_get_type (const gchar * type_name) { PyObject *module, *dict; module = PyImport_ImportModule ("gi.repository.Gst"); if (module == NULL) { PyErr_SetString (PyExc_KeyError, "Could not get module for gi.repository.Gst"); return NULL; } dict = PyModule_GetDict (module); Py_DECREF (module); /* For some reason we need this intermediary step */ module = PyMapping_GetItemString (dict, "_overrides_module"); if (module == NULL) { PyErr_SetString (PyExc_KeyError, "Could not get module for _overrides_module"); return NULL; } dict = PyModule_GetDict (module); return PyMapping_GetItemString (dict, type_name); } static PyObject * gi_gst_fraction_from_value (const GValue * value) { PyObject *fraction_type, *args, *fraction; gint numerator, denominator; numerator = gst_value_get_fraction_numerator (value); denominator = gst_value_get_fraction_denominator (value); fraction_type = gi_gst_get_type ("Fraction"); args = Py_BuildValue ("(ii)", numerator, denominator); fraction = PyObject_Call (fraction_type, args, NULL); Py_DECREF (args); return fraction; } static int gi_gst_fraction_to_value (GValue * value, PyObject * object) { PyObject *numerator, *denominator; numerator = PyObject_GetAttrString (object, "num"); if (numerator == NULL) goto fail; denominator = PyObject_GetAttrString (object, "denom"); if (denominator == NULL) goto fail; gst_value_set_fraction (value, PyLong_AsLong (numerator), PyLong_AsLong (denominator)); return 0; fail: return -1; } static PyObject * gi_gst_int_range_from_value (const GValue * value) { gint min, max, step; PyObject *int_range_type, *int_range, *range; min = gst_value_get_int_range_min (value); max = gst_value_get_int_range_max (value); step = gst_value_get_int_range_step (value); int_range_type = gi_gst_get_type ("IntRange"); range = PyObject_CallFunction ((PyObject *) & PyRange_Type, "iii", min, max, step); int_range = PyObject_CallFunction (int_range_type, "O", range); Py_DECREF (int_range_type); Py_DECREF (range); return int_range; } static int gi_gst_int_range_to_value (GValue * value, PyObject * object) { PyObject *range, *min, *max, *step; range = PyObject_GetAttrString (object, "range"); if (range == NULL) goto fail; min = PyObject_GetAttrString (range, "start"); if (min == NULL) goto fail; max = PyObject_GetAttrString (range, "stop"); if (max == NULL) goto fail; step = PyObject_GetAttrString (range, "step"); if (step == NULL) goto fail; gst_value_set_int_range_step (value, PyLong_AsLong (min), PyLong_AsLong (max), PyLong_AsLong (step)); return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.IntRange"); return -1; } static PyObject * gi_gst_int64_range_from_value (const GValue * value) { gint64 min, max, step; PyObject *int64_range_type, *int64_range, *range; min = gst_value_get_int64_range_min (value); max = gst_value_get_int64_range_max (value); step = gst_value_get_int64_range_step (value); range = PyObject_CallFunction ((PyObject *) & PyRange_Type, "LLL", min, max, step); int64_range_type = gi_gst_get_type ("Int64Range"); int64_range = PyObject_CallFunction (int64_range_type, "O", range); Py_DECREF (int64_range_type); Py_DECREF (range); return int64_range; } static int gi_gst_int64_range_to_value (GValue * value, PyObject * object) { PyObject *range, *min, *max, *step; range = PyObject_GetAttrString (object, "range"); if (range == NULL) goto fail; min = PyObject_GetAttrString (range, "start"); if (min == NULL) goto fail; max = PyObject_GetAttrString (range, "stop"); if (max == NULL) goto fail; step = PyObject_GetAttrString (range, "step"); if (step == NULL) goto fail; gst_value_set_int64_range_step (value, PyLong_AsLongLong (min), PyLong_AsLongLong (max), PyLong_AsLongLong (step)); return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.Int64Range"); return -1; } static PyObject * gi_gst_double_range_from_value (const GValue * value) { PyObject *double_range_type, *double_range; gdouble min, max; min = gst_value_get_double_range_min (value); max = gst_value_get_double_range_max (value); double_range_type = gi_gst_get_type ("DoubleRange"); double_range = PyObject_CallFunction (double_range_type, "dd", min, max); Py_DECREF (double_range_type); return double_range; } static int gi_gst_double_range_to_value (GValue * value, PyObject * object) { PyObject *min, *max; min = PyObject_GetAttrString (object, "start"); if (min == NULL) goto fail; max = PyObject_GetAttrString (object, "stop"); if (max == NULL) goto fail; gst_value_set_double_range (value, PyFloat_AsDouble (min), PyFloat_AsDouble (max)); return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.DoubleRange"); return -1; } static PyObject * gi_gst_fraction_range_from_value (const GValue * value) { PyObject *min, *max, *fraction_range_type, *fraction_range; const GValue *fraction; fraction = gst_value_get_fraction_range_min (value); min = gi_gst_fraction_from_value (fraction); fraction = gst_value_get_fraction_range_max (value); max = gi_gst_fraction_from_value (fraction); fraction_range_type = gi_gst_get_type ("FractionRange"); fraction_range = PyObject_CallFunction (fraction_range_type, "NN", min, max); Py_DECREF (fraction_range_type); return fraction_range; } static int gi_gst_fraction_range_to_value (GValue * value, PyObject * object) { PyObject *min, *max; GValue vmin = G_VALUE_INIT, vmax = G_VALUE_INIT; min = PyObject_GetAttrString (object, "start"); if (min == NULL) goto fail; max = PyObject_GetAttrString (object, "stop"); if (max == NULL) goto fail; g_value_init (&vmin, GST_TYPE_FRACTION); if (gi_gst_fraction_to_value (&vmin, min) < 0) goto fail; g_value_init (&vmax, GST_TYPE_FRACTION); if (gi_gst_fraction_to_value (&vmax, max) < 0) { g_value_unset (&vmin); goto fail; } gst_value_set_fraction_range (value, &vmin, &vmax); g_value_unset (&vmin); g_value_unset (&vmax); return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.FractionRange"); return -1; } static PyObject * gi_gst_array_from_value (const GValue * value) { PyObject *list, *array_type, *array; gint i; list = PyList_New (gst_value_array_get_size (value)); for (i = 0; i < gst_value_array_get_size (value); i++) { const GValue *v = gst_value_array_get_value (value, i); PyList_SET_ITEM (list, i, pyg_value_as_pyobject (v, TRUE)); } array_type = gi_gst_get_type ("ValueArray"); array = PyObject_CallFunction (array_type, "N", list); Py_DECREF (array_type); return array; } static int gi_gst_array_to_value (GValue * value, PyObject * object) { gint len, i; len = PySequence_Length (object); for (i = 0; i < len; i++) { GValue v = G_VALUE_INIT; GType type; PyObject *item; item = PySequence_GetItem (object, i); if (item == Py_None) type = G_TYPE_POINTER; else type = pyg_type_from_object ((PyObject *) Py_TYPE (item)); if (type == G_TYPE_NONE) { Py_DECREF (item); goto fail; } g_value_init (&v, type); if (pyg_value_from_pyobject (&v, item) < 0) { Py_DECREF (item); goto fail; } gst_value_array_append_and_take_value (value, &v); Py_DECREF (item); } return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.ValueArray"); return -1; } static PyObject * gi_gst_bitmask_from_value (const GValue * value) { PyObject *val, *bitmask_type; bitmask_type = gi_gst_get_type ("Bitmask"); val = PyObject_CallFunction (bitmask_type, "L", gst_value_get_bitmask (value)); Py_DECREF (bitmask_type); return val; } static int gi_gst_bitmask_to_value (GValue * value, PyObject * object) { PyObject *v = PyObject_GetAttrString (object, "v"); if (v == NULL) goto fail; gst_value_set_bitmask (value, PyLong_AsLong (v)); return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.Bitmask"); return -1; } static PyObject * gi_gst_list_from_value (const GValue * value) { PyObject *list, *value_list_type, *value_list; gint i; list = PyList_New (gst_value_list_get_size (value)); for (i = 0; i < gst_value_list_get_size (value); i++) { const GValue *v = gst_value_list_get_value (value, i); PyList_SET_ITEM (list, i, pyg_value_as_pyobject (v, TRUE)); } value_list_type = gi_gst_get_type ("ValueList"); value_list = PyObject_CallFunction (value_list_type, "N", list); Py_DECREF (value_list_type); return value_list; } static int gi_gst_list_to_value (GValue * value, PyObject * object) { gint len, i; len = PySequence_Length (object); for (i = 0; i < len; i++) { GValue v = G_VALUE_INIT; GType type; PyObject *item; item = PySequence_GetItem (object, i); if (item == Py_None) type = G_TYPE_POINTER; else type = pyg_type_from_object ((PyObject *) Py_TYPE (item)); if (type == G_TYPE_NONE) { Py_DECREF (item); goto fail; } g_value_init (&v, type); if (pyg_value_from_pyobject (&v, item) < 0) { Py_DECREF (item); goto fail; } gst_value_list_append_and_take_value (value, &v); Py_DECREF (item); } return 0; fail: PyErr_SetString (PyExc_KeyError, "Object is not compatible with Gst.ValueList"); return -1; } static void gi_gst_register_types (PyObject * d) { pyg_register_gtype_custom (GST_TYPE_FRACTION, gi_gst_fraction_from_value, gi_gst_fraction_to_value); pyg_register_gtype_custom (GST_TYPE_INT_RANGE, gi_gst_int_range_from_value, gi_gst_int_range_to_value); pyg_register_gtype_custom (GST_TYPE_INT64_RANGE, gi_gst_int64_range_from_value, gi_gst_int64_range_to_value); pyg_register_gtype_custom (GST_TYPE_DOUBLE_RANGE, gi_gst_double_range_from_value, gi_gst_double_range_to_value); pyg_register_gtype_custom (GST_TYPE_FRACTION_RANGE, gi_gst_fraction_range_from_value, gi_gst_fraction_range_to_value); pyg_register_gtype_custom (GST_TYPE_ARRAY, gi_gst_array_from_value, gi_gst_array_to_value); pyg_register_gtype_custom (GST_TYPE_LIST, gi_gst_list_from_value, gi_gst_list_to_value); #if 0 /* TODO */ pyg_register_gtype_custom (GST_TYPE_DATE_TIME, gi_gst_date_time_from_value, gi_gst_date_time_to_value); pyg_register_gtype_custom (GST_TYPE_FLAG_SET, gi_gst_flag_set_from_value, gi_gst_flag_set_to_value); #endif pyg_register_gtype_custom (GST_TYPE_BITMASK, gi_gst_bitmask_from_value, gi_gst_bitmask_to_value); } static int add_templates (gpointer gclass, PyObject * templates) { if (PyTuple_Check (templates)) { gint i, len; PyGObject *templ; len = PyTuple_Size (templates); if (len == 0) return 0; for (i = 0; i < len; i++) { templ = (PyGObject *) PyTuple_GetItem (templates, i); if (!pygobject_check (templ, &PyGObject_Type)) { PyObject *repr = PyObject_Repr ((PyObject *) templ); #if PY_VERSION_HEX < 0x03000000 PyErr_Format (PyExc_TypeError, "expected GObject but got %s", PyString_AsString (repr)); #else PyErr_Format (PyExc_TypeError, "expected GObject but got %s", _PyUnicode_AsString (repr)); #endif Py_DECREF (repr); return -1; } else if (!GST_IS_PAD_TEMPLATE (pygobject_get (templ))) { gchar *error = g_strdup_printf ("entries for __gsttemplates__ must be of type GstPadTemplate (%s)", G_OBJECT_TYPE_NAME (pygobject_get (templ))); PyErr_SetString (PyExc_TypeError, error); g_free (error); return -1; } } for (i = 0; i < len; i++) { templ = (PyGObject *) PyTuple_GetItem (templates, i); gst_element_class_add_pad_template (gclass, GST_PAD_TEMPLATE (templ->obj)); } return 0; } else if (!pygobject_check (templates, &PyGObject_Type) || GST_IS_PAD_TEMPLATE (pygobject_get (templates)) == FALSE) { PyErr_SetString (PyExc_TypeError, "entry for __gsttemplates__ must be of type GstPadTemplate"); return -1; } gst_element_class_add_pad_template (gclass, GST_PAD_TEMPLATE (pygobject_get (templates))); return 0; } static int _pygst_element_set_metadata (gpointer gclass, PyObject * metadata) { const gchar *longname, *classification, *description, *author; if (!PyTuple_Check (metadata)) { PyErr_SetString (PyExc_TypeError, "__gstmetadata__ must be a tuple"); return -1; } if (PyTuple_Size (metadata) != 4) { PyErr_SetString (PyExc_TypeError, "__gstmetadata__ must contain 4 elements"); return -1; } if (!PyArg_ParseTuple (metadata, "ssss", &longname, &classification, &description, &author)) { PyErr_SetString (PyExc_TypeError, "__gstmetadata__ must contain 4 strings"); return -1; } GST_DEBUG ("setting metadata on gclass %p from __gstmetadata__, longname %s", gclass, longname); gst_element_class_set_metadata (gclass, longname, classification, description, author); return 0; } static int _pygst_element_init (gpointer gclass, PyTypeObject * pyclass) { PyObject *templates, *metadata; GST_DEBUG ("_pygst_element_init for gclass %p", gclass); templates = PyDict_GetItemString (pyclass->tp_dict, "__gsttemplates__"); if (templates) { if (add_templates (gclass, templates) != 0) return -1; } else { PyErr_Clear (); } metadata = PyDict_GetItemString (pyclass->tp_dict, "__gstmetadata__"); if (metadata) { if (_pygst_element_set_metadata (gclass, metadata) != 0) return -1; PyDict_DelItemString (pyclass->tp_dict, "__gstmetadata__"); } else { PyErr_Clear (); } return 0; } #include static PyObject * pygst_debug_log (PyObject * pyobject, PyObject * string, GstDebugLevel level, gboolean isgstobject) { #ifndef GST_DISABLE_GST_DEBUG gchar *str; gchar *function; gchar *filename; int lineno; PyFrameObject *frame; GObject *object = NULL; if (!PyArg_ParseTuple (string, "s:gst.debug_log", &str)) { PyErr_SetString (PyExc_TypeError, "Need a string!"); return NULL; } frame = PyEval_GetFrame (); { PyObject *utf8; const gchar *utf8_str; utf8 = PyUnicode_AsUTF8String (frame->f_code->co_name); utf8_str = PyBytes_AS_STRING (utf8); function = g_strdup (utf8_str); Py_DECREF (utf8); utf8 = PyUnicode_AsUTF8String (frame->f_code->co_filename); utf8_str = PyBytes_AS_STRING (utf8); filename = g_strdup (utf8_str); Py_DECREF (utf8); } lineno = PyCode_Addr2Line (frame->f_code, frame->f_lasti); /* gst_debug_log : category, level, file, function, line, object, format, va_list */ if (isgstobject) object = G_OBJECT (pygobject_get (pyobject)); gst_debug_log (python_debug, level, filename, function, lineno, object, "%s", str); if (function) g_free (function); if (filename) g_free (filename); #endif Py_INCREF (Py_None); return Py_None; } static PyObject * _wrap_gst_trace (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_TRACE, FALSE); } static PyObject * _wrap_gst_log (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_LOG, FALSE); } static PyObject * _wrap_gst_debug (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_DEBUG, FALSE); } static PyObject * _wrap_gst_info (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_INFO, FALSE); } static PyObject * _wrap_gst_warning (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_WARNING, FALSE); } static PyObject * _wrap_gst_error (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_ERROR, FALSE); } static PyObject * _wrap_gst_fixme (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_FIXME, FALSE); } static PyObject * _wrap_gst_memdump (PyObject * whatever, PyObject * string) { return pygst_debug_log (whatever, string, GST_LEVEL_MEMDUMP, FALSE); } static PyObject * _remap (GstMapInfo * mapinfo, PyObject * py_mapinfo) { PyObject *mview, *py_memory; /* Create memoryview with compatible flags */ int flags; flags = (mapinfo->flags & GST_MAP_WRITE) ? PyBUF_WRITE : PyBUF_READ; mview = PyMemoryView_FromMemory ((char *) mapinfo->data, mapinfo->size, flags); /* Box GstMemory into a Gst.Memory */ py_memory = pyg_boxed_new (_gst_memory_type, mapinfo->memory, TRUE, TRUE); /* Fill out Gst.MapInfo with values corresponding to GstMapInfo */ if (PyObject_SetAttrString (py_mapinfo, "memory", py_memory) == -1) return NULL; if (PyObject_SetAttrString (py_mapinfo, "flags", Py_BuildValue ("i", mapinfo->flags)) == -1) return NULL; if (PyObject_SetAttrString (py_mapinfo, "data", mview) == -1) return NULL; if (PyObject_SetAttrString (py_mapinfo, "size", Py_BuildValue ("i", mapinfo->size)) == -1) return NULL; if (PyObject_SetAttrString (py_mapinfo, "maxsize", Py_BuildValue ("i", mapinfo->maxsize)) == -1) return NULL; if (PyObject_SetAttrString (py_mapinfo, "__cmapinfo", PyCapsule_New (mapinfo, "__cmapinfo", NULL)) == -1) return NULL; Py_INCREF (Py_True); return Py_True; } static PyObject * _gst_memory_override_map (PyObject * self, PyObject * args) { PyTypeObject *gst_memory_type; PyObject *py_memory, *py_mapinfo, *success; int flags; GstMemory *memory; GstMapInfo *mapinfo; _Bool ok; /* Look up Gst.memory, Gst.MapInfo, and Gst.MapFlags parameters */ gst_memory_type = pygobject_lookup_class (_gst_memory_type); if (!PyArg_ParseTuple (args, "O!Oi", gst_memory_type, &py_memory, &py_mapinfo, &flags)) return NULL; /* Since Python does only support r/o or r/w it has to be changed to either */ flags = (flags & GST_MAP_WRITE) ? GST_MAP_READWRITE : GST_MAP_READ; /* Extract GstMemory from Gst.Memory parameter */ memory = GST_MEMORY_CAST (pygobject_get (py_memory)); /* Map the memory, fill out GstMapInfo */ mapinfo = g_new0 (GstMapInfo, 1); ok = gst_memory_map (memory, mapinfo, flags); if (!ok) { g_free (mapinfo); goto err; } success = _remap (mapinfo, py_mapinfo); if (!success) { gst_memory_unmap (memory, mapinfo); g_free (mapinfo); } return success; err: Py_INCREF (Py_False); return Py_False; } static PyObject * _gst_memory_override_unmap (PyObject * self, PyObject * args) { PyTypeObject *gst_memory_type; PyObject *py_memory, *py_cmapinfo, *py_mapinfo, *mview, *ret; GstMemory *memory; GstMapInfo *mapinfo; /* Look up Gst.Buffer and Gst.Mapinfo parameters */ gst_memory_type = pygobject_lookup_class (_gst_memory_type); if (!PyArg_ParseTuple (args, "O!O", gst_memory_type, &py_memory, &py_mapinfo)) { PyErr_BadArgument (); return NULL; } /* Extract attributes from Gst.MapInfo */ if (!(mview = PyObject_GetAttrString (py_mapinfo, "data"))) goto err; if (!PyObject_HasAttrString (py_mapinfo, "__cmapinfo")) goto end; if (!(py_cmapinfo = PyObject_GetAttrString (py_mapinfo, "__cmapinfo"))) goto err; /* Extract GstBuffer from Gst.Buffer parameter */ memory = GST_MEMORY_CAST (pygobject_get (py_memory)); /* Reconstruct GstMapInfo from Gst.MapInfo contents */ mapinfo = PyCapsule_GetPointer (py_cmapinfo, "__cmapinfo"); /* Call the memoryview.release() Python method, there is no C API */ ret = PyObject_CallMethod (mview, "release", NULL); if (!ret) goto err; Py_DECREF (ret); Py_DECREF (py_cmapinfo); PyObject_SetAttrString (py_mapinfo, "__cmapinfo", NULL); /* Unmap the buffer, using reconstructed GstMapInfo */ gst_memory_unmap (memory, mapinfo); g_free (mapinfo); end: Py_DECREF (mview); Py_INCREF (Py_True); return Py_True; err: Py_INCREF (Py_False); return Py_False; } static PyObject * _gst_buffer_override_map_range (PyObject * self, PyObject * args) { PyTypeObject *gst_buffer_type; PyObject *py_buffer, *py_mapinfo, *success; int flags, range; unsigned int idx; GstBuffer *buffer; GstMapInfo *mapinfo; _Bool ok; /* Look up Gst.Buffer, Gst.MapInfo, idx, range, and Gst.MapFlags parameters */ gst_buffer_type = pygobject_lookup_class (_gst_buffer_type); if (!PyArg_ParseTuple (args, "O!OIii", gst_buffer_type, &py_buffer, &py_mapinfo, &idx, &range, &flags)) goto err; /* Since Python does only support r/o or r/w it has to be changed to either */ flags = (flags & GST_MAP_WRITE) ? GST_MAP_READWRITE : GST_MAP_READ; /* Extract GstBuffer from Gst.Buffer parameter */ buffer = GST_BUFFER (pygobject_get (py_buffer)); /* Map the buffer, fill out GstMapInfo */ mapinfo = g_new0 (GstMapInfo, 1); ok = gst_buffer_map_range (buffer, idx, range, mapinfo, flags); if (!ok) { g_free (mapinfo); goto err; } success = _remap (mapinfo, py_mapinfo); if (!success) { gst_buffer_unmap (buffer, mapinfo); g_free (mapinfo); } return success; err: Py_INCREF (Py_False); return Py_False; } static PyObject * _gst_buffer_override_map (PyObject * self, PyObject * args) { PyTypeObject *gst_buffer_type; PyObject *py_buffer, *py_mapinfo, *success; int flags; GstBuffer *buffer; GstMapInfo *mapinfo; _Bool ok; /* Look up Gst.Buffer, Gst.MapInfo, and Gst.MapFlags parameters */ gst_buffer_type = pygobject_lookup_class (_gst_buffer_type); if (!PyArg_ParseTuple (args, "O!Oi", gst_buffer_type, &py_buffer, &py_mapinfo, &flags)) { PyErr_BadArgument (); return NULL; } /* Since Python does only support r/o or r/w it has to be changed to either */ flags = (flags & GST_MAP_WRITE) ? GST_MAP_READWRITE : GST_MAP_READ; /* Extract GstBuffer from Gst.Buffer parameter */ buffer = GST_BUFFER (pygobject_get (py_buffer)); /* Map the buffer, fill out GstMapInfo */ mapinfo = g_new0 (GstMapInfo, 1); ok = gst_buffer_map (buffer, mapinfo, flags); if (!ok) { g_free (mapinfo); goto err; } success = _remap (mapinfo, py_mapinfo); if (!success) { gst_buffer_unmap (buffer, mapinfo); g_free (mapinfo); } return success; err: Py_INCREF (Py_False); return Py_False; } static PyObject * _gst_buffer_override_unmap (PyObject * self, PyObject * args) { PyTypeObject *gst_buf_type; PyObject *py_buffer, *py_cmapinfo, *py_mapinfo, *mview, *ret; GstBuffer *buffer; GstMapInfo *mapinfo; /* Look up Gst.Buffer and Gst.Mapinfo parameters */ gst_buf_type = pygobject_lookup_class (_gst_buffer_type); if (!PyArg_ParseTuple (args, "O!O", gst_buf_type, &py_buffer, &py_mapinfo)) { PyErr_BadArgument (); return NULL; } /* Extract attributes from Gst.MapInfo */ if (!(mview = PyObject_GetAttrString (py_mapinfo, "data"))) goto err; if (!PyObject_HasAttrString (py_mapinfo, "__cmapinfo")) goto end; if (!(py_cmapinfo = PyObject_GetAttrString (py_mapinfo, "__cmapinfo"))) goto err; /* Extract GstBuffer from Gst.Buffer parameter */ buffer = GST_BUFFER (pygobject_get (py_buffer)); /* Reconstruct GstMapInfo from Gst.MapInfo contents */ mapinfo = PyCapsule_GetPointer (py_cmapinfo, "__cmapinfo"); /* Call the memoryview.release() Python method, there is no C API */ ret = PyObject_CallMethod (mview, "release", NULL); if (!ret) { GST_ERROR ("Could not call `.release()` on the memoryview."); goto err; } Py_DECREF (ret); Py_DECREF (py_cmapinfo); PyObject_SetAttrString (py_mapinfo, "__cmapinfo", NULL); /* Unmap the buffer, using reconstructed GstMapInfo */ gst_buffer_unmap (buffer, mapinfo); g_free (mapinfo); end: Py_DECREF (mview); Py_INCREF (Py_True); return Py_True; err: Py_INCREF (Py_False); return Py_False; } static PyMethodDef _gi_gst_functions[] = { {"trace", (PyCFunction) _wrap_gst_trace, METH_VARARGS, NULL}, {"log", (PyCFunction) _wrap_gst_log, METH_VARARGS, NULL}, {"debug", (PyCFunction) _wrap_gst_debug, METH_VARARGS, NULL}, {"info", (PyCFunction) _wrap_gst_info, METH_VARARGS, NULL}, {"warning", (PyCFunction) _wrap_gst_warning, METH_VARARGS, NULL}, {"error", (PyCFunction) _wrap_gst_error, METH_VARARGS, NULL}, {"fixme", (PyCFunction) _wrap_gst_fixme, METH_VARARGS, NULL}, {"memdump", (PyCFunction) _wrap_gst_memdump, METH_VARARGS, NULL}, {"buffer_override_map_range", (PyCFunction) _gst_buffer_override_map_range, METH_VARARGS, NULL}, {"buffer_override_map", (PyCFunction) _gst_buffer_override_map, METH_VARARGS, NULL}, {"buffer_override_unmap", (PyCFunction) _gst_buffer_override_unmap, METH_VARARGS, NULL}, {"memory_override_map", (PyCFunction) _gst_memory_override_map, METH_VARARGS, NULL}, {"memory_override_unmap", (PyCFunction) _gst_memory_override_unmap, METH_VARARGS, NULL}, {NULL, NULL, 0, NULL} }; PYGLIB_MODULE_START (_gi_gst, "_gi_gst") { PyObject *d; /* gst should have been initialized already */ /* Initialize debugging category */ GST_DEBUG_CATEGORY_INIT (pygst_debug, "pygst", 0, "GStreamer python bindings"); GST_DEBUG_CATEGORY_INIT (python_debug, "python", GST_DEBUG_FG_GREEN, "python code using gst-python"); pygobject_init (3, 0, 0); d = PyModule_GetDict (module); gi_gst_register_types (d); pyg_register_class_init (GST_TYPE_ELEMENT, _pygst_element_init); } PYGLIB_MODULE_END;