Skip to content

Commit

Permalink
kvm: fix unaligned slots
Browse files Browse the repository at this point in the history
kvm_set_phys_mem() may be passed sections that are not aligned to a page
boundary.  The current code simply brute-forces the alignment which leads
to an inconsistency and an abort().

Fix by aligning the start and the end of the section correctly, discarding
and unaligned head or tail.

This was triggered by a guest sizing a 64-bit BAR that is smaller than a page
with PCI_COMMAND_MEMORY enabled and the upper dword clear.

Signed-off-by: Avi Kivity <[email protected]>
  • Loading branch information
avikivity committed Mar 1, 2012
1 parent 7c51c1a commit 8f6f962
Showing 1 changed file with 12 additions and 3 deletions.
15 changes: 12 additions & 3 deletions kvm-all.c
Original file line number Diff line number Diff line change
Expand Up @@ -541,17 +541,26 @@ static void kvm_set_phys_mem(MemoryRegionSection *section, bool add)
target_phys_addr_t start_addr = section->offset_within_address_space;
ram_addr_t size = section->size;
void *ram = NULL;
unsigned delta;

/* kvm works in page size chunks, but the function may be called
with sub-page size and unaligned start address. */
size = TARGET_PAGE_ALIGN(size);
start_addr = TARGET_PAGE_ALIGN(start_addr);
delta = TARGET_PAGE_ALIGN(size) - size;
if (delta > size) {
return;
}
start_addr += delta;
size -= delta;
size &= TARGET_PAGE_MASK;
if (!size || (start_addr & ~TARGET_PAGE_MASK)) {
return;
}

if (!memory_region_is_ram(mr)) {
return;
}

ram = memory_region_get_ram_ptr(mr) + section->offset_within_region;
ram = memory_region_get_ram_ptr(mr) + section->offset_within_region + delta;

while (1) {
mem = kvm_lookup_overlapping_slot(s, start_addr, start_addr + size);
Expand Down

0 comments on commit 8f6f962

Please sign in to comment.