File: test-posix_spawnp-vfork.c

package info (click to toggle)
coreutils 9.10-1
  • links: PTS
  • area: main
  • in suites: sid
  • size: 70,560 kB
  • sloc: ansic: 253,546; sh: 30,931; perl: 8,141; yacc: 1,846; makefile: 198; python: 47; sed: 16
file content (106 lines) | stat: -rw-r--r-- 3,745 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
/* Test of posix_spawnp() function.
   Copyright (C) 2020-2026 Free Software Foundation, Inc.

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation, either version 3, or (at your option)
   any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program; if not, see <https://www.gnu.org/licenses/>.  */

#include <config.h>

#include <spawn.h>

#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>

#include "macros.h"

/* Check an invocation via vfork() with a large $PATH (ca. 4 KB).
   When configured with gl_cv_func_posix_spawnp_secure_exec=no, this test used
   to crash on Linux/SPARC, Linux/ppc64, Linux/ppc64le, and Solaris/SPARC.  */

int
main ()
{
#if defined _WIN32 && !defined __CYGWIN__
  /* There is no 'false' program on this platform, but no vfork() either.  */
  fprintf (stderr, "Skipping test: no vfork() on this platform.\n");
  return 77;
#else

  /* Components of file names are limited to ca. 255 characters on most
     platforms.  (Cf. NAME_MAX in <limits.h>.)
     Here is one with 250 characters.  */
# define OOO "oooooooooooooooooooooooooooooooooooooooooooooooooo" \
             "oooooooooooooooooooooooooooooooooooooooooooooooooo" \
             "oooooooooooooooooooooooooooooooooooooooooooooooooo" \
             "oooooooooooooooooooooooooooooooooooooooooooooooooo" \
             "oooooooooooooooooooooooooooooooooooooooooooooooooo"
  /* File names are limited to ca. 1024 characters on many platforms,
     to ca. 4096 characters only on Linux and Cygwin, and unlimited only
     on GNU/Hurd.  (Cf. PATH_MAX in <limits.h>.)
     We prepend several directory names below this 1024 limit.  */
# define ADD "/usr/bin/p" OOO "/" OOO "/" OOO "/" OOO ":" \
             "/usr/bin/q" OOO "/" OOO "/" OOO "/" OOO ":" \
             "/usr/bin/r" OOO "/" OOO "/" OOO "/" OOO ":" \
             "/usr/bin/s" OOO "/" OOO "/" OOO "/" OOO ":"
  const char *old_path = getenv ("PATH");
  char *new_path = (char *) malloc (strlen (ADD) + strlen (old_path) + 1);
  ASSERT (new_path != NULL);
  memcpy (new_path, ADD, strlen (ADD));
  strcpy (new_path + strlen (ADD), old_path);
  ASSERT (setenv ("PATH", new_path, 1) == 0);

  posix_spawnattr_t attr;
  ASSERT (posix_spawnattr_init (&attr) == 0);
  ASSERT (posix_spawnattr_setflags (&attr, POSIX_SPAWN_USEVFORK) == 0);

  if (test_exit_status != EXIT_SUCCESS)
    return test_exit_status;

  /* Can't use 'false' here, because its exit status is only guaranteed to be
     non-zero, and is in fact 255 on Solaris.  */
  const char *prog = "sh";
  const char *prog_argv[4] = { prog, "-c", "exit 1", NULL };
  pid_t child;
  int err = posix_spawnp (&child, prog, NULL, &attr, (char **) prog_argv, environ);
  if (err != 0)
    {
      errno = err;
      perror ("posix_spawn");
      return 1;
    }

  /* Wait for child.  */
  int status = 0;
  while (waitpid (child, &status, 0) != child)
    ;
  if (!WIFEXITED (status))
    {
      fprintf (stderr, "subprocess terminated with unexpected wait status %d\n", status);
      return 1;
    }
  int exitstatus = WEXITSTATUS (status);
  if (exitstatus != 1)
    {
      fprintf (stderr, "subprocess terminated with unexpected exit status %d\n", exitstatus);
      return 1;
    }

  return test_exit_status;

#endif
}