Re: Some IDLE improvements
- From: Philip Van Hoof <spam pvanhoof be>
- To: tinymail-devel-list gnome org
- Subject: Re: Some IDLE improvements
- Date: Tue, 23 Oct 2007 18:51:21 +0200
On Tue, 2007-10-23 at 15:13 +0200, Philip Van Hoof wrote:
And another version :)
--
Philip Van Hoof, software developer
home: me at pvanhoof dot be
gnome: pvanhoof at gnome dot org
http://www.pvanhoof.be/blog
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c (revision 2872)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.c (working copy)
@@ -195,49 +195,48 @@
}
static void
-let_idle_die (CamelImapStore *imap_store, gboolean connect_buz)
+let_idle_die (CamelImapStore *store, gboolean connect_buz)
{
- imap_store->idle_cont = FALSE;
- g_static_rec_mutex_lock (imap_store->idle_prefix_lock);
- g_static_rec_mutex_lock (imap_store->idle_lock);
+ idle_debug ("let_idle_die starts\n");
- imap_store->idle_cont = FALSE;
+ g_static_rec_mutex_lock (store->idle_prefix_lock);
+ g_static_rec_mutex_lock (store->idle_lock);
- /* This one can get called from within the thread! This would deadlock
+ store->idle_kill = TRUE;
+ store->idle_cont = FALSE;
- if (imap_store->in_idle && imap_store->idle_thread) {
- g_thread_join (imap_store->idle_thread);
- imap_store->idle_thread = NULL;
- } */
-
- if (imap_store->idle_prefix)
+ if (store->idle_prefix)
{
gchar *resp = NULL;
int nwritten = 0;
CamelException ex = CAMEL_EXCEPTION_INITIALISER;
- g_free (imap_store->idle_prefix);
- imap_store->idle_prefix=NULL;
-
idle_debug ("Sending DONE in let_idle_die\n");
- nwritten = camel_stream_printf (imap_store->ostream, "DONE\r\n");
- if (nwritten != -1)
- {
+ CAMEL_SERVICE_REC_LOCK (store, connect_lock);
+ nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
+ store->in_idle = FALSE;
+ if (nwritten != -1) {
resp = NULL;
- while ((camel_imap_command_response_idle (imap_store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED)
- {
- idle_debug ("(.., ..) <- %s | in idle_deal_with_stuff in let_idle_die\n", resp);
+ while ((camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) {
+ idle_debug ("(.., ..) <- %s | in let_idle_die\n", resp);
g_free (resp); resp=NULL;
}
- if (resp)
+ if (resp) {
+ idle_debug ("(.., ..) <- %s\n", resp);
g_free (resp);
+ }
}
+ CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
+ g_free (store->idle_prefix);
+ store->idle_prefix=NULL;
}
- g_static_rec_mutex_unlock (imap_store->idle_lock);
- g_static_rec_mutex_unlock (imap_store->idle_prefix_lock);
+ g_static_rec_mutex_unlock (store->idle_lock);
+ g_static_rec_mutex_unlock (store->idle_prefix_lock);
+ idle_debug ("let_idle_die finished\n");
+
return;
}
@@ -247,41 +246,33 @@
if (store->current_folder && CAMEL_IS_IMAP_FOLDER (store->current_folder))
camel_imap_folder_stop_idle (store->current_folder);
else {
- store->idle_cont = FALSE;
-
g_static_rec_mutex_lock (store->idle_prefix_lock);
g_static_rec_mutex_lock (store->idle_lock);
+ store->idle_kill = TRUE;
store->idle_cont = FALSE;
- if (store->in_idle && store->idle_thread) {
- g_thread_join (store->idle_thread);
- store->idle_thread = NULL;
- }
- if (store->idle_prefix)
- {
+ if (store->idle_prefix) {
gchar *resp = NULL;
int nwritten = 0;
CamelException ex = CAMEL_EXCEPTION_INITIALISER;
- idle_debug ("Sending DONE in camel_imap_store_stop_idle (no current folder?)\n");
+ idle_debug ("Sending DONE in camel_imap_store_stop_idle\n");
CAMEL_SERVICE_REC_LOCK (store, connect_lock);
nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
-
- if (nwritten != -1)
- {
+ store->in_idle = FALSE;
+ if (nwritten != -1) {
resp = NULL;
- while ((camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED)
- {
+ while ((camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED) {
idle_debug ("(.., ..) <- %s | in idle_deal_with_stuff (no current folder?)\n", resp);
g_free (resp); resp=NULL;
}
- if (resp)
+ if (resp) {
+ idle_debug ("(.., ..) <- %s\n", resp);
g_free (resp);
+ }
}
-
CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-
g_free (store->idle_prefix);
store->idle_prefix = NULL;
}
@@ -312,6 +303,7 @@
camel_imap_folder_selected (folder, response2, &nex, TRUE);
camel_imap_response_free (imap_store, response2);
}
+ camel_imap_store_start_idle (imap_store);
}
return;
@@ -441,7 +433,7 @@
imap_store->dontdistridlehack = FALSE;
- imap_store->idle_sleep = 20; /* default of 20s */
+ imap_store->idle_sleep = 600; /* default of 10m */
imap_store->getsrv_sleep = 100; /* default of 100s */
imap_store->in_idle = FALSE;
@@ -2588,8 +2580,8 @@
}
imap_status_item_free (items);
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
return;
}
@@ -2630,16 +2622,18 @@
const char *c;
if (camel_exception_get_id(ex) == CAMEL_EXCEPTION_USER_CANCEL) {
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+
return NULL;
}
camel_exception_clear (ex);
if (!(flags & CAMEL_STORE_FOLDER_CREATE)) {
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+
camel_exception_setv (ex, CAMEL_EXCEPTION_STORE_NO_FOLDER,
_("No such folder %s"), folder_name);
return NULL;
@@ -2651,11 +2645,11 @@
c++;
if (*c != '\0') {
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
camel_exception_setv (ex, CAMEL_EXCEPTION_FOLDER_INVALID_PATH,
_("The folder name \"%s\" is invalid because it contains the character \"%c\""),
folder_name, *c);
+ camel_imap_store_start_idle (imap_store);
return NULL;
}
@@ -2673,10 +2667,12 @@
int i;
if (!(response = camel_imap_command (imap_store, NULL, ex, "LIST \"\" %G", parent_real))) {
- camel_imap_store_start_idle (imap_store);
- CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
g_free (parent_name);
g_free (parent_real);
+
+ CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+
return NULL;
}
@@ -2717,12 +2713,13 @@
imap_status_item_free (items);
if (messages > 0) {
- camel_imap_store_start_idle (imap_store);
camel_exception_set (ex, CAMEL_EXCEPTION_FOLDER_INVALID_STATE,
_("The parent folder is not allowed to contain subfolders"));
- CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
g_free (parent_name);
g_free (parent_real);
+ CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+
return NULL;
}
@@ -2730,11 +2727,11 @@
camel_exception_init (&lex);
delete_folder (store, parent_name, &lex);
if (camel_exception_is_set (&lex)) {
- camel_imap_store_start_idle (imap_store);
- CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
- camel_exception_xfer (ex, &lex);
g_free (parent_name);
g_free (parent_real);
+ CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+ camel_exception_xfer (ex, &lex);
return NULL;
}
@@ -2745,10 +2742,11 @@
g_free (name);
if (!response) {
- camel_imap_store_start_idle (imap_store);
- CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
g_free (parent_name);
g_free (parent_real);
+ CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+
return NULL;
} else
camel_imap_response_free (imap_store, response);
@@ -2770,8 +2768,9 @@
}
g_free(folder_real);
if (!response) {
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
+ camel_imap_store_start_idle (imap_store);
+
return NULL;
}
} else if (flags & CAMEL_STORE_FOLDER_EXCL) {
@@ -2780,9 +2779,9 @@
folder_name);
camel_imap_response_free_without_processing (imap_store, response);
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
-
+ camel_imap_store_start_idle (imap_store);
+
return NULL;
}
@@ -2809,9 +2808,9 @@
}
camel_imap_response_free_without_processing (imap_store, response);
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK (imap_store, connect_lock);
-
+ camel_imap_store_start_idle (imap_store);
+
return new_folder;
}
@@ -2850,7 +2849,6 @@
{
CamelImapStore *imap_store = CAMEL_IMAP_STORE (store);
CamelImapResponse *response;
- gboolean did_start_idle = FALSE;
CamelFolder *old_in_case;
CAMEL_SERVICE_REC_LOCK (imap_store, connect_lock);
@@ -2881,7 +2879,6 @@
if (response2) {
camel_imap_folder_selected (old_in_case, response2, &nex, TRUE);
camel_imap_response_free (imap_store, response2);
- did_start_idle = TRUE;
}
}
@@ -2890,9 +2887,10 @@
imap_forget_folder (imap_store, folder_name, ex);
}
fail:
- if (!did_start_idle)
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK(imap_store, connect_lock);
+
+ camel_imap_store_start_idle (imap_store);
+
}
static void
@@ -3033,7 +3031,6 @@
char *oldpath, *newpath, *storage_path;
char *tpath, *lslash;
CamelFolder *old_in_case = NULL;
- gboolean did_start_idle = FALSE;
CAMEL_SERVICE_REC_LOCK (imap_store, connect_lock);
@@ -3078,7 +3075,6 @@
if (response2) {
camel_imap_folder_selected (old_in_case, response2, &nex, TRUE);
camel_imap_response_free (imap_store, response2);
- did_start_idle = TRUE;
}
}
@@ -3126,10 +3122,11 @@
g_free (newpath);
fail:
imap_store->renaming = FALSE;
- if (!did_start_idle)
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK(imap_store, connect_lock);
camel_operation_end (NULL);
+
+ camel_imap_store_start_idle (imap_store);
+
}
static CamelFolderInfo *
@@ -3955,8 +3952,10 @@
imap_folder_effectively_unsubscribed (imap_store, folder_name, ex);
done:
- camel_imap_store_start_idle (imap_store);
CAMEL_SERVICE_REC_UNLOCK(store, connect_lock);
+
+ camel_imap_store_start_idle (imap_store);
+
}
#if 0
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h (revision 2872)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h (working copy)
@@ -165,7 +165,7 @@
GStaticRecMutex *idle_prefix_lock, *idle_lock, *sum_lock;
GThread *idle_thread;
- gboolean idle_cont, in_idle;
+ gboolean idle_cont, in_idle, idle_kill;
guint idle_sleep, getsrv_sleep;
gboolean courier_crap;
gboolean going_online, got_online, clean_exit;
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-folder.c
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-folder.c (revision 2872)
+++ libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-folder.c (working copy)
@@ -38,6 +38,8 @@
#include <config.h>
+#include <sched.h>
+
#include <ctype.h>
#include <errno.h>
#include <fcntl.h>
@@ -760,8 +762,6 @@
camel_folder_change_info_free(changes);
}
- if (idle)
- camel_imap_folder_start_idle (folder);
}
@@ -907,18 +907,17 @@
* messages. */
imap_rescan (folder, camel_folder_summary_count (folder->summary), ex);
} else {
+
#if 0
/* on some servers need to CHECKpoint INBOX to recieve new messages?? */
/* rfc2060 suggests this, but havent seen a server that requires it */
if (g_ascii_strcasecmp(folder->full_name, "INBOX") == 0) {
response = camel_imap_command (imap_store, folder, ex, "CHECK");
camel_imap_response_free (imap_store, response);
- camel_imap_folder_start_idle (folder);
}
#endif
response = camel_imap_command (imap_store, folder, ex, "NOOP");
camel_imap_response_free (imap_store, response);
- camel_imap_folder_start_idle (folder);
}
si = camel_store_summary_path((CamelStoreSummary *)((CamelImapStore *)folder->parent_store)->summary, folder->full_name);
@@ -939,8 +938,10 @@
CAMEL_FOLDER_REC_UNLOCK(folder, lock);
camel_folder_summary_save(folder->summary, ex);
+ camel_store_summary_save((CamelStoreSummary *)((CamelImapStore *)folder->parent_store)->summary, ex);
- camel_store_summary_save((CamelStoreSummary *)((CamelImapStore *)folder->parent_store)->summary, ex);
+ camel_imap_folder_start_idle (folder);
+
}
#if 0
@@ -1522,19 +1523,17 @@
}
g_ptr_array_free (matches, TRUE);
- if (camel_exception_is_set (&local_ex))
- camel_imap_folder_start_idle (folder);
-
/* We unlock here so that other threads can have a chance to grab the connect_lock */
CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-
+
/* check for an exception */
if (camel_exception_is_set (&local_ex)) {
- /* Look up */
+
camel_exception_xfer (ex, &local_ex);
+ camel_imap_folder_start_idle (folder);
return;
}
-
+
/* Re-lock the connect_lock */
CAMEL_SERVICE_REC_LOCK (store, connect_lock);
}
@@ -1542,9 +1541,9 @@
/* Save the summary */
imap_sync_offline (folder, ex);
+ CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
camel_imap_folder_start_idle (folder);
- CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
- /* camel_imap_folder_start_idle (folder); */
+
}
static int
@@ -1643,7 +1642,6 @@
CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
-
camel_imap_folder_start_idle (folder);
}
@@ -3591,11 +3589,8 @@
idle_response_new (CamelFolder *folder)
{
IdleResponse *idle_resp = g_slice_new0 (IdleResponse);
-
+ idle_debug ("idle_response_new\n");
idle_resp->vanished = NULL;
-
- idle_debug ("idle_response_new\n");
-
idle_resp->folder = folder;
camel_object_ref (CAMEL_OBJECT (folder));
return idle_resp;
@@ -3604,27 +3599,23 @@
static void
idle_response_free (IdleResponse *idle_resp)
{
- guint i=0;
-
idle_debug ("idle_response_free\n");
-
if (idle_resp->expunged)
g_array_free (idle_resp->expunged, TRUE);
-
if (idle_resp->vanished) {
- g_list_foreach (idle_resp->vanished, (GFunc)g_free, NULL);
+ g_list_foreach (idle_resp->vanished, (GFunc) g_free, NULL);
g_list_free (idle_resp->vanished);
idle_resp->vanished = NULL;
}
-
- if (idle_resp->fetch)
+ if (idle_resp->fetch) {
+ guint i=0;
for (i=0 ;i < idle_resp->fetch->len; i++)
g_slice_free (FetchIdleResponse, idle_resp->fetch->pdata[i]);
-
+ }
if (idle_resp->folder)
camel_object_unref (CAMEL_OBJECT (idle_resp->folder));
-
g_slice_free (IdleResponse, idle_resp);
+ return;
}
@@ -3633,18 +3624,18 @@
{
char *resp;
CamelException ex = CAMEL_EXCEPTION_INITIALISER;
-
+ gboolean l = CAMEL_SERVICE_REC_TRYLOCK (store, connect_lock);
if (store->ostream && store->istream && CAMEL_IS_STREAM (store->ostream))
{
+ int nwritten = 0;
if (store->idle_prefix)
g_free (store->idle_prefix);
store->idle_prefix = g_strdup_printf ("%c%.5u",
store->tag_prefix, store->command++);
-
- idle_debug ("(.., ..) -> %s IDLE | in idle_real_start\n", store->idle_prefix);
-
- camel_stream_printf (store->ostream, "%s IDLE\r\n",
+ nwritten = camel_stream_printf (store->ostream, "%s IDLE\r\n",
store->idle_prefix);
+ idle_debug ("(%d, %d) -> %s IDLE\n", strlen (store->idle_prefix)+5,
+ nwritten, store->idle_prefix);
} else {
idle_debug ("idle_real_start connection lost\n");
goto errh;
@@ -3658,7 +3649,6 @@
* active, the server is now free to send untagged EXISTS, EXPUNGE, and
* other messages at any time. */
-
/* So according to the RFC, we will wait for the server for its +
* continuation. If the server doesn't do this, it's an incorrect
* IDLE implementation at the server. Right? */
@@ -3680,248 +3670,236 @@
}
if (resp)
g_free (resp);
-
errh:
+ if (l)
+ CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
+ return;
+}
+
+static void
+consume_idle_line (CamelImapStore *store, CamelFolder *folder, char *resp, IdleResponse *idle_resp)
+{
+ if (strchr (resp, '*') != NULL && (camel_strstrcase (resp, "EXISTS") ||
+ camel_strstrcase (resp, "FETCH")|| camel_strstrcase (resp, "EXPUNGE") ||
+ camel_strstrcase (resp, "VANISHED") || camel_strstrcase (resp, "RECENT")))
+ {
+ if (!idle_resp)
+ idle_resp = idle_response_new (folder);
+ read_idle_response (folder, resp, idle_resp);
+ }
+ idle_debug ("(%d, ..) <- %s\n",
+ strlen (resp), resp);
return;
}
-static IdleResponse*
-idle_deal_with_stuff (CamelFolder *folder, CamelImapStore *store, gboolean *had_err, gboolean *had_lock)
+typedef struct {
+ CamelFolder *folder;
+ GCond *condition;
+ GMutex *mutex;
+ gboolean had_cond;
+} IdleThreadInfo;
+
+static gpointer
+idle_thread (gpointer data)
{
- IdleResponse *idle_resp = NULL;
- CamelException ex = CAMEL_EXCEPTION_INITIALISER;
- char *resp = NULL;
- int nwritten=0;
- CamelImapResponseType type;
- gboolean hlock = FALSE;
+ IdleThreadInfo *info = (IdleThreadInfo *) data;
+ CamelFolder *folder = (CamelFolder *) info->folder;
+ CamelImapFolder *imap_folder;
+ CamelImapStore *store;
+ gboolean tfirst = TRUE, first = TRUE, my_cont;
+ int cnt = 0;
+ int nwritten=0;
- idle_debug ("idle_deal_with_stuff\n");
+ idle_debug ("idle_thread starts\n");
- if (!camel_disco_store_check_online ((CamelDiscoStore*)store, &ex))
- return NULL;
+ if (!folder || !folder->parent_store) {
+ g_thread_exit (NULL);
+ return NULL;
+ }
- hlock = CAMEL_SERVICE_REC_TRYLOCK (store, connect_lock);
+ if (!CAMEL_IS_FOLDER (folder) || !CAMEL_IS_STORE (folder->parent_store)) {
+ g_thread_exit (NULL);
+ return NULL;
+ }
- if (hlock)
- {
- if (store->current_folder)
- {
- /* We read-away everything non-blocking and process it */
- resp = NULL;
- while (camel_imap_store_readline_nb (store, &resp, &ex) > 0)
- {
- if (strchr (resp, '*') != NULL && (camel_strstrcase (resp, "EXISTS") ||
- camel_strstrcase (resp, "FETCH")|| camel_strstrcase (resp, "EXPUNGE") ||
- camel_strstrcase (resp, "VANISHED") || camel_strstrcase (resp, "RECENT")))
- {
- if (!idle_resp)
- idle_resp = idle_response_new (folder);
- read_idle_response (folder, resp, idle_resp);
- }
- idle_debug ("(%d, ..) <- %s | in idle_deal_with_stuff at nb\n",
- strlen (resp), resp);
- g_free (resp); resp=NULL;
- }
- if (resp)
- g_free (resp);
+ store = CAMEL_IMAP_STORE (folder->parent_store);
+ imap_folder = CAMEL_IMAP_FOLDER (folder);
+ if (!imap_folder->do_push_email) {
+ idle_debug ("Folder set not to do idle\n");
+ g_thread_exit (NULL);
+ return NULL;
+ }
- /* Here we force the server to tell us about the changes: */
+ if (!(store->capabilities & IMAP_CAPABILITY_IDLE)) {
+ idle_debug ("Server has no IDLE capabilities\n");
+ g_thread_exit (NULL);
+ return NULL;
+ }
- /* The IDLE command is terminated by the receipt of a "DONE"
- * continuation from the client; such response satisfies the server's
- * continuation request. At that point, the server MAY send any
- * remaining queued untagged responses and then MUST immediately send
- * the tagged response to the IDLE command and prepare to process other
- * commands. */
- if (store->ostream && CAMEL_IS_STREAM (store->ostream)) {
- nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
- idle_debug ("(%d, 8) -> DONE | Sending DONE in idle_deal_with_stuff (nb)\n",
- nwritten);
+ camel_object_ref (folder);
+
+ my_cont = store->idle_cont;
+
+ if (my_cont) {
+ idle_debug ("idle_thread starting\n");
+ } else {
+ idle_debug ("idle_thread not starting\n");
+ }
+
+ while (my_cont && !store->idle_kill)
+ {
+ CamelException ex = CAMEL_EXCEPTION_INITIALISER;
+ char *resp = NULL;
+ IdleResponse *idle_resp = NULL;
+
+ if (store->idle_cont && first) {
+ gboolean l = g_static_rec_mutex_trylock (store->idle_lock);
+ if (!store->idle_kill)
+ idle_real_start (store);
+ if (l)
+ g_static_rec_mutex_unlock (store->idle_lock);
+ first = FALSE;
}
- if (nwritten == -1)
- goto outofhere;
+ if (tfirst) {
+ if (info->condition) {
+ g_mutex_lock (info->mutex);
+ g_cond_broadcast (info->condition);
+ info->had_cond = TRUE;
+ g_mutex_unlock (info->mutex);
+ }
+ tfirst = FALSE;
+ }
- resp = NULL;
- while ((type = camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED)
- {
- /* printf ("D resp: %s\n", resp); */
- if (strchr (resp, '*') != NULL && (camel_strstrcase (resp, "EXISTS") ||
- camel_strstrcase (resp, "FETCH") || camel_strstrcase (resp, "EXPUNGE") ||
- camel_strstrcase (resp, "RECENT")))
- {
+ if (g_static_rec_mutex_trylock (store->idle_lock)) {
+ if (!store->idle_kill)
+ while (camel_imap_store_readline_nb (store, &resp, &ex) > 0)
+ {
if (!idle_resp)
idle_resp = idle_response_new (folder);
- read_idle_response (folder, resp, idle_resp);
- }
- idle_debug ("(%d, ..) <- %s | in idle_deal_with_stuff at idle\n",
- strlen (resp), resp);
- g_free (resp); resp=NULL;
+ consume_idle_line (store, folder, resp, idle_resp);
+ g_free (resp);
+ resp = NULL;
+ }
+ g_static_rec_mutex_unlock (store->idle_lock);
}
- if (type == CAMEL_IMAP_RESPONSE_ERROR)
- *had_err = TRUE;
-
if (resp)
g_free (resp);
- } else {
- /* Trying to deal while the current folder is gone: just read away everything */
- if (store->ostream && CAMEL_IS_STREAM (store->ostream)) {
- nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
- idle_debug ("(%d, 8) -> DONE | Sending DONE in idle_deal_with_stuff (b)\n",
- nwritten);
+ if (idle_resp) {
+ process_idle_response (idle_resp);
+ idle_response_free (idle_resp);
}
- if (nwritten == -1)
- goto outofhere;
- resp = NULL;
- while ((type = camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED)
+
+ idle_resp = NULL;
+
+ if ((cnt > store->idle_sleep) || (!store->idle_cont))
{
- idle_debug ("(.., ..) <- %s | in idle_deal_with_stuff in else\n", resp);
- g_free (resp); resp=NULL;
+ if (store->idle_prefix)
+ {
+ CamelImapResponseType type;
+ gboolean l = g_static_rec_mutex_trylock (store->idle_lock);
+
+ if (!store->idle_kill) {
+ nwritten = camel_stream_printf (store->ostream, "DONE\r\n");
+ idle_debug ("(%d, 8) -> DONE\n", nwritten);
+ while ((type = camel_imap_command_response_idle (store, &resp, &ex)) == CAMEL_IMAP_RESPONSE_UNTAGGED)
+ {
+ if (!idle_resp)
+ idle_resp = idle_response_new (folder);
+ consume_idle_line (store, folder, resp, idle_resp);
+ g_free (resp);
+ resp = NULL;
+ }
+ }
+
+ if (l)
+ g_static_rec_mutex_unlock (store->idle_lock);
+ if (resp) {
+ idle_debug ("(.., ..) <- %s\n", resp);
+ g_free (resp);
+ }
+
+ if (idle_resp) {
+ process_idle_response (idle_resp);
+ idle_response_free (idle_resp);
+ }
+ }
+ if (store->idle_cont)
+ first = TRUE;
+ else
+ my_cont = FALSE;
+
+ cnt = 0;
}
- if (resp)
- g_free (resp);
+
+ if (my_cont)
+ usleep (500000);
+ cnt++;
+
}
-outofhere:
+ if (!info->had_cond && info->condition) {
+ g_mutex_lock (info->mutex);
+ g_cond_broadcast (info->condition);
+ info->had_cond = TRUE;
+ g_mutex_unlock (info->mutex);
+ }
- CAMEL_SERVICE_REC_UNLOCK (store, connect_lock);
- }
+ camel_object_unref (folder);
- *had_lock = hlock;
+ g_thread_exit (NULL);
- return idle_resp;
+ return NULL;
}
+
void
camel_imap_folder_stop_idle (CamelFolder *folder)
{
CamelImapStore *store;
- IdleResponse *idle_resp = NULL;
- gboolean had_err = FALSE;
CamelException ex = CAMEL_EXCEPTION_INITIALISER;
idle_debug ("camel_imap_folder_stop_idle\n");
store = CAMEL_IMAP_STORE (folder->parent_store);
+ store->idle_cont = FALSE;
+
if (!camel_disco_store_check_online ((CamelDiscoStore*)store, &ex))
return;
- store->idle_cont = FALSE;
-
- g_static_rec_mutex_lock (store->idle_lock);
- g_static_rec_mutex_lock (store->idle_prefix_lock);
-
- if ((store->capabilities & IMAP_CAPABILITY_IDLE) && store->idle_prefix)
+ if ((store->capabilities & IMAP_CAPABILITY_IDLE))
{
- gboolean had_lock = FALSE;
-
- store->idle_cont = FALSE;
if (store->in_idle && store->idle_thread) {
+ store->idle_cont = FALSE;
g_thread_join (store->idle_thread);
+ g_static_rec_mutex_lock (store->idle_prefix_lock);
+ g_static_rec_mutex_lock (store->idle_lock);
+ store->in_idle = FALSE;
store->idle_thread = NULL;
+ if (store->idle_prefix)
+ g_free (store->idle_prefix);
+ g_static_rec_mutex_unlock (store->idle_lock);
+ g_static_rec_mutex_unlock (store->idle_prefix_lock);
}
-
- if (store->idle_prefix)
- g_free (store->idle_prefix);
store->idle_prefix = NULL;
-
- idle_resp = idle_deal_with_stuff (folder, store, &had_err, &had_lock);
-
- if (idle_resp && !had_err)
- process_idle_response (idle_resp);
-
- if (idle_resp)
- idle_response_free (idle_resp);
}
- g_static_rec_mutex_unlock (store->idle_prefix_lock);
- g_static_rec_mutex_unlock (store->idle_lock);
-
+ return;
}
-static gpointer
-idle_thread (gpointer data)
-{
- CamelFolder *folder = (CamelFolder *) data;
- CamelImapFolder *imap_folder;
- CamelImapStore *store;
- gboolean had_err = FALSE, hadlock = FALSE;
-
- idle_debug ("idle_thread\n");
-
- if (!folder || !folder->parent_store)
- { g_thread_exit (NULL); return NULL; }
-
- if (!CAMEL_IS_FOLDER (folder) || !CAMEL_IS_STORE (folder->parent_store))
- { g_thread_exit (NULL); return NULL; }
-
- store = CAMEL_IMAP_STORE (folder->parent_store);
- imap_folder = CAMEL_IMAP_FOLDER (folder);
-
- if (!imap_folder->do_push_email) {
- idle_debug ("Folder set not to do idle\n");
- return NULL;
- }
-
- if (!(store->capabilities & IMAP_CAPABILITY_IDLE)) {
- idle_debug ("Server has no IDLE capabilities\n");
- return NULL;
- }
-
- idle_debug ("idle_thread starting (%s)\n", store->idle_prefix?store->idle_prefix:"(none)");
-
- store->idle_cont = TRUE;
-
- while (store->idle_cont && store->idle_prefix != NULL)
- {
- int x = 0;
-
- g_static_rec_mutex_lock (store->idle_lock);
- g_static_rec_mutex_lock (store->idle_prefix_lock);
-
- store->in_idle = TRUE;
- if (store->idle_prefix != NULL)
- {
- IdleResponse *idle_resp = idle_deal_with_stuff (folder, store, &had_err,&hadlock);
-
- if (idle_resp && !had_err)
- process_idle_response (idle_resp);
-
- if (hadlock)
- idle_real_start (store);
-
- if (idle_resp)
- idle_response_free (idle_resp);
- }
-
- g_static_rec_mutex_unlock (store->idle_prefix_lock);
- g_static_rec_mutex_unlock (store->idle_lock);
-
- idle_debug ("idle checked in idle_thread, waiting %ds for new check\n", store->idle_sleep);
-
- for (x=0; x<1000 && store->idle_cont; x++)
- usleep (store->idle_sleep * 1000);
- }
-
- store->in_idle = FALSE;
-
- g_thread_exit (NULL);
- return NULL;
-}
-
-
void
camel_imap_folder_start_idle (CamelFolder *folder)
{
CamelImapStore *store;
CamelException ex = CAMEL_EXCEPTION_INITIALISER;
+ CamelImapFolder *imap_folder = (CamelImapFolder *) folder;
idle_debug ("camel_imap_folder_start_idle\n");
@@ -3930,6 +3908,9 @@
if (!camel_disco_store_check_online ((CamelDiscoStore*)store, &ex))
return;
+ if (!imap_folder->do_push_email)
+ return;
+
g_static_rec_mutex_lock (store->idle_lock);
if (store->capabilities & IMAP_CAPABILITY_IDLE)
@@ -3939,17 +3920,36 @@
{
folder->folder_flags |= CAMEL_FOLDER_HAS_PUSHEMAIL_CAPABILITY;
- if (!store->in_idle && store->idle_thread)
- {
+ if (!store->in_idle && store->idle_thread) {
store->idle_cont = FALSE;
g_thread_join (store->idle_thread);
store->idle_thread = NULL;
}
if (!store->in_idle) {
- idle_real_start (store);
+ IdleThreadInfo *info = g_slice_new0 (IdleThreadInfo);
+ store->idle_kill = FALSE;
+ store->in_idle = TRUE;
+ store->idle_cont = TRUE;
+
+ info->mutex = g_mutex_new ();
+ info->condition = g_cond_new ();
+ info->had_cond = FALSE;
+ info->folder = folder;
+ camel_object_ref (info->folder);
+
store->idle_thread = g_thread_create (idle_thread,
- folder, TRUE, NULL);
+ info, TRUE, NULL);
+ g_mutex_lock (info->mutex);
+ if (!info->had_cond)
+ g_cond_wait (info->condition, info->mutex);
+ g_mutex_unlock (info->mutex);
+
+ camel_object_unref (info->folder);
+ g_mutex_free (info->mutex);
+ g_cond_free (info->condition);
+ g_slice_free (IdleThreadInfo, info);
+
}
}
@@ -3977,6 +3977,8 @@
return;
}
+
+
/* Called with the store's connect_lock locked */
void
camel_imap_folder_changed (CamelFolder *folder, int exists,
@@ -4018,9 +4020,9 @@
len = camel_folder_summary_count (folder->summary);
if (exists > len) {
+ /* TNY Question: is this stop really necessary? Why? */
camel_imap_folder_stop_idle (folder);
imap_update_summary (folder, exists, changes, ex);
- /* camel_imap_folder_start_idle (folder); will happen later? */
}
if (camel_folder_change_info_changed (changes))
Index: ChangeLog
===================================================================
--- ChangeLog (revision 2874)
+++ ChangeLog (working copy)
@@ -3,6 +3,7 @@
* Reference count problem in TnyCamelHeader
* priv->folder_name in TnyCamelFolder sometimes is NULL, which doesn't
seem right (and is racy).
+ * Improvements for the IDLE support
2007-10-19 Philip Van Hoof <pvanhoof gnome org>
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]