1 /* Test of posix_spawn() function with 'chdir' action.
2 Copyright (C) 2008-2020 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>, 2018. */
31 #include <sys/types.h>
39 if (0 <= fd
&& fd
<= 2)
41 int f
= fd_safer (dup (fd
));
52 test (const char *pwd_prog
)
54 char *argv
[2] = { (char *) "pwd", NULL
};
56 sigset_t blocked_signals
;
57 sigset_t fatal_signal_set
;
58 posix_spawn_file_actions_t actions
;
59 bool actions_allocated
;
60 posix_spawnattr_t attrs
;
70 if (pipe (ifd
) < 0 || (ifd
[0] = fd_safer (ifd
[0])) < 0)
72 perror ("cannot create pipe");
75 sigprocmask (SIG_SETMASK
, NULL
, &blocked_signals
);
76 sigemptyset (&fatal_signal_set
);
77 sigaddset (&fatal_signal_set
, SIGINT
);
78 sigaddset (&fatal_signal_set
, SIGTERM
);
79 sigaddset (&fatal_signal_set
, SIGHUP
);
80 sigaddset (&fatal_signal_set
, SIGPIPE
);
81 sigprocmask (SIG_BLOCK
, &fatal_signal_set
, NULL
);
82 actions_allocated
= false;
83 attrs_allocated
= false;
84 if ((err
= posix_spawn_file_actions_init (&actions
)) != 0
85 || (actions_allocated
= true,
86 (err
= posix_spawn_file_actions_adddup2 (&actions
, ifd
[1], STDOUT_FILENO
)) != 0
87 || (err
= posix_spawn_file_actions_addclose (&actions
, ifd
[1])) != 0
88 || (err
= posix_spawn_file_actions_addclose (&actions
, ifd
[0])) != 0
89 || (err
= posix_spawn_file_actions_addopen (&actions
, STDIN_FILENO
, "/dev/null", O_RDONLY
, 0)) != 0
90 || (err
= posix_spawn_file_actions_addchdir (&actions
, "/")) != 0
91 || (err
= posix_spawnattr_init (&attrs
)) != 0
92 || (attrs_allocated
= true,
93 (err
= posix_spawnattr_setsigmask (&attrs
, &blocked_signals
)) != 0
94 || (err
= posix_spawnattr_setflags (&attrs
, POSIX_SPAWN_SETSIGMASK
)) != 0)
95 || (err
= posix_spawnp (&child
, pwd_prog
, &actions
, &attrs
, argv
, environ
)) != 0))
97 if (actions_allocated
)
98 posix_spawn_file_actions_destroy (&actions
);
100 posix_spawnattr_destroy (&attrs
);
101 sigprocmask (SIG_UNBLOCK
, &fatal_signal_set
, NULL
);
103 perror ("subprocess failed");
106 posix_spawn_file_actions_destroy (&actions
);
107 posix_spawnattr_destroy (&attrs
);
108 sigprocmask (SIG_UNBLOCK
, &fatal_signal_set
, NULL
);
111 fp
= fdopen (fd
, "r");
114 fprintf (stderr
, "fdopen() failed\n");
117 if (fread (line
, 1, 80, fp
) < 2)
119 fprintf (stderr
, "could not read expected output\n");
122 if (memcmp (line
, "/\n", 2) != 0)
124 fprintf (stderr
, "read output is not the expected output");
129 while (waitpid (child
, &status
, 0) != child
)
131 if (!WIFEXITED (status
))
133 fprintf (stderr
, "subprocess terminated with unexpected wait status %d\n", status
);
136 exitstatus
= WEXITSTATUS (status
);
139 fprintf (stderr
, "subprocess terminated with unexpected exit status %d\n", exitstatus
);
149 /* Verify that if a program is given as a relative file name with at least one
150 slash, it is interpreted w.r.t. the current directory after chdir has been
153 const char *abs_pwd_prog
= find_in_path ("pwd");
155 if (abs_pwd_prog
!= NULL
156 && abs_pwd_prog
[0] == '/'
157 && abs_pwd_prog
[1] != '0' && abs_pwd_prog
[1] != '/')
158 test (&abs_pwd_prog
[1]);