aboutsummaryrefslogtreecommitdiffstats
path: root/camel/camel-object.c
diff options
context:
space:
mode:
Diffstat (limited to 'camel/camel-object.c')
-rw-r--r--camel/camel-object.c231
1 files changed, 204 insertions, 27 deletions
diff --git a/camel/camel-object.c b/camel/camel-object.c
index ed8c255eaf..fde35f3be1 100644
--- a/camel/camel-object.c
+++ b/camel/camel-object.c
@@ -30,6 +30,13 @@
#include <string.h>
#include "camel-object.h"
+/* FIXME: Make the code so this isn't necessary, its just the hook locking thats the problem */
+#ifndef ENABLE_THREADS
+#error "Threads must be enabled to compile this version of camel"
+#endif
+
+#include <pthread.h>
+
/* I just mashed the keyboard for these... */
#define CAMEL_OBJECT_MAGIC_VALUE 0x77A344EF
#define CAMEL_OBJECT_CLASS_MAGIC_VALUE 0xEE26A990
@@ -63,12 +70,27 @@ typedef struct _CamelTypeInfo
}
CamelTypeInfo;
+/* A 'locked' hooklist, that is only allocated on demand */
+typedef struct _CamelHookList {
+ pthread_mutex_t lock;
+
+ struct _CamelHookPair *list;
+} CamelHookList;
+
+/* a 'hook pair', actually a hook tuple, we just store all hooked events in the same list,
+ and just comapre as we go, rather than storing separate lists for each hook type
+
+ the name field just points directly to the key field in the class's preplist hashtable.
+ This way we can just use a direct pointer compare when scanning it, and also saves
+ copying the string */
typedef struct _CamelHookPair
{
+ struct _CamelHookPair *next; /* next MUST be the first member */
+
+ const char *name; /* points to the key field in the classes preplist, static memory */
CamelObjectEventHookFunc func;
- gpointer user_data;
-}
-CamelHookPair;
+ void *data;
+} CamelHookPair;
/* ************************************************************************ */
@@ -84,6 +106,8 @@ static gboolean shared_is_of_type (CamelObjectShared * sh, CamelType ctype,
gboolean is_obj);
static void make_global_classfuncs (CamelTypeInfo * type_info);
+static void camel_object_free_hooks(CamelObject *o);
+
/* ************************************************************************ */
G_LOCK_DEFINE_STATIC (type_system);
@@ -322,7 +346,7 @@ obj_init (CamelObject * obj)
{
obj->s.magic = CAMEL_OBJECT_MAGIC_VALUE;
obj->ref_count = 1;
- obj->event_to_hooklist = NULL;
+ obj->hooks = NULL;
obj->in_event = 0;
obj->destroying = 0;
}
@@ -336,11 +360,7 @@ obj_finalize (CamelObject * obj)
obj->s.magic = CAMEL_OBJECT_FINALIZED_VALUE;
- if (obj->event_to_hooklist) {
- g_hash_table_foreach (obj->event_to_hooklist, (GHFunc) g_free, NULL);
- g_hash_table_destroy (obj->event_to_hooklist);
- obj->event_to_hooklist = NULL;
- }
+ camel_object_free_hooks(obj);
}
static void
@@ -359,6 +379,7 @@ obj_class_finalize (CamelObjectClass * class)
class->s.magic = CAMEL_OBJECT_CLASS_FINALIZED_VALUE;
if (class->event_to_preplist) {
+ /* FIXME: This leaks the preplist slist entries */
g_hash_table_foreach (class->event_to_preplist,
(GHFunc) g_free, NULL);
g_hash_table_destroy (class->event_to_preplist);
@@ -411,6 +432,7 @@ camel_object_new (CamelType type)
type_info->free_instances =
g_list_remove_link (type_info->free_instances, first);
g_list_free_1 (first);
+ memset (instance, 0, type_info->instance_size);
} else {
instance = g_mem_chunk_alloc0 (type_info->instance_chunk);
}
@@ -557,10 +579,11 @@ camel_object_unref (CamelObject * obj)
/* A little bit of cleaning up.
* Don't erase the type, so we can peek at it if a finalized object
- * is check_cast'ed somewhere.
+ * is check_cast'ed somewhere. Fill it with gunk to help detect
+ * other invalid ref's of it.
*/
- memset (obj, 0, type_info->instance_size);
+ memset (obj, 0xEB, type_info->instance_size);
obj->s.type = type_info->self;
obj->s.magic = CAMEL_OBJECT_FINALIZED_VALUE;
@@ -682,18 +705,98 @@ camel_object_class_declare_event (CamelObjectClass * class,
g_hash_table_insert (class->event_to_preplist, g_strdup (name), prep);
}
+/* free hook data */
+static void camel_object_free_hooks(CamelObject *o)
+{
+ CamelHookPair *pair, *next;
+
+ if (o->hooks) {
+ pair = o->hooks->list;
+ while (pair) {
+ next = pair->next;
+ g_free(pair);
+ pair = next;
+ }
+ g_free(o->hooks);
+ o->hooks = NULL;
+ }
+}
+
+/* return (allocate if required) the object's hook list, locking at the same time */
+static CamelHookList *camel_object_get_hooks(CamelObject *o)
+{
+#ifdef ENABLE_THREADS
+ static pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
+#endif
+ CamelHookList *hooks;
+
+ /* if we have it, we dont have to do any other locking,
+ otherwise use a global lock to setup the object's hook data */
+#ifdef ENABLE_THREADS
+ if (o->hooks == NULL) {
+ pthread_mutex_lock(&lock);
+#endif
+ if (o->hooks == NULL) {
+ hooks = g_malloc(sizeof(*o->hooks));
+#ifdef ENABLE_THREADS
+ pthread_mutex_init(&hooks->lock, NULL);
+#endif
+ hooks->list = NULL;
+ o->hooks = hooks;
+ }
+#ifdef ENABLE_THREADS
+ pthread_mutex_unlock(&lock);
+ }
+#endif
+
+#ifdef ENABLE_THREADS
+ pthread_mutex_lock(&o->hooks->lock);
+#endif
+ return o->hooks;
+}
+
+/* unlock object hooks' list */
+#ifdef ENABLE_THREADS
+#define camel_object_unget_hooks(o) (pthread_mutex_unlock(&(CAMEL_OBJECT(o)->hooks->lock)))
+#else
+#define camel_object_unget_hooks(o)
+#endif
+
void
-camel_object_hook_event (CamelObject * obj, const gchar * name,
- CamelObjectEventHookFunc hook, gpointer user_data)
+camel_object_hook_event (CamelObject * obj, const char * name,
+ CamelObjectEventHookFunc func, void *data)
{
- GSList *hooklist;
CamelHookPair *pair;
- gpointer old_name, old_hooklist;
+ const char *prepname;
+ CamelObjectEventPrepFunc prep;
+ CamelHookList *hooks;
g_return_if_fail (CAMEL_IS_OBJECT (obj));
- g_return_if_fail (name);
- g_return_if_fail (hook);
+ g_return_if_fail (name != NULL);
+ g_return_if_fail (func != NULL);
+
+ /* first, does this event exist? */
+ if (obj->classfuncs->event_to_preplist == NULL
+ || !g_hash_table_lookup_extended(obj->classfuncs->event_to_preplist, name,
+ (void **)&prepname, (void **)&prep)) {
+ g_warning("camel_object_hook_event: trying to hook event `%s' in class `%s' with no defined events.",
+ name, camel_type_to_name (obj->s.type));
+ return;
+ }
+
+ /* setup hook pair */
+ pair = g_malloc(sizeof(*pair));
+ pair->name = prepname; /* effectively static! */
+ pair->func = func;
+ pair->data = data;
+ /* get the hook list object, locked, link in new event hook, unlock */
+ hooks = camel_object_get_hooks(obj);
+ pair->next = hooks->list;
+ hooks->list = pair;
+ camel_object_unget_hooks(obj);
+
+#if 0
if (obj->event_to_hooklist == NULL)
obj->event_to_hooklist =
g_hash_table_new (g_str_hash, g_str_equal);
@@ -712,19 +815,61 @@ camel_object_hook_event (CamelObject * obj, const gchar * name,
g_hash_table_insert (obj->event_to_hooklist, g_strdup (name),
hooklist);
}
+#endif
}
void
-camel_object_unhook_event (CamelObject * obj, const gchar * name,
- CamelObjectEventHookFunc hook, gpointer user_data)
+camel_object_unhook_event (CamelObject * obj, const char * name,
+ CamelObjectEventHookFunc func, void *data)
{
+#if 0
GSList *hooklist;
GSList *head;
+#endif
+ char *prepname;
+ CamelObjectEventPrepFunc prep;
+ CamelHookList *hooks;
+ CamelHookPair *pair, *parent;
g_return_if_fail (CAMEL_IS_OBJECT (obj));
- g_return_if_fail (name);
- g_return_if_fail (hook);
+ g_return_if_fail (name != NULL);
+ g_return_if_fail (func != NULL);
+ if (obj->hooks == NULL) {
+ g_warning("camel_object_unhook_event: trying to unhook `%s` from an instance of `%s' with no hooks",
+ name, camel_type_to_name(obj->s.type));
+ return;
+ }
+
+ /* get event name static pointer */
+ if (obj->classfuncs->event_to_preplist == NULL
+ || !g_hash_table_lookup_extended(obj->classfuncs->event_to_preplist, name,
+ (void **)&prepname, (void **)&prep)) {
+ g_warning("camel_object_hook_event: trying to hook event `%s' in class `%s' with no defined events.",
+ name, camel_type_to_name (obj->s.type));
+ return;
+ }
+
+ /* scan hooks for this event, remove it */
+ hooks = camel_object_get_hooks(obj);
+ parent = (CamelHookPair *)&hooks->list;
+ pair = parent->next;
+ while (pair) {
+ if (pair->name == prepname && pair->func == func && pair->data == data) {
+ parent->next = pair->next;
+ g_free(pair);
+ camel_object_unget_hooks(obj);
+ return;
+ }
+ parent = pair;
+ pair = pair->next;
+ }
+ camel_object_unget_hooks(obj);
+
+ g_warning("camel_object_unhook_event: cannot find hook/data pair %p/%p in an instance of `%s' attached to `%s'",
+ func, data, camel_type_to_name (obj->s.type), name);
+
+#if 0
if (obj->event_to_hooklist == NULL) {
g_warning
("camel_object_unhook_event: trying to unhook `%s' from an instance "
@@ -764,23 +909,54 @@ camel_object_unhook_event (CamelObject * obj, const gchar * name,
("camel_object_unhook_event: cannot find hook/data pair %p/%p in an "
"instance of `%s' attached to `%s'", hook, user_data,
camel_type_to_name (obj->s.type), name);
+#endif
}
void
-camel_object_trigger_event (CamelObject * obj, const gchar * name,
- gpointer event_data)
+camel_object_trigger_event (CamelObject * obj, const char * name, void *event_data)
{
- GSList *hooklist;
CamelHookPair *pair;
CamelObjectEventPrepFunc prep;
+ const char *prepname;
+ CamelHookList *hooks;
g_return_if_fail (CAMEL_IS_OBJECT (obj));
g_return_if_fail (name);
+ /* no hooks, dont bother going anywhere */
+ if (obj->hooks == NULL)
+ return;
+
+ /* get event name static pointer/prep func */
+ if (obj->classfuncs->event_to_preplist == NULL
+ || !g_hash_table_lookup_extended(obj->classfuncs->event_to_preplist, name,
+ (void **)&prepname, (void **)&prep)) {
+ g_warning("camel_object_hook_event: trying to hook event `%s' in class `%s' with no defined events.",
+ name, camel_type_to_name (obj->s.type));
+ return;
+ }
+
+ /* lock the object for hook emission */
+ camel_object_ref(obj);
+ hooks = camel_object_get_hooks(obj);
+
+ if (prep == NULL_PREP_VALUE || prep(obj, event_data)) {
+ pair = hooks->list;
+ while (pair) {
+ if (pair->name == prepname)
+ (pair->func) (obj, event_data, pair->data);
+
+ pair = pair->next;
+ }
+ }
+
+ camel_object_unget_hooks(obj);
+ camel_object_unref(obj);
+
+#if 0
if (obj->in_event) {
- g_warning
- ("camel_object_trigger_event: trying to trigger `%s' in class "
- "`%s' while already triggering another event", name,
+ g_warning("camel_object_trigger_event: trying to trigger `%s' in class "
+ "`%s' while already triggering another event", name,
camel_type_to_name (obj->s.type));
return;
}
@@ -827,6 +1003,7 @@ camel_object_trigger_event (CamelObject * obj, const gchar * name,
obj->in_event = 0;
camel_object_unref (obj);
+#endif
}
/* ** Static helpers ****************************************************** */