usleep tests: Avoid failure due to known Cygwin 3.5.3 bug.
[gnulib.git] / tests / test-posix_spawn-open1.c
blob0f83e3407807d7e12fd1c5ad2e29256e916fc983
1 /* Test of posix_spawn() function with 'open' action.
2 Copyright (C) 2008-2024 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 of the License, or
7 (at your option) any later version.
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 /* Written by Bruno Haible <bruno@clisp.org>, 2008. */
19 /* Test whether posix_spawn_file_actions_addopen supports filename arguments
20 that contain special characters such as '*'. */
22 #include <config.h>
24 #include <spawn.h>
26 #include "signature.h"
27 SIGNATURE_CHECK (posix_spawn, int, (pid_t *, char const *,
28 posix_spawn_file_actions_t const *,
29 posix_spawnattr_t const *,
30 char *const[], char *const[]));
31 #include <errno.h>
32 #include <fcntl.h>
33 #include <signal.h>
34 #include <stdio.h>
35 #include <string.h>
36 #include <unistd.h>
37 #include <sys/types.h>
38 #include <sys/wait.h>
40 #define CHILD_PROGRAM_FILENAME "test-posix_spawn-open1" EXEEXT
41 #define DATA_FILENAME "t!#$%&'()*+,-;=?@[\\]^_`{|}~.tmp"
42 /* On Windows (including Cygwin), '*' '?' '\\' '|' cannot be used in file
43 names. */
44 #if defined _WIN32 || defined __CYGWIN__
45 # undef DATA_FILENAME
46 # define DATA_FILENAME "t!#$%&'()+,-;=@[]^_`{}~.tmp"
47 #endif
49 static int
50 parent_main (void)
52 FILE *fp;
53 char *argv[3] = { CHILD_PROGRAM_FILENAME, "-child", NULL };
54 posix_spawn_file_actions_t actions;
55 bool actions_allocated;
56 int err;
57 pid_t child;
58 int status;
59 int exitstatus;
61 /* Create a data file with specific contents. */
62 fp = fopen (DATA_FILENAME, "wb");
63 if (fp == NULL)
65 perror ("cannot create data file");
66 return 1;
68 fwrite ("Halle Potta", 1, 11, fp);
69 if (fflush (fp) || fclose (fp))
71 perror ("cannot prepare data file");
72 return 1;
75 /* Avoid reading from our stdin, as it could block. */
76 if (freopen ("/dev/null", "rb", stdin) == NULL)
78 perror ("cannot redirect stdin");
79 return 1;
82 /* Test whether posix_spawn_file_actions_addopen with this file name
83 actually works, by spawning a child that reads from this file. */
84 actions_allocated = false;
85 if ((err = posix_spawn_file_actions_init (&actions)) != 0
86 || (actions_allocated = true,
87 (err = posix_spawn_file_actions_addopen (&actions, STDIN_FILENO, DATA_FILENAME, O_RDONLY, 0600)) != 0
88 || (err = posix_spawn (&child, CHILD_PROGRAM_FILENAME, &actions, NULL, argv, environ)) != 0))
90 if (actions_allocated)
91 posix_spawn_file_actions_destroy (&actions);
92 errno = err;
93 perror ("subprocess failed");
94 return 1;
96 posix_spawn_file_actions_destroy (&actions);
97 status = 0;
98 while (waitpid (child, &status, 0) != child)
100 if (!WIFEXITED (status))
102 fprintf (stderr, "subprocess terminated with unexpected wait status %d\n", status);
103 return 1;
105 exitstatus = WEXITSTATUS (status);
106 if (exitstatus != 0)
108 fprintf (stderr, "subprocess terminated with unexpected exit status %d\n", exitstatus);
109 return 1;
111 return 0;
114 static int
115 child_main (void)
117 char buf[1024];
119 /* See if reading from STDIN_FILENO yields the expected contents. */
120 if (fread (buf, 1, sizeof (buf), stdin) == 11
121 && memcmp (buf, "Halle Potta", 11) == 0)
122 return 0;
123 else
124 return 2;
127 static void
128 cleanup_then_die (int sig)
130 /* Clean up data file. */
131 unlink (DATA_FILENAME);
133 /* Re-raise the signal and die from it. */
134 signal (sig, SIG_DFL);
135 raise (sig);
139 main (int argc, char *argv[])
141 int exitstatus;
143 if (!(argc > 1 && strcmp (argv[1], "-child") == 0))
145 /* This is the parent process. */
146 signal (SIGINT, cleanup_then_die);
147 signal (SIGTERM, cleanup_then_die);
148 #ifdef SIGHUP
149 signal (SIGHUP, cleanup_then_die);
150 #endif
152 exitstatus = parent_main ();
154 else
156 /* This is the child process. */
158 exitstatus = child_main ();
160 unlink (DATA_FILENAME);
161 return exitstatus;