| /* Skeleton for test programs. | 
 |    Copyright (C) 1998-2016 Free Software Foundation, Inc. | 
 |    This file is part of the GNU C Library. | 
 |    Contributed by Ulrich Drepper <drepper@cygnus.com>, 1998. | 
 |  | 
 |    The GNU C Library is free software; you can redistribute it and/or | 
 |    modify it under the terms of the GNU Lesser General Public | 
 |    License as published by the Free Software Foundation; either | 
 |    version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |    The GNU C Library is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |    Lesser General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU Lesser General Public | 
 |    License along with the GNU C Library; if not, see | 
 |    <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include <assert.h> | 
 | #include <errno.h> | 
 | #include <fcntl.h> | 
 | #include <getopt.h> | 
 | #include <malloc.h> | 
 | #include <paths.h> | 
 | #include <search.h> | 
 | #include <signal.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 | #include <unistd.h> | 
 | #include <sys/resource.h> | 
 | #include <sys/wait.h> | 
 | #include <sys/param.h> | 
 | #include <time.h> | 
 |  | 
 | /* The test function is normally called `do_test' and it is called | 
 |    with argc and argv as the arguments.  We nevertheless provide the | 
 |    possibility to overwrite this name.  */ | 
 | #ifndef TEST_FUNCTION | 
 | # define TEST_FUNCTION do_test (argc, argv) | 
 | #endif | 
 |  | 
 | #ifndef TEST_DATA_LIMIT | 
 | # define TEST_DATA_LIMIT (64 << 20) /* Data limit (bytes) to run with.  */ | 
 | #endif | 
 |  | 
 | #ifndef TIMEOUT | 
 |   /* Default timeout is two seconds.  */ | 
 | # define TIMEOUT 2 | 
 | #endif | 
 |  | 
 | #define OPT_DIRECT 1000 | 
 | #define OPT_TESTDIR 1001 | 
 |  | 
 | static struct option options[] = | 
 | { | 
 | #ifdef CMDLINE_OPTIONS | 
 |   CMDLINE_OPTIONS | 
 | #endif | 
 |   { "direct", no_argument, NULL, OPT_DIRECT }, | 
 |   { "test-dir", required_argument, NULL, OPT_TESTDIR }, | 
 |   { NULL, 0, NULL, 0 } | 
 | }; | 
 |  | 
 | /* PID of the test itself.  */ | 
 | static pid_t pid; | 
 |  | 
 | /* Directory to place temporary files in.  */ | 
 | static const char *test_dir; | 
 |  | 
 | /* List of temporary files.  */ | 
 | struct temp_name_list | 
 | { | 
 |   struct qelem q; | 
 |   char *name; | 
 | } *temp_name_list; | 
 |  | 
 | /* Add temporary files in list.  */ | 
 | static void | 
 | __attribute__ ((unused)) | 
 | add_temp_file (const char *name) | 
 | { | 
 |   struct temp_name_list *newp | 
 |     = (struct temp_name_list *) calloc (sizeof (*newp), 1); | 
 |   char *newname = strdup (name); | 
 |   if (newp != NULL && newname != NULL) | 
 |     { | 
 |       newp->name = newname; | 
 |       if (temp_name_list == NULL) | 
 | 	temp_name_list = (struct temp_name_list *) &newp->q; | 
 |       else | 
 | 	insque (newp, temp_name_list); | 
 |     } | 
 |   else | 
 |     free (newp); | 
 | } | 
 |  | 
 | /* Delete all temporary files.  */ | 
 | static void | 
 | delete_temp_files (void) | 
 | { | 
 |   while (temp_name_list != NULL) | 
 |     { | 
 |       remove (temp_name_list->name); | 
 |       free (temp_name_list->name); | 
 |  | 
 |       struct temp_name_list *next | 
 | 	= (struct temp_name_list *) temp_name_list->q.q_forw; | 
 |       free (temp_name_list); | 
 |       temp_name_list = next; | 
 |     } | 
 | } | 
 |  | 
 | /* Create a temporary file.  Return the opened file descriptor on | 
 |    success, or -1 on failure.  Write the file name to *FILENAME if | 
 |    FILENAME is not NULL.  In this case, the caller is expected to free | 
 |    *FILENAME.  */ | 
 | static int | 
 | __attribute__ ((unused)) | 
 | create_temp_file (const char *base, char **filename) | 
 | { | 
 |   char *fname; | 
 |   int fd; | 
 |  | 
 |   fname = (char *) malloc (strlen (test_dir) + 1 + strlen (base) | 
 | 			   + sizeof ("XXXXXX")); | 
 |   if (fname == NULL) | 
 |     { | 
 |       puts ("out of memory"); | 
 |       return -1; | 
 |     } | 
 |   strcpy (stpcpy (stpcpy (stpcpy (fname, test_dir), "/"), base), "XXXXXX"); | 
 |  | 
 |   fd = mkstemp (fname); | 
 |   if (fd == -1) | 
 |     { | 
 |       printf ("cannot open temporary file '%s': %m\n", fname); | 
 |       free (fname); | 
 |       return -1; | 
 |     } | 
 |  | 
 |   add_temp_file (fname); | 
 |   if (filename != NULL) | 
 |     *filename = fname; | 
 |   else | 
 |     free (fname); | 
 |  | 
 |   return fd; | 
 | } | 
 |  | 
 | /* Timeout handler.  We kill the child and exit with an error.  */ | 
 | static void | 
 | __attribute__ ((noreturn)) | 
 | signal_handler (int sig __attribute__ ((unused))) | 
 | { | 
 |   int killed; | 
 |   int status; | 
 |  | 
 |   assert (pid > 1); | 
 |   /* Kill the whole process group.  */ | 
 |   kill (-pid, SIGKILL); | 
 |   /* In case setpgid failed in the child, kill it individually too.  */ | 
 |   kill (pid, SIGKILL); | 
 |  | 
 |   /* Wait for it to terminate.  */ | 
 |   int i; | 
 |   for (i = 0; i < 5; ++i) | 
 |     { | 
 |       killed = waitpid (pid, &status, WNOHANG|WUNTRACED); | 
 |       if (killed != 0) | 
 | 	break; | 
 |  | 
 |       /* Delay, give the system time to process the kill.  If the | 
 | 	 nanosleep() call return prematurely, all the better.  We | 
 | 	 won't restart it since this probably means the child process | 
 | 	 finally died.  */ | 
 |       struct timespec ts; | 
 |       ts.tv_sec = 0; | 
 |       ts.tv_nsec = 100000000; | 
 |       nanosleep (&ts, NULL); | 
 |     } | 
 |   if (killed != 0 && killed != pid) | 
 |     { | 
 |       printf ("Failed to kill test process: %m\n"); | 
 |       exit (1); | 
 |     } | 
 |  | 
 | #ifdef CLEANUP_HANDLER | 
 |   CLEANUP_HANDLER; | 
 | #endif | 
 |  | 
 |   if (sig == SIGINT) | 
 |     { | 
 |       signal (sig, SIG_DFL); | 
 |       raise (sig); | 
 |     } | 
 |  | 
 |   /* If we expected this signal: good!  */ | 
 | #ifdef EXPECTED_SIGNAL | 
 |   if (EXPECTED_SIGNAL == SIGALRM) | 
 |     exit (0); | 
 | #endif | 
 |  | 
 |   if (killed == 0 || (WIFSIGNALED (status) && WTERMSIG (status) == SIGKILL)) | 
 |     puts ("Timed out: killed the child process"); | 
 |   else if (WIFSTOPPED (status)) | 
 |     printf ("Timed out: the child process was %s\n", | 
 | 	    strsignal (WSTOPSIG (status))); | 
 |   else if (WIFSIGNALED (status)) | 
 |     printf ("Timed out: the child process got signal %s\n", | 
 | 	    strsignal (WTERMSIG (status))); | 
 |   else | 
 |     printf ("Timed out: killed the child process but it exited %d\n", | 
 | 	    WEXITSTATUS (status)); | 
 |  | 
 |   /* Exit with an error.  */ | 
 |   exit (1); | 
 | } | 
 |  | 
 | /* Avoid all the buffer overflow messages on stderr.  */ | 
 | static void | 
 | __attribute__ ((unused)) | 
 | ignore_stderr (void) | 
 | { | 
 |   int fd = open (_PATH_DEVNULL, O_WRONLY); | 
 |   if (fd == -1) | 
 |     close (STDERR_FILENO); | 
 |   else | 
 |     { | 
 |       dup2 (fd, STDERR_FILENO); | 
 |       close (fd); | 
 |     } | 
 |   setenv ("LIBC_FATAL_STDERR_", "1", 1); | 
 | } | 
 |  | 
 | /* Set fortification error handler.  Used when tests want to verify that bad | 
 |    code is caught by the library.  */ | 
 | static void | 
 | __attribute__ ((unused)) | 
 | set_fortify_handler (void (*handler) (int sig)) | 
 | { | 
 |   struct sigaction sa; | 
 |  | 
 |   sa.sa_handler = handler; | 
 |   sa.sa_flags = 0; | 
 |   sigemptyset (&sa.sa_mask); | 
 |  | 
 |   sigaction (SIGABRT, &sa, NULL); | 
 |   ignore_stderr (); | 
 | } | 
 |  | 
 | /* Show people how to run the program.  */ | 
 | static void | 
 | usage (void) | 
 | { | 
 |   size_t i; | 
 |  | 
 |   printf ("Usage: %s [options]\n" | 
 | 	  "\n" | 
 | 	  "Environment Variables:\n" | 
 | 	  "  TIMEOUTFACTOR          An integer used to scale the timeout\n" | 
 | 	  "  TMPDIR                 Where to place temporary files\n" | 
 | 	  "\n", | 
 | 	  program_invocation_short_name); | 
 |   printf ("Options:\n"); | 
 |   for (i = 0; options[i].name; ++i) | 
 |     { | 
 |       int indent; | 
 |  | 
 |       indent = printf ("  --%s", options[i].name); | 
 |       if (options[i].has_arg == required_argument) | 
 | 	indent += printf (" <arg>"); | 
 |       printf ("%*s", 25 - indent, ""); | 
 |       switch (options[i].val) | 
 | 	{ | 
 | 	case OPT_DIRECT: | 
 | 	  printf ("Run the test directly (instead of forking & monitoring)"); | 
 | 	  break; | 
 | 	case OPT_TESTDIR: | 
 | 	  printf ("Override the TMPDIR env var"); | 
 | 	  break; | 
 | 	} | 
 |       printf ("\n"); | 
 |     } | 
 | } | 
 |  | 
 | /* We provide the entry point here.  */ | 
 | int | 
 | main (int argc, char *argv[]) | 
 | { | 
 |   int direct = 0;	/* Directly call the test function?  */ | 
 |   int status; | 
 |   int opt; | 
 |   unsigned int timeoutfactor = 1; | 
 |   pid_t termpid; | 
 |  | 
 |   /* Make uses of freed and uninitialized memory known.  */ | 
 |   mallopt (M_PERTURB, 42); | 
 |  | 
 | #ifdef STDOUT_UNBUFFERED | 
 |   setbuf (stdout, NULL); | 
 | #endif | 
 |  | 
 |   while ((opt = getopt_long (argc, argv, "+", options, NULL)) != -1) | 
 |     switch (opt) | 
 |       { | 
 |       case '?': | 
 | 	usage (); | 
 | 	exit (1); | 
 |       case OPT_DIRECT: | 
 | 	direct = 1; | 
 | 	break; | 
 |       case OPT_TESTDIR: | 
 | 	test_dir = optarg; | 
 | 	break; | 
 | #ifdef CMDLINE_PROCESS | 
 | 	CMDLINE_PROCESS | 
 | #endif | 
 |       } | 
 |  | 
 |   /* If set, read the test TIMEOUTFACTOR value from the environment. | 
 |      This value is used to scale the default test timeout values. */ | 
 |   char *envstr_timeoutfactor = getenv ("TIMEOUTFACTOR"); | 
 |   if (envstr_timeoutfactor != NULL) | 
 |     { | 
 |       char *envstr_conv = envstr_timeoutfactor; | 
 |       unsigned long int env_fact; | 
 |  | 
 |       env_fact = strtoul (envstr_timeoutfactor, &envstr_conv, 0); | 
 |       if (*envstr_conv == '\0' && envstr_conv != envstr_timeoutfactor) | 
 | 	timeoutfactor = MAX (env_fact, 1); | 
 |     } | 
 |  | 
 |   /* Set TMPDIR to specified test directory.  */ | 
 |   if (test_dir != NULL) | 
 |     { | 
 |       setenv ("TMPDIR", test_dir, 1); | 
 |  | 
 |       if (chdir (test_dir) < 0) | 
 | 	{ | 
 | 	  printf ("chdir: %m\n"); | 
 | 	  exit (1); | 
 | 	} | 
 |     } | 
 |   else | 
 |     { | 
 |       test_dir = getenv ("TMPDIR"); | 
 |       if (test_dir == NULL || test_dir[0] == '\0') | 
 | 	test_dir = "/tmp"; | 
 |     } | 
 |  | 
 |   /* Make sure we see all message, even those on stdout.  */ | 
 |   setvbuf (stdout, NULL, _IONBF, 0); | 
 |  | 
 |   /* Make sure temporary files are deleted.  */ | 
 |   atexit (delete_temp_files); | 
 |  | 
 |   /* Correct for the possible parameters.  */ | 
 |   argv[optind - 1] = argv[0]; | 
 |   argv += optind - 1; | 
 |   argc -= optind - 1; | 
 |  | 
 |   /* Call the initializing function, if one is available.  */ | 
 | #ifdef PREPARE | 
 |   PREPARE (argc, argv); | 
 | #endif | 
 |  | 
 |   const char *envstr_direct = getenv ("TEST_DIRECT"); | 
 |   if (envstr_direct != NULL) | 
 |     { | 
 |       FILE *f = fopen (envstr_direct, "w"); | 
 |       if (f == NULL) | 
 |         { | 
 |           printf ("cannot open TEST_DIRECT output file '%s': %m\n", | 
 |                   envstr_direct); | 
 |           exit (1); | 
 |         } | 
 |  | 
 |       fprintf (f, "timeout=%u\ntimeoutfactor=%u\n", TIMEOUT, timeoutfactor); | 
 | #ifdef EXPECTED_STATUS | 
 |       fprintf (f, "exit=%u\n", EXPECTED_STATUS); | 
 | #endif | 
 | #ifdef EXPECTED_SIGNAL | 
 |       switch (EXPECTED_SIGNAL) | 
 |         { | 
 |         default: abort (); | 
 | # define init_sig(signo, name, text) \ | 
 |         case signo: fprintf (f, "signal=%s\n", name); break; | 
 | # include <siglist.h> | 
 | # undef init_sig | 
 |         } | 
 | #endif | 
 |  | 
 |       if (temp_name_list != NULL) | 
 |         { | 
 |           struct temp_name_list *n; | 
 |           fprintf (f, "temp_files=(\n"); | 
 |           for (n = temp_name_list; | 
 |                n != NULL; | 
 |                n = (struct temp_name_list *) n->q.q_forw) | 
 |             fprintf (f, "  '%s'\n", n->name); | 
 |           fprintf (f, ")\n"); | 
 |         } | 
 |  | 
 |       fclose (f); | 
 |       direct = 1; | 
 |     } | 
 |  | 
 |   /* If we are not expected to fork run the function immediately.  */ | 
 |   if (direct) | 
 |     return TEST_FUNCTION; | 
 |  | 
 |   /* Set up the test environment: | 
 |      - prevent core dumps | 
 |      - set up the timer | 
 |      - fork and execute the function.  */ | 
 |  | 
 |   pid = fork (); | 
 |   if (pid == 0) | 
 |     { | 
 |       /* This is the child.  */ | 
 | #ifdef RLIMIT_CORE | 
 |       /* Try to avoid dumping core.  */ | 
 |       struct rlimit core_limit; | 
 |       core_limit.rlim_cur = 0; | 
 |       core_limit.rlim_max = 0; | 
 |       setrlimit (RLIMIT_CORE, &core_limit); | 
 | #endif | 
 |  | 
 | #ifdef RLIMIT_DATA | 
 |       /* Try to avoid eating all memory if a test leaks.  */ | 
 |       struct rlimit data_limit; | 
 |       if (getrlimit (RLIMIT_DATA, &data_limit) == 0) | 
 | 	{ | 
 | 	  if (TEST_DATA_LIMIT == RLIM_INFINITY) | 
 | 	    data_limit.rlim_cur = data_limit.rlim_max; | 
 | 	  else if (data_limit.rlim_cur > (rlim_t) TEST_DATA_LIMIT) | 
 | 	    data_limit.rlim_cur = MIN ((rlim_t) TEST_DATA_LIMIT, | 
 | 				       data_limit.rlim_max); | 
 | 	  if (setrlimit (RLIMIT_DATA, &data_limit) < 0) | 
 | 	    printf ("setrlimit: RLIMIT_DATA: %m\n"); | 
 | 	} | 
 |       else | 
 | 	printf ("getrlimit: RLIMIT_DATA: %m\n"); | 
 | #endif | 
 |  | 
 |       /* We put the test process in its own pgrp so that if it bogusly | 
 | 	 generates any job control signals, they won't hit the whole build.  */ | 
 |       if (setpgid (0, 0) != 0) | 
 | 	printf ("Failed to set the process group ID: %m\n"); | 
 |  | 
 |       /* Execute the test function and exit with the return value.   */ | 
 |       exit (TEST_FUNCTION); | 
 |     } | 
 |   else if (pid < 0) | 
 |     { | 
 |       printf ("Cannot fork test program: %m\n"); | 
 |       exit (1); | 
 |     } | 
 |  | 
 |   /* Set timeout.  */ | 
 |   signal (SIGALRM, signal_handler); | 
 |   alarm (TIMEOUT * timeoutfactor); | 
 |  | 
 |   /* Make sure we clean up if the wrapper gets interrupted.  */ | 
 |   signal (SIGINT, signal_handler); | 
 |  | 
 |   /* Wait for the regular termination.  */ | 
 |   termpid = TEMP_FAILURE_RETRY (waitpid (pid, &status, 0)); | 
 |   if (termpid == -1) | 
 |     { | 
 |       printf ("Waiting for test program failed: %m\n"); | 
 |       exit (1); | 
 |     } | 
 |   if (termpid != pid) | 
 |     { | 
 |       printf ("Oops, wrong test program terminated: expected %ld, got %ld\n", | 
 | 	      (long int) pid, (long int) termpid); | 
 |       exit (1); | 
 |     } | 
 |  | 
 |   /* Process terminated normaly without timeout etc.  */ | 
 |   if (WIFEXITED (status)) | 
 |     { | 
 | #ifndef EXPECTED_STATUS | 
 | # ifndef EXPECTED_SIGNAL | 
 |       /* Simply exit with the return value of the test.  */ | 
 |       return WEXITSTATUS (status); | 
 | # else | 
 |       printf ("Expected signal '%s' from child, got none\n", | 
 | 	      strsignal (EXPECTED_SIGNAL)); | 
 |       exit (1); | 
 | # endif | 
 | #else | 
 |       if (WEXITSTATUS (status) != EXPECTED_STATUS) | 
 |         { | 
 |           printf ("Expected status %d, got %d\n", | 
 | 	          EXPECTED_STATUS, WEXITSTATUS (status)); | 
 |           exit (1); | 
 |         } | 
 |  | 
 |       return 0; | 
 | #endif | 
 |     } | 
 |   /* Process was killed by timer or other signal.  */ | 
 |   else | 
 |     { | 
 | #ifndef EXPECTED_SIGNAL | 
 |       printf ("Didn't expect signal from child: got `%s'\n", | 
 | 	      strsignal (WTERMSIG (status))); | 
 |       exit (1); | 
 | #else | 
 |       if (WTERMSIG (status) != EXPECTED_SIGNAL) | 
 | 	{ | 
 | 	  printf ("Incorrect signal from child: got `%s', need `%s'\n", | 
 | 		  strsignal (WTERMSIG (status)), | 
 | 		  strsignal (EXPECTED_SIGNAL)); | 
 | 	  exit (1); | 
 | 	} | 
 |  | 
 |       return 0; | 
 | #endif | 
 |     } | 
 | } |