Commit 2cd282a1 authored by Sergei Antonov's avatar Sergei Antonov Committed by Linus Torvalds

hfsplus: fix "unused node is not erased" error

Zero newly allocated extents in the catalog tree if volume attributes
tell us to.  Not doing so we risk getting the "unused node is not
erased" error.  See kHFSUnusedNodeFix flag in Apple's source code for
reference.

There was a previous commit clearing the node when it is freed: commit
899bed05 ("hfsplus: fix issue with unzeroed unused b-tree nodes").
But it did not handle newly allocated extents (this patch fixes it).
And it zeroed nodes in all trees unconditionally which is an overkill.

This patch adds a condition and also switches to 'tree->node_size' as a
simpler method of getting the length to zero.
Signed-off-by: default avatarSergei Antonov <saproj@gmail.com>
Cc: Anton Altaparmakov <aia21@cam.ac.uk>
Cc: Al Viro <viro@zeniv.linux.org.uk>
Cc: Christoph Hellwig <hch@infradead.org>
Cc: Vyacheslav Dubeyko <slava@dubeyko.com>
Cc: Hin-Tak Leung <htl10@users.sourceforge.net>
Cc: Kyle Laracey <kalaracey@gmail.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 915ab236
...@@ -646,8 +646,8 @@ void hfs_bnode_put(struct hfs_bnode *node) ...@@ -646,8 +646,8 @@ void hfs_bnode_put(struct hfs_bnode *node)
if (test_bit(HFS_BNODE_DELETED, &node->flags)) { if (test_bit(HFS_BNODE_DELETED, &node->flags)) {
hfs_bnode_unhash(node); hfs_bnode_unhash(node);
spin_unlock(&tree->hash_lock); spin_unlock(&tree->hash_lock);
hfs_bnode_clear(node, 0, if (hfs_bnode_need_zeroout(tree))
PAGE_CACHE_SIZE * tree->pages_per_bnode); hfs_bnode_clear(node, 0, tree->node_size);
hfs_bmap_free(node); hfs_bmap_free(node);
hfs_bnode_free(node); hfs_bnode_free(node);
return; return;
...@@ -656,3 +656,16 @@ void hfs_bnode_put(struct hfs_bnode *node) ...@@ -656,3 +656,16 @@ void hfs_bnode_put(struct hfs_bnode *node)
} }
} }
/*
* Unused nodes have to be zeroed if this is the catalog tree and
* a corresponding flag in the volume header is set.
*/
bool hfs_bnode_need_zeroout(struct hfs_btree *tree)
{
struct super_block *sb = tree->inode->i_sb;
struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb);
const u32 volume_attr = be32_to_cpu(sbi->s_vhdr->attributes);
return tree->cnid == HFSPLUS_CAT_CNID &&
volume_attr & HFSPLUS_VOL_UNUSED_NODE_FIX;
}
...@@ -358,7 +358,7 @@ struct hfs_bnode *hfs_bmap_alloc(struct hfs_btree *tree) ...@@ -358,7 +358,7 @@ struct hfs_bnode *hfs_bmap_alloc(struct hfs_btree *tree)
u32 count; u32 count;
int res; int res;
res = hfsplus_file_extend(inode); res = hfsplus_file_extend(inode, hfs_bnode_need_zeroout(tree));
if (res) if (res)
return ERR_PTR(res); return ERR_PTR(res);
hip->phys_size = inode->i_size = hip->phys_size = inode->i_size =
......
...@@ -235,7 +235,7 @@ int hfsplus_get_block(struct inode *inode, sector_t iblock, ...@@ -235,7 +235,7 @@ int hfsplus_get_block(struct inode *inode, sector_t iblock,
if (iblock > hip->fs_blocks || !create) if (iblock > hip->fs_blocks || !create)
return -EIO; return -EIO;
if (ablock >= hip->alloc_blocks) { if (ablock >= hip->alloc_blocks) {
res = hfsplus_file_extend(inode); res = hfsplus_file_extend(inode, false);
if (res) if (res)
return res; return res;
} }
...@@ -425,7 +425,7 @@ int hfsplus_free_fork(struct super_block *sb, u32 cnid, ...@@ -425,7 +425,7 @@ int hfsplus_free_fork(struct super_block *sb, u32 cnid,
return res; return res;
} }
int hfsplus_file_extend(struct inode *inode) int hfsplus_file_extend(struct inode *inode, bool zeroout)
{ {
struct super_block *sb = inode->i_sb; struct super_block *sb = inode->i_sb;
struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb); struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb);
...@@ -463,6 +463,12 @@ int hfsplus_file_extend(struct inode *inode) ...@@ -463,6 +463,12 @@ int hfsplus_file_extend(struct inode *inode)
} }
} }
if (zeroout) {
res = sb_issue_zeroout(sb, start, len, GFP_NOFS);
if (res)
goto out;
}
hfs_dbg(EXTENT, "extend %lu: %u,%u\n", inode->i_ino, start, len); hfs_dbg(EXTENT, "extend %lu: %u,%u\n", inode->i_ino, start, len);
if (hip->alloc_blocks <= hip->first_blocks) { if (hip->alloc_blocks <= hip->first_blocks) {
......
...@@ -414,6 +414,7 @@ void hfs_bnode_free(struct hfs_bnode *); ...@@ -414,6 +414,7 @@ void hfs_bnode_free(struct hfs_bnode *);
struct hfs_bnode *hfs_bnode_create(struct hfs_btree *, u32); struct hfs_bnode *hfs_bnode_create(struct hfs_btree *, u32);
void hfs_bnode_get(struct hfs_bnode *); void hfs_bnode_get(struct hfs_bnode *);
void hfs_bnode_put(struct hfs_bnode *); void hfs_bnode_put(struct hfs_bnode *);
bool hfs_bnode_need_zeroout(struct hfs_btree *);
/* brec.c */ /* brec.c */
u16 hfs_brec_lenoff(struct hfs_bnode *, u16, u16 *); u16 hfs_brec_lenoff(struct hfs_bnode *, u16, u16 *);
...@@ -460,7 +461,7 @@ int hfsplus_ext_write_extent(struct inode *); ...@@ -460,7 +461,7 @@ int hfsplus_ext_write_extent(struct inode *);
int hfsplus_get_block(struct inode *, sector_t, struct buffer_head *, int); int hfsplus_get_block(struct inode *, sector_t, struct buffer_head *, int);
int hfsplus_free_fork(struct super_block *, u32, int hfsplus_free_fork(struct super_block *, u32,
struct hfsplus_fork_raw *, int); struct hfsplus_fork_raw *, int);
int hfsplus_file_extend(struct inode *); int hfsplus_file_extend(struct inode *, bool zeroout);
void hfsplus_file_truncate(struct inode *); void hfsplus_file_truncate(struct inode *);
/* inode.c */ /* inode.c */
......
...@@ -144,6 +144,7 @@ struct hfsplus_vh { ...@@ -144,6 +144,7 @@ struct hfsplus_vh {
#define HFSPLUS_VOL_NODEID_REUSED (1 << 12) #define HFSPLUS_VOL_NODEID_REUSED (1 << 12)
#define HFSPLUS_VOL_JOURNALED (1 << 13) #define HFSPLUS_VOL_JOURNALED (1 << 13)
#define HFSPLUS_VOL_SOFTLOCK (1 << 15) #define HFSPLUS_VOL_SOFTLOCK (1 << 15)
#define HFSPLUS_VOL_UNUSED_NODE_FIX (1 << 31)
/* HFS+ BTree node descriptor */ /* HFS+ BTree node descriptor */
struct hfs_bnode_desc { struct hfs_bnode_desc {
......
...@@ -196,7 +196,7 @@ static int hfsplus_create_attributes_file(struct super_block *sb) ...@@ -196,7 +196,7 @@ static int hfsplus_create_attributes_file(struct super_block *sb)
} }
while (hip->alloc_blocks < hip->clump_blocks) { while (hip->alloc_blocks < hip->clump_blocks) {
err = hfsplus_file_extend(attr_file); err = hfsplus_file_extend(attr_file, false);
if (unlikely(err)) { if (unlikely(err)) {
pr_err("failed to extend attributes file\n"); pr_err("failed to extend attributes file\n");
goto end_attr_file_creation; goto end_attr_file_creation;
......
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