Luna/luna/src/Bitmap.cpp

135 lines
3.3 KiB
C++
Raw Normal View History

#include <luna/Bitmap.h>
#include <luna/CString.h>
#include <luna/Check.h>
Bitmap::Bitmap()
{
}
Bitmap::Bitmap(void* location, usize size_in_bytes) : m_location((u8*)location), m_size_in_bytes(size_in_bytes)
{
}
void Bitmap::initialize(void* location, usize size_in_bytes)
{
m_location = (u8*)location;
m_size_in_bytes = size_in_bytes;
}
void* Bitmap::move(void* new_location, usize new_location_size_in_bytes)
{
expect(initialized(), "Bitmap was never initialized");
if (new_location_size_in_bytes > m_size_in_bytes) memcpy(new_location, m_location, m_size_in_bytes);
else
memcpy(new_location, m_location, new_location_size_in_bytes);
void* old_location = (void*)m_location;
m_location = (u8*)new_location;
m_size_in_bytes = new_location_size_in_bytes;
return old_location;
}
void Bitmap::set(usize index, bool value)
{
expect(initialized(), "Bitmap was never initialized");
expect(index < size(), "Bitmap access out of range");
u64 byte_index = index / 8;
u8 bit_mask = (u8)(0b10000000 >> (index % 8));
m_location[byte_index] &= (u8)(~bit_mask);
if (value) { m_location[byte_index] |= bit_mask; }
}
bool Bitmap::get(usize index) const
{
expect(initialized(), "Bitmap was never initialized");
expect(index < size(), "Bitmap access out of range");
usize byte_index = index / 8;
u8 bit_mask = (u8)(0b10000000 >> (index % 8));
return (m_location[byte_index] & bit_mask) > 0;
}
void Bitmap::clear(bool value)
{
expect(initialized(), "Bitmap was never initialized");
2022-12-05 11:53:16 +00:00
memset(m_location, value_byte(value), m_size_in_bytes);
}
void Bitmap::clear_region(usize start, usize bits, bool value)
{
expect(initialized(), "Bitmap was never initialized");
expect((start + bits) <= size(), "Bitmap clear out of range");
2022-12-07 13:32:41 +00:00
if (!bits) return;
// Set individual bits while not on a byte boundary.
while ((start % 8) && bits)
{
set(start, value);
start++;
bits--;
}
// Clear out the rest in bytes.
usize bytes = bits / 8;
2022-12-05 11:53:16 +00:00
memset(&m_location[start / 8], value_byte(value), bytes);
start += bytes * 8;
bits -= bytes * 8;
// Set the remaining individual bits.
while (bits--)
{
set(start, value);
start++;
}
2023-01-02 12:07:29 +00:00
}
Option<usize> Bitmap::find(bool value, usize begin) const
{
expect(initialized(), "Bitmap was never initialized");
usize size = this->size();
expect(begin < size, "Start index out of range");
while (begin % 8)
{
if (get(begin) == value) return begin;
begin++;
}
if (begin == size) return {};
usize i = begin / 8;
u8 byte_that_does_not_contain_value = value_byte(!value);
while (i < m_size_in_bytes)
{
if (m_location[i] == byte_that_does_not_contain_value)
{
i++;
continue;
}
usize index = i * 8;
for (usize j = 0; j < 8; j++, index++)
{
if (get(index) == value) return index;
}
// Once we've located a byte that contains the value, we should succeed in finding it.
unreachable();
}
return {};
}
Option<usize> Bitmap::find_and_toggle(bool value, usize begin)
{
usize index = TRY(find(value, begin));
set(index, !value);
return index;
}