mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-04-02 22:03:07 +02:00
Add a GirParser object to hold the state of a compilation
Add a toplevel GirParser object to hold state that is global across a compilation. Currently just holds the include path, but will eventually also keep a cached list of parsed modules. svn path=/trunk/; revision=901
This commit is contained in:
parent
092901f3ab
commit
0faf142d0a
90
girparser.c
90
girparser.c
@ -28,6 +28,12 @@
|
|||||||
#include "girnode.h"
|
#include "girnode.h"
|
||||||
#include "gtypelib.h"
|
#include "gtypelib.h"
|
||||||
|
|
||||||
|
struct _GIrParser
|
||||||
|
{
|
||||||
|
gchar **includes;
|
||||||
|
GList *include_modules; /* All previously parsed include modules */
|
||||||
|
};
|
||||||
|
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
STATE_START,
|
STATE_START,
|
||||||
@ -66,11 +72,11 @@ typedef enum
|
|||||||
typedef struct _ParseContext ParseContext;
|
typedef struct _ParseContext ParseContext;
|
||||||
struct _ParseContext
|
struct _ParseContext
|
||||||
{
|
{
|
||||||
|
GIrParser *parser;
|
||||||
|
|
||||||
ParseState state;
|
ParseState state;
|
||||||
ParseState prev_state;
|
ParseState prev_state;
|
||||||
|
|
||||||
const char * const*includes;
|
|
||||||
|
|
||||||
GList *modules;
|
GList *modules;
|
||||||
GList *include_modules;
|
GList *include_modules;
|
||||||
gboolean prefix_aliases;
|
gboolean prefix_aliases;
|
||||||
@ -107,7 +113,7 @@ static void cleanup (GMarkupParseContext *context,
|
|||||||
GError *error,
|
GError *error,
|
||||||
gpointer user_data);
|
gpointer user_data);
|
||||||
|
|
||||||
static GMarkupParser parser =
|
static GMarkupParser markup_parser =
|
||||||
{
|
{
|
||||||
start_element_handler,
|
start_element_handler,
|
||||||
end_element_handler,
|
end_element_handler,
|
||||||
@ -128,6 +134,34 @@ static const gchar *find_attribute (const gchar *name,
|
|||||||
const gchar **attribute_names,
|
const gchar **attribute_names,
|
||||||
const gchar **attribute_values);
|
const gchar **attribute_values);
|
||||||
|
|
||||||
|
|
||||||
|
GIrParser *
|
||||||
|
g_ir_parser_new (void)
|
||||||
|
{
|
||||||
|
GIrParser *parser = g_slice_new0 (GIrParser);
|
||||||
|
|
||||||
|
return parser;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
g_ir_parser_free (GIrParser *parser)
|
||||||
|
{
|
||||||
|
if (parser->includes)
|
||||||
|
g_strfreev (parser->includes);
|
||||||
|
|
||||||
|
g_slice_free (GIrParser, parser);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
g_ir_parser_set_includes (GIrParser *parser,
|
||||||
|
const gchar *const *includes)
|
||||||
|
{
|
||||||
|
if (parser->includes)
|
||||||
|
g_strfreev (parser->includes);
|
||||||
|
|
||||||
|
parser->includes = g_strdupv ((char **)includes);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
firstpass_start_element_handler (GMarkupParseContext *context,
|
firstpass_start_element_handler (GMarkupParseContext *context,
|
||||||
const gchar *element_name,
|
const gchar *element_name,
|
||||||
@ -187,7 +221,9 @@ static GMarkupParser firstpass_parser =
|
|||||||
};
|
};
|
||||||
|
|
||||||
static char *
|
static char *
|
||||||
locate_gir (const char *name, const char *version, const char * const* extra_paths)
|
locate_gir (GIrParser *parser,
|
||||||
|
const char *name,
|
||||||
|
const char *version)
|
||||||
{
|
{
|
||||||
const gchar *const *datadirs;
|
const gchar *const *datadirs;
|
||||||
const gchar *const *dir;
|
const gchar *const *dir;
|
||||||
@ -198,9 +234,9 @@ locate_gir (const char *name, const char *version, const char * const* extra_pat
|
|||||||
|
|
||||||
girname = g_strdup_printf ("%s-%s.gir", name, version);
|
girname = g_strdup_printf ("%s-%s.gir", name, version);
|
||||||
|
|
||||||
if (extra_paths != NULL)
|
if (parser->includes != NULL)
|
||||||
{
|
{
|
||||||
for (dir = extra_paths; *dir; dir++)
|
for (dir = (const gchar *const *)parser->includes; *dir; dir++)
|
||||||
{
|
{
|
||||||
path = g_build_filename (*dir, girname, NULL);
|
path = g_build_filename (*dir, girname, NULL);
|
||||||
if (g_file_test (path, G_FILE_TEST_EXISTS | G_FILE_TEST_IS_REGULAR))
|
if (g_file_test (path, G_FILE_TEST_EXISTS | G_FILE_TEST_IS_REGULAR))
|
||||||
@ -2183,7 +2219,7 @@ parse_include (GMarkupParseContext *context,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
girpath = locate_gir (name, version, ctx->includes);
|
girpath = locate_gir (ctx->parser, name, version);
|
||||||
|
|
||||||
if (girpath == NULL)
|
if (girpath == NULL)
|
||||||
{
|
{
|
||||||
@ -2204,8 +2240,8 @@ parse_include (GMarkupParseContext *context,
|
|||||||
}
|
}
|
||||||
g_free (girpath);
|
g_free (girpath);
|
||||||
|
|
||||||
|
sub_ctx.parser = ctx->parser;
|
||||||
sub_ctx.state = STATE_START;
|
sub_ctx.state = STATE_START;
|
||||||
sub_ctx.includes = ctx->includes;
|
|
||||||
sub_ctx.prefix_aliases = TRUE;
|
sub_ctx.prefix_aliases = TRUE;
|
||||||
sub_ctx.namespace = name;
|
sub_ctx.namespace = name;
|
||||||
sub_ctx.aliases = ctx->aliases;
|
sub_ctx.aliases = ctx->aliases;
|
||||||
@ -2222,7 +2258,7 @@ parse_include (GMarkupParseContext *context,
|
|||||||
|
|
||||||
g_markup_parse_context_free (context);
|
g_markup_parse_context_free (context);
|
||||||
|
|
||||||
context = g_markup_parse_context_new (&parser, 0, &sub_ctx, NULL);
|
context = g_markup_parse_context_new (&markup_parser, 0, &sub_ctx, NULL);
|
||||||
if (!g_markup_parse_context_parse (context, buffer, length, error))
|
if (!g_markup_parse_context_parse (context, buffer, length, error))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
@ -2958,9 +2994,20 @@ post_filter (GIrModule *module)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* g_ir_parser_parse_string:
|
||||||
|
* @parser: a #GIrParser
|
||||||
|
* @error: return location for a #GError, or %NULL
|
||||||
|
*
|
||||||
|
* Parse a string that holds a complete GIR XML file, and return a list of a
|
||||||
|
* a #GirModule for each <namespace/> element within the file.
|
||||||
|
*
|
||||||
|
* @returns: a newly allocated list of #GIrModule. The modules themselves
|
||||||
|
* are owned by the #GIrParser and will be freed along with the parser.
|
||||||
|
*/
|
||||||
GList *
|
GList *
|
||||||
g_ir_parse_string (const gchar *namespace,
|
g_ir_parser_parse_string (GIrParser *parser,
|
||||||
const gchar *const *includes,
|
const gchar *namespace,
|
||||||
const gchar *buffer,
|
const gchar *buffer,
|
||||||
gssize length,
|
gssize length,
|
||||||
GError **error)
|
GError **error)
|
||||||
@ -2968,8 +3015,8 @@ g_ir_parse_string (const gchar *namespace,
|
|||||||
ParseContext ctx = { 0 };
|
ParseContext ctx = { 0 };
|
||||||
GMarkupParseContext *context;
|
GMarkupParseContext *context;
|
||||||
|
|
||||||
|
ctx.parser = parser;
|
||||||
ctx.state = STATE_START;
|
ctx.state = STATE_START;
|
||||||
ctx.includes = includes;
|
|
||||||
ctx.prefix_aliases = FALSE;
|
ctx.prefix_aliases = FALSE;
|
||||||
ctx.namespace = namespace;
|
ctx.namespace = namespace;
|
||||||
ctx.aliases = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
ctx.aliases = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
||||||
@ -2988,7 +3035,7 @@ g_ir_parse_string (const gchar *namespace,
|
|||||||
|
|
||||||
g_markup_parse_context_free (context);
|
g_markup_parse_context_free (context);
|
||||||
|
|
||||||
context = g_markup_parse_context_new (&parser, 0, &ctx, NULL);
|
context = g_markup_parse_context_new (&markup_parser, 0, &ctx, NULL);
|
||||||
if (!g_markup_parse_context_parse (context, buffer, length, error))
|
if (!g_markup_parse_context_parse (context, buffer, length, error))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
@ -3005,9 +3052,20 @@ g_ir_parse_string (const gchar *namespace,
|
|||||||
return ctx.modules;
|
return ctx.modules;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* g_ir_parser_parse_file:
|
||||||
|
* @parser: a #GIrParser
|
||||||
|
* @error: return location for a #GError, or %NULL
|
||||||
|
*
|
||||||
|
* Parse GIR XML file, and return a list of a a #GirModule for each
|
||||||
|
* <namespace/> element within the file.
|
||||||
|
*
|
||||||
|
* @returns: a newly allocated list of #GIrModule. The modules themselves
|
||||||
|
* are owned by the #GIrParser and will be freed along with the parser.
|
||||||
|
*/
|
||||||
GList *
|
GList *
|
||||||
g_ir_parse_file (const gchar *filename,
|
g_ir_parser_parse_file (GIrParser *parser,
|
||||||
const gchar *const *includes,
|
const gchar *filename,
|
||||||
GError **error)
|
GError **error)
|
||||||
{
|
{
|
||||||
gchar *buffer;
|
gchar *buffer;
|
||||||
@ -3038,7 +3096,7 @@ g_ir_parse_file (const gchar *filename,
|
|||||||
if (!g_file_get_contents (filename, &buffer, &length, error))
|
if (!g_file_get_contents (filename, &buffer, &length, error))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
modules = g_ir_parse_string (namespace, includes, buffer, length, error);
|
modules = g_ir_parser_parse_string (parser, namespace, buffer, length, error);
|
||||||
|
|
||||||
for (iter = modules; iter; iter = iter->next)
|
for (iter = modules; iter; iter = iter->next)
|
||||||
{
|
{
|
||||||
|
15
girparser.h
15
girparser.h
@ -25,17 +25,22 @@
|
|||||||
|
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
typedef struct _GIrParser GIrParser;
|
||||||
|
|
||||||
GList *g_ir_parse_string (const gchar *namespace,
|
GIrParser *g_ir_parser_new (void);
|
||||||
const gchar *const *includes,
|
void g_ir_parser_free (GIrParser *parser);
|
||||||
|
void g_ir_parser_set_includes (GIrParser *parser,
|
||||||
|
const gchar *const *includes);
|
||||||
|
|
||||||
|
GList *g_ir_parser_parse_string (GIrParser *parser,
|
||||||
|
const gchar *namespace,
|
||||||
const gchar *buffer,
|
const gchar *buffer,
|
||||||
gssize length,
|
gssize length,
|
||||||
GError **error);
|
GError **error);
|
||||||
GList *g_ir_parse_file (const gchar *filename,
|
GList *g_ir_parser_parse_file (GIrParser *parser,
|
||||||
const gchar *const *includes,
|
const gchar *filename,
|
||||||
GError **error);
|
GError **error);
|
||||||
|
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
#endif /* __G_GIR_PARSER_H__ */
|
#endif /* __G_GIR_PARSER_H__ */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user