Re: Some IDLE improvements



On Tue, 2007-10-23 at 14:51 +0200, Philip Van Hoof wrote:
> On Tue, 2007-10-23 at 13:51 +0200, Philip Van Hoof wrote:
> > This is a patch that improves IDLE.
> > 
> > Please review.
> 
> New version.

And again new version

> 
> > ps. Don't tell Dave, but this is an awesome IMAP server:
> > 
> > turner.dave.cridland.net
> > u:pvanhoof
> > p:qresync
> > 
> > :-)
> > 
> > 
> > _______________________________________________
> > tinymail-devel-list mailing list
> > tinymail-devel-list gnome org
> > http://mail.gnome.org/mailman/listinfo/tinymail-devel-list
> _______________________________________________
> tinymail-devel-list mailing list
> tinymail-devel-list gnome org
> http://mail.gnome.org/mailman/listinfo/tinymail-devel-list
-- 
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 2870)
+++ 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;
 		}
@@ -441,7 +432,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;
Index: libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h
===================================================================
--- libtinymail-camel/camel-lite/camel/providers/imap/camel-imap-store.h	(revision 2867)
+++ 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 2870)
+++ 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>
@@ -3591,11 +3593,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 +3603,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 +3628,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 +3653,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 +3674,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 +3912,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 +3924,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 +3981,8 @@
 
 	return;
 }
+
+
 /* Called with the store's connect_lock locked */
 void
 camel_imap_folder_changed (CamelFolder *folder, int exists,
Index: ChangeLog
===================================================================
--- ChangeLog	(revision 2872)
+++ ChangeLog	(working copy)
@@ -1,6 +1,7 @@
 2007-10-23  Philip Van Hoof  <pvanhoof gnome org>
 
 	* Reference count problem in TnyCamelHeader
+	* 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]