diff options
-rw-r--r-- | fs/btrfs/btrfs_inode.h | 15 | ||||
-rw-r--r-- | fs/btrfs/ctree.h | 8 | ||||
-rw-r--r-- | fs/btrfs/dir-item.c | 4 | ||||
-rw-r--r-- | fs/btrfs/inode-item.c | 10 | ||||
-rw-r--r-- | fs/btrfs/inode.c | 143 | ||||
-rw-r--r-- | fs/btrfs/ioctl.c | 4 | ||||
-rw-r--r-- | fs/btrfs/transaction.c | 4 |
7 files changed, 155 insertions, 33 deletions
diff --git a/fs/btrfs/btrfs_inode.h b/fs/btrfs/btrfs_inode.h index 3bf40591742a..b7bd60e4fdd7 100644 --- a/fs/btrfs/btrfs_inode.h +++ b/fs/btrfs/btrfs_inode.h | |||
@@ -43,6 +43,21 @@ struct btrfs_inode { | |||
43 | u64 delalloc_bytes; | 43 | u64 delalloc_bytes; |
44 | u64 disk_i_size; | 44 | u64 disk_i_size; |
45 | u32 flags; | 45 | u32 flags; |
46 | |||
47 | /* | ||
48 | * if this is a directory then index_cnt is the counter for the index | ||
49 | * number for new files that are created | ||
50 | */ | ||
51 | u64 index_cnt; | ||
52 | |||
53 | /* | ||
54 | * index holds the directory index for this inode on creation, so | ||
55 | * add_link can do what its supposed to. This isn't populated when the | ||
56 | * inode is read because there isn't really a reason to know this unless | ||
57 | * we are creating the directory index or deleting it, and deletion | ||
58 | * reads the index off of the inode reference at unlink time. | ||
59 | */ | ||
60 | u64 index; | ||
46 | }; | 61 | }; |
47 | 62 | ||
48 | static inline struct btrfs_inode *BTRFS_I(struct inode *inode) | 63 | static inline struct btrfs_inode *BTRFS_I(struct inode *inode) |
diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h index 6675e916ebcd..beb05b1de54c 100644 --- a/fs/btrfs/ctree.h +++ b/fs/btrfs/ctree.h | |||
@@ -372,6 +372,7 @@ struct btrfs_dev_extent { | |||
372 | } __attribute__ ((__packed__)); | 372 | } __attribute__ ((__packed__)); |
373 | 373 | ||
374 | struct btrfs_inode_ref { | 374 | struct btrfs_inode_ref { |
375 | __le64 index; | ||
375 | __le16 name_len; | 376 | __le16 name_len; |
376 | /* name goes here */ | 377 | /* name goes here */ |
377 | } __attribute__ ((__packed__)); | 378 | } __attribute__ ((__packed__)); |
@@ -902,6 +903,7 @@ BTRFS_SETGET_STACK_FUNCS(block_group_flags, | |||
902 | 903 | ||
903 | /* struct btrfs_inode_ref */ | 904 | /* struct btrfs_inode_ref */ |
904 | BTRFS_SETGET_FUNCS(inode_ref_name_len, struct btrfs_inode_ref, name_len, 16); | 905 | BTRFS_SETGET_FUNCS(inode_ref_name_len, struct btrfs_inode_ref, name_len, 16); |
906 | BTRFS_SETGET_FUNCS(inode_ref_index, struct btrfs_inode_ref, index, 64); | ||
905 | 907 | ||
906 | /* struct btrfs_inode_item */ | 908 | /* struct btrfs_inode_item */ |
907 | BTRFS_SETGET_FUNCS(inode_generation, struct btrfs_inode_item, generation, 64); | 909 | BTRFS_SETGET_FUNCS(inode_generation, struct btrfs_inode_item, generation, 64); |
@@ -1528,7 +1530,7 @@ int btrfs_find_dead_roots(struct btrfs_root *root, u64 objectid, | |||
1528 | /* dir-item.c */ | 1530 | /* dir-item.c */ |
1529 | int btrfs_insert_dir_item(struct btrfs_trans_handle *trans, struct btrfs_root | 1531 | int btrfs_insert_dir_item(struct btrfs_trans_handle *trans, struct btrfs_root |
1530 | *root, const char *name, int name_len, u64 dir, | 1532 | *root, const char *name, int name_len, u64 dir, |
1531 | struct btrfs_key *location, u8 type); | 1533 | struct btrfs_key *location, u8 type, u64 index); |
1532 | struct btrfs_dir_item *btrfs_lookup_dir_item(struct btrfs_trans_handle *trans, | 1534 | struct btrfs_dir_item *btrfs_lookup_dir_item(struct btrfs_trans_handle *trans, |
1533 | struct btrfs_root *root, | 1535 | struct btrfs_root *root, |
1534 | struct btrfs_path *path, u64 dir, | 1536 | struct btrfs_path *path, u64 dir, |
@@ -1566,11 +1568,11 @@ int btrfs_find_highest_inode(struct btrfs_root *fs_root, u64 *objectid); | |||
1566 | int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans, | 1568 | int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans, |
1567 | struct btrfs_root *root, | 1569 | struct btrfs_root *root, |
1568 | const char *name, int name_len, | 1570 | const char *name, int name_len, |
1569 | u64 inode_objectid, u64 ref_objectid); | 1571 | u64 inode_objectid, u64 ref_objectid, u64 index); |
1570 | int btrfs_del_inode_ref(struct btrfs_trans_handle *trans, | 1572 | int btrfs_del_inode_ref(struct btrfs_trans_handle *trans, |
1571 | struct btrfs_root *root, | 1573 | struct btrfs_root *root, |
1572 | const char *name, int name_len, | 1574 | const char *name, int name_len, |
1573 | u64 inode_objectid, u64 ref_objectid); | 1575 | u64 inode_objectid, u64 ref_objectid, u64 *index); |
1574 | int btrfs_insert_empty_inode(struct btrfs_trans_handle *trans, | 1576 | int btrfs_insert_empty_inode(struct btrfs_trans_handle *trans, |
1575 | struct btrfs_root *root, | 1577 | struct btrfs_root *root, |
1576 | struct btrfs_path *path, u64 objectid); | 1578 | struct btrfs_path *path, u64 objectid); |
diff --git a/fs/btrfs/dir-item.c b/fs/btrfs/dir-item.c index 7a73dc59dc4d..eb4dd3d75cf9 100644 --- a/fs/btrfs/dir-item.c +++ b/fs/btrfs/dir-item.c | |||
@@ -110,7 +110,7 @@ int btrfs_insert_xattr_item(struct btrfs_trans_handle *trans, | |||
110 | 110 | ||
111 | int btrfs_insert_dir_item(struct btrfs_trans_handle *trans, struct btrfs_root | 111 | int btrfs_insert_dir_item(struct btrfs_trans_handle *trans, struct btrfs_root |
112 | *root, const char *name, int name_len, u64 dir, | 112 | *root, const char *name, int name_len, u64 dir, |
113 | struct btrfs_key *location, u8 type) | 113 | struct btrfs_key *location, u8 type, u64 index) |
114 | { | 114 | { |
115 | int ret = 0; | 115 | int ret = 0; |
116 | int ret2 = 0; | 116 | int ret2 = 0; |
@@ -156,7 +156,7 @@ second_insert: | |||
156 | btrfs_release_path(root, path); | 156 | btrfs_release_path(root, path); |
157 | 157 | ||
158 | btrfs_set_key_type(&key, BTRFS_DIR_INDEX_KEY); | 158 | btrfs_set_key_type(&key, BTRFS_DIR_INDEX_KEY); |
159 | key.offset = location->objectid; | 159 | key.offset = index; |
160 | dir_item = insert_with_overflow(trans, root, path, &key, data_size, | 160 | dir_item = insert_with_overflow(trans, root, path, &key, data_size, |
161 | name, name_len); | 161 | name, name_len); |
162 | if (IS_ERR(dir_item)) { | 162 | if (IS_ERR(dir_item)) { |
diff --git a/fs/btrfs/inode-item.c b/fs/btrfs/inode-item.c index cba30b6cc6fe..d93451c66ba1 100644 --- a/fs/btrfs/inode-item.c +++ b/fs/btrfs/inode-item.c | |||
@@ -52,7 +52,7 @@ int find_name_in_backref(struct btrfs_path *path, const char * name, | |||
52 | int btrfs_del_inode_ref(struct btrfs_trans_handle *trans, | 52 | int btrfs_del_inode_ref(struct btrfs_trans_handle *trans, |
53 | struct btrfs_root *root, | 53 | struct btrfs_root *root, |
54 | const char *name, int name_len, | 54 | const char *name, int name_len, |
55 | u64 inode_objectid, u64 ref_objectid) | 55 | u64 inode_objectid, u64 ref_objectid, u64 *index) |
56 | { | 56 | { |
57 | struct btrfs_path *path; | 57 | struct btrfs_path *path; |
58 | struct btrfs_key key; | 58 | struct btrfs_key key; |
@@ -86,6 +86,10 @@ int btrfs_del_inode_ref(struct btrfs_trans_handle *trans, | |||
86 | } | 86 | } |
87 | leaf = path->nodes[0]; | 87 | leaf = path->nodes[0]; |
88 | item_size = btrfs_item_size_nr(leaf, path->slots[0]); | 88 | item_size = btrfs_item_size_nr(leaf, path->slots[0]); |
89 | |||
90 | if (index) | ||
91 | *index = btrfs_inode_ref_index(leaf, ref); | ||
92 | |||
89 | if (del_len == item_size) { | 93 | if (del_len == item_size) { |
90 | ret = btrfs_del_item(trans, root, path); | 94 | ret = btrfs_del_item(trans, root, path); |
91 | goto out; | 95 | goto out; |
@@ -106,7 +110,7 @@ out: | |||
106 | int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans, | 110 | int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans, |
107 | struct btrfs_root *root, | 111 | struct btrfs_root *root, |
108 | const char *name, int name_len, | 112 | const char *name, int name_len, |
109 | u64 inode_objectid, u64 ref_objectid) | 113 | u64 inode_objectid, u64 ref_objectid, u64 index) |
110 | { | 114 | { |
111 | struct btrfs_path *path; | 115 | struct btrfs_path *path; |
112 | struct btrfs_key key; | 116 | struct btrfs_key key; |
@@ -138,6 +142,7 @@ int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans, | |||
138 | struct btrfs_inode_ref); | 142 | struct btrfs_inode_ref); |
139 | ref = (struct btrfs_inode_ref *)((unsigned long)ref + old_size); | 143 | ref = (struct btrfs_inode_ref *)((unsigned long)ref + old_size); |
140 | btrfs_set_inode_ref_name_len(path->nodes[0], ref, name_len); | 144 | btrfs_set_inode_ref_name_len(path->nodes[0], ref, name_len); |
145 | btrfs_set_inode_ref_index(path->nodes[0], ref, index); | ||
141 | ptr = (unsigned long)(ref + 1); | 146 | ptr = (unsigned long)(ref + 1); |
142 | ret = 0; | 147 | ret = 0; |
143 | } else if (ret < 0) { | 148 | } else if (ret < 0) { |
@@ -146,6 +151,7 @@ int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans, | |||
146 | ref = btrfs_item_ptr(path->nodes[0], path->slots[0], | 151 | ref = btrfs_item_ptr(path->nodes[0], path->slots[0], |
147 | struct btrfs_inode_ref); | 152 | struct btrfs_inode_ref); |
148 | btrfs_set_inode_ref_name_len(path->nodes[0], ref, name_len); | 153 | btrfs_set_inode_ref_name_len(path->nodes[0], ref, name_len); |
154 | btrfs_set_inode_ref_index(path->nodes[0], ref, index); | ||
149 | ptr = (unsigned long)(ref + 1); | 155 | ptr = (unsigned long)(ref + 1); |
150 | } | 156 | } |
151 | write_extent_buffer(path->nodes[0], name, ptr, name_len); | 157 | write_extent_buffer(path->nodes[0], name, ptr, name_len); |
diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c index 0e90315ea803..8d371d6fe551 100644 --- a/fs/btrfs/inode.c +++ b/fs/btrfs/inode.c | |||
@@ -872,6 +872,8 @@ void btrfs_read_locked_inode(struct inode *inode) | |||
872 | inode->i_rdev = 0; | 872 | inode->i_rdev = 0; |
873 | rdev = btrfs_inode_rdev(leaf, inode_item); | 873 | rdev = btrfs_inode_rdev(leaf, inode_item); |
874 | 874 | ||
875 | BTRFS_I(inode)->index_cnt = (u64)-1; | ||
876 | |||
875 | alloc_group_block = btrfs_inode_block_group(leaf, inode_item); | 877 | alloc_group_block = btrfs_inode_block_group(leaf, inode_item); |
876 | BTRFS_I(inode)->block_group = btrfs_lookup_block_group(root->fs_info, | 878 | BTRFS_I(inode)->block_group = btrfs_lookup_block_group(root->fs_info, |
877 | alloc_group_block); | 879 | alloc_group_block); |
@@ -993,6 +995,7 @@ static int btrfs_unlink_trans(struct btrfs_trans_handle *trans, | |||
993 | struct extent_buffer *leaf; | 995 | struct extent_buffer *leaf; |
994 | struct btrfs_dir_item *di; | 996 | struct btrfs_dir_item *di; |
995 | struct btrfs_key key; | 997 | struct btrfs_key key; |
998 | u64 index; | ||
996 | 999 | ||
997 | path = btrfs_alloc_path(); | 1000 | path = btrfs_alloc_path(); |
998 | if (!path) { | 1001 | if (!path) { |
@@ -1017,8 +1020,19 @@ static int btrfs_unlink_trans(struct btrfs_trans_handle *trans, | |||
1017 | goto err; | 1020 | goto err; |
1018 | btrfs_release_path(root, path); | 1021 | btrfs_release_path(root, path); |
1019 | 1022 | ||
1023 | ret = btrfs_del_inode_ref(trans, root, name, name_len, | ||
1024 | dentry->d_inode->i_ino, | ||
1025 | dentry->d_parent->d_inode->i_ino, &index); | ||
1026 | if (ret) { | ||
1027 | printk("failed to delete reference to %.*s, " | ||
1028 | "inode %lu parent %lu\n", name_len, name, | ||
1029 | dentry->d_inode->i_ino, | ||
1030 | dentry->d_parent->d_inode->i_ino); | ||
1031 | goto err; | ||
1032 | } | ||
1033 | |||
1020 | di = btrfs_lookup_dir_index_item(trans, root, path, dir->i_ino, | 1034 | di = btrfs_lookup_dir_index_item(trans, root, path, dir->i_ino, |
1021 | key.objectid, name, name_len, -1); | 1035 | index, name, name_len, -1); |
1022 | if (IS_ERR(di)) { | 1036 | if (IS_ERR(di)) { |
1023 | ret = PTR_ERR(di); | 1037 | ret = PTR_ERR(di); |
1024 | goto err; | 1038 | goto err; |
@@ -1031,15 +1045,6 @@ static int btrfs_unlink_trans(struct btrfs_trans_handle *trans, | |||
1031 | btrfs_release_path(root, path); | 1045 | btrfs_release_path(root, path); |
1032 | 1046 | ||
1033 | dentry->d_inode->i_ctime = dir->i_ctime; | 1047 | dentry->d_inode->i_ctime = dir->i_ctime; |
1034 | ret = btrfs_del_inode_ref(trans, root, name, name_len, | ||
1035 | dentry->d_inode->i_ino, | ||
1036 | dentry->d_parent->d_inode->i_ino); | ||
1037 | if (ret) { | ||
1038 | printk("failed to delete reference to %.*s, " | ||
1039 | "inode %lu parent %lu\n", name_len, name, | ||
1040 | dentry->d_inode->i_ino, | ||
1041 | dentry->d_parent->d_inode->i_ino); | ||
1042 | } | ||
1043 | err: | 1048 | err: |
1044 | btrfs_free_path(path); | 1049 | btrfs_free_path(path); |
1045 | if (!ret) { | 1050 | if (!ret) { |
@@ -1625,6 +1630,7 @@ static int btrfs_init_locked_inode(struct inode *inode, void *p) | |||
1625 | BTRFS_I(inode)->root = args->root; | 1630 | BTRFS_I(inode)->root = args->root; |
1626 | BTRFS_I(inode)->delalloc_bytes = 0; | 1631 | BTRFS_I(inode)->delalloc_bytes = 0; |
1627 | BTRFS_I(inode)->disk_i_size = 0; | 1632 | BTRFS_I(inode)->disk_i_size = 0; |
1633 | BTRFS_I(inode)->index_cnt = (u64)-1; | ||
1628 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); | 1634 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); |
1629 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, | 1635 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, |
1630 | inode->i_mapping, GFP_NOFS); | 1636 | inode->i_mapping, GFP_NOFS); |
@@ -1901,8 +1907,77 @@ void btrfs_dirty_inode(struct inode *inode) | |||
1901 | btrfs_end_transaction(trans, root); | 1907 | btrfs_end_transaction(trans, root); |
1902 | } | 1908 | } |
1903 | 1909 | ||
1910 | static int btrfs_set_inode_index_count(struct inode *inode) | ||
1911 | { | ||
1912 | struct btrfs_root *root = BTRFS_I(inode)->root; | ||
1913 | struct btrfs_key key, found_key; | ||
1914 | struct btrfs_path *path; | ||
1915 | struct extent_buffer *leaf; | ||
1916 | int ret; | ||
1917 | |||
1918 | key.objectid = inode->i_ino; | ||
1919 | btrfs_set_key_type(&key, BTRFS_DIR_INDEX_KEY); | ||
1920 | key.offset = (u64)-1; | ||
1921 | |||
1922 | path = btrfs_alloc_path(); | ||
1923 | if (!path) | ||
1924 | return -ENOMEM; | ||
1925 | |||
1926 | ret = btrfs_search_slot(NULL, root, &key, path, 0, 0); | ||
1927 | if (ret < 0) | ||
1928 | goto out; | ||
1929 | /* FIXME: we should be able to handle this */ | ||
1930 | if (ret == 0) | ||
1931 | goto out; | ||
1932 | ret = 0; | ||
1933 | |||
1934 | /* | ||
1935 | * MAGIC NUMBER EXPLANATION: | ||
1936 | * since we search a directory based on f_pos we have to start at 2 | ||
1937 | * since '.' and '..' have f_pos of 0 and 1 respectively, so everybody | ||
1938 | * else has to start at 2 | ||
1939 | */ | ||
1940 | if (path->slots[0] == 0) { | ||
1941 | BTRFS_I(inode)->index_cnt = 2; | ||
1942 | goto out; | ||
1943 | } | ||
1944 | |||
1945 | path->slots[0]--; | ||
1946 | |||
1947 | leaf = path->nodes[0]; | ||
1948 | btrfs_item_key_to_cpu(leaf, &found_key, path->slots[0]); | ||
1949 | |||
1950 | if (found_key.objectid != inode->i_ino || | ||
1951 | btrfs_key_type(&found_key) != BTRFS_DIR_INDEX_KEY) { | ||
1952 | BTRFS_I(inode)->index_cnt = 2; | ||
1953 | goto out; | ||
1954 | } | ||
1955 | |||
1956 | BTRFS_I(inode)->index_cnt = found_key.offset + 1; | ||
1957 | out: | ||
1958 | btrfs_free_path(path); | ||
1959 | return ret; | ||
1960 | } | ||
1961 | |||
1962 | static int btrfs_set_inode_index(struct inode *dir, struct inode *inode) | ||
1963 | { | ||
1964 | int ret = 0; | ||
1965 | |||
1966 | if (BTRFS_I(dir)->index_cnt == (u64)-1) { | ||
1967 | ret = btrfs_set_inode_index_count(dir); | ||
1968 | if (ret) | ||
1969 | return ret; | ||
1970 | } | ||
1971 | |||
1972 | BTRFS_I(inode)->index = BTRFS_I(dir)->index_cnt; | ||
1973 | BTRFS_I(dir)->index_cnt++; | ||
1974 | |||
1975 | return ret; | ||
1976 | } | ||
1977 | |||
1904 | static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans, | 1978 | static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans, |
1905 | struct btrfs_root *root, | 1979 | struct btrfs_root *root, |
1980 | struct inode *dir, | ||
1906 | const char *name, int name_len, | 1981 | const char *name, int name_len, |
1907 | u64 ref_objectid, | 1982 | u64 ref_objectid, |
1908 | u64 objectid, | 1983 | u64 objectid, |
@@ -1928,6 +2003,20 @@ static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans, | |||
1928 | if (!inode) | 2003 | if (!inode) |
1929 | return ERR_PTR(-ENOMEM); | 2004 | return ERR_PTR(-ENOMEM); |
1930 | 2005 | ||
2006 | if (dir) { | ||
2007 | ret = btrfs_set_inode_index(dir, inode); | ||
2008 | if (ret) | ||
2009 | return ERR_PTR(ret); | ||
2010 | } else { | ||
2011 | BTRFS_I(inode)->index = 0; | ||
2012 | } | ||
2013 | /* | ||
2014 | * index_cnt is ignored for everything but a dir, | ||
2015 | * btrfs_get_inode_index_count has an explanation for the magic | ||
2016 | * number | ||
2017 | */ | ||
2018 | BTRFS_I(inode)->index_cnt = 2; | ||
2019 | |||
1931 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); | 2020 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); |
1932 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, | 2021 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, |
1933 | inode->i_mapping, GFP_NOFS); | 2022 | inode->i_mapping, GFP_NOFS); |
@@ -1984,6 +2073,7 @@ static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans, | |||
1984 | ref = btrfs_item_ptr(path->nodes[0], path->slots[0] + 1, | 2073 | ref = btrfs_item_ptr(path->nodes[0], path->slots[0] + 1, |
1985 | struct btrfs_inode_ref); | 2074 | struct btrfs_inode_ref); |
1986 | btrfs_set_inode_ref_name_len(path->nodes[0], ref, name_len); | 2075 | btrfs_set_inode_ref_name_len(path->nodes[0], ref, name_len); |
2076 | btrfs_set_inode_ref_index(path->nodes[0], ref, BTRFS_I(inode)->index); | ||
1987 | ptr = (unsigned long)(ref + 1); | 2077 | ptr = (unsigned long)(ref + 1); |
1988 | write_extent_buffer(path->nodes[0], name, ptr, name_len); | 2078 | write_extent_buffer(path->nodes[0], name, ptr, name_len); |
1989 | 2079 | ||
@@ -1998,6 +2088,8 @@ static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans, | |||
1998 | insert_inode_hash(inode); | 2088 | insert_inode_hash(inode); |
1999 | return inode; | 2089 | return inode; |
2000 | fail: | 2090 | fail: |
2091 | if (dir) | ||
2092 | BTRFS_I(dir)->index_cnt--; | ||
2001 | btrfs_free_path(path); | 2093 | btrfs_free_path(path); |
2002 | return ERR_PTR(ret); | 2094 | return ERR_PTR(ret); |
2003 | } | 2095 | } |
@@ -2014,7 +2106,7 @@ static int btrfs_add_link(struct btrfs_trans_handle *trans, | |||
2014 | int ret; | 2106 | int ret; |
2015 | struct btrfs_key key; | 2107 | struct btrfs_key key; |
2016 | struct btrfs_root *root = BTRFS_I(dentry->d_parent->d_inode)->root; | 2108 | struct btrfs_root *root = BTRFS_I(dentry->d_parent->d_inode)->root; |
2017 | struct inode *parent_inode; | 2109 | struct inode *parent_inode = dentry->d_parent->d_inode; |
2018 | 2110 | ||
2019 | key.objectid = inode->i_ino; | 2111 | key.objectid = inode->i_ino; |
2020 | btrfs_set_key_type(&key, BTRFS_INODE_ITEM_KEY); | 2112 | btrfs_set_key_type(&key, BTRFS_INODE_ITEM_KEY); |
@@ -2023,16 +2115,17 @@ static int btrfs_add_link(struct btrfs_trans_handle *trans, | |||
2023 | ret = btrfs_insert_dir_item(trans, root, | 2115 | ret = btrfs_insert_dir_item(trans, root, |
2024 | dentry->d_name.name, dentry->d_name.len, | 2116 | dentry->d_name.name, dentry->d_name.len, |
2025 | dentry->d_parent->d_inode->i_ino, | 2117 | dentry->d_parent->d_inode->i_ino, |
2026 | &key, btrfs_inode_type(inode)); | 2118 | &key, btrfs_inode_type(inode), |
2119 | BTRFS_I(inode)->index); | ||
2027 | if (ret == 0) { | 2120 | if (ret == 0) { |
2028 | if (add_backref) { | 2121 | if (add_backref) { |
2029 | ret = btrfs_insert_inode_ref(trans, root, | 2122 | ret = btrfs_insert_inode_ref(trans, root, |
2030 | dentry->d_name.name, | 2123 | dentry->d_name.name, |
2031 | dentry->d_name.len, | 2124 | dentry->d_name.len, |
2032 | inode->i_ino, | 2125 | inode->i_ino, |
2033 | dentry->d_parent->d_inode->i_ino); | 2126 | parent_inode->i_ino, |
2127 | BTRFS_I(inode)->index); | ||
2034 | } | 2128 | } |
2035 | parent_inode = dentry->d_parent->d_inode; | ||
2036 | btrfs_i_size_write(parent_inode, parent_inode->i_size + | 2129 | btrfs_i_size_write(parent_inode, parent_inode->i_size + |
2037 | dentry->d_name.len * 2); | 2130 | dentry->d_name.len * 2); |
2038 | parent_inode->i_mtime = parent_inode->i_ctime = CURRENT_TIME; | 2131 | parent_inode->i_mtime = parent_inode->i_ctime = CURRENT_TIME; |
@@ -2083,7 +2176,7 @@ static int btrfs_mknod(struct inode *dir, struct dentry *dentry, | |||
2083 | goto out_unlock; | 2176 | goto out_unlock; |
2084 | } | 2177 | } |
2085 | 2178 | ||
2086 | inode = btrfs_new_inode(trans, root, dentry->d_name.name, | 2179 | inode = btrfs_new_inode(trans, root, dir, dentry->d_name.name, |
2087 | dentry->d_name.len, | 2180 | dentry->d_name.len, |
2088 | dentry->d_parent->d_inode->i_ino, objectid, | 2181 | dentry->d_parent->d_inode->i_ino, objectid, |
2089 | BTRFS_I(dir)->block_group, mode); | 2182 | BTRFS_I(dir)->block_group, mode); |
@@ -2138,7 +2231,7 @@ static int btrfs_create(struct inode *dir, struct dentry *dentry, | |||
2138 | goto out_unlock; | 2231 | goto out_unlock; |
2139 | } | 2232 | } |
2140 | 2233 | ||
2141 | inode = btrfs_new_inode(trans, root, dentry->d_name.name, | 2234 | inode = btrfs_new_inode(trans, root, dir, dentry->d_name.name, |
2142 | dentry->d_name.len, | 2235 | dentry->d_name.len, |
2143 | dentry->d_parent->d_inode->i_ino, | 2236 | dentry->d_parent->d_inode->i_ino, |
2144 | objectid, BTRFS_I(dir)->block_group, mode); | 2237 | objectid, BTRFS_I(dir)->block_group, mode); |
@@ -2203,10 +2296,15 @@ static int btrfs_link(struct dentry *old_dentry, struct inode *dir, | |||
2203 | err = btrfs_check_free_space(root, 1, 0); | 2296 | err = btrfs_check_free_space(root, 1, 0); |
2204 | if (err) | 2297 | if (err) |
2205 | goto fail; | 2298 | goto fail; |
2299 | err = btrfs_set_inode_index(dir, inode); | ||
2300 | if (err) | ||
2301 | goto fail; | ||
2302 | |||
2206 | trans = btrfs_start_transaction(root, 1); | 2303 | trans = btrfs_start_transaction(root, 1); |
2207 | 2304 | ||
2208 | btrfs_set_trans_block_group(trans, dir); | 2305 | btrfs_set_trans_block_group(trans, dir); |
2209 | atomic_inc(&inode->i_count); | 2306 | atomic_inc(&inode->i_count); |
2307 | |||
2210 | err = btrfs_add_nondir(trans, dentry, inode, 1); | 2308 | err = btrfs_add_nondir(trans, dentry, inode, 1); |
2211 | 2309 | ||
2212 | if (err) | 2310 | if (err) |
@@ -2258,7 +2356,7 @@ static int btrfs_mkdir(struct inode *dir, struct dentry *dentry, int mode) | |||
2258 | goto out_unlock; | 2356 | goto out_unlock; |
2259 | } | 2357 | } |
2260 | 2358 | ||
2261 | inode = btrfs_new_inode(trans, root, dentry->d_name.name, | 2359 | inode = btrfs_new_inode(trans, root, dir, dentry->d_name.name, |
2262 | dentry->d_name.len, | 2360 | dentry->d_name.len, |
2263 | dentry->d_parent->d_inode->i_ino, objectid, | 2361 | dentry->d_parent->d_inode->i_ino, objectid, |
2264 | BTRFS_I(dir)->block_group, S_IFDIR | mode); | 2362 | BTRFS_I(dir)->block_group, S_IFDIR | mode); |
@@ -2886,9 +2984,8 @@ int btrfs_create_subvol_root(struct btrfs_root *new_root, | |||
2886 | struct btrfs_block_group_cache *block_group) | 2984 | struct btrfs_block_group_cache *block_group) |
2887 | { | 2985 | { |
2888 | struct inode *inode; | 2986 | struct inode *inode; |
2889 | int ret; | ||
2890 | 2987 | ||
2891 | inode = btrfs_new_inode(trans, new_root, "..", 2, new_dirid, | 2988 | inode = btrfs_new_inode(trans, new_root, NULL, "..", 2, new_dirid, |
2892 | new_dirid, block_group, S_IFDIR | 0700); | 2989 | new_dirid, block_group, S_IFDIR | 0700); |
2893 | if (IS_ERR(inode)) | 2990 | if (IS_ERR(inode)) |
2894 | return PTR_ERR(inode); | 2991 | return PTR_ERR(inode); |
@@ -2896,8 +2993,6 @@ int btrfs_create_subvol_root(struct btrfs_root *new_root, | |||
2896 | inode->i_fop = &btrfs_dir_file_operations; | 2993 | inode->i_fop = &btrfs_dir_file_operations; |
2897 | new_root->inode = inode; | 2994 | new_root->inode = inode; |
2898 | 2995 | ||
2899 | ret = btrfs_insert_inode_ref(trans, new_root, "..", 2, new_dirid, | ||
2900 | new_dirid); | ||
2901 | inode->i_nlink = 1; | 2996 | inode->i_nlink = 1; |
2902 | btrfs_i_size_write(inode, 0); | 2997 | btrfs_i_size_write(inode, 0); |
2903 | 2998 | ||
@@ -3078,6 +3173,10 @@ static int btrfs_rename(struct inode * old_dir, struct dentry *old_dentry, | |||
3078 | if (ret) | 3173 | if (ret) |
3079 | goto out_fail; | 3174 | goto out_fail; |
3080 | } | 3175 | } |
3176 | ret = btrfs_set_inode_index(new_dir, old_inode); | ||
3177 | if (ret) | ||
3178 | goto out_fail; | ||
3179 | |||
3081 | ret = btrfs_add_link(trans, new_dentry, old_inode, 1); | 3180 | ret = btrfs_add_link(trans, new_dentry, old_inode, 1); |
3082 | if (ret) | 3181 | if (ret) |
3083 | goto out_fail; | 3182 | goto out_fail; |
@@ -3123,7 +3222,7 @@ static int btrfs_symlink(struct inode *dir, struct dentry *dentry, | |||
3123 | goto out_unlock; | 3222 | goto out_unlock; |
3124 | } | 3223 | } |
3125 | 3224 | ||
3126 | inode = btrfs_new_inode(trans, root, dentry->d_name.name, | 3225 | inode = btrfs_new_inode(trans, root, dir, dentry->d_name.name, |
3127 | dentry->d_name.len, | 3226 | dentry->d_name.len, |
3128 | dentry->d_parent->d_inode->i_ino, objectid, | 3227 | dentry->d_parent->d_inode->i_ino, objectid, |
3129 | BTRFS_I(dir)->block_group, S_IFLNK|S_IRWXUGO); | 3228 | BTRFS_I(dir)->block_group, S_IFLNK|S_IRWXUGO); |
diff --git a/fs/btrfs/ioctl.c b/fs/btrfs/ioctl.c index a61f2e7e2db5..faf081302d02 100644 --- a/fs/btrfs/ioctl.c +++ b/fs/btrfs/ioctl.c | |||
@@ -129,13 +129,13 @@ static noinline int create_subvol(struct btrfs_root *root, char *name, | |||
129 | dir = root->fs_info->sb->s_root->d_inode; | 129 | dir = root->fs_info->sb->s_root->d_inode; |
130 | ret = btrfs_insert_dir_item(trans, root->fs_info->tree_root, | 130 | ret = btrfs_insert_dir_item(trans, root->fs_info->tree_root, |
131 | name, namelen, dir->i_ino, &key, | 131 | name, namelen, dir->i_ino, &key, |
132 | BTRFS_FT_DIR); | 132 | BTRFS_FT_DIR, 0); |
133 | if (ret) | 133 | if (ret) |
134 | goto fail; | 134 | goto fail; |
135 | 135 | ||
136 | ret = btrfs_insert_inode_ref(trans, root->fs_info->tree_root, | 136 | ret = btrfs_insert_inode_ref(trans, root->fs_info->tree_root, |
137 | name, namelen, objectid, | 137 | name, namelen, objectid, |
138 | root->fs_info->sb->s_root->d_inode->i_ino); | 138 | root->fs_info->sb->s_root->d_inode->i_ino, 0); |
139 | if (ret) | 139 | if (ret) |
140 | goto fail; | 140 | goto fail; |
141 | 141 | ||
diff --git a/fs/btrfs/transaction.c b/fs/btrfs/transaction.c index 0f756e0175c0..768b0d223e68 100644 --- a/fs/btrfs/transaction.c +++ b/fs/btrfs/transaction.c | |||
@@ -577,14 +577,14 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans, | |||
577 | ret = btrfs_insert_dir_item(trans, root->fs_info->tree_root, | 577 | ret = btrfs_insert_dir_item(trans, root->fs_info->tree_root, |
578 | pending->name, namelen, | 578 | pending->name, namelen, |
579 | root->fs_info->sb->s_root->d_inode->i_ino, | 579 | root->fs_info->sb->s_root->d_inode->i_ino, |
580 | &key, BTRFS_FT_DIR); | 580 | &key, BTRFS_FT_DIR, 0); |
581 | 581 | ||
582 | if (ret) | 582 | if (ret) |
583 | goto fail; | 583 | goto fail; |
584 | 584 | ||
585 | ret = btrfs_insert_inode_ref(trans, root->fs_info->tree_root, | 585 | ret = btrfs_insert_inode_ref(trans, root->fs_info->tree_root, |
586 | pending->name, strlen(pending->name), objectid, | 586 | pending->name, strlen(pending->name), objectid, |
587 | root->fs_info->sb->s_root->d_inode->i_ino); | 587 | root->fs_info->sb->s_root->d_inode->i_ino, 0); |
588 | 588 | ||
589 | /* Invalidate existing dcache entry for new snapshot. */ | 589 | /* Invalidate existing dcache entry for new snapshot. */ |
590 | btrfs_invalidate_dcache_root(root, pending->name, namelen); | 590 | btrfs_invalidate_dcache_root(root, pending->name, namelen); |