Commit 81056dd0 authored by Jan Kara's avatar Jan Kara

udf: Perform preallocation only for regular files

So far we preallocated blocks also for directories but that brings a
problem, when to get rid of preallocated blocks we don't need. So far
we removed them in udf_clear_inode() which has a disadvantage that
1) blocks are unavailable long after writing to a directory finished
   and thus one can get out of space unnecessarily early
2) releasing blocks from udf_clear_inode is problematic because VFS
   does not expect us to redirty inode there and it also slows down
   memory reclaim.

So preallocate blocks only for regular files where we can drop preallocation
in udf_release_file.
Signed-off-by: default avatarJan Kara <jack@suse.cz>
parent 7c6e3d1a
...@@ -90,19 +90,16 @@ void udf_delete_inode(struct inode *inode) ...@@ -90,19 +90,16 @@ void udf_delete_inode(struct inode *inode)
} }
/* /*
* If we are going to release inode from memory, we discard preallocation and * If we are going to release inode from memory, we truncate last inode extent
* truncate last inode extent to proper length. We could use drop_inode() but * to proper length. We could use drop_inode() but it's called under inode_lock
* it's called under inode_lock and thus we cannot mark inode dirty there. We * and thus we cannot mark inode dirty there. We use clear_inode() but we have
* use clear_inode() but we have to make sure to write inode as it's not written * to make sure to write inode as it's not written automatically.
* automatically.
*/ */
void udf_clear_inode(struct inode *inode) void udf_clear_inode(struct inode *inode)
{ {
struct udf_inode_info *iinfo; struct udf_inode_info *iinfo;
if (!(inode->i_sb->s_flags & MS_RDONLY)) { if (!(inode->i_sb->s_flags & MS_RDONLY)) {
lock_kernel(); lock_kernel();
/* Discard preallocation for directories, symlinks, etc. */
udf_discard_prealloc(inode);
udf_truncate_tail_extent(inode); udf_truncate_tail_extent(inode);
unlock_kernel(); unlock_kernel();
write_inode_now(inode, 0); write_inode_now(inode, 0);
...@@ -664,8 +661,12 @@ static struct buffer_head *inode_getblk(struct inode *inode, sector_t block, ...@@ -664,8 +661,12 @@ static struct buffer_head *inode_getblk(struct inode *inode, sector_t block,
udf_split_extents(inode, &c, offset, newblocknum, laarr, &endnum); udf_split_extents(inode, &c, offset, newblocknum, laarr, &endnum);
#ifdef UDF_PREALLOCATE #ifdef UDF_PREALLOCATE
/* preallocate blocks */ /* We preallocate blocks only for regular files. It also makes sense
udf_prealloc_extents(inode, c, lastblock, laarr, &endnum); * for directories but there's a problem when to drop the
* preallocation. We might use some delayed work for that but I feel
* it's overengineering for a filesystem like UDF. */
if (S_ISREG(inode->i_mode))
udf_prealloc_extents(inode, c, lastblock, laarr, &endnum);
#endif #endif
/* merge any continuous blocks in laarr */ /* merge any continuous blocks in laarr */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment