• Al Viro's avatar
    fix dget_parent() fastpath race · 98e151ba
    Al Viro authored
    [ Upstream commit e8400933 ]
    
    We are overoptimistic about taking the fast path there; seeing
    the same value in ->d_parent after having grabbed a reference
    to that parent does *not* mean that it has remained our parent
    all along.
    
    That wouldn't be a big deal (in the end it is our parent and
    we have grabbed the reference we are about to return), but...
    the situation with barriers is messed up.
    
    We might have hit the following sequence:
    
    d is a dentry of /tmp/a/b
    CPU1:					CPU2:
    parent = d->d_parent (i.e. dentry of /tmp/a)
    					rename /tmp/a/b to /tmp/b
    					rmdir /tmp/a, making its dentry negative
    grab reference to parent,
    end up with cached parent->d_inode (NULL)
    					mkdir /tmp/a, rename /tmp/b to /tmp/a/b
    recheck d->d_parent, which is back to original
    decide that everything's fine and return the reference we'd got.
    
    The trouble is, caller (on CPU1) will observe dget_parent()
    returning an apparently negative dentry.  It actually is positive,
    but CPU1 has stale ->d_inode cached.
    
    Use d->d_seq to see if it has been moved instead of rechecking ->d_parent.
    NOTE: we are *NOT* going to retry on any kind of ->d_seq mismatch;
    we just go into the slow path in such case.  We don't wait for ->d_seq
    to become even either - again, if we are racing with renames, we
    can bloody well go to slow path anyway.
    Signed-off-by: default avatarAl Viro <viro@zeniv.linux.org.uk>
    Signed-off-by: default avatarSasha Levin <sashal@kernel.org>
    98e151ba
dcache.c 81.8 KB