os/ossrv/glib/tests/child-test.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /* GLIB - Library of useful routines for C programming
     2  * Copyright (C) 1995-1997  Peter Mattis, Spencer Kimball and Josh MacDonald
     3  *
     4  * Portions copyright (c) 2009 Nokia Corporation.  All rights reserved.
     5  * This library is free software; you can redistribute it and/or
     6  * modify it under the terms of the GNU Lesser General Public
     7  * License as published by the Free Software Foundation; either
     8  * version 2 of the License, or (at your option) any later version.
     9  *
    10  * This library is distributed in the hope that it will be useful,
    11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
    13  * Lesser General Public License for more details.
    14  *
    15  * You should have received a copy of the GNU Lesser General Public
    16  * License along with this library; if not, write to the
    17  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
    18  * Boston, MA 02111-1307, USA.
    19  */
    20 
    21 /*
    22  * Modified by the GLib Team and others 1997-2000.  See the AUTHORS
    23  * file for a list of people on the GLib Team.  See the ChangeLog
    24  * files for a list of changes.  These files are distributed with
    25  * GLib at ftp://ftp.gtk.org/pub/gtk/. 
    26  */
    27 
    28 #include "config.h"
    29 
    30 #include <sys/types.h>
    31 #ifdef HAVE_UNISTD_H
    32 #include <unistd.h>
    33 #endif
    34 #include <stdlib.h>
    35 #include <spawn.h>
    36 #include <sys/wait.h>
    37 #include <glib.h>
    38 
    39 #ifdef __SYMBIAN32__
    40 #include "mrt2_glib2_test.h"
    41 #endif /*__SYMBIAN32__*/
    42 
    43 #ifdef G_OS_WIN32
    44 #include <windows.h>
    45 #endif
    46 
    47 #ifdef G_OS_WIN32
    48 #define GPID_FORMAT "%p"
    49 #else
    50 #define GPID_FORMAT "%d"
    51 #endif
    52 
    53 GMainLoop *main_loop;
    54 gint alive;
    55 
    56 #ifdef G_OS_WIN32
    57 char *argv0;
    58 #endif
    59 
    60 GPid
    61 get_a_child (gint ttl)
    62 {
    63   GPid pid;
    64 #ifdef __SYMBIAN32__  
    65   gboolean retval;
    66   char **argv = NULL;
    67   GError *error = NULL;
    68 #endif//__SYMBIAN32__  
    69 #ifdef G_OS_WIN32
    70   STARTUPINFO si;
    71   PROCESS_INFORMATION pi;
    72   gchar *cmdline;
    73 
    74   memset (&si, 0, sizeof (si));
    75   si.cb = sizeof (&si);
    76   memset (&pi, 0, sizeof (pi));
    77 
    78   cmdline = g_strdup_printf( "child-test -c%d", ttl);
    79 
    80   if (!CreateProcess (argv0, cmdline, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi))
    81     g_error ("CreateProcess failed: %s\n", g_win32_error_message (GetLastError ()));
    82 
    83   g_free(cmdline);
    84 
    85   CloseHandle (pi.hThread);
    86   pid = pi.hProcess;
    87 
    88   return pid;
    89 #endif //G_OS_WIN32
    90 #ifndef __SYMBIAN32__
    91   pid = fork ();
    92 #else  
    93   argv = (char **)malloc(3*sizeof(char *));    
    94   argv[0] = "Helloworld.exe"; // wrong exe name. Should cause g_spawn_async to fail
    95   argv[1] = NULL;
    96   retval = g_spawn_async(NULL, argv, NULL, G_SPAWN_DO_NOT_REAP_CHILD | G_SPAWN_SEARCH_PATH, NULL, "1234", &pid, &error);
    97   if (pid < 0)
    98     exit (1);
    99 
   100   if (pid > 0)
   101     return pid;
   102 
   103   sleep (ttl);
   104   _exit (0);
   105 #endif /*__SYMBIAN32__*/
   106 }
   107 
   108 gboolean
   109 child_watch_callback (GPid pid, gint status, gpointer data)
   110 {
   111 #ifdef VERBOSE
   112   gint ttl = GPOINTER_TO_INT (data);
   113 
   114   g_print ("child " GPID_FORMAT " (ttl %d) exited, status %d\n", pid, ttl, status);
   115 #endif
   116 
   117   g_spawn_close_pid (pid);
   118 
   119   if (--alive == 0)
   120     g_main_loop_quit (main_loop);
   121 
   122   return TRUE;
   123 }
   124 
   125 static gboolean
   126 quit_loop (gpointer data)
   127 {
   128   GMainLoop *main_loop = data;
   129 
   130   g_main_loop_quit (main_loop);
   131 
   132   return TRUE;
   133 }
   134 
   135 #ifdef TEST_THREAD
   136 static gpointer
   137 test_thread (gpointer data)
   138 {
   139   GMainLoop *new_main_loop;
   140   GSource *source;
   141   GPid pid;
   142   gint ttl = GPOINTER_TO_INT (data);
   143 
   144   new_main_loop = g_main_loop_new (NULL, FALSE);
   145 
   146   pid = get_a_child (ttl);
   147   source = g_child_watch_source_new (pid);
   148   g_source_set_callback (source, (GSourceFunc) child_watch_callback, data, NULL);
   149   g_source_attach (source, g_main_loop_get_context (new_main_loop));
   150   g_source_unref (source);
   151 
   152 #ifdef VERBOSE
   153   g_print ("whee! created pid: " GPID_FORMAT " (ttl %d)\n", pid, ttl);
   154 #endif
   155 
   156   g_main_loop_run (new_main_loop);
   157 
   158   return NULL;
   159 }
   160 #endif
   161 
   162 int
   163 main (int argc, char *argv[])
   164 {
   165 #ifndef TEST_THREAD
   166   GPid pid;
   167 #endif
   168 #ifdef G_OS_WIN32
   169   argv0 = argv[0];
   170   if (argc > 1 && argv[1][0] == '-' && argv[1][1] == 'c')
   171     {
   172       int ttl = atoi (argv[1] + 2);
   173       Sleep (ttl * 1000);
   174       /* Exit on purpose with STILL_ACTIVE (which isn't a very common
   175        * exit status) to verify that g_child_watch_check() in gmain.c
   176        * doesn't believe a child still to be active if it happens to
   177        * exit with that status.
   178        */
   179       exit (STILL_ACTIVE);
   180     }
   181 #endif
   182 
   183 	#ifdef __SYMBIAN32__
   184 	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);
   185 	g_set_print_handler(mrtPrintHandler);
   186 	#endif /*__SYMBIAN32__*/
   187 
   188   /* Only run the test, if threads are enabled and a default thread
   189    * implementation is available.
   190    */
   191 #if defined(G_THREADS_ENABLED) && ! defined(G_THREADS_IMPL_NONE)
   192 #ifdef TEST_THREAD
   193   g_thread_init (NULL);
   194 #endif
   195   main_loop = g_main_loop_new (NULL, FALSE);
   196 
   197 #ifdef G_OS_WIN32
   198   system ("ipconfig /all");
   199 #else
   200   system ("/bin/true");
   201 #endif
   202 
   203   alive = 2;
   204   g_timeout_add (30000, quit_loop, main_loop);
   205 
   206 #ifdef TEST_THREAD
   207   g_thread_create (test_thread, GINT_TO_POINTER (10), FALSE, NULL);
   208   g_thread_create (test_thread, GINT_TO_POINTER (20), FALSE, NULL);
   209 #else
   210   pid = get_a_child (10);
   211   g_child_watch_add (pid, (GChildWatchFunc) child_watch_callback,
   212 		     GINT_TO_POINTER (10));
   213   pid = get_a_child (20);
   214   g_child_watch_add (pid, (GChildWatchFunc) child_watch_callback,
   215 		     GINT_TO_POINTER (20));
   216 #endif
   217   
   218   g_main_loop_run (main_loop);
   219 
   220   if (alive > 0)
   221     {
   222       g_warning ("%d children still alive\n", alive);
   223       return 1;
   224     }
   225     
   226 #endif
   227 
   228    #if __SYMBIAN32__
   229    testResultXml("child-test1");
   230    #endif
   231    
   232    return 0;
   233 }