memory: fix usage of find_next_bit and find_next_zero_bit

The last two arguments to these functions are the last and first bit to
check relative to the base. The code was using incorrectly the first
bit and the number of bits. Fix this in cpu_physical_memory_get_dirty
and cpu_physical_memory_all_dirty. This requires a few changes in the
iteration; change the code in cpu_physical_memory_set_dirty_range to
match.

Backports commit 88c73d16ad1b6c22a2ab082064d0d521f756296a from qemu
This commit is contained in:
Paolo Bonzini 2018-02-22 19:51:24 -05:00 committed by Lioncash
parent 171d267209
commit 9479199c6b
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7

View file

@ -76,6 +76,7 @@ static inline bool cpu_physical_memory_get_dirty(struct uc_struct *uc, ram_addr_
{ {
DirtyMemoryBlocks *blocks; DirtyMemoryBlocks *blocks;
unsigned long end, page; unsigned long end, page;
unsigned long idx, offset, base;
bool dirty = false; bool dirty = false;
assert(client < DIRTY_MEMORY_NUM); assert(client < DIRTY_MEMORY_NUM);
@ -89,17 +90,22 @@ static inline bool cpu_physical_memory_get_dirty(struct uc_struct *uc, ram_addr_
// Unicorn: atomic_read used instead of atomic_rcu_read // Unicorn: atomic_read used instead of atomic_rcu_read
blocks = atomic_read(&uc->ram_list.dirty_memory[client]); blocks = atomic_read(&uc->ram_list.dirty_memory[client]);
idx = page / DIRTY_MEMORY_BLOCK_SIZE;
offset = page % DIRTY_MEMORY_BLOCK_SIZE;
base = page - offset;
while (page < end) { while (page < end) {
unsigned long idx = page / DIRTY_MEMORY_BLOCK_SIZE; unsigned long next = MIN(end, base + DIRTY_MEMORY_BLOCK_SIZE);
unsigned long offset = page % DIRTY_MEMORY_BLOCK_SIZE; unsigned long num = next - base;
unsigned long num = MIN(end - page, DIRTY_MEMORY_BLOCK_SIZE - offset); unsigned long found = find_next_bit(blocks->blocks[idx], num, offset);
if (found < num) {
if (find_next_bit(blocks->blocks[idx], offset, num) < num) {
dirty = true; dirty = true;
break; break;
} }
page += num; page = next;
idx++;
offset = 0;
base += DIRTY_MEMORY_BLOCK_SIZE;
} }
// Unicorn: commented out // Unicorn: commented out
@ -115,6 +121,7 @@ static inline bool cpu_physical_memory_all_dirty(struct uc_struct *uc, ram_addr_
{ {
DirtyMemoryBlocks *blocks; DirtyMemoryBlocks *blocks;
unsigned long end, page; unsigned long end, page;
unsigned long idx, offset, base;
bool dirty = true; bool dirty = true;
assert(client < DIRTY_MEMORY_NUM); assert(client < DIRTY_MEMORY_NUM);
@ -128,17 +135,22 @@ static inline bool cpu_physical_memory_all_dirty(struct uc_struct *uc, ram_addr_
// Unicorn: atomic_read used instead of atomic_rcu_read // Unicorn: atomic_read used instead of atomic_rcu_read
blocks = atomic_read(&uc->ram_list.dirty_memory[client]); blocks = atomic_read(&uc->ram_list.dirty_memory[client]);
idx = page / DIRTY_MEMORY_BLOCK_SIZE;
offset = page % DIRTY_MEMORY_BLOCK_SIZE;
base = page - offset;
while (page < end) { while (page < end) {
unsigned long idx = page / DIRTY_MEMORY_BLOCK_SIZE; unsigned long next = MIN(end, base + DIRTY_MEMORY_BLOCK_SIZE);
unsigned long offset = page % DIRTY_MEMORY_BLOCK_SIZE; unsigned long num = next - base;
unsigned long num = MIN(end - page, DIRTY_MEMORY_BLOCK_SIZE - offset); unsigned long found = find_next_zero_bit(blocks->blocks[idx], num, offset);
if (found < num) {
if (find_next_zero_bit(blocks->blocks[idx], offset, num) < num) {
dirty = false; dirty = false;
break; break;
} }
page += num; page = next;
idx++;
offset = 0;
base += DIRTY_MEMORY_BLOCK_SIZE;
} }
// Unicorn: commented out // Unicorn: commented out
@ -200,6 +212,7 @@ static inline void cpu_physical_memory_set_dirty_range(struct uc_struct *uc, ram
{ {
DirtyMemoryBlocks *blocks[DIRTY_MEMORY_NUM]; DirtyMemoryBlocks *blocks[DIRTY_MEMORY_NUM];
unsigned long end, page; unsigned long end, page;
unsigned long idx, offset, base;
int i; int i;
if (!mask && !xen_enabled()) { if (!mask && !xen_enabled()) {
@ -217,17 +230,21 @@ static inline void cpu_physical_memory_set_dirty_range(struct uc_struct *uc, ram
blocks[i] = atomic_read(&uc->ram_list.dirty_memory[i]); blocks[i] = atomic_read(&uc->ram_list.dirty_memory[i]);
} }
idx = page / DIRTY_MEMORY_BLOCK_SIZE;
offset = page % DIRTY_MEMORY_BLOCK_SIZE;
base = page - offset;
while (page < end) { while (page < end) {
unsigned long idx = page / DIRTY_MEMORY_BLOCK_SIZE; unsigned long next = MIN(end, base + DIRTY_MEMORY_BLOCK_SIZE);
unsigned long offset = page % DIRTY_MEMORY_BLOCK_SIZE;
unsigned long num = MIN(end - page, DIRTY_MEMORY_BLOCK_SIZE - offset);
if (unlikely(mask & (1 << DIRTY_MEMORY_CODE))) { if (unlikely(mask & (1 << DIRTY_MEMORY_CODE))) {
bitmap_set_atomic(blocks[DIRTY_MEMORY_CODE]->blocks[idx], bitmap_set_atomic(blocks[DIRTY_MEMORY_CODE]->blocks[idx],
offset, num); offset, next - page);
} }
page += num; page = next;
idx++;
offset = 0;
base += DIRTY_MEMORY_BLOCK_SIZE;
} }
// Unicorn: commented out // Unicorn: commented out