1 /* Child program invoked by test-execute-main.
2 Copyright (C) 2009-2025 Free Software Foundation, Inc.
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3, or (at your option)
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, see <https://www.gnu.org/licenses/>. */
17 /* If the user's config.h happens to include <sys/stat.h>, let it include only
18 the system's <sys/stat.h> here. */
19 #define __need_system_sys_stat_h
22 /* Get the original definition of fstat. It might be defined as a macro.
23 Also, 'stat' might be defined as a macro. */
24 #include <sys/types.h>
26 #undef __need_system_sys_stat_h
28 /* Return non-zero if FD is opened to a device. */
32 #if defined _WIN32 && ! defined __CYGWIN__
34 return _fstat (fd
, &st
) >= 0 && !((st
.st_mode
& S_IFMT
) == S_IFREG
);
37 return fstat (fd
, &st
) >= 0 && !S_ISREG (st
.st_mode
);
41 /* Now include the other header files. */
50 #if defined _WIN32 && ! defined __CYGWIN__
51 /* Get declarations of the native Windows API functions. */
52 # define WIN32_LEAN_AND_MEAN
54 /* Get _get_osfhandle, _isatty, _chdir, _getcwd. */
58 /* In this file, we use only system functions, no overrides from gnulib. */
79 /* macOS 12's "warning: 'sprintf' is deprecated" is pointless,
80 as sprintf is used safely here. */
81 #if defined __APPLE__ && defined __MACH__ && _GL_GNUC_PREREQ (4, 2)
82 # pragma GCC diagnostic ignored "-Wdeprecated-declarations"
87 #if HAVE_MSVC_INVALID_PARAMETER_HANDLER
89 gl_msvc_invalid_parameter_handler (const wchar_t *expression
,
90 const wchar_t *function
,
98 /* Return non-zero if FD is open. */
102 #if defined _WIN32 && ! defined __CYGWIN__
103 /* On native Windows, the initial state of unassigned standard file
104 descriptors is that they are open but point to an
105 INVALID_HANDLE_VALUE, and there is no fcntl. */
106 return (HANDLE
) _get_osfhandle (fd
) != INVALID_HANDLE_VALUE
;
109 # error Please port fcntl to your platform
111 return 0 <= fcntl (fd
, F_GETFL
);
116 main (int argc
, char *argv
[])
119 /* Check an invocation without arguments. Check the exit code. */
122 int test
= atoi (argv
[1]);
126 /* Check argument passing. */
128 && strcmp (argv
[2], "abc def") == 0
129 && strcmp (argv
[3], "abc\"def\"ghi") == 0
130 && strcmp (argv
[4], "xyz\"") == 0
131 && strcmp (argv
[5], "abc\\def\\ghi") == 0
132 && strcmp (argv
[6], "xyz\\") == 0
133 && strcmp (argv
[7], "???") == 0
134 && strcmp (argv
[8], "***") == 0
135 && strcmp (argv
[9], "") == 0
136 && strcmp (argv
[10], "foo") == 0
137 && strcmp (argv
[11], "") == 0);
138 #if !(defined _WIN32 && !defined __CYGWIN__)
140 /* Check SIGPIPE handling with ignore_sigpipe = false. */
142 /* Check SIGPIPE handling with ignore_sigpipe = true. */
147 /* Check other signal. */
151 /* Check stdin is inherited. */
152 return !(fgetc (stdin
) == 'F' && fgetc (stdin
) == 'o');
154 /* Check null_stdin = true. */
155 return !(fgetc (stdin
) == EOF
);
157 /* Check stdout is inherited, part 1 (regular file). */
158 return !(fputs ("bar", stdout
) != EOF
&& fflush (stdout
) == 0);
160 /* Check stdout is inherited, part 2 (device). */
162 /* Check null_stdout = true. */
163 return !is_device (STDOUT_FILENO
);
165 /* Check stderr is inherited, part 1 (regular file). */
166 return !(fputs ("bar", stderr
) != EOF
&& fflush (stderr
) == 0);
168 /* Check stderr is inherited, part 2 (device). */
170 /* Check null_stderr = true. */
171 return !is_device (STDERR_FILENO
);
174 /* Check file descriptors >= 3 can be inherited. */
176 /* Check file descriptors >= 3 with O_CLOEXEC bit are not inherited. */
177 #if HAVE_MSVC_INVALID_PARAMETER_HANDLER
178 /* Avoid exceptions from within _get_osfhandle. */
179 _set_invalid_parameter_handler (gl_msvc_invalid_parameter_handler
);
182 /* QEMU 6.1 in user-mode passes an open fd = 3, that references
183 /dev/urandom. We need to ignore this fd. */
184 bool is_qemu
= is_running_under_qemu_user ();
189 for (fd
= 0; fd
< 20; fd
++)
190 if (is_open (fd
) && !(is_qemu
&& fd
== 3))
192 sprintf (p
, "%d ", fd
);
195 const char *expected
= (test
< 16 ? "0 1 2 10 " : "0 1 2 ");
196 if (strcmp (buf
, expected
) == 0)
200 fprintf (stderr
, "Test case %d: %s\n", test
, buf
); fflush (stderr
);
205 /* Check that file descriptors >= 3, open for reading, can be inherited,
206 including the file position. */
209 int n
= read (15, buf
, sizeof (buf
));
210 return !(n
== 4 && memcmp (buf
, "obar", 4) == 0);
213 /* Check that file descriptors >= 3, open for writing, can be inherited,
214 including the file position. */
216 int n
= write (15, "bar", 3);
220 /* Check that file descriptors >= 3, when inherited, preserve their
221 isatty() property, part 1 (regular file). */
223 /* Check that file descriptors >= 3, when inherited, preserve their
224 isatty() property, part 2 (character devices). */
226 #if defined _WIN32 && ! defined __CYGWIN__
227 return 4 + 2 * (_isatty (15) != 0) + (_isatty (16) != 0);
229 return 4 + 2 * (isatty (15) != 0) + (isatty (16) != 0);
233 /* Check execution in a different directory. */
236 #if defined _WIN32 && ! defined __CYGWIN__
237 if (_chdir ("..") != 0)
239 if (_getcwd (cwd
, sizeof (cwd
)) == NULL
)
242 if (chdir ("..") != 0)
244 if (getcwd (cwd
, sizeof (cwd
)) == NULL
)
247 return (argc
== 3 && strcmp (argv
[2], cwd
) == 0 ? 0 : 3);