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. static void *
  26. tf (void *arg)
  27. {
  28. pthread_t th = (pthread_t) arg;
  29. if (pthread_join (th, NULL) == 0)
  30. {
  31. puts ("thread in parent joined!?");
  32. exit (1);
  33. }
  34. puts ("join in thread in parent returned!?");
  35. exit (1);
  36. }
  37. static int
  38. do_test (void)
  39. {
  40. int fd[2];
  41. if (pipe (fd) != 0)
  42. {
  43. puts ("pipe failed");
  44. exit (1);
  45. }
  46. /* Not interested in knowing when the pipe is closed. */
  47. if (sigignore (SIGPIPE) != 0)
  48. {
  49. puts ("sigignore failed");
  50. exit (1);
  51. }
  52. pid_t pid = fork ();
  53. if (pid == -1)
  54. {
  55. puts ("fork failed");
  56. exit (1);
  57. }
  58. if (pid == 0)
  59. {
  60. /* Use the fd for stdout. This is kind of ugly because it
  61. substitutes the fd of stdout but we know what we are doing
  62. here... */
  63. if (dup2 (fd[1], STDOUT_FILENO) != STDOUT_FILENO)
  64. {
  65. puts ("dup2 failed");
  66. exit (1);
  67. }
  68. close (fd[0]);
  69. pthread_t th;
  70. if (pthread_create (&th, NULL, tf, (void *) pthread_self ()) != 0)
  71. {
  72. puts ("create failed");
  73. exit (1);
  74. }
  75. execl (_PATH_BSHELL, _PATH_BSHELL, "-c", "echo $$", NULL);
  76. puts ("execl failed");
  77. exit (1);
  78. }
  79. close (fd[1]);
  80. char buf[200];
  81. ssize_t n;
  82. bool seen_pid = false;
  83. while (TEMP_FAILURE_RETRY ((n = read (fd[0], buf, sizeof (buf)))) > 0)
  84. {
  85. /* We only expect to read the PID. */
  86. char *endp;
  87. long int rpid = strtol (buf, &endp, 10);
  88. if (*endp != '\n')
  89. {
  90. printf ("didn't parse whole line: \"%s\"\n", buf);
  91. exit (1);
  92. }
  93. if (endp == buf)
  94. {
  95. puts ("read empty line");
  96. exit (1);
  97. }
  98. if (rpid != pid)
  99. {
  100. printf ("found \"%s\", expected PID %ld\n", buf, (long int) pid);
  101. exit (1);
  102. }
  103. if (seen_pid)
  104. {
  105. puts ("found more than one PID line");
  106. exit (1);
  107. }
  108. seen_pid = true;
  109. }
  110. close (fd[0]);
  111. int status;
  112. int err = waitpid (pid, &status, 0);
  113. if (err != pid)
  114. {
  115. puts ("waitpid failed");
  116. exit (1);
  117. }
  118. if (!seen_pid)
  119. {
  120. puts ("didn't get PID");
  121. exit (1);
  122. }
  123. return 0;
  124. }
  125. #define TEST_FUNCTION do_test ()
  126. #include "../test-skeleton.c"