Received: by 2002:a25:1506:0:0:0:0:0 with SMTP id 6csp6210351ybv; Tue, 18 Feb 2020 12:05:45 -0800 (PST) X-Google-Smtp-Source: APXvYqxjWL3F8rGKdo37NYev+DMNJz0T0d7p9ujNjsESQztTlRNhNufny66QUE1mPi0OtnCmSUUB X-Received: by 2002:a9d:6f85:: with SMTP id h5mr16867959otq.19.1582056344893; Tue, 18 Feb 2020 12:05:44 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1582056344; cv=none; d=google.com; s=arc-20160816; b=q4Ckmr+o7Qw9UpstKky1LSoa18EokRj1J9Fgau7Kj7PBQYAMnZpdHxA6i4KQMhVP3u NmqbFqnAbvhOjWOmBXp4+vwNhVhmZjlLjTYEmgXY2uzzuXGkzoLBkMVzr75Peh5l40Sw +LMR0nm+PaS+Msbt7LQo8bzlAsE9HwQfO1hxE1Xvocm8iph//SjnPa70SWD/p+wLXqN2 Gs132ZNXsbZT1ex7Mp7MoQAd7o7grez47ScVxDGgH/UxFGBOBF5WBUNDyudfd9XqVBfP TgKN7EBOUjw6ZSztQwzL5uomQzMMXLA0Go9Cu2XTXYmyh6ZxN18fnI1F0gMNAo2SJqy3 GkxA== 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:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=Xd//PvYtrBRZvooDVmWQAUal5PCBTmD/kd+rTWRuEmo=; b=k8N2s3UrfqmFHY2EyUQgmaqUBOnwknvBisKW0CQoFqzsmp8LT9flGqSsnbUbmzHxKj lTzWz17qFr8q7o4JgcgtkICIlPH6q4NlCDJHqEkxNZjy7ngC0FtJhtJG9WWTEuXSwFHU 5ED43sAQKY1wQWi2/zvuFlgth8CNgRz1DOvYUgaryu/xkjOSw363uWxv8n4dDYvSdJD/ F9OAWsHmzsMa/YJQfDdf4b2FIGswdwS/thWDmFVf9d9PcyTtkfKM2LzRjfpbrPSYQQGU a1+9xG9ienL8pDDPr0Yujan0YB45C+AkmhMNnDps4MMr60Gv9gF1RjWX3I42CYs4Sn8/ OPHA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=jw8p+AVB; 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 10si8206711ois.76.2020.02.18.12.05.32; Tue, 18 Feb 2020 12:05:44 -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=@kernel.org header.s=default header.b=jw8p+AVB; 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 S1728944AbgBRUDj (ORCPT + 99 others); Tue, 18 Feb 2020 15:03:39 -0500 Received: from mail.kernel.org ([198.145.29.99]:44966 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728964AbgBRUDi (ORCPT ); Tue, 18 Feb 2020 15:03:38 -0500 Received: from localhost (83-86-89-107.cable.dynamic.v4.ziggo.nl [83.86.89.107]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 4C1B921D56; Tue, 18 Feb 2020 20:03:36 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1582056216; bh=RJptZS0OKYne+gFv/LGiaLgBKwZ9QaewdZfuLKFKubs=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=jw8p+AVBzQypSeOopJ8CrgNDARSAaXwp6MSL8R9rsZWl93k9iaTbjo8R11EVnZyMV iJYxYhcGnUq3LmKKRvM3kF1TT0LGDDqXXTHiPn9jA2pOJfN1nfMM6BYb9fD2W8hrTF /D2wZkbSb8zXqwWTsjdvbSO65zqx+bhvHUWi4Fg4= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Jan Kara , "zhangyi (F)" , Theodore Tso , stable@kernel.org Subject: [PATCH 5.5 32/80] jbd2: do not clear the BH_Mapped flag when forgetting a metadata buffer Date: Tue, 18 Feb 2020 20:54:53 +0100 Message-Id: <20200218190435.536576320@linuxfoundation.org> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20200218190432.043414522@linuxfoundation.org> References: <20200218190432.043414522@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: zhangyi (F) commit c96dceeabf765d0b1b1f29c3bf50a5c01315b820 upstream. Commit 904cdbd41d74 ("jbd2: clear dirty flag when revoking a buffer from an older transaction") set the BH_Freed flag when forgetting a metadata buffer which belongs to the committing transaction, it indicate the committing process clear dirty bits when it is done with the buffer. But it also clear the BH_Mapped flag at the same time, which may trigger below NULL pointer oops when block_size < PAGE_SIZE. rmdir 1 kjournald2 mkdir 2 jbd2_journal_commit_transaction commit transaction N jbd2_journal_forget set_buffer_freed(bh1) jbd2_journal_commit_transaction commit transaction N+1 ... clear_buffer_mapped(bh1) ext4_getblk(bh2 ummapped) ... grow_dev_page init_page_buffers bh1->b_private=NULL bh2->b_private=NULL jbd2_journal_put_journal_head(jh1) __journal_remove_journal_head(hb1) jh1 is NULL and trigger oops *) Dir entry block bh1 and bh2 belongs to one page, and the bh2 has already been unmapped. For the metadata buffer we forgetting, we should always keep the mapped flag and clear the dirty flags is enough, so this patch pick out the these buffers and keep their BH_Mapped flag. Link: https://lore.kernel.org/r/20200213063821.30455-3-yi.zhang@huawei.com Fixes: 904cdbd41d74 ("jbd2: clear dirty flag when revoking a buffer from an older transaction") Reviewed-by: Jan Kara Signed-off-by: zhangyi (F) Signed-off-by: Theodore Ts'o Cc: stable@kernel.org Signed-off-by: Greg Kroah-Hartman --- fs/jbd2/commit.c | 25 +++++++++++++++++++++---- 1 file changed, 21 insertions(+), 4 deletions(-) --- a/fs/jbd2/commit.c +++ b/fs/jbd2/commit.c @@ -985,12 +985,29 @@ restart_loop: * pagesize and it is attached to the last partial page. */ if (buffer_freed(bh) && !jh->b_next_transaction) { + struct address_space *mapping; + clear_buffer_freed(bh); clear_buffer_jbddirty(bh); - clear_buffer_mapped(bh); - clear_buffer_new(bh); - clear_buffer_req(bh); - bh->b_bdev = NULL; + + /* + * Block device buffers need to stay mapped all the + * time, so it is enough to clear buffer_jbddirty and + * buffer_freed bits. For the file mapping buffers (i.e. + * journalled data) we need to unmap buffer and clear + * more bits. We also need to be careful about the check + * because the data page mapping can get cleared under + * out hands, which alse need not to clear more bits + * because the page and buffers will be freed and can + * never be reused once we are done with them. + */ + mapping = READ_ONCE(bh->b_page->mapping); + if (mapping && !sb_is_blkdev_sb(mapping->host->i_sb)) { + clear_buffer_mapped(bh); + clear_buffer_new(bh); + clear_buffer_req(bh); + bh->b_bdev = NULL; + } } if (buffer_jbddirty(bh)) {