]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
ext4: make sure to reset inode lockdep class when quota enabling fails
authorJan Kara <jack@suse.cz>
Thu, 7 Oct 2021 15:53:36 +0000 (17:53 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 27 Jan 2022 08:19:51 +0000 (09:19 +0100)
commit 4013d47a5307fdb5c13370b5392498b00fedd274 upstream.

When we succeed in enabling some quota type but fail to enable another
one with quota feature, we correctly disable all enabled quota types.
However we forget to reset i_data_sem lockdep class. When the inode gets
freed and reused, it will inherit this lockdep class (i_data_sem is
initialized only when a slab is created) and thus eventually lockdep
barfs about possible deadlocks.

Reported-and-tested-by: syzbot+3b6f9218b1301ddda3e2@syzkaller.appspotmail.com
Signed-off-by: Jan Kara <jack@suse.cz>
Cc: stable@kernel.org
Link: https://lore.kernel.org/r/20211007155336.12493-3-jack@suse.cz
Signed-off-by: Theodore Ts'o <tytso@mit.edu>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
fs/ext4/super.c

index f68dfef5939f4cda8ede99dcc5f97b42f9200f5f..e9ded68812816b25beacda8d07c47a2b7855cdd9 100644 (file)
@@ -5998,8 +5998,19 @@ static int ext4_enable_quotas(struct super_block *sb)
                                        "Failed to enable quota tracking "
                                        "(type=%d, err=%d). Please run "
                                        "e2fsck to fix.", type, err);
-                               for (type--; type >= 0; type--)
+                               for (type--; type >= 0; type--) {
+                                       struct inode *inode;
+
+                                       inode = sb_dqopt(sb)->files[type];
+                                       if (inode)
+                                               inode = igrab(inode);
                                        dquot_quota_off(sb, type);
+                                       if (inode) {
+                                               lockdep_set_quota_inode(inode,
+                                                       I_DATA_SEM_NORMAL);
+                                               iput(inode);
+                                       }
+                               }
 
                                return err;
                        }