/* -*- 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 (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; } 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 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}, {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;