Remove g_irepository_register_file in favor of g_irepository_require.

2008-08-20  Colin Walters  <walters@verbum.org>

	* girepository/girepository.c: Remove
	g_irepository_register_file in favor of
	g_irepository_require.  There are two
	possible deployment scenarios for typelibs:
	First, separate in $DATADIR/gitypelibs/.  Second,
	they may be embedded in shlibs.  However since
	the first is now the normal case, the API is
	optimized around it.

	Refactor internals to look up typelibs for
	namespaces just-in-time, but we expect
	consumers to call g_irepository_require.

	Also, add some docs.  No one has died from that
	before.
	* gir/Makefile.am: Need --library for glib.
	* giscanner/girwriter.py: Write out shared-library.
	* tools/g-ir-writer: Take the first --library
	argument as the target of shared-library.  In
	the future we should make this nicer with pkg-config
	probably.

svn path=/trunk/; revision=426
This commit is contained in:
Colin Walters 2008-08-20 23:56:40 +00:00 committed by Colin Walters
parent 4de606f07d
commit 61ae81c9c8
2 changed files with 90 additions and 40 deletions

View File

@ -28,6 +28,7 @@
#include "girepository.h" #include "girepository.h"
#include "gtypelib.h" #include "gtypelib.h"
static GStaticMutex globals_lock = G_STATIC_MUTEX_INIT;
static GIRepository *default_repository = NULL; static GIRepository *default_repository = NULL;
static GHashTable *default_typelib = NULL; static GHashTable *default_typelib = NULL;
static GSList *search_path = NULL; static GSList *search_path = NULL;
@ -68,6 +69,39 @@ g_irepository_class_init (GIRepositoryClass *class)
g_type_class_add_private (class, sizeof (GIRepositoryPrivate)); g_type_class_add_private (class, sizeof (GIRepositoryPrivate));
} }
static void
init_globals ()
{
g_static_mutex_lock (&globals_lock);
if (default_repository == NULL)
{
default_repository = g_object_new (G_TYPE_IREPOSITORY, NULL);
if (default_typelib == NULL)
default_typelib = g_hash_table_new_full (g_str_hash, g_str_equal,
(GDestroyNotify) NULL,
(GDestroyNotify) g_typelib_free);
default_repository->priv->typelib = default_typelib;
}
if (search_path == NULL)
{
const gchar *const *datadirs;
const gchar *const *dir;
datadirs = g_get_system_data_dirs ();
search_path = NULL;
for (dir = datadirs; *dir; dir++) {
char *path = g_build_filename (*dir, "gitypelibs", NULL);
search_path = g_slist_prepend (search_path, path);
}
search_path = g_slist_reverse (search_path);
}
g_static_mutex_unlock (&globals_lock);
}
const gchar * const gchar *
g_irepository_register (GIRepository *repository, g_irepository_register (GIRepository *repository,
GTypelib *typelib) GTypelib *typelib)
@ -93,10 +127,7 @@ g_irepository_register (GIRepository *repository,
} }
else else
{ {
if (default_typelib == NULL) init_globals ();
default_typelib = g_hash_table_new_full (g_str_hash, g_str_equal,
(GDestroyNotify) NULL,
(GDestroyNotify) g_typelib_free);
table = default_typelib; table = default_typelib;
} }
@ -126,8 +157,11 @@ g_irepository_unregister (GIRepository *repository,
if (repository != NULL) if (repository != NULL)
table = repository->priv->typelib; table = repository->priv->typelib;
else else
table = default_typelib; {
init_globals ();
table = default_typelib;
}
if (!g_hash_table_remove (table, namespace)) if (!g_hash_table_remove (table, namespace))
{ {
@ -144,7 +178,10 @@ g_irepository_is_registered (GIRepository *repository,
if (repository != NULL) if (repository != NULL)
table = repository->priv->typelib; table = repository->priv->typelib;
else else
table = default_typelib; {
init_globals ();
table = default_typelib;
}
return g_hash_table_lookup (table, namespace) != NULL; return g_hash_table_lookup (table, namespace) != NULL;
} }
@ -152,16 +189,7 @@ g_irepository_is_registered (GIRepository *repository,
GIRepository * GIRepository *
g_irepository_get_default (void) g_irepository_get_default (void)
{ {
if (default_repository == NULL) init_globals ();
{
default_repository = g_object_new (G_TYPE_IREPOSITORY, NULL);
if (default_typelib == NULL)
default_typelib = g_hash_table_new_full (g_str_hash, g_str_equal,
(GDestroyNotify) NULL,
(GDestroyNotify) g_typelib_free);
default_repository->priv->typelib = default_typelib;
}
return default_repository; return default_repository;
} }
@ -315,6 +343,17 @@ g_irepository_find_by_gtype (GIRepository *repository,
return data.iface; return data.iface;
} }
/**
* g_irepository_find_by_name
* @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace to search in, may be %NULL for all
* @name: Name to find
*
* Searches for a particular name in one or all namespaces.
* See #g_irepository_require to load metadata for namespaces.
* Returns: #GIBaseInfo representing metadata about @name, or %NULL
*/
GIBaseInfo * GIBaseInfo *
g_irepository_find_by_name (GIRepository *repository, g_irepository_find_by_name (GIRepository *repository,
const gchar *namespace, const gchar *namespace,
@ -352,6 +391,16 @@ collect_namespaces (gpointer key,
*list = g_list_append (*list, key); *list = g_list_append (*list, key);
} }
/**
* g_irepository_get_namespaces
* @repository: A #GIRepository, may be %NULL for the default
*
* Return the list of currently known namespaces. Normally
* if you want a particular namespace, you should call
* #g_irepository_require to load it in.
* Returns: List of namespaces
*/
gchar ** gchar **
g_irepository_get_namespaces (GIRepository *repository) g_irepository_get_namespaces (GIRepository *repository)
{ {
@ -390,23 +439,25 @@ g_irepository_get_shared_library (GIRepository *repository,
static inline void static inline void
g_irepository_build_search_path (void) g_irepository_build_search_path (void)
{ {
const gchar *const *datadirs;
const gchar *const *dir;
datadirs = g_get_system_data_dirs ();
search_path = NULL;
for (dir = datadirs; *dir; dir++) {
char *path = g_build_filename (*dir, "gitypelibs", NULL);
search_path = g_slist_prepend (search_path, path);
}
search_path = g_slist_reverse (search_path);
} }
/**
* g_irepository_require
* @repository: Repository, may be null for the default
* @namespace: GI namespace to use, e.g. "Gtk"
* @error: a #GError.
*
* Force the namespace @namespace to be loaded if it isn't
* already. If @namespace is not loaded, this function will
* search for a ".typelib" file using the repository search
* path.
*
* Returns: Namespace if successful, NULL otherwise
*/
const gchar * const gchar *
g_irepository_register_file (GIRepository *repository, g_irepository_require (GIRepository *repository,
const gchar *namespace, const gchar *namespace,
GError **error) GError **error)
{ {
GSList *ldir; GSList *ldir;
const char *dir; const char *dir;
@ -422,14 +473,14 @@ g_irepository_register_file (GIRepository *repository,
if (repository != NULL) if (repository != NULL)
table = repository->priv->typelib; table = repository->priv->typelib;
else else
table = default_typelib; {
init_globals ();
table = default_typelib;
}
/* don't bother loading a namespace if already registered */ /* don't bother loading a namespace if already registered */
if (g_hash_table_lookup (table, namespace)) if (g_hash_table_lookup (table, namespace))
return NULL; return namespace;
if (search_path == NULL)
g_irepository_build_search_path ();
fname = g_strconcat (namespace, ".typelib", NULL); fname = g_strconcat (namespace, ".typelib", NULL);
@ -438,7 +489,6 @@ g_irepository_register_file (GIRepository *repository,
full_path = g_build_filename (dir, fname, NULL); full_path = g_build_filename (dir, fname, NULL);
mfile = g_mapped_file_new (full_path, FALSE, &error1); mfile = g_mapped_file_new (full_path, FALSE, &error1);
if (error1) { if (error1) {
g_debug ("Failed to mmap \"%s\": %s", full_path, error1->message);
g_clear_error (&error1); g_clear_error (&error1);
g_free (full_path); g_free (full_path);
continue; continue;

View File

@ -76,14 +76,14 @@ const gchar * g_irepository_register (GIRepository *repository,
GTypelib *typelib); GTypelib *typelib);
void g_irepository_unregister (GIRepository *repository, void g_irepository_unregister (GIRepository *repository,
const gchar *namespace); const gchar *namespace);
const gchar * g_irepository_register_file (GIRepository *repository,
const gchar *filename,
GError **error);
gboolean g_irepository_is_registered (GIRepository *repository, gboolean g_irepository_is_registered (GIRepository *repository,
const gchar *namespace); const gchar *namespace);
GIBaseInfo * g_irepository_find_by_name (GIRepository *repository, GIBaseInfo * g_irepository_find_by_name (GIRepository *repository,
const gchar *namespace, const gchar *namespace,
const gchar *name); const gchar *name);
const char * g_irepository_require (GIRepository *repository,
const char *namespace,
GError **error);
gchar ** g_irepository_get_namespaces (GIRepository *repository); gchar ** g_irepository_get_namespaces (GIRepository *repository);
GIBaseInfo * g_irepository_find_by_gtype (GIRepository *repository, GIBaseInfo * g_irepository_find_by_gtype (GIRepository *repository,
GType gtype); GType gtype);