aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/btrfs/ctree.c57
-rw-r--r--fs/btrfs/ctree.h21
-rw-r--r--fs/btrfs/extent-tree.c20
-rw-r--r--fs/btrfs/file-item.c61
-rw-r--r--fs/btrfs/super.c284
5 files changed, 394 insertions, 49 deletions
diff --git a/fs/btrfs/ctree.c b/fs/btrfs/ctree.c
index 6b76fd967153..43d4779abdd1 100644
--- a/fs/btrfs/ctree.c
+++ b/fs/btrfs/ctree.c
@@ -1249,6 +1249,63 @@ static int split_leaf(struct btrfs_trans_handle *trans, struct btrfs_root
1249 return ret; 1249 return ret;
1250} 1250}
1251 1251
1252int btrfs_truncate_item(struct btrfs_trans_handle *trans,
1253 struct btrfs_root *root,
1254 struct btrfs_path *path,
1255 u32 new_size)
1256{
1257 int ret = 0;
1258 int slot;
1259 int slot_orig;
1260 struct btrfs_leaf *leaf;
1261 struct buffer_head *leaf_buf;
1262 u32 nritems;
1263 unsigned int data_end;
1264 unsigned int old_data_start;
1265 unsigned int old_size;
1266 unsigned int size_diff;
1267 int i;
1268
1269 slot_orig = path->slots[0];
1270 leaf_buf = path->nodes[0];
1271 leaf = btrfs_buffer_leaf(leaf_buf);
1272
1273 nritems = btrfs_header_nritems(&leaf->header);
1274 data_end = leaf_data_end(root, leaf);
1275
1276 slot = path->slots[0];
1277 old_data_start = btrfs_item_offset(leaf->items + slot);
1278 old_size = btrfs_item_size(leaf->items + slot);
1279 BUG_ON(old_size <= new_size);
1280 size_diff = old_size - new_size;
1281
1282 BUG_ON(slot < 0);
1283 BUG_ON(slot >= nritems);
1284
1285 /*
1286 * item0..itemN ... dataN.offset..dataN.size .. data0.size
1287 */
1288 /* first correct the data pointers */
1289 for (i = slot; i < nritems; i++) {
1290 u32 ioff = btrfs_item_offset(leaf->items + i);
1291 btrfs_set_item_offset(leaf->items + i,
1292 ioff + size_diff);
1293 }
1294 /* shift the data */
1295printk("truncate item, new_size %u old_size %u, diff %u, bufp %p, dst, %p, num %u, old_data_start %u, data_end %u\n", new_size, old_size, size_diff, leaf, btrfs_leaf_data(leaf) + data_end + size_diff, old_data_start-data_end, old_data_start, data_end);
1296 btrfs_memmove(root, leaf, btrfs_leaf_data(leaf) +
1297 data_end + size_diff, btrfs_leaf_data(leaf) +
1298 data_end, old_data_start + new_size - data_end);
1299 btrfs_set_item_size(leaf->items + slot, new_size);
1300 btrfs_mark_buffer_dirty(leaf_buf);
1301
1302 ret = 0;
1303 if (btrfs_leaf_free_space(root, leaf) < 0)
1304 BUG();
1305 check_leaf(root, path, 0);
1306 return ret;
1307}
1308
1252int btrfs_extend_item(struct btrfs_trans_handle *trans, struct btrfs_root 1309int btrfs_extend_item(struct btrfs_trans_handle *trans, struct btrfs_root
1253 *root, struct btrfs_path *path, u32 data_size) 1310 *root, struct btrfs_path *path, u32 data_size)
1254{ 1311{
diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h
index 796f19d03ab5..ca3ab160f460 100644
--- a/fs/btrfs/ctree.h
+++ b/fs/btrfs/ctree.h
@@ -999,7 +999,7 @@ static inline void btrfs_mark_buffer_dirty(struct buffer_head *bh)
999 ((type *)(btrfs_leaf_data(leaf) + \ 999 ((type *)(btrfs_leaf_data(leaf) + \
1000 btrfs_item_offset((leaf)->items + (slot)))) 1000 btrfs_item_offset((leaf)->items + (slot))))
1001 1001
1002/* extent-item.c */ 1002/* extent-tree.c */
1003int btrfs_inc_root_ref(struct btrfs_trans_handle *trans, 1003int btrfs_inc_root_ref(struct btrfs_trans_handle *trans,
1004 struct btrfs_root *root); 1004 struct btrfs_root *root);
1005struct buffer_head *btrfs_alloc_free_block(struct btrfs_trans_handle *trans, 1005struct buffer_head *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
@@ -1013,9 +1013,16 @@ int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
1013 *root, u64 blocknr, u64 num_blocks, int pin); 1013 *root, u64 blocknr, u64 num_blocks, int pin);
1014int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans, struct 1014int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans, struct
1015 btrfs_root *root); 1015 btrfs_root *root);
1016int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
1017 struct btrfs_root *root,
1018 u64 blocknr, u64 num_blocks);
1016/* ctree.c */ 1019/* ctree.c */
1017int btrfs_extend_item(struct btrfs_trans_handle *trans, struct btrfs_root 1020int btrfs_extend_item(struct btrfs_trans_handle *trans, struct btrfs_root
1018 *root, struct btrfs_path *path, u32 data_size); 1021 *root, struct btrfs_path *path, u32 data_size);
1022int btrfs_truncate_item(struct btrfs_trans_handle *trans,
1023 struct btrfs_root *root,
1024 struct btrfs_path *path,
1025 u32 new_size);
1019int btrfs_search_slot(struct btrfs_trans_handle *trans, struct btrfs_root 1026int btrfs_search_slot(struct btrfs_trans_handle *trans, struct btrfs_root
1020 *root, struct btrfs_key *key, struct btrfs_path *p, int 1027 *root, struct btrfs_key *key, struct btrfs_path *p, int
1021 ins_len, int cow); 1028 ins_len, int cow);
@@ -1073,11 +1080,10 @@ int btrfs_lookup_inode(struct btrfs_trans_handle *trans, struct btrfs_root
1073 struct btrfs_key *location, int mod); 1080 struct btrfs_key *location, int mod);
1074 1081
1075/* file-item.c */ 1082/* file-item.c */
1076int btrfs_alloc_file_extent(struct btrfs_trans_handle *trans, 1083int btrfs_insert_file_extent(struct btrfs_trans_handle *trans,
1077 struct btrfs_root *root, 1084 struct btrfs_root *root,
1078 u64 objectid, u64 offset, 1085 u64 objectid, u64 pos, u64 offset,
1079 u64 num_blocks, u64 hint_block, 1086 u64 num_blocks);
1080 u64 *result);
1081int btrfs_lookup_file_extent(struct btrfs_trans_handle *trans, 1087int btrfs_lookup_file_extent(struct btrfs_trans_handle *trans,
1082 struct btrfs_root *root, 1088 struct btrfs_root *root,
1083 struct btrfs_path *path, u64 objectid, 1089 struct btrfs_path *path, u64 objectid,
@@ -1090,6 +1096,11 @@ int btrfs_csum_file_block(struct btrfs_trans_handle *trans,
1090int btrfs_csum_verify_file_block(struct btrfs_root *root, 1096int btrfs_csum_verify_file_block(struct btrfs_root *root,
1091 u64 objectid, u64 offset, 1097 u64 objectid, u64 offset,
1092 char *data, size_t len); 1098 char *data, size_t len);
1099struct btrfs_csum_item *btrfs_lookup_csum(struct btrfs_trans_handle *trans,
1100 struct btrfs_root *root,
1101 struct btrfs_path *path,
1102 u64 objectid, u64 offset,
1103 int cow);
1093/* super.c */ 1104/* super.c */
1094extern struct subsystem btrfs_subsys; 1105extern struct subsystem btrfs_subsys;
1095 1106
diff --git a/fs/btrfs/extent-tree.c b/fs/btrfs/extent-tree.c
index d560831c10a7..2cee9df001f6 100644
--- a/fs/btrfs/extent-tree.c
+++ b/fs/btrfs/extent-tree.c
@@ -12,8 +12,9 @@ static int finish_current_insert(struct btrfs_trans_handle *trans, struct
12static int del_pending_extents(struct btrfs_trans_handle *trans, struct 12static int del_pending_extents(struct btrfs_trans_handle *trans, struct
13 btrfs_root *extent_root); 13 btrfs_root *extent_root);
14 14
15static int inc_block_ref(struct btrfs_trans_handle *trans, struct btrfs_root 15int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
16 *root, u64 blocknr, u64 num_blocks) 16 struct btrfs_root *root,
17 u64 blocknr, u64 num_blocks)
17{ 18{
18 struct btrfs_path *path; 19 struct btrfs_path *path;
19 int ret; 20 int ret;
@@ -50,8 +51,9 @@ static int inc_block_ref(struct btrfs_trans_handle *trans, struct btrfs_root
50 return 0; 51 return 0;
51} 52}
52 53
53static int lookup_block_ref(struct btrfs_trans_handle *trans, struct btrfs_root 54static int lookup_extent_ref(struct btrfs_trans_handle *trans,
54 *root, u64 blocknr, u64 num_blocks, u32 *refs) 55 struct btrfs_root *root, u64 blocknr,
56 u64 num_blocks, u32 *refs)
55{ 57{
56 struct btrfs_path *path; 58 struct btrfs_path *path;
57 int ret; 59 int ret;
@@ -80,7 +82,7 @@ static int lookup_block_ref(struct btrfs_trans_handle *trans, struct btrfs_root
80int btrfs_inc_root_ref(struct btrfs_trans_handle *trans, 82int btrfs_inc_root_ref(struct btrfs_trans_handle *trans,
81 struct btrfs_root *root) 83 struct btrfs_root *root)
82{ 84{
83 return inc_block_ref(trans, root, bh_blocknr(root->node), 1); 85 return btrfs_inc_extent_ref(trans, root, bh_blocknr(root->node), 1);
84} 86}
85 87
86int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root, 88int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
@@ -107,13 +109,13 @@ int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
107 continue; 109 continue;
108 fi = btrfs_item_ptr(buf_leaf, i, 110 fi = btrfs_item_ptr(buf_leaf, i,
109 struct btrfs_file_extent_item); 111 struct btrfs_file_extent_item);
110 ret = inc_block_ref(trans, root, 112 ret = btrfs_inc_extent_ref(trans, root,
111 btrfs_file_extent_disk_blocknr(fi), 113 btrfs_file_extent_disk_blocknr(fi),
112 btrfs_file_extent_disk_num_blocks(fi)); 114 btrfs_file_extent_disk_num_blocks(fi));
113 BUG_ON(ret); 115 BUG_ON(ret);
114 } else { 116 } else {
115 blocknr = btrfs_node_blockptr(buf_node, i); 117 blocknr = btrfs_node_blockptr(buf_node, i);
116 ret = inc_block_ref(trans, root, blocknr, 1); 118 ret = btrfs_inc_extent_ref(trans, root, blocknr, 1);
117 BUG_ON(ret); 119 BUG_ON(ret);
118 } 120 }
119 } 121 }
@@ -563,7 +565,7 @@ static int walk_down_tree(struct btrfs_trans_handle *trans, struct btrfs_root
563 565
564 WARN_ON(*level < 0); 566 WARN_ON(*level < 0);
565 WARN_ON(*level >= BTRFS_MAX_LEVEL); 567 WARN_ON(*level >= BTRFS_MAX_LEVEL);
566 ret = lookup_block_ref(trans, root, bh_blocknr(path->nodes[*level]), 568 ret = lookup_extent_ref(trans, root, bh_blocknr(path->nodes[*level]),
567 1, &refs); 569 1, &refs);
568 BUG_ON(ret); 570 BUG_ON(ret);
569 if (refs > 1) 571 if (refs > 1)
@@ -587,7 +589,7 @@ static int walk_down_tree(struct btrfs_trans_handle *trans, struct btrfs_root
587 } 589 }
588 blocknr = btrfs_node_blockptr(btrfs_buffer_node(cur), 590 blocknr = btrfs_node_blockptr(btrfs_buffer_node(cur),
589 path->slots[*level]); 591 path->slots[*level]);
590 ret = lookup_block_ref(trans, root, blocknr, 1, &refs); 592 ret = lookup_extent_ref(trans, root, blocknr, 1, &refs);
591 BUG_ON(ret); 593 BUG_ON(ret);
592 if (refs != 1) { 594 if (refs != 1) {
593 path->slots[*level]++; 595 path->slots[*level]++;
diff --git a/fs/btrfs/file-item.c b/fs/btrfs/file-item.c
index 93d42d650820..f49968ad0a07 100644
--- a/fs/btrfs/file-item.c
+++ b/fs/btrfs/file-item.c
@@ -6,13 +6,11 @@
6#define MAX_CSUM_ITEMS(r) ((((BTRFS_LEAF_DATA_SIZE(r) - \ 6#define MAX_CSUM_ITEMS(r) ((((BTRFS_LEAF_DATA_SIZE(r) - \
7 sizeof(struct btrfs_item)) / \ 7 sizeof(struct btrfs_item)) / \
8 sizeof(struct btrfs_csum_item)) - 1)) 8 sizeof(struct btrfs_csum_item)) - 1))
9int btrfs_alloc_file_extent(struct btrfs_trans_handle *trans, 9int btrfs_insert_file_extent(struct btrfs_trans_handle *trans,
10 struct btrfs_root *root, 10 struct btrfs_root *root,
11 u64 objectid, u64 offset, 11 u64 objectid, u64 pos,
12 u64 num_blocks, u64 hint_block, 12 u64 offset, u64 num_blocks)
13 u64 *result)
14{ 13{
15 struct btrfs_key ins;
16 int ret = 0; 14 int ret = 0;
17 struct btrfs_file_extent_item *item; 15 struct btrfs_file_extent_item *item;
18 struct btrfs_key file_key; 16 struct btrfs_key file_key;
@@ -21,11 +19,13 @@ int btrfs_alloc_file_extent(struct btrfs_trans_handle *trans,
21 path = btrfs_alloc_path(); 19 path = btrfs_alloc_path();
22 BUG_ON(!path); 20 BUG_ON(!path);
23 btrfs_init_path(path); 21 btrfs_init_path(path);
22 /*
24 ret = btrfs_alloc_extent(trans, root, num_blocks, hint_block, 23 ret = btrfs_alloc_extent(trans, root, num_blocks, hint_block,
25 (u64)-1, &ins); 24 (u64)-1, &ins);
25 */
26 BUG_ON(ret); 26 BUG_ON(ret);
27 file_key.objectid = objectid; 27 file_key.objectid = objectid;
28 file_key.offset = offset; 28 file_key.offset = pos;
29 file_key.flags = 0; 29 file_key.flags = 0;
30 btrfs_set_key_type(&file_key, BTRFS_EXTENT_DATA_KEY); 30 btrfs_set_key_type(&file_key, BTRFS_EXTENT_DATA_KEY);
31 31
@@ -34,21 +34,22 @@ int btrfs_alloc_file_extent(struct btrfs_trans_handle *trans,
34 BUG_ON(ret); 34 BUG_ON(ret);
35 item = btrfs_item_ptr(btrfs_buffer_leaf(path->nodes[0]), path->slots[0], 35 item = btrfs_item_ptr(btrfs_buffer_leaf(path->nodes[0]), path->slots[0],
36 struct btrfs_file_extent_item); 36 struct btrfs_file_extent_item);
37 btrfs_set_file_extent_disk_blocknr(item, ins.objectid); 37 btrfs_set_file_extent_disk_blocknr(item, offset);
38 btrfs_set_file_extent_disk_num_blocks(item, ins.offset); 38 btrfs_set_file_extent_disk_num_blocks(item, num_blocks);
39 btrfs_set_file_extent_offset(item, 0); 39 btrfs_set_file_extent_offset(item, 0);
40 btrfs_set_file_extent_num_blocks(item, ins.offset); 40 btrfs_set_file_extent_num_blocks(item, num_blocks);
41 btrfs_set_file_extent_generation(item, trans->transid); 41 btrfs_set_file_extent_generation(item, trans->transid);
42 btrfs_mark_buffer_dirty(path->nodes[0]); 42 btrfs_mark_buffer_dirty(path->nodes[0]);
43 *result = ins.objectid;
44 btrfs_release_path(root, path); 43 btrfs_release_path(root, path);
45 btrfs_free_path(path); 44 btrfs_free_path(path);
46 return 0; 45 return 0;
47} 46}
48 47
49static struct btrfs_csum_item *__lookup_csum_item(struct btrfs_root *root, 48struct btrfs_csum_item *btrfs_lookup_csum(struct btrfs_trans_handle *trans,
50 struct btrfs_path *path, 49 struct btrfs_root *root,
51 u64 objectid, u64 offset) 50 struct btrfs_path *path,
51 u64 objectid, u64 offset,
52 int cow)
52{ 53{
53 int ret; 54 int ret;
54 struct btrfs_key file_key; 55 struct btrfs_key file_key;
@@ -61,19 +62,23 @@ static struct btrfs_csum_item *__lookup_csum_item(struct btrfs_root *root,
61 file_key.offset = offset; 62 file_key.offset = offset;
62 file_key.flags = 0; 63 file_key.flags = 0;
63 btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY); 64 btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY);
64 ret = btrfs_search_slot(NULL, root, &file_key, path, 0, 0); 65printk("__lookup for %Lu\n", offset);
66 ret = btrfs_search_slot(trans, root, &file_key, path, 0, cow);
65 if (ret < 0) 67 if (ret < 0)
66 goto fail; 68 goto fail;
67 leaf = btrfs_buffer_leaf(path->nodes[0]); 69 leaf = btrfs_buffer_leaf(path->nodes[0]);
68 if (ret > 0) { 70 if (ret > 0) {
69 ret = 1; 71 ret = 1;
70 if (path->slots[0] == 0) 72 if (path->slots[0] == 0) {
73printk("fail1\n");
71 goto fail; 74 goto fail;
75 }
72 path->slots[0]--; 76 path->slots[0]--;
73 btrfs_disk_key_to_cpu(&found_key, 77 btrfs_disk_key_to_cpu(&found_key,
74 &leaf->items[path->slots[0]].key); 78 &leaf->items[path->slots[0]].key);
75 if (btrfs_key_type(&found_key) != BTRFS_CSUM_ITEM_KEY || 79 if (btrfs_key_type(&found_key) != BTRFS_CSUM_ITEM_KEY ||
76 found_key.objectid != objectid) { 80 found_key.objectid != objectid) {
81printk("fail2 type %u %Lu %Lu\n", btrfs_key_type(&found_key), found_key.objectid, objectid);
77 goto fail; 82 goto fail;
78 } 83 }
79 csum_offset = (offset - found_key.offset) >> 84 csum_offset = (offset - found_key.offset) >>
@@ -81,6 +86,7 @@ static struct btrfs_csum_item *__lookup_csum_item(struct btrfs_root *root,
81 if (csum_offset >= 86 if (csum_offset >=
82 btrfs_item_size(leaf->items + path->slots[0]) / 87 btrfs_item_size(leaf->items + path->slots[0]) /
83 sizeof(struct btrfs_csum_item)) { 88 sizeof(struct btrfs_csum_item)) {
89printk("fail3, csum offset %lu size %u\n", csum_offset, btrfs_item_size(leaf->items + path->slots[0]) / sizeof(struct btrfs_csum_item));
84 goto fail; 90 goto fail;
85 } 91 }
86 } 92 }
@@ -89,7 +95,7 @@ static struct btrfs_csum_item *__lookup_csum_item(struct btrfs_root *root,
89 return item; 95 return item;
90fail: 96fail:
91 if (ret > 0) 97 if (ret > 0)
92 ret = -EIO; 98 ret = -ENOENT;
93 return ERR_PTR(ret); 99 return ERR_PTR(ret);
94} 100}
95 101
@@ -105,7 +111,7 @@ int btrfs_lookup_file_extent(struct btrfs_trans_handle *trans,
105 int cow = mod != 0; 111 int cow = mod != 0;
106 struct btrfs_csum_item *csum_item; 112 struct btrfs_csum_item *csum_item;
107 113
108 csum_item = __lookup_csum_item(root, path, objectid, offset); 114 csum_item = btrfs_lookup_csum(trans, root, path, objectid, offset, 0);
109 if (IS_ERR(csum_item)) 115 if (IS_ERR(csum_item))
110 return PTR_ERR(csum_item); 116 return PTR_ERR(csum_item);
111 file_key.objectid = objectid; 117 file_key.objectid = objectid;
@@ -113,7 +119,9 @@ int btrfs_lookup_file_extent(struct btrfs_trans_handle *trans,
113 file_key.flags = 0; 119 file_key.flags = 0;
114 btrfs_set_key_type(&file_key, BTRFS_EXTENT_DATA_KEY); 120 btrfs_set_key_type(&file_key, BTRFS_EXTENT_DATA_KEY);
115 btrfs_release_path(root, path); 121 btrfs_release_path(root, path);
122printk("lookup file extent searches for %Lu\n", file_key.offset);
116 ret = btrfs_search_slot(trans, root, &file_key, path, ins_len, cow); 123 ret = btrfs_search_slot(trans, root, &file_key, path, ins_len, cow);
124printk("ret is %d\n", ret);
117 return ret; 125 return ret;
118} 126}
119 127
@@ -134,17 +142,23 @@ int btrfs_csum_file_block(struct btrfs_trans_handle *trans,
134 path = btrfs_alloc_path(); 142 path = btrfs_alloc_path();
135 BUG_ON(!path); 143 BUG_ON(!path);
136 btrfs_init_path(path); 144 btrfs_init_path(path);
145
146 item = btrfs_lookup_csum(trans, root, path, objectid, offset, 0);
147 if (!IS_ERR(item))
148 goto found;
149 btrfs_release_path(root, path);
137 file_key.objectid = objectid; 150 file_key.objectid = objectid;
138 file_key.offset = offset; 151 file_key.offset = offset;
139 file_key.flags = 0; 152 file_key.flags = 0;
140 btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY); 153 btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY);
154printk("searching for csum %Lu %Lu\n", objectid, offset);
141 ret = btrfs_search_slot(trans, root, &file_key, path, 155 ret = btrfs_search_slot(trans, root, &file_key, path,
142 sizeof(struct btrfs_csum_item), 1); 156 sizeof(struct btrfs_csum_item), 1);
157printk("ret %d\n", ret);
143 if (ret < 0) 158 if (ret < 0)
144 goto fail; 159 goto fail;
145 if (ret == 0) { 160 if (ret == 0) {
146 csum_offset = 0; 161 BUG();
147 goto csum;
148 } 162 }
149 if (path->slots[0] == 0) { 163 if (path->slots[0] == 0) {
150 btrfs_release_path(root, path); 164 btrfs_release_path(root, path);
@@ -153,12 +167,15 @@ int btrfs_csum_file_block(struct btrfs_trans_handle *trans,
153 path->slots[0]--; 167 path->slots[0]--;
154 leaf = btrfs_buffer_leaf(path->nodes[0]); 168 leaf = btrfs_buffer_leaf(path->nodes[0]);
155 btrfs_disk_key_to_cpu(&found_key, &leaf->items[path->slots[0]].key); 169 btrfs_disk_key_to_cpu(&found_key, &leaf->items[path->slots[0]].key);
170printk("found key %Lu %Lu %u\n", found_key.objectid, found_key.offset, found_key.flags);
156 csum_offset = (offset - found_key.offset) >> 171 csum_offset = (offset - found_key.offset) >>
157 root->fs_info->sb->s_blocksize_bits; 172 root->fs_info->sb->s_blocksize_bits;
173printk("csum_offset %Lu\n", csum_offset);
158 if (btrfs_key_type(&found_key) != BTRFS_CSUM_ITEM_KEY || 174 if (btrfs_key_type(&found_key) != BTRFS_CSUM_ITEM_KEY ||
159 found_key.objectid != objectid || 175 found_key.objectid != objectid ||
160 csum_offset >= MAX_CSUM_ITEMS(root)) { 176 csum_offset >= MAX_CSUM_ITEMS(root)) {
161 btrfs_release_path(root, path); 177 btrfs_release_path(root, path);
178printk("insert1\n");
162 goto insert; 179 goto insert;
163 } 180 }
164 if (csum_offset >= btrfs_item_size(leaf->items + path->slots[0]) / 181 if (csum_offset >= btrfs_item_size(leaf->items + path->slots[0]) /
@@ -166,11 +183,13 @@ int btrfs_csum_file_block(struct btrfs_trans_handle *trans,
166 ret = btrfs_extend_item(trans, root, path, 183 ret = btrfs_extend_item(trans, root, path,
167 sizeof(struct btrfs_csum_item)); 184 sizeof(struct btrfs_csum_item));
168 BUG_ON(ret); 185 BUG_ON(ret);
186printk("item extended\n");
169 goto csum; 187 goto csum;
170 } 188 }
171 189
172insert: 190insert:
173 csum_offset = 0; 191 csum_offset = 0;
192printk("inserting item %Lu %Lu %u\n", file_key.objectid, file_key.offset, file_key.flags);
174 ret = btrfs_insert_empty_item(trans, root, path, &file_key, 193 ret = btrfs_insert_empty_item(trans, root, path, &file_key,
175 sizeof(struct btrfs_csum_item)); 194 sizeof(struct btrfs_csum_item));
176 if (ret != 0 && ret != -EEXIST) 195 if (ret != 0 && ret != -EEXIST)
@@ -180,12 +199,14 @@ csum:
180 struct btrfs_csum_item); 199 struct btrfs_csum_item);
181 ret = 0; 200 ret = 0;
182 item += csum_offset; 201 item += csum_offset;
202found:
183 ret = btrfs_csum_data(root, data, len, item->csum); 203 ret = btrfs_csum_data(root, data, len, item->csum);
184 btrfs_set_csum_extent_offset(item, extent_offset); 204 btrfs_set_csum_extent_offset(item, extent_offset);
185 btrfs_mark_buffer_dirty(path->nodes[0]); 205 btrfs_mark_buffer_dirty(path->nodes[0]);
186fail: 206fail:
187 btrfs_release_path(root, path); 207 btrfs_release_path(root, path);
188 btrfs_free_path(path); 208 btrfs_free_path(path);
209printk("return ret %d\n", ret);
189 return ret; 210 return ret;
190} 211}
191 212
@@ -208,7 +229,7 @@ int btrfs_csum_verify_file_block(struct btrfs_root *root,
208 btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY); 229 btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY);
209 mutex_lock(&root->fs_info->fs_mutex); 230 mutex_lock(&root->fs_info->fs_mutex);
210 231
211 item = __lookup_csum_item(root, path, objectid, offset); 232 item = btrfs_lookup_csum(NULL, root, path, objectid, offset, 0);
212 if (IS_ERR(item)) { 233 if (IS_ERR(item)) {
213 ret = PTR_ERR(item); 234 ret = PTR_ERR(item);
214 goto fail; 235 goto fail;
diff --git a/fs/btrfs/super.c b/fs/btrfs/super.c
index ec689992fdf4..6a56416147e6 100644
--- a/fs/btrfs/super.c
+++ b/fs/btrfs/super.c
@@ -1089,7 +1089,6 @@ static int btrfs_get_block_lock(struct inode *inode, sector_t iblock,
1089 1089
1090 extent_start = btrfs_disk_key_offset(&leaf->items[path->slots[0]].key); 1090 extent_start = btrfs_disk_key_offset(&leaf->items[path->slots[0]].key);
1091 extent_start = extent_start >> inode->i_blkbits; 1091 extent_start = extent_start >> inode->i_blkbits;
1092 extent_start += btrfs_file_extent_offset(item);
1093 extent_end = extent_start + btrfs_file_extent_num_blocks(item); 1092 extent_end = extent_start + btrfs_file_extent_num_blocks(item);
1094 if (iblock >= extent_start && iblock < extent_end) { 1093 if (iblock >= extent_start && iblock < extent_end) {
1095 err = 0; 1094 err = 0;
@@ -1103,6 +1102,7 @@ allocate:
1103 err = 0; 1102 err = 0;
1104 goto out; 1103 goto out;
1105 } 1104 }
1105#if 0
1106 ret = btrfs_alloc_file_extent(trans, root, objectid, 1106 ret = btrfs_alloc_file_extent(trans, root, objectid,
1107 iblock << inode->i_blkbits, 1107 iblock << inode->i_blkbits,
1108 1, extent_end, &blocknr); 1108 1, extent_end, &blocknr);
@@ -1115,9 +1115,11 @@ allocate:
1115 map_bh(result, inode->i_sb, blocknr); 1115 map_bh(result, inode->i_sb, blocknr);
1116 1116
1117 btrfs_map_bh_to_logical(root, result, blocknr); 1117 btrfs_map_bh_to_logical(root, result, blocknr);
1118#endif
1118out: 1119out:
1119 btrfs_release_path(root, path); 1120 btrfs_release_path(root, path);
1120 btrfs_free_path(path); 1121 btrfs_free_path(path);
1122printk("mapping iblock %lu to %lu\n", iblock, result->b_blocknr);
1121 if (trans) 1123 if (trans)
1122 btrfs_end_transaction(trans, root); 1124 btrfs_end_transaction(trans, root);
1123 return err; 1125 return err;
@@ -1273,8 +1275,244 @@ failed:
1273 return err; 1275 return err;
1274} 1276}
1275 1277
1276static int prepare_pages(struct btrfs_trans_handle *trans, 1278static int drop_csums(struct btrfs_trans_handle *trans,
1277 struct btrfs_root *root, 1279 struct btrfs_root *root,
1280 struct inode *inode,
1281 u64 start, u64 end)
1282{
1283 struct btrfs_path *path;
1284 struct btrfs_leaf *leaf;
1285 struct btrfs_key key;
1286 int slot;
1287 struct btrfs_csum_item *item;
1288 char *old_block = NULL;
1289 u64 cur = start;
1290 u64 found_end;
1291 u64 num_csums;
1292 u64 item_size;
1293 int ret;
1294
1295 path = btrfs_alloc_path();
1296 if (!path)
1297 return -ENOMEM;
1298 while(cur < end) {
1299 item = btrfs_lookup_csum(trans, root, path,
1300 inode->i_ino, cur, 1);
1301 if (IS_ERR(item)) {
1302 cur += root->blocksize;
1303 continue;
1304 }
1305 leaf = btrfs_buffer_leaf(path->nodes[0]);
1306 slot = path->slots[0];
1307 btrfs_disk_key_to_cpu(&key, &leaf->items[slot].key);
1308 item_size = btrfs_item_size(leaf->items + slot);
1309 num_csums = item_size / sizeof(struct btrfs_csum_item);
1310 found_end = key.offset + (num_csums << inode->i_blkbits);
1311 cur = found_end;
1312
1313 if (found_end > end) {
1314 char *src;
1315 old_block = kmalloc(root->blocksize, GFP_NOFS);
1316 src = btrfs_item_ptr(leaf, slot, char);
1317 memcpy(old_block, src, item_size);
1318 }
1319 if (key.offset < start) {
1320 u64 new_size = (start - key.offset) >>
1321 inode->i_blkbits;
1322 new_size *= sizeof(struct btrfs_csum_item);
1323 ret = btrfs_truncate_item(trans, root, path, new_size);
1324 BUG_ON(ret);
1325 } else {
1326 btrfs_del_item(trans, root, path);
1327 }
1328 btrfs_release_path(root, path);
1329 if (found_end > end) {
1330 char *dst;
1331 int i;
1332 int new_size;
1333
1334 num_csums = (found_end - end) >> inode->i_blkbits;
1335 new_size = num_csums * sizeof(struct btrfs_csum_item);
1336 key.offset = end;
1337 ret = btrfs_insert_empty_item(trans, root, path,
1338 &key, new_size);
1339 BUG_ON(ret);
1340 dst = btrfs_item_ptr(btrfs_buffer_leaf(path->nodes[0]),
1341 path->slots[0], char);
1342 memcpy(dst, old_block + item_size - new_size,
1343 new_size);
1344 item = (struct btrfs_csum_item *)dst;
1345 for (i = 0; i < num_csums; i++) {
1346 btrfs_set_csum_extent_offset(item, end);
1347 item++;
1348 }
1349 mark_buffer_dirty(path->nodes[0]);
1350 kfree(old_block);
1351 break;
1352 }
1353 }
1354 btrfs_free_path(path);
1355 return 0;
1356}
1357
1358static int drop_extents(struct btrfs_trans_handle *trans,
1359 struct btrfs_root *root,
1360 struct inode *inode,
1361 u64 start, u64 end)
1362{
1363 int ret;
1364 struct btrfs_key key;
1365 struct btrfs_leaf *leaf;
1366 int slot;
1367 struct btrfs_file_extent_item *extent;
1368 u64 extent_end;
1369 int keep;
1370 struct btrfs_file_extent_item old;
1371 struct btrfs_path *path;
1372 u64 search_start = start;
1373 int bookend;
1374
1375 path = btrfs_alloc_path();
1376 if (!path)
1377 return -ENOMEM;
1378search_again:
1379printk("drop extent inode %lu start %Lu end %Lu\n", inode->i_ino, start, end);
1380 ret = btrfs_lookup_file_extent(trans, root, path, inode->i_ino,
1381 search_start, -1);
1382 if (ret != 0) {
1383printk("lookup failed\n");
1384 goto out;
1385 }
1386 while(1) {
1387 keep = 0;
1388 bookend = 0;
1389 leaf = btrfs_buffer_leaf(path->nodes[0]);
1390 slot = path->slots[0];
1391 btrfs_disk_key_to_cpu(&key, &leaf->items[slot].key);
1392
1393printk("found key %Lu %Lu %u\n", key.objectid, key.offset, key.flags);
1394
1395 extent = btrfs_item_ptr(leaf, slot,
1396 struct btrfs_file_extent_item);
1397 extent_end = key.offset +
1398 (btrfs_file_extent_num_blocks(extent) <<
1399 inode->i_blkbits);
1400printk("extent end is %Lu\n", extent_end);
1401 if (key.offset >= end || key.objectid != inode->i_ino) {
1402 ret = 0;
1403 goto out;
1404 }
1405 if (btrfs_key_type(&key) != BTRFS_EXTENT_DATA_KEY)
1406 goto next_leaf;
1407
1408 if (end < extent_end && end >= key.offset) {
1409 memcpy(&old, extent, sizeof(old));
1410 ret = btrfs_inc_extent_ref(trans, root,
1411 btrfs_file_extent_disk_blocknr(&old),
1412 btrfs_file_extent_disk_num_blocks(&old));
1413 BUG_ON(ret);
1414 bookend = 1;
1415 }
1416
1417 if (start > key.offset) {
1418 u64 new_num;
1419 /* truncate existing extent */
1420 keep = 1;
1421 WARN_ON(start & (root->blocksize - 1));
1422 new_num = (start - key.offset) >> inode->i_blkbits;
1423printk("truncating existing extent, was %Lu ", btrfs_file_extent_num_blocks(extent));
1424 btrfs_set_file_extent_num_blocks(extent, new_num);
1425printk("now %Lu\n", btrfs_file_extent_num_blocks(extent));
1426
1427 mark_buffer_dirty(path->nodes[0]);
1428 }
1429 if (!keep) {
1430 u64 disk_blocknr;
1431 u64 disk_num_blocks;
1432printk("del old\n");
1433 disk_blocknr = btrfs_file_extent_disk_blocknr(extent);
1434 disk_num_blocks =
1435 btrfs_file_extent_disk_num_blocks(extent);
1436 search_start = key.offset +
1437 (btrfs_file_extent_num_blocks(extent) <<
1438 inode->i_blkbits);
1439 ret = btrfs_del_item(trans, root, path);
1440 BUG_ON(ret);
1441 btrfs_release_path(root, path);
1442
1443 ret = btrfs_free_extent(trans, root, disk_blocknr,
1444 disk_num_blocks, 0);
1445
1446 BUG_ON(ret);
1447 if (!bookend && search_start >= end) {
1448 ret = 0;
1449 goto out;
1450 }
1451 if (!bookend)
1452 goto search_again;
1453 }
1454 if (bookend) {
1455 /* create bookend */
1456 struct btrfs_key ins;
1457printk("bookend! extent end %Lu\n", extent_end);
1458 ins.objectid = inode->i_ino;
1459 ins.offset = end;
1460 ins.flags = 0;
1461 btrfs_set_key_type(&ins, BTRFS_EXTENT_DATA_KEY);
1462
1463 btrfs_release_path(root, path);
1464 ret = drop_csums(trans, root, inode, start, end);
1465 BUG_ON(ret);
1466 ret = btrfs_insert_empty_item(trans, root, path, &ins,
1467 sizeof(*extent));
1468 BUG_ON(ret);
1469 extent = btrfs_item_ptr(
1470 btrfs_buffer_leaf(path->nodes[0]),
1471 path->slots[0],
1472 struct btrfs_file_extent_item);
1473 btrfs_set_file_extent_disk_blocknr(extent,
1474 btrfs_file_extent_disk_blocknr(&old));
1475 btrfs_set_file_extent_disk_num_blocks(extent,
1476 btrfs_file_extent_disk_num_blocks(&old));
1477
1478 btrfs_set_file_extent_offset(extent,
1479 btrfs_file_extent_offset(&old) +
1480 ((end - key.offset) >> inode->i_blkbits));
1481 WARN_ON(btrfs_file_extent_num_blocks(&old) <
1482 (end - key.offset) >> inode->i_blkbits);
1483 btrfs_set_file_extent_num_blocks(extent,
1484 btrfs_file_extent_num_blocks(&old) -
1485 ((end - key.offset) >> inode->i_blkbits));
1486
1487 btrfs_set_file_extent_generation(extent,
1488 btrfs_file_extent_generation(&old));
1489printk("new bookend at offset %Lu, file_extent_offset %Lu, file_extent_num_blocks %Lu\n", end, btrfs_file_extent_offset(extent), btrfs_file_extent_num_blocks(extent));
1490 btrfs_mark_buffer_dirty(path->nodes[0]);
1491 ret = 0;
1492 goto out_nocsum;
1493 }
1494next_leaf:
1495 if (slot >= btrfs_header_nritems(&leaf->header) - 1) {
1496 ret = btrfs_next_leaf(root, path);
1497 if (ret) {
1498 ret = 0;
1499 goto out;
1500 }
1501 } else {
1502 path->slots[0]++;
1503 }
1504 }
1505
1506out:
1507 ret = drop_csums(trans, root, inode, start, end);
1508 BUG_ON(ret);
1509
1510out_nocsum:
1511 btrfs_free_path(path);
1512 return ret;
1513}
1514
1515static int prepare_pages(struct btrfs_root *root,
1278 struct file *file, 1516 struct file *file,
1279 struct page **pages, 1517 struct page **pages,
1280 size_t num_pages, 1518 size_t num_pages,
@@ -1289,7 +1527,6 @@ static int prepare_pages(struct btrfs_trans_handle *trans,
1289 struct inode *inode = file->f_path.dentry->d_inode; 1527 struct inode *inode = file->f_path.dentry->d_inode;
1290 int offset; 1528 int offset;
1291 int err = 0; 1529 int err = 0;
1292 int ret;
1293 int this_write; 1530 int this_write;
1294 struct buffer_head *bh; 1531 struct buffer_head *bh;
1295 struct buffer_head *head; 1532 struct buffer_head *head;
@@ -1305,18 +1542,21 @@ static int prepare_pages(struct btrfs_trans_handle *trans,
1305 } 1542 }
1306 offset = pos & (PAGE_CACHE_SIZE -1); 1543 offset = pos & (PAGE_CACHE_SIZE -1);
1307 this_write = min(PAGE_CACHE_SIZE - offset, write_bytes); 1544 this_write = min(PAGE_CACHE_SIZE - offset, write_bytes);
1308 if (!PageUptodate(pages[i]) && 1545#if 0
1309 (pages[i]->index == first_index || 1546 if ((pages[i]->index == first_index ||
1310 pages[i]->index == last_index) && pos < isize) { 1547 pages[i]->index == last_index) && pos < isize &&
1548 !PageUptodate(pages[i])) {
1311 ret = mpage_readpage(pages[i], btrfs_get_block); 1549 ret = mpage_readpage(pages[i], btrfs_get_block);
1312 BUG_ON(ret); 1550 BUG_ON(ret);
1313 lock_page(pages[i]); 1551 lock_page(pages[i]);
1314 } 1552 }
1553#endif
1315 create_empty_buffers(pages[i], root->fs_info->sb->s_blocksize, 1554 create_empty_buffers(pages[i], root->fs_info->sb->s_blocksize,
1316 (1 << BH_Uptodate)); 1555 (1 << BH_Uptodate));
1317 head = page_buffers(pages[i]); 1556 head = page_buffers(pages[i]);
1318 bh = head; 1557 bh = head;
1319 do { 1558 do {
1559printk("mapping page %lu to block %Lu\n", pages[i]->index, alloc_extent_start);
1320 err = btrfs_map_bh_to_logical(root, bh, 1560 err = btrfs_map_bh_to_logical(root, bh,
1321 alloc_extent_start); 1561 alloc_extent_start);
1322 BUG_ON(err); 1562 BUG_ON(err);
@@ -1351,7 +1591,7 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf,
1351 int ret = 0; 1591 int ret = 0;
1352 struct inode *inode = file->f_path.dentry->d_inode; 1592 struct inode *inode = file->f_path.dentry->d_inode;
1353 struct btrfs_root *root = BTRFS_I(inode)->root; 1593 struct btrfs_root *root = BTRFS_I(inode)->root;
1354 struct page *pages[1]; 1594 struct page *pages[8];
1355 unsigned long first_index; 1595 unsigned long first_index;
1356 unsigned long last_index; 1596 unsigned long last_index;
1357 u64 start_pos; 1597 u64 start_pos;
@@ -1359,6 +1599,7 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf,
1359 u64 alloc_extent_start; 1599 u64 alloc_extent_start;
1360 u64 orig_extent_start; 1600 u64 orig_extent_start;
1361 struct btrfs_trans_handle *trans; 1601 struct btrfs_trans_handle *trans;
1602 struct btrfs_key ins;
1362 1603
1363 if (file->f_flags & O_DIRECT) 1604 if (file->f_flags & O_DIRECT)
1364 return -EINVAL; 1605 return -EINVAL;
@@ -1390,16 +1631,24 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf,
1390 trans = btrfs_start_transaction(root, 1); 1631 trans = btrfs_start_transaction(root, 1);
1391 if (!trans) { 1632 if (!trans) {
1392 err = -ENOMEM; 1633 err = -ENOMEM;
1634 mutex_unlock(&root->fs_info->fs_mutex);
1393 goto out_unlock; 1635 goto out_unlock;
1394 } 1636 }
1395 ret = btrfs_alloc_file_extent(trans, root, inode->i_ino, 1637 if (start_pos < inode->i_size) {
1396 start_pos, num_blocks, 1, 1638 ret = drop_extents(trans, root, inode,
1397 &alloc_extent_start); 1639 start_pos,
1398 BUG_ON(ret); 1640 (pos + count + root->blocksize -1) &
1399 1641 ~(root->blocksize - 1));
1642 }
1400 orig_extent_start = start_pos; 1643 orig_extent_start = start_pos;
1401 ret = btrfs_end_transaction(trans, root); 1644 ret = btrfs_alloc_extent(trans, root, num_blocks, 1,
1645 (u64)-1, &ins);
1646 BUG_ON(ret);
1647 ret = btrfs_insert_file_extent(trans, root, inode->i_ino,
1648 start_pos, ins.objectid, ins.offset);
1402 BUG_ON(ret); 1649 BUG_ON(ret);
1650 alloc_extent_start = ins.objectid;
1651 ret = btrfs_end_transaction(trans, root);
1403 mutex_unlock(&root->fs_info->fs_mutex); 1652 mutex_unlock(&root->fs_info->fs_mutex);
1404 1653
1405 while(count > 0) { 1654 while(count > 0) {
@@ -1407,16 +1656,21 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf,
1407 size_t write_bytes = min(count, PAGE_CACHE_SIZE - offset); 1656 size_t write_bytes = min(count, PAGE_CACHE_SIZE - offset);
1408 size_t num_pages = (write_bytes + PAGE_CACHE_SIZE - 1) >> 1657 size_t num_pages = (write_bytes + PAGE_CACHE_SIZE - 1) >>
1409 PAGE_CACHE_SHIFT; 1658 PAGE_CACHE_SHIFT;
1410 ret = prepare_pages(NULL, root, file, pages, num_pages, 1659printk("num_pages is %lu\n", num_pages);
1660
1661 memset(pages, 0, sizeof(pages));
1662 ret = prepare_pages(root, file, pages, num_pages,
1411 pos, first_index, last_index, 1663 pos, first_index, last_index,
1412 write_bytes, alloc_extent_start); 1664 write_bytes, alloc_extent_start);
1413 BUG_ON(ret); 1665 BUG_ON(ret);
1666
1414 /* FIXME blocks != pagesize */ 1667 /* FIXME blocks != pagesize */
1415 alloc_extent_start += num_pages; 1668 alloc_extent_start += num_pages;
1416 ret = btrfs_copy_from_user(pos, num_pages, 1669 ret = btrfs_copy_from_user(pos, num_pages,
1417 write_bytes, pages, buf); 1670 write_bytes, pages, buf);
1418 BUG_ON(ret); 1671 BUG_ON(ret);
1419 1672
1673printk("2num_pages is %lu\n", num_pages);
1420 ret = dirty_and_release_pages(NULL, root, file, pages, 1674 ret = dirty_and_release_pages(NULL, root, file, pages,
1421 num_pages, orig_extent_start, 1675 num_pages, orig_extent_start,
1422 pos, write_bytes); 1676 pos, write_bytes);