/* gst-python * Copyright (C) 2009 Edward Hervey * 2005 Benjamin Otte * * 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., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif /* include this first, before NO_IMPORT_PYGOBJECT is defined */ #include #include #include void *_PyGstElement_Type; GST_DEBUG_CATEGORY_STATIC (pyplugindebug); #define GST_CAT_DEFAULT pyplugindebug #define GST_ORIGIN "http://gstreamer.freedesktop.org" static PyObject *element; static inline gboolean np_init_pygobject (void) { gboolean res = FALSE; PyObject *gobject = NULL; PyObject *main_module = NULL; PyObject *mdict = NULL; PyObject *pygtk = NULL; pygtk = PyImport_ImportModule ("pygtk"); if (pygtk == NULL) { PyErr_Print (); GST_WARNING ("could not import pygtk"); goto beach; } if (!(PyObject_CallMethod (pygtk, "require", "s", "2.0"))) { GST_WARNING ("could not run pygtk.require"); PyErr_Print (); goto beach; } gobject = PyImport_ImportModule ("gobject"); if (gobject == NULL) { PyErr_Print (); GST_WARNING ("could not import gobject"); goto beach; } main_module = PyImport_AddModule ("__main__"); mdict = PyModule_GetDict (gobject); PyObject *cobject = PyMapping_GetItemString (mdict, "_PyGObject_API"); if (cobject == NULL) { GST_WARNING ("could not find _PyGObject_API"); goto beach; } _PyGObject_API = (struct _PyGObject_Functions *) PyCObject_AsVoidPtr (cobject); if (_PyGObject_API == NULL) { PyErr_Print (); GST_WARNING ("_PyGObject_API is not a valid CObject"); goto beach; } if (!(PyObject_CallMethod (gobject, "threads_init", NULL, NULL))) { PyErr_Print (); GST_WARNING ("could not initialize threads"); goto beach; } res = TRUE; beach: Py_XDECREF (pygtk); Py_XDECREF (gobject); return res; } static gboolean gst_python_plugin_load_file (GstPlugin * plugin, const char *name) { PyObject *main_module, *main_locals; PyObject *elementfactory; int pos = 0; GType gtype; PyObject *module; const gchar *facname; guint rank; PyObject *class; GST_DEBUG ("loading plugin %s", name); main_module = PyImport_AddModule ("__main__"); if (main_module == NULL) { GST_WARNING ("Could not get __main__, ignoring plugin %s", name); PyErr_Print (); PyErr_Clear (); return FALSE; } main_locals = PyModule_GetDict (main_module); module = PyImport_ImportModuleEx ((char *) name, main_locals, main_locals, NULL); if (!module) { GST_DEBUG ("Could not load module, ignoring plugin %s", name); PyErr_Print (); PyErr_Clear (); return FALSE; } /* Get __gstelementfactory__ from file */ elementfactory = PyObject_GetAttrString (module, "__gstelementfactory__"); if (!elementfactory) { GST_DEBUG ("python file doesn't contain __gstelementfactory__"); PyErr_Clear (); return FALSE; } /* parse tuple : name, rank, gst.ElementClass */ if (!PyArg_ParseTuple (elementfactory, "sIO", &facname, &rank, &class)) { GST_WARNING ("__gstelementfactory__ isn't correctly formatted"); PyErr_Print (); PyErr_Clear (); Py_DECREF (elementfactory); return FALSE; } if (!(PyObject_IsSubclass (class, (PyObject *) _PyGstElement_Type))) { GST_WARNING ("the class provided isn't a subclass of gst.Element"); PyErr_Print (); PyErr_Clear (); Py_DECREF (elementfactory); Py_DECREF (class); return FALSE; } GST_LOG ("Valid plugin"); Py_DECREF (elementfactory); return gst_element_register (plugin, facname, rank, pyg_type_from_object (class)); } static gboolean gst_python_load_directory (GstPlugin * plugin, gchar * path) { GST_LOG ("Checking for python plugins in %s", path); GDir *dir; const gchar *file; GError *error = NULL; gboolean ret = TRUE; dir = g_dir_open (path, 0, &error); if (!dir) { /*retval should probably be depending on error, but since we ignore it... */ GST_WARNING ("Couldn't open Python plugin dir: %s", error->message); g_error_free (error); return FALSE; } while ((file = g_dir_read_name (dir))) { /* FIXME : go down in subdirectories */ if (g_str_has_suffix (file, ".py")) { gsize len = strlen (file) - 3; gchar *name = g_strndup (file, len); ret &= gst_python_plugin_load_file (plugin, name); g_free (name); } } return TRUE; } static gboolean gst_python_plugin_load (GstPlugin * plugin) { PyObject *sys_path; const gchar *plugin_path; gboolean ret = TRUE; sys_path = PySys_GetObject ("path"); /* Mimic the order in which the registry is checked in core */ /* 1. check env_variable GST_PLUGIN_PATH */ plugin_path = g_getenv ("GST_PLUGIN_PATH"); if (plugin_path) { char **list; int i; GST_DEBUG ("GST_PLUGIN_PATH set to %s", plugin_path); list = g_strsplit (plugin_path, G_SEARCHPATH_SEPARATOR_S, 0); for (i = 0; list[i]; i++) { gchar *sysdir = g_build_filename (list[i], "python", NULL); PyList_Insert (sys_path, 0, PyString_FromString (sysdir)); gst_python_load_directory (plugin, sysdir); g_free (sysdir); } g_strfreev (list); } /* 2. Check for GST_PLUGIN_SYSTEM_PATH */ plugin_path = g_getenv ("GST_PLUGIN_SYSTEM_PATH"); if (plugin_path == NULL) { char *home_plugins; /* 2.a. Scan user and system-wide plugin directory */ GST_DEBUG ("GST_PLUGIN_SYSTEM_PATH not set"); /* plugins in the user's home directory take precedence over * system-installed ones */ home_plugins = g_build_filename (g_get_home_dir (), ".gstreamer-" GST_MAJORMINOR, "plugins", "python", NULL); PyList_Insert (sys_path, 0, PyString_FromString (home_plugins)); gst_python_load_directory (plugin, home_plugins); g_free (home_plugins); /* add the main (installed) library path */ PyList_Insert (sys_path, 0, PyString_FromString (PLUGINDIR "/python")); gst_python_load_directory (plugin, PLUGINDIR "/python"); } else { gchar **list; gint i; /* 2.b. Scan GST_PLUGIN_SYSTEM_PATH */ GST_DEBUG ("GST_PLUGIN_SYSTEM_PATH set to %s", plugin_path, plugin_path); list = g_strsplit (plugin_path, G_SEARCHPATH_SEPARATOR_S, 0); for (i = 0; list[i]; i++) { gchar *sysdir; sysdir = g_build_filename (list[i], "python", NULL); PyList_Insert (sys_path, 0, PyString_FromString (sysdir)); gst_python_load_directory (plugin, sysdir); g_free (sysdir); } g_strfreev (list); } return ret; } /** *pygst_require: * @version: the version required * * Checks if the pygst/gst python modules are available. * Requests the specified version. * * Returns: the gst-python module, or NULL if there was an error. */ static PyObject * pygst_require (gchar * version) { PyObject *pygst, *gst; PyObject *require; PyObject *modules; gboolean doupdate = TRUE; const gchar *regupd; modules = PySys_GetObject ("modules"); /* Try to see if 'gst' is already imported */ if (!(gst = PyMapping_GetItemString (modules, "gst"))) { PyErr_Clear (); /* if not, see if 'pygst' was already imported. If so, we assume that * 'pygst.require' has already been called. */ if (!(pygst = PyMapping_GetItemString (modules, "pygst"))) { PyErr_Clear (); if (!(pygst = PyImport_ImportModule ("pygst"))) { GST_ERROR ("the pygst module is not available!"); goto error; } if (!(PyObject_CallMethod (pygst, "require", "s", version))) { GST_ERROR ("the required version, %s, of gst-python is not available!", version); Py_DECREF (pygst); goto error; } } /* We don't want the registry to be loaded when we import gst */ if ((regupd = g_getenv ("GST_REGISTRY_UPDATE")) && (!strcmp (regupd, "no"))) doupdate = FALSE; g_setenv ("GST_REGISTRY_UPDATE", "no", TRUE); if (!(gst = PyImport_ImportModule ("gst"))) { GST_ERROR ("couldn't import the gst module"); Py_DECREF (pygst); if (doupdate) g_unsetenv ("GST_REGISTRY_UPDATE"); goto error; } } if (doupdate) g_unsetenv ("GST_REGISTRY_UPDATE"); #define IMPORT(x, y) \ _PyGst##x##_Type = (void *)PyObject_GetAttrString(gst, y); \ if (_PyGst##x##_Type == NULL) { \ PyErr_Print(); \ return NULL; \ } IMPORT (Element, "Element"); return gst; error: { PyErr_Print (); PyErr_Clear (); return NULL; } } static gboolean plugin_init (GstPlugin * plugin) { PyGILState_STATE state; PyObject *gst, *dict, *pyplugin; gboolean we_initialized = FALSE; GModule *libpython; gpointer has_python = NULL; GST_DEBUG_CATEGORY_INIT (pyplugindebug, "pyplugin", 0, "Python plugin loader"); gst_plugin_add_dependency_simple (plugin, "HOME/.gstreamer-0.10/plugins/python:GST_PLUGIN_SYSTEM_PATH/python:GST_PLUGIN_PATH/python", PLUGINDIR "/python:HOME/.gstreamer-0.10/plugins/python:" "GST_PLUGIN_SYSTEM_PATH/python:GST_PLUGIN_PATH/python", NULL, GST_PLUGIN_DEPENDENCY_FLAG_NONE); GST_LOG ("Checking to see if libpython is already loaded"); g_module_symbol (g_module_open (NULL, G_MODULE_BIND_LOCAL), "_Py_NoneStruct", &has_python); if (has_python) { GST_LOG ("libpython is already loaded"); } else { GST_LOG ("loading libpython"); libpython = g_module_open (PY_LIB_LOC "/libpython" PYTHON_VERSION "." PY_LIB_SUFFIX, 0); if (!libpython) { GST_WARNING ("Couldn't g_module_open libpython. Reason: %s", g_module_error ()); return FALSE; } } if (!Py_IsInitialized ()) { GST_LOG ("python wasn't initialized"); /* set the correct plugin for registering stuff */ Py_Initialize (); we_initialized = TRUE; } else { GST_LOG ("python was already initialized"); state = PyGILState_Ensure (); } GST_LOG ("initializing pygobject"); if (!np_init_pygobject ()) { GST_WARNING ("pygobject initialization failed"); return FALSE; } if (!(gst = pygst_require ("0.10"))) { return FALSE; } if (we_initialized) { pyplugin = pygobject_new (G_OBJECT (plugin)); if (!pyplugin || PyModule_AddObject (gst, "__plugin__", pyplugin) != 0) { g_warning ("Couldn't set plugin"); Py_DECREF (pyplugin); } } dict = PyModule_GetDict (gst); if (!dict) { GST_ERROR ("no dict?!"); return FALSE; } gst_python_plugin_load (plugin); if (we_initialized) { /* We need to release the GIL since we're going back to C land */ PyEval_SaveThread (); } else PyGILState_Release (state); return TRUE; } GST_PLUGIN_DEFINE (GST_VERSION_MAJOR, GST_VERSION_MINOR, "python", "loader for plugins written in python", plugin_init, VERSION, "LGPL", PACKAGE_NAME, GST_ORIGIN)