Compare commits
No commits in common. "a1ab44d39fd14a79f9cf5e3bd3155aecabf28b9d" and "2d3810874d023e9f0cf41cfbdc0873c407b70224" have entirely different histories.
a1ab44d39f
...
2d3810874d
|
@ -32,6 +32,9 @@ namespace Kernel
|
||||||
|
|
||||||
virtual dev_t rdev() const override { return m_rdev; }
|
virtual dev_t rdev() const override { return m_rdev; }
|
||||||
|
|
||||||
|
virtual BAN::ErrorOr<size_t> read_impl(off_t, BAN::ByteSpan) override;
|
||||||
|
virtual BAN::ErrorOr<size_t> write_impl(off_t, BAN::ConstByteSpan) override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
ATABaseDevice();
|
ATABaseDevice();
|
||||||
BAN::ErrorOr<void> initialize(BAN::Span<const uint16_t> identify_data);
|
BAN::ErrorOr<void> initialize(BAN::Span<const uint16_t> identify_data);
|
||||||
|
|
|
@ -40,9 +40,6 @@ namespace Kernel
|
||||||
virtual BAN::ErrorOr<void> write_sectors_impl(uint64_t lba, uint64_t sector_count, BAN::ConstByteSpan) = 0;
|
virtual BAN::ErrorOr<void> write_sectors_impl(uint64_t lba, uint64_t sector_count, BAN::ConstByteSpan) = 0;
|
||||||
void add_disk_cache();
|
void add_disk_cache();
|
||||||
|
|
||||||
virtual BAN::ErrorOr<size_t> read_impl(off_t, BAN::ByteSpan) override;
|
|
||||||
virtual BAN::ErrorOr<size_t> write_impl(off_t, BAN::ConstByteSpan) override;
|
|
||||||
|
|
||||||
virtual bool can_read_impl() const override { return true; }
|
virtual bool can_read_impl() const override { return true; }
|
||||||
virtual bool can_write_impl() const override { return true; }
|
virtual bool can_write_impl() const override { return true; }
|
||||||
virtual bool has_error_impl() const override { return false; }
|
virtual bool has_error_impl() const override { return false; }
|
||||||
|
|
|
@ -77,6 +77,26 @@ namespace Kernel
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BAN::ErrorOr<size_t> detail::ATABaseDevice::read_impl(off_t offset, BAN::ByteSpan buffer)
|
||||||
|
{
|
||||||
|
if (offset % sector_size())
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
|
if (buffer.size() % sector_size())
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
|
TRY(read_sectors(offset / sector_size(), buffer.size() / sector_size(), buffer));
|
||||||
|
return buffer.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
BAN::ErrorOr<size_t> detail::ATABaseDevice::write_impl(off_t offset, BAN::ConstByteSpan buffer)
|
||||||
|
{
|
||||||
|
if (offset % sector_size())
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
|
if (buffer.size() % sector_size())
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
|
TRY(write_sectors(offset / sector_size(), buffer.size() / sector_size(), buffer));
|
||||||
|
return buffer.size();
|
||||||
|
}
|
||||||
|
|
||||||
BAN::ErrorOr<BAN::RefPtr<ATADevice>> ATADevice::create(BAN::RefPtr<ATABus> bus, ATABus::DeviceType type, bool is_secondary, BAN::Span<const uint16_t> identify_data)
|
BAN::ErrorOr<BAN::RefPtr<ATADevice>> ATADevice::create(BAN::RefPtr<ATABus> bus, ATABus::DeviceType type, bool is_secondary, BAN::Span<const uint16_t> identify_data)
|
||||||
{
|
{
|
||||||
auto* device_ptr = new ATADevice(bus, type, is_secondary);
|
auto* device_ptr = new ATADevice(bus, type, is_secondary);
|
||||||
|
|
|
@ -224,36 +224,20 @@ namespace Kernel
|
||||||
{
|
{
|
||||||
ASSERT(buffer.size() >= sector_count * sector_size());
|
ASSERT(buffer.size() >= sector_count * sector_size());
|
||||||
|
|
||||||
|
{
|
||||||
LockGuard _(m_mutex);
|
LockGuard _(m_mutex);
|
||||||
|
|
||||||
if (!m_disk_cache.has_value())
|
if (!m_disk_cache.has_value())
|
||||||
return read_sectors_impl(lba, sector_count, buffer);
|
return read_sectors_impl(lba, sector_count, buffer);
|
||||||
|
|
||||||
uint64_t sectors_done = 0;
|
|
||||||
while (sectors_done < sector_count)
|
|
||||||
{
|
|
||||||
const uint32_t segment_sector_count = BAN::Math::min<uint32_t>(sector_count - sectors_done, 64);
|
|
||||||
uint64_t needed_sector_bitmask = (static_cast<uint64_t>(1) << segment_sector_count) - 1;
|
|
||||||
for (uint32_t i = 0; i < segment_sector_count; i++)
|
|
||||||
if (m_disk_cache->read_from_cache(lba + sectors_done + i, buffer.slice((sectors_done + i) * sector_size(), sector_size())))
|
|
||||||
needed_sector_bitmask &= ~(static_cast<uint64_t>(1) << i);
|
|
||||||
|
|
||||||
for (uint32_t i = 0; i < segment_sector_count && needed_sector_bitmask; i++)
|
|
||||||
{
|
|
||||||
if (!(needed_sector_bitmask & (static_cast<uint64_t>(1) << i)))
|
|
||||||
continue;
|
|
||||||
uint32_t len = 1;
|
|
||||||
while (needed_sector_bitmask & (static_cast<uint64_t>(1) << (i + len)))
|
|
||||||
len++;
|
|
||||||
auto segment_buffer = buffer.slice((sectors_done + i) * sector_size(), len * sector_size());
|
|
||||||
TRY(read_sectors_impl(lba + sectors_done + i, len, segment_buffer));
|
|
||||||
for (uint32_t j = 0; j < len; j++)
|
|
||||||
(void)m_disk_cache->write_to_cache(lba + sectors_done + i + j, segment_buffer.slice(j * sector_size(), sector_size()), false);
|
|
||||||
needed_sector_bitmask &= ~(((static_cast<uint64_t>(1) << len) - 1) << i);
|
|
||||||
i += len;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sectors_done += segment_sector_count;
|
for (uint64_t offset = 0; offset < sector_count; offset++)
|
||||||
|
{
|
||||||
|
LockGuard _(m_mutex);
|
||||||
|
auto sector_buffer = buffer.slice(offset * sector_size(), sector_size());
|
||||||
|
if (m_disk_cache->read_from_cache(lba + offset, sector_buffer))
|
||||||
|
continue;
|
||||||
|
TRY(read_sectors_impl(lba + offset, 1, sector_buffer));
|
||||||
|
(void)m_disk_cache->write_to_cache(lba + offset, sector_buffer, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
|
@ -263,13 +247,15 @@ namespace Kernel
|
||||||
{
|
{
|
||||||
ASSERT(buffer.size() >= sector_count * sector_size());
|
ASSERT(buffer.size() >= sector_count * sector_size());
|
||||||
|
|
||||||
|
{
|
||||||
LockGuard _(m_mutex);
|
LockGuard _(m_mutex);
|
||||||
|
|
||||||
if (!m_disk_cache.has_value())
|
if (!m_disk_cache.has_value())
|
||||||
return write_sectors_impl(lba, sector_count, buffer);
|
return write_sectors_impl(lba, sector_count, buffer);
|
||||||
|
}
|
||||||
|
|
||||||
for (uint8_t offset = 0; offset < sector_count; offset++)
|
for (uint8_t offset = 0; offset < sector_count; offset++)
|
||||||
{
|
{
|
||||||
|
LockGuard _(m_mutex);
|
||||||
auto sector_buffer = buffer.slice(offset * sector_size(), sector_size());
|
auto sector_buffer = buffer.slice(offset * sector_size(), sector_size());
|
||||||
if (m_disk_cache->write_to_cache(lba + offset, sector_buffer, true).is_error())
|
if (m_disk_cache->write_to_cache(lba + offset, sector_buffer, true).is_error())
|
||||||
TRY(write_sectors_impl(lba + offset, 1, sector_buffer));
|
TRY(write_sectors_impl(lba + offset, 1, sector_buffer));
|
||||||
|
@ -278,24 +264,4 @@ namespace Kernel
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
BAN::ErrorOr<size_t> StorageDevice::read_impl(off_t offset, BAN::ByteSpan buffer)
|
|
||||||
{
|
|
||||||
if (offset % sector_size())
|
|
||||||
return BAN::Error::from_errno(EINVAL);
|
|
||||||
if (buffer.size() % sector_size())
|
|
||||||
return BAN::Error::from_errno(EINVAL);
|
|
||||||
TRY(read_sectors(offset / sector_size(), buffer.size() / sector_size(), buffer));
|
|
||||||
return buffer.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
BAN::ErrorOr<size_t> StorageDevice::write_impl(off_t offset, BAN::ConstByteSpan buffer)
|
|
||||||
{
|
|
||||||
if (offset % sector_size())
|
|
||||||
return BAN::Error::from_errno(EINVAL);
|
|
||||||
if (buffer.size() % sector_size())
|
|
||||||
return BAN::Error::from_errno(EINVAL);
|
|
||||||
TRY(write_sectors(offset / sector_size(), buffer.size() / sector_size(), buffer));
|
|
||||||
return buffer.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue