]> git.baikalelectronics.ru Git - kernel.git/commitdiff
proc: Use new_inode not new_inode_pseudo
authorEric W. Biederman <ebiederm@xmission.com>
Fri, 12 Jun 2020 14:42:03 +0000 (09:42 -0500)
committerEric W. Biederman <ebiederm@xmission.com>
Fri, 12 Jun 2020 19:13:33 +0000 (14:13 -0500)
Recently syzbot reported that unmounting proc when there is an ongoing
inotify watch on the root directory of proc could result in a use
after free when the watch is removed after the unmount of proc
when the watcher exits.

Commit 55878bfe1385 ("proc: Remove the now unnecessary internal mount
of proc") made it easier to unmount proc and allowed syzbot to see the
problem, but looking at the code it has been around for a long time.

Looking at the code the fsnotify watch should have been removed by
fsnotify_sb_delete in generic_shutdown_super.  Unfortunately the inode
was allocated with new_inode_pseudo instead of new_inode so the inode
was not on the sb->s_inodes list.  Which prevented
fsnotify_unmount_inodes from finding the inode and removing the watch
as well as made it so the "VFS: Busy inodes after unmount" warning
could not find the inodes to warn about them.

Make all of the inodes in proc visible to generic_shutdown_super,
and fsnotify_sb_delete by using new_inode instead of new_inode_pseudo.
The only functional difference is that new_inode places the inodes
on the sb->s_inodes list.

I wrote a small test program and I can verify that without changes it
can trigger this issue, and by replacing new_inode_pseudo with
new_inode the issues goes away.

Cc: stable@vger.kernel.org
Link: https://lkml.kernel.org/r/000000000000d788c905a7dfa3f4@google.com
Reported-by: syzbot+7d2debdcdb3cb93c1e5e@syzkaller.appspotmail.com
Fixes: e1db54deb96e ("proc: Implement /proc/thread-self to point at the directory of the current thread")
Fixes: ebc401282106 ("procfs: switch /proc/self away from proc_dir_entry")
Fixes: 667a365bb0c4 ("vfs,proc: guarantee unique inodes in /proc")
Signed-off-by: "Eric W. Biederman" <ebiederm@xmission.com>
fs/proc/inode.c
fs/proc/self.c
fs/proc/thread_self.c

index f40c2532c057699773a2631838d7494778e190cc..28d6105e908e4c3f47803b519e9e68d6a8f3c7da 100644 (file)
@@ -617,7 +617,7 @@ const struct inode_operations proc_link_inode_operations = {
 
 struct inode *proc_get_inode(struct super_block *sb, struct proc_dir_entry *de)
 {
-       struct inode *inode = new_inode_pseudo(sb);
+       struct inode *inode = new_inode(sb);
 
        if (inode) {
                inode->i_ino = de->low_ino;
index ca5158fa561c0d6fc20fe048073d61547725da0f..72cd69bcaf4ad977ab7c3b134d91cfaaeee34026 100644 (file)
@@ -43,7 +43,7 @@ int proc_setup_self(struct super_block *s)
        inode_lock(root_inode);
        self = d_alloc_name(s->s_root, "self");
        if (self) {
-               struct inode *inode = new_inode_pseudo(s);
+               struct inode *inode = new_inode(s);
                if (inode) {
                        inode->i_ino = self_inum;
                        inode->i_mtime = inode->i_atime = inode->i_ctime = current_time(inode);
index ac284f409568ceb2b8f8f0559ccf55db4bad7d3b..a553273fbd41758df711b85c9647134b9eb1c631 100644 (file)
@@ -43,7 +43,7 @@ int proc_setup_thread_self(struct super_block *s)
        inode_lock(root_inode);
        thread_self = d_alloc_name(s->s_root, "thread-self");
        if (thread_self) {
-               struct inode *inode = new_inode_pseudo(s);
+               struct inode *inode = new_inode(s);
                if (inode) {
                        inode->i_ino = thread_self_inum;
                        inode->i_mtime = inode->i_atime = inode->i_ctime = current_time(inode);