diff options
Diffstat (limited to 'lib/ephy-shlib-loader.c')
-rw-r--r-- | lib/ephy-shlib-loader.c | 256 |
1 files changed, 256 insertions, 0 deletions
diff --git a/lib/ephy-shlib-loader.c b/lib/ephy-shlib-loader.c new file mode 100644 index 000000000..68276bb67 --- /dev/null +++ b/lib/ephy-shlib-loader.c @@ -0,0 +1,256 @@ +/* + * Copyright (C) 2003 Marco Pesenti Gritti + * Copyright (C) 2003, 2004 Christian Persch + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2, or (at your option) + * any later version. + * + * This program 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + * + * $Id$ + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include "ephy-shlib-loader.h" +#include "ephy-loader.h" +#include "ephy-module.h" +#include "ephy-debug.h" + +#include <string.h> + +#define DATA_KEY "EphyShlibLoader::LoaderData" + +typedef struct +{ + EphyModule *module; + GObject *object; +} LoaderData; + +static GQuark library_quark = 0; + +#define EPHY_SHLIB_LOADER_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_SHLIB_LOADER, EphyShlibLoaderPrivate)) + +struct _EphyShlibLoaderPrivate +{ + GSList *data; +}; + +static void ephy_shlib_loader_class_init (EphyShlibLoaderClass *klass); +static void ephy_shlib_loader_iface_init (EphyLoaderIface *iface); +static void ephy_shlib_loader_init (EphyShlibLoader *loader); + +static GObjectClass *parent_class = NULL; + +GType +ephy_shlib_loader_get_type (void) +{ + static GType type = 0; + + if (type == 0) + { + static const GTypeInfo our_info = + { + sizeof (EphyShlibLoaderClass), + NULL, /* base_init */ + NULL, /* base_finalize */ + (GClassInitFunc) ephy_shlib_loader_class_init, + NULL, + NULL, /* class_data */ + sizeof (EphyShlibLoader), + 0, /* n_preallocs */ + (GInstanceInitFunc) ephy_shlib_loader_init + }; + + static const GInterfaceInfo loader_info = + { + (GInterfaceInitFunc) ephy_shlib_loader_iface_init, + NULL, + NULL + }; + + type = g_type_register_static (G_TYPE_OBJECT, + "EphyShlibLoader", + &our_info, 0); + + g_type_add_interface_static (type, + EPHY_TYPE_LOADER, + &loader_info); + } + + return type; +} + +static void +free_loader_data (LoaderData *data) +{ + g_return_if_fail (data != NULL); + + /* data->module must NOT be unreffed! */ + + if (data->object != NULL) + { + g_object_unref (data->object); + } + + g_free (data); +} + +static void +ephy_shlib_loader_init (EphyShlibLoader *loader) +{ + loader->priv = EPHY_SHLIB_LOADER_GET_PRIVATE (loader); + + LOG ("EphyShlibLoader initialising") +} + +static void +ephy_shlib_loader_finalize (GObject *object) +{ + EphyShlibLoader *loader = EPHY_SHLIB_LOADER (object); + + LOG ("EphyShlibLoader finalising") + + g_slist_foreach (loader->priv->data, (GFunc) free_loader_data, NULL); + g_slist_free (loader->priv->data); + + parent_class->finalize (object); +} + +static int +find_library (const LoaderData *data, + const char *library) +{ + return strcmp (ephy_module_get_path (data->module), library); +} + +static int +find_object (const LoaderData *data, + const GObject *object) +{ + return data->object != object; +} + +static gboolean +idle_unref (GObject *object) +{ + g_object_unref (object); + + return FALSE; +} + +static GObject * +impl_get_object (EphyLoader *eloader, + GData **attributes) +{ + EphyShlibLoader *loader = EPHY_SHLIB_LOADER (eloader); + GSList *l; + LoaderData *data = NULL; + const char *library; + + library = g_datalist_id_get_data (attributes, library_quark); + if (library == NULL) + { + g_warning ("NULL library name!\n"); + return NULL; + } + + l = g_slist_find_custom (loader->priv->data, library, + (GCompareFunc) find_library); + + if (l != NULL) + { + data = l->data; + g_return_val_if_fail (data != NULL, NULL); + + if (data->object != NULL) + { + return g_object_ref (data->object); + } + } + else + { + data = g_new0 (LoaderData, 1); + loader->priv->data = g_slist_prepend (loader->priv->data, data); + } + + if (data->module == NULL) + { + data->module = ephy_module_new (library); + } + + g_return_val_if_fail (data->object == NULL, data->object); + + if (g_type_module_use (G_TYPE_MODULE (data->module)) == FALSE) + { + g_warning ("Could not load extension file at %s\n", + ephy_module_get_path (data->module)); + return NULL; + } + + data->object = ephy_module_new_object (data->module); + + g_type_module_unuse (G_TYPE_MODULE (data->module)); + + if (data->object != NULL) + { + g_object_set_data (G_OBJECT (data->object), DATA_KEY, data); + } + + return data->object; +} + +static void +impl_release_object (EphyLoader *eloader, + GObject *object) +{ + EphyShlibLoader *loader = EPHY_SHLIB_LOADER (eloader); + GSList *l; + LoaderData *data; + + l = g_slist_find_custom (loader->priv->data, object, + (GCompareFunc) find_object); + g_return_if_fail (l != NULL); + data = l->data; + + /* FIXME: should we consider those extensions broken? + * Only unref the extension in the idle loop; if the extension has its + * own functions queued in the idle loop, the functions must exist in + * memory before being called. + */ + g_idle_add ((GSourceFunc) idle_unref, data->object); + data->object = NULL; +} + +static void +ephy_shlib_loader_iface_init (EphyLoaderIface *iface) +{ + iface->type = "shlib"; + iface->get_object = impl_get_object; + iface->release_object = impl_release_object; +} + +static void +ephy_shlib_loader_class_init (EphyShlibLoaderClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + parent_class = g_type_class_peek_parent (klass); + + object_class->finalize = ephy_shlib_loader_finalize; + + g_type_class_add_private (object_class, sizeof (EphyShlibLoaderPrivate)); + + library_quark = g_quark_from_string ("library"); +} |