Provide functions to retrieve rules from Grammar
[cascardo/grammar.git] / grammar.c
index b73cd39..9795468 100644 (file)
--- a/grammar.c
+++ b/grammar.c
@@ -14,6 +14,23 @@ symbol_t* symbol_new (gboolean terminal, gint value)
   return symbol;
 }
 
+guint symbol_hash (gconstpointer data)
+{
+  symbol_t* symbol;
+  symbol = (symbol_t*) data;
+  return g_direct_hash ((gpointer)symbol->value);
+}
+
+gboolean symbol_equal (gconstpointer data1, gconstpointer data2)
+{
+  symbol_t* symbol1;
+  symbol_t* symbol2;
+  symbol1 = (symbol_t*) data1;
+  symbol2 = (symbol_t*) data2;
+  return symbol1->value == symbol2->value &&
+    symbol1->terminal == symbol2->terminal;
+}
+
 rule_t* rule_new ()
 {
   rule_t* rule;
@@ -27,28 +44,34 @@ void rule_append (rule_t* rule, symbol_t* right)
   rule->right = g_list_append (rule->right, right);
 }
 
-guint symbol_hash (gconstpointer data)
+void rule_delete (rule_t* rule)
 {
-  symbol_t* symbol;
-  symbol = (symbol_t*) data;
-  return g_direct_hash ((gpointer)symbol->value);
+  GList* l;
+  for (l = g_list_first (rule->right); l != NULL; l = g_list_next (l))
+    {
+      g_free (l->data);
+    }
+  g_list_free (rule->right);
+  g_free (rule);
 }
 
-gboolean symbol_equal (gconstpointer data1, gconstpointer data2)
+void rules_delete (GList** list)
 {
-  symbol_t* symbol1;
-  symbol_t* symbol2;
-  symbol1 = (symbol_t*) data1;
-  symbol2 = (symbol_t*) data2;
-  return symbol1->value == symbol2->value &&
-    symbol1->terminal == symbol2->terminal;
+  GList* l;
+  for (l = g_list_first (*list); l != NULL; l = g_list_next (l))
+    {
+      rule_delete (l->data);
+    }
+  g_list_free (*list);
+  g_free (list);
 }
 
 static void grammar_init (GTypeInstance* instance, gpointer g_class)
 {
   Grammar* self = GRAMMAR(instance);
   self->grammar = g_hash_table_new_full (symbol_hash, symbol_equal,
-                                        g_free, g_free);
+                                        g_free,
+                                        (GDestroyNotify) rules_delete);
 }
 
 static void grammar_finalize (GObject* obj)
@@ -101,6 +124,7 @@ rule_t* grammar_rule_new (Grammar* grammar, symbol_t* left)
                                     left, NULL, (gpointer*)&l))
     {
       l = g_malloc (sizeof (GList**));
+      *l = NULL;
       g_hash_table_insert (grammar->grammar, left, l);
     }
 
@@ -116,3 +140,19 @@ void grammar_rule_append (rule_t* rule, symbol_t* right)
 {
   rule_append (rule, right);
 }
+
+GList* grammar_get_rules (Grammar* grammar, symbol_t* left)
+{
+  GList** l;
+  if (!g_hash_table_lookup_extended (grammar->grammar,
+                                    left, NULL, (gpointer*)&l))
+    {
+      return NULL;
+    }
+  return g_list_first (*l);
+}
+
+GList* grammar_get_rule (rule_t* rule)
+{
+  return rule->right;
+}