1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-07-25 06:07:34 +00:00

Kernel: Move VirtIO code away from using a scatter gather list

Currently, when passing buffers into VirtIOQueues, we use scatter-gather
lists, which contain an internal vector of buffers. This vector is
allocated, filled and the destroy whenever we try to provide buffers
into a virtqueue, which would happen a lot in performance cricital code
(the main transport mechanism for certain paravirtualized devices).

This commit moves it over to using VirtIOQueueChains and building the
chain in place in the VirtIOQueue. Also included are a bunch of fixups
for the VirtIO Console device, making it use an internal VM::RingBuffer
instead.
This commit is contained in:
Sahan Fernando 2021-04-24 11:05:51 +10:00 committed by Andreas Kling
parent 13d5cdcd08
commit ed0e7b53a5
11 changed files with 312 additions and 159 deletions

View file

@ -19,40 +19,4 @@ ScatterGatherList::ScatterGatherList(AsyncBlockDeviceRequest& request, NonnullRe
m_dma_region = MM.allocate_kernel_region_with_vmobject(m_vm_object, page_round_up((request.block_count() * device_block_size)), "AHCI Scattered DMA", Region::Access::Read | Region::Access::Write, Region::Cacheable::Yes);
}
ScatterGatherRefList ScatterGatherRefList::create_from_buffer(const u8* buffer, size_t size)
{
VERIFY(buffer && size);
ScatterGatherRefList new_list;
auto* region = MM.find_region_from_vaddr(VirtualAddress(buffer));
VERIFY(region);
while (size > 0) {
size_t offset_in_page = (VirtualAddress(buffer) - region->vaddr()).get() % PAGE_SIZE;
size_t size_in_page = min(PAGE_SIZE - offset_in_page, size);
VERIFY(offset_in_page + size_in_page - 1 <= PAGE_SIZE);
new_list.add_entry(region->physical_page(region->page_index_from_address(VirtualAddress(buffer)))->paddr().get(), offset_in_page, size_in_page);
size -= size_in_page;
buffer += size_in_page;
}
return new_list;
}
ScatterGatherRefList ScatterGatherRefList::create_from_physical(PhysicalAddress paddr, size_t size)
{
VERIFY(!paddr.is_null() && size);
ScatterGatherRefList new_list;
new_list.add_entry(paddr.page_base().get(), paddr.offset_in_page(), size);
return new_list;
}
void ScatterGatherRefList::add_entry(FlatPtr addr, size_t offset, size_t size)
{
m_entries.append({ addr, offset, size });
}
void ScatterGatherRefList::for_each_entry(Function<void(const FlatPtr, const size_t)> callback) const
{
for (auto& entry : m_entries)
callback(entry.page_base + entry.offset, entry.length);
}
}