Linux-2.6.12-rc2
Initial git repository build. I'm not bothering with the full history, even though we have it. We can create a separate "historical" git archive of that later if we want to, and in the meantime it's about 3.2GB when imported into git - space that would just make the early git days unnecessarily complicated, when we don't have a lot of good infrastructure for it. Let it rip!
This commit is contained in:
243
fs/hfs/bitmap.c
Normal file
243
fs/hfs/bitmap.c
Normal file
@@ -0,0 +1,243 @@
|
||||
/*
|
||||
* 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;
|
||||
}
|
Reference in New Issue
Block a user