120 lines
3.3 KiB
C
120 lines
3.3 KiB
C
#include <linux/module.h>
|
|
#include "ctree.h"
|
|
#include "disk-io.h"
|
|
#include "transaction.h"
|
|
|
|
int btrfs_alloc_file_extent(struct btrfs_trans_handle *trans,
|
|
struct btrfs_root *root,
|
|
u64 objectid, u64 offset,
|
|
u64 num_blocks, u64 hint_block,
|
|
u64 *result)
|
|
{
|
|
struct btrfs_key ins;
|
|
int ret = 0;
|
|
struct btrfs_file_extent_item *item;
|
|
struct btrfs_key file_key;
|
|
struct btrfs_path path;
|
|
|
|
btrfs_init_path(&path);
|
|
ret = btrfs_alloc_extent(trans, root, num_blocks, hint_block,
|
|
(u64)-1, objectid, &ins);
|
|
BUG_ON(ret);
|
|
file_key.objectid = objectid;
|
|
file_key.offset = offset;
|
|
file_key.flags = 0;
|
|
btrfs_set_key_type(&file_key, BTRFS_EXTENT_DATA_KEY);
|
|
|
|
ret = btrfs_insert_empty_item(trans, root, &path, &file_key,
|
|
sizeof(*item));
|
|
BUG_ON(ret);
|
|
item = btrfs_item_ptr(btrfs_buffer_leaf(path.nodes[0]), path.slots[0],
|
|
struct btrfs_file_extent_item);
|
|
btrfs_set_file_extent_disk_blocknr(item, ins.objectid);
|
|
btrfs_set_file_extent_disk_num_blocks(item, ins.offset);
|
|
btrfs_set_file_extent_offset(item, 0);
|
|
btrfs_set_file_extent_num_blocks(item, ins.offset);
|
|
btrfs_set_file_extent_generation(item, trans->transid);
|
|
mark_buffer_dirty(path.nodes[0]);
|
|
*result = ins.objectid;
|
|
btrfs_release_path(root, &path);
|
|
return 0;
|
|
}
|
|
|
|
int btrfs_lookup_file_extent(struct btrfs_trans_handle *trans,
|
|
struct btrfs_root *root,
|
|
struct btrfs_path *path, u64 objectid,
|
|
u64 offset, int mod)
|
|
{
|
|
int ret;
|
|
struct btrfs_key file_key;
|
|
int ins_len = mod < 0 ? -1 : 0;
|
|
int cow = mod != 0;
|
|
|
|
file_key.objectid = objectid;
|
|
file_key.offset = offset;
|
|
file_key.flags = 0;
|
|
btrfs_set_key_type(&file_key, BTRFS_EXTENT_DATA_KEY);
|
|
ret = btrfs_search_slot(trans, root, &file_key, path, ins_len, cow);
|
|
return ret;
|
|
}
|
|
|
|
int btrfs_csum_file_block(struct btrfs_trans_handle *trans,
|
|
struct btrfs_root *root,
|
|
u64 objectid, u64 offset,
|
|
char *data, size_t len)
|
|
{
|
|
int ret;
|
|
struct btrfs_key file_key;
|
|
struct btrfs_path path;
|
|
struct btrfs_csum_item *item;
|
|
|
|
btrfs_init_path(&path);
|
|
file_key.objectid = objectid;
|
|
file_key.offset = offset;
|
|
file_key.flags = 0;
|
|
btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY);
|
|
ret = btrfs_insert_empty_item(trans, root, &path, &file_key,
|
|
BTRFS_CSUM_SIZE);
|
|
if (ret != 0 && ret != -EEXIST)
|
|
goto fail;
|
|
item = btrfs_item_ptr(btrfs_buffer_leaf(path.nodes[0]), path.slots[0],
|
|
struct btrfs_csum_item);
|
|
ret = 0;
|
|
ret = btrfs_csum_data(root, data, len, item->csum);
|
|
mark_buffer_dirty(path.nodes[0]);
|
|
fail:
|
|
btrfs_release_path(root, &path);
|
|
return ret;
|
|
}
|
|
|
|
int btrfs_csum_verify_file_block(struct btrfs_root *root,
|
|
u64 objectid, u64 offset,
|
|
char *data, size_t len)
|
|
{
|
|
int ret;
|
|
struct btrfs_key file_key;
|
|
struct btrfs_path path;
|
|
struct btrfs_csum_item *item;
|
|
char result[BTRFS_CSUM_SIZE];
|
|
|
|
btrfs_init_path(&path);
|
|
file_key.objectid = objectid;
|
|
file_key.offset = offset;
|
|
file_key.flags = 0;
|
|
btrfs_set_key_type(&file_key, BTRFS_CSUM_ITEM_KEY);
|
|
ret = btrfs_search_slot(NULL, root, &file_key, &path, 0, 0);
|
|
if (ret)
|
|
goto fail;
|
|
item = btrfs_item_ptr(btrfs_buffer_leaf(path.nodes[0]), path.slots[0],
|
|
struct btrfs_csum_item);
|
|
ret = 0;
|
|
ret = btrfs_csum_data(root, data, len, result);
|
|
WARN_ON(ret);
|
|
if (memcmp(result, item->csum, BTRFS_CSUM_SIZE))
|
|
ret = 1;
|
|
fail:
|
|
btrfs_release_path(root, &path);
|
|
return ret;
|
|
}
|
|
|