mufs: properly free intermediate pages when empty in mufs_do_free_page()

tags/0.1.0
S.J.R. van Schaik 8 years ago
parent 37a25d7384
commit 2a62bd1f15
  1. 44
      source/fs/mufs/tree.c

@ -109,47 +109,61 @@ int mufs_alloc_page(struct mufs_tree *tree, uint32_t *page,
return mufs_do_lookup(fs, page, tree->root, tree->depth, va, 0, 1);
}
static int mufs_do_free_page(struct mufs *fs, uint32_t base,
static int mufs_do_free_page(struct mufs *fs, uint32_t *base,
uint8_t depth, uint32_t va, int free_page)
{
struct flash_dev *dev = fs->dev;
uint32_t entry;
size_t index;
uint32_t offset, entry;
if (!base)
if (!base || *base == 0)
return -1;
if (depth == 0) {
if (free_page)
mufs_free_block(fs, base);
if (free_page) {
mufs_free_block(fs, *base);
*base = 0;
}
return 0;
}
base <<= dev->log2_block_size;
index = mufs_get_index(fs, va, depth);
base += index * sizeof entry;
offset = mufs_get_index(fs, va, depth) * sizeof entry;
offset += (*base << dev->log2_block_size);
if (flash_read(fs->dev, offset, &entry, sizeof entry) == 0)
return -1;
if (flash_read(fs->dev, base, &entry, sizeof entry) == 0)
if (mufs_do_free_page(fs, &entry, depth - 1, va, free_page) < 0)
return -1;
if (mufs_do_free_page(fs, entry, depth - 1, va, free_page) < 0)
if (entry != 0)
return 0;
if (flash_write0(fs->dev, offset, sizeof entry) == 0)
return -1;
if (flash_write0(fs->dev, base, sizeof entry) == 0)
if (flash_sync(fs->dev) == 0)
return -1;
return flash_sync(fs->dev);
if (!flash_is_erased(fs->dev, *base, 1))
return 0;
if (mufs_free_block(fs, *base) < 0)
return -1;
*base = 0;
return 0;
}
void mufs_unmap_page(struct mufs_tree *tree, uint32_t va)
{
mufs_do_free_page(tree->fs, tree->root, tree->depth, va, 0);
mufs_do_free_page(tree->fs, &tree->root, tree->depth, va, 0);
}
void mufs_free_page(struct mufs_tree *tree, uint32_t va)
{
mufs_do_free_page(tree->fs, tree->root, tree->depth, va, 1);
mufs_do_free_page(tree->fs, &tree->root, tree->depth, va, 1);
}
int mufs_extend_tree(struct mufs_tree *tree, uint8_t depth)

Loading…
Cancel
Save