/* Test of posix_spawn() function with an inherited file descriptor 0. Copyright (C) 2008-2023 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 of the License, 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 . */ /* Written by Bruno Haible , 2020. */ /* Test whether passing a file descriptor open for reading, including the current file position, works. */ #include #include #include #include #include #include #include #include #include #define CHILD_PROGRAM_FILENAME "test-posix_spawn-inherit0" EXEEXT #define DATA_FILENAME "test-posix_spawn-inh0-data.tmp" static int parent_main (void) { FILE *fp; char *argv[3] = { CHILD_PROGRAM_FILENAME, "-child", NULL }; int err; pid_t child; int status; int exitstatus; /* Create a data file with specific contents. */ fp = fopen (DATA_FILENAME, "wb"); if (fp == NULL) { perror ("cannot create data file"); return 1; } fwrite ("Halle Potta", 1, 11, fp); if (fflush (fp) || fclose (fp)) { perror ("cannot prepare data file"); return 1; } /* Open the data file for reading. */ fp = freopen (DATA_FILENAME, "rb", stdin); if (fp == NULL) { perror ("cannot open data file"); return 1; } if (fflush (fp) || fseek (fp, 6, SEEK_SET)) /* needs gnulib module 'fflush' */ { perror ("cannot seek in data file"); return 1; } /* Test whether the child reads from fd 0 at the current file position. */ if ((err = posix_spawn (&child, CHILD_PROGRAM_FILENAME, NULL, NULL, argv, environ)) != 0) { errno = err; perror ("subprocess failed"); return 1; } status = 0; while (waitpid (child, &status, 0) != child) ; if (!WIFEXITED (status)) { fprintf (stderr, "subprocess terminated with unexpected wait status %d\n", status); return 1; } exitstatus = WEXITSTATUS (status); if (exitstatus != 0) { fprintf (stderr, "subprocess terminated with unexpected exit status %d\n", exitstatus); return 1; } if (fclose (fp)) { perror ("cannot close data file"); return 1; } /* Clean up data file. */ unlink (DATA_FILENAME); return 0; } static int child_main (void) { /* Read from STDIN_FILENO. */ char buf[1024]; int nread = fread (buf, 1, sizeof (buf), stdin); if (!(nread == 5 && memcmp (buf, "Potta", 5) == 0)) { fprintf (stderr, "child: read %d bytes, expected %d bytes\n", nread, 5); return 1; } return 0; } static void cleanup_then_die (int sig) { /* Clean up data file. */ unlink (DATA_FILENAME); /* Re-raise the signal and die from it. */ signal (sig, SIG_DFL); raise (sig); } int main (int argc, char *argv[]) { int exitstatus; if (!(argc > 1 && strcmp (argv[1], "-child") == 0)) { /* This is the parent process. */ signal (SIGINT, cleanup_then_die); signal (SIGTERM, cleanup_then_die); #ifdef SIGHUP signal (SIGHUP, cleanup_then_die); #endif exitstatus = parent_main (); } else { /* This is the child process. */ exitstatus = child_main (); } return exitstatus; }