ceph: don't clear I_NEW until inode metadata is fully populated
Currently, we could have an open-by-handle (or NFS server) call into the filesystem and start working with an inode before it's properly filled out. Don't clear I_NEW until we have filled out the inode, and discard it properly if that fails. Note that we occasionally take an extra reference to the inode to ensure that we don't put the last reference in discard_new_inode, but rather leave it for ceph_async_iput. Signed-off-by: Jeff Layton <jlayton@kernel.org> Signed-off-by: Ilya Dryomov <idryomov@gmail.com>
This commit is contained in:
parent
c4853e9776
commit
893e456b2c
@ -55,11 +55,9 @@ struct inode *ceph_get_inode(struct super_block *sb, struct ceph_vino vino)
|
|||||||
inode = iget5_locked(sb, t, ceph_ino_compare, ceph_set_ino_cb, &vino);
|
inode = iget5_locked(sb, t, ceph_ino_compare, ceph_set_ino_cb, &vino);
|
||||||
if (!inode)
|
if (!inode)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
if (inode->i_state & I_NEW) {
|
if (inode->i_state & I_NEW)
|
||||||
dout("get_inode created new inode %p %llx.%llx ino %llx\n",
|
dout("get_inode created new inode %p %llx.%llx ino %llx\n",
|
||||||
inode, ceph_vinop(inode), (u64)inode->i_ino);
|
inode, ceph_vinop(inode), (u64)inode->i_ino);
|
||||||
unlock_new_inode(inode);
|
|
||||||
}
|
|
||||||
|
|
||||||
dout("get_inode on %lu=%llx.%llx got %p\n", inode->i_ino, vino.ino,
|
dout("get_inode on %lu=%llx.%llx got %p\n", inode->i_ino, vino.ino,
|
||||||
vino.snap, inode);
|
vino.snap, inode);
|
||||||
@ -88,6 +86,10 @@ struct inode *ceph_get_snapdir(struct inode *parent)
|
|||||||
inode->i_fop = &ceph_snapdir_fops;
|
inode->i_fop = &ceph_snapdir_fops;
|
||||||
ci->i_snap_caps = CEPH_CAP_PIN; /* so we can open */
|
ci->i_snap_caps = CEPH_CAP_PIN; /* so we can open */
|
||||||
ci->i_rbytes = 0;
|
ci->i_rbytes = 0;
|
||||||
|
|
||||||
|
if (inode->i_state & I_NEW)
|
||||||
|
unlock_new_inode(inode);
|
||||||
|
|
||||||
return inode;
|
return inode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1304,7 +1306,6 @@ retry_lookup:
|
|||||||
err = PTR_ERR(in);
|
err = PTR_ERR(in);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
req->r_target_inode = in;
|
|
||||||
|
|
||||||
err = fill_inode(in, req->r_locked_page, &rinfo->targeti, NULL,
|
err = fill_inode(in, req->r_locked_page, &rinfo->targeti, NULL,
|
||||||
session,
|
session,
|
||||||
@ -1314,8 +1315,13 @@ retry_lookup:
|
|||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
pr_err("fill_inode badness %p %llx.%llx\n",
|
pr_err("fill_inode badness %p %llx.%llx\n",
|
||||||
in, ceph_vinop(in));
|
in, ceph_vinop(in));
|
||||||
|
if (in->i_state & I_NEW)
|
||||||
|
discard_new_inode(in);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
req->r_target_inode = in;
|
||||||
|
if (in->i_state & I_NEW)
|
||||||
|
unlock_new_inode(in);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1499,7 +1505,14 @@ static int readdir_prepopulate_inodes_only(struct ceph_mds_request *req,
|
|||||||
if (rc < 0) {
|
if (rc < 0) {
|
||||||
pr_err("fill_inode badness on %p got %d\n", in, rc);
|
pr_err("fill_inode badness on %p got %d\n", in, rc);
|
||||||
err = rc;
|
err = rc;
|
||||||
|
if (in->i_state & I_NEW) {
|
||||||
|
ihold(in);
|
||||||
|
discard_new_inode(in);
|
||||||
|
}
|
||||||
|
} else if (in->i_state & I_NEW) {
|
||||||
|
unlock_new_inode(in);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* avoid calling iput_final() in mds dispatch threads */
|
/* avoid calling iput_final() in mds dispatch threads */
|
||||||
ceph_async_iput(in);
|
ceph_async_iput(in);
|
||||||
}
|
}
|
||||||
@ -1701,12 +1714,18 @@ retry_lookup:
|
|||||||
if (d_really_is_negative(dn)) {
|
if (d_really_is_negative(dn)) {
|
||||||
/* avoid calling iput_final() in mds
|
/* avoid calling iput_final() in mds
|
||||||
* dispatch threads */
|
* dispatch threads */
|
||||||
|
if (in->i_state & I_NEW) {
|
||||||
|
ihold(in);
|
||||||
|
discard_new_inode(in);
|
||||||
|
}
|
||||||
ceph_async_iput(in);
|
ceph_async_iput(in);
|
||||||
}
|
}
|
||||||
d_drop(dn);
|
d_drop(dn);
|
||||||
err = ret;
|
err = ret;
|
||||||
goto next_item;
|
goto next_item;
|
||||||
}
|
}
|
||||||
|
if (in->i_state & I_NEW)
|
||||||
|
unlock_new_inode(in);
|
||||||
|
|
||||||
if (d_really_is_negative(dn)) {
|
if (d_really_is_negative(dn)) {
|
||||||
if (ceph_security_xattr_deadlock(in)) {
|
if (ceph_security_xattr_deadlock(in)) {
|
||||||
|
Loading…
Reference in New Issue
Block a user