[GFS2] Fix recursive locking attempt with NFS
In certain cases, its possible for NFS to call the lookup code while holding the glock (when doing a readdirplus operation) so we need to check for that and not try and lock the glock twice. This also fixes a typo in a previous NFS related GFS2 patch. Signed-off-by: Steven Whitehouse <swhiteho@redhat.com>
This commit is contained in:
parent
b790c3b7c3
commit
d7c103d0bd
|
@ -395,8 +395,10 @@ struct inode *gfs2_lookup_simple(struct inode *dip, const char *name)
|
|||
* @is_root: If 1, ignore the caller's permissions
|
||||
* @i_gh: An uninitialized holder for the new inode glock
|
||||
*
|
||||
* There will always be a vnode (Linux VFS inode) for the d_gh inode unless
|
||||
* @is_root is true.
|
||||
* This can be called via the VFS filldir function when NFS is doing
|
||||
* a readdirplus and the inode which its intending to stat isn't
|
||||
* already in cache. In this case we must not take the directory glock
|
||||
* again, since the readdir call will have already taken that lock.
|
||||
*
|
||||
* Returns: errno
|
||||
*/
|
||||
|
@ -409,8 +411,9 @@ struct inode *gfs2_lookupi(struct inode *dir, const struct qstr *name,
|
|||
struct gfs2_holder d_gh;
|
||||
struct gfs2_inum_host inum;
|
||||
unsigned int type;
|
||||
int error = 0;
|
||||
int error;
|
||||
struct inode *inode = NULL;
|
||||
int unlock = 0;
|
||||
|
||||
if (!name->len || name->len > GFS2_FNAMESIZE)
|
||||
return ERR_PTR(-ENAMETOOLONG);
|
||||
|
@ -422,9 +425,12 @@ struct inode *gfs2_lookupi(struct inode *dir, const struct qstr *name,
|
|||
return dir;
|
||||
}
|
||||
|
||||
error = gfs2_glock_nq_init(dip->i_gl, LM_ST_SHARED, 0, &d_gh);
|
||||
if (error)
|
||||
return ERR_PTR(error);
|
||||
if (gfs2_glock_is_locked_by_me(dip->i_gl) == 0) {
|
||||
error = gfs2_glock_nq_init(dip->i_gl, LM_ST_SHARED, 0, &d_gh);
|
||||
if (error)
|
||||
return ERR_PTR(error);
|
||||
unlock = 1;
|
||||
}
|
||||
|
||||
if (!is_root) {
|
||||
error = permission(dir, MAY_EXEC, NULL);
|
||||
|
@ -439,10 +445,11 @@ struct inode *gfs2_lookupi(struct inode *dir, const struct qstr *name,
|
|||
inode = gfs2_inode_lookup(sb, &inum, type);
|
||||
|
||||
out:
|
||||
gfs2_glock_dq_uninit(&d_gh);
|
||||
if (unlock)
|
||||
gfs2_glock_dq_uninit(&d_gh);
|
||||
if (error == -ENOENT)
|
||||
return NULL;
|
||||
return inode;
|
||||
return inode ? inode : ERR_PTR(error);
|
||||
}
|
||||
|
||||
static int pick_formal_ino_1(struct gfs2_sbd *sdp, u64 *formal_ino)
|
||||
|
|
|
@ -1018,7 +1018,7 @@ static int gfs2_getattr(struct vfsmount *mnt, struct dentry *dentry,
|
|||
}
|
||||
|
||||
generic_fillattr(inode, stat);
|
||||
if (unlock);
|
||||
if (unlock)
|
||||
gfs2_glock_dq_uninit(&gh);
|
||||
|
||||
return 0;
|
||||
|
|
Loading…
Reference in New Issue