forked from Minki/linux
244 lines
5.8 KiB
C
244 lines
5.8 KiB
C
|
/*
|
||
|
* linux/fs/hfs/bitmap.c
|
||
|
*
|
||
|
* Copyright (C) 1996-1997 Paul H. Hargrove
|
||
|
* (C) 2003 Ardis Technologies <roman@ardistech.com>
|
||
|
* This file may be distributed under the terms of the GNU General Public License.
|
||
|
*
|
||
|
* Based on GPLed code Copyright (C) 1995 Michael Dreher
|
||
|
*
|
||
|
* This file contains the code to modify the volume bitmap:
|
||
|
* search/set/clear bits.
|
||
|
*/
|
||
|
|
||
|
#include "hfs_fs.h"
|
||
|
|
||
|
/*
|
||
|
* hfs_find_zero_bit()
|
||
|
*
|
||
|
* Description:
|
||
|
* Given a block of memory, its length in bits, and a starting bit number,
|
||
|
* determine the number of the first zero bits (in left-to-right ordering)
|
||
|
* in that range.
|
||
|
*
|
||
|
* Returns >= 'size' if no zero bits are found in the range.
|
||
|
*
|
||
|
* Accesses memory in 32-bit aligned chunks of 32-bits and thus
|
||
|
* may read beyond the 'size'th bit.
|
||
|
*/
|
||
|
static u32 hfs_find_set_zero_bits(__be32 *bitmap, u32 size, u32 offset, u32 *max)
|
||
|
{
|
||
|
__be32 *curr, *end;
|
||
|
u32 mask, start, len, n;
|
||
|
__be32 val;
|
||
|
int i;
|
||
|
|
||
|
len = *max;
|
||
|
if (!len)
|
||
|
return size;
|
||
|
|
||
|
curr = bitmap + (offset / 32);
|
||
|
end = bitmap + ((size + 31) / 32);
|
||
|
|
||
|
/* scan the first partial u32 for zero bits */
|
||
|
val = *curr;
|
||
|
if (~val) {
|
||
|
n = be32_to_cpu(val);
|
||
|
i = offset % 32;
|
||
|
mask = (1U << 31) >> i;
|
||
|
for (; i < 32; mask >>= 1, i++) {
|
||
|
if (!(n & mask))
|
||
|
goto found;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* scan complete u32s for the first zero bit */
|
||
|
while (++curr < end) {
|
||
|
val = *curr;
|
||
|
if (~val) {
|
||
|
n = be32_to_cpu(val);
|
||
|
mask = 1 << 31;
|
||
|
for (i = 0; i < 32; mask >>= 1, i++) {
|
||
|
if (!(n & mask))
|
||
|
goto found;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return size;
|
||
|
|
||
|
found:
|
||
|
start = (curr - bitmap) * 32 + i;
|
||
|
if (start >= size)
|
||
|
return start;
|
||
|
/* do any partial u32 at the start */
|
||
|
len = min(size - start, len);
|
||
|
while (1) {
|
||
|
n |= mask;
|
||
|
if (++i >= 32)
|
||
|
break;
|
||
|
mask >>= 1;
|
||
|
if (!--len || n & mask)
|
||
|
goto done;
|
||
|
}
|
||
|
if (!--len)
|
||
|
goto done;
|
||
|
*curr++ = cpu_to_be32(n);
|
||
|
/* do full u32s */
|
||
|
while (1) {
|
||
|
n = be32_to_cpu(*curr);
|
||
|
if (len < 32)
|
||
|
break;
|
||
|
if (n) {
|
||
|
len = 32;
|
||
|
break;
|
||
|
}
|
||
|
*curr++ = cpu_to_be32(0xffffffff);
|
||
|
len -= 32;
|
||
|
}
|
||
|
/* do any partial u32 at end */
|
||
|
mask = 1U << 31;
|
||
|
for (i = 0; i < len; i++) {
|
||
|
if (n & mask)
|
||
|
break;
|
||
|
n |= mask;
|
||
|
mask >>= 1;
|
||
|
}
|
||
|
done:
|
||
|
*curr = cpu_to_be32(n);
|
||
|
*max = (curr - bitmap) * 32 + i - start;
|
||
|
return start;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* hfs_vbm_search_free()
|
||
|
*
|
||
|
* Description:
|
||
|
* Search for 'num_bits' consecutive cleared bits in the bitmap blocks of
|
||
|
* the hfs MDB. 'mdb' had better be locked or the returned range
|
||
|
* may be no longer free, when this functions returns!
|
||
|
* XXX Currently the search starts from bit 0, but it should start with
|
||
|
* the bit number stored in 's_alloc_ptr' of the MDB.
|
||
|
* Input Variable(s):
|
||
|
* struct hfs_mdb *mdb: Pointer to the hfs MDB
|
||
|
* u16 *num_bits: Pointer to the number of cleared bits
|
||
|
* to search for
|
||
|
* Output Variable(s):
|
||
|
* u16 *num_bits: The number of consecutive clear bits of the
|
||
|
* returned range. If the bitmap is fragmented, this will be less than
|
||
|
* requested and it will be zero, when the disk is full.
|
||
|
* Returns:
|
||
|
* The number of the first bit of the range of cleared bits which has been
|
||
|
* found. When 'num_bits' is zero, this is invalid!
|
||
|
* Preconditions:
|
||
|
* 'mdb' points to a "valid" (struct hfs_mdb).
|
||
|
* 'num_bits' points to a variable of type (u16), which contains
|
||
|
* the number of cleared bits to find.
|
||
|
* Postconditions:
|
||
|
* 'num_bits' is set to the length of the found sequence.
|
||
|
*/
|
||
|
u32 hfs_vbm_search_free(struct super_block *sb, u32 goal, u32 *num_bits)
|
||
|
{
|
||
|
void *bitmap;
|
||
|
u32 pos;
|
||
|
|
||
|
/* make sure we have actual work to perform */
|
||
|
if (!*num_bits)
|
||
|
return 0;
|
||
|
|
||
|
down(&HFS_SB(sb)->bitmap_lock);
|
||
|
bitmap = HFS_SB(sb)->bitmap;
|
||
|
|
||
|
pos = hfs_find_set_zero_bits(bitmap, HFS_SB(sb)->fs_ablocks, goal, num_bits);
|
||
|
if (pos >= HFS_SB(sb)->fs_ablocks) {
|
||
|
if (goal)
|
||
|
pos = hfs_find_set_zero_bits(bitmap, goal, 0, num_bits);
|
||
|
if (pos >= HFS_SB(sb)->fs_ablocks) {
|
||
|
*num_bits = pos = 0;
|
||
|
goto out;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dprint(DBG_BITMAP, "alloc_bits: %u,%u\n", pos, *num_bits);
|
||
|
HFS_SB(sb)->free_ablocks -= *num_bits;
|
||
|
hfs_bitmap_dirty(sb);
|
||
|
out:
|
||
|
up(&HFS_SB(sb)->bitmap_lock);
|
||
|
return pos;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* hfs_clear_vbm_bits()
|
||
|
*
|
||
|
* Description:
|
||
|
* Clear the requested bits in the volume bitmap of the hfs filesystem
|
||
|
* Input Variable(s):
|
||
|
* struct hfs_mdb *mdb: Pointer to the hfs MDB
|
||
|
* u16 start: The offset of the first bit
|
||
|
* u16 count: The number of bits
|
||
|
* Output Variable(s):
|
||
|
* None
|
||
|
* Returns:
|
||
|
* 0: no error
|
||
|
* -1: One of the bits was already clear. This is a strange
|
||
|
* error and when it happens, the filesystem must be repaired!
|
||
|
* -2: One or more of the bits are out of range of the bitmap.
|
||
|
* Preconditions:
|
||
|
* 'mdb' points to a "valid" (struct hfs_mdb).
|
||
|
* Postconditions:
|
||
|
* Starting with bit number 'start', 'count' bits in the volume bitmap
|
||
|
* are cleared. The affected bitmap blocks are marked "dirty", the free
|
||
|
* block count of the MDB is updated and the MDB is marked dirty.
|
||
|
*/
|
||
|
int hfs_clear_vbm_bits(struct super_block *sb, u16 start, u16 count)
|
||
|
{
|
||
|
__be32 *curr;
|
||
|
u32 mask;
|
||
|
int i, len;
|
||
|
|
||
|
/* is there any actual work to be done? */
|
||
|
if (!count)
|
||
|
return 0;
|
||
|
|
||
|
dprint(DBG_BITMAP, "clear_bits: %u,%u\n", start, count);
|
||
|
/* are all of the bits in range? */
|
||
|
if ((start + count) > HFS_SB(sb)->fs_ablocks)
|
||
|
return -2;
|
||
|
|
||
|
down(&HFS_SB(sb)->bitmap_lock);
|
||
|
/* bitmap is always on a 32-bit boundary */
|
||
|
curr = HFS_SB(sb)->bitmap + (start / 32);
|
||
|
len = count;
|
||
|
|
||
|
/* do any partial u32 at the start */
|
||
|
i = start % 32;
|
||
|
if (i) {
|
||
|
int j = 32 - i;
|
||
|
mask = 0xffffffffU << j;
|
||
|
if (j > count) {
|
||
|
mask |= 0xffffffffU >> (i + count);
|
||
|
*curr &= cpu_to_be32(mask);
|
||
|
goto out;
|
||
|
}
|
||
|
*curr++ &= cpu_to_be32(mask);
|
||
|
count -= j;
|
||
|
}
|
||
|
|
||
|
/* do full u32s */
|
||
|
while (count >= 32) {
|
||
|
*curr++ = 0;
|
||
|
count -= 32;
|
||
|
}
|
||
|
/* do any partial u32 at end */
|
||
|
if (count) {
|
||
|
mask = 0xffffffffU >> count;
|
||
|
*curr &= cpu_to_be32(mask);
|
||
|
}
|
||
|
out:
|
||
|
HFS_SB(sb)->free_ablocks += len;
|
||
|
up(&HFS_SB(sb)->bitmap_lock);
|
||
|
hfs_bitmap_dirty(sb);
|
||
|
|
||
|
return 0;
|
||
|
}
|