diff options
Diffstat (limited to 'camel/providers')
-rw-r--r-- | camel/providers/imap/camel-imap-folder.c | 59 | ||||
-rw-r--r-- | camel/providers/imap/camel-imap-store.c | 45 | ||||
-rw-r--r-- | camel/providers/imap/camel-imap-store.h | 22 |
3 files changed, 123 insertions, 3 deletions
diff --git a/camel/providers/imap/camel-imap-folder.c b/camel/providers/imap/camel-imap-folder.c index 1e74461ba0..0e4c0dd44e 100644 --- a/camel/providers/imap/camel-imap-folder.c +++ b/camel/providers/imap/camel-imap-folder.c @@ -1797,6 +1797,44 @@ imap_update_summary (CamelFolder *folder, int exists, } } +struct _filter_msg { + CamelImapMsg msg; + + GPtrArray *recents; + CamelFolder *folder; + CamelFilterDriver *driver; +}; + +static void +filter_proc(CamelImapStore *store, CamelImapMsg *mm) +{ + struct _filter_msg *msg = (struct _filter_msg *)mm; + + printf("executing filtering %d messages folder %p\n", msg->recents->len, msg->folder); + + /* what about exceptions? */ + camel_filter_driver_filter_folder(msg->driver, msg->folder, NULL, msg->recents, FALSE, NULL); +} + +static void +filter_free(CamelImapStore *store, CamelImapMsg *mm) +{ + struct _filter_msg *msg = (struct _filter_msg *)mm; + int i; + + printf("freeing filtering %d messages folder %p\n", msg->recents->len, msg->folder); + + camel_object_unref((CamelObject *)msg->driver); + + camel_folder_thaw(msg->folder); + camel_object_unref((CamelObject *)msg->folder); + + for (i=0;i<msg->recents->len;i++) + g_free(msg->recents->pdata[i]); + + g_ptr_array_free(msg->recents, TRUE); +} + /* Called with the store's command_lock locked */ void camel_imap_folder_changed (CamelFolder *folder, int exists, @@ -1853,11 +1891,26 @@ camel_imap_folder_changed (CamelFolder *folder, int exists, driver = camel_session_get_filter_driver (CAMEL_SERVICE (folder->parent_store)->session, "incoming", ex); if (driver) { +#ifdef ENABLE_THREADS + int i; + struct _filter_msg *msg = (struct _filter_msg *)camel_imap_msg_new(filter_proc, filter_free, sizeof(*msg)); + + msg->recents = g_ptr_array_new(); + for (i=0;i<recents->len;i++) + g_ptr_array_add(msg->recents, g_strdup(recents->pdata[i])); + + camel_object_ref((CamelObject *)folder); + msg->folder = folder; + msg->driver = driver; + printf("queueing filtering %d messages folder %p\n", msg->recents->len, folder); + camel_imap_msg_queue((CamelImapStore *)folder->parent_store, (CamelImapMsg *)msg); +#else camel_filter_driver_filter_folder (driver, folder, NULL, recents, FALSE, ex); + camel_folder_thaw (folder); camel_object_unref (CAMEL_OBJECT (driver)); - } - - camel_folder_thaw (folder); +#endif + } else + camel_folder_thaw (folder); } else { if (camel_folder_change_info_changed (changes)) camel_object_trigger_event (CAMEL_OBJECT (folder), "folder_changed", changes); diff --git a/camel/providers/imap/camel-imap-store.c b/camel/providers/imap/camel-imap-store.c index 10794f1270..ee5c0800fb 100644 --- a/camel/providers/imap/camel-imap-store.c +++ b/camel/providers/imap/camel-imap-store.c @@ -158,10 +158,52 @@ camel_imap_store_finalize (CamelObject *object) #ifdef ENABLE_THREADS e_mutex_destroy (imap_store->priv->command_lock); + e_thread_destroy(imap_store->async_thread); #endif g_free (imap_store->priv); } +#ifdef ENABLE_THREADS +static void async_destroy(EThread *et, EMsg *em, void *data) +{ + CamelImapStore *imap_store = data; + CamelImapMsg *msg = (CamelImapMsg *)em; + + if (msg->free) + msg->free(imap_store, msg); + + g_free(msg); +} + +static void async_received(EThread *et, EMsg *em, void *data) +{ + CamelImapStore *imap_store = data; + CamelImapMsg *msg = (CamelImapMsg *)em; + + if (msg->receive) + msg->receive(imap_store, msg); +} + +CamelImapMsg *camel_imap_msg_new(void (*receive)(CamelImapStore *store, struct _CamelImapMsg *m), + void (*free)(CamelImapStore *store, struct _CamelImapMsg *m), + size_t size) +{ + CamelImapMsg *msg; + + g_assert(size >= sizeof(*msg)); + + msg = g_malloc0(size); + msg->receive = receive; + msg->free = free; +} + +void camel_imap_msg_queue(CamelImapStore *store, CamelImapMsg *msg) +{ + e_thread_put(store->async_thread, (EMsg *)msg); +} + +#endif + static void camel_imap_store_init (gpointer object, gpointer klass) { @@ -183,6 +225,9 @@ camel_imap_store_init (gpointer object, gpointer klass) imap_store->priv = g_malloc0 (sizeof (*imap_store->priv)); #ifdef ENABLE_THREADS imap_store->priv->command_lock = e_mutex_new (E_MUTEX_REC); + imap_store->async_thread = e_thread_new(E_THREAD_QUEUE); + e_thread_set_msg_destroy(imap_store->async_thread, async_destroy, imap_store); + e_thread_set_msg_received(imap_store->async_thread, async_received, imap_store); #endif } diff --git a/camel/providers/imap/camel-imap-store.h b/camel/providers/imap/camel-imap-store.h index c467d6066b..458c1aa525 100644 --- a/camel/providers/imap/camel-imap-store.h +++ b/camel/providers/imap/camel-imap-store.h @@ -35,6 +35,25 @@ extern "C" { #include "camel-imap-types.h" #include "camel-disco-store.h" +#ifdef ENABLE_THREADS +#include "e-util/e-msgport.h" + +typedef struct _CamelImapMsg CamelImapMsg; + +struct _CamelImapMsg { + EMsg msg; + + void (*receive)(CamelImapStore *store, struct _CamelImapMsg *m); + void (*free)(CamelImapStore *store, struct _CamelImapMsg *m); +}; + +CamelImapMsg *camel_imap_msg_new(void (*receive)(CamelImapStore *store, struct _CamelImapMsg *m), + void (*free)(CamelImapStore *store, struct _CamelImapMsg *m), + size_t size); +void camel_imap_msg_queue(CamelImapStore *store, CamelImapMsg *msg); + +#endif + #define CAMEL_IMAP_STORE_TYPE (camel_imap_store_get_type ()) #define CAMEL_IMAP_STORE(obj) (CAMEL_CHECK_CAST((obj), CAMEL_IMAP_STORE_TYPE, CamelImapStore)) #define CAMEL_IMAP_STORE_CLASS(k) (CAMEL_CHECK_CLASS_CAST ((k), CAMEL_IMAP_STORE_TYPE, CamelImapStoreClass)) @@ -73,6 +92,9 @@ struct _CamelImapStore { guint32 capabilities, parameters; char *namespace, dir_sep, *base_url, *storage_path; GHashTable *authtypes, *subscribed_folders; +#ifdef ENABLE_THREADS + EThread *async_thread; +#endif }; |