diff options
author | James Booth <boothj5@gmail.com> | 2016-08-17 23:21:48 +0100 |
---|---|---|
committer | James Booth <boothj5@gmail.com> | 2016-08-17 23:21:48 +0100 |
commit | 9945246a648998e374c0c3a5ac56cdffc63ef13a (patch) | |
tree | 551e54c41787cdb8fe2f1bf3361b6dcd2c484c1c /src/plugins | |
parent | 8b6549b36c233fc9ac08dae73bc90e9672020efb (diff) | |
download | profani-tty-9945246a648998e374c0c3a5ac56cdffc63ef13a.tar.gz |
Use hashtables for plugin disco features
Diffstat (limited to 'src/plugins')
-rw-r--r-- | src/plugins/disco.c | 92 | ||||
-rw-r--r-- | src/plugins/disco.h | 2 |
2 files changed, 66 insertions, 28 deletions
diff --git a/src/plugins/disco.c b/src/plugins/disco.c index bf42d4c0..320b7fa7 100644 --- a/src/plugins/disco.c +++ b/src/plugins/disco.c @@ -37,76 +37,112 @@ #include <glib.h> +// features to reference count map +static GHashTable *features = NULL; + +// plugin to feature map static GHashTable *plugin_to_features = NULL; static void -_free_features(GList *features) +_free_features(GHashTable *features) { - g_list_free_full(features, free); + g_hash_table_destroy(features); } void -disco_add_feature(const char *plugin_name, char* feature) +disco_add_feature(const char *plugin_name, char *feature) { if (feature == NULL || plugin_name == NULL) { return; } + if (!features) { + features = g_hash_table_new_full(g_str_hash, g_str_equal, free, NULL); + } if (!plugin_to_features) { plugin_to_features = g_hash_table_new_full(g_str_hash, g_str_equal, free, (GDestroyNotify)_free_features); } - GList *features = g_hash_table_lookup(plugin_to_features, plugin_name); - if (!features) { - features = g_list_append(features, strdup(feature)); - g_hash_table_insert(plugin_to_features, strdup(plugin_name), features); + GHashTable *plugin_features = g_hash_table_lookup(plugin_to_features, plugin_name); + gboolean added = FALSE; + if (plugin_features == NULL) { + plugin_features = g_hash_table_new_full(g_str_hash, g_str_equal, free, NULL); + g_hash_table_add(plugin_features, strdup(feature)); + g_hash_table_insert(plugin_to_features, strdup(plugin_name), plugin_features); + added = TRUE; + } else if (!g_hash_table_contains(plugin_features, feature)) { + g_hash_table_add(plugin_features, strdup(feature)); + added = TRUE; + } + + if (added == FALSE) { + return; + } + + if (!g_hash_table_contains(features, feature)) { + g_hash_table_insert(features, strdup(feature), GINT_TO_POINTER(1)); } else { - features = g_list_append(features, strdup(feature)); + void *refcountp = g_hash_table_lookup(features, feature); + int refcount = GPOINTER_TO_INT(refcountp); + refcount++; + g_hash_table_replace(features, strdup(feature), GINT_TO_POINTER(refcount)); } } void disco_remove_features(const char *plugin_name) { + if (!features) { + return; + } if (!plugin_to_features) { return; } - if (!g_hash_table_contains(plugin_to_features, plugin_name)) { + GHashTable *plugin_features_set = g_hash_table_lookup(plugin_to_features, plugin_name); + if (!plugin_features_set) { return; } - g_hash_table_remove(plugin_to_features, plugin_name); + GList *plugin_feature_list = g_hash_table_get_keys(plugin_features_set); + GList *curr = plugin_feature_list; + while (curr) { + char *feature = curr->data; + if (g_hash_table_contains(features, feature)) { + void *refcountp = g_hash_table_lookup(features, feature); + int refcount = GPOINTER_TO_INT(refcountp); + if (refcount == 1) { + g_hash_table_remove(features, feature); + } else { + refcount--; + g_hash_table_replace(features, strdup(feature), GINT_TO_POINTER(refcount)); + } + } + + curr = g_list_next(curr); + } + g_list_free(plugin_feature_list); + } GList* disco_get_features(void) { - GList *result = NULL; - if (!plugin_to_features) { - return result; + if (features == NULL) { + return NULL; } - GList *lists = g_hash_table_get_values(plugin_to_features); - GList *curr_list = lists; - while (curr_list) { - GList *features = curr_list->data; - GList *curr = features; - while (curr) { - result = g_list_append(result, curr->data); - curr = g_list_next(curr); - } - curr_list = g_list_next(curr_list); - } - - g_list_free(lists); - - return result; + return g_hash_table_get_keys(features); } void disco_close(void) { + if (features) { + g_hash_table_destroy(features); + features = NULL; + } + if (plugin_to_features) { g_hash_table_destroy(plugin_to_features); plugin_to_features = NULL; diff --git a/src/plugins/disco.h b/src/plugins/disco.h index f16e2642..d08b6537 100644 --- a/src/plugins/disco.h +++ b/src/plugins/disco.h @@ -35,6 +35,8 @@ #ifndef PLUGINS_DISCO_H #define PLUGINS_DISCO_H +#include <glib.h> + void disco_add_feature(const char* plugin_name, char *feature); void disco_remove_features(const char *plugin_name); GList* disco_get_features(void); |