9bb5d6
tst-env-setuid: Use support_capture_subprogram_self_sgid
9bb5d6
9bb5d6
Use the support_capture_subprogram_self_sgid to spawn an sgid child.
9bb5d6
Reviewed-by: Carlos O'Donell <carlos@redhat.com>
9bb5d6
9bb5d6
(cherry picked from commit ca335281068a1ed549a75ee64f90a8310755956f)
9bb5d6
9bb5d6
diff --git a/elf/tst-env-setuid.c b/elf/tst-env-setuid.c
9bb5d6
index 183a6dd133cfa16e..eda87f9dda293e79 100644
9bb5d6
--- a/elf/tst-env-setuid.c
9bb5d6
+++ b/elf/tst-env-setuid.c
9bb5d6
@@ -29,173 +29,12 @@
9bb5d6
 #include <sys/wait.h>
9bb5d6
 #include <unistd.h>
9bb5d6
 
9bb5d6
+#include <support/check.h>
9bb5d6
 #include <support/support.h>
9bb5d6
 #include <support/test-driver.h>
9bb5d6
+#include <support/capture_subprocess.h>
9bb5d6
 
9bb5d6
 static char SETGID_CHILD[] = "setgid-child";
9bb5d6
-#define CHILD_STATUS 42
9bb5d6
-
9bb5d6
-/* Return a GID which is not our current GID, but is present in the
9bb5d6
-   supplementary group list.  */
9bb5d6
-static gid_t
9bb5d6
-choose_gid (void)
9bb5d6
-{
9bb5d6
-  const int count = 64;
9bb5d6
-  gid_t groups[count];
9bb5d6
-  int ret = getgroups (count, groups);
9bb5d6
-  if (ret < 0)
9bb5d6
-    {
9bb5d6
-      printf ("getgroups: %m\n");
9bb5d6
-      exit (1);
9bb5d6
-    }
9bb5d6
-  gid_t current = getgid ();
9bb5d6
-  for (int i = 0; i < ret; ++i)
9bb5d6
-    {
9bb5d6
-      if (groups[i] != current)
9bb5d6
-	return groups[i];
9bb5d6
-    }
9bb5d6
-  return 0;
9bb5d6
-}
9bb5d6
-
9bb5d6
-/* Spawn and execute a program and verify that it returns the CHILD_STATUS.  */
9bb5d6
-static pid_t
9bb5d6
-do_execve (char **args)
9bb5d6
-{
9bb5d6
-  pid_t kid = vfork ();
9bb5d6
-
9bb5d6
-  if (kid < 0)
9bb5d6
-    {
9bb5d6
-      printf ("vfork: %m\n");
9bb5d6
-      return -1;
9bb5d6
-    }
9bb5d6
-
9bb5d6
-  if (kid == 0)
9bb5d6
-    {
9bb5d6
-      /* Child process.  */
9bb5d6
-      execve (args[0], args, environ);
9bb5d6
-      _exit (-errno);
9bb5d6
-    }
9bb5d6
-
9bb5d6
-  if (kid < 0)
9bb5d6
-    return 1;
9bb5d6
-
9bb5d6
-  int status;
9bb5d6
-
9bb5d6
-  if (waitpid (kid, &status, 0) < 0)
9bb5d6
-    {
9bb5d6
-      printf ("waitpid: %m\n");
9bb5d6
-      return 1;
9bb5d6
-    }
9bb5d6
-
9bb5d6
-  if (WEXITSTATUS (status) == EXIT_UNSUPPORTED)
9bb5d6
-    return EXIT_UNSUPPORTED;
9bb5d6
-
9bb5d6
-  if (!WIFEXITED (status) || WEXITSTATUS (status) != CHILD_STATUS)
9bb5d6
-    {
9bb5d6
-      printf ("Unexpected exit status %d from child process\n",
9bb5d6
-	      WEXITSTATUS (status));
9bb5d6
-      return 1;
9bb5d6
-    }
9bb5d6
-  return 0;
9bb5d6
-}
9bb5d6
-
9bb5d6
-/* Copies the executable into a restricted directory, so that we can
9bb5d6
-   safely make it SGID with the TARGET group ID.  Then runs the
9bb5d6
-   executable.  */
9bb5d6
-static int
9bb5d6
-run_executable_sgid (gid_t target)
9bb5d6
-{
9bb5d6
-  char *dirname = xasprintf ("%s/tst-tunables-setuid.%jd",
9bb5d6
-			     test_dir, (intmax_t) getpid ());
9bb5d6
-  char *execname = xasprintf ("%s/bin", dirname);
9bb5d6
-  int infd = -1;
9bb5d6
-  int outfd = -1;
9bb5d6
-  int ret = 0;
9bb5d6
-  if (mkdir (dirname, 0700) < 0)
9bb5d6
-    {
9bb5d6
-      printf ("mkdir: %m\n");
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-  infd = open ("/proc/self/exe", O_RDONLY);
9bb5d6
-  if (infd < 0)
9bb5d6
-    {
9bb5d6
-      printf ("open (/proc/self/exe): %m\n");
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-  outfd = open (execname, O_WRONLY | O_CREAT | O_EXCL, 0700);
9bb5d6
-  if (outfd < 0)
9bb5d6
-    {
9bb5d6
-      printf ("open (%s): %m\n", execname);
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-  char buf[4096];
9bb5d6
-  for (;;)
9bb5d6
-    {
9bb5d6
-      ssize_t rdcount = read (infd, buf, sizeof (buf));
9bb5d6
-      if (rdcount < 0)
9bb5d6
-	{
9bb5d6
-	  printf ("read: %m\n");
9bb5d6
-	  goto err;
9bb5d6
-	}
9bb5d6
-      if (rdcount == 0)
9bb5d6
-	break;
9bb5d6
-      char *p = buf;
9bb5d6
-      char *end = buf + rdcount;
9bb5d6
-      while (p != end)
9bb5d6
-	{
9bb5d6
-	  ssize_t wrcount = write (outfd, buf, end - p);
9bb5d6
-	  if (wrcount == 0)
9bb5d6
-	    errno = ENOSPC;
9bb5d6
-	  if (wrcount <= 0)
9bb5d6
-	    {
9bb5d6
-	      printf ("write: %m\n");
9bb5d6
-	      goto err;
9bb5d6
-	    }
9bb5d6
-	  p += wrcount;
9bb5d6
-	}
9bb5d6
-    }
9bb5d6
-  if (fchown (outfd, getuid (), target) < 0)
9bb5d6
-    {
9bb5d6
-      printf ("fchown (%s): %m\n", execname);
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-  if (fchmod (outfd, 02750) < 0)
9bb5d6
-    {
9bb5d6
-      printf ("fchmod (%s): %m\n", execname);
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-  if (close (outfd) < 0)
9bb5d6
-    {
9bb5d6
-      printf ("close (outfd): %m\n");
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-  if (close (infd) < 0)
9bb5d6
-    {
9bb5d6
-      printf ("close (infd): %m\n");
9bb5d6
-      goto err;
9bb5d6
-    }
9bb5d6
-
9bb5d6
-  char *args[] = {execname, SETGID_CHILD, NULL};
9bb5d6
-
9bb5d6
-  ret = do_execve (args);
9bb5d6
-
9bb5d6
-err:
9bb5d6
-  if (outfd >= 0)
9bb5d6
-    close (outfd);
9bb5d6
-  if (infd >= 0)
9bb5d6
-    close (infd);
9bb5d6
-  if (execname)
9bb5d6
-    {
9bb5d6
-      unlink (execname);
9bb5d6
-      free (execname);
9bb5d6
-    }
9bb5d6
-  if (dirname)
9bb5d6
-    {
9bb5d6
-      rmdir (dirname);
9bb5d6
-      free (dirname);
9bb5d6
-    }
9bb5d6
-  return ret;
9bb5d6
-}
9bb5d6
 
9bb5d6
 #ifndef test_child
9bb5d6
 static int
9bb5d6
@@ -256,40 +95,32 @@ do_test (int argc, char **argv)
9bb5d6
   if (argc == 2 && strcmp (argv[1], SETGID_CHILD) == 0)
9bb5d6
     {
9bb5d6
       if (getgid () == getegid ())
9bb5d6
-	{
9bb5d6
-	  /* This can happen if the file system is mounted nosuid.  */
9bb5d6
-	  fprintf (stderr, "SGID failed: GID and EGID match (%jd)\n",
9bb5d6
-		   (intmax_t) getgid ());
9bb5d6
-	  exit (EXIT_UNSUPPORTED);
9bb5d6
-	}
9bb5d6
+	/* This can happen if the file system is mounted nosuid.  */
9bb5d6
+	FAIL_UNSUPPORTED ("SGID failed: GID and EGID match (%jd)\n",
9bb5d6
+			  (intmax_t) getgid ());
9bb5d6
 
9bb5d6
       int ret = test_child ();
9bb5d6
 
9bb5d6
       if (ret != 0)
9bb5d6
 	exit (1);
9bb5d6
 
9bb5d6
-      exit (CHILD_STATUS);
9bb5d6
+      exit (EXIT_SUCCESS);
9bb5d6
     }
9bb5d6
   else
9bb5d6
     {
9bb5d6
       if (test_parent () != 0)
9bb5d6
 	exit (1);
9bb5d6
 
9bb5d6
-      /* Try running a setgid program.  */
9bb5d6
-      gid_t target = choose_gid ();
9bb5d6
-      if (target == 0)
9bb5d6
-	{
9bb5d6
-	  fprintf (stderr,
9bb5d6
-		   "Could not find a suitable GID for user %jd, skipping test\n",
9bb5d6
-		   (intmax_t) getuid ());
9bb5d6
-	  exit (0);
9bb5d6
-	}
9bb5d6
+      int status = support_capture_subprogram_self_sgid (SETGID_CHILD);
9bb5d6
 
9bb5d6
-      return run_executable_sgid (target);
9bb5d6
-    }
9bb5d6
+      if (WEXITSTATUS (status) == EXIT_UNSUPPORTED)
9bb5d6
+	return EXIT_UNSUPPORTED;
9bb5d6
+
9bb5d6
+      if (!WIFEXITED (status))
9bb5d6
+	FAIL_EXIT1 ("Unexpected exit status %d from child process\n", status);
9bb5d6
 
9bb5d6
-  /* Something went wrong and our argv was corrupted.  */
9bb5d6
-  _exit (1);
9bb5d6
+      return 0;
9bb5d6
+    }
9bb5d6
 }
9bb5d6
 
9bb5d6
 #define TEST_FUNCTION_ARGV do_test