| /* GStreamer |
| * Copyright (C) 2006 Stefan Kost <ensonic@users.sf.net> |
| * |
| * gstpreset.c: helper interface for element presets |
| * |
| * 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 St, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| /** |
| * SECTION:gstpreset |
| * @title: GstPreset |
| * @short_description: helper interface for element presets |
| * |
| * This interface offers methods to query and manipulate parameter preset sets. |
| * A preset is a bunch of property settings, together with meta data and a name. |
| * The name of a preset serves as key for subsequent method calls to manipulate |
| * single presets. |
| * All instances of one type will share the list of presets. The list is created |
| * on demand, if presets are not used, the list is not created. |
| * |
| * The interface comes with a default implementation that serves most plugins. |
| * Wrapper plugins will override most methods to implement support for the |
| * native preset format of those wrapped plugins. |
| * One method that is useful to be overridden is gst_preset_get_property_names(). |
| * With that one can control which properties are saved and in which order. |
| * When implementing support for read-only presets, one should set the vmethods |
| * for gst_preset_save_preset() and gst_preset_delete_preset() to %NULL. |
| * Applications can use gst_preset_is_editable() to check for that. |
| * |
| * The default implementation supports presets located in a system directory, |
| * application specific directory and in the users home directory. When getting |
| * a list of presets individual presets are read and overlaid in 1) system, |
| * 2) application and 3) user order. Whenever an earlier entry is newer, the |
| * later entries will be updated. Since 1.8 you can also provide extra paths |
| * where to find presets through the GST_PRESET_PATH environment variable. |
| * Presets found in those paths will be concidered as "app presets". |
| */ |
| /* FIXME: |
| * - non racyness |
| * - we need to avoid two instances writing the preset file |
| * -> flock(fileno()), http://www.ecst.csuchico.edu/~beej/guide/ipc/flock.html |
| * -> open exclusive |
| * -> better save the new file to a tempfile and then rename? |
| * - we like to know when any other instance makes changes to the keyfile |
| * - then ui can be updated |
| * - and we make sure that we don't lose edits |
| * -> its the same problem actually, once for inside a process, once system- |
| * wide |
| * - can we use a lock inside a names shared memory segment? |
| * |
| * - should there be a 'preset-list' property to get the preset list |
| * (and to connect a notify:: to to listen for changes) |
| * we could use gnome_vfs_monitor_add() to monitor the user preset_file. |
| * |
| * - should there be a 'preset-name' property so that we can set a preset via |
| * gst-launch, or should we handle this with special syntax in gst-launch: |
| * gst-launch element preset:<preset-name> property=value ... |
| * - this would allow to have preset-bundles too (a preset on bins that |
| * specifies presets for children |
| */ |
| |
| #include "gst_private.h" |
| |
| #include "gstpreset.h" |
| #include "gstchildproxy.h" |
| #include "gstinfo.h" |
| #include "gstvalue.h" |
| |
| #ifdef HAVE_UNISTD_H |
| #include <unistd.h> |
| #endif |
| #include <glib/gstdio.h> |
| |
| #ifdef G_OS_WIN32 |
| #define WIN32_LEAN_AND_MEAN |
| #include <windows.h> |
| |
| extern HMODULE _priv_gst_dll_handle; |
| #endif |
| |
| #define GST_CAT_DEFAULT preset_debug |
| GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT); |
| |
| /* defines for keyfile usage, this group contains the element type name and |
| * version these presets belong to. */ |
| #define PRESET_HEADER "_presets_" |
| |
| /* keys of the preset header section */ |
| #define PRESET_HEADER_ELEMENT_NAME "element-name" |
| #define PRESET_HEADER_VERSION "version" |
| |
| static GQuark preset_user_path_quark = 0; |
| static GQuark preset_app_path_quark = 0; |
| static GQuark preset_system_path_quark = 0; |
| static GQuark preset_quark = 0; |
| |
| /* the application can set a custom path that is checked in addition to standard |
| * system and user dirs. This helps to develop new presets first local to the |
| * application. |
| */ |
| static gchar *preset_app_dir = NULL; |
| |
| /* default iface implementation */ |
| |
| static gboolean gst_preset_default_save_presets_file (GstPreset * preset); |
| |
| /* |
| * preset_get_paths: |
| * @preset: a #GObject that implements #GstPreset |
| * @preset_user_path: (out) (allow-none): location for path or %NULL |
| * @preset_app_path: (out) (allow-none): location for path or %NULL |
| * @preset_system_path: (out) (allow-none): location for path or %NULL |
| * |
| * Fetch the preset_path for user local, application specific and system wide |
| * settings. Don't free after use. |
| * |
| * Returns: %FALSE if no paths could be found. |
| */ |
| static gboolean |
| preset_get_paths (GstPreset * preset, const gchar ** preset_user_path, |
| const gchar ** preset_app_path, const gchar ** preset_system_path) |
| { |
| GType type = G_TYPE_FROM_INSTANCE (preset); |
| gchar *preset_path; |
| const gchar *element_name; |
| |
| /* we use the element name when we must construct the paths */ |
| element_name = G_OBJECT_TYPE_NAME (preset); |
| GST_INFO_OBJECT (preset, "element_name: '%s'", element_name); |
| |
| if (preset_user_path) { |
| /* preset user path requested, see if we have it cached in the qdata */ |
| if (!(preset_path = g_type_get_qdata (type, preset_user_path_quark))) { |
| gchar *preset_dir; |
| |
| /* user presets go in user's XDG data directory. */ |
| preset_dir = g_build_filename (g_get_user_data_dir (), |
| "gstreamer-" GST_API_VERSION, "presets", NULL); |
| GST_INFO_OBJECT (preset, "user_preset_dir: '%s'", preset_dir); |
| preset_path = |
| g_strdup_printf ("%s" G_DIR_SEPARATOR_S "%s.prs", preset_dir, |
| element_name); |
| GST_INFO_OBJECT (preset, "user_preset_path: '%s'", preset_path); |
| /* create dirs */ |
| g_mkdir_with_parents (preset_dir, 0755); |
| g_free (preset_dir); |
| |
| /* cache the preset path to the type */ |
| g_type_set_qdata (type, preset_user_path_quark, preset_path); |
| } |
| *preset_user_path = preset_path; |
| } |
| |
| if (preset_app_path) { |
| if (preset_app_dir) { |
| if (!(preset_path = g_type_get_qdata (type, preset_system_path_quark))) { |
| preset_path = g_strdup_printf ("%s" G_DIR_SEPARATOR_S "%s.prs", |
| preset_app_dir, element_name); |
| GST_INFO_OBJECT (preset, "application_preset_path: '%s'", preset_path); |
| |
| /* cache the preset path to the type */ |
| g_type_set_qdata (type, preset_app_path_quark, preset_path); |
| } |
| *preset_app_path = preset_path; |
| } else { |
| *preset_app_path = NULL; |
| } |
| } |
| |
| if (preset_system_path) { |
| /* preset system path requested, see if we have it cached in the qdata */ |
| if (!(preset_path = g_type_get_qdata (type, preset_system_path_quark))) { |
| gchar *preset_dir; |
| |
| /* system presets in '$GST_DATADIR/gstreamer-1.0/presets/GstAudioPanorama.prs' */ |
| #ifdef G_OS_WIN32 |
| gchar *basedir = |
| g_win32_get_package_installation_directory_of_module |
| (_priv_gst_dll_handle); |
| preset_dir = |
| g_build_filename (basedir, "share", "gstreamer-" GST_API_VERSION, |
| "presets", NULL); |
| g_free (basedir); |
| #else |
| preset_dir = g_build_filename (GST_DATADIR, "gstreamer-" GST_API_VERSION, |
| "presets", NULL); |
| #endif |
| GST_INFO_OBJECT (preset, "system_preset_dir: '%s'", preset_dir); |
| preset_path = g_strdup_printf ("%s" G_DIR_SEPARATOR_S "%s.prs", |
| preset_dir, element_name); |
| GST_INFO_OBJECT (preset, "system_preset_path: '%s'", preset_path); |
| /* create dirs */ |
| g_mkdir_with_parents (preset_dir, 0755); |
| g_free (preset_dir); |
| |
| /* cache the preset path to the type */ |
| g_type_set_qdata (type, preset_system_path_quark, preset_path); |
| } |
| *preset_system_path = preset_path; |
| } |
| return TRUE; |
| } |
| |
| static gboolean |
| preset_skip_property (GParamSpec * property) |
| { |
| if (((property->flags & G_PARAM_READWRITE) != G_PARAM_READWRITE) || |
| (property->flags & G_PARAM_CONSTRUCT_ONLY)) |
| return TRUE; |
| /* FIXME: skip GST_PARAM_NOT_PRESETABLE, see #522205 */ |
| return FALSE; |
| } |
| |
| static guint64 |
| preset_parse_version (const gchar * str_version) |
| { |
| guint major, minor, micro, nano; |
| gint num; |
| |
| major = minor = micro = nano = 0; |
| |
| /* parse version (e.g. 0.10.15.1) to guint64 */ |
| num = sscanf (str_version, "%u.%u.%u.%u", &major, &minor, µ, &nano); |
| /* make sure we have at least "major.minor" */ |
| if (num > 1) { |
| guint64 version; |
| |
| version = ((((major << 8 | minor) << 8) | micro) << 8) | nano; |
| GST_DEBUG ("version %s -> %" G_GUINT64_FORMAT, str_version, version); |
| return version; |
| } |
| return G_GUINT64_CONSTANT (0); |
| } |
| |
| static GKeyFile * |
| preset_open_and_parse_header (GstPreset * preset, const gchar * preset_path, |
| guint64 * preset_version) |
| { |
| GKeyFile *in; |
| GError *error = NULL; |
| gboolean res; |
| const gchar *element_name; |
| gchar *name; |
| |
| in = g_key_file_new (); |
| |
| res = g_key_file_load_from_file (in, preset_path, |
| G_KEY_FILE_KEEP_COMMENTS | G_KEY_FILE_KEEP_TRANSLATIONS, &error); |
| if (!res || error != NULL) |
| goto load_error; |
| |
| /* element type name and preset name must match or we are dealing with a wrong |
| * preset file */ |
| element_name = G_OBJECT_TYPE_NAME (preset); |
| name = |
| g_key_file_get_value (in, PRESET_HEADER, PRESET_HEADER_ELEMENT_NAME, |
| NULL); |
| |
| if (!name || strcmp (name, element_name)) |
| goto wrong_name; |
| |
| g_free (name); |
| |
| /* get the version now so that the caller can check it */ |
| if (preset_version) { |
| gchar *str = |
| g_key_file_get_value (in, PRESET_HEADER, PRESET_HEADER_VERSION, NULL); |
| *preset_version = preset_parse_version (str); |
| g_free (str); |
| } |
| |
| return in; |
| |
| /* ERRORS */ |
| load_error: |
| { |
| GST_INFO_OBJECT (preset, "Unable to read preset file %s: %s", |
| preset_path, error->message); |
| g_error_free (error); |
| g_key_file_free (in); |
| return NULL; |
| } |
| wrong_name: |
| { |
| GST_WARNING_OBJECT (preset, |
| "Wrong element name in preset file %s. Expected %s, got %s", |
| preset_path, element_name, GST_STR_NULL (name)); |
| g_free (name); |
| g_key_file_free (in); |
| return NULL; |
| } |
| } |
| |
| static void |
| preset_merge (GKeyFile * system, GKeyFile * user) |
| { |
| gchar *str; |
| gchar **groups, **keys; |
| gsize i, j, num_groups, num_keys; |
| |
| /* copy file comment if there is any */ |
| if ((str = g_key_file_get_comment (user, NULL, NULL, NULL))) { |
| g_key_file_set_comment (system, NULL, NULL, str, NULL); |
| g_free (str); |
| } |
| |
| /* get groups in user and copy into system */ |
| groups = g_key_file_get_groups (user, &num_groups); |
| for (i = 0; i < num_groups; i++) { |
| /* copy group comment if there is any */ |
| if ((str = g_key_file_get_comment (user, groups[i], NULL, NULL))) { |
| g_key_file_set_comment (system, groups[i], NULL, str, NULL); |
| g_free (str); |
| } |
| |
| /* ignore private groups */ |
| if (groups[i][0] == '_') |
| continue; |
| |
| /* if group already exists in system, remove and re-add keys from user */ |
| if (g_key_file_has_group (system, groups[i])) { |
| g_key_file_remove_group (system, groups[i], NULL); |
| } |
| |
| keys = g_key_file_get_keys (user, groups[i], &num_keys, NULL); |
| for (j = 0; j < num_keys; j++) { |
| /* copy key comment if there is any */ |
| if ((str = g_key_file_get_comment (user, groups[i], keys[j], NULL))) { |
| g_key_file_set_comment (system, groups[i], keys[j], str, NULL); |
| g_free (str); |
| } |
| str = g_key_file_get_value (user, groups[i], keys[j], NULL); |
| g_key_file_set_value (system, groups[i], keys[j], str); |
| g_free (str); |
| } |
| g_strfreev (keys); |
| } |
| g_strfreev (groups); |
| } |
| |
| typedef struct |
| { |
| GKeyFile *preset; |
| guint64 version; |
| } PresetAndVersion; |
| |
| static gint |
| compare_preset_and_version (gconstpointer a, gconstpointer b, |
| gpointer user_data) |
| { |
| const PresetAndVersion *pa = a, *pb = b; |
| |
| if (pa->version > pb->version) |
| return -1; |
| if (pa->version < pb->version) |
| return 1; |
| else |
| return 0; |
| } |
| |
| /* reads the user and system presets files and merges them together. This |
| * function caches the GKeyFile on the element type. If there is no existing |
| * preset file, a new in-memory GKeyFile will be created. */ |
| static GKeyFile * |
| preset_get_keyfile (GstPreset * preset) |
| { |
| GKeyFile *presets; |
| GType type = G_TYPE_FROM_INSTANCE (preset); |
| |
| /* first see if the have a cached version for the type */ |
| if (!(presets = g_type_get_qdata (type, preset_quark))) { |
| const gchar *preset_user_path, *preset_app_path, *preset_system_path; |
| guint64 version_system = G_GUINT64_CONSTANT (0); |
| guint64 version_app = G_GUINT64_CONSTANT (0); |
| guint64 version_user = G_GUINT64_CONSTANT (0); |
| guint64 version = G_GUINT64_CONSTANT (0); |
| gboolean merged = FALSE; |
| GKeyFile *in_user, *in_app = NULL, *in_system; |
| GQueue in_env = G_QUEUE_INIT; |
| gboolean have_env = FALSE; |
| const gchar *envvar; |
| |
| /* try to load the user, app and system presets, we do this to get the |
| * versions of all files. */ |
| preset_get_paths (preset, &preset_user_path, &preset_app_path, |
| &preset_system_path); |
| in_user = preset_open_and_parse_header (preset, preset_user_path, |
| &version_user); |
| |
| if (preset_app_path) { |
| in_app = preset_open_and_parse_header (preset, preset_app_path, |
| &version_app); |
| } |
| |
| envvar = g_getenv ("GST_PRESET_PATH"); |
| if (envvar) { |
| gint i; |
| gchar **preset_dirs = g_strsplit (envvar, G_SEARCHPATH_SEPARATOR_S, -1); |
| |
| for (i = 0; preset_dirs[i]; i++) { |
| gchar *preset_path = g_strdup_printf ("%s" G_DIR_SEPARATOR_S "%s.prs", |
| preset_dirs[i], G_OBJECT_TYPE_NAME (preset)); |
| GKeyFile *env_file; |
| guint64 env_version; |
| |
| env_file = preset_open_and_parse_header (preset, preset_path, |
| &env_version); |
| g_free (preset_path); |
| if (env_file) { |
| PresetAndVersion *pv = g_new (PresetAndVersion, 1); |
| pv->preset = env_file; |
| pv->version = env_version; |
| g_queue_push_tail (&in_env, pv); |
| have_env = TRUE; |
| } |
| } |
| g_strfreev (preset_dirs); |
| } |
| |
| in_system = preset_open_and_parse_header (preset, preset_system_path, |
| &version_system); |
| |
| /* compare version to check for merge */ |
| if (in_system) { |
| presets = in_system; |
| version = version_system; |
| } |
| |
| if (have_env) { |
| GList *l; |
| |
| /* merge the ones from the environment paths. If any of them has a |
| * higher version, take that as the "master" version. Lower versions are |
| * then just merged in. */ |
| g_queue_sort (&in_env, compare_preset_and_version, NULL); |
| /* highest version to lowest */ |
| for (l = in_env.head; l; l = l->next) { |
| PresetAndVersion *pv = l->data; |
| |
| if (version > pv->version) { |
| preset_merge (presets, pv->preset); |
| g_key_file_free (pv->preset); |
| } else { |
| if (presets) |
| g_key_file_free (presets); |
| presets = pv->preset; |
| version = pv->version; |
| } |
| g_free (pv); |
| } |
| g_queue_clear (&in_env); |
| } |
| |
| if (in_app) { |
| /* if system/env version is higher, merge */ |
| if (version > version_app) { |
| preset_merge (presets, in_app); |
| g_key_file_free (in_app); |
| } else { |
| if (presets) |
| g_key_file_free (presets); |
| presets = in_app; |
| version = version_app; |
| } |
| } |
| if (in_user) { |
| /* if system/env or app version is higher, merge */ |
| if (version > version_user) { |
| preset_merge (presets, in_user); |
| g_key_file_free (in_user); |
| merged = TRUE; |
| } else { |
| if (presets) |
| g_key_file_free (presets); |
| presets = in_user; |
| } |
| } |
| |
| if (!presets) { |
| /* we did not load a user, app or system presets file, create a new one */ |
| presets = g_key_file_new (); |
| g_key_file_set_string (presets, PRESET_HEADER, PRESET_HEADER_ELEMENT_NAME, |
| G_OBJECT_TYPE_NAME (preset)); |
| } |
| |
| /* attach the preset to the type */ |
| g_type_set_qdata (type, preset_quark, (gpointer) presets); |
| |
| if (merged) { |
| gst_preset_default_save_presets_file (preset); |
| } |
| } |
| return presets; |
| } |
| |
| static gint |
| compare_strings (gchar ** a, gchar ** b, gpointer user_data) |
| { |
| return g_strcmp0 (*a, *b); |
| } |
| |
| /* get a list of all supported preset names for an element */ |
| static gchar ** |
| gst_preset_default_get_preset_names (GstPreset * preset) |
| { |
| GKeyFile *presets; |
| gsize i, num_groups; |
| gchar **groups; |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| /* get the groups, which are also the preset names */ |
| if (!(groups = g_key_file_get_groups (presets, &num_groups))) |
| goto no_groups; |
| |
| /* remove all private group names starting with '_' from the array */ |
| for (i = 0; i < num_groups; i++) { |
| if (groups[i][0] == '_') { |
| /* free private group */ |
| g_free (groups[i]); |
| /* move last element of list down */ |
| num_groups--; |
| /* move last element into removed element */ |
| groups[i] = groups[num_groups]; |
| groups[num_groups] = NULL; |
| } |
| } |
| if (!num_groups) { |
| GST_INFO_OBJECT (preset, "Empty preset file"); |
| g_strfreev (groups); |
| return NULL; |
| } |
| |
| /* sort the array now */ |
| g_qsort_with_data (groups, num_groups, sizeof (gchar *), |
| (GCompareDataFunc) compare_strings, NULL); |
| |
| return groups; |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "Could not load presets"); |
| return NULL; |
| } |
| no_groups: |
| { |
| GST_WARNING_OBJECT (preset, "Could not find preset groups"); |
| return NULL; |
| } |
| } |
| |
| /* get a list of all property names that are used for presets */ |
| static gchar ** |
| gst_preset_default_get_property_names (GstPreset * preset) |
| { |
| GParamSpec **props; |
| guint i, j = 0, n_props; |
| GObjectClass *gclass; |
| gboolean is_child_proxy; |
| gchar **result = NULL; |
| |
| gclass = G_OBJECT_CLASS (GST_ELEMENT_GET_CLASS (preset)); |
| is_child_proxy = GST_IS_CHILD_PROXY (preset); |
| |
| /* get a list of object properties */ |
| props = g_object_class_list_properties (gclass, &n_props); |
| if (props) { |
| /* allocate array big enough to hold the worst case, including a terminating |
| * NULL pointer. */ |
| result = g_new (gchar *, n_props + 1); |
| |
| /* now filter out the properties that we can use for presets */ |
| GST_DEBUG_OBJECT (preset, " filtering properties: %u", n_props); |
| for (i = 0; i < n_props; i++) { |
| if (preset_skip_property (props[i])) |
| continue; |
| GST_DEBUG_OBJECT (preset, " using: %s", props[i]->name); |
| result[j++] = g_strdup (props[i]->name); |
| } |
| g_free (props); |
| } |
| |
| if (is_child_proxy) { |
| guint c, n_children; |
| GObject *child; |
| |
| n_children = gst_child_proxy_get_children_count ((GstChildProxy *) preset); |
| for (c = 0; c < n_children; c++) { |
| child = gst_child_proxy_get_child_by_index ((GstChildProxy *) preset, c); |
| gclass = G_OBJECT_CLASS (GST_ELEMENT_GET_CLASS (child)); |
| |
| props = g_object_class_list_properties (gclass, &n_props); |
| if (props) { |
| /* resize property name array */ |
| result = g_renew (gchar *, result, j + n_props + 1); |
| |
| /* now filter out the properties that we can use for presets */ |
| GST_DEBUG_OBJECT (preset, " filtering properties: %u", n_props); |
| for (i = 0; i < n_props; i++) { |
| if (preset_skip_property (props[i])) |
| continue; |
| GST_DEBUG_OBJECT (preset, " using: %s::%s", |
| GST_OBJECT_NAME (child), props[i]->name); |
| result[j++] = g_strdup_printf ("%s::%s", GST_OBJECT_NAME (child), |
| props[i]->name); |
| } |
| g_free (props); |
| } |
| |
| g_object_unref (child); |
| } |
| } |
| if (!result) { |
| GST_INFO_OBJECT (preset, "object has no properties"); |
| } else { |
| result[j] = NULL; |
| } |
| return result; |
| } |
| |
| /* load the presets of @name for the instance @preset. Returns %FALSE if something |
| * failed. */ |
| static gboolean |
| gst_preset_default_load_preset (GstPreset * preset, const gchar * name) |
| { |
| GKeyFile *presets; |
| gchar **props; |
| guint i; |
| GObjectClass *gclass; |
| gboolean is_child_proxy; |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| /* get the preset name */ |
| if (!g_key_file_has_group (presets, name)) |
| goto no_group; |
| |
| GST_DEBUG_OBJECT (preset, "loading preset : '%s'", name); |
| |
| /* get the properties that we can configure in this element */ |
| if (!(props = gst_preset_get_property_names (preset))) |
| goto no_properties; |
| |
| gclass = G_OBJECT_CLASS (GST_ELEMENT_GET_CLASS (preset)); |
| is_child_proxy = GST_IS_CHILD_PROXY (preset); |
| |
| /* for each of the property names, find the preset parameter and try to |
| * configure the property with its value */ |
| for (i = 0; props[i]; i++) { |
| gchar *str; |
| GValue gvalue = { 0, }; |
| GParamSpec *property = NULL; |
| |
| /* check if we have a settings for this element property */ |
| if (!(str = g_key_file_get_value (presets, name, props[i], NULL))) { |
| /* the element has a property but the parameter is not in the keyfile */ |
| GST_INFO_OBJECT (preset, "parameter '%s' not in preset", props[i]); |
| continue; |
| } |
| |
| GST_DEBUG_OBJECT (preset, "setting value '%s' for property '%s'", str, |
| props[i]); |
| |
| if (is_child_proxy) { |
| gst_child_proxy_lookup ((GstChildProxy *) preset, props[i], NULL, |
| &property); |
| } else { |
| property = g_object_class_find_property (gclass, props[i]); |
| } |
| if (!property) { |
| /* the parameter was in the keyfile, the element said it supported it but |
| * then the property was not found in the element. This should not happen. */ |
| GST_WARNING_OBJECT (preset, "property '%s' not in object", props[i]); |
| g_free (str); |
| continue; |
| } |
| |
| /* try to deserialize the property value from the keyfile and set it as |
| * the object property */ |
| g_value_init (&gvalue, property->value_type); |
| if (gst_value_deserialize (&gvalue, str)) { |
| if (is_child_proxy) { |
| gst_child_proxy_set_property ((GstChildProxy *) preset, props[i], |
| &gvalue); |
| } else { |
| g_object_set_property ((GObject *) preset, props[i], &gvalue); |
| } |
| } else { |
| GST_WARNING_OBJECT (preset, |
| "deserialization of value '%s' for property '%s' failed", str, |
| props[i]); |
| } |
| g_value_unset (&gvalue); |
| g_free (str); |
| } |
| g_strfreev (props); |
| |
| return TRUE; |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "no presets"); |
| return FALSE; |
| } |
| no_group: |
| { |
| GST_WARNING_OBJECT (preset, "no preset named '%s'", name); |
| return FALSE; |
| } |
| no_properties: |
| { |
| GST_INFO_OBJECT (preset, "no properties"); |
| return FALSE; |
| } |
| } |
| |
| /* save the presets file. A copy of the existing presets file is stored in a |
| * .bak file */ |
| static gboolean |
| gst_preset_default_save_presets_file (GstPreset * preset) |
| { |
| GKeyFile *presets; |
| const gchar *preset_path; |
| GError *error = NULL; |
| gchar *bak_file_name; |
| gboolean backup = TRUE; |
| gchar *data; |
| gsize data_size; |
| |
| preset_get_paths (preset, &preset_path, NULL, NULL); |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| GST_DEBUG_OBJECT (preset, "saving preset file: '%s'", preset_path); |
| |
| /* create backup if possible */ |
| bak_file_name = g_strdup_printf ("%s.bak", preset_path); |
| if (g_file_test (bak_file_name, G_FILE_TEST_EXISTS)) { |
| if (g_unlink (bak_file_name)) { |
| backup = FALSE; |
| GST_INFO_OBJECT (preset, "cannot remove old backup file : %s", |
| bak_file_name); |
| } |
| } |
| if (backup) { |
| if (g_rename (preset_path, bak_file_name)) { |
| GST_INFO_OBJECT (preset, "cannot backup file : %s -> %s", preset_path, |
| bak_file_name); |
| } |
| } |
| g_free (bak_file_name); |
| |
| /* update gstreamer version */ |
| g_key_file_set_string (presets, PRESET_HEADER, PRESET_HEADER_VERSION, |
| PACKAGE_VERSION); |
| |
| /* get new contents, wee need this to save it */ |
| if (!(data = g_key_file_to_data (presets, &data_size, &error))) |
| goto convert_failed; |
| |
| /* write presets */ |
| if (!g_file_set_contents (preset_path, data, data_size, &error)) |
| goto write_failed; |
| |
| g_free (data); |
| |
| return TRUE; |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, |
| "no presets, trying to unlink possibly existing preset file: '%s'", |
| preset_path); |
| g_unlink (preset_path); |
| return FALSE; |
| } |
| convert_failed: |
| { |
| GST_WARNING_OBJECT (preset, "can not get the keyfile contents: %s", |
| error->message); |
| g_error_free (error); |
| g_free (data); |
| return FALSE; |
| } |
| write_failed: |
| { |
| GST_WARNING_OBJECT (preset, "Unable to store preset file %s: %s", |
| preset_path, error->message); |
| g_error_free (error); |
| g_free (data); |
| return FALSE; |
| } |
| } |
| |
| /* save the preset with the given name */ |
| static gboolean |
| gst_preset_default_save_preset (GstPreset * preset, const gchar * name) |
| { |
| GKeyFile *presets; |
| gchar **props; |
| guint i; |
| GObjectClass *gclass; |
| gboolean is_child_proxy; |
| |
| GST_INFO_OBJECT (preset, "saving new preset: %s", name); |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| /* take copies of current gobject properties from preset */ |
| if (!(props = gst_preset_get_property_names (preset))) |
| goto no_properties; |
| |
| gclass = G_OBJECT_CLASS (GST_ELEMENT_GET_CLASS (preset)); |
| is_child_proxy = GST_IS_CHILD_PROXY (preset); |
| |
| /* loop over the object properties and store the property value in the |
| * keyfile */ |
| for (i = 0; props[i]; i++) { |
| GValue gvalue = { 0, }; |
| gchar *str; |
| GParamSpec *property = NULL; |
| |
| if (is_child_proxy) { |
| gst_child_proxy_lookup ((GstChildProxy *) preset, props[i], NULL, |
| &property); |
| } else { |
| property = g_object_class_find_property (gclass, props[i]); |
| } |
| if (!property) { |
| /* the element said it supported the property but then it does not have |
| * that property. This should not happen. */ |
| GST_WARNING_OBJECT (preset, "property '%s' not in object", props[i]); |
| continue; |
| } |
| |
| if (property->flags & G_PARAM_DEPRECATED) { |
| GST_INFO_OBJECT (preset, "Not saving property %s as it is deprecated", |
| property->name); |
| continue; |
| } |
| |
| g_value_init (&gvalue, property->value_type); |
| if (is_child_proxy) { |
| gst_child_proxy_get_property ((GstChildProxy *) preset, props[i], |
| &gvalue); |
| } else { |
| g_object_get_property ((GObject *) preset, props[i], &gvalue); |
| } |
| |
| if ((str = gst_value_serialize (&gvalue))) { |
| g_key_file_set_string (presets, name, props[i], (gpointer) str); |
| g_free (str); |
| } else { |
| GST_WARNING_OBJECT (preset, "serialization for property '%s' failed", |
| props[i]); |
| } |
| g_value_unset (&gvalue); |
| } |
| GST_INFO_OBJECT (preset, " saved"); |
| g_strfreev (props); |
| |
| /* save updated version */ |
| return gst_preset_default_save_presets_file (preset); |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "no presets"); |
| return FALSE; |
| } |
| no_properties: |
| { |
| GST_INFO_OBJECT (preset, "no properties"); |
| return FALSE; |
| } |
| } |
| |
| /* copies all keys and comments from one group to another, deleting the old |
| * group. */ |
| static gboolean |
| gst_preset_default_rename_preset (GstPreset * preset, const gchar * old_name, |
| const gchar * new_name) |
| { |
| GKeyFile *presets; |
| gchar *str; |
| gchar **keys; |
| gsize i, num_keys; |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| if (!g_key_file_has_group (presets, old_name)) |
| goto no_group; |
| |
| /* copy group comment if there is any */ |
| if ((str = g_key_file_get_comment (presets, old_name, NULL, NULL))) { |
| g_key_file_set_comment (presets, new_name, NULL, str, NULL); |
| g_free (str); |
| } |
| |
| /* get all keys from the old group and copy them in the new group */ |
| keys = g_key_file_get_keys (presets, old_name, &num_keys, NULL); |
| for (i = 0; i < num_keys; i++) { |
| /* copy key comment if there is any */ |
| if ((str = g_key_file_get_comment (presets, old_name, keys[i], NULL))) { |
| g_key_file_set_comment (presets, new_name, keys[i], str, NULL); |
| g_free (str); |
| } |
| /* copy key value */ |
| str = g_key_file_get_value (presets, old_name, keys[i], NULL); |
| g_key_file_set_value (presets, new_name, keys[i], str); |
| g_free (str); |
| } |
| g_strfreev (keys); |
| |
| /* remove old group */ |
| g_key_file_remove_group (presets, old_name, NULL); |
| |
| /* save updated version */ |
| return gst_preset_default_save_presets_file (preset); |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "no presets"); |
| return FALSE; |
| } |
| no_group: |
| { |
| GST_WARNING_OBJECT (preset, "no preset named %s", old_name); |
| return FALSE; |
| } |
| } |
| |
| /* delete a group from the keyfile */ |
| static gboolean |
| gst_preset_default_delete_preset (GstPreset * preset, const gchar * name) |
| { |
| GKeyFile *presets; |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| /* get the group */ |
| if (!g_key_file_has_group (presets, name)) |
| goto no_group; |
| |
| /* remove the group */ |
| g_key_file_remove_group (presets, name, NULL); |
| |
| /* save updated version */ |
| return gst_preset_default_save_presets_file (preset); |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "no presets"); |
| return FALSE; |
| } |
| no_group: |
| { |
| GST_WARNING_OBJECT (preset, "no preset named %s", name); |
| return FALSE; |
| } |
| } |
| |
| static gboolean |
| gst_preset_default_set_meta (GstPreset * preset, const gchar * name, |
| const gchar * tag, const gchar * value) |
| { |
| GKeyFile *presets; |
| gchar *key; |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| key = g_strdup_printf ("_meta/%s", tag); |
| if (value && *value) { |
| g_key_file_set_value (presets, name, key, value); |
| } else { |
| g_key_file_remove_key (presets, name, key, NULL); |
| } |
| g_free (key); |
| |
| /* save updated keyfile */ |
| return gst_preset_default_save_presets_file (preset); |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "no presets"); |
| return FALSE; |
| } |
| } |
| |
| /* the caller must free @value after usage */ |
| static gboolean |
| gst_preset_default_get_meta (GstPreset * preset, const gchar * name, |
| const gchar * tag, gchar ** value) |
| { |
| GKeyFile *presets; |
| gchar *key; |
| |
| /* get the presets from the type */ |
| if (!(presets = preset_get_keyfile (preset))) |
| goto no_presets; |
| |
| key = g_strdup_printf ("_meta/%s", tag); |
| *value = g_key_file_get_value (presets, name, key, NULL); |
| g_free (key); |
| |
| return TRUE; |
| |
| /* ERRORS */ |
| no_presets: |
| { |
| GST_WARNING_OBJECT (preset, "no presets"); |
| *value = NULL; |
| return FALSE; |
| } |
| } |
| |
| /* wrapper */ |
| |
| /** |
| * gst_preset_get_preset_names: |
| * @preset: a #GObject that implements #GstPreset |
| * |
| * Get a copy of preset names as a %NULL terminated string array. |
| * |
| * Returns: (transfer full) (array zero-terminated=1) (element-type gchar*): |
| * list with names, use g_strfreev() after usage. |
| */ |
| gchar ** |
| gst_preset_get_preset_names (GstPreset * preset) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), NULL); |
| |
| return (GST_PRESET_GET_INTERFACE (preset)->get_preset_names (preset)); |
| } |
| |
| /** |
| * gst_preset_get_property_names: |
| * @preset: a #GObject that implements #GstPreset |
| * |
| * Get a the names of the GObject properties that can be used for presets. |
| * |
| * Returns: (transfer full) (array zero-terminated=1) (element-type gchar*): an |
| * array of property names which should be freed with g_strfreev() after use. |
| */ |
| gchar ** |
| gst_preset_get_property_names (GstPreset * preset) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), NULL); |
| |
| return (GST_PRESET_GET_INTERFACE (preset)->get_property_names (preset)); |
| } |
| |
| /** |
| * gst_preset_load_preset: |
| * @preset: a #GObject that implements #GstPreset |
| * @name: preset name to load |
| * |
| * Load the given preset. |
| * |
| * Returns: %TRUE for success, %FALSE if e.g. there is no preset with that @name |
| */ |
| gboolean |
| gst_preset_load_preset (GstPreset * preset, const gchar * name) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), FALSE); |
| g_return_val_if_fail (name, FALSE); |
| |
| return (GST_PRESET_GET_INTERFACE (preset)->load_preset (preset, name)); |
| } |
| |
| /** |
| * gst_preset_save_preset: |
| * @preset: a #GObject that implements #GstPreset |
| * @name: preset name to save |
| * |
| * Save the current object settings as a preset under the given name. If there |
| * is already a preset by this @name it will be overwritten. |
| * |
| * Returns: %TRUE for success, %FALSE |
| */ |
| gboolean |
| gst_preset_save_preset (GstPreset * preset, const gchar * name) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), FALSE); |
| g_return_val_if_fail (name, FALSE); |
| |
| return (GST_PRESET_GET_INTERFACE (preset)->save_preset (preset, name)); |
| } |
| |
| /** |
| * gst_preset_rename_preset: |
| * @preset: a #GObject that implements #GstPreset |
| * @old_name: current preset name |
| * @new_name: new preset name |
| * |
| * Renames a preset. If there is already a preset by the @new_name it will be |
| * overwritten. |
| * |
| * Returns: %TRUE for success, %FALSE if e.g. there is no preset with @old_name |
| */ |
| gboolean |
| gst_preset_rename_preset (GstPreset * preset, const gchar * old_name, |
| const gchar * new_name) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), FALSE); |
| g_return_val_if_fail (old_name, FALSE); |
| g_return_val_if_fail (new_name, FALSE); |
| |
| return (GST_PRESET_GET_INTERFACE (preset)->rename_preset (preset, old_name, |
| new_name)); |
| } |
| |
| /** |
| * gst_preset_delete_preset: |
| * @preset: a #GObject that implements #GstPreset |
| * @name: preset name to remove |
| * |
| * Delete the given preset. |
| * |
| * Returns: %TRUE for success, %FALSE if e.g. there is no preset with that @name |
| */ |
| gboolean |
| gst_preset_delete_preset (GstPreset * preset, const gchar * name) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), FALSE); |
| g_return_val_if_fail (name, FALSE); |
| |
| return (GST_PRESET_GET_INTERFACE (preset)->delete_preset (preset, name)); |
| } |
| |
| /** |
| * gst_preset_set_meta: |
| * @preset: a #GObject that implements #GstPreset |
| * @name: preset name |
| * @tag: meta data item name |
| * @value: (allow-none): new value |
| * |
| * Sets a new @value for an existing meta data item or adds a new item. Meta |
| * data @tag names can be something like e.g. "comment". Supplying %NULL for the |
| * @value will unset an existing value. |
| * |
| * Returns: %TRUE for success, %FALSE if e.g. there is no preset with that @name |
| */ |
| gboolean |
| gst_preset_set_meta (GstPreset * preset, const gchar * name, const gchar * tag, |
| const gchar * value) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), FALSE); |
| g_return_val_if_fail (name, FALSE); |
| g_return_val_if_fail (tag, FALSE); |
| |
| return GST_PRESET_GET_INTERFACE (preset)->set_meta (preset, name, tag, value); |
| } |
| |
| /** |
| * gst_preset_get_meta: |
| * @preset: a #GObject that implements #GstPreset |
| * @name: preset name |
| * @tag: meta data item name |
| * @value: (out callee-allocates): value |
| * |
| * Gets the @value for an existing meta data @tag. Meta data @tag names can be |
| * something like e.g. "comment". Returned values need to be released when done. |
| * |
| * Returns: %TRUE for success, %FALSE if e.g. there is no preset with that @name |
| * or no value for the given @tag |
| */ |
| gboolean |
| gst_preset_get_meta (GstPreset * preset, const gchar * name, const gchar * tag, |
| gchar ** value) |
| { |
| g_return_val_if_fail (GST_IS_PRESET (preset), FALSE); |
| g_return_val_if_fail (name, FALSE); |
| g_return_val_if_fail (tag, FALSE); |
| g_return_val_if_fail (value, FALSE); |
| |
| return GST_PRESET_GET_INTERFACE (preset)->get_meta (preset, name, tag, value); |
| } |
| |
| /** |
| * gst_preset_set_app_dir: |
| * @app_dir: (type filename): the application specific preset dir |
| * |
| * Sets an extra directory as an absolute path that should be considered when |
| * looking for presets. Any presets in the application dir will shadow the |
| * system presets. |
| * |
| * Returns: %TRUE for success, %FALSE if the dir already has been set |
| */ |
| gboolean |
| gst_preset_set_app_dir (const gchar * app_dir) |
| { |
| g_return_val_if_fail (app_dir, FALSE); |
| |
| if (!preset_app_dir) { |
| preset_app_dir = g_strdup (app_dir); |
| return TRUE; |
| } |
| return FALSE; |
| } |
| |
| /** |
| * gst_preset_get_app_dir: |
| * |
| * Gets the directory for application specific presets if set by the |
| * application. |
| * |
| * Returns: (nullable) (type filename): the directory or %NULL, don't free or modify |
| * the string |
| */ |
| const gchar * |
| gst_preset_get_app_dir (void) |
| { |
| return preset_app_dir; |
| } |
| |
| /** |
| * gst_preset_is_editable: |
| * @preset: a #GObject that implements #GstPreset |
| * |
| * Check if one can add new presets, change existing ones and remove presets. |
| * |
| * Returns: %TRUE if presets are editable or %FALSE if they are static |
| * |
| * Since: 1.6 |
| */ |
| gboolean |
| gst_preset_is_editable (GstPreset * preset) |
| { |
| GstPresetInterface *iface = GST_PRESET_GET_INTERFACE (preset); |
| |
| return iface->save_preset && iface->delete_preset; |
| } |
| |
| /* class internals */ |
| |
| static void |
| gst_preset_class_init (GstPresetInterface * iface) |
| { |
| iface->get_preset_names = gst_preset_default_get_preset_names; |
| iface->get_property_names = gst_preset_default_get_property_names; |
| |
| iface->load_preset = gst_preset_default_load_preset; |
| iface->save_preset = gst_preset_default_save_preset; |
| iface->rename_preset = gst_preset_default_rename_preset; |
| iface->delete_preset = gst_preset_default_delete_preset; |
| |
| iface->set_meta = gst_preset_default_set_meta; |
| iface->get_meta = gst_preset_default_get_meta; |
| } |
| |
| static void |
| gst_preset_base_init (gpointer g_class) |
| { |
| static gboolean initialized = FALSE; |
| |
| if (!initialized) { |
| /* init default implementation */ |
| GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT, "preset", |
| GST_DEBUG_FG_WHITE | GST_DEBUG_BG_BLACK, "preset interface"); |
| |
| /* create quarks for use with g_type_{g,s}et_qdata() */ |
| preset_quark = g_quark_from_static_string ("GstPreset::presets"); |
| preset_user_path_quark = |
| g_quark_from_static_string ("GstPreset::user_path"); |
| preset_app_path_quark = g_quark_from_static_string ("GstPreset::app_path"); |
| preset_system_path_quark = |
| g_quark_from_static_string ("GstPreset::system_path"); |
| |
| #if 0 |
| /* create interface properties, each element would need to override this |
| * g_object_class_override_property(gobject_class, PROP_PRESET_NAME, "preset-name"); |
| * and in _set_property() do |
| * case PROP_PRESET_NAME: { |
| * gchar *name = g_value_get_string (value); |
| * if (name) |
| * gst_preset_load_preset(preset, name); |
| * } break; |
| */ |
| g_object_interface_install_property (g_class, |
| g_param_spec_string ("preset-name", |
| "preset-name property", |
| "load given preset", NULL, G_PARAM_WRITABLE)); |
| #endif |
| |
| initialized = TRUE; |
| } |
| } |
| |
| GType |
| gst_preset_get_type (void) |
| { |
| static volatile gsize type = 0; |
| |
| if (g_once_init_enter (&type)) { |
| GType _type; |
| const GTypeInfo info = { |
| sizeof (GstPresetInterface), |
| (GBaseInitFunc) gst_preset_base_init, /* base_init */ |
| NULL, /* base_finalize */ |
| (GClassInitFunc) gst_preset_class_init, /* class_init */ |
| NULL, /* class_finalize */ |
| NULL, /* class_data */ |
| 0, |
| 0, /* n_preallocs */ |
| NULL /* instance_init */ |
| }; |
| _type = g_type_register_static (G_TYPE_INTERFACE, "GstPreset", &info, 0); |
| g_once_init_leave (&type, _type); |
| } |
| return type; |
| } |