Commit 5da1d479 authored by Chris Mason's avatar Chris Mason Committed by Ben Hutchings

Btrfs: setup inode location during btrfs_init_inode_locked

commit 90d3e592 upstream.

We have a race during inode init because the BTRFS_I(inode)->location is setup
after the inode hash table lock is dropped.  btrfs_find_actor uses the location
field, so our search might not find an existing inode in the hash table if we
race with the inode init code.

This commit changes things to setup the location field sooner.  Also the find actor now
uses only the location objectid to match inodes.  For inode hashing, we just
need a unique and stable test, it doesn't have to reflect the inode numbers we
show to userland.
Signed-off-by: default avatarChris Mason <clm@fb.com>
[bwh: Backported to 3.2:
 - No hashval in btrfs_iget_locked()]
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent 1d4b777f
...@@ -56,7 +56,7 @@ ...@@ -56,7 +56,7 @@
#include "inode-map.h" #include "inode-map.h"
struct btrfs_iget_args { struct btrfs_iget_args {
u64 ino; struct btrfs_key *location;
struct btrfs_root *root; struct btrfs_root *root;
}; };
...@@ -3847,7 +3847,9 @@ int btrfs_invalidate_inodes(struct btrfs_root *root) ...@@ -3847,7 +3847,9 @@ int btrfs_invalidate_inodes(struct btrfs_root *root)
static int btrfs_init_locked_inode(struct inode *inode, void *p) static int btrfs_init_locked_inode(struct inode *inode, void *p)
{ {
struct btrfs_iget_args *args = p; struct btrfs_iget_args *args = p;
inode->i_ino = args->ino; inode->i_ino = args->location->objectid;
memcpy(&BTRFS_I(inode)->location, args->location,
sizeof(*args->location));
BTRFS_I(inode)->root = args->root; BTRFS_I(inode)->root = args->root;
btrfs_set_inode_space_info(args->root, inode); btrfs_set_inode_space_info(args->root, inode);
return 0; return 0;
...@@ -3856,20 +3858,20 @@ static int btrfs_init_locked_inode(struct inode *inode, void *p) ...@@ -3856,20 +3858,20 @@ static int btrfs_init_locked_inode(struct inode *inode, void *p)
static int btrfs_find_actor(struct inode *inode, void *opaque) static int btrfs_find_actor(struct inode *inode, void *opaque)
{ {
struct btrfs_iget_args *args = opaque; struct btrfs_iget_args *args = opaque;
return args->ino == btrfs_ino(inode) && return args->location->objectid == BTRFS_I(inode)->location.objectid &&
args->root == BTRFS_I(inode)->root; args->root == BTRFS_I(inode)->root;
} }
static struct inode *btrfs_iget_locked(struct super_block *s, static struct inode *btrfs_iget_locked(struct super_block *s,
u64 objectid, struct btrfs_key *location,
struct btrfs_root *root) struct btrfs_root *root)
{ {
struct inode *inode; struct inode *inode;
struct btrfs_iget_args args; struct btrfs_iget_args args;
args.ino = objectid; args.location = location;
args.root = root; args.root = root;
inode = iget5_locked(s, objectid, btrfs_find_actor, inode = iget5_locked(s, location->objectid, btrfs_find_actor,
btrfs_init_locked_inode, btrfs_init_locked_inode,
(void *)&args); (void *)&args);
return inode; return inode;
...@@ -3883,13 +3885,11 @@ struct inode *btrfs_iget(struct super_block *s, struct btrfs_key *location, ...@@ -3883,13 +3885,11 @@ struct inode *btrfs_iget(struct super_block *s, struct btrfs_key *location,
{ {
struct inode *inode; struct inode *inode;
inode = btrfs_iget_locked(s, location->objectid, root); inode = btrfs_iget_locked(s, location, root);
if (!inode) if (!inode)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
if (inode->i_state & I_NEW) { if (inode->i_state & I_NEW) {
BTRFS_I(inode)->root = root;
memcpy(&BTRFS_I(inode)->location, location, sizeof(*location));
btrfs_read_locked_inode(inode); btrfs_read_locked_inode(inode);
if (!is_bad_inode(inode)) { if (!is_bad_inode(inode)) {
inode_tree_add(inode); inode_tree_add(inode);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment