/*
* extent.c --- ext2 extent abstraction
*
* This abstraction is used to provide a compact way of representing a
* translation table, for moving multiple contiguous ranges (extents)
* of blocks or inodes.
*
* Copyright (C) 1997, 1998 by Theodore Ts'o and
* PowerQuest, Inc.
*
* Copyright (C) 1999, 2000 by Theosore Ts'o
*
* %Begin-Header%
* This file may be redistributed under the terms of the GNU Public
* License.
* %End-Header%
*/
#include "resize2fs.h"
struct ext2_extent_entry {
__u64 old_loc, new_loc;
__u64 size;
};
struct _ext2_extent {
struct ext2_extent_entry *list;
__u64 cursor;
__u64 size;
__u64 num;
__u64 sorted;
};
/*
* Create an extent table
*/
errcode_t ext2fs_create_extent_table(ext2_extent *ret_extent, __u64 size)
{
ext2_extent extent;
errcode_t retval;
retval = ext2fs_get_mem(sizeof(struct _ext2_extent), &extent);
if (retval)
return retval;
memset(extent, 0, sizeof(struct _ext2_extent));
extent->size = size ? size : 50;
extent->cursor = 0;
extent->num = 0;
extent->sorted = 1;
retval = ext2fs_get_array(sizeof(struct ext2_extent_entry),
extent->size, &extent->list);
if (retval) {
ext2fs_free_mem(&extent);
return retval;
}
memset(extent->list, 0,
sizeof(struct ext2_extent_entry) * extent->size);
*ret_extent = extent;
return 0;
}
/*
* Free an extent table
*/
void ext2fs_free_extent_table(ext2_extent extent)
{
if (extent->list)
ext2fs_free_mem(&extent->list);
extent->list = 0;
extent->size = 0;
extent->num = 0;
ext2fs_free_mem(&extent);
}
/*
* Add an entry to the extent table
*/
errcode_t ext2fs_add_extent_entry(ext2_extent extent, __u64 old_loc, __u64 new_loc)
{
struct ext2_extent_entry *ent;
errcode_t retval;
__u64 newsize;
__u64 curr;
if (extent->num >= extent->size) {
newsize = extent->size + 100;
retval = ext2fs_resize_mem(sizeof(struct ext2_extent_entry) *
extent->size,
sizeof(struct ext2_extent_entry) *
newsize, &extent->list);
if (retval)
return retval;
extent->size = newsize;
}
curr = extent->num;
ent = extent->list + curr;
if (curr) {
/*
* Check to see if this can be coalesced with the last
* extent
*/
ent--;
if ((ent->old_loc + ent->size == old_loc) &&
(ent->new_loc + ent->size == new_loc)) {
ent->size++;
return 0;
}
/*
* Now see if we're going to ruin the sorting
*/
if (ent->old_loc + ent->size > old_loc)
extent->sorted = 0;
ent++;
}
ent->old_loc = old_loc;
ent->new_loc = new_loc;
ent->size = 1;
extent->num++;
return 0;
}
/*
* Helper function for qsort
*/
static EXT2_QSORT_TYPE extent_cmp(const void *a, const void *b)
{
const struct ext2_extent_entry *db_a;
const struct ext2_extent_entry *db_b;
db_a = (const struct ext2_extent_entry *) a;
db_b = (const struct ext2_extent_entry *) b;
return (db_a->old_loc - db_b->old_loc);
}
/*
* Given an inode map and inode number, look up the old inode number
* and return the new inode number.
*/
__u64 ext2fs_extent_translate(ext2_extent extent, __u64 old_loc)
{
__s64 low, high, mid;
__u64 lowval, highval;
float range;
if (!extent->sorted) {
qsort(extent->list, extent->num,
sizeof(struct ext2_extent_entry), extent_cmp);
extent->sorted = 1;
}
low = 0;
high = extent->num-1;
while (low <= high) {
#if 0
mid = (low+high)/2;
#else
if (low == high)
mid = low;
else {
/* Interpolate for efficiency */
lowval = extent->list[low].old_loc;
highval = extent->list[high].old_loc;
if (old_loc < lowval)
range = 0;
else if (old_loc > highval)
range = 1;
else {
range = ((float) (old_loc - lowval)) /
(highval - lowval);
if (range > 0.9)
range = 0.9;
if (range < 0.1)
range = 0.1;
}
mid = low + ((__u64) (range * (high-low)));
}
#endif
if ((old_loc >= extent->list[mid].old_loc) &&
(old_loc < extent->list[mid].old_loc + extent->list[mid].size))
return (extent->list[mid].new_loc +
(old_loc - extent->list[mid].old_loc));
if (old_loc < extent->list[mid].old_loc)
high = mid-1;
else
low = mid+1;
}
return 0;
}
/*
* For debugging only
*/
void ext2fs_extent_dump(ext2_extent extent, FILE *out)
{
__u64 i;
struct ext2_extent_entry *ent;
fputs(_("# Extent dump:\n"), out);
fprintf(out, _("#\tNum=%llu, Size=%llu, Cursor=%llu, Sorted=%llu\n"),
extent->num, extent->size, extent->cursor, extent->sorted);
for (i=0, ent=extent->list; i < extent->num; i++, ent++) {
fprintf(out, "#\t\t %llu -> %llu (%llu)\n", ent->old_loc,
ent->new_loc, ent->size);
}
}
/*
* Iterate over the contents of the extent table
*/
errcode_t ext2fs_iterate_extent(ext2_extent extent, __u64 *old_loc,
__u64 *new_loc, __u64 *size)
{
struct ext2_extent_entry *ent;
if (!old_loc) {
extent->cursor = 0;
return 0;
}
if (extent->cursor >= extent->num) {
*old_loc = 0;
*new_loc = 0;
*size = 0;
return 0;
}
ent = extent->list + extent->cursor++;
*old_loc = ent->old_loc;
*new_loc = ent->new_loc;
*size = ent->size;
return 0;
}