On Fri, Jan 29, 2016 at 01:03:26PM +0800, Qu Wenruo wrote:
> Introduce new function, migrate_reserved_ranges() to migrate used fs
> data in btrfs reserved space.
>
> Unlike old implement, which will need to relocate all the complicated
> csum and reference relocation, previous patches already ensure such
> reserved ranges won't be allocated.
> So here we only need copy these data out and create new
> extent/csum/reference.
>
> Signed-off-by: Qu Wenruo <quwenruo@xxxxxxxxxxxxxx>
> Signed-off-by: David Sterba <dsterba@xxxxxxxx>
> ---
> btrfs-convert.c | 124 +++++++++++++++++++++++++++++++++++++++++++++++++++++++-
> 1 file changed, 122 insertions(+), 2 deletions(-)
>
> diff --git a/btrfs-convert.c b/btrfs-convert.c
> index 16e2309..f6126db 100644
> --- a/btrfs-convert.c
> +++ b/btrfs-convert.c
> @@ -1679,6 +1679,123 @@ static int create_image_file_range_v2(struct btrfs_trans_handle *trans,
> return ret;
> }
>
> +
> +/*
> + * Relocate old fs data in one reserved ranges
> + *
> + * Since all old fs data in reserved range is not covered by any chunk nor
> + * data extent, we don't need to handle any reference but add new
> + * extent/reference, which makes codes more clear
> + */
> +static int migrate_one_reserved_range(struct btrfs_trans_handle *trans,
> + struct btrfs_root *root,
> + struct cache_tree *used,
> + struct btrfs_inode_item *inode, int fd,
> + u64 ino, u64 start, u64 len, int datacsum)
> +{
> + u64 cur_off = start;
> + u64 cur_len = len;
> + struct cache_extent *cache;
> + struct btrfs_key key;
> + struct extent_buffer *eb;
> + int ret = 0;
> +
> + while (cur_off < start + len) {
> + cache = lookup_cache_extent(used, cur_off, cur_len);
> + if (!cache)
> + break;
> + cur_off = max(cache->start, cur_off);
> + cur_len = min(cache->start + cache->size, start + len) -
> + cur_off;
> + BUG_ON(cur_len < root->sectorsize);
> +
> + /* reserve extent for the data */
> + ret = btrfs_reserve_extent(trans, root, cur_len, 0, 0, (u64)-1,
> + &key, 1);
> + if (ret < 0)
> + break;
> +
> + eb = malloc(sizeof(*eb) + cur_len);
> + if (!eb) {
> + ret = -ENOMEM;
> + break;
> + }
> +
> + ret = pread(fd, eb->data, cur_len, cur_off);
> + if (ret < cur_len) {
> + ret = (ret < 0 ? ret : -EIO);
> + free(eb);
> + break;
> + }
> + eb->start = key.objectid;
> + eb->len = key.offset;
> +
> + /* Write the data */
> + ret = write_and_map_eb(trans, root, eb);
> + free(eb);
> + if (ret < 0)
> + break;
With write_data_to_disk(), we don't have to create eb for write.
Thanks,
-liubo
> +
> + /* Now handle extent item and file extent things */
> + ret = btrfs_record_file_extent(trans, root, ino, inode, cur_off,
> + key.objectid, key.offset);
> + if (ret < 0)
> + break;
> + /* Finally, insert csum items */
> + if (datacsum)
> + ret = csum_disk_extent(trans, root, key.objectid,
> + key.offset);
> +
> + cur_off += key.offset;
> + cur_len = start + len - cur_off;
> + }
> + return ret;
> +}
> +
> +/*
> + * Relocate the used ext2 data in reserved ranges
> + * [0,1M)
> + * [btrfs_sb_offset(1), +BTRFS_STRIPE_LEN)
> + * [btrfs_sb_offset(2), +BTRFS_STRIPE_LEN)
> + */
> +static int migrate_reserved_ranges(struct btrfs_trans_handle *trans,
> + struct btrfs_root *root,
> + struct cache_tree *used,
> + struct btrfs_inode_item *inode, int fd,
> + u64 ino, u64 total_bytes, int datacsum)
> +{
> + u64 cur_off;
> + u64 cur_len;
> + int ret = 0;
> +
> + /* 0 ~ 1M */
> + cur_off = 0;
> + cur_len = 1024 * 1024;
> + ret = migrate_one_reserved_range(trans, root, used, inode, fd, ino,
> + cur_off, cur_len, datacsum);
> + if (ret < 0)
> + return ret;
> +
> + /* second sb(fisrt sb is included in 0~1M) */
> + cur_off = btrfs_sb_offset(1);
> + cur_len = min(total_bytes, cur_off + BTRFS_STRIPE_LEN) - cur_off;
> + if (cur_off < total_bytes)
> + return ret;
> + ret = migrate_one_reserved_range(trans, root, used, inode, fd, ino,
> + cur_off, cur_len, datacsum);
> + if (ret < 0)
> + return ret;
> +
> + /* Last sb */
> + cur_off = btrfs_sb_offset(2);
> + cur_len = min(total_bytes, cur_off + BTRFS_STRIPE_LEN) - cur_off;
> + if (cur_off < total_bytes)
> + return ret;
> + ret = migrate_one_reserved_range(trans, root, used, inode, fd, ino,
> + cur_off, cur_len, datacsum);
> + return ret;
> +}
> +
> static int wipe_reserved_ranges(struct cache_tree *tree, u64 min_stripe_size,
> int ensure_size);
>
> @@ -1687,11 +1804,10 @@ static int wipe_reserved_ranges(struct cache_tree *tree, u64 min_stripe_size,
> *
> * This is completely fs independent as we have cctx->used, only
> * need to create file extents point to all the posistions.
> - * TODO: Add handler for reserved ranges in next patch
> */
> static int create_image_v2(struct btrfs_root *root,
> struct btrfs_mkfs_config *cfg,
> - struct btrfs_convert_context *cctx,
> + struct btrfs_convert_context *cctx, int fd,
> u64 size, char *name, int datacsum)
> {
> struct btrfs_inode_item buf;
> @@ -1769,6 +1885,10 @@ static int create_image_v2(struct btrfs_root *root,
> goto out;
> cur += len;
> }
> + /* Handle the reserved ranges */
> + ret = migrate_reserved_ranges(trans, root, &cctx->used, &buf, fd, ino,
> + cfg->num_bytes, datacsum);
> +
>
> key.objectid = ino;
> key.type = BTRFS_INODE_ITEM_KEY;
> --
> 2.7.0
>
>
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-btrfs" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at http://vger.kernel.org/majordomo-info.html
--
To unsubscribe from this list: send the line "unsubscribe linux-btrfs" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html