tst-exec2.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. /* Thread with running thread calls exec.
  2. Copyright (C) 2002 Free Software Foundation, Inc.
  3. This file is part of the GNU C Library.
  4. Contributed by Ulrich Drepper <drepper@redhat.com>, 2002.
  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.
  9. The GNU C Library 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 GNU
  12. Lesser General Public License for more details.
  13. You should have received a copy of the GNU Lesser General Public
  14. License along with the GNU C Library; if not, see
  15. <http://www.gnu.org/licenses/>. */
  16. #include <errno.h>
  17. #include <paths.h>
  18. #include <pthread.h>
  19. #include <signal.h>
  20. #include <stdbool.h>
  21. #include <stdio.h>
  22. #include <stdlib.h>
  23. #include <unistd.h>
  24. #include <sys/wait.h>
  25. #include "../test-skeleton.h"
  26. static void *
  27. tf (void *arg)
  28. {
  29. pthread_t th = (pthread_t) arg;
  30. if (pthread_join (th, NULL) == 0)
  31. {
  32. puts ("thread in parent joined!?");
  33. exit (1);
  34. }
  35. puts ("join in thread in parent returned!?");
  36. exit (1);
  37. }
  38. static int
  39. do_test (void)
  40. {
  41. int fd[2];
  42. if (pipe (fd) != 0)
  43. {
  44. puts ("pipe failed");
  45. exit (1);
  46. }
  47. /* Not interested in knowing when the pipe is closed. */
  48. if (sigignore (SIGPIPE) != 0)
  49. {
  50. puts ("sigignore failed");
  51. exit (1);
  52. }
  53. pid_t pid = fork ();
  54. if (pid == -1)
  55. {
  56. puts ("fork failed");
  57. exit (1);
  58. }
  59. if (pid == 0)
  60. {
  61. /* Use the fd for stdout. This is kind of ugly because it
  62. substitutes the fd of stdout but we know what we are doing
  63. here... */
  64. if (dup2 (fd[1], STDOUT_FILENO) != STDOUT_FILENO)
  65. {
  66. puts ("dup2 failed");
  67. exit (1);
  68. }
  69. close (fd[0]);
  70. pthread_t th;
  71. if (pthread_create (&th, NULL, tf, (void *) pthread_self ()) != 0)
  72. {
  73. puts ("create failed");
  74. exit (1);
  75. }
  76. execl (_PATH_BSHELL, _PATH_BSHELL, "-c", "echo $$", NULL);
  77. puts ("execl failed");
  78. exit (1);
  79. }
  80. close (fd[1]);
  81. char buf[200];
  82. ssize_t n;
  83. bool seen_pid = false;
  84. while (TEMP_FAILURE_RETRY ((n = read (fd[0], buf, sizeof (buf)))) > 0)
  85. {
  86. /* We only expect to read the PID. */
  87. char *endp;
  88. long int rpid = strtol (buf, &endp, 10);
  89. if (*endp != '\n')
  90. {
  91. printf ("didn't parse whole line: \"%s\"\n", buf);
  92. exit (1);
  93. }
  94. if (endp == buf)
  95. {
  96. puts ("read empty line");
  97. exit (1);
  98. }
  99. if (rpid != pid)
  100. {
  101. printf ("found \"%s\", expected PID %ld\n", buf, (long int) pid);
  102. exit (1);
  103. }
  104. if (seen_pid)
  105. {
  106. puts ("found more than one PID line");
  107. exit (1);
  108. }
  109. seen_pid = true;
  110. }
  111. close (fd[0]);
  112. int status;
  113. int err = waitpid (pid, &status, 0);
  114. if (err != pid)
  115. {
  116. puts ("waitpid failed");
  117. exit (1);
  118. }
  119. if (!seen_pid)
  120. {
  121. puts ("didn't get PID");
  122. exit (1);
  123. }
  124. return 0;
  125. }
  126. #define TEST_FUNCTION do_test ()
  127. #include "../test-skeleton.c"