os/ossrv/glib/tests/timeloop-closure.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /* Portion Copyright © 2008-09 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.*/
     2 #undef G_DISABLE_ASSERT
     3 #undef G_LOG_DOMAIN
     4 
     5 #include <errno.h>
     6 #include <stdlib.h>
     7 #include <unistd.h>
     8 #include <stdio.h>
     9 #include <sys/time.h>
    10 #include <sys/resource.h>
    11 
    12 #include <glib.h>
    13 #include <glib-object.h>
    14 
    15 #ifdef __SYMBIAN32__
    16 #include "mrt2_glib2_test.h"
    17 #endif /*__SYMBIAN32__*/
    18 
    19 
    20 static int n_children = 3;
    21 static int n_active_children;
    22 static int n_iters = 10000;
    23 static GMainLoop *loop;
    24 
    25 static void
    26 io_pipe (GIOChannel **channels)
    27 {
    28   int fds[2];
    29 
    30   if (pipe(fds) < 0)
    31     {
    32       fprintf (stderr, "Cannot create pipe %s\n", g_strerror (errno));
    33       exit (1);
    34     }
    35 
    36   channels[0] = g_io_channel_unix_new (fds[0]);
    37   channels[1] = g_io_channel_unix_new (fds[1]);
    38 }
    39 
    40 static gboolean
    41 read_all (GIOChannel *channel, char *buf, int len)
    42 {
    43   gsize bytes_read = 0;
    44   gsize count;
    45   GIOError err;
    46 
    47   while (bytes_read < len)
    48     {
    49       err = g_io_channel_read (channel, buf + bytes_read, len - bytes_read, &count);
    50       if (err)
    51 	{
    52 	  if (err != G_IO_ERROR_AGAIN)
    53 	    return FALSE;
    54 	}
    55       else if (count == 0)
    56 	return FALSE;
    57 
    58       bytes_read += count;
    59     }
    60 
    61   return TRUE;
    62 }
    63 
    64 static gboolean
    65 write_all (GIOChannel *channel, char *buf, int len)
    66 {
    67   gsize bytes_written = 0;
    68   gsize count;
    69   GIOError err;
    70 
    71   while (bytes_written < len)
    72     {
    73       err = g_io_channel_write (channel, buf + bytes_written, len - bytes_written, &count);
    74       if (err && err != G_IO_ERROR_AGAIN)
    75 	return FALSE;
    76 
    77       bytes_written += count;
    78     }
    79 
    80   return TRUE;
    81 }
    82 
    83 #ifndef __SYMBIAN32__
    84 static void
    85 run_child (GIOChannel *in_channel, GIOChannel *out_channel)
    86 #else
    87 gpointer
    88 run_child (gpointer data)
    89 #endif//__SYMBIAN32__
    90 {
    91   int i;
    92   int val = 1;
    93 #ifdef  __SYMBIAN32__ 
    94   GIOChannel *in_channel,*out_channel;
    95 #endif//__SYMBIAN32__  
    96   GTimer *timer = g_timer_new();
    97 #ifdef __SYMBIAN32__  
    98   GIOChannel **channels = data;
    99   
   100   in_channel = channels[0];
   101   out_channel = channels[1];
   102 #endif//__SYMBIAN32__
   103   for (i = 0; i < n_iters; i++)
   104     {
   105       write_all (out_channel, (char *)&val, sizeof (val));
   106       read_all (in_channel, (char *)&val, sizeof (val));
   107     }
   108 
   109   val = 0;
   110   write_all (out_channel, (char *)&val, sizeof (val));
   111 
   112   val = g_timer_elapsed (timer, NULL) * 1000;
   113   
   114   write_all (out_channel, (char *)&val, sizeof (val));
   115   g_timer_destroy (timer);
   116 #ifndef  __SYMBIAN32__
   117   exit (0);
   118 #else
   119   return NULL;
   120 #endif//__SYMBIAN32__    
   121 }
   122 
   123 static gboolean
   124 input_callback (GIOChannel   *source,
   125 		GIOCondition  condition,
   126 		gpointer      data)
   127 {
   128   int val;
   129   GIOChannel *dest = (GIOChannel *)data;
   130   
   131   if (!read_all (source, (char *)&val, sizeof(val)))
   132     {
   133       fprintf (stderr, "Unexpected EOF\n");
   134       exit (1);
   135     }
   136 
   137   if (val)
   138     {
   139       write_all (dest, (char *)&val, sizeof(val));
   140       
   141       return TRUE;
   142     }
   143   else
   144     {
   145       g_io_channel_close (source);
   146       g_io_channel_close (dest);
   147       
   148       g_io_channel_unref (source);
   149       g_io_channel_unref (dest);
   150 
   151       n_active_children--;
   152       if (n_active_children == 0)
   153 	g_main_loop_quit (loop);
   154       
   155       return FALSE;
   156     }
   157 }
   158 
   159 static void
   160 create_child (void)
   161 {
   162   GError *err;
   163   GIOChannel *in_channels[2];
   164   GIOChannel *out_channels[2];
   165   GIOChannel **sub_channels;
   166   GSource *source;
   167   
   168   sub_channels = g_new (GIOChannel *, 2);
   169   
   170   io_pipe (in_channels);
   171   io_pipe (out_channels);
   172   
   173   sub_channels[0] = in_channels[0];
   174   sub_channels[1] = out_channels[1];
   175 
   176   source = g_io_create_watch (out_channels[0], G_IO_IN | G_IO_HUP);
   177   g_assert(source != NULL);
   178   g_source_set_closure (source,
   179   g_cclosure_new (G_CALLBACK (input_callback), in_channels[1], NULL));
   180   g_source_attach (source, NULL);
   181 
   182   g_thread_create(run_child,sub_channels,FALSE,&err);
   183 
   184 }
   185 
   186 static double 
   187 difftimeval (struct timeval *old, struct timeval *new)
   188 {
   189   return
   190     (new->tv_sec - old->tv_sec) * 1000. + (new->tv_usec - old->tv_usec) / 1000;
   191 }
   192 
   193 int 
   194 main (int argc, char **argv)
   195 {
   196   int i;
   197 
   198   #ifdef __SYMBIAN32__
   199   g_log_set_handler (NULL,  G_LOG_FLAG_FATAL| G_LOG_FLAG_RECURSION | G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_WARNING | G_LOG_LEVEL_MESSAGE | G_LOG_LEVEL_INFO | G_LOG_LEVEL_DEBUG, &mrtLogHandler, NULL);
   200   g_set_print_handler(mrtPrintHandler);
   201   #endif /*__SYMBIAN32__*/
   202    
   203   g_thread_init(NULL);
   204 	  
   205   g_type_init ();
   206   
   207   if (argc > 1)
   208     n_children = atoi(argv[1]);
   209 
   210   if (argc > 2)
   211     n_iters = atoi(argv[2]);
   212 
   213   printf ("Children: %d     Iters: %d\n", n_children, n_iters);
   214 
   215   n_active_children = n_children;
   216   for (i = 0; i < n_children; i++)
   217     create_child ();
   218 
   219   loop = g_main_loop_new (NULL, FALSE);
   220   
   221   g_assert(loop != NULL);
   222   
   223   g_main_loop_run (loop);
   224   
   225   g_assert(n_active_children == 0);
   226   
   227   
   228 #ifdef __SYMBIAN32__
   229   testResultXml("timeloop-closure");
   230 #endif /* EMULATOR */
   231   return 0;
   232 }