Received: by 2002:a05:6a10:af89:0:0:0:0 with SMTP id iu9csp3654716pxb; Mon, 24 Jan 2022 14:32:58 -0800 (PST) X-Google-Smtp-Source: ABdhPJyOop/6d2GwEbj98cDpGnh/pI+JJ7MMt+YJck0W7MlU+m/ByCu9eAlzekerM2UFvR7Z58Jp X-Received: by 2002:aa7:888c:0:b0:4ca:75a3:aa1c with SMTP id z12-20020aa7888c000000b004ca75a3aa1cmr2595865pfe.65.1643063577944; Mon, 24 Jan 2022 14:32:57 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1643063577; cv=none; d=google.com; s=arc-20160816; b=OqjwWLPiCxeqUE6gc4+sXxEQkr2HDx2pzi5c+YkJAt/KvzybIRyu2syagn4U1B8jq0 US8WsFUyMQaWCT0RhpNH7YjjBd2m8xxlZjbNLB2vvz9fSH8CelbrtFad+aTOf56tiD8p r5HGe6mKlM1msZlciiJIJHP/60iIGPeXOS5hKQySbnF3P+Z+BkPc5knvtseVn65hVvwP sRo4xyDn0QQrTdU1Fg66jOWeI2l9CI86tXvr0YevuL4nP3qgyAO2BmsyUdGfRZpnYw22 dPmfPE6uqhPQRxxU5II92/uHi1DkiWpyQzSlPPRPHkYhqlBvpeQvx64ENPME6npA5uRS fVQA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=rZeFuDJ2Kl2ig95xmTBVgxNXtHln5o1mWnDlsVZEJJk=; b=wJuOU3FkJfmB1MsswacNyfD3idlsHvkABkI5OJ53UeESigucrVtWNva2NCce6b728+ yJwpDu2ZTN0FYavhuY6ZhRmDefgufuT/BGVSHtlLvVi5rqxaH+o7uS3vRZquTTAzxrvs DWg+SmZ5jmDjUcdxBEagQVbYcZrlkYpvRV8RGhbpHbtfRmx80UbBbRPop6AhaVQcBL52 GWnXCJnmUizN1Qlp3RGNRFGGbYRfO2y4S6QIyIxXmjJZ4+FczeKlvD307Cc3QsowVsrU sp2YxCY5Hy1+PunrOiszSjxsNOlNFkXBTaulp0Fg9kFeUPxVmUHu6zpG2xp9pK3g/OYP MINA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=DeJU6osT; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id m13si13560308pls.622.2022.01.24.14.32.45; Mon, 24 Jan 2022 14:32:57 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=DeJU6osT; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1587901AbiAXWaD (ORCPT + 99 others); Mon, 24 Jan 2022 17:30:03 -0500 Received: from ams.source.kernel.org ([145.40.68.75]:47772 "EHLO ams.source.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1456084AbiAXVhs (ORCPT ); Mon, 24 Jan 2022 16:37:48 -0500 Received: from smtp.kernel.org (relay.kernel.org [52.25.139.140]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by ams.source.kernel.org (Postfix) with ESMTPS id 4957CB812A7; Mon, 24 Jan 2022 21:37:43 +0000 (UTC) Received: by smtp.kernel.org (Postfix) with ESMTPSA id 617EAC340E4; Mon, 24 Jan 2022 21:37:41 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=linuxfoundation.org; s=korg; t=1643060262; bh=TrEyxXScUQN2ndIprFfTUzN2MsbyjUrfc//izk3BMCk=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=DeJU6osTOOhIoC29NF3vjDJ+1aRVrQI/+NQV4Ch9H8YyHm4BN0hH4XRF8dGo8EOYI xWll1l4ba5KQWc89WyLa4qqqP6mA/E59T0daOW+FwkO8wbKOgkjjEjx+VCuY3vT1OE pBAt8+b5EqXrTtrHOPEhgSBW4nqJZhtytMUtvcDg= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Chunguang Xu , kernel test robot , Jan Kara , Theodore Tso , stable@kernel.org Subject: [PATCH 5.16 0891/1039] ext4: fix a possible ABBA deadlock due to busy PA Date: Mon, 24 Jan 2022 19:44:40 +0100 Message-Id: <20220124184155.237744472@linuxfoundation.org> X-Mailer: git-send-email 2.34.1 In-Reply-To: <20220124184125.121143506@linuxfoundation.org> References: <20220124184125.121143506@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Chunguang Xu commit 8c80fb312d7abf8bcd66cca1d843a80318a2c522 upstream. We found on older kernel (3.10) that in the scenario of insufficient disk space, system may trigger an ABBA deadlock problem, it seems that this problem still exists in latest kernel, try to fix it here. The main process triggered by this problem is that task A occupies the PA and waits for the jbd2 transaction finish, the jbd2 transaction waits for the completion of task B's IO (plug_list), but task B waits for the release of PA by task A to finish discard, which indirectly forms an ABBA deadlock. The related calltrace is as follows: Task A vfs_write ext4_mb_new_blocks() ext4_mb_mark_diskspace_used() JBD2 jbd2_journal_get_write_access() -> jbd2_journal_commit_transaction() ->schedule() filemap_fdatawait() | | | Task B | | do_unlinkat() | | ext4_evict_inode() | | jbd2_journal_begin_ordered_truncate() | | filemap_fdatawrite_range() | | ext4_mb_new_blocks() | -ext4_mb_discard_group_preallocations() <----- Here, try to cancel ext4_mb_discard_group_preallocations() internal retry due to PA busy, and do a limited number of retries inside ext4_mb_discard_preallocations(), which can circumvent the above problems, but also has some advantages: 1. Since the PA is in a busy state, if other groups have free PAs, keeping the current PA may help to reduce fragmentation. 2. Continue to traverse forward instead of waiting for the current group PA to be released. In most scenarios, the PA discard time can be reduced. However, in the case of smaller free space, if only a few groups have space, then due to multiple traversals of the group, it may increase CPU overhead. But in contrast, I feel that the overall benefit is better than the cost. Signed-off-by: Chunguang Xu Reported-by: kernel test robot Reviewed-by: Jan Kara Link: https://lore.kernel.org/r/1637630277-23496-1-git-send-email-brookxu.cn@gmail.com Signed-off-by: Theodore Ts'o Cc: stable@kernel.org Signed-off-by: Greg Kroah-Hartman --- fs/ext4/mballoc.c | 40 ++++++++++++++++++---------------------- 1 file changed, 18 insertions(+), 22 deletions(-) --- a/fs/ext4/mballoc.c +++ b/fs/ext4/mballoc.c @@ -4814,7 +4814,7 @@ ext4_mb_release_group_pa(struct ext4_bud */ static noinline_for_stack int ext4_mb_discard_group_preallocations(struct super_block *sb, - ext4_group_t group, int needed) + ext4_group_t group, int *busy) { struct ext4_group_info *grp = ext4_get_group_info(sb, group); struct buffer_head *bitmap_bh = NULL; @@ -4822,8 +4822,7 @@ ext4_mb_discard_group_preallocations(str struct list_head list; struct ext4_buddy e4b; int err; - int busy = 0; - int free, free_total = 0; + int free = 0; mb_debug(sb, "discard preallocation for group %u\n", group); if (list_empty(&grp->bb_prealloc_list)) @@ -4846,19 +4845,14 @@ ext4_mb_discard_group_preallocations(str goto out_dbg; } - if (needed == 0) - needed = EXT4_CLUSTERS_PER_GROUP(sb) + 1; - INIT_LIST_HEAD(&list); -repeat: - free = 0; ext4_lock_group(sb, group); list_for_each_entry_safe(pa, tmp, &grp->bb_prealloc_list, pa_group_list) { spin_lock(&pa->pa_lock); if (atomic_read(&pa->pa_count)) { spin_unlock(&pa->pa_lock); - busy = 1; + *busy = 1; continue; } if (pa->pa_deleted) { @@ -4898,22 +4892,13 @@ repeat: call_rcu(&(pa)->u.pa_rcu, ext4_mb_pa_callback); } - free_total += free; - - /* if we still need more blocks and some PAs were used, try again */ - if (free_total < needed && busy) { - ext4_unlock_group(sb, group); - cond_resched(); - busy = 0; - goto repeat; - } ext4_unlock_group(sb, group); ext4_mb_unload_buddy(&e4b); put_bh(bitmap_bh); out_dbg: mb_debug(sb, "discarded (%d) blocks preallocated for group %u bb_free (%d)\n", - free_total, group, grp->bb_free); - return free_total; + free, group, grp->bb_free); + return free; } /* @@ -5455,13 +5440,24 @@ static int ext4_mb_discard_preallocation { ext4_group_t i, ngroups = ext4_get_groups_count(sb); int ret; - int freed = 0; + int freed = 0, busy = 0; + int retry = 0; trace_ext4_mb_discard_preallocations(sb, needed); + + if (needed == 0) + needed = EXT4_CLUSTERS_PER_GROUP(sb) + 1; + repeat: for (i = 0; i < ngroups && needed > 0; i++) { - ret = ext4_mb_discard_group_preallocations(sb, i, needed); + ret = ext4_mb_discard_group_preallocations(sb, i, &busy); freed += ret; needed -= ret; + cond_resched(); + } + + if (needed > 0 && busy && ++retry < 3) { + busy = 0; + goto repeat; } return freed;