2 Copyright (C) 2000-2023 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
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.
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.
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/>. */
28 /* Nonzero if the program gets called via `exec'. */
32 #define CMDLINE_OPTIONS \
33 { "restart", no_argument, &restart, 1 },
35 /* Prototype for our test function. */
36 extern void do_prepare (int argc
, char *argv
[]);
37 extern int do_test (int argc
, char *argv
[]);
39 /* We have a preparation function. */
40 #define PREPARE do_prepare
42 #include "../test-skeleton.c"
45 /* Name of the temporary files. */
49 /* File descriptors for these temporary files. */
50 static int temp_fd1
= -1;
51 static int temp_fd2
= -1;
53 /* The contents of our files. */
54 static const char fd1string
[] = "This file should get closed";
55 static const char fd2string
[] = "This file should stay opened";
58 /* We have a preparation function. */
60 do_prepare (int argc
, char *argv
[])
62 /* We must not open any files in the restart case. */
66 temp_fd1
= create_temp_file ("exec", &name1
);
67 temp_fd2
= create_temp_file ("exec", &name2
);
68 if (temp_fd1
< 0 || temp_fd2
< 0)
74 handle_restart (const char *fd1s
, const char *fd2s
, const char *name
)
80 /* First get the descriptors. */
86 error (EXIT_FAILURE
, 0, "value of fd1 and fd2 is the same");
88 /* First the easy part: read from the file descriptor which is
89 supposed to be open. */
90 if (lseek (fd2
, 0, SEEK_CUR
) != strlen (fd2string
))
91 error (EXIT_FAILURE
, errno
, "file 2 not in right position");
92 if (lseek (fd2
, 0, SEEK_SET
) != 0)
93 error (EXIT_FAILURE
, 0, "cannot reset position in file 2");
94 if (read (fd2
, buf
, sizeof buf
) != strlen (fd2string
))
95 error (EXIT_FAILURE
, 0, "cannot read file 2");
96 if (memcmp (fd2string
, buf
, strlen (fd2string
)) != 0)
97 error (EXIT_FAILURE
, 0, "file 2 does not match");
99 /* No try to read the first file. First make sure it is not opened. */
100 if (lseek (fd1
, 0, SEEK_CUR
) != (off_t
) -1 || errno
!= EBADF
)
101 error (EXIT_FAILURE
, 0, "file 1 (%d) is not closed", fd1
);
103 /* Now open the file and read it. */
104 fd1
= open (name
, O_RDONLY
);
106 error (EXIT_FAILURE
, errno
,
107 "cannot open first file \"%s\" for verification", name
);
109 if (read (fd1
, buf
, sizeof buf
) != strlen (fd1string
))
110 error (EXIT_FAILURE
, errno
, "cannot read file 1");
111 if (memcmp (fd1string
, buf
, strlen (fd1string
)) != 0)
112 error (EXIT_FAILURE
, 0, "file 1 does not match");
119 do_test (int argc
, char *argv
[])
126 - one or four parameters left if called initially
127 + path for ld.so optional
128 + "--library-path" optional
129 + the library path optional
130 + the application name
131 - three parameters left if called through re-execution
132 + file descriptor number which is supposed to be closed
133 + the open file descriptor
134 + the name of the closed descriptor
140 error (EXIT_FAILURE
, 0, "wrong number of arguments (%d)", argc
);
142 return handle_restart (argv
[1], argv
[2], argv
[3]);
145 if (argc
!= 2 && argc
!= 5)
146 error (EXIT_FAILURE
, 0, "wrong number of arguments (%d)", argc
);
148 /* Prepare the test. We are creating two files: one which file descriptor
149 will be marked with FD_CLOEXEC, another which is not. */
152 flags
= fcntl (temp_fd1
, F_GETFD
, 0);
154 error (EXIT_FAILURE
, errno
, "cannot get flags");
156 if (fcntl (temp_fd1
, F_SETFD
, flags
) < 0)
157 error (EXIT_FAILURE
, errno
, "cannot set flags");
159 /* Write something in the files. */
160 if (write (temp_fd1
, fd1string
, strlen (fd1string
)) != strlen (fd1string
))
161 error (EXIT_FAILURE
, errno
, "cannot write to first file");
162 if (write (temp_fd2
, fd2string
, strlen (fd2string
)) != strlen (fd2string
))
163 error (EXIT_FAILURE
, errno
, "cannot write to second file");
165 /* We want to test the `exec' function. To do this we restart the program
166 with an additional parameter. But first create another process. */
173 snprintf (fd1name
, sizeof fd1name
, "%d", temp_fd1
);
174 snprintf (fd2name
, sizeof fd2name
, "%d", temp_fd2
);
176 /* This is the child. Construct the command line. */
178 execl (argv
[1], argv
[1], argv
[2], argv
[3], argv
[4], "--direct",
179 "--restart", fd1name
, fd2name
, name1
, NULL
);
181 execl (argv
[1], argv
[1], "--direct",
182 "--restart", fd1name
, fd2name
, name1
, NULL
);
184 error (EXIT_FAILURE
, errno
, "cannot exec");
186 else if (pid
== (pid_t
) -1)
187 error (EXIT_FAILURE
, errno
, "cannot fork");
189 /* Wait for the child. */
190 if (waitpid (pid
, &status
, 0) != pid
)
191 error (EXIT_FAILURE
, errno
, "wrong child");
193 if (WTERMSIG (status
) != 0)
194 error (EXIT_FAILURE
, 0, "Child terminated incorrectly");
195 status
= WEXITSTATUS (status
);