1 /* Thread calls exec.
2    Copyright (C) 2002-2022 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4 
5    The GNU C 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.1 of the License, or (at your option) any later version.
9 
10    The GNU C 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 the GNU C Library; if not, see
17    <https://www.gnu.org/licenses/>.  */
18 
19 #include <errno.h>
20 #include <paths.h>
21 #include <pthread.h>
22 #include <signal.h>
23 #include <spawn.h>
24 #include <stdbool.h>
25 #include <stdio.h>
26 #include <stdlib.h>
27 #include <unistd.h>
28 #include <sys/wait.h>
29 #include <support/xsignal.h>
30 
31 
32 static void *
tf(void * arg)33 tf (void *arg)
34 {
35   execl (_PATH_BSHELL, _PATH_BSHELL, "-c", "echo $$", NULL);
36 
37   puts ("execl failed");
38   exit (1);
39 }
40 
41 
42 static int
do_test(void)43 do_test (void)
44 {
45   int fd[2];
46   if (pipe (fd) != 0)
47     {
48       puts ("pipe failed");
49       exit (1);
50     }
51 
52   /* Not interested in knowing when the pipe is closed.  */
53   xsignal (SIGPIPE, SIG_IGN);
54 
55   pid_t pid = fork ();
56   if (pid == -1)
57     {
58       puts ("fork failed");
59       exit (1);
60     }
61 
62   if (pid == 0)
63     {
64       /* Use the fd for stdout.  This is kind of ugly because it
65 	 substitutes the fd of stdout but we know what we are doing
66 	 here...  */
67       if (dup2 (fd[1], STDOUT_FILENO) != STDOUT_FILENO)
68 	{
69 	  puts ("dup2 failed");
70 	  exit (1);
71 	}
72 
73       close (fd[0]);
74 
75       pthread_t th;
76       if (pthread_create (&th, NULL, tf, NULL) != 0)
77 	{
78 	  puts ("create failed");
79 	  exit (1);
80 	}
81 
82       if (pthread_join (th, NULL) == 0)
83 	{
84 	  puts ("join succeeded!?");
85 	  exit (1);
86 	}
87 
88       puts ("join returned!?");
89       exit (1);
90     }
91 
92   close (fd[1]);
93 
94   char buf[200];
95   ssize_t n;
96   bool seen_pid = false;
97   while (TEMP_FAILURE_RETRY ((n = read (fd[0], buf, sizeof (buf)))) > 0)
98     {
99       /* We only expect to read the PID.  */
100       char *endp;
101       long int rpid = strtol (buf, &endp, 10);
102 
103       if (*endp != '\n')
104 	{
105 	  printf ("didn't parse whole line: \"%s\"\n", buf);
106 	  exit (1);
107 	}
108       if (endp == buf)
109 	{
110 	  puts ("read empty line");
111 	  exit (1);
112 	}
113 
114       if (rpid != pid)
115 	{
116 	  printf ("found \"%s\", expected PID %ld\n", buf, (long int) pid);
117 	  exit (1);
118 	}
119 
120       if (seen_pid)
121 	{
122 	  puts ("found more than one PID line");
123 	  exit (1);
124 	}
125       seen_pid = true;
126     }
127 
128   close (fd[0]);
129 
130   int status;
131   int err = waitpid (pid, &status, 0);
132   if (err != pid)
133     {
134       puts ("waitpid failed");
135       exit (1);
136     }
137 
138   if (!seen_pid)
139     {
140       puts ("didn't get PID");
141       exit (1);
142     }
143 
144   return 0;
145 }
146 
147 #define TEST_FUNCTION do_test ()
148 #include "../test-skeleton.c"
149