Line data Source code
1 : /*
2 : * Tests for tfork
3 : *
4 : * Copyright Ralph Boehme <slow@samba.org> 2017
5 : *
6 : * This program is free software; you can redistribute it and/or modify
7 : * it under the terms of the GNU General Public License as published by
8 : * the Free Software Foundation; either version 3 of the License, or
9 : * (at your option) any later version.
10 : *
11 : * This program is distributed in the hope that it will be useful,
12 : * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 : * GNU General Public License for more details.
15 : *
16 : * You should have received a copy of the GNU General Public License
17 : * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 : */
19 :
20 : #include "replace.h"
21 : #include <talloc.h>
22 : #include <tevent.h>
23 : #include "system/filesys.h"
24 : #include "system/wait.h"
25 : #include "system/select.h"
26 : #include "libcli/util/ntstatus.h"
27 : #include "torture/torture.h"
28 : #include "lib/util/data_blob.h"
29 : #include "torture/local/proto.h"
30 : #include "lib/util/tfork.h"
31 : #include "lib/util/samba_util.h"
32 : #include "lib/util/sys_rw.h"
33 : #ifdef HAVE_PTHREAD
34 : #include <pthread.h>
35 : #endif
36 :
37 0 : static bool test_tfork_simple(struct torture_context *tctx)
38 : {
39 0 : pid_t parent = getpid();
40 0 : struct tfork *t = NULL;
41 : pid_t child;
42 : int ret;
43 :
44 0 : t = tfork_create();
45 0 : if (t == NULL) {
46 0 : torture_fail(tctx, "tfork failed\n");
47 : return false;
48 : }
49 0 : child = tfork_child_pid(t);
50 0 : if (child == 0) {
51 0 : torture_comment(tctx, "my parent pid is %d\n", parent);
52 0 : torture_assert(tctx, getpid() != parent, "tfork failed\n");
53 0 : _exit(0);
54 : }
55 :
56 0 : ret = tfork_destroy(&t);
57 0 : torture_assert(tctx, ret == 0, "tfork_destroy failed\n");
58 :
59 0 : return true;
60 : }
61 :
62 0 : static bool test_tfork_status(struct torture_context *tctx)
63 : {
64 0 : struct tfork *t = NULL;
65 : int status;
66 : pid_t child;
67 0 : bool ok = true;
68 :
69 0 : t = tfork_create();
70 0 : if (t == NULL) {
71 0 : torture_fail(tctx, "tfork failed\n");
72 : return false;
73 : }
74 0 : child = tfork_child_pid(t);
75 0 : if (child == 0) {
76 0 : _exit(123);
77 : }
78 :
79 0 : status = tfork_status(&t, true);
80 0 : if (status == -1) {
81 0 : torture_fail(tctx, "tfork_status failed\n");
82 : }
83 :
84 0 : torture_assert_goto(tctx, WIFEXITED(status) == true, ok, done,
85 : "tfork failed\n");
86 0 : torture_assert_goto(tctx, WEXITSTATUS(status) == 123, ok, done,
87 : "tfork failed\n");
88 :
89 0 : torture_comment(tctx, "exit status [%d]\n", WEXITSTATUS(status));
90 :
91 0 : done:
92 0 : return ok;
93 : }
94 :
95 0 : static bool test_tfork_sigign(struct torture_context *tctx)
96 : {
97 0 : struct tfork *t = NULL;
98 : struct sigaction act;
99 : pid_t child;
100 : int status;
101 0 : bool ok = true;
102 : int ret;
103 :
104 0 : act = (struct sigaction) {
105 : .sa_flags = SA_NOCLDWAIT,
106 : .sa_handler = SIG_IGN,
107 : };
108 :
109 0 : ret = sigaction(SIGCHLD, &act, NULL);
110 0 : torture_assert_goto(tctx, ret == 0, ok, done, "sigaction failed\n");
111 :
112 0 : t = tfork_create();
113 0 : if (t == NULL) {
114 0 : torture_fail(tctx, "tfork failed\n");
115 : return false;
116 : }
117 0 : child = tfork_child_pid(t);
118 0 : if (child == 0) {
119 0 : sleep(1);
120 0 : _exit(123);
121 : }
122 :
123 0 : child = fork();
124 0 : if (child == -1) {
125 0 : torture_fail(tctx, "fork failed\n");
126 : return false;
127 : }
128 0 : if (child == 0) {
129 0 : _exit(0);
130 : }
131 :
132 0 : status = tfork_status(&t, true);
133 0 : if (status == -1) {
134 0 : torture_fail(tctx, "tfork_status failed\n");
135 : }
136 :
137 0 : torture_assert_goto(tctx, WIFEXITED(status) == true, ok, done,
138 : "tfork failed\n");
139 0 : torture_assert_goto(tctx, WEXITSTATUS(status) == 123, ok, done,
140 : "tfork failed\n");
141 0 : torture_comment(tctx, "exit status [%d]\n", WEXITSTATUS(status));
142 :
143 0 : done:
144 0 : return ok;
145 : }
146 :
147 0 : static void sigchld_handler1(int signum, siginfo_t *si, void *u)
148 : {
149 : pid_t pid;
150 : int status;
151 :
152 0 : if (signum != SIGCHLD) {
153 0 : abort();
154 : }
155 :
156 0 : pid = waitpid(si->si_pid, &status, 0);
157 0 : if (pid != si->si_pid) {
158 0 : abort();
159 : }
160 0 : }
161 :
162 0 : static bool test_tfork_sighandler(struct torture_context *tctx)
163 : {
164 0 : struct tfork *t = NULL;
165 : struct sigaction act;
166 : struct sigaction oldact;
167 : pid_t child;
168 : int status;
169 0 : bool ok = true;
170 : int ret;
171 :
172 0 : act = (struct sigaction) {
173 : .sa_flags = SA_SIGINFO,
174 : .sa_sigaction = sigchld_handler1,
175 : };
176 :
177 0 : ret = sigaction(SIGCHLD, &act, &oldact);
178 0 : torture_assert_goto(tctx, ret == 0, ok, done, "sigaction failed\n");
179 :
180 0 : t = tfork_create();
181 0 : if (t == NULL) {
182 0 : torture_fail(tctx, "tfork failed\n");
183 : return false;
184 : }
185 0 : child = tfork_child_pid(t);
186 0 : if (child == 0) {
187 0 : sleep(1);
188 0 : _exit(123);
189 : }
190 :
191 0 : child = fork();
192 0 : if (child == -1) {
193 0 : torture_fail(tctx, "fork failed\n");
194 : return false;
195 : }
196 0 : if (child == 0) {
197 0 : _exit(0);
198 : }
199 :
200 0 : status = tfork_status(&t, true);
201 0 : if (status == -1) {
202 0 : torture_fail(tctx, "tfork_status failed\n");
203 : }
204 :
205 0 : torture_assert_goto(tctx, WIFEXITED(status) == true, ok, done,
206 : "tfork failed\n");
207 0 : torture_assert_goto(tctx, WEXITSTATUS(status) == 123, ok, done,
208 : "tfork failed\n");
209 0 : torture_comment(tctx, "exit status [%d]\n", WEXITSTATUS(status));
210 :
211 0 : done:
212 0 : sigaction(SIGCHLD, &oldact, NULL);
213 :
214 0 : return ok;
215 : }
216 :
217 0 : static bool test_tfork_process_hierarchy(struct torture_context *tctx)
218 : {
219 0 : struct tfork *t = NULL;
220 0 : pid_t pid = getpid();
221 : pid_t child;
222 0 : pid_t pgid = getpgid(0);
223 0 : pid_t sid = getsid(0);
224 0 : char *procpath = NULL;
225 : int status;
226 : struct stat st;
227 : int ret;
228 0 : bool ok = true;
229 :
230 0 : procpath = talloc_asprintf(tctx, "/proc/%d/status", getpid());
231 0 : torture_assert_not_null(tctx, procpath, "talloc_asprintf failed\n");
232 :
233 0 : ret = stat(procpath, &st);
234 0 : TALLOC_FREE(procpath);
235 0 : if (ret != 0) {
236 0 : if (errno == ENOENT) {
237 0 : torture_skip(tctx, "/proc missing\n");
238 : }
239 0 : torture_fail(tctx, "stat failed\n");
240 : }
241 :
242 0 : t = tfork_create();
243 0 : if (t == NULL) {
244 0 : torture_fail(tctx, "tfork failed\n");
245 : return false;
246 : }
247 0 : child = tfork_child_pid(t);
248 0 : if (child == 0) {
249 0 : char *cmd = NULL;
250 0 : FILE *fp = NULL;
251 : char line[64];
252 : char *p;
253 : pid_t ppid;
254 :
255 0 : torture_assert_goto(tctx, pgid == getpgid(0), ok, child_fail, "tfork failed\n");
256 0 : torture_assert_goto(tctx, sid == getsid(0), ok, child_fail, "tfork failed\n");
257 :
258 0 : cmd = talloc_asprintf(tctx, "cat /proc/%d/status | awk '/^PPid:/ {print $2}'", getppid());
259 0 : torture_assert_goto(tctx, cmd != NULL, ok, child_fail, "talloc_asprintf failed\n");
260 :
261 0 : fp = popen(cmd, "r");
262 0 : torture_assert_goto(tctx, fp != NULL, ok, child_fail, "popen failed\n");
263 :
264 0 : p = fgets(line, sizeof(line) - 1, fp);
265 0 : pclose(fp);
266 0 : torture_assert_goto(tctx, p != NULL, ok, child_fail, "popen failed\n");
267 :
268 0 : ret = sscanf(line, "%d", &ppid);
269 0 : torture_assert_goto(tctx, ret == 1, ok, child_fail, "sscanf failed\n");
270 0 : torture_assert_goto(tctx, ppid == pid, ok, child_fail, "process hierarchy not rooted at caller\n");
271 :
272 0 : _exit(0);
273 :
274 0 : child_fail:
275 0 : _exit(1);
276 : }
277 :
278 0 : status = tfork_status(&t, true);
279 0 : if (status == -1) {
280 0 : torture_fail(tctx, "tfork_status failed\n");
281 : }
282 :
283 0 : torture_assert_goto(tctx, WIFEXITED(status) == true, ok, done,
284 : "tfork failed\n");
285 0 : torture_assert_goto(tctx, WEXITSTATUS(status) == 0, ok, done,
286 : "tfork failed\n");
287 0 : torture_comment(tctx, "exit status [%d]\n", WEXITSTATUS(status));
288 :
289 0 : done:
290 0 : return ok;
291 : }
292 :
293 0 : static bool test_tfork_pipe(struct torture_context *tctx)
294 : {
295 0 : struct tfork *t = NULL;
296 : int status;
297 : pid_t child;
298 : int up[2];
299 : int down[2];
300 : char c;
301 : int ret;
302 0 : bool ok = true;
303 :
304 0 : ret = pipe(&up[0]);
305 0 : torture_assert(tctx, ret == 0, "pipe failed\n");
306 :
307 0 : ret = pipe(&down[0]);
308 0 : torture_assert(tctx, ret == 0, "pipe failed\n");
309 :
310 0 : t = tfork_create();
311 0 : if (t == NULL) {
312 0 : torture_fail(tctx, "tfork failed\n");
313 : return false;
314 : }
315 0 : child = tfork_child_pid(t);
316 0 : if (child == 0) {
317 0 : close(up[0]);
318 0 : close(down[1]);
319 :
320 0 : ret = read(down[0], &c, 1);
321 0 : torture_assert_goto(tctx, ret == 1, ok, child_fail, "read failed\n");
322 0 : torture_assert_goto(tctx, c == 1, ok, child_fail, "read failed\n");
323 :
324 0 : ret = write(up[1], &(char){2}, 1);
325 0 : torture_assert_goto(tctx, ret == 1, ok, child_fail, "write failed\n");
326 :
327 0 : _exit(0);
328 :
329 0 : child_fail:
330 0 : _exit(1);
331 : }
332 :
333 0 : close(up[1]);
334 0 : close(down[0]);
335 :
336 0 : ret = write(down[1], &(char){1}, 1);
337 0 : torture_assert(tctx, ret == 1, "read failed\n");
338 :
339 0 : ret = read(up[0], &c, 1);
340 0 : torture_assert(tctx, ret == 1, "read failed\n");
341 0 : torture_assert(tctx, c == 2, "read failed\n");
342 :
343 0 : status = tfork_status(&t, true);
344 0 : if (status == -1) {
345 0 : torture_fail(tctx, "tfork_status failed\n");
346 : }
347 :
348 0 : torture_assert_goto(tctx, WIFEXITED(status) == true, ok, done,
349 : "tfork failed\n");
350 0 : torture_assert_goto(tctx, WEXITSTATUS(status) == 0, ok, done,
351 : "tfork failed\n");
352 0 : done:
353 0 : return ok;
354 : }
355 :
356 0 : static bool test_tfork_twice(struct torture_context *tctx)
357 : {
358 0 : struct tfork *t = NULL;
359 : int status;
360 : pid_t child;
361 : pid_t pid;
362 : int up[2];
363 : int ret;
364 0 : bool ok = true;
365 :
366 0 : ret = pipe(&up[0]);
367 0 : torture_assert(tctx, ret == 0, "pipe failed\n");
368 :
369 0 : t = tfork_create();
370 0 : if (t == NULL) {
371 0 : torture_fail(tctx, "tfork failed\n");
372 : return false;
373 : }
374 0 : child = tfork_child_pid(t);
375 0 : if (child == 0) {
376 0 : close(up[0]);
377 :
378 0 : t = tfork_create();
379 0 : if (t == NULL) {
380 0 : torture_fail(tctx, "tfork failed\n");
381 : return false;
382 : }
383 0 : child = tfork_child_pid(t);
384 0 : if (child == 0) {
385 0 : sleep(1);
386 0 : pid = getpid();
387 0 : ret = write(up[1], &pid, sizeof(pid_t));
388 0 : torture_assert_goto(tctx, ret == sizeof(pid_t), ok, child_fail, "write failed\n");
389 :
390 0 : _exit(0);
391 :
392 0 : child_fail:
393 0 : _exit(1);
394 : }
395 :
396 0 : _exit(0);
397 : }
398 :
399 0 : close(up[1]);
400 :
401 0 : ret = read(up[0], &pid, sizeof(pid_t));
402 0 : torture_assert(tctx, ret == sizeof(pid_t), "read failed\n");
403 :
404 0 : status = tfork_status(&t, true);
405 0 : torture_assert_goto(tctx, status != -1, ok, done, "tfork_status failed\n");
406 :
407 0 : torture_assert_goto(tctx, WIFEXITED(status) == true, ok, done,
408 : "tfork failed\n");
409 0 : torture_assert_goto(tctx, WEXITSTATUS(status) == 0, ok, done,
410 : "tfork failed\n");
411 0 : done:
412 0 : return ok;
413 : }
414 :
415 0 : static void *tfork_thread(void *p)
416 : {
417 0 : struct tfork *t = NULL;
418 : int status;
419 : pid_t child;
420 0 : uint64_t tid = (uint64_t)pthread_self();
421 0 : uint64_t *result = NULL;
422 : int up[2];
423 : ssize_t nread;
424 : int ret;
425 :
426 0 : ret = pipe(up);
427 0 : if (ret != 0) {
428 0 : pthread_exit(NULL);
429 : }
430 :
431 0 : t = tfork_create();
432 0 : if (t == NULL) {
433 0 : pthread_exit(NULL);
434 : }
435 0 : child = tfork_child_pid(t);
436 0 : if (child == 0) {
437 : ssize_t nwritten;
438 :
439 0 : close(up[0]);
440 0 : tid++;
441 0 : nwritten = sys_write(up[1], &tid, sizeof(uint64_t));
442 0 : if (nwritten != sizeof(uint64_t)) {
443 0 : _exit(1);
444 : }
445 0 : _exit(0);
446 : }
447 0 : close(up[1]);
448 :
449 0 : result = malloc(sizeof(uint64_t));
450 0 : if (result == NULL) {
451 0 : pthread_exit(NULL);
452 : }
453 :
454 0 : nread = sys_read(up[0], result, sizeof(uint64_t));
455 0 : if (nread != sizeof(uint64_t)) {
456 0 : pthread_exit(NULL);
457 : }
458 :
459 0 : status = tfork_status(&t, true);
460 0 : if (status == -1) {
461 0 : pthread_exit(NULL);
462 : }
463 :
464 0 : pthread_exit(result);
465 : }
466 :
467 0 : static bool test_tfork_threads(struct torture_context *tctx)
468 0 : {
469 : int ret;
470 0 : bool ok = true;
471 0 : const int num_threads = 64;
472 0 : pthread_t threads[num_threads];
473 : sigset_t set;
474 : int i;
475 :
476 : #ifndef HAVE_PTHREAD
477 : torture_skip(tctx, "no pthread support\n");
478 : #endif
479 :
480 : /*
481 : * Be nasty and taste for the worst case: ensure all threads start with
482 : * SIGCHLD unblocked so we have the most fun with SIGCHLD being
483 : * delivered to a random thread. :)
484 : */
485 0 : sigemptyset(&set);
486 0 : sigaddset(&set, SIGCHLD);
487 : #ifdef HAVE_PTHREAD
488 0 : ret = pthread_sigmask(SIG_UNBLOCK, &set, NULL);
489 : #else
490 : ret = sigprocmask(SIG_UNBLOCK, &set, NULL);
491 : #endif
492 0 : if (ret != 0) {
493 0 : return false;
494 : }
495 :
496 0 : for (i = 0; i < num_threads; i++) {
497 0 : ret = pthread_create(&threads[i], NULL, tfork_thread, NULL);
498 0 : torture_assert_goto(tctx, ret == 0, ok, done,
499 : "pthread_create failed\n");
500 : }
501 :
502 0 : for (i = 0; i < num_threads; i++) {
503 : void *p;
504 : uint64_t *result;
505 :
506 0 : ret = pthread_join(threads[i], &p);
507 0 : torture_assert_goto(tctx, ret == 0, ok, done,
508 : "pthread_join failed\n");
509 0 : result = (uint64_t *)p;
510 0 : torture_assert_goto(tctx, *result == (uint64_t)threads[i] + 1,
511 : ok, done, "thread failed\n");
512 0 : free(p);
513 : }
514 :
515 0 : done:
516 0 : return ok;
517 : }
518 :
519 0 : static bool test_tfork_cmd_send(struct torture_context *tctx)
520 : {
521 0 : struct tevent_context *ev = NULL;
522 0 : struct tevent_req *req = NULL;
523 0 : const char *cmd[2] = { NULL, NULL };
524 0 : bool ok = true;
525 :
526 0 : ev = tevent_context_init(tctx);
527 0 : torture_assert_goto(tctx, ev != NULL, ok, done,
528 : "tevent_context_init failed\n");
529 :
530 0 : cmd[0] = talloc_asprintf(tctx, "%s/testprogs/blackbox/tfork.sh", SRCDIR);
531 0 : torture_assert_goto(tctx, cmd[0] != NULL, ok, done,
532 : "talloc_asprintf failed\n");
533 :
534 0 : req = samba_runcmd_send(tctx, ev, timeval_zero(), 0, 0,
535 : cmd, "foo", NULL);
536 0 : torture_assert_goto(tctx, req != NULL, ok, done,
537 : "samba_runcmd_send failed\n");
538 :
539 0 : ok = tevent_req_poll(req, ev);
540 0 : torture_assert_goto(tctx, ok, ok, done, "tevent_req_poll failed\n");
541 :
542 0 : torture_comment(tctx, "samba_runcmd_send test finished\n");
543 :
544 0 : done:
545 0 : TALLOC_FREE(ev);
546 :
547 0 : return ok;
548 : }
549 :
550 : /*
551 : * Test to ensure that the event_fd becomes readable after
552 : * a tfork_process terminates.
553 : */
554 0 : static bool test_tfork_event_file_handle(struct torture_context *tctx)
555 : {
556 0 : bool ok = true;
557 :
558 0 : struct tfork *t1 = NULL;
559 : pid_t child1;
560 0 : struct pollfd poll1[] = {
561 : {
562 : .fd = -1,
563 : .events = POLLIN,
564 : },
565 : };
566 :
567 0 : struct tfork *t2 = NULL;
568 : pid_t child2;
569 0 : struct pollfd poll2[] = {
570 : {
571 : .fd = -1,
572 : .events = POLLIN,
573 : },
574 : };
575 :
576 :
577 0 : t1 = tfork_create();
578 0 : if (t1 == NULL) {
579 0 : torture_fail(tctx, "tfork failed\n");
580 : return false;
581 : }
582 :
583 0 : child1 = tfork_child_pid(t1);
584 0 : if (child1 == 0) {
585 : /*
586 : * Parent process will kill this with a SIGTERM
587 : * so 10 seconds should be plenty
588 : */
589 0 : sleep(10);
590 0 : exit(1);
591 : }
592 0 : poll1[0].fd = tfork_event_fd(t1);
593 :
594 0 : t2 = tfork_create();
595 0 : if (t2 == NULL) {
596 0 : torture_fail(tctx, "tfork failed\n");
597 : return false;
598 : }
599 0 : child2 = tfork_child_pid(t2);
600 0 : if (child2 == 0) {
601 : /*
602 : * Parent process will kill this with a SIGTERM
603 : * so 10 seconds should be plenty
604 : */
605 0 : sleep(10);
606 0 : exit(2);
607 : }
608 0 : poll2[0].fd = tfork_event_fd(t2);
609 :
610 : /*
611 : * Have forked two process and are in the master process
612 : * Expect that both event_fds are unreadable
613 : */
614 0 : poll(poll1, 1, 0);
615 0 : ok = !(poll1[0].revents & POLLIN);
616 0 : torture_assert_goto(tctx, ok, ok, done,
617 : "tfork process 1 event fd readable\n");
618 0 : poll(poll2, 1, 0);
619 0 : ok = !(poll2[0].revents & POLLIN);
620 0 : torture_assert_goto(tctx, ok, ok, done,
621 : "tfork process 1 event fd readable\n");
622 :
623 : /* Kill the first child process */
624 0 : kill(child1, SIGKILL);
625 0 : sleep(1);
626 :
627 : /*
628 : * Have killed the first child, so expect it's event_fd to have gone
629 : * readable.
630 : *
631 : */
632 0 : poll(poll1, 1, 0);
633 0 : ok = (poll1[0].revents & POLLIN);
634 0 : torture_assert_goto(tctx, ok, ok, done,
635 : "tfork process 1 event fd not readable\n");
636 0 : poll(poll2, 1, 0);
637 0 : ok = !(poll2[0].revents & POLLIN);
638 0 : torture_assert_goto(tctx, ok, ok, done,
639 : "tfork process 2 event fd readable\n");
640 :
641 : /* Kill the secind child process */
642 0 : kill(child2, SIGKILL);
643 0 : sleep(1);
644 : /*
645 : * Have killed the children, so expect their event_fd's to have gone
646 : * readable.
647 : *
648 : */
649 0 : poll(poll1, 1, 0);
650 0 : ok = (poll1[0].revents & POLLIN);
651 0 : torture_assert_goto(tctx, ok, ok, done,
652 : "tfork process 1 event fd not readable\n");
653 0 : poll(poll2, 1, 0);
654 0 : ok = (poll2[0].revents & POLLIN);
655 0 : torture_assert_goto(tctx, ok, ok, done,
656 : "tfork process 2 event fd not readable\n");
657 :
658 0 : done:
659 0 : free(t1);
660 0 : free(t2);
661 :
662 0 : return ok;
663 : }
664 :
665 : /*
666 : * Test to ensure that the status calls behave as expected after a process
667 : * terminates.
668 : *
669 : * As the parent process owns the status fd's they get passed to all
670 : * subsequent children after a tfork. So it's possible for another
671 : * child process to hold the status pipe open.
672 : *
673 : * The event fd needs to be left open by tfork, as a close in the status
674 : * code can cause issues in tevent code.
675 : *
676 : */
677 0 : static bool test_tfork_status_handle(struct torture_context *tctx)
678 : {
679 0 : bool ok = true;
680 :
681 0 : struct tfork *t1 = NULL;
682 : pid_t child1;
683 :
684 0 : struct tfork *t2 = NULL;
685 : pid_t child2;
686 :
687 : int status;
688 : int fd;
689 : int ev1_fd;
690 : int ev2_fd;
691 :
692 :
693 0 : t1 = tfork_create();
694 0 : if (t1 == NULL) {
695 0 : torture_fail(tctx, "tfork failed\n");
696 : return false;
697 : }
698 :
699 0 : child1 = tfork_child_pid(t1);
700 0 : if (child1 == 0) {
701 : /*
702 : * Parent process will kill this with a SIGTERM
703 : * so 10 seconds should be plenty
704 : */
705 0 : sleep(10);
706 0 : exit(1);
707 : }
708 0 : ev1_fd = tfork_event_fd(t1);
709 :
710 0 : t2 = tfork_create();
711 0 : if (t2 == NULL) {
712 0 : torture_fail(tctx, "tfork failed\n");
713 : return false;
714 : }
715 0 : child2 = tfork_child_pid(t2);
716 0 : if (child2 == 0) {
717 : /*
718 : * Parent process will kill this with a SIGTERM
719 : * so 10 seconds should be plenty
720 : */
721 0 : sleep(10);
722 0 : exit(2);
723 : }
724 0 : ev2_fd = tfork_event_fd(t2);
725 :
726 : /*
727 : * Have forked two process and are in the master process
728 : * expect that the status call will block, and hence return -1
729 : * as the processes are still running
730 : * The event fd's should be open.
731 : */
732 0 : status = tfork_status(&t1, false);
733 0 : ok = status == -1;
734 0 : torture_assert_goto(tctx, ok, ok, done,
735 : "tfork status available for non terminated "
736 : "process 1\n");
737 : /* Is the event fd open? */
738 0 : fd = dup(ev1_fd);
739 0 : ok = fd != -1;
740 0 : torture_assert_goto(tctx, ok, ok, done,
741 : "tfork process 1 event fd is not open");
742 :
743 0 : status = tfork_status(&t2, false);
744 0 : ok = status == -1;
745 0 : torture_assert_goto(tctx, ok, ok, done,
746 : "tfork status available for non terminated "
747 : "process 2\n");
748 : /* Is the event fd open? */
749 0 : fd = dup(ev2_fd);
750 0 : ok = fd != -1;
751 0 : torture_assert_goto(tctx, ok, ok, done,
752 : "tfork process 2 event fd is not open");
753 :
754 : /*
755 : * Kill the first process, it's status should be readable
756 : * and it's event_fd should be open
757 : * The second process's status should be unreadable.
758 : */
759 0 : kill(child1, SIGTERM);
760 0 : sleep(1);
761 0 : status = tfork_status(&t1, false);
762 0 : ok = status != -1;
763 0 : torture_assert_goto(tctx, ok, ok, done,
764 : "tfork status for child 1 not available after "
765 : "termination\n");
766 : /* Is the event fd open? */
767 0 : fd = dup(ev2_fd);
768 0 : ok = fd != -1;
769 0 : torture_assert_goto(tctx, ok, ok, done,
770 : "tfork process 1 event fd is not open");
771 :
772 0 : status = tfork_status(&t2, false);
773 0 : ok = status == -1;
774 0 : torture_assert_goto(tctx, ok, ok, done,
775 : "tfork status available for child 2 after "
776 : "termination of child 1\n");
777 :
778 : /*
779 : * Kill the second process, it's status should be readable
780 : */
781 0 : kill(child2, SIGTERM);
782 0 : sleep(1);
783 0 : status = tfork_status(&t2, false);
784 0 : ok = status != -1;
785 0 : torture_assert_goto(tctx, ok, ok, done,
786 : "tfork status for child 2 not available after "
787 : "termination\n");
788 :
789 : /* Check that the event fd's are still open */
790 : /* Is the event fd open? */
791 0 : fd = dup(ev1_fd);
792 0 : ok = fd != -1;
793 0 : torture_assert_goto(tctx, ok, ok, done,
794 : "tfork process 1 event fd is not open");
795 : /* Is the event fd open? */
796 0 : fd = dup(ev2_fd);
797 0 : ok = fd != -1;
798 0 : torture_assert_goto(tctx, ok, ok, done,
799 : "tfork process 2 event fd is not open");
800 :
801 0 : done:
802 0 : return ok;
803 : }
804 :
805 964 : struct torture_suite *torture_local_tfork(TALLOC_CTX *mem_ctx)
806 : {
807 738 : struct torture_suite *suite =
808 226 : torture_suite_create(mem_ctx, "tfork");
809 :
810 964 : torture_suite_add_simple_test(suite,
811 : "tfork_simple",
812 : test_tfork_simple);
813 :
814 964 : torture_suite_add_simple_test(suite,
815 : "tfork_status",
816 : test_tfork_status);
817 :
818 964 : torture_suite_add_simple_test(suite,
819 : "tfork_sigign",
820 : test_tfork_sigign);
821 :
822 964 : torture_suite_add_simple_test(suite,
823 : "tfork_sighandler",
824 : test_tfork_sighandler);
825 :
826 964 : torture_suite_add_simple_test(suite,
827 : "tfork_process_hierarchy",
828 : test_tfork_process_hierarchy);
829 :
830 964 : torture_suite_add_simple_test(suite,
831 : "tfork_pipe",
832 : test_tfork_pipe);
833 :
834 964 : torture_suite_add_simple_test(suite,
835 : "tfork_twice",
836 : test_tfork_twice);
837 :
838 964 : torture_suite_add_simple_test(suite,
839 : "tfork_threads",
840 : test_tfork_threads);
841 :
842 964 : torture_suite_add_simple_test(suite,
843 : "tfork_cmd_send",
844 : test_tfork_cmd_send);
845 :
846 964 : torture_suite_add_simple_test(suite,
847 : "tfork_event_file_handle",
848 : test_tfork_event_file_handle);
849 :
850 964 : torture_suite_add_simple_test(suite,
851 : "tfork_status_handle",
852 : test_tfork_status_handle);
853 :
854 964 : return suite;
855 : }
|