* gnu/packages/patches/glibc-2.35-CVE-2023-4911.patch: New file. * gnu/local.mk: Register it here. * gnu/packages/base.scm (glibc/fixed): New variable. (glibc): Use it as replacement.
		
			
				
	
	
		
			160 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
| From 1056e5b4c3f2d90ed2b4a55f96add28da2f4c8fa Mon Sep 17 00:00:00 2001
 | |
| From: Siddhesh Poyarekar <siddhesh@sourceware.org>
 | |
| Date: Tue, 19 Sep 2023 18:39:32 -0400
 | |
| Subject: [PATCH 1/1] tunables: Terminate if end of input is reached
 | |
|  (CVE-2023-4911)
 | |
| 
 | |
| The string parsing routine may end up writing beyond bounds of tunestr
 | |
| if the input tunable string is malformed, of the form name=name=val.
 | |
| This gets processed twice, first as name=name=val and next as name=val,
 | |
| resulting in tunestr being name=name=val:name=val, thus overflowing
 | |
| tunestr.
 | |
| 
 | |
| Terminate the parsing loop at the first instance itself so that tunestr
 | |
| does not overflow.
 | |
| 
 | |
| This also fixes up tst-env-setuid-tunables to actually handle failures
 | |
| correct and add new tests to validate the fix for this CVE.
 | |
| 
 | |
| Signed-off-by: Siddhesh Poyarekar <siddhesh@sourceware.org>
 | |
| Reviewed-by: Carlos O'Donell <carlos@redhat.com>
 | |
| ---
 | |
| Backported to 2.35 by Liliana Marie Prikler <liliana.prikler@gmail.com>
 | |
| 
 | |
|  NEWS                          |  5 +++++
 | |
|  elf/dl-tunables.c             | 17 +++++++++-------
 | |
|  elf/tst-env-setuid-tunables.c | 37 +++++++++++++++++++++++++++--------
 | |
|  3 files changed, 44 insertions(+), 15 deletions(-)
 | |
| 
 | |
| Index: glibc-2.35/NEWS
 | |
| ===================================================================
 | |
| --- glibc-2.35.orig/NEWS
 | |
| +++ glibc-2.35/NEWS
 | |
| @@ -199,6 +199,11 @@ Security related changes:
 | |
|    corresponds to the / directory through an unprivileged mount
 | |
|    namespace.  Reported by Qualys.
 | |
|  
 | |
| +  CVE-2023-4911: If a tunable of the form NAME=NAME=VAL is passed in the
 | |
| +  environment of a setuid program and NAME is valid, it may result in a
 | |
| +  buffer overflow, which could be exploited to achieve escalated
 | |
| +  privileges.  This flaw was introduced in glibc 2.34.
 | |
| +
 | |
|  The following bugs are resolved with this release:
 | |
|  
 | |
|    [12889] nptl: Race condition in pthread_kill
 | |
| Index: glibc-2.35/elf/dl-tunables.c
 | |
| ===================================================================
 | |
| --- glibc-2.35.orig/elf/dl-tunables.c
 | |
| +++ glibc-2.35/elf/dl-tunables.c
 | |
| @@ -187,11 +187,7 @@ parse_tunables (char *tunestr, char *val
 | |
|        /* If we reach the end of the string before getting a valid name-value
 | |
|  	 pair, bail out.  */
 | |
|        if (p[len] == '\0')
 | |
| -	{
 | |
| -	  if (__libc_enable_secure)
 | |
| -	    tunestr[off] = '\0';
 | |
| -	  return;
 | |
| -	}
 | |
| +	break;
 | |
|  
 | |
|        /* We did not find a valid name-value pair before encountering the
 | |
|  	 colon.  */
 | |
| @@ -251,9 +247,16 @@ parse_tunables (char *tunestr, char *val
 | |
|  	    }
 | |
|  	}
 | |
|  
 | |
| -      if (p[len] != '\0')
 | |
| -	p += len + 1;
 | |
| +      /* We reached the end while processing the tunable string.  */
 | |
| +      if (p[len] == '\0')
 | |
| +	break;
 | |
| +
 | |
| +      p += len + 1;
 | |
|      }
 | |
| +
 | |
| +  /* Terminate tunestr before we leave.  */
 | |
| +  if (__libc_enable_secure)
 | |
| +    tunestr[off] = '\0';
 | |
|  }
 | |
|  #endif
 | |
|  
 | |
| Index: glibc-2.35/elf/tst-env-setuid-tunables.c
 | |
| ===================================================================
 | |
| --- glibc-2.35.orig/elf/tst-env-setuid-tunables.c
 | |
| +++ glibc-2.35/elf/tst-env-setuid-tunables.c
 | |
| @@ -52,6 +52,8 @@ const char *teststrings[] =
 | |
|    "glibc.malloc.perturb=0x800:not_valid.malloc.check=2:glibc.malloc.mmap_threshold=4096",
 | |
|    "glibc.not_valid.check=2:glibc.malloc.mmap_threshold=4096",
 | |
|    "not_valid.malloc.check=2:glibc.malloc.mmap_threshold=4096",
 | |
| +  "glibc.malloc.mmap_threshold=glibc.malloc.mmap_threshold=4096",
 | |
| +  "glibc.malloc.check=2",
 | |
|    "glibc.malloc.garbage=2:glibc.maoc.mmap_threshold=4096:glibc.malloc.check=2",
 | |
|    "glibc.malloc.check=4:glibc.malloc.garbage=2:glibc.maoc.mmap_threshold=4096",
 | |
|    ":glibc.malloc.garbage=2:glibc.malloc.check=1",
 | |
| @@ -70,6 +72,8 @@ const char *resultstrings[] =
 | |
|    "glibc.malloc.perturb=0x800:glibc.malloc.mmap_threshold=4096",
 | |
|    "glibc.malloc.mmap_threshold=4096",
 | |
|    "glibc.malloc.mmap_threshold=4096",
 | |
| +  "glibc.malloc.mmap_threshold=glibc.malloc.mmap_threshold=4096",
 | |
| +  "",
 | |
|    "",
 | |
|    "",
 | |
|    "",
 | |
| @@ -89,6 +93,8 @@ test_child (int off)
 | |
|  
 | |
|    if (val != NULL)
 | |
|      printf ("[%d] Unexpected GLIBC_TUNABLES VALUE %s\n", off, val);
 | |
| +  else
 | |
| +    printf ("[%d] GLIBC_TUNABLES environment variable absent\n", off);
 | |
|  
 | |
|    return 1;
 | |
|  #else
 | |
| @@ -117,21 +123,26 @@ do_test (int argc, char **argv)
 | |
|        if (ret != 0)
 | |
|  	exit (1);
 | |
|  
 | |
| -      exit (EXIT_SUCCESS);
 | |
| +      /* Special return code to make sure that the child executed all the way
 | |
| +	 through.  */
 | |
| +      exit (42);
 | |
|      }
 | |
|    else
 | |
|      {
 | |
| -      int ret = 0;
 | |
| -
 | |
|        /* Spawn tests.  */
 | |
|        for (int i = 0; i < array_length (teststrings); i++)
 | |
|  	{
 | |
|  	  char buf[INT_BUFSIZE_BOUND (int)];
 | |
|  
 | |
| -	  printf ("Spawned test for %s (%d)\n", teststrings[i], i);
 | |
| +	  printf ("[%d] Spawned test for %s\n", i, teststrings[i]);
 | |
|  	  snprintf (buf, sizeof (buf), "%d\n", i);
 | |
| +	  fflush (stdout);
 | |
|  	  if (setenv ("GLIBC_TUNABLES", teststrings[i], 1) != 0)
 | |
| -	    exit (1);
 | |
| +	    {
 | |
| +	      printf ("    [%d] Failed to set GLIBC_TUNABLES: %m", i);
 | |
| +	      support_record_failure ();
 | |
| +	      continue;
 | |
| +	    }
 | |
|  
 | |
|  	  int status = support_capture_subprogram_self_sgid (buf);
 | |
|  
 | |
| @@ -139,9 +150,14 @@ do_test (int argc, char **argv)
 | |
|  	  if (WEXITSTATUS (status) == EXIT_UNSUPPORTED)
 | |
|  	    return EXIT_UNSUPPORTED;
 | |
|  
 | |
| -	  ret |= status;
 | |
| +	  if (WEXITSTATUS (status) != 42)
 | |
| +	    {
 | |
| +	      printf ("    [%d] child failed with status %d\n", i,
 | |
| +		      WEXITSTATUS (status));
 | |
| +	      support_record_failure ();
 | |
| +	    }
 | |
|  	}
 | |
| -      return ret;
 | |
| +      return 0;
 | |
|      }
 | |
|  }
 | |
|  
 |