Skip to content

Commit

Permalink
pnfs/blocklayout: serialize GETDEVICEINFO calls
Browse files Browse the repository at this point in the history
The rpc_pipefs code isn't thread safe, leading to occasional use after
frees when running xfstests generic/241 (dbench).

Signed-off-by: Christoph Hellwig <[email protected]>
Link: http://lkml.kernel.org/r/[email protected]
Cc: [email protected] # 3.17.x
Signed-off-by: Trond Myklebust <[email protected]>
  • Loading branch information
Christoph Hellwig authored and trondmypd committed Nov 12, 2014
1 parent 8c393f9 commit e0d4ed7
Show file tree
Hide file tree
Showing 2 changed files with 10 additions and 5 deletions.
14 changes: 9 additions & 5 deletions fs/nfs/blocklayout/rpc_pipefs.c
Original file line number Diff line number Diff line change
Expand Up @@ -65,17 +65,18 @@ bl_resolve_deviceid(struct nfs_server *server, struct pnfs_block_volume *b,

dprintk("%s CREATING PIPEFS MESSAGE\n", __func__);

mutex_lock(&nn->bl_mutex);
bl_pipe_msg.bl_wq = &nn->bl_wq;

b->simple.len += 4; /* single volume */
if (b->simple.len > PAGE_SIZE)
return -EIO;
goto out_unlock;

memset(msg, 0, sizeof(*msg));
msg->len = sizeof(*bl_msg) + b->simple.len;
msg->data = kzalloc(msg->len, gfp_mask);
if (!msg->data)
goto out;
goto out_free_data;

bl_msg = msg->data;
bl_msg->type = BL_DEVICE_MOUNT,
Expand All @@ -87,7 +88,7 @@ bl_resolve_deviceid(struct nfs_server *server, struct pnfs_block_volume *b,
rc = rpc_queue_upcall(nn->bl_device_pipe, msg);
if (rc < 0) {
remove_wait_queue(&nn->bl_wq, &wq);
goto out;
goto out_free_data;
}

set_current_state(TASK_UNINTERRUPTIBLE);
Expand All @@ -97,12 +98,14 @@ bl_resolve_deviceid(struct nfs_server *server, struct pnfs_block_volume *b,
if (reply->status != BL_DEVICE_REQUEST_PROC) {
printk(KERN_WARNING "%s failed to decode device: %d\n",
__func__, reply->status);
goto out;
goto out_free_data;
}

dev = MKDEV(reply->major, reply->minor);
out:
out_free_data:
kfree(msg->data);
out_unlock:
mutex_unlock(&nn->bl_mutex);
return dev;
}

Expand Down Expand Up @@ -232,6 +235,7 @@ static int nfs4blocklayout_net_init(struct net *net)
struct nfs_net *nn = net_generic(net, nfs_net_id);
struct dentry *dentry;

mutex_init(&nn->bl_mutex);
init_waitqueue_head(&nn->bl_wq);
nn->bl_device_pipe = rpc_mkpipe_data(&bl_upcall_ops, 0);
if (IS_ERR(nn->bl_device_pipe))
Expand Down
1 change: 1 addition & 0 deletions fs/nfs/netns.h
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ struct nfs_net {
struct rpc_pipe *bl_device_pipe;
struct bl_dev_msg bl_mount_reply;
wait_queue_head_t bl_wq;
struct mutex bl_mutex;
struct list_head nfs_client_list;
struct list_head nfs_volume_list;
#if IS_ENABLED(CONFIG_NFS_V4)
Expand Down

0 comments on commit e0d4ed7

Please sign in to comment.