mirror of git://sourceware.org/git/glibc.git
				
				
				
			
		
			
				
	
	
		
			238 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			238 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
| /* Check for file descriptor leak in alias :include: processing (bug 23521).
 | |
|    Copyright (C) 2018-2020 Free Software Foundation, Inc.
 | |
|    This file is part of the GNU C Library.
 | |
| 
 | |
|    The GNU C Library is free software; you can redistribute it and/or
 | |
|    modify it under the terms of the GNU Lesser General Public
 | |
|    License as published by the Free Software Foundation; either
 | |
|    version 2.1 of the License, or (at your option) any later version.
 | |
| 
 | |
|    The GNU C Library 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
 | |
|    Lesser General Public License for more details.
 | |
| 
 | |
|    You should have received a copy of the GNU Lesser General Public
 | |
|    License along with the GNU C Library; if not, see
 | |
|    <https://www.gnu.org/licenses/>.  */
 | |
| 
 | |
| #include <aliases.h>
 | |
| #include <array_length.h>
 | |
| #include <dlfcn.h>
 | |
| #include <errno.h>
 | |
| #include <gnu/lib-names.h>
 | |
| #include <nss.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <support/check.h>
 | |
| #include <support/namespace.h>
 | |
| #include <support/support.h>
 | |
| #include <support/temp_file.h>
 | |
| #include <support/test-driver.h>
 | |
| #include <support/xstdio.h>
 | |
| #include <support/xunistd.h>
 | |
| 
 | |
| static struct support_chroot *chroot_env;
 | |
| 
 | |
| /* Number of the aliases for the "many" user.  This must be large
 | |
|    enough to trigger reallocation for the pointer array, but result in
 | |
|    answers below the maximum size tried in do_test.  */
 | |
| enum { many_aliases = 30 };
 | |
| 
 | |
| static void
 | |
| prepare (int argc, char **argv)
 | |
| {
 | |
|   chroot_env = support_chroot_create
 | |
|     ((struct support_chroot_configuration) { } );
 | |
| 
 | |
|   char *path = xasprintf ("%s/etc/aliases", chroot_env->path_chroot);
 | |
|   add_temp_file (path);
 | |
|   support_write_file_string
 | |
|     (path,
 | |
|      "user1: :include:/etc/aliases.user1\n"
 | |
|      "user2: :include:/etc/aliases.user2\n"
 | |
|      "comment: comment1, :include:/etc/aliases.comment\n"
 | |
|      "many: :include:/etc/aliases.many\n");
 | |
|   free (path);
 | |
| 
 | |
|   path = xasprintf ("%s/etc/aliases.user1", chroot_env->path_chroot);
 | |
|   add_temp_file (path);
 | |
|   support_write_file_string (path, "alias1\n");
 | |
|   free (path);
 | |
| 
 | |
|   path = xasprintf ("%s/etc/aliases.user2", chroot_env->path_chroot);
 | |
|   add_temp_file (path);
 | |
|   support_write_file_string (path, "alias1a, alias2\n");
 | |
|   free (path);
 | |
| 
 | |
|   path = xasprintf ("%s/etc/aliases.comment", chroot_env->path_chroot);
 | |
|   add_temp_file (path);
 | |
|   support_write_file_string
 | |
|     (path,
 | |
|      /* The line must be longer than the line with the :include:
 | |
|         directive in /etc/aliases.  */
 | |
|      "# Long line.  ##############################################\n"
 | |
|      "comment2\n");
 | |
|   free (path);
 | |
| 
 | |
|   path = xasprintf ("%s/etc/aliases.many", chroot_env->path_chroot);
 | |
|   add_temp_file (path);
 | |
|   FILE *fp = xfopen (path, "w");
 | |
|   for (int i = 0; i < many_aliases; ++i)
 | |
|     fprintf (fp, "a%d\n", i);
 | |
|   TEST_VERIFY_EXIT (! ferror (fp));
 | |
|   xfclose (fp);
 | |
|   free (path);
 | |
| }
 | |
| 
 | |
| /* The names of the users to test.  */
 | |
| static const char *users[] = { "user1", "user2", "comment", "many" };
 | |
| 
 | |
| static void
 | |
| check_aliases (int id, const struct aliasent *e)
 | |
| {
 | |
|   TEST_VERIFY_EXIT (id >= 0 || id < array_length (users));
 | |
|   const char *name = users[id];
 | |
|   TEST_COMPARE_BLOB (e->alias_name, strlen (e->alias_name),
 | |
|                      name, strlen (name));
 | |
| 
 | |
|   switch (id)
 | |
|     {
 | |
|     case 0:
 | |
|       TEST_COMPARE (e->alias_members_len, 1);
 | |
|       TEST_COMPARE_BLOB (e->alias_members[0], strlen (e->alias_members[0]),
 | |
|                          "alias1", strlen ("alias1"));
 | |
|       break;
 | |
| 
 | |
|     case 1:
 | |
|       TEST_COMPARE (e->alias_members_len, 2);
 | |
|       TEST_COMPARE_BLOB (e->alias_members[0], strlen (e->alias_members[0]),
 | |
|                          "alias1a", strlen ("alias1a"));
 | |
|       TEST_COMPARE_BLOB (e->alias_members[1], strlen (e->alias_members[1]),
 | |
|                          "alias2", strlen ("alias2"));
 | |
|       break;
 | |
| 
 | |
|     case 2:
 | |
|       TEST_COMPARE (e->alias_members_len, 2);
 | |
|       TEST_COMPARE_BLOB (e->alias_members[0], strlen (e->alias_members[0]),
 | |
|                          "comment1", strlen ("comment1"));
 | |
|       TEST_COMPARE_BLOB (e->alias_members[1], strlen (e->alias_members[1]),
 | |
|                          "comment2", strlen ("comment2"));
 | |
|       break;
 | |
| 
 | |
|     case 3:
 | |
|       TEST_COMPARE (e->alias_members_len, many_aliases);
 | |
|       for (int i = 0; i < e->alias_members_len; ++i)
 | |
|         {
 | |
|           char alias[30];
 | |
|           int len = snprintf (alias, sizeof (alias), "a%d", i);
 | |
|           TEST_VERIFY_EXIT (len > 0);
 | |
|           TEST_COMPARE_BLOB (e->alias_members[i], strlen (e->alias_members[i]),
 | |
|                              alias, len);
 | |
|         }
 | |
|       break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static int
 | |
| do_test (void)
 | |
| {
 | |
|   /* Make sure we don't try to load the module in the chroot.  */
 | |
|   if (dlopen (LIBNSS_FILES_SO, RTLD_NOW) == NULL)
 | |
|     FAIL_EXIT1 ("could not load " LIBNSS_FILES_SO ": %s", dlerror ());
 | |
| 
 | |
|   /* Some of these descriptors will become unavailable if there is a
 | |
|      file descriptor leak.  10 is chosen somewhat arbitrarily.  The
 | |
|      array must be longer than the number of files opened by nss_files
 | |
|      at the same time (currently that number is 2).  */
 | |
|   int next_descriptors[10];
 | |
|   for (size_t i = 0; i < array_length (next_descriptors); ++i)
 | |
|     {
 | |
|       next_descriptors[i] = dup (0);
 | |
|       TEST_VERIFY_EXIT (next_descriptors[i] > 0);
 | |
|     }
 | |
|   for (size_t i = 0; i < array_length (next_descriptors); ++i)
 | |
|     xclose (next_descriptors[i]);
 | |
| 
 | |
|   support_become_root ();
 | |
|   if (!support_can_chroot ())
 | |
|     return EXIT_UNSUPPORTED;
 | |
| 
 | |
|   __nss_configure_lookup ("aliases", "files");
 | |
| 
 | |
|   xchroot (chroot_env->path_chroot);
 | |
| 
 | |
|   /* Attempt various buffer sizes.  If the operation succeeds, we
 | |
|      expect correct data.  */
 | |
|   for (int id = 0; id < array_length (users); ++id)
 | |
|     {
 | |
|       bool found = false;
 | |
|       for (size_t size = 1; size <= 1000; ++size)
 | |
|         {
 | |
|           void *buffer = malloc (size);
 | |
|           struct aliasent result;
 | |
|           struct aliasent *res;
 | |
|           errno = EINVAL;
 | |
|           int ret = getaliasbyname_r (users[id], &result, buffer, size, &res);
 | |
|           if (ret == 0)
 | |
|             {
 | |
|               if (res != NULL)
 | |
|                 {
 | |
|                   found = true;
 | |
|                   check_aliases (id, res);
 | |
|                 }
 | |
|               else
 | |
|                 {
 | |
|                   support_record_failure ();
 | |
|                   printf ("error: failed lookup for user \"%s\", size %zu\n",
 | |
|                           users[id], size);
 | |
|                 }
 | |
|             }
 | |
|           else if (ret != ERANGE)
 | |
|             {
 | |
|               support_record_failure ();
 | |
|               printf ("error: invalid return code %d (user \"%s\", size %zu)\n",
 | |
|                       ret, users[id], size);
 | |
|             }
 | |
|           free (buffer);
 | |
| 
 | |
|           /* Make sure that we did not have a file descriptor leak.  */
 | |
|           for (size_t i = 0; i < array_length (next_descriptors); ++i)
 | |
|             {
 | |
|               int new_fd = dup (0);
 | |
|               if (new_fd != next_descriptors[i])
 | |
|                 {
 | |
|                   support_record_failure ();
 | |
|                   printf ("error: descriptor %d at index %zu leaked"
 | |
|                           " (user \"%s\", size %zu)\n",
 | |
|                           next_descriptors[i], i, users[id], size);
 | |
| 
 | |
|                   /* Close unexpected descriptor, the leak probing
 | |
|                      descriptors, and the leaked descriptor
 | |
|                      next_descriptors[i].  */
 | |
|                   xclose (new_fd);
 | |
|                   for (size_t j = 0; j <= i; ++j)
 | |
|                     xclose (next_descriptors[j]);
 | |
|                   goto next_size;
 | |
|                 }
 | |
|             }
 | |
|           for (size_t i = 0; i < array_length (next_descriptors); ++i)
 | |
|             xclose (next_descriptors[i]);
 | |
| 
 | |
|         next_size:
 | |
|           ;
 | |
|         }
 | |
|       if (!found)
 | |
|         {
 | |
|           support_record_failure ();
 | |
|           printf ("error: user %s not found\n", users[id]);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   support_chroot_free (chroot_env);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| #define PREPARE prepare
 | |
| #include <support/test-driver.c>
 |