Re: [PATCH] ext4:Make FIEMAP and delayed allocation play well together.

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi,

Can this patch be merged now?

On Sat, Feb 26, 2011 at 11:05 PM, Yongqiang Yang <xiaoqiangnk@xxxxxxxxx> wrote:
> 1. lookup dirty pages with specified range in pagecache.
>   If no page is got, then there is no delayed-extent and
>   return with EXT_CONTINUE.
> 2. find the 1st mapped buffer,
> 3. check if the mapped buffer is both in the request range
>   and a delayed buffer. If not, there is no delayed-extent,
>   then return.
> 4. a delayed-extent is found, the extent will be collected.
>
> Reported by Chris Mason <chris.mason@xxxxxxxxxx>:
> We've had reports on btrfs that cp is giving us files full of zeros
> instead of actually copying them.  It was tracked down to a bug with
> the btrfs fiemap implementation where it was returning holes for
> delalloc ranges.
>
> Newer versions of cp are trusting fiemap to tell it where the holes
> are, which does seem like a pretty neat trick.
>
> I decided to give xfs and ext4 a shot with a few tests cases too, xfs
> passed with all the ones btrfs was getting wrong, and ext4 got the basic
> delalloc case right.
> $ mkfs.ext4 /dev/xxx
> $ mount /dev/xxx /mnt
> $ dd if=/dev/zero of=/mnt/foo bs=1M count=1
> $ fiemap-test foo
> ext:   0 logical: [       0..     255] phys:        0..     255
> flags: 0x007 tot: 256
>
> Horray!  But once we throw a hole in, things go bad:
> $ mkfs.ext4 /dev/xxx
> $ mount /dev/xxx /mnt
> $ dd if=/dev/zero of=/mnt/foo bs=1M count=1 seek=1
> $ fiemap-test foo
> < no output >
>
> We've got a delalloc extent after the hole and ext4 fiemap didn't find
> it.  If I run sync to kick the delalloc out:
> $sync
> $ fiemap-test foo
> ext:   0 logical: [     256..     511] phys:    34048..   34303
> flags: 0x001 tot: 256
>
> fiemap-test is sitting in my /usr/local/bin, and I have no idea how it
> got there.  It's full of pretty comments so I know it isn't mine, but
> you can grab it here:
>
> http://oss.oracle.com/~mason/fiemap-test.c
>
> xfsqa has a fiemap program too.
>
> After Fix, test results are as follows:
> ext:   0 logical: [     256..     511] phys:        0..     255
> flags: 0x007 tot: 256
> ext:   0 logical: [     256..     511] phys:    33280..   33535
> flags: 0x001 tot: 256
>
> $ mkfs.ext4 /dev/xxx
> $ mount /dev/xxx /mnt
> $ dd if=/dev/zero of=/mnt/foo bs=1M count=1 seek=1
> $ sync
> $ dd if=/dev/zero of=/mnt/foo bs=1M count=1 seek=3
> $ dd if=/dev/zero of=/mnt/foo bs=1M count=1 seek=5
> $ fiemap-test foo
> ext:   0 logical: [     256..     511] phys:    33280..   33535
> flags: 0x000 tot: 256
> ext:   1 logical: [     768..    1023] phys:        0..     255
> flags: 0x006 tot: 256
> ext:   2 logical: [    1280..    1535] phys:        0..     255
> flags: 0x007 tot: 256
>
> Signed-off-by: Yongqiang Yang <xiaoqiangnk@xxxxxxxxx>
> ---
>  fs/ext4/extents.c |  187 ++++++++++++++++++++++++++++++++++++++++++-----------
>  1 files changed, 148 insertions(+), 39 deletions(-)
>
> diff --git a/fs/ext4/extents.c b/fs/ext4/extents.c
> index ccce8a7..402a7e8 100644
> --- a/fs/ext4/extents.c
> +++ b/fs/ext4/extents.c
> @@ -3775,6 +3775,7 @@ int ext4_convert_unwritten_extents(struct inode *inode, loff_t offset,
>        }
>        return ret > 0 ? ret2 : ret;
>  }
> +
>  /*
>  * Callback function called for each extent to gather FIEMAP information.
>  */
> @@ -3782,38 +3783,162 @@ static int ext4_ext_fiemap_cb(struct inode *inode, struct ext4_ext_path *path,
>                       struct ext4_ext_cache *newex, struct ext4_extent *ex,
>                       void *data)
>  {
> -       struct fiemap_extent_info *fieinfo = data;
> -       unsigned char blksize_bits = inode->i_sb->s_blocksize_bits;
>        __u64   logical;
>        __u64   physical;
>        __u64   length;
> +       loff_t  size;
>        __u32   flags = 0;
> -       int     error;
> +       int             ret = 0;
> +       struct fiemap_extent_info *fieinfo = data;
> +       unsigned char blksize_bits;
>
> -       logical =  (__u64)newex->ec_block << blksize_bits;
> +       blksize_bits = inode->i_sb->s_blocksize_bits;
> +       logical = (__u64)newex->ec_block << blksize_bits;
>
>        if (newex->ec_start == 0) {
> -               pgoff_t offset;
> -               struct page *page;
> +               /*
> +                * No extent in extent-tree contains block @newex->ec_start,
> +                * then the block may stay in 1)a hole or 2)delayed-extent.
> +                *
> +                * Holes or delayed-extents are processed as follows.
> +                * 1. lookup dirty pages with specified range in pagecache.
> +                *    If no page is got, then there is no delayed-extent and
> +                *    return with EXT_CONTINUE.
> +                * 2. find the 1st mapped buffer,
> +                * 3. check if the mapped buffer is both in the request range
> +                *    and a delayed buffer. If not, there is no delayed-extent,
> +                *    then return.
> +                * 4. a delayed-extent is found, the extent will be collected.
> +                */
> +               ext4_lblk_t     end = 0;
> +               pgoff_t         last_offset;
> +               pgoff_t         offset;
> +               pgoff_t         index;
> +               struct page     **pages = NULL;
>                struct buffer_head *bh = NULL;
> +               struct buffer_head *head = NULL;
> +               unsigned int nr_pages = PAGE_SIZE / sizeof(struct page *);
> +
> +               pages = kmalloc(PAGE_SIZE, GFP_KERNEL);
> +               if (pages == NULL)
> +                       return -EFAULT;
>
>                offset = logical >> PAGE_SHIFT;
> -               page = find_get_page(inode->i_mapping, offset);
> -               if (!page || !page_has_buffers(page))
> -                       return EXT_CONTINUE;
> +repeat:
> +               last_offset = offset;
> +               head = NULL;
> +               ret = find_get_pages_tag(inode->i_mapping, &offset,
> +                                       PAGECACHE_TAG_DIRTY, nr_pages, pages);
> +
> +               if (!(flags & FIEMAP_EXTENT_DELALLOC)) {
> +                       /* First time, try to find a mapped buffer. */
> +                       if (ret == 0) {
> +out:
> +                               for (index = 0; index < ret; index++)
> +                                       page_cache_release(pages[index]);
> +                               /* just a hole. */
> +                               kfree(pages);
> +                               return EXT_CONTINUE;
> +                       }
>
> -               bh = page_buffers(page);
> +                       /* Try to find the 1st mapped buffer. */
> +                       end = ((__u64)pages[0]->index << PAGE_SHIFT) >>
> +                                 blksize_bits;
> +                       if (!page_has_buffers(pages[0]))
> +                               goto out;
> +                       head = page_buffers(pages[0]);
> +                       if (!head)
> +                               goto out;
>
> -               if (!bh)
> -                       return EXT_CONTINUE;
> +                       bh = head;
> +                       do {
> +                               if (buffer_mapped(bh)) {
> +                                       /* get the 1st mapped buffer. */
> +                                       if (end > newex->ec_block +
> +                                               newex->ec_len)
> +                                               /* The buffer is out of
> +                                                * the request range.
> +                                                */
> +                                               goto out;
> +                                       goto found_mapped_buffer;
> +                               }
> +                               bh = bh->b_this_page;
> +                               end++;
> +                       } while (bh != head);
>
> -               if (buffer_delay(bh)) {
> -                       flags |= FIEMAP_EXTENT_DELALLOC;
> -                       page_cache_release(page);
> +                       /* No mapped buffer found. */
> +                       goto out;
>                } else {
> -                       page_cache_release(page);
> -                       return EXT_CONTINUE;
> +                       /*Find contiguous delayed buffers. */
> +                       if (ret > 0 && pages[0]->index == last_offset)
> +                               head = page_buffers(pages[0]);
> +                       bh = head;
> +               }
> +
> +found_mapped_buffer:
> +               if (bh != NULL && buffer_delay(bh)) {
> +                       /* 1st or contiguous delayed buffer found. */
> +                       if (!(flags & FIEMAP_EXTENT_DELALLOC)) {
> +                               /*
> +                                * 1st delayed buffer found, record
> +                                * the start of extent.
> +                                */
> +                               flags |= FIEMAP_EXTENT_DELALLOC;
> +                               newex->ec_block = end;
> +                               logical = (__u64)end << blksize_bits;
> +                       }
> +                       /* Find contiguous delayed buffers. */
> +                       do {
> +                               if (!buffer_delay(bh))
> +                                       goto found_delayed_extent;
> +                               bh = bh->b_this_page;
> +                               end++;
> +                       } while (bh != head);
> +
> +                       for (index = 1; index < ret; index++) {
> +                               if (!page_has_buffers(pages[index])) {
> +                                       bh = NULL;
> +                                       break;
> +                               }
> +                               head = page_buffers(pages[index]);
> +                               if (!head) {
> +                                       bh = NULL;
> +                                       break;
> +                               }
> +                               if (pages[index]->index !=
> +                                       pages[0]->index + index) {
> +                                       /* Blocks are not contiguous. */
> +                                       bh = NULL;
> +                                       break;
> +                               }
> +                               bh = head;
> +                               do {
> +                                       if (!buffer_delay(bh))
> +                                               /* Delayed-extent ends. */
> +                                               goto found_delayed_extent;
> +                                       bh = bh->b_this_page;
> +                                       end++;
> +                               } while (bh != head);
> +                       }
> +               } else if (!(flags & FIEMAP_EXTENT_DELALLOC))
> +                       /* a hole found. */
> +                       goto out;
> +
> +found_delayed_extent:
> +               newex->ec_len = min(end - newex->ec_block,
> +                                               (ext4_lblk_t)EXT_INIT_MAX_LEN);
> +               if (ret == nr_pages && bh != NULL &&
> +                       newex->ec_len < EXT_INIT_MAX_LEN &&
> +                       buffer_delay(bh)) {
> +                       /* Have not collected an extent and continue. */
> +                       for (index = 0; index < ret; index++)
> +                               page_cache_release(pages[index]);
> +                       goto repeat;
>                }
> +
> +               for (index = 0; index < ret; index++)
> +                       page_cache_release(pages[index]);
> +               kfree(pages);
>        }
>
>        physical = (__u64)newex->ec_start << blksize_bits;
> @@ -3822,32 +3947,16 @@ static int ext4_ext_fiemap_cb(struct inode *inode, struct ext4_ext_path *path,
>        if (ex && ext4_ext_is_uninitialized(ex))
>                flags |= FIEMAP_EXTENT_UNWRITTEN;
>
> -       /*
> -        * If this extent reaches EXT_MAX_BLOCK, it must be last.
> -        *
> -        * Or if ext4_ext_next_allocated_block is EXT_MAX_BLOCK,
> -        * this also indicates no more allocated blocks.
> -        *
> -        * XXX this might miss a single-block extent at EXT_MAX_BLOCK
> -        */
> -       if (ext4_ext_next_allocated_block(path) == EXT_MAX_BLOCK ||
> -           newex->ec_block + newex->ec_len - 1 == EXT_MAX_BLOCK) {
> -               loff_t size = i_size_read(inode);
> -               loff_t bs = EXT4_BLOCK_SIZE(inode->i_sb);
> -
> +       size = i_size_read(inode);
> +       if (logical + length >= size)
>                flags |= FIEMAP_EXTENT_LAST;
> -               if ((flags & FIEMAP_EXTENT_DELALLOC) &&
> -                   logical+length > size)
> -                       length = (size - logical + bs - 1) & ~(bs-1);
> -       }
>
> -       error = fiemap_fill_next_extent(fieinfo, logical, physical,
> +       ret = fiemap_fill_next_extent(fieinfo, logical, physical,
>                                        length, flags);
> -       if (error < 0)
> -               return error;
> -       if (error == 1)
> +       if (ret < 0)
> +               return ret;
> +       if (ret == 1)
>                return EXT_BREAK;
> -
>        return EXT_CONTINUE;
>  }
>
> --
> 1.7.4
>
>



-- 
Best Wishes
Yongqiang Yang
--
To unsubscribe from this list: send the line "unsubscribe linux-ext4" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Reiser Filesystem Development]     [Ceph FS]     [Kernel Newbies]     [Security]     [Netfilter]     [Bugtraq]     [Linux FS]     [Yosemite National Park]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Device Mapper]     [Linux Media]

  Powered by Linux