/* * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) version 3. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with the program; if not, see * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) */ #include #include #include #include #include #include #include #include #include "shell/e-shell.h" #include "e-util/e-alert-activity.h" #include "e-util/e-alert-dialog.h" #include "mail-mt.h" /*#define MALLOC_CHECK*/ #define d(x) /* XXX This is a dirty hack on a dirty hack. We really need * to rework or get rid of the functions that use this. */ const gchar *shell_builtin_backend = "mail"; /* background operation status stuff */ struct _MailMsgPrivate { EActivity *activity; GtkWidget *error; gboolean cancelable; }; static guint mail_msg_seq; /* sequence number of each message */ /* Table of active messages. Must hold mail_msg_lock to access. */ static GHashTable *mail_msg_active_table; static GMutex *mail_msg_lock; static GCond *mail_msg_cond; MailAsyncEvent *mail_async_event; static void mail_msg_cancelled (CamelOperation *operation, gpointer user_data) { mail_msg_cancel (GPOINTER_TO_UINT (user_data)); } static gboolean mail_msg_submit (EActivity *activity) { EShell *shell; EShellBackend *shell_backend; shell = e_shell_get_default (); shell_backend = e_shell_get_backend_by_name ( shell, shell_builtin_backend); e_shell_backend_add_activity (shell_backend, activity); return FALSE; } gpointer mail_msg_new (MailMsgInfo *info) { MailMsg *msg; g_mutex_lock (mail_msg_lock); msg = g_slice_alloc0 (info->size); msg->info = info; msg->ref_count = 1; msg->seq = mail_msg_seq++; msg->cancel = camel_operation_new (); msg->priv = g_slice_new0 (MailMsgPrivate); msg->priv->activity = e_activity_new (); msg->priv->cancelable = TRUE; e_activity_set_percent (msg->priv->activity, 0.0); e_activity_set_cancellable ( msg->priv->activity, G_CANCELLABLE (msg->cancel)); g_signal_connect ( msg->cancel, "cancelled", G_CALLBACK (mail_msg_cancelled), GINT_TO_POINTER (msg->seq)); g_hash_table_insert ( mail_msg_active_table, GINT_TO_POINTER (msg->seq), msg); d(printf("New message %p\n", msg)); g_mutex_unlock (mail_msg_lock); return msg; } #ifdef MALLOC_CHECK #include static void checkmem (gpointer p) { if (p) { gint status = mprobe (p); switch (status) { case MCHECK_HEAD: printf("Memory underrun at %p\n", p); abort (); case MCHECK_TAIL: printf("Memory overrun at %p\n", p); abort (); case MCHECK_FREE: printf("Double free %p\n", p); abort (); } } } #endif static gboolean mail_msg_free (MailMsg *mail_msg) { EShell *shell; EShellBackend *shell_backend; /* This is an idle callback. */ shell = e_shell_get_default (); shell_backend = e_shell_get_backend_by_name ( shell, shell_builtin_backend); if (mail_msg->priv->activity != NULL) { e_activity_complete (mail_msg->priv->activity); g_object_unref (mail_msg->priv->activity); } if (mail_msg->cancel != NULL) g_object_unref (mail_msg->cancel); if (mail_msg->error != NULL) g_error_free (mail_msg->error); if (mail_msg->priv->error != NULL) { EActivity *activity; GtkWidget *widget; widget = mail_msg->priv->error; activity = e_alert_activity_new_warning (widget); e_shell_backend_add_activity (shell_backend, activity); g_object_unref (activity); } g_slice_free (MailMsgPrivate, mail_msg->priv); g_slice_free1 (mail_msg->info->size, mail_msg); return FALSE; } gpointer mail_msg_ref (gpointer msg) { MailMsg *mail_msg = msg; g_return_val_if_fail (mail_msg != NULL, msg); g_return_val_if_fail (mail_msg->ref_count > 0, msg); g_atomic_int_add (&mail_msg->ref_count, 1); return msg; } void mail_msg_unref (gpointer msg) { MailMsg *mail_msg = msg; g_return_if_fail (mail_msg != NULL); g_return_if_fail (mail_msg->ref_count > 0); if (g_atomic_int_exchange_and_add (&mail_msg->ref_count, -1) > 1) return; #ifdef MALLOC_CHECK checkmem (mail_msg); checkmem (mail_msg->cancel); checkmem (mail_msg->priv); #endif d(printf("Free message %p\n", msg)); if (mail_msg->info->free) mail_msg->info->free (mail_msg); g_mutex_lock (mail_msg_lock); g_hash_table_remove ( mail_msg_active_table, GINT_TO_POINTER (mail_msg->seq)); g_cond_broadcast (mail_msg_cond); g_mutex_unlock (mail_msg_lock); /* Destroy the message from an idle callback * so we know we're in the main loop thread. */ g_idle_add ((GSourceFunc) mail_msg_free, mail_msg); } /* hash table of ops->dialogue of active errors */ static GHashTable *active_errors = NULL; static void error_destroy (GtkObject *o, gpointer data) { g_hash_table_remove (active_errors, data); } static void error_response (GtkObject *o, gint button, gpointer data) { gtk_widget_destroy ((GtkWidget *)o); } void mail_msg_check_error (gpointer msg) { GtkWindow *parent; MailMsg *m = msg; gchar *what; GtkDialog *gd; #ifdef MALLOC_CHECK checkmem (m); checkmem (m->cancel); checkmem (m->priv); #endif if (m->error == NULL || g_error_matches (m->error, G_IO_ERROR, G_IO_ERROR_CANCELLED) || g_error_matches (m->error, CAMEL_FOLDER_ERROR, CAMEL_FOLDER_ERROR_INVALID_UID) || (m->cancel && camel_operation_cancel_check (m->cancel))) return; if (active_errors == NULL) active_errors = g_hash_table_new (NULL, NULL); /* check to see if we have dialogue already running for this operation */ /* we key on the operation pointer, which is at least accurate enough for the operation type, although it could be on a different object. */ if (g_hash_table_lookup (active_errors, m->info)) { g_message ( "Error occurred while existing dialogue active:\n%s", m->error->message); return; } parent = e_shell_get_active_window (NULL); if (m->info->desc && (what = m->info->desc (m))) { gd = (GtkDialog *) e_alert_dialog_new_for_args ( parent, "mail:async-error", what, m->error->message, NULL); g_free (what); } else gd = (GtkDialog *) e_alert_dialog_new_for_args ( parent, "mail:async-error-nodescribe", m->error->message, NULL); g_hash_table_insert (active_errors, m->info, gd); g_signal_connect(gd, "response", G_CALLBACK(error_response), m->info); g_signal_connect(gd, "destroy", G_CALLBACK(error_destroy), m->info); if (m->priv->cancelable) m->priv->error = (GtkWidget *) gd; else gtk_widget_show ((GtkWidget *)gd); } void mail_msg_cancel (guint msgid) { MailMsg *m; g_mutex_lock (mail_msg_lock); m = g_hash_table_lookup (mail_msg_active_table, GINT_TO_POINTER (msgid)); if (m != NULL && !camel_operation_cancel_check (m->cancel)) camel_operation_cancel (m->cancel); g_mutex_unlock (mail_msg_lock); } /* waits for a message to be finished processing (freed) the messageid is from MailMsg->seq */ void mail_msg_wait (guint msgid) { MailMsg *m; if (mail_in_main_thread ()) { g_mutex_lock (mail_msg_lock); m = g_hash_table_lookup (mail_msg_active_table, GINT_TO_POINTER (msgid)); while (m) { g_mutex_unlock (mail_msg_lock); gtk_main_iteration (); g_mutex_lock (mail_msg_lock); m = g_hash_table_lookup (mail_msg_active_table, GINT_TO_POINTER (msgid)); } g_mutex_unlock (mail_msg_lock); } else { g_mutex_lock (mail_msg_lock); m = g_hash_table_lookup (mail_msg_active_table, GINT_TO_POINTER (msgid)); while (m) { g_cond_wait (mail_msg_cond, mail_msg_lock); m = g_hash_table_lookup (mail_msg_active_table, GINT_TO_POINTER (msgid)); } g_mutex_unlock (mail_msg_lock); } } gint mail_msg_active (guint msgid) { gint active; g_mutex_lock (mail_msg_lock); if (msgid == (guint)-1) active = g_hash_table_size (mail_msg_active_table) > 0; else active = g_hash_table_lookup (mail_msg_active_table, GINT_TO_POINTER (msgid)) != NULL; g_mutex_unlock (mail_msg_lock); return active; } void mail_msg_wait_all (void) { if (mail_in_main_thread ()) { g_mutex_lock (mail_msg_lock); while (g_hash_table_size (mail_msg_active_table) > 0) { g_mutex_unlock (mail_msg_lock); gtk_main_iteration (); g_mutex_lock (mail_msg_lock); } g_mutex_unlock (mail_msg_lock); } else { g_mutex_lock (mail_msg_lock); while (g_hash_table_size (mail_msg_active_table) > 0) { g_cond_wait (mail_msg_cond, mail_msg_lock); } g_mutex_unlock (mail_msg_lock); } } /* **************************************** */ static GHookList cancel_hook_list; GHook * mail_cancel_hook_add (GHookFunc func, gpointer data) { GHook *hook; g_mutex_lock (mail_msg_lock); if (!cancel_hook_list.is_setup) g_hook_list_init (&cancel_hook_list, sizeof (GHook)); hook = g_hook_alloc (&cancel_hook_list); hook->func = func; hook->data = data; g_hook_append (&cancel_hook_list, hook); g_mutex_unlock (mail_msg_lock); return hook; } void mail_cancel_hook_remove (GHook *hook) { g_mutex_lock (mail_msg_lock); g_return_if_fail (cancel_hook_list.is_setup); g_hook_destroy_link (&cancel_hook_list, hook); g_mutex_unlock (mail_msg_lock); } void mail_cancel_all (void) { camel_operation_cancel (NULL); g_mutex_lock (mail_msg_lock); if (cancel_hook_list.is_setup) g_hook_list_invoke (&cancel_hook_list, FALSE); g_mutex_unlock (mail_msg_lock); } void mail_msg_set_cancelable (gpointer msg, gboolean status) { MailMsg *mail_msg = msg; mail_msg->priv->cancelable = status; } static guint idle_source_id = 0; G_LOCK_DEFINE_STATIC (idle_source_id); static GAsyncQueue *main_loop_queue = NULL; static GAsyncQueue *msg_reply_queue = NULL; static GThread *main_thread = NULL; static gboolean mail_msg_idle_cb (void) { MailMsg *msg; g_return_val_if_fail (main_loop_queue != NULL, FALSE); g_return_val_if_fail (msg_reply_queue != NULL, FALSE); G_LOCK (idle_source_id); idle_source_id = 0; G_UNLOCK (idle_source_id); /* check the main loop queue */ while ((msg = g_async_queue_try_pop (main_loop_queue)) != NULL) { g_idle_add_full ( G_PRIORITY_DEFAULT, (GSourceFunc) mail_msg_submit, g_object_ref (msg->priv->activity), (GDestroyNotify) g_object_unref); if (msg->info->exec != NULL) msg->info->exec (msg); if (msg->info->done != NULL) msg->info->done (msg); mail_msg_unref (msg); } /* check the reply queue */ while ((msg = g_async_queue_try_pop (msg_reply_queue)) != NULL) { if (msg->info->done != NULL) msg->info->done (msg); mail_msg_check_error (msg); mail_msg_unref (msg); } return FALSE; } static void mail_msg_proxy (MailMsg *msg) { if (msg->info->desc != NULL && msg->cancel) { gchar *text = msg->info->desc (msg); camel_operation_register (msg->cancel); camel_operation_start (msg->cancel, "%s", text); g_free (text); } g_idle_add_full ( G_PRIORITY_DEFAULT, (GSourceFunc) mail_msg_submit, g_object_ref (msg->priv->activity), (GDestroyNotify) g_object_unref); if (msg->info->exec != NULL) msg->info->exec (msg); if (msg->info->desc != NULL && msg->cancel) { camel_operation_end (msg->cancel); camel_operation_unregister (); } g_async_queue_push (msg_reply_queue, msg); G_LOCK (idle_source_id); if (idle_source_id == 0) idle_source_id = g_idle_add ( (GSourceFunc) mail_msg_idle_cb, NULL); G_UNLOCK (idle_source_id); } void mail_msg_cleanup (void) { mail_msg_wait_all (); G_LOCK (idle_source_id); if (idle_source_id != 0) { GSource *source; /* Cancel the idle source. */ source = g_main_context_find_source_by_id ( g_main_context_default (), idle_source_id); g_source_destroy (source); idle_source_id = 0; } G_UNLOCK (idle_source_id); g_async_queue_unref (main_loop_queue); main_loop_queue = NULL; g_async_queue_unref (msg_reply_queue); msg_reply_queue = NULL; } void mail_msg_init (void) { mail_msg_lock = g_mutex_new (); mail_msg_cond = g_cond_new (); main_loop_queue = g_async_queue_new (); msg_reply_queue = g_async_queue_new (); mail_msg_active_table = g_hash_table_new (NULL, NULL); main_thread = g_thread_self (); mail_async_event = mail_async_event_new (); } static gint mail_msg_compare (const MailMsg *msg1, const MailMsg *msg2) { gint priority1 = msg1->priority; gint priority2 = msg2->priority; if (priority1 == priority2) return 0; return (priority1 < priority2) ? 1 : -1; } static gpointer create_thread_pool (gpointer data) { GThreadPool *thread_pool; gint max_threads = GPOINTER_TO_INT (data); /* once created, run forever */ thread_pool = g_thread_pool_new ( (GFunc) mail_msg_proxy, NULL, max_threads, FALSE, NULL); g_thread_pool_set_sort_function ( thread_pool, (GCompareDataFunc) mail_msg_compare, NULL); return thread_pool; } void mail_msg_main_loop_push (gpointer msg) { g_async_queue_push_sorted (main_loop_queue, msg, (GCompareDataFunc) mail_msg_compare, NULL); G_LOCK (idle_source_id); if (idle_source_id == 0) idle_source_id = g_idle_add ( (GSourceFunc) mail_msg_idle_cb, NULL); G_UNLOCK (idle_source_id); } void mail_msg_unordered_push (gpointer msg) { static GOnce once = G_ONCE_INIT; g_once (&once, (GThreadFunc) create_thread_pool, GINT_TO_POINTER (10)); g_thread_pool_push ((GThreadPool *) once.retval, msg, NULL); } void mail_msg_fast_ordered_push (gpointer msg) { static GOnce once = G_ONCE_INIT; g_once (&once, (GThreadFunc) create_thread_pool, GINT_TO_POINTER (1)); g_thread_pool_push ((GThreadPool *) once.retval, msg, NULL); } void mail_msg_slow_ordered_push (gpointer msg) { static GOnce once = G_ONCE_INIT; g_once (&once, (GThreadFunc) create_thread_pool, GINT_TO_POINTER (1)); g_thread_pool_push ((GThreadPool *) once.retval, msg, NULL); } gboolean mail_in_main_thread (void) { return (g_thread_self () == main_thread); } /* ********************************************************************** */ struct _proxy_msg { MailMsg base; MailAsyncEvent *ea; mail_async_event_t type; GThread *thread; guint idle_id; MailAsyncFunc func; gpointer o; gpointer event_data; gpointer data; }; static void do_async_event (struct _proxy_msg *m) { m->thread = g_thread_self (); m->func (m->o, m->event_data, m->data); m->thread = NULL; g_mutex_lock (m->ea->lock); m->ea->tasks = g_slist_remove (m->ea->tasks, m); g_mutex_unlock (m->ea->lock); } static gint idle_async_event (struct _proxy_msg *m) { m->idle_id = 0; do_async_event (m); mail_msg_unref (m); return FALSE; } static MailMsgInfo async_event_info = { sizeof (struct _proxy_msg), (MailMsgDescFunc) NULL, (MailMsgExecFunc) do_async_event, (MailMsgDoneFunc) NULL, (MailMsgFreeFunc) NULL }; MailAsyncEvent * mail_async_event_new (void) { MailAsyncEvent *ea; ea = g_malloc0 (sizeof (*ea)); ea->lock = g_mutex_new (); return ea; } guint mail_async_event_emit (MailAsyncEvent *ea, mail_async_event_t type, MailAsyncFunc func, gpointer o, gpointer event_data, gpointer data) { struct _proxy_msg *m; guint id; /* We dont have a reply port for this, we dont * care when/if it gets executed, just queue it. */ m = mail_msg_new (&async_event_info); m->func = func; m->o = o; m->event_data = event_data; m->data = data; m->ea = ea; m->type = type; m->thread = NULL; id = m->base.seq; g_mutex_lock (ea->lock); ea->tasks = g_slist_prepend (ea->tasks, m); g_mutex_unlock (ea->lock); /* We use an idle function instead of our own message port only * because the gui message ports's notification buffer might * overflow and deadlock us. */ if (type == MAIL_ASYNC_GUI) { if (mail_in_main_thread ()) m->idle_id = g_idle_add ( (GSourceFunc) idle_async_event, m); else mail_msg_main_loop_push (m); } else mail_msg_fast_ordered_push (m); return id; } gint mail_async_event_destroy (MailAsyncEvent *ea) { gint id; struct _proxy_msg *m; g_mutex_lock (ea->lock); while (ea->tasks) { m = ea->tasks->data; id = m->base.seq; if (m->thread == g_thread_self ()) { g_warning("Destroying async event from inside an event, returning EDEADLK"); g_mutex_unlock (ea->lock); errno = EDEADLK; return -1; } if (m->idle_id > 0) { g_source_remove (m->idle_id); m->idle_id = 0; } g_mutex_unlock (ea->lock); mail_msg_wait (id); g_mutex_lock (ea->lock); } g_mutex_unlock (ea->lock); g_mutex_free (ea->lock); g_free (ea); return 0; } /* ********************************************************************** */ struct _call_msg { MailMsg base; mail_call_t type; MailMainFunc func; gpointer ret; va_list ap; EFlag *done; }; static void do_call (struct _call_msg *m) { gpointer p1, *p2, *p3, *p4, *p5; gint i1; va_list ap; G_VA_COPY (ap, m->ap); switch (m->type) { case MAIL_CALL_p_p: p1 = va_arg (ap, gpointer ); m->ret = m->func (p1); break; case MAIL_CALL_p_pp: p1 = va_arg (ap, gpointer ); p2 = va_arg (ap, gpointer ); m->ret = m->func (p1, p2); break; case MAIL_CALL_p_ppp: p1 = va_arg (ap, gpointer ); p2 = va_arg (ap, gpointer ); p3 = va_arg (ap, gpointer ); m->ret = m->func (p1, p2, p3); break; case MAIL_CALL_p_pppp: p1 = va_arg (ap, gpointer ); p2 = va_arg (ap, gpointer ); p3 = va_arg (ap, gpointer ); p4 = va_arg (ap, gpointer ); m->ret = m->func (p1, p2, p3, p4); break; case MAIL_CALL_p_ppppp: p1 = va_arg (ap, gpointer ); p2 = va_arg (ap, gpointer ); p3 = va_arg (ap, gpointer ); p4 = va_arg (ap, gpointer ); p5 = va_arg (ap, gpointer ); m->ret = m->func (p1, p2, p3, p4, p5); break; case MAIL_CALL_p_ppippp: p1 = va_arg (ap, gpointer ); p2 = va_arg (ap, gpointer ); i1 = va_arg (ap, gint); p3 = va_arg (ap, gpointer ); p4 = va_arg (ap, gpointer ); p5 = va_arg (ap, gpointer ); m->ret = m->func (p1, p2, i1, p3, p4, p5); break; } if (m->done != NULL) e_flag_set (m->done); } static MailMsgInfo mail_call_info = { sizeof (struct _call_msg), (MailMsgDescFunc) NULL, (MailMsgExecFunc) do_call, (MailMsgDoneFunc) NULL, (MailMsgFreeFunc) NULL }; gpointer mail_call_main (mail_call_t type, MailMainFunc func, ...) { struct _call_msg *m; gpointer ret; va_list ap; va_start (ap, func); m = mail_msg_new (&mail_call_info); m->type = type; m->func = func; G_VA_COPY (m->ap, ap); if (mail_in_main_thread ()) do_call (m); else { mail_msg_ref (m); m->done = e_flag_new (); mail_msg_main_loop_push (m); e_flag_wait (m->done); e_flag_free (m->done); } va_end (ap); ret = m->ret; mail_msg_unref (m); return ret; } void mail_mt_set_backend (gchar *backend) { shell_builtin_backend = backend; }