]> git.ipfire.org Git - people/teissler/ipfire-2.x.git/blame - src/patches/glibc/glibc-rh794817.patch
Merge branch 'ppp-update' into beyond-next
[people/teissler/ipfire-2.x.git] / src / patches / glibc / glibc-rh794817.patch
CommitLineData
12788f63
MT
1From libc-alpha-return-25252-listarch-libc-alpha=sources dot redhat dot com at sourceware dot org Thu Feb 16 16:21:17 2012
2Return-Path: <libc-alpha-return-25252-listarch-libc-alpha=sources dot redhat dot com at sourceware dot org>
3Delivered-To: listarch-libc-alpha at sources dot redhat dot com
4Received: (qmail 5187 invoked by alias); 16 Feb 2012 16:21:14 -0000
5Delivered-To: moderator for libc-alpha at sourceware dot org
6Received: (qmail 2174 invoked by uid 22791); 16 Feb 2012 16:17:18 -0000
7X-SWARE-Spam-Status: No, hits=-2.0 required=5.0
8 tests=AWL,BAYES_00,RCVD_IN_DNSWL_NONE,SPF_HELO_PASS,TW_TV,TW_VB,TW_VF,T_RP_MATCHES_RCVD
9X-Spam-Check-By: sourceware.org
10Date: Thu, 16 Feb 2012 08:16:13 -0800
11From: Kees Cook <kees at outflux dot net>
12To: "Ryan S dot Arnold" <ryan dot arnold at gmail dot com>
13Cc: libc-alpha at sourceware dot org, Paul Eggert <eggert at cs dot ucla dot edu>,
14 Roland McGrath <roland at hack dot frob dot com>,
15 Andreas Schwab <schwab at linux-m68k dot org>
16Subject: Re: [PATCH] vfprintf: validate nargs and maybe allocate from heap
17Message-ID: <20120216161613.GZ20420@outflux.net>
18References: <20120206062537.GM4979@outflux.net>
19 <20120207000509 dot GP4989 at outflux dot net>
20 <20120210192457 dot GF20420 at outflux dot net>
21 <CAAKybw8AgkGsKAx=kvX4Tsi74f+HtuVnatTCB0VfsHi7vVFi1Q at mail dot gmail dot com>
22 <20120214223048 dot GM20420 at outflux dot net>
23 <CAAKybw_HS+cav+YcDw3ns7UXu6_xA7EHPrkiB87P+OGwEB0PVQ at mail dot gmail dot com>
24 <20120214224543 dot GN20420 at outflux dot net>
25MIME-Version: 1.0
26Content-Type: text/plain; charset=us-ascii
27Content-Disposition: inline
28In-Reply-To: <20120214224543 dot GN20420 at outflux dot net>
29X-MIMEDefang-Filter: outflux$Revision: 1.316 $
30X-HELO: www.outflux.net
31Mailing-List: contact libc-alpha-help at sourceware dot org; run by ezmlm
32Precedence: bulk
33List-Id: <libc-alpha.sourceware.org>
34List-Subscribe: <mailto:libc-alpha-subscribe at sourceware dot org>
35List-Archive: <http://sourceware.org/ml/libc-alpha/>
36List-Post: <mailto:libc-alpha at sourceware dot org>
37List-Help: <mailto:libc-alpha-help at sourceware dot org>, <http://sourceware dot org/ml/#faqs>
38Sender: libc-alpha-owner at sourceware dot org
39Delivered-To: mailing list libc-alpha at sourceware dot org
40
41The nargs value can overflow when doing allocations, allowing arbitrary
42memory writes via format strings, bypassing _FORTIFY_SOURCE:
43http://www.phrack.org/issues.html?issue=67&id=9
44
45This checks for nargs overflow and possibly allocates from heap instead of
46stack, and adds a regression test for the situation.
47
48I have FSF assignment via Google. (Sent from @outflux since that's how I'm
49subscribed here, but CL shows @chromium.org as part of my Google work.)
50
51This version disables the useless test on non-32-bit platforms.
52
532012-02-16 Kees Cook <keescook@chromium.org>
54
55 [BZ #13656]
56 * stdio-common/vfprintf.c (vfprintf): Check for nargs overflow and
57 possibly allocate from heap instead of stack.
58 * stdio-common/bug-vfprintf-nargs.c: New file.
59 * stdio-common/Makefile (tests): Add nargs overflow test.
60
61
62diff -rup a/stdio-common/Makefile b/stdio-common/Makefile
63--- a/stdio-common/Makefile 2010-05-04 05:27:23.000000000 -0600
64+++ b/stdio-common/Makefile 2012-02-20 21:57:52.983040992 -0700
65@@ -60,7 +60,7 @@ tests := tstscanf test_rdwr test-popen t
66 tst-popen tst-unlockedio tst-fmemopen2 tst-put-error tst-fgets \
67 tst-fwrite bug16 bug17 tst-swscanf tst-sprintf2 bug18 bug18a \
68 bug19 bug19a tst-popen2 scanf13 scanf14 scanf15 bug20 bug21 bug22 \
69- scanf16 scanf17 tst-setvbuf1
70+ scanf16 scanf17 tst-setvbuf1 bug-vfprintf-nargs
71
72 test-srcs = tst-unbputc tst-printf
73
74diff --git a/stdio-common/bug-vfprintf-nargs.c b/stdio-common/bug-vfprintf-nargs.c
75new file mode 100644
76index 0000000..13c66c0
77--- /dev/null
78+++ b/stdio-common/bug-vfprintf-nargs.c
79@@ -0,0 +1,78 @@
80+/* Test for vfprintf nargs allocation overflow (BZ #13656).
81+ Copyright (C) 2012 Free Software Foundation, Inc.
82+ This file is part of the GNU C Library.
83+ Contributed by Kees Cook <keescook@chromium.org>, 2012.
84+
85+ The GNU C Library is free software; you can redistribute it and/or
86+ modify it under the terms of the GNU Lesser General Public
87+ License as published by the Free Software Foundation; either
88+ version 2.1 of the License, or (at your option) any later version.
89+
90+ The GNU C Library is distributed in the hope that it will be useful,
91+ but WITHOUT ANY WARRANTY; without even the implied warranty of
92+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
93+ Lesser General Public License for more details.
94+
95+ You should have received a copy of the GNU Lesser General Public
96+ License along with the GNU C Library; if not, write to the Free
97+ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
98+ 02111-1307 USA. */
99+
100+#include <stdio.h>
101+#include <stdlib.h>
102+#include <stdint.h>
103+#include <unistd.h>
104+#include <inttypes.h>
105+#include <string.h>
106+#include <signal.h>
107+
108+static int
109+format_failed (const char *fmt, const char *expected)
110+{
111+ char output[80];
112+
113+ printf ("%s : ", fmt);
114+
115+ memset (output, 0, sizeof output);
116+ /* Having sprintf itself detect a failure is good. */
117+ if (sprintf (output, fmt, 1, 2, 3, "test") > 0
118+ && strcmp (output, expected) != 0)
119+ {
120+ printf ("FAIL (output '%s' != expected '%s')\n", output, expected);
121+ return 1;
122+ }
123+ puts ("ok");
124+ return 0;
125+}
126+
127+static int
128+do_test (void)
129+{
130+ int rc = 0;
131+ char buf[64];
132+
133+ /* Regular positionals work. */
134+ if (format_failed ("%1$d", "1") != 0)
135+ rc = 1;
136+
137+ /* Regular width positionals work. */
138+ if (format_failed ("%1$*2$d", " 1") != 0)
139+ rc = 1;
140+
141+ /* Positional arguments are constructed via read_int, so nargs can only
142+ overflow on 32-bit systems. On 64-bit systems, it will attempt to
143+ allocate a giant amount of memory and possibly crash, which is the
144+ expected situation. Since the 64-bit behavior is arch-specific, only
145+ test this on 32-bit systems. */
146+ if (sizeof (long int) == 4)
147+ {
148+ sprintf (buf, "%%1$d %%%" PRIdPTR "$d", UINT32_MAX / sizeof (int));
149+ if (format_failed (buf, "1 %$d") != 0)
150+ rc = 1;
151+ }
152+
153+ return rc;
154+}
155+
156+#define TEST_FUNCTION do_test ()
157+#include "../test-skeleton.c"
158diff --git a/stdio-common/vfprintf.c b/stdio-common/vfprintf.c
159index 863cd5d..022e72b 100644
160--- a/stdio-common/vfprintf.c
161+++ b/stdio-common/vfprintf.c
162@@ -235,6 +235,9 @@ vfprintf (FILE *s, const CHAR_T *format, va_list ap)
163 0 if unknown. */
164 int readonly_format = 0;
165
166+ /* For the argument descriptions, which may be allocated on the heap. */
167+ void *args_malloced = NULL;
168+
169 /* This table maps a character into a number representing a
170 class. In each step there is a destination label for each
171 class. */
172@@ -1647,9 +1650,10 @@ do_positional:
173 determine the size of the array needed to store the argument
174 attributes. */
175 size_t nargs = 0;
176- int *args_type;
177- union printf_arg *args_value = NULL;
178+ size_t bytes_per_arg;
179+ union printf_arg *args_value;
180 int *args_size;
181+ int *args_type;
182
183 /* Positional parameters refer to arguments directly. This could
184 also determine the maximum number of arguments. Track the
185@@ -1698,13 +1702,33 @@ do_positional:
186
187 /* Determine the number of arguments the format string consumes. */
188 nargs = MAX (nargs, max_ref_arg);
189+ bytes_per_arg = sizeof (*args_value) + sizeof (*args_size)
190+ + sizeof (*args_type);
191+
192+ /* Check for potential integer overflow. */
193+ if (nargs > SIZE_MAX / bytes_per_arg)
194+ {
195+ done = -1;
196+ goto all_done;
197+ }
198
199 /* Allocate memory for the argument descriptions. */
200- args_type = alloca (nargs * sizeof (int));
201+ if (__libc_use_alloca (nargs * bytes_per_arg))
202+ args_value = alloca (nargs * bytes_per_arg);
203+ else
204+ {
205+ args_value = args_malloced = malloc (nargs * bytes_per_arg);
206+ if (args_value == NULL)
207+ {
208+ done = -1;
209+ goto all_done;
210+ }
211+ }
212+
213+ args_size = &args_value[nargs].pa_int;
214+ args_type = &args_size[nargs];
215 memset (args_type, s->_flags2 & _IO_FLAGS2_FORTIFY ? '\xff' : '\0',
216- nargs * sizeof (int));
217- args_value = alloca (nargs * sizeof (union printf_arg));
218- args_size = alloca (nargs * sizeof (int));
219+ nargs * sizeof (*args_type));
220
221 /* XXX Could do sanity check here: If any element in ARGS_TYPE is
222 still zero after this loop, format is invalid. For now we
223@@ -1973,8 +1997,8 @@ do_positional:
224 }
225
226 all_done:
227- if (__builtin_expect (workstart != NULL, 0))
228- free (workstart);
229+ free (args_malloced);
230+ free (workstart);
231 /* Unlock the stream. */
232 _IO_funlockfile (s);
233 _IO_cleanup_region_end (0);
234--
2351.7.5.4
236
237--
238Kees Cook @outflux.net
239