Fix the regression created by "set S_DEAD on unlink()..." commit
authorAl Viro <viro@zeniv.linux.org.uk>
Fri, 30 Apr 2010 21:17:09 +0000 (17:17 -0400)
committerAl Viro <viro@zeniv.linux.org.uk>
Sat, 15 May 2010 11:16:33 +0000 (07:16 -0400)
commitd83c49f3e36cecd2e8823b6c48ffba083b8a5704
treea304de4eb43652d2a9528d4b43f798ab821c8b93
parent6a251b0ab67989f468f4cb65179e0cf40cf8c295
Fix the regression created by "set S_DEAD on unlink()..." commit

1) i_flags simply doesn't work for mount/unlink race prevention;
we may have many links to file and rm on one of those obviously
shouldn't prevent bind on top of another later on.  To fix it
right way we need to mark _dentry_ as unsuitable for mounting
upon; new flag (DCACHE_CANT_MOUNT) is protected by d_flags and
i_mutex on the inode in question.  Set it (with dont_mount(dentry))
in unlink/rmdir/etc., check (with cant_mount(dentry)) in places
in namespace.c that used to check for S_DEAD.  Setting S_DEAD
is still needed in places where we used to set it (for directories
getting killed), since we rely on it for readdir/rmdir race
prevention.

2) rename()/mount() protection has another bogosity - we unhash
the target before we'd checked that it's not a mountpoint.  Fixed.

3) ancient bogosity in pivot_root() - we locked i_mutex on the
right directory, but checked S_DEAD on the different (and wrong)
one.  Noticed and fixed.

Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
drivers/usb/core/inode.c
fs/configfs/dir.c
fs/namei.c
fs/namespace.c
include/linux/dcache.h