提交 62a2744c 编写于 作者: S Sheng Yang 提交者: Marcelo Tosatti

kvm: Flush coalesced MMIO buffer periodly

The default action of coalesced MMIO is, cache the writing in buffer, until:
1. The buffer is full.
2. Or the exit to QEmu due to other reasons.

But this would result in a very late writing in some condition.
1. The each time write to MMIO content is small.
2. The writing interval is big.
3. No need for input or accessing other devices frequently.

This issue was observed in a experimental embbed system. The test image
simply print "test" every 1 seconds. The output in QEmu meets expectation,
but the output in KVM is delayed for seconds.

Per Avi's suggestion, I hooked flushing coalesced MMIO buffer in VGA update
handler. By this way, We don't need vcpu explicit exit to QEmu to
handle this issue.
Signed-off-by: NSheng Yang <sheng@linux.intel.com>
Signed-off-by: NMarcelo Tosatti <mtosatti@redhat.com>
上级 117f8eb8
...@@ -915,6 +915,8 @@ void qemu_register_coalesced_mmio(target_phys_addr_t addr, ram_addr_t size); ...@@ -915,6 +915,8 @@ void qemu_register_coalesced_mmio(target_phys_addr_t addr, ram_addr_t size);
void qemu_unregister_coalesced_mmio(target_phys_addr_t addr, ram_addr_t size); void qemu_unregister_coalesced_mmio(target_phys_addr_t addr, ram_addr_t size);
void qemu_flush_coalesced_mmio_buffer(void);
/*******************************************/ /*******************************************/
/* host CPU ticks (if available) */ /* host CPU ticks (if available) */
......
...@@ -2406,6 +2406,12 @@ void qemu_unregister_coalesced_mmio(target_phys_addr_t addr, ram_addr_t size) ...@@ -2406,6 +2406,12 @@ void qemu_unregister_coalesced_mmio(target_phys_addr_t addr, ram_addr_t size)
kvm_uncoalesce_mmio_region(addr, size); kvm_uncoalesce_mmio_region(addr, size);
} }
void qemu_flush_coalesced_mmio_buffer(void)
{
if (kvm_enabled())
kvm_flush_coalesced_mmio_buffer();
}
ram_addr_t qemu_ram_alloc(ram_addr_t size) ram_addr_t qemu_ram_alloc(ram_addr_t size)
{ {
RAMBlock *new_block; RAMBlock *new_block;
......
...@@ -59,6 +59,9 @@ struct KVMState ...@@ -59,6 +59,9 @@ struct KVMState
int vmfd; int vmfd;
int regs_modified; int regs_modified;
int coalesced_mmio; int coalesced_mmio;
#ifdef KVM_CAP_COALESCED_MMIO
struct kvm_coalesced_mmio_ring *coalesced_mmio_ring;
#endif
int broken_set_mem_region; int broken_set_mem_region;
int migration_log; int migration_log;
int vcpu_events; int vcpu_events;
...@@ -200,6 +203,12 @@ int kvm_init_vcpu(CPUState *env) ...@@ -200,6 +203,12 @@ int kvm_init_vcpu(CPUState *env)
goto err; goto err;
} }
#ifdef KVM_CAP_COALESCED_MMIO
if (s->coalesced_mmio && !s->coalesced_mmio_ring)
s->coalesced_mmio_ring = (void *) env->kvm_run +
s->coalesced_mmio * PAGE_SIZE;
#endif
ret = kvm_arch_init_vcpu(env); ret = kvm_arch_init_vcpu(env);
if (ret == 0) { if (ret == 0) {
qemu_register_reset(kvm_reset_vcpu, env); qemu_register_reset(kvm_reset_vcpu, env);
...@@ -466,10 +475,10 @@ int kvm_init(int smp_cpus) ...@@ -466,10 +475,10 @@ int kvm_init(int smp_cpus)
goto err; goto err;
} }
s->coalesced_mmio = 0;
#ifdef KVM_CAP_COALESCED_MMIO #ifdef KVM_CAP_COALESCED_MMIO
s->coalesced_mmio = kvm_check_extension(s, KVM_CAP_COALESCED_MMIO); s->coalesced_mmio = kvm_check_extension(s, KVM_CAP_COALESCED_MMIO);
#else s->coalesced_mmio_ring = NULL;
s->coalesced_mmio = 0;
#endif #endif
s->broken_set_mem_region = 1; s->broken_set_mem_region = 1;
...@@ -544,14 +553,12 @@ static int kvm_handle_io(uint16_t port, void *data, int direction, int size, ...@@ -544,14 +553,12 @@ static int kvm_handle_io(uint16_t port, void *data, int direction, int size,
return 1; return 1;
} }
static void kvm_run_coalesced_mmio(CPUState *env, struct kvm_run *run) void kvm_flush_coalesced_mmio_buffer(void)
{ {
#ifdef KVM_CAP_COALESCED_MMIO #ifdef KVM_CAP_COALESCED_MMIO
KVMState *s = kvm_state; KVMState *s = kvm_state;
if (s->coalesced_mmio) { if (s->coalesced_mmio_ring) {
struct kvm_coalesced_mmio_ring *ring; struct kvm_coalesced_mmio_ring *ring = s->coalesced_mmio_ring;
ring = (void *)run + (s->coalesced_mmio * TARGET_PAGE_SIZE);
while (ring->first != ring->last) { while (ring->first != ring->last) {
struct kvm_coalesced_mmio *ent; struct kvm_coalesced_mmio *ent;
...@@ -609,7 +616,7 @@ int kvm_cpu_exec(CPUState *env) ...@@ -609,7 +616,7 @@ int kvm_cpu_exec(CPUState *env)
abort(); abort();
} }
kvm_run_coalesced_mmio(env, run); kvm_flush_coalesced_mmio_buffer();
ret = 0; /* exit loop */ ret = 0; /* exit loop */
switch (run->exit_reason) { switch (run->exit_reason) {
......
...@@ -53,6 +53,7 @@ void kvm_setup_guest_memory(void *start, size_t size); ...@@ -53,6 +53,7 @@ void kvm_setup_guest_memory(void *start, size_t size);
int kvm_coalesce_mmio_region(target_phys_addr_t start, ram_addr_t size); int kvm_coalesce_mmio_region(target_phys_addr_t start, ram_addr_t size);
int kvm_uncoalesce_mmio_region(target_phys_addr_t start, ram_addr_t size); int kvm_uncoalesce_mmio_region(target_phys_addr_t start, ram_addr_t size);
void kvm_flush_coalesced_mmio_buffer(void);
int kvm_insert_breakpoint(CPUState *current_env, target_ulong addr, int kvm_insert_breakpoint(CPUState *current_env, target_ulong addr,
target_ulong len, int type); target_ulong len, int type);
......
...@@ -2996,6 +2996,7 @@ static void gui_update(void *opaque) ...@@ -2996,6 +2996,7 @@ static void gui_update(void *opaque)
DisplayState *ds = opaque; DisplayState *ds = opaque;
DisplayChangeListener *dcl = ds->listeners; DisplayChangeListener *dcl = ds->listeners;
qemu_flush_coalesced_mmio_buffer();
dpy_refresh(ds); dpy_refresh(ds);
while (dcl != NULL) { while (dcl != NULL) {
...@@ -3011,6 +3012,7 @@ static void nographic_update(void *opaque) ...@@ -3011,6 +3012,7 @@ static void nographic_update(void *opaque)
{ {
uint64_t interval = GUI_REFRESH_INTERVAL; uint64_t interval = GUI_REFRESH_INTERVAL;
qemu_flush_coalesced_mmio_buffer();
qemu_mod_timer(nographic_timer, interval + qemu_get_clock(rt_clock)); qemu_mod_timer(nographic_timer, interval + qemu_get_clock(rt_clock));
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册