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:
Owen Taylor 2008-11-12 17:16:42 +00:00
parent 092901f3ab
commit 0faf142d0a
2 changed files with 92 additions and 29 deletions

View File

@ -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)
{ {

View File

@ -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__ */