Received: by 10.223.185.116 with SMTP id b49csp104532wrg; Thu, 8 Mar 2018 13:42:39 -0800 (PST) X-Google-Smtp-Source: AG47ELv2eBpbG6IZ8bcy1ShVePcn0dZ01YgpCCWQ07LikmtZKA9WLmgyS894oXdU21OERf22oU/m X-Received: by 2002:a17:902:5489:: with SMTP id e9-v6mr22848600pli.81.1520545359462; Thu, 08 Mar 2018 13:42:39 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1520545359; cv=none; d=google.com; s=arc-20160816; b=AoFBLZkVBFefjC4D4dnBdG3uGcSMFi11fFn/GHB026T/PkCjw8gdOhmzkzGOIMB5AV cJD+6nG0ATOYeBTmuikpon6Zu7bG4RD8/CdcynH1lXRaa4OvBzWwaSNgT1jSiUwjpX/N YWLRXaR/06F6QmdAYc2HNWoQYcBTKgKVJDoV40U1zBQvk/2NJgu5sYKljBULsjU+eak6 UFX8EU6nJhz0/DQHC/PRODKRSgfBXTeS6MsnxEfXxU9Q8b/87hu+jBD+0Z21tCceF/Pa 0is0bvmpvqcyBKT8Hfrmr8otTZ39Rfc8hQ7h9Hhk2tftTn015oBcNYdTBe8thepb5GOP YBHg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-disposition:mime-version:message-id:subject:cc:to:from:date :dkim-signature:arc-authentication-results; bh=mXKp1jldFF0wj1CgXLNWuiL66o4SlunJLezhwmIPwLk=; b=VaRqYp6UR1SzeFChqvlGH9SW1+bFi+aEiM4hHjEz2W7dE70/uhtYyS9tyPyQ0z9zkm /eG57Bn7hefajCGBF4kE9iX5N8Xw9sup3N6q4k6j7G9rMHPYBbzgLZtvuFjiOAfYTZhV dwYAwLEqDY66Mg5xUXjUOA+BxEekTOJZVK73gt4aorNPpJC/xRoGnwgG4yhFYrWsW7cA XLV0d/oH8hdipw4SnVs8OvIKafainqhAnLUhHGp19Db5KmdYDmrSyIYfWmq3d9pnOd79 1G5dadRNZe+VGXUs+pCo6j9fVMwtViS/ZlqSwz8tqCVjmwEVHRj5rsrcbG/9VzvpMcX5 o6SQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=NVYCcqt0; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id v62si14357170pfi.156.2018.03.08.13.42.24; Thu, 08 Mar 2018 13:42:39 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=NVYCcqt0; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1751144AbeCHVkv (ORCPT + 99 others); Thu, 8 Mar 2018 16:40:51 -0500 Received: from mail-pl0-f66.google.com ([209.85.160.66]:40478 "EHLO mail-pl0-f66.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751143AbeCHVkt (ORCPT ); Thu, 8 Mar 2018 16:40:49 -0500 Received: by mail-pl0-f66.google.com with SMTP id i6-v6so4062181plt.7 for ; Thu, 08 Mar 2018 13:40:48 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=chromium.org; s=google; h=date:from:to:cc:subject:message-id:mime-version:content-disposition :content-transfer-encoding; bh=mXKp1jldFF0wj1CgXLNWuiL66o4SlunJLezhwmIPwLk=; b=NVYCcqt0c8y/dlWLaFZSvZ2a/KKqfBhCd532U6XsGaT73ISDHNylhLzQLC2FzUkYv9 JUA7HJZqSKOdEdDPzCVAxPi82bPjywGys2Ph42By9BhjtROkhuXiJ9/kxhf9ZCC4Ayj/ 7HG8hAmVdtGRTztfip3NanGbuz+OTeAUNUY18= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:message-id:mime-version :content-disposition:content-transfer-encoding; bh=mXKp1jldFF0wj1CgXLNWuiL66o4SlunJLezhwmIPwLk=; b=aIXgyj26gAq6FUnakgrPvVmUmielBWLMc5ZvU5g4kwOTk3IV2zXLeZqEflxrb83Emm TklFurg6Hquttc5EZRiZavU06zKYOhcY0TfA6ecn91UWSK+DkJzVWd5TrLfW05EVv1V9 Zo0ZHbf4MRKDJSHjrgwi6mTMgDGvad+D+/YrmD+I0g9DQgw5o5IgLeQr1DIcg2uBJeiS dKN6iRexikTZoMutVyJ/QKsX/dT3dUK4BLBMu9RylBCw7r1KzGidog7rScxA2A4KZiXu ArGg+Fegw7SljMKcIxOgKxbWUxFWYRz/QV5UAUoBds3Z+q9pwiy+qNsDojas25lqBzCW HesA== X-Gm-Message-State: APf1xPCsI0FhI2vI/7VDjeOQ94S6T19CrXMQkPaMn4PcTQ3rcHdzw+Fc 7alE2i4WESwE7hK8+u3h8FT/8w== X-Received: by 2002:a17:902:4601:: with SMTP id o1-v6mr25229793pld.210.1520545248540; Thu, 08 Mar 2018 13:40:48 -0800 (PST) Received: from www.outflux.net (173-164-112-133-Oregon.hfc.comcastbusiness.net. [173.164.112.133]) by smtp.gmail.com with ESMTPSA id a3sm42421417pfh.120.2018.03.08.13.40.46 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Thu, 08 Mar 2018 13:40:46 -0800 (PST) Date: Thu, 8 Mar 2018 13:40:45 -0800 From: Kees Cook To: Andrew Morton Cc: Josh Poimboeuf , Rasmus Villemoes , "Gustavo A. R. Silva" , "Tobin C. Harding" , rostedt@goodmis.org, corbet@lwn.net, Chris Mason , Josef Bacik , David Sterba , "David S. Miller" , Alexey Kuznetsov , Hideaki YOSHIFUJI , Ingo Molnar , Peter Zijlstra , Thomas Gleixner , Masahiro Yamada , Borislav Petkov , Randy Dunlap , Ian Abbott , Sergey Senozhatsky , Petr Mladek , Andy Shevchenko , Pantelis Antoniou , linux-btrfs@vger.kernel.org, netdev@vger.kernel.org, linux-kernel@vger.kernel.org, kernel-hardening@lists.openwall.com Subject: [PATCH] kernel.h: Skip single-eval logic on literals in min()/max() Message-ID: <20180308214045.GA6787@beast> MIME-Version: 1.0 Content-Type: text/plain; charset=utf-8 Content-Disposition: inline Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org When max() is used in stack array size calculations from literal values (e.g. "char foo[max(sizeof(struct1), sizeof(struct2))]", the compiler thinks this is a dynamic calculation due to the single-eval logic, which is not needed in the literal case. This change removes several accidental stack VLAs from an x86 allmodconfig build: $ diff -u before.txt after.txt | grep ^- -drivers/input/touchscreen/cyttsp4_core.c:871:2: warning: ISO C90 forbids variable length array ‘ids’ [-Wvla] -fs/btrfs/tree-checker.c:344:4: warning: ISO C90 forbids variable length array ‘namebuf’ [-Wvla] -lib/vsprintf.c:747:2: warning: ISO C90 forbids variable length array ‘sym’ [-Wvla] -net/ipv4/proc.c:403:2: warning: ISO C90 forbids variable length array ‘buff’ [-Wvla] -net/ipv6/proc.c:198:2: warning: ISO C90 forbids variable length array ‘buff’ [-Wvla] -net/ipv6/proc.c:218:2: warning: ISO C90 forbids variable length array ‘buff64’ [-Wvla] Based on an earlier patch from Josh Poimboeuf. Signed-off-by: Kees Cook --- include/linux/kernel.h | 42 ++++++++++++++++++++++++++++++------------ 1 file changed, 30 insertions(+), 12 deletions(-) diff --git a/include/linux/kernel.h b/include/linux/kernel.h index 3fd291503576..e0b39d461582 100644 --- a/include/linux/kernel.h +++ b/include/linux/kernel.h @@ -787,37 +787,57 @@ static inline void ftrace_dump(enum ftrace_dump_mode oops_dump_mode) { } * strict type-checking.. See the * "unnecessary" pointer comparison. */ -#define __min(t1, t2, min1, min2, x, y) ({ \ +#define __single_eval_min(t1, t2, min1, min2, x, y) ({ \ t1 min1 = (x); \ t2 min2 = (y); \ (void) (&min1 == &min2); \ min1 < min2 ? min1 : min2; }) +/* + * In the case of builtin constant values, there is no need to do the + * double-evaluation protection, so the raw comparison can be made. + * This allows min()/max() to be used in stack array allocations and + * avoid the compiler thinking it is a dynamic value leading to an + * accidental VLA. + */ +#define __min(t1, t2, x, y) \ + __builtin_choose_expr(__builtin_constant_p(x) && \ + __builtin_constant_p(y) && \ + __builtin_types_compatible_p(t1, t2), \ + (t1)(x) < (t2)(y) ? (t1)(x) : (t2)(y), \ + __single_eval_min(t1, t2, \ + __UNIQUE_ID(max1_), \ + __UNIQUE_ID(max2_), \ + x, y)) + /** * min - return minimum of two values of the same or compatible types * @x: first value * @y: second value */ -#define min(x, y) \ - __min(typeof(x), typeof(y), \ - __UNIQUE_ID(min1_), __UNIQUE_ID(min2_), \ - x, y) +#define min(x, y) __min(typeof(x), typeof(y), x, y) -#define __max(t1, t2, max1, max2, x, y) ({ \ +#define __single_eval_max(t1, t2, max1, max2, x, y) ({ \ t1 max1 = (x); \ t2 max2 = (y); \ (void) (&max1 == &max2); \ max1 > max2 ? max1 : max2; }) +#define __max(t1, t2, x, y) \ + __builtin_choose_expr(__builtin_constant_p(x) && \ + __builtin_constant_p(y) && \ + __builtin_types_compatible_p(t1, t2), \ + (t1)(x) > (t2)(y) ? (t1)(x) : (t2)(y), \ + __single_eval_max(t1, t2, \ + __UNIQUE_ID(max1_), \ + __UNIQUE_ID(max2_), \ + x, y)) /** * max - return maximum of two values of the same or compatible types * @x: first value * @y: second value */ -#define max(x, y) \ - __max(typeof(x), typeof(y), \ - __UNIQUE_ID(max1_), __UNIQUE_ID(max2_), \ - x, y) +#define max(x, y) __max(typeof(x), typeof(y), x, y) /** * min3 - return minimum of three values @@ -871,7 +891,6 @@ static inline void ftrace_dump(enum ftrace_dump_mode oops_dump_mode) { } */ #define min_t(type, x, y) \ __min(type, type, \ - __UNIQUE_ID(min1_), __UNIQUE_ID(min2_), \ x, y) /** @@ -882,7 +901,6 @@ static inline void ftrace_dump(enum ftrace_dump_mode oops_dump_mode) { } */ #define max_t(type, x, y) \ __max(type, type, \ - __UNIQUE_ID(min1_), __UNIQUE_ID(min2_), \ x, y) /** -- 2.7.4 -- Kees Cook Pixel Security