Received: by 10.213.65.68 with SMTP id h4csp669570imn; Fri, 6 Apr 2018 07:03:55 -0700 (PDT) X-Google-Smtp-Source: AIpwx4+Np50K+nZhQIBvAppEqES86fdVi+1c1neSr+ADxR95wFiiQfLRCcLPdK0iJZVivYOkFGC3 X-Received: by 10.99.61.15 with SMTP id k15mr4996598pga.361.1523023435051; Fri, 06 Apr 2018 07:03:55 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1523023435; cv=none; d=google.com; s=arc-20160816; b=KcWJcCBls7OcVNW1jPBEN2/T+yeroWvLrIhJr7KmP+DXxQ/AuCdmhp+rVFmb1xJzOV XVnUM5mus7YpZkp4+kaqGWw4/OiOAm/72t12Jd2/eAzGOcFY5aJ/pmKkgtJLStEH+IUL 84Avo7HMwXlhywpYM3o5DJW4WOVld8HhJdUOB1gmt+9cmFfzNiGumWytZD/9RKS0xiSb bC9whvzBa2oWuAdHy5fEFIKmkwyG6u9uocQ/KRMzQyZmbOvfQqyU4QJ2cnUCB5uiYWZT WgQ9j3iOAz5skqEejrpHjU0NSDXQboUxPZaBII0azzVckYnPBxX5iJ2dHFKqBS4WJO5b i6Ew== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:mime-version:user-agent:references :in-reply-to:message-id:date:subject:cc:to:from :arc-authentication-results; bh=tr6c58aEcwKQ9tnuhiOPxKMWGjszOlEli4c02To2DoE=; b=cwHDXor5hHX/E5ZUUU4L/xZQGYJfgx3wB62Ztwk/yhVpK3PWDEsMDSZvUleq3E/sov kzXLrGRnl6y4vD0Po9CeZo9kAew388+Ypi1b360YSlrFnMxXxfyLWdf4ekQ0gnttrEx5 wdvdVw/aN6lHd8oPYYFf7DnKGBFFRbU2UzKIfuWwFpFWX1KyHdliQs7jaPOxQmgGgBub csRkvdo3svUniaWgVtA6FFgM89dxUq6/zTmVUxQYhuF4wDyH8n17RuP0roIqZMdQYQ60 zJexSfuiuSVcyyNJXtMARuVubY02x8MlTikzpby1vpFIwVWK6K8D3bid9p8zNWBzz4bH Rg6Q== ARC-Authentication-Results: i=1; mx.google.com; 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 Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id i4si7214966pgr.820.2018.04.06.07.03.40; Fri, 06 Apr 2018 07:03:54 -0700 (PDT) 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; 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 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1756627AbeDFOBd (ORCPT + 99 others); Fri, 6 Apr 2018 10:01:33 -0400 Received: from mail.linuxfoundation.org ([140.211.169.12]:36882 "EHLO mail.linuxfoundation.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1752281AbeDFNmR (ORCPT ); Fri, 6 Apr 2018 09:42:17 -0400 Received: from localhost (LFbn-1-12247-202.w90-92.abo.wanadoo.fr [90.92.61.202]) by mail.linuxfoundation.org (Postfix) with ESMTPSA id 4CA3DC64; Fri, 6 Apr 2018 13:42:16 +0000 (UTC) From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, stable@kernel.org, "Erhard F." , Matthew Wilcox , Rasmus Villemoes , Andrew Morton , Arnd Bergmann , Omar Sandoval , Linus Torvalds Subject: [PATCH 4.15 40/72] bitmap: fix memset optimization on big-endian systems Date: Fri, 6 Apr 2018 15:24:15 +0200 Message-Id: <20180406084352.504891273@linuxfoundation.org> X-Mailer: git-send-email 2.17.0 In-Reply-To: <20180406084349.367583460@linuxfoundation.org> References: <20180406084349.367583460@linuxfoundation.org> User-Agent: quilt/0.65 X-stable: review MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org 4.15-stable review patch. If anyone has any objections, please let me know. ------------------ From: Omar Sandoval commit 21035965f60b0502fc6537b232839389bb4ce664 upstream. Commit 2a98dc028f91 ("include/linux/bitmap.h: turn bitmap_set and bitmap_clear into memset when possible") introduced an optimization to bitmap_{set,clear}() which uses memset() when the start and length are constants aligned to a byte. This is wrong on big-endian systems; our bitmaps are arrays of unsigned long, so bit n is not at byte n / 8 in memory. This was caught by the Btrfs selftests, but the bitmap selftests also fail when run on a big-endian machine. We can still use memset if the start and length are aligned to an unsigned long, so do that on big-endian. The same problem applies to the memcmp in bitmap_equal(), so fix it there, too. Fixes: 2a98dc028f91 ("include/linux/bitmap.h: turn bitmap_set and bitmap_clear into memset when possible") Fixes: 2c6deb01525a ("bitmap: use memcmp optimisation in more situations") Cc: stable@kernel.org Reported-by: "Erhard F." Cc: Matthew Wilcox Cc: Rasmus Villemoes Cc: Andrew Morton Cc: Arnd Bergmann Signed-off-by: Omar Sandoval Signed-off-by: Linus Torvalds Signed-off-by: Greg Kroah-Hartman --- include/linux/bitmap.h | 22 +++++++++++++++++----- 1 file changed, 17 insertions(+), 5 deletions(-) --- a/include/linux/bitmap.h +++ b/include/linux/bitmap.h @@ -271,12 +271,20 @@ static inline void bitmap_complement(uns __bitmap_complement(dst, src, nbits); } +#ifdef __LITTLE_ENDIAN +#define BITMAP_MEM_ALIGNMENT 8 +#else +#define BITMAP_MEM_ALIGNMENT (8 * sizeof(unsigned long)) +#endif +#define BITMAP_MEM_MASK (BITMAP_MEM_ALIGNMENT - 1) + static inline int bitmap_equal(const unsigned long *src1, const unsigned long *src2, unsigned int nbits) { if (small_const_nbits(nbits)) return !((*src1 ^ *src2) & BITMAP_LAST_WORD_MASK(nbits)); - if (__builtin_constant_p(nbits & 7) && IS_ALIGNED(nbits, 8)) + if (__builtin_constant_p(nbits & BITMAP_MEM_MASK) && + IS_ALIGNED(nbits, BITMAP_MEM_ALIGNMENT)) return !memcmp(src1, src2, nbits / 8); return __bitmap_equal(src1, src2, nbits); } @@ -327,8 +335,10 @@ static __always_inline void bitmap_set(u { if (__builtin_constant_p(nbits) && nbits == 1) __set_bit(start, map); - else if (__builtin_constant_p(start & 7) && IS_ALIGNED(start, 8) && - __builtin_constant_p(nbits & 7) && IS_ALIGNED(nbits, 8)) + else if (__builtin_constant_p(start & BITMAP_MEM_MASK) && + IS_ALIGNED(start, BITMAP_MEM_ALIGNMENT) && + __builtin_constant_p(nbits & BITMAP_MEM_MASK) && + IS_ALIGNED(nbits, BITMAP_MEM_ALIGNMENT)) memset((char *)map + start / 8, 0xff, nbits / 8); else __bitmap_set(map, start, nbits); @@ -339,8 +349,10 @@ static __always_inline void bitmap_clear { if (__builtin_constant_p(nbits) && nbits == 1) __clear_bit(start, map); - else if (__builtin_constant_p(start & 7) && IS_ALIGNED(start, 8) && - __builtin_constant_p(nbits & 7) && IS_ALIGNED(nbits, 8)) + else if (__builtin_constant_p(start & BITMAP_MEM_MASK) && + IS_ALIGNED(start, BITMAP_MEM_ALIGNMENT) && + __builtin_constant_p(nbits & BITMAP_MEM_MASK) && + IS_ALIGNED(nbits, BITMAP_MEM_ALIGNMENT)) memset((char *)map + start / 8, 0, nbits / 8); else __bitmap_clear(map, start, nbits);