| 
									
										
										
										
											2002-07-04 15:19:30 +00:00
										 |  |  | #undef G_DISABLE_ASSERT
 | 
					
						
							|  |  |  | #undef G_LOG_DOMAIN
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <glib.h>
 | 
					
						
							| 
									
										
										
										
											2001-01-06 03:09:46 +00:00
										 |  |  | #ifdef G_OS_UNIX
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2001-01-06 03:09:46 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-06 03:09:46 +00:00
										 |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  | #include <fcntl.h>		/* For _O_BINARY used by pipe() macro */
 | 
					
						
							| 
									
										
										
										
											2001-07-20 17:13:52 +00:00
										 |  |  | #include <io.h>			/* for _pipe() */
 | 
					
						
							| 
									
										
										
										
											2007-11-07 09:49:25 +00:00
										 |  |  | #define pipe(fds) _pipe(fds, 4096, _O_BINARY)
 | 
					
						
							| 
									
										
										
										
											2001-01-06 03:09:46 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | #define ITERS 10000
 | 
					
						
							|  |  |  | #define INCREMENT 10
 | 
					
						
							|  |  |  | #define NTHREADS 4
 | 
					
						
							|  |  |  | #define NCRAWLERS 4
 | 
					
						
							|  |  |  | #define CRAWLER_TIMEOUT_RANGE 40
 | 
					
						
							|  |  |  | #define RECURSER_TIMEOUT 50
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-02 02:56:08 +00:00
										 |  |  | /* The partial ordering between the context array mutex and
 | 
					
						
							|  |  |  |  * crawler array mutex is that the crawler array mutex cannot | 
					
						
							|  |  |  |  * be locked while the context array mutex is locked | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | GPtrArray *context_array; | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  | GMutex context_array_mutex; | 
					
						
							|  |  |  | GCond context_array_cond; | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | GMainLoop *main_loop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | G_LOCK_DEFINE_STATIC (crawler_array_lock); | 
					
						
							|  |  |  | GPtrArray *crawler_array; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _AddrData AddrData; | 
					
						
							|  |  |  | typedef struct _TestData TestData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct _AddrData | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GMainLoop *loop; | 
					
						
							|  |  |  |   GIOChannel *dest; | 
					
						
							|  |  |  |   gint count; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct _TestData | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   gint current_val; | 
					
						
							|  |  |  |   gint iters; | 
					
						
							|  |  |  |   GIOChannel *in; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void cleanup_crawlers (GMainContext *context); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2002-09-29 19:16:31 +00:00
										 |  |  | read_all (GIOChannel *channel, char *buf, gsize len) | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-09-29 19:16:31 +00:00
										 |  |  |   gsize bytes_read = 0; | 
					
						
							|  |  |  |   gsize count; | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   GIOError err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (bytes_read < len) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       err = g_io_channel_read (channel, buf + bytes_read, len - bytes_read, &count); | 
					
						
							|  |  |  |       if (err) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (err != G_IO_ERROR_AGAIN) | 
					
						
							|  |  |  | 	    return FALSE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (count == 0) | 
					
						
							|  |  |  | 	return FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       bytes_read += count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2002-09-29 19:16:31 +00:00
										 |  |  | write_all (GIOChannel *channel, char *buf, gsize len) | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-09-29 19:16:31 +00:00
										 |  |  |   gsize bytes_written = 0; | 
					
						
							|  |  |  |   gsize count; | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   GIOError err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (bytes_written < len) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       err = g_io_channel_write (channel, buf + bytes_written, len - bytes_written, &count); | 
					
						
							|  |  |  |       if (err && err != G_IO_ERROR_AGAIN) | 
					
						
							|  |  |  | 	return FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       bytes_written += count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | adder_callback (GIOChannel   *source, | 
					
						
							|  |  |  | 		GIOCondition  condition, | 
					
						
							|  |  |  | 		gpointer      data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char buf1[32]; | 
					
						
							|  |  |  |   char buf2[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char result[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   AddrData *addr_data = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!read_all (source, buf1, 32) || | 
					
						
							|  |  |  |       !read_all (source, buf2, 32)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       g_main_loop_quit (addr_data->loop); | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sprintf (result, "%d", atoi(buf1) + atoi(buf2)); | 
					
						
							|  |  |  |   write_all (addr_data->dest, result, 32); | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | timeout_callback (gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   AddrData *addr_data = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   addr_data->count++; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static gpointer | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | adder_thread (gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GMainContext *context; | 
					
						
							|  |  |  |   GSource *adder_source; | 
					
						
							|  |  |  |   GSource *timeout_source; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   GIOChannel **channels = data; | 
					
						
							|  |  |  |   AddrData addr_data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-30 19:56:47 +00:00
										 |  |  |   context = g_main_context_new (); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_lock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_ptr_array_add (context_array, context); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (context_array->len == NTHREADS) | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |     g_cond_broadcast (&context_array_cond); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_unlock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   addr_data.dest = channels[1]; | 
					
						
							|  |  |  |   addr_data.loop = g_main_loop_new (context, FALSE); | 
					
						
							|  |  |  |   addr_data.count = 0; | 
					
						
							| 
									
										
										
										
											2010-04-20 17:47:44 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   adder_source = g_io_create_watch (channels[0], G_IO_IN | G_IO_HUP); | 
					
						
							| 
									
										
										
										
											2010-05-26 16:21:15 -04:00
										 |  |  |   g_source_set_name (adder_source, "Adder I/O"); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_source_set_callback (adder_source, (GSourceFunc)adder_callback, &addr_data, NULL); | 
					
						
							|  |  |  |   g_source_attach (adder_source, context); | 
					
						
							| 
									
										
										
										
											2000-12-07 20:29:58 +00:00
										 |  |  |   g_source_unref (adder_source); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   timeout_source = g_timeout_source_new (10); | 
					
						
							| 
									
										
										
										
											2010-05-26 16:21:15 -04:00
										 |  |  |   g_source_set_name (timeout_source, "Adder timeout"); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_source_set_callback (timeout_source, (GSourceFunc)timeout_callback, &addr_data, NULL); | 
					
						
							|  |  |  |   g_source_set_priority (timeout_source, G_PRIORITY_HIGH); | 
					
						
							|  |  |  |   g_source_attach (timeout_source, context); | 
					
						
							| 
									
										
										
										
											2000-12-07 20:29:58 +00:00
										 |  |  |   g_source_unref (timeout_source); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-15 22:16:57 +00:00
										 |  |  |   g_main_loop_run (addr_data.loop); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   g_io_channel_unref (channels[0]); | 
					
						
							|  |  |  |   g_io_channel_unref (channels[1]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_free (channels); | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2001-01-05 21:22:47 +00:00
										 |  |  |   g_main_loop_unref (addr_data.loop); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-14 05:20:14 +00:00
										 |  |  | #ifdef VERBOSE
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_print ("Timeout run %d times\n", addr_data.count); | 
					
						
							| 
									
										
										
										
											2005-07-14 05:20:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_lock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_ptr_array_remove (context_array, context); | 
					
						
							|  |  |  |   if (context_array->len == 0) | 
					
						
							|  |  |  |     g_main_loop_quit (main_loop); | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_unlock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   cleanup_crawlers (context); | 
					
						
							| 
									
										
										
										
											2001-05-08 08:23:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return NULL; | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | io_pipe (GIOChannel **channels) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   gint fds[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (pipe(fds) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       g_warning ("Cannot create pipe %s\n", g_strerror (errno)); | 
					
						
							|  |  |  |       exit (1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   channels[0] = g_io_channel_unix_new (fds[0]); | 
					
						
							|  |  |  |   channels[1] = g_io_channel_unix_new (fds[1]); | 
					
						
							| 
									
										
										
										
											2002-10-15 22:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   g_io_channel_set_close_on_unref (channels[0], TRUE); | 
					
						
							|  |  |  |   g_io_channel_set_close_on_unref (channels[1], TRUE); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | do_add (GIOChannel *in, gint a, gint b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char buf1[32]; | 
					
						
							|  |  |  |   char buf2[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sprintf (buf1, "%d", a); | 
					
						
							|  |  |  |   sprintf (buf2, "%d", b); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   write_all (in, buf1, 32); | 
					
						
							|  |  |  |   write_all (in, buf2, 32); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | adder_response (GIOChannel   *source, | 
					
						
							|  |  |  | 		GIOCondition  condition, | 
					
						
							|  |  |  | 		gpointer      data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char result[32]; | 
					
						
							|  |  |  |   TestData *test_data = data; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   if (!read_all (source, result, 32)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   test_data->current_val = atoi (result); | 
					
						
							|  |  |  |   test_data->iters--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (test_data->iters == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (test_data->current_val != ITERS * INCREMENT) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  g_print ("Addition failed: %d != %d\n", | 
					
						
							|  |  |  | 		   test_data->current_val, ITERS * INCREMENT); | 
					
						
							|  |  |  | 	  exit (1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       g_io_channel_unref (source); | 
					
						
							|  |  |  |       g_io_channel_unref (test_data->in); | 
					
						
							| 
									
										
										
										
											2002-10-14 19:33:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       g_free (test_data); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |        | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   do_add (test_data->in, test_data->current_val, INCREMENT); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 15:45:14 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | create_adder_thread (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GError *err = NULL; | 
					
						
							|  |  |  |   TestData *test_data; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   GIOChannel *in_channels[2]; | 
					
						
							|  |  |  |   GIOChannel *out_channels[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   GIOChannel **sub_channels; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sub_channels = g_new (GIOChannel *, 2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   io_pipe (in_channels); | 
					
						
							|  |  |  |   io_pipe (out_channels); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sub_channels[0] = in_channels[0]; | 
					
						
							|  |  |  |   sub_channels[1] = out_channels[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-18 08:44:57 +00:00
										 |  |  |   g_thread_create (adder_thread, sub_channels, FALSE, &err); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (err) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       g_warning ("Cannot create thread: %s", err->message); | 
					
						
							|  |  |  |       exit (1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   test_data = g_new (TestData, 1); | 
					
						
							|  |  |  |   test_data->in = in_channels[1]; | 
					
						
							|  |  |  |   test_data->current_val = 0; | 
					
						
							|  |  |  |   test_data->iters = ITERS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_io_add_watch (out_channels[0], G_IO_IN | G_IO_HUP, | 
					
						
							|  |  |  | 		  adder_response, test_data); | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   do_add (test_data->in, test_data->current_val, INCREMENT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void create_crawler (void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | remove_crawler (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GSource *other_source; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (crawler_array->len > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       other_source = crawler_array->pdata[g_random_int_range (0, crawler_array->len)]; | 
					
						
							|  |  |  |       g_source_destroy (other_source); | 
					
						
							|  |  |  |       g_assert (g_ptr_array_remove_fast (crawler_array, other_source)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static gint | 
					
						
							|  |  |  | crawler_callback (gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GSource *source = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   G_LOCK (crawler_array_lock); | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   if (!g_ptr_array_remove_fast (crawler_array, source)) | 
					
						
							|  |  |  |     remove_crawler(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   remove_crawler(); | 
					
						
							|  |  |  |   G_UNLOCK (crawler_array_lock); | 
					
						
							|  |  |  | 	     | 
					
						
							|  |  |  |   create_crawler(); | 
					
						
							|  |  |  |   create_crawler(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | create_crawler (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GSource *source = g_timeout_source_new (g_random_int_range (0, CRAWLER_TIMEOUT_RANGE)); | 
					
						
							| 
									
										
										
										
											2010-04-20 17:47:44 -04:00
										 |  |  |   g_source_set_name (source, "Crawler timeout"); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_source_set_callback (source, (GSourceFunc)crawler_callback, source, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-02 02:56:08 +00:00
										 |  |  |   G_LOCK (crawler_array_lock); | 
					
						
							|  |  |  |   g_ptr_array_add (crawler_array, source); | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_lock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_source_attach (source, context_array->pdata[g_random_int_range (0, context_array->len)]); | 
					
						
							| 
									
										
										
										
											2000-12-07 20:29:58 +00:00
										 |  |  |   g_source_unref (source); | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_unlock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   G_UNLOCK (crawler_array_lock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | cleanup_crawlers (GMainContext *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   gint i; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   G_LOCK (crawler_array_lock); | 
					
						
							|  |  |  |   for (i=0; i < crawler_array->len; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (g_source_get_context (crawler_array->pdata[i]) == context) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  g_source_destroy (g_ptr_array_remove_index (crawler_array, i)); | 
					
						
							|  |  |  | 	  i--; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   G_UNLOCK (crawler_array_lock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static gboolean | 
					
						
							|  |  |  | recurser_idle (gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GMainContext *context = data; | 
					
						
							|  |  |  |   gint i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < 10; i++) | 
					
						
							| 
									
										
										
										
											2001-11-02 02:56:08 +00:00
										 |  |  |     g_main_context_iteration (context, FALSE); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static gboolean | 
					
						
							|  |  |  | recurser_start (gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GMainContext *context; | 
					
						
							|  |  |  |   GSource *source; | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_lock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   context = context_array->pdata[g_random_int_range (0, context_array->len)]; | 
					
						
							|  |  |  |   source = g_idle_source_new (); | 
					
						
							| 
									
										
										
										
											2010-04-20 17:47:44 -04:00
										 |  |  |   g_source_set_name (source, "Recursing idle source"); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   g_source_set_callback (source, recurser_idle, context, NULL); | 
					
						
							|  |  |  |   g_source_attach (source, context); | 
					
						
							| 
									
										
										
										
											2000-12-07 20:29:58 +00:00
										 |  |  |   g_source_unref (source); | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_unlock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-30 00:04:05 -04:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | main (int   argc, | 
					
						
							|  |  |  |       char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   gint i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   context_array = g_ptr_array_new (); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   crawler_array = g_ptr_array_new (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   main_loop = g_main_loop_new (NULL, FALSE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < NTHREADS; i++) | 
					
						
							|  |  |  |     create_adder_thread (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  |   /* Wait for all threads to start
 | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_lock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  |    | 
					
						
							|  |  |  |   if (context_array->len < NTHREADS) | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |     g_cond_wait (&context_array_cond, &context_array_mutex); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2011-10-04 19:04:41 -04:00
										 |  |  |   g_mutex_unlock (&context_array_mutex); | 
					
						
							| 
									
										
										
										
											2001-04-17 14:48:11 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  |   for (i = 0; i < NCRAWLERS; i++) | 
					
						
							|  |  |  |     create_crawler (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_timeout_add (RECURSER_TIMEOUT, recurser_start, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_main_loop_run (main_loop); | 
					
						
							| 
									
										
										
										
											2001-01-05 21:22:47 +00:00
										 |  |  |   g_main_loop_unref (main_loop); | 
					
						
							| 
									
										
										
										
											2000-12-05 20:45:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } |