Skip to content

Commit

Permalink
staging/lustre/llite: fix open lock matching in ll_md_blocking_ast()
Browse files Browse the repository at this point in the history
In ll_md_blocking_ast() match open locks before all others, ensuring
that MDS_INODELOCK_OPEN is not cleared from bits by another open lock
with a different mode. Change the int flags parameter of
ll_md_real_close() to fmode_t fmode. Clean up verious style issues in
both functions.

Signed-off-by: John L. Hammond <[email protected]>
Reviewed-on: http://review.whamcloud.com/8718
Intel-bug-id: https://jira.hpdd.intel.com/browse/LU-4429
Reviewed-by: Niu Yawei <[email protected]>
Reviewed-by: Jinshan Xiong <[email protected]>
Reviewed-by: Oleg Drokin <[email protected]>
Signed-off-by: Oleg Drokin <[email protected]>
Signed-off-by: Greg Kroah-Hartman <[email protected]>
  • Loading branch information
jhammond-intel authored and gregkh committed Mar 1, 2014
1 parent da62fa2 commit 45b2a01
Show file tree
Hide file tree
Showing 3 changed files with 54 additions and 45 deletions.
19 changes: 11 additions & 8 deletions drivers/staging/lustre/lustre/llite/file.c
Original file line number Diff line number Diff line change
Expand Up @@ -205,38 +205,41 @@ static int ll_close_inode_openhandle(struct obd_export *md_exp,
return rc;
}

int ll_md_real_close(struct inode *inode, int flags)
int ll_md_real_close(struct inode *inode, fmode_t fmode)
{
struct ll_inode_info *lli = ll_i2info(inode);
struct obd_client_handle **och_p;
struct obd_client_handle *och;
__u64 *och_usecount;
int rc = 0;

if (flags & FMODE_WRITE) {
if (fmode & FMODE_WRITE) {
och_p = &lli->lli_mds_write_och;
och_usecount = &lli->lli_open_fd_write_count;
} else if (flags & FMODE_EXEC) {
} else if (fmode & FMODE_EXEC) {
och_p = &lli->lli_mds_exec_och;
och_usecount = &lli->lli_open_fd_exec_count;
} else {
LASSERT(flags & FMODE_READ);
LASSERT(fmode & FMODE_READ);
och_p = &lli->lli_mds_read_och;
och_usecount = &lli->lli_open_fd_read_count;
}

mutex_lock(&lli->lli_och_mutex);
if (*och_usecount) { /* There are still users of this handle, so
skip freeing it. */
if (*och_usecount > 0) {
/* There are still users of this handle, so skip
* freeing it. */
mutex_unlock(&lli->lli_och_mutex);
return 0;
}

och=*och_p;
*och_p = NULL;
mutex_unlock(&lli->lli_och_mutex);

if (och) { /* There might be a race and somebody have freed this och
already */
if (och != NULL) {
/* There might be a race and this handle may already
be closed. */
rc = ll_close_inode_openhandle(ll_i2sbi(inode)->ll_md_exp,
inode, och, NULL);
}
Expand Down
2 changes: 1 addition & 1 deletion drivers/staging/lustre/lustre/llite/llite_internal.h
Original file line number Diff line number Diff line change
Expand Up @@ -775,7 +775,7 @@ int ll_local_open(struct file *file,
int ll_release_openhandle(struct dentry *, struct lookup_intent *);
int ll_md_close(struct obd_export *md_exp, struct inode *inode,
struct file *file);
int ll_md_real_close(struct inode *inode, int flags);
int ll_md_real_close(struct inode *inode, fmode_t fmode);
void ll_ioepoch_close(struct inode *inode, struct md_op_data *op_data,
struct obd_client_handle **och, unsigned long flags);
void ll_done_writing_attr(struct inode *inode, struct md_op_data *op_data);
Expand Down
78 changes: 42 additions & 36 deletions drivers/staging/lustre/lustre/llite/namei.c
Original file line number Diff line number Diff line change
Expand Up @@ -195,101 +195,107 @@ static void ll_invalidate_negative_children(struct inode *dir)
int ll_md_blocking_ast(struct ldlm_lock *lock, struct ldlm_lock_desc *desc,
void *data, int flag)
{
int rc;
struct lustre_handle lockh;
int rc;

switch (flag) {
case LDLM_CB_BLOCKING:
ldlm_lock2handle(lock, &lockh);
rc = ldlm_cli_cancel(&lockh, LCF_ASYNC);
if (rc < 0) {
CDEBUG(D_INODE, "ldlm_cli_cancel: %d\n", rc);
CDEBUG(D_INODE, "ldlm_cli_cancel: rc = %d\n", rc);
return rc;
}
break;
case LDLM_CB_CANCELING: {
struct inode *inode = ll_inode_from_resource_lock(lock);
struct ll_inode_info *lli;
__u64 bits = lock->l_policy_data.l_inodebits.bits;
struct lu_fid *fid;
ldlm_mode_t mode = lock->l_req_mode;

/* Inode is set to lock->l_resource->lr_lvb_inode
* for mdc - bug 24555 */
LASSERT(lock->l_ast_data == NULL);

/* Invalidate all dentries associated with this inode */
if (inode == NULL)
break;

/* Invalidate all dentries associated with this inode */
LASSERT(lock->l_flags & LDLM_FL_CANCELING);

if (bits & MDS_INODELOCK_XATTR)
if (!fid_res_name_eq(ll_inode2fid(inode),
&lock->l_resource->lr_name)) {
LDLM_ERROR(lock, "data mismatch with object "DFID"(%p)",
PFID(ll_inode2fid(inode)), inode);
LBUG();
}

if (bits & MDS_INODELOCK_XATTR) {
ll_xattr_cache_destroy(inode);
bits &= ~MDS_INODELOCK_XATTR;
}

/* For OPEN locks we differentiate between lock modes
* LCK_CR, LCK_CW, LCK_PR - bug 22891 */
if (bits & (MDS_INODELOCK_LOOKUP | MDS_INODELOCK_UPDATE |
MDS_INODELOCK_LAYOUT | MDS_INODELOCK_PERM))
ll_have_md_lock(inode, &bits, LCK_MINMODE);

if (bits & MDS_INODELOCK_OPEN)
ll_have_md_lock(inode, &bits, mode);

fid = ll_inode2fid(inode);
if (!fid_res_name_eq(fid, &lock->l_resource->lr_name))
LDLM_ERROR(lock, "data mismatch with object "
DFID" (%p)", PFID(fid), inode);
ll_have_md_lock(inode, &bits, lock->l_req_mode);

if (bits & MDS_INODELOCK_OPEN) {
int flags = 0;
fmode_t fmode;

switch (lock->l_req_mode) {
case LCK_CW:
flags = FMODE_WRITE;
fmode = FMODE_WRITE;
break;
case LCK_PR:
flags = FMODE_EXEC;
fmode = FMODE_EXEC;
break;
case LCK_CR:
flags = FMODE_READ;
fmode = FMODE_READ;
break;
default:
CERROR("Unexpected lock mode for OPEN lock "
"%d, inode %ld\n", lock->l_req_mode,
inode->i_ino);
LDLM_ERROR(lock, "bad lock mode for OPEN lock");
LBUG();
}
ll_md_real_close(inode, flags);

ll_md_real_close(inode, fmode);
}

lli = ll_i2info(inode);
if (bits & (MDS_INODELOCK_LOOKUP | MDS_INODELOCK_UPDATE |
MDS_INODELOCK_LAYOUT | MDS_INODELOCK_PERM))
ll_have_md_lock(inode, &bits, LCK_MINMODE);

if (bits & MDS_INODELOCK_LAYOUT) {
struct cl_object_conf conf = { { 0 } };
struct cl_object_conf conf = {
.coc_opc = OBJECT_CONF_INVALIDATE,
.coc_inode = inode,
};

conf.coc_opc = OBJECT_CONF_INVALIDATE;
conf.coc_inode = inode;
rc = ll_layout_conf(inode, &conf);
if (rc)
CDEBUG(D_INODE, "invaliding layout %d.\n", rc);
if (rc < 0)
CDEBUG(D_INODE, "cannot invalidate layout of "
DFID": rc = %d\n",
PFID(ll_inode2fid(inode)), rc);
}

if (bits & MDS_INODELOCK_UPDATE) {
struct ll_inode_info *lli = ll_i2info(inode);

spin_lock(&lli->lli_lock);
lli->lli_flags &= ~LLIF_MDS_SIZE_LOCK;
spin_unlock(&lli->lli_lock);
}

if (S_ISDIR(inode->i_mode) &&
(bits & MDS_INODELOCK_UPDATE)) {
if ((bits & MDS_INODELOCK_UPDATE) && S_ISDIR(inode->i_mode)) {
CDEBUG(D_INODE, "invalidating inode %lu\n",
inode->i_ino);
truncate_inode_pages(inode->i_mapping, 0);
ll_invalidate_negative_children(inode);
}

if (inode->i_sb->s_root &&
inode != inode->i_sb->s_root->d_inode &&
(bits & (MDS_INODELOCK_LOOKUP | MDS_INODELOCK_PERM)))
if ((bits & (MDS_INODELOCK_LOOKUP | MDS_INODELOCK_PERM)) &&
inode->i_sb->s_root != NULL &&
inode != inode->i_sb->s_root->d_inode)
ll_invalidate_aliases(inode);

iput(inode);
break;
}
Expand Down

0 comments on commit 45b2a01

Please sign in to comment.