提交 7677e24f 编写于 作者: A Anthony Liguori

Merge remote-tracking branch 'qemu-kvm/uq/master' into staging

* qemu-kvm/uq/master:
  kvm: i8254: Fix conversion of in-kernel to userspace state
  kvm/apic: correct short memset
...@@ -30,7 +30,7 @@ void kvm_put_apic_state(DeviceState *d, struct kvm_lapic_state *kapic) ...@@ -30,7 +30,7 @@ void kvm_put_apic_state(DeviceState *d, struct kvm_lapic_state *kapic)
APICCommonState *s = DO_UPCAST(APICCommonState, busdev.qdev, d); APICCommonState *s = DO_UPCAST(APICCommonState, busdev.qdev, d);
int i; int i;
memset(kapic, 0, sizeof(kapic)); memset(kapic, 0, sizeof(*kapic));
kvm_apic_set_reg(kapic, 0x2, s->id << 24); kvm_apic_set_reg(kapic, 0x2, s->id << 24);
kvm_apic_set_reg(kapic, 0x8, s->tpr); kvm_apic_set_reg(kapic, 0x8, s->tpr);
kvm_apic_set_reg(kapic, 0xd, s->log_dest << 24); kvm_apic_set_reg(kapic, 0xd, s->log_dest << 24);
......
...@@ -23,31 +23,63 @@ ...@@ -23,31 +23,63 @@
* THE SOFTWARE. * THE SOFTWARE.
*/ */
#include "qemu-timer.h" #include "qemu-timer.h"
#include "sysemu.h"
#include "hw/i8254.h" #include "hw/i8254.h"
#include "hw/i8254_internal.h" #include "hw/i8254_internal.h"
#include "kvm.h" #include "kvm.h"
#define KVM_PIT_REINJECT_BIT 0 #define KVM_PIT_REINJECT_BIT 0
#define CALIBRATION_ROUNDS 3
typedef struct KVMPITState { typedef struct KVMPITState {
PITCommonState pit; PITCommonState pit;
LostTickPolicy lost_tick_policy; LostTickPolicy lost_tick_policy;
bool state_valid;
} KVMPITState; } KVMPITState;
static void kvm_pit_get(PITCommonState *s) static int64_t abs64(int64_t v)
{ {
return v < 0 ? -v : v;
}
static void kvm_pit_get(PITCommonState *pit)
{
KVMPITState *s = DO_UPCAST(KVMPITState, pit, pit);
struct kvm_pit_state2 kpit; struct kvm_pit_state2 kpit;
struct kvm_pit_channel_state *kchan; struct kvm_pit_channel_state *kchan;
struct PITChannelState *sc; struct PITChannelState *sc;
int64_t offset, clock_offset;
struct timespec ts;
int i, ret; int i, ret;
if (s->state_valid) {
return;
}
/*
* Measure the delta between CLOCK_MONOTONIC, the base used for
* kvm_pit_channel_state::count_load_time, and vm_clock. Take the
* minimum of several samples to filter out scheduling noise.
*/
clock_offset = INT64_MAX;
for (i = 0; i < CALIBRATION_ROUNDS; i++) {
offset = qemu_get_clock_ns(vm_clock);
clock_gettime(CLOCK_MONOTONIC, &ts);
offset -= ts.tv_nsec;
offset -= (int64_t)ts.tv_sec * 1000000000;
if (abs64(offset) < abs64(clock_offset)) {
clock_offset = offset;
}
}
if (kvm_has_pit_state2()) { if (kvm_has_pit_state2()) {
ret = kvm_vm_ioctl(kvm_state, KVM_GET_PIT2, &kpit); ret = kvm_vm_ioctl(kvm_state, KVM_GET_PIT2, &kpit);
if (ret < 0) { if (ret < 0) {
fprintf(stderr, "KVM_GET_PIT2 failed: %s\n", strerror(ret)); fprintf(stderr, "KVM_GET_PIT2 failed: %s\n", strerror(ret));
abort(); abort();
} }
s->channels[0].irq_disabled = kpit.flags & KVM_PIT_FLAGS_HPET_LEGACY; pit->channels[0].irq_disabled = kpit.flags & KVM_PIT_FLAGS_HPET_LEGACY;
} else { } else {
/* /*
* kvm_pit_state2 is superset of kvm_pit_state struct, * kvm_pit_state2 is superset of kvm_pit_state struct,
...@@ -61,7 +93,7 @@ static void kvm_pit_get(PITCommonState *s) ...@@ -61,7 +93,7 @@ static void kvm_pit_get(PITCommonState *s)
} }
for (i = 0; i < 3; i++) { for (i = 0; i < 3; i++) {
kchan = &kpit.channels[i]; kchan = &kpit.channels[i];
sc = &s->channels[i]; sc = &pit->channels[i];
sc->count = kchan->count; sc->count = kchan->count;
sc->latched_count = kchan->latched_count; sc->latched_count = kchan->latched_count;
sc->count_latched = kchan->count_latched; sc->count_latched = kchan->count_latched;
...@@ -74,10 +106,10 @@ static void kvm_pit_get(PITCommonState *s) ...@@ -74,10 +106,10 @@ static void kvm_pit_get(PITCommonState *s)
sc->mode = kchan->mode; sc->mode = kchan->mode;
sc->bcd = kchan->bcd; sc->bcd = kchan->bcd;
sc->gate = kchan->gate; sc->gate = kchan->gate;
sc->count_load_time = kchan->count_load_time; sc->count_load_time = kchan->count_load_time + clock_offset;
} }
sc = &s->channels[0]; sc = &pit->channels[0];
sc->next_transition_time = sc->next_transition_time =
pit_get_next_transition_time(sc, sc->count_load_time); pit_get_next_transition_time(sc, sc->count_load_time);
} }
...@@ -173,6 +205,19 @@ static void kvm_pit_irq_control(void *opaque, int n, int enable) ...@@ -173,6 +205,19 @@ static void kvm_pit_irq_control(void *opaque, int n, int enable)
kvm_pit_put(pit); kvm_pit_put(pit);
} }
static void kvm_pit_vm_state_change(void *opaque, int running,
RunState state)
{
KVMPITState *s = opaque;
if (running) {
s->state_valid = false;
} else {
kvm_pit_get(&s->pit);
s->state_valid = true;
}
}
static int kvm_pit_initfn(PITCommonState *pit) static int kvm_pit_initfn(PITCommonState *pit)
{ {
KVMPITState *s = DO_UPCAST(KVMPITState, pit, pit); KVMPITState *s = DO_UPCAST(KVMPITState, pit, pit);
...@@ -215,6 +260,8 @@ static int kvm_pit_initfn(PITCommonState *pit) ...@@ -215,6 +260,8 @@ static int kvm_pit_initfn(PITCommonState *pit)
qdev_init_gpio_in(&pit->dev.qdev, kvm_pit_irq_control, 1); qdev_init_gpio_in(&pit->dev.qdev, kvm_pit_irq_control, 1);
qemu_add_vm_change_state_handler(kvm_pit_vm_state_change, s);
return 0; return 0;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册