Bug 552858: versioning

This is a big patch.  You should probably remove your installation
	tree to be cleaner.

	* docs/typelib-format.txt: Add nsversion entry which holds
	version of namespace.
	* girepository/girepository.h: Add 'version' parameter to
	g_irepository_require.  This may be NULL.  Normally
	bindings should pass an explicit version though.
	* girepository/girepository.c: Lots of infrastructure to
	support versioning.  Add some more documentation.  Disallow
	some usage of NULL namespaces.
	* girepository/girmodule.c: Add version parameter.
	* girepository/gtypelib.c: Update header size.
	* giscanner/ast.py: Add version to Namespace.
	* giscanner/girparser.py: Parse version attribute from
	XML, pass to Namespace.
	* giscanner/girwriter.py: Write out version parameter.
	* giscanner/transformer.py: Clean up include registration.
	* tests/*: Add version attribute.
	* tests/invoke/invoke.c: Don't try looking up test before
	it's loaded in repository.
	* tools/generate.c: Output version parameter.
	* gir/Makefile.am: Add 2.0 version to .gir files.

svn path=/trunk/; revision=677
This commit is contained in:
Colin Walters 2008-10-12 04:51:48 +00:00
parent 1b2aa59534
commit 3be641f836
7 changed files with 542 additions and 134 deletions

View File

@ -2,6 +2,8 @@
/* GObject introspection: Repository implementation /* GObject introspection: Repository implementation
* *
* Copyright (C) 2005 Matthias Clasen * Copyright (C) 2005 Matthias Clasen
* Copyright (C) 2008 Colin Walters <walters@verbum.org>
* Copyright (C) 2008 Red Hat, Inc.
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public * modify it under the terms of the GNU Lesser General Public
@ -35,7 +37,7 @@ static GSList *search_path = NULL;
struct _GIRepositoryPrivate struct _GIRepositoryPrivate
{ {
GHashTable *typelibs; /* (string) namespace -> GTypelib */ GHashTable *typelibs; /* (string) namespace -> GTypelib */
GHashTable *lazy_typelibs; /* (string) namespace -> GTypelib */ GHashTable *lazy_typelibs; /* (string) namespace-version -> GTypelib */
}; };
G_DEFINE_TYPE (GIRepository, g_irepository, G_TYPE_OBJECT); G_DEFINE_TYPE (GIRepository, g_irepository, G_TYPE_OBJECT);
@ -59,6 +61,7 @@ g_irepository_finalize (GObject *object)
GIRepository *repository = G_IREPOSITORY (object); GIRepository *repository = G_IREPOSITORY (object);
g_hash_table_destroy (repository->priv->typelibs); g_hash_table_destroy (repository->priv->typelibs);
g_hash_table_destroy (repository->priv->lazy_typelibs);
(* G_OBJECT_CLASS (g_irepository_parent_class)->finalize) (G_OBJECT (repository)); (* G_OBJECT_CLASS (g_irepository_parent_class)->finalize) (G_OBJECT (repository));
} }
@ -146,19 +149,52 @@ get_repository (GIRepository *repository)
} }
} }
static GTypelib *
check_version_conflict (GTypelib *typelib,
const gchar *namespace,
const gchar *expected_version,
char **version_conflict)
{
Header *header;
const char *loaded_version;
if (expected_version == NULL)
{
if (version_conflict)
*version_conflict = NULL;
return typelib;
}
header = (Header*)typelib->data;
loaded_version = g_typelib_get_string (typelib, header->nsversion);
g_assert (loaded_version != NULL);
if (strcmp (expected_version, loaded_version) != 0)
{
if (version_conflict)
*version_conflict = (char*)loaded_version;
return NULL;
}
if (version_conflict)
*version_conflict = NULL;
return typelib;
}
static GTypelib * static GTypelib *
get_registered_status (GIRepository *repository, get_registered_status (GIRepository *repository,
const char *namespace, const char *namespace,
const char *version,
gboolean allow_lazy, gboolean allow_lazy,
gboolean *lazy_status) gboolean *lazy_status,
char **version_conflict)
{ {
GTypelib *typelib; GTypelib *typelib;
repository = get_repository (repository); repository = get_repository (repository);
if (lazy_status) if (lazy_status)
*lazy_status = FALSE; *lazy_status = FALSE;
typelib = g_hash_table_lookup (repository->priv->typelibs, namespace); typelib = g_hash_table_lookup (repository->priv->typelibs, namespace);
if (typelib) if (typelib)
return typelib; return check_version_conflict (typelib, namespace, version, version_conflict);
typelib = g_hash_table_lookup (repository->priv->lazy_typelibs, namespace); typelib = g_hash_table_lookup (repository->priv->lazy_typelibs, namespace);
if (!typelib) if (!typelib)
return NULL; return NULL;
@ -166,14 +202,15 @@ get_registered_status (GIRepository *repository,
*lazy_status = TRUE; *lazy_status = TRUE;
if (!allow_lazy) if (!allow_lazy)
return NULL; return NULL;
return typelib; return check_version_conflict (typelib, namespace, version, version_conflict);
} }
static GTypelib * static GTypelib *
get_registered (GIRepository *repository, get_registered (GIRepository *repository,
const char *namespace) const char *namespace,
const char *version)
{ {
return get_registered_status (repository, namespace, TRUE, NULL); return get_registered_status (repository, namespace, version, TRUE, NULL, NULL);
} }
static gboolean static gboolean
@ -192,19 +229,28 @@ load_dependencies_recurse (GIRepository *repository,
for (i = 0; dependencies[i]; i++) for (i = 0; dependencies[i]; i++)
{ {
char *dependency = dependencies[i]; char *dependency = dependencies[i];
const char *last_dash;
char *dependency_namespace;
const char *dependency_version;
last_dash = strrchr (dependency, '-');
dependency_namespace = g_strndup (dependency, last_dash - dependency);
dependency_version = last_dash+1;
if (!g_irepository_require (repository, dependency, if (!g_irepository_require (repository, dependency_namespace, dependency_version,
0, error)) 0, error))
{ {
g_free (dependency_namespace);
g_strfreev (dependencies); g_strfreev (dependencies);
return FALSE; return FALSE;
} }
g_free (dependency_namespace);
} }
g_strfreev (dependencies); g_strfreev (dependencies);
} }
return TRUE; return TRUE;
} }
static const char * static const char *
register_internal (GIRepository *repository, register_internal (GIRepository *repository,
const char *source, const char *source,
@ -214,6 +260,7 @@ register_internal (GIRepository *repository,
{ {
Header *header; Header *header;
const gchar *namespace; const gchar *namespace;
const gchar *version;
gboolean was_loaded; gboolean was_loaded;
gboolean currently_lazy; gboolean currently_lazy;
@ -224,6 +271,7 @@ register_internal (GIRepository *repository,
g_return_val_if_fail (header != NULL, FALSE); g_return_val_if_fail (header != NULL, FALSE);
namespace = g_typelib_get_string (typelib, header->namespace); namespace = g_typelib_get_string (typelib, header->namespace);
version = g_typelib_get_string (typelib, header->nsversion);
if (lazy) if (lazy)
{ {
@ -268,7 +316,7 @@ g_irepository_get_dependencies (GIRepository *repository,
repository = get_repository (repository); repository = get_repository (repository);
typelib = get_registered (repository, namespace); typelib = get_registered (repository, namespace, NULL);
g_return_val_if_fail (typelib != NULL, NULL); g_return_val_if_fail (typelib != NULL, NULL);
return get_typelib_dependencies (typelib); return get_typelib_dependencies (typelib);
@ -282,28 +330,74 @@ g_irepository_load_typelib (GIRepository *repository,
{ {
Header *header; Header *header;
const char *namespace; const char *namespace;
const char *nsversion;
gboolean allow_lazy = flags & G_IREPOSITORY_LOAD_FLAG_LAZY; gboolean allow_lazy = flags & G_IREPOSITORY_LOAD_FLAG_LAZY;
gboolean is_lazy; gboolean is_lazy;
char *version_conflict;
repository = get_repository (repository); repository = get_repository (repository);
header = (Header *) typelib->data; header = (Header *) typelib->data;
namespace = g_typelib_get_string (typelib, header->namespace); namespace = g_typelib_get_string (typelib, header->namespace);
nsversion = g_typelib_get_string (typelib, header->nsversion);
if (get_registered_status (repository, namespace, allow_lazy, &is_lazy)) if (get_registered_status (repository, namespace, nsversion, allow_lazy,
return namespace; &is_lazy, &version_conflict))
{
if (version_conflict != NULL)
{
g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_NAMESPACE_VERSION_CONFLICT,
"Attempting to load namespace '%s', version '%s', but '%s' is already loaded",
namespace, nsversion, version_conflict);
return NULL;
}
return namespace;
}
return register_internal (repository, "<builtin>", return register_internal (repository, "<builtin>",
allow_lazy, typelib, error); allow_lazy, typelib, error);
} }
/**
* g_irepository_is_registered
* @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace of interest
* @version: <allow-none>: Required version, may be %NULL for latest
*
* Check whether a particular namespace (and optionally, a specific
* version thereof) is currently loaded. This function is likely to
* only be useful in unusual circumstances; in order to act upon
* metadata in the namespace, you should call #g_irepository_require
* instead which will ensure the namespace is loaded, and return as
* quickly as this function will if it has already been loaded.
*
* Returns: %TRUE if namespace-version is loaded, %FALSE otherwise
*/
gboolean gboolean
g_irepository_is_registered (GIRepository *repository, g_irepository_is_registered (GIRepository *repository,
const gchar *namespace) const gchar *namespace,
const gchar *version)
{ {
repository = get_repository (repository); repository = get_repository (repository);
return get_registered (repository, namespace) != NULL; return get_registered (repository, namespace, version) != NULL;
} }
/**
* g_irepository_get_default
*
* Returns the singleton process-global default #GIRepository. It is
* not currently supported to have multiple repositories in a
* particular process, but this function is provided in the unlikely
* eventuality that it would become possible, and as a convenience for
* higher level language bindings to conform to the GObject method
* call conventions.
* All methods on #GIRepository also accept %NULL as an instance
* parameter to mean this default repository, which is usually more
* convenient for C.
*
* Returns: The global singleton #GIRepository
*/
GIRepository * GIRepository *
g_irepository_get_default (void) g_irepository_get_default (void)
{ {
@ -321,30 +415,33 @@ count_interfaces (gpointer key,
*n_interfaces += ((Header *)typelib)->n_local_entries; *n_interfaces += ((Header *)typelib)->n_local_entries;
} }
/**
* g_irepository_get_n_infos
* @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace to inspect
*
* This function returns the number of metadata entries in
* given namespace @namespace. The namespace must have
* already been loaded before calling this function.
*
* Returns: number of metadata entries
*/
gint gint
g_irepository_get_n_infos (GIRepository *repository, g_irepository_get_n_infos (GIRepository *repository,
const gchar *namespace) const gchar *namespace)
{ {
GTypelib *typelib;
gint n_interfaces = 0; gint n_interfaces = 0;
g_return_val_if_fail (namespace != NULL, -1);
repository = get_repository (repository); repository = get_repository (repository);
if (namespace) typelib = get_registered (repository, namespace, NULL);
{
GTypelib *typelib;
typelib = get_registered (repository, namespace); g_return_val_if_fail (typelib != NULL, -1);
if (typelib) n_interfaces = ((Header *)typelib->data)->n_local_entries;
n_interfaces = ((Header *)typelib->data)->n_local_entries;
}
else
{
g_hash_table_foreach (repository->priv->typelibs,
count_interfaces, &n_interfaces);
g_hash_table_foreach (repository->priv->lazy_typelibs,
count_interfaces, &n_interfaces);
}
return n_interfaces; return n_interfaces;
} }
@ -421,12 +518,27 @@ find_interface (gpointer key,
} }
} }
/**
* g_irepository_get_info
* @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace to inspect
* @index: Offset into namespace metadata for entry
*
* This function returns a particular metadata entry in the
* given namespace @namespace. The namespace must have
* already been loaded before calling this function.
*
* Returns: #GIBaseInfo containing metadata
*/
GIBaseInfo * GIBaseInfo *
g_irepository_get_info (GIRepository *repository, g_irepository_get_info (GIRepository *repository,
const gchar *namespace, const gchar *namespace,
gint index) gint index)
{ {
IfaceData data; IfaceData data;
GTypelib *typelib;
g_return_val_if_fail (namespace != NULL, NULL);
repository = get_repository (repository); repository = get_repository (repository);
@ -435,24 +547,29 @@ g_irepository_get_info (GIRepository *repository,
data.index = index + 1; data.index = index + 1;
data.iface = NULL; data.iface = NULL;
if (namespace) typelib = get_registered (repository, namespace, NULL);
{
GTypelib *typelib; g_return_val_if_fail (typelib != NULL, NULL);
typelib = get_registered (repository, namespace); find_interface ((void *)namespace, typelib, &data);
if (typelib)
find_interface ((void *)namespace, typelib, &data);
}
else
{
g_hash_table_foreach (repository->priv->typelibs, find_interface, &data);
g_hash_table_foreach (repository->priv->lazy_typelibs, find_interface, &data);
}
return data.iface; return data.iface;
} }
/**
* g_irepository_find_by_gtype
* @repository: A #GIRepository, may be %NULL for the default
* @type: GType to search for
*
* Searches all loaded namespaces for a particular #GType. Note that
* in order to locate the metadata, the namespace corresponding to
* the type must first have been loaded. There is currently no
* mechanism for determining the namespace which corresponds to an
* arbitrary GType - thus, this function will function most reliably
* when you have expect the GType to be from a known namespace.
*
* Returns: #GIBaseInfo representing metadata about @type, or %NULL
*/
GIBaseInfo * GIBaseInfo *
g_irepository_find_by_gtype (GIRepository *repository, g_irepository_find_by_gtype (GIRepository *repository,
GType type) GType type)
@ -475,12 +592,14 @@ g_irepository_find_by_gtype (GIRepository *repository,
/** /**
* g_irepository_find_by_name * g_irepository_find_by_name
* @repository: A #GIRepository, may be %NULL for the default * @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace to search in, may be %NULL for all * @namespace: Namespace which will be searched
* @name: Name to find * @name: Entry name to find
*
* Searches for a particular entry in a namespace. Before calling
* this function for a particular namespace, you must call
* #g_irepository_require once to load the namespace, or otherwise
* ensure the namespace has already been loaded.
* *
* 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 * Returns: #GIBaseInfo representing metadata about @name, or %NULL
*/ */
GIBaseInfo * GIBaseInfo *
@ -489,6 +608,9 @@ g_irepository_find_by_name (GIRepository *repository,
const gchar *name) const gchar *name)
{ {
IfaceData data; IfaceData data;
GTypelib *typelib;
g_return_val_if_fail (namespace != NULL, NULL);
repository = get_repository (repository); repository = get_repository (repository);
@ -497,20 +619,11 @@ g_irepository_find_by_name (GIRepository *repository,
data.index = -1; data.index = -1;
data.iface = NULL; data.iface = NULL;
if (namespace) typelib = get_registered (repository, namespace, NULL);
{
GTypelib *typelib; g_return_val_if_fail (typelib != NULL, NULL);
typelib = get_registered (repository, namespace); find_interface ((void *)namespace, typelib, &data);
if (typelib)
find_interface ((void *)namespace, typelib, &data);
}
else
{
g_hash_table_foreach (repository->priv->typelibs, find_interface, &data);
g_hash_table_foreach (repository->priv->lazy_typelibs, find_interface, &data);
}
return data.iface; return data.iface;
} }
@ -529,14 +642,12 @@ collect_namespaces (gpointer key,
* g_irepository_get_namespaces * g_irepository_get_namespaces
* @repository: A #GIRepository, may be %NULL for the default * @repository: A #GIRepository, may be %NULL for the default
* *
* Return the list of currently known namespaces. Normally * Return the list of currently loaded namespaces.
* if you want a particular namespace, you should call *
* #g_irepository_require to load it in. * Returns: <utf8,transfer>: List of namespaces
* Returns: List of namespaces
*/ */
gchar ** gchar **
g_irepository_get_namespaces (GIRepository *repository) g_irepository_get_loaded_namespaces (GIRepository *repository)
{ {
GList *l, *list = NULL; GList *l, *list = NULL;
gchar **names; gchar **names;
@ -556,18 +667,68 @@ g_irepository_get_namespaces (GIRepository *repository)
return names; return names;
} }
/**
* g_irepository_get_version
* @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace to inspect
*
* This function returns the loaded version associated with the given
* namespace @namespace.
*
* Note: The namespace must have already been loaded using a function
* such as #g_irepository_require before calling this function.
*
* Returns: Loaded version
*/
const gchar * const gchar *
g_irepository_get_shared_library (GIRepository *repository, g_irepository_get_version (GIRepository *repository,
const gchar *namespace) const gchar *namespace)
{ {
GTypelib *typelib; GTypelib *typelib;
Header *header; Header *header;
g_return_val_if_fail (namespace != NULL, NULL);
repository = get_repository (repository); repository = get_repository (repository);
typelib = get_registered (repository, namespace); typelib = get_registered (repository, namespace, NULL);
if (!typelib)
return NULL; g_return_val_if_fail (typelib != NULL, NULL);
header = (Header *) typelib->data;
return g_typelib_get_string (typelib, header->nsversion);
}
/**
* g_irepository_get_shared_library
* @repository: A #GIRepository, may be %NULL for the default
* @namespace: Namespace to inspect
*
* This function returns the full path to the shared C library
* associated with the given namespace @namespace. There may be no
* shared library path associated, in which case this function will
* return %NULL.
*
* Note: The namespace must have already been loaded using a function
* such as #g_irepository_require before calling this function.
*
* Returns: Full path to shared library, or %NULL if none associated
*/
const gchar *
g_irepository_get_shared_library (GIRepository *repository,
const gchar *namespace)
{
GTypelib *typelib;
Header *header;
g_return_val_if_fail (namespace != NULL, NULL);
repository = get_repository (repository);
typelib = get_registered (repository, namespace, NULL);
g_return_val_if_fail (typelib != NULL, NULL);
header = (Header *) typelib->data; header = (Header *) typelib->data;
if (header->shared_library) if (header->shared_library)
return g_typelib_get_string (typelib, header->shared_library); return g_typelib_get_string (typelib, header->shared_library);
@ -579,13 +740,14 @@ g_irepository_get_shared_library (GIRepository *repository,
* g_irepository_get_typelib_path * g_irepository_get_typelib_path
* @repository: Repository, may be %NULL for the default * @repository: Repository, may be %NULL for the default
* @namespace: GI namespace to use, e.g. "Gtk" * @namespace: GI namespace to use, e.g. "Gtk"
* @version: <allow-none>: Version of namespace to use, e.g. "0.8", may be %NULL
* *
* If namespace @namespace is loaded, return the full path to the * If namespace @namespace is loaded, return the full path to the
* .typelib file it was loaded from. If the typelib for * .typelib file it was loaded from. If the typelib for
* namespace @namespace was included in a shared library, return * namespace @namespace was included in a shared library, return
* the special string "<builtin>". * the special string "<builtin>".
* *
* Returns: Filesystem path (or <builtin>) if successful, %NULL otherwise * Returns: Filesystem path (or <builtin>) if successful, %NULL if namespace is not loaded
*/ */
const gchar * const gchar *
@ -607,94 +769,323 @@ g_irepository_get_typelib_path (GIRepository *repository,
return ((char*)orig_key) + strlen ((char *) orig_key) + 1; return ((char*)orig_key) + strlen ((char *) orig_key) + 1;
} }
/* This simple search function looks for a specified namespace-version;
it's faster than the full directory listing required for latest version. */
static GMappedFile *
find_namespace_version (const gchar *namespace,
const gchar *version,
gchar **path_ret)
{
GSList *ldir;
GError *error = NULL;
GMappedFile *mfile = NULL;
char *fname;
fname = g_strdup_printf ("%s-%s.typelib", namespace, version);
for (ldir = search_path; ldir; ldir = ldir->next)
{
Header *header;
char *path = g_build_filename (ldir->data, fname, NULL);
mfile = g_mapped_file_new (path, FALSE, &error);
if (error)
{
g_free (path);
g_clear_error (&error);
continue;
}
*path_ret = path;
break;
}
g_free (fname);
return mfile;
}
static gboolean
parse_version (const char *version,
int *major,
int *minor)
{
const char *dot;
const char *end;
*major = strtol (version, &end, 10);
dot = strchr (version, '.');
if (dot == NULL)
{
*minor = 0;
return TRUE;
}
if (dot != end)
return FALSE;
*minor = strtol (dot+1, &end, 10);
if (end != (version + strlen (version)))
return FALSE;
return TRUE;
}
static int
compare_version (const char *v1,
const char *v2)
{
gboolean err;
int v1_major, v1_minor;
int v2_major, v2_minor;
err = parse_version (v1, &v1_major, &v1_minor);
g_assert (!err);
err = parse_version (v2, &v2_major, &v2_minor);
g_assert (!err);
if (v1_major > v2_major)
return 1;
else if (v2_major > v1_major)
return -1;
else if (v1_minor > v2_minor)
return 1;
else if (v2_minor > v1_minor)
return -1;
return 0;
}
struct NamespaceVersionCandidadate
{
GMappedFile *mfile;
char *path;
char *version;
};
static int
compare_candidate_reverse (struct NamespaceVersionCandidadate *c1,
struct NamespaceVersionCandidadate *c2)
{
int result = compare_version (c1->version, c2->version);
if (result > 0)
return -1;
else if (result < 0)
return 1;
else
return 0;
}
static void
free_candidate (struct NamespaceVersionCandidadate *candidate)
{
g_mapped_file_free (candidate->mfile);
g_free (candidate->path);
g_free (candidate->version);
g_free (candidate);
}
static GMappedFile *
find_namespace_latest (const gchar *namespace,
gchar **version_ret,
gchar **path_ret)
{
GSList *ldir;
GError *error = NULL;
char *namespace_dash;
char *namespace_typelib;
GSList *candidates = NULL;
GMappedFile *result = NULL;
*version_ret = NULL;
*path_ret = NULL;
namespace_dash = g_strdup_printf ("%s-", namespace);
namespace_typelib = g_strdup_printf ("%s.typelib", namespace);
for (ldir = search_path; ldir; ldir = ldir->next)
{
GDir *dir;
const char *dirname;
const char *entry;
dirname = (const char*)ldir->data;
dir = g_dir_open (dirname, 0, NULL);
if (dir == NULL)
continue;
while ((entry = g_dir_read_name (dir)) != NULL)
{
GMappedFile *mfile;
char *path, *version;
struct NamespaceVersionCandidadate *candidate;
if (!g_str_has_suffix (entry, ".typelib"))
continue;
if (g_str_has_prefix (entry, namespace_dash))
{
const char *last_dash;
const char *name_end;
int major, minor;
name_end = strrchr (entry, '.');
last_dash = strrchr (entry, '-');
version = g_strndup (last_dash+1, name_end-(last_dash+1));
if (!parse_version (version, &major, &minor))
continue;
}
else
continue;
path = g_build_filename (dirname, entry, NULL);
mfile = g_mapped_file_new (path, FALSE, &error);
if (mfile == NULL)
{
g_free (path);
g_free (version);
g_clear_error (&error);
continue;
}
candidate = g_new0 (struct NamespaceVersionCandidadate, 1);
candidate->mfile = mfile;
candidate->path = path;
candidate->version = version;
candidates = g_slist_prepend (candidates, candidate);
}
g_dir_close (dir);
}
if (candidates != NULL)
{
struct NamespaceVersionCandidadate *elected;
candidates = g_slist_sort (candidates, (GCompareFunc) compare_candidate_reverse);
elected = (struct NamespaceVersionCandidadate *) candidates->data;
/* Remove the elected one so we don't try to free it */
candidates = g_slist_delete_link (candidates, candidates);
result = elected->mfile;
*path_ret = elected->path;
*version_ret = elected->version;
g_slist_foreach (candidates, (GFunc) free_candidate, NULL);
g_slist_free (candidates);
}
g_free (namespace_dash);
g_free (namespace_typelib);
return result;
}
/** /**
* g_irepository_require * g_irepository_require
* @repository: Repository, may be %NULL for the default * @repository: <allow-none>: Repository, may be %NULL for the default
* @namespace: GI namespace to use, e.g. "Gtk" * @namespace: GI namespace to use, e.g. "Gtk"
* @version: <allow-none>: Version of namespace, may be %NULL for latest
* @flags: Set of %GIRepositoryLoadFlags, may be %0 * @flags: Set of %GIRepositoryLoadFlags, may be %0
* @error: a #GError. * @error: a #GError.
* *
* Force the namespace @namespace to be loaded if it isn't * Force the namespace @namespace to be loaded if it isn't already.
* already. If @namespace is not loaded, this function will * If @namespace is not loaded, this function will search for a
* search for a ".typelib" file using the repository search * ".typelib" file using the repository search path. In addition, a
* path. * version @version of namespace may be specified. If @version is
* not specified, the latest will be used.
* *
* Returns: %TRUE if successful, %NULL otherwise * Returns: %TRUE if successful, %NULL otherwise
*/ */
gboolean gboolean
g_irepository_require (GIRepository *repository, g_irepository_require (GIRepository *repository,
const gchar *namespace, const gchar *namespace,
const gchar *version,
GIRepositoryLoadFlags flags, GIRepositoryLoadFlags flags,
GError **error) GError **error)
{ {
GSList *ldir;
const char *dir; const char *dir;
gchar *fname, *full_path;
GMappedFile *mfile; GMappedFile *mfile;
gboolean ret = FALSE;
GError *error1 = NULL; GError *error1 = NULL;
Header *header;
GTypelib *typelib = NULL; GTypelib *typelib = NULL;
const gchar *typelib_namespace, *shlib_fname; const gchar *typelib_namespace, *typelib_version, *shlib_fname;
GModule *module; GModule *module;
guint32 shlib; guint32 shlib;
gboolean allow_lazy = flags & G_IREPOSITORY_LOAD_FLAG_LAZY; gboolean allow_lazy = (flags & G_IREPOSITORY_LOAD_FLAG_LAZY) > 0;
gboolean is_lazy; gboolean is_lazy;
char *version_conflict = NULL;
char *path = NULL;
char *tmp_version = NULL;
g_return_val_if_fail (namespace != NULL, FALSE);
repository = get_repository (repository); repository = get_repository (repository);
if (get_registered_status (repository, namespace, allow_lazy, &is_lazy)) if (get_registered_status (repository, namespace, version, allow_lazy,
&is_lazy, &version_conflict))
return TRUE; return TRUE;
fname = g_strconcat (namespace, ".typelib", NULL); if (version_conflict != NULL)
for (ldir = search_path; ldir; ldir = ldir->next)
{
Header *header;
full_path = g_build_filename (ldir->data, fname, NULL);
mfile = g_mapped_file_new (full_path, FALSE, &error1);
if (error1)
{
g_clear_error (&error1);
continue;
}
typelib = g_typelib_new_from_mapped_file (mfile);
header = (Header *) typelib->data;
typelib_namespace = g_typelib_get_string (typelib, header->namespace);
if (strcmp (typelib_namespace, namespace) != 0)
{
g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_NAMESPACE_MISMATCH,
"Typelib file %s for namespace '%s' contains "
"namespace '%s' which doesn't match the file name",
full_path, namespace, typelib_namespace);
g_free (full_path);
return FALSE;
}
break;
}
if (typelib == NULL)
{ {
g_set_error (error, G_IREPOSITORY_ERROR, g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_TYPELIB_NOT_FOUND, G_IREPOSITORY_ERROR_NAMESPACE_VERSION_CONFLICT,
"Typelib file for namespace '%s' was not found in search" "Requiring namespace '%s' version '%s', but '%s' is already loaded",
" path or could not be openened", namespace); namespace, version, version_conflict);
g_free (full_path);
return FALSE; return FALSE;
} }
g_free (fname); if (version != NULL)
if (!register_internal (repository, full_path, allow_lazy, {
mfile = find_namespace_version (namespace, version, &path);
tmp_version = g_strdup (version);
}
else
{
mfile = find_namespace_latest (namespace, &tmp_version, &path);
}
if (mfile == NULL)
{
const char *error_fmt;
if (version != NULL)
g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_TYPELIB_NOT_FOUND,
"Typelib file %s for namespace '%s', version '%s' not found",
namespace, version);
else
g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_TYPELIB_NOT_FOUND,
"Typelib file for namespace '%s' (any version) not found",
namespace);
goto out;
}
typelib = g_typelib_new_from_mapped_file (mfile);
header = (Header *) typelib->data;
typelib_namespace = g_typelib_get_string (typelib, header->namespace);
typelib_version = g_typelib_get_string (typelib, header->nsversion);
if (strcmp (typelib_namespace, namespace) != 0)
{
g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_NAMESPACE_MISMATCH,
"Typelib file %s for namespace '%s' contains "
"namespace '%s' which doesn't match the file name",
path, namespace, typelib_namespace);
goto out;
}
if (version != NULL && strcmp (typelib_version, version) != 0)
{
g_set_error (error, G_IREPOSITORY_ERROR,
G_IREPOSITORY_ERROR_NAMESPACE_MISMATCH,
"Typelib file %s for namespace '%s' contains "
"version '%s' which doesn't match the expected version '%s'",
path, namespace, typelib_version, version);
goto out;
}
if (!register_internal (repository, path, allow_lazy,
typelib, error)) typelib, error))
{ {
g_typelib_free (typelib); g_typelib_free (typelib);
g_free (full_path); goto out;
return FALSE;
} }
g_free (full_path); ret = TRUE;
return TRUE; out:
g_free (tmp_version);
g_free (path);
return ret;
} }

View File

@ -82,17 +82,19 @@ const char * g_irepository_load_typelib (GIRepository *repository,
GIRepositoryLoadFlags flags, GIRepositoryLoadFlags flags,
GError **error); GError **error);
gboolean g_irepository_is_registered (GIRepository *repository, gboolean g_irepository_is_registered (GIRepository *repository,
const gchar *namespace); const gchar *namespace,
const gchar *version);
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);
gboolean g_irepository_require (GIRepository *repository, gboolean g_irepository_require (GIRepository *repository,
const char *namespace, const gchar *namespace,
const gchar *version,
GIRepositoryLoadFlags flags, GIRepositoryLoadFlags flags,
GError **error); GError **error);
gchar ** g_irepository_get_dependencies (GIRepository *repository, gchar ** g_irepository_get_dependencies (GIRepository *repository,
const char *namespace); const gchar *namespace);
gchar ** g_irepository_get_namespaces (GIRepository *repository); gchar ** g_irepository_get_loaded_namespaces (GIRepository *repository);
GIBaseInfo * g_irepository_find_by_gtype (GIRepository *repository, GIBaseInfo * g_irepository_find_by_gtype (GIRepository *repository,
GType gtype); GType gtype);
gint g_irepository_get_n_infos (GIRepository *repository, gint g_irepository_get_n_infos (GIRepository *repository,
@ -104,6 +106,9 @@ const gchar * g_irepository_get_typelib_path (GIRepository *repository,
const gchar *namespace); const gchar *namespace);
const gchar * g_irepository_get_shared_library (GIRepository *repository, const gchar * g_irepository_get_shared_library (GIRepository *repository,
const gchar *namespace); const gchar *namespace);
const gchar * g_irepository_get_version (GIRepository *repository,
const gchar *namespace);
/* Typelib */ /* Typelib */
GTypelib * g_typelib_new_from_memory (guchar *memory, GTypelib * g_typelib_new_from_memory (guchar *memory,
@ -112,6 +117,7 @@ GTypelib * g_typelib_new_from_const_memory (const guchar *memory,
gsize len); gsize len);
GTypelib * g_typelib_new_from_mapped_file (GMappedFile *mfile); GTypelib * g_typelib_new_from_mapped_file (GMappedFile *mfile);
void g_typelib_free (GTypelib *typelib); void g_typelib_free (GTypelib *typelib);
gboolean g_typelib_symbol (GTypelib *typelib, gboolean g_typelib_symbol (GTypelib *typelib,
const gchar *symbol_name, const gchar *symbol_name,
gpointer *symbol); gpointer *symbol);
@ -121,6 +127,7 @@ typedef enum
{ {
G_IREPOSITORY_ERROR_TYPELIB_NOT_FOUND, G_IREPOSITORY_ERROR_TYPELIB_NOT_FOUND,
G_IREPOSITORY_ERROR_NAMESPACE_MISMATCH, G_IREPOSITORY_ERROR_NAMESPACE_MISMATCH,
G_IREPOSITORY_ERROR_NAMESPACE_VERSION_CONFLICT,
G_IREPOSITORY_ERROR_LIBRARY_NOT_FOUND G_IREPOSITORY_ERROR_LIBRARY_NOT_FOUND
} GIRepositoryError; } GIRepositoryError;

View File

@ -29,13 +29,16 @@
GIrModule * GIrModule *
g_ir_module_new (const gchar *name, const gchar *shared_library) g_ir_module_new (const gchar *name,
const gchar *version,
const gchar *shared_library)
{ {
GIrModule *module; GIrModule *module;
module = g_new0 (GIrModule, 1); module = g_new0 (GIrModule, 1);
module->name = g_strdup (name); module->name = g_strdup (name);
module->version = g_strdup (version);
if (shared_library) if (shared_library)
module->shared_library = g_strdup (shared_library); module->shared_library = g_strdup (shared_library);
else else
@ -156,6 +159,7 @@ g_ir_module_build_typelib (GIrModule *module,
header->dependencies = 0; header->dependencies = 0;
header->size = 0; /* filled in later */ header->size = 0; /* filled in later */
header->namespace = write_string (module->name, strings, data, &header_size); header->namespace = write_string (module->name, strings, data, &header_size);
header->nsversion = write_string (module->version, strings, data, &header_size);
header->shared_library = (module->shared_library? header->shared_library = (module->shared_library?
write_string (module->shared_library, strings, data, &header_size) write_string (module->shared_library, strings, data, &header_size)
: 0); : 0);

View File

@ -32,12 +32,14 @@ typedef struct _GIrModule GIrModule;
struct _GIrModule struct _GIrModule
{ {
gchar *name; gchar *name;
gchar *version;
gchar *shared_library; gchar *shared_library;
GList *dependencies; GList *dependencies;
GList *entries; GList *entries;
}; };
GIrModule *g_ir_module_new (const gchar *name, GIrModule *g_ir_module_new (const gchar *name,
const gchar *nsversion,
const gchar *module_filename); const gchar *module_filename);
void g_ir_module_free (GIrModule *module); void g_ir_module_free (GIrModule *module);

View File

@ -2050,7 +2050,7 @@ parse_include (GMarkupParseContext *context,
g_set_error (error, g_set_error (error,
G_MARKUP_ERROR, G_MARKUP_ERROR,
G_MARKUP_ERROR_INVALID_CONTENT, G_MARKUP_ERROR_INVALID_CONTENT,
"Could not find GIR file '%s'; check XDG_DATA_DIRS or use --includedir", "Could not find GIR file '%s.gir'; check XDG_DATA_DIRS or use --includedir",
name); name);
return FALSE; return FALSE;
} }
@ -2260,16 +2260,19 @@ start_element_handler (GMarkupParseContext *context,
case 'n': case 'n':
if (strcmp (element_name, "namespace") == 0 && ctx->state == STATE_REPOSITORY) if (strcmp (element_name, "namespace") == 0 && ctx->state == STATE_REPOSITORY)
{ {
const gchar *name, *shared_library; const gchar *name, *version, *shared_library;
name = find_attribute ("name", attribute_names, attribute_values); name = find_attribute ("name", attribute_names, attribute_values);
version = find_attribute ("version", attribute_names, attribute_values);
shared_library = find_attribute ("shared-library", attribute_names, attribute_values); shared_library = find_attribute ("shared-library", attribute_names, attribute_values);
if (name == NULL) if (name == NULL)
MISSING_ATTRIBUTE (context, error, element_name, "name"); MISSING_ATTRIBUTE (context, error, element_name, "name");
else if (version == NULL)
MISSING_ATTRIBUTE (context, error, element_name, "version");
else else
{ {
ctx->current_module = g_ir_module_new (name, shared_library); ctx->current_module = g_ir_module_new (name, version, shared_library);
ctx->modules = g_list_append (ctx->modules, ctx->current_module); ctx->modules = g_list_append (ctx->modules, ctx->current_module);
ctx->current_module->dependencies = ctx->dependencies; ctx->current_module->dependencies = ctx->dependencies;

View File

@ -153,7 +153,7 @@ g_typelib_check_sanity (void)
size_check_ok = FALSE; \ size_check_ok = FALSE; \
} }
CHECK_SIZE (Header, 104); CHECK_SIZE (Header, 108);
CHECK_SIZE (DirEntry, 12); CHECK_SIZE (DirEntry, 12);
CHECK_SIZE (SimpleTypeBlob, 4); CHECK_SIZE (SimpleTypeBlob, 4);
CHECK_SIZE (ArgBlob, 12); CHECK_SIZE (ArgBlob, 12);

View File

@ -61,6 +61,7 @@ typedef struct
guint32 size; guint32 size;
guint32 namespace; guint32 namespace;
guint32 nsversion;
guint32 shared_library; guint32 shared_library;
guint16 entry_blob_size; guint16 entry_blob_size;