diff options
author | Allison Henderson <achender@linux.vnet.ibm.com> | 2011-05-25 07:41:50 -0400 |
---|---|---|
committer | Theodore Ts'o <tytso@mit.edu> | 2011-05-25 07:41:50 -0400 |
commit | a4bb6b64e39abc0e41ca077725f2a72c868e7622 (patch) | |
tree | 0d911caa13d445d64cce5ea37f424bf066731ea6 /fs/ext4/inode.c | |
parent | e861304b8ed83fe43e36d46794d72641c82d4636 (diff) |
ext4: enable "punch hole" functionality
This patch adds new routines: "ext4_punch_hole" "ext4_ext_punch_hole"
and "ext4_ext_check_cache"
fallocate has been modified to call ext4_punch_hole when the punch hole
flag is passed. At the moment, we only support punching holes in
extents, so this routine is pretty much a wrapper for the ext4_ext_punch_hole
routine.
The ext4_ext_punch_hole routine first completes all outstanding writes
with the associated pages, and then releases them. The unblock
aligned data is zeroed, and all blocks in between are punched out.
The ext4_ext_check_cache routine is very similar to ext4_ext_in_cache
except it accepts a ext4_ext_cache parameter instead of a ext4_extent
parameter. This routine is used by ext4_ext_punch_hole to check and
see if a block in a hole that has been cached. The ext4_ext_cache
parameter is necessary because the members ext4_extent structure are
not large enough to hold a 32 bit value. The existing
ext4_ext_in_cache routine has become a wrapper to this new function.
[ext4 punch hole patch series 5/5 v7]
Signed-off-by: Allison Henderson <achender@us.ibm.com>
Signed-off-by: "Theodore Ts'o" <tytso@mit.edu>
Reviewed-by: Mingming Cao <cmm@us.ibm.com>
Diffstat (limited to 'fs/ext4/inode.c')
-rw-r--r-- | fs/ext4/inode.c | 25 |
1 files changed, 25 insertions, 0 deletions
diff --git a/fs/ext4/inode.c b/fs/ext4/inode.c index fbe11e676624..65576c02c6f5 100644 --- a/fs/ext4/inode.c +++ b/fs/ext4/inode.c | |||
@@ -4419,6 +4419,31 @@ int ext4_can_truncate(struct inode *inode) | |||
4419 | } | 4419 | } |
4420 | 4420 | ||
4421 | /* | 4421 | /* |
4422 | * ext4_punch_hole: punches a hole in a file by releaseing the blocks | ||
4423 | * associated with the given offset and length | ||
4424 | * | ||
4425 | * @inode: File inode | ||
4426 | * @offset: The offset where the hole will begin | ||
4427 | * @len: The length of the hole | ||
4428 | * | ||
4429 | * Returns: 0 on sucess or negative on failure | ||
4430 | */ | ||
4431 | |||
4432 | int ext4_punch_hole(struct file *file, loff_t offset, loff_t length) | ||
4433 | { | ||
4434 | struct inode *inode = file->f_path.dentry->d_inode; | ||
4435 | if (!S_ISREG(inode->i_mode)) | ||
4436 | return -ENOTSUPP; | ||
4437 | |||
4438 | if (!ext4_test_inode_flag(inode, EXT4_INODE_EXTENTS)) { | ||
4439 | /* TODO: Add support for non extent hole punching */ | ||
4440 | return -ENOTSUPP; | ||
4441 | } | ||
4442 | |||
4443 | return ext4_ext_punch_hole(file, offset, length); | ||
4444 | } | ||
4445 | |||
4446 | /* | ||
4422 | * ext4_truncate() | 4447 | * ext4_truncate() |
4423 | * | 4448 | * |
4424 | * We block out ext4_get_block() block instantiations across the entire | 4449 | * We block out ext4_get_block() block instantiations across the entire |