提交 cc921867 编写于 作者: A Anthony Liguori

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

* qemu-kvm/uq/master:
  update-linux-headers.sh: Pull in asm-generic/kvm_para.h
  kvmvapic: Disable if there is insufficient memory
  kvm: i8254: Finish time conversion fix
  kvm: i8254: Cache kernel clock offset in KVMPITState
...@@ -299,7 +299,9 @@ static int apic_init_common(SysBusDevice *dev) ...@@ -299,7 +299,9 @@ static int apic_init_common(SysBusDevice *dev)
sysbus_init_mmio(dev, &s->io_memory); sysbus_init_mmio(dev, &s->io_memory);
if (!vapic && s->vapic_control & VAPIC_ENABLE_MASK) { /* Note: We need at least 1M to map the VAPIC option ROM */
if (!vapic && s->vapic_control & VAPIC_ENABLE_MASK &&
ram_size >= 1024 * 1024) {
vapic = sysbus_create_simple("kvmvapic", -1, NULL); vapic = sysbus_create_simple("kvmvapic", -1, NULL);
} }
s->vapic = vapic; s->vapic = vapic;
......
...@@ -35,7 +35,8 @@ ...@@ -35,7 +35,8 @@
typedef struct KVMPITState { typedef struct KVMPITState {
PITCommonState pit; PITCommonState pit;
LostTickPolicy lost_tick_policy; LostTickPolicy lost_tick_policy;
bool state_valid; bool vm_stopped;
int64_t kernel_clock_offset;
} KVMPITState; } KVMPITState;
static int64_t abs64(int64_t v) static int64_t abs64(int64_t v)
...@@ -43,19 +44,11 @@ static int64_t abs64(int64_t v) ...@@ -43,19 +44,11 @@ static int64_t abs64(int64_t v)
return v < 0 ? -v : v; return v < 0 ? -v : v;
} }
static void kvm_pit_get(PITCommonState *pit) static void kvm_pit_update_clock_offset(KVMPITState *s)
{ {
KVMPITState *s = DO_UPCAST(KVMPITState, pit, pit);
struct kvm_pit_state2 kpit;
struct kvm_pit_channel_state *kchan;
struct PITChannelState *sc;
int64_t offset, clock_offset; int64_t offset, clock_offset;
struct timespec ts; struct timespec ts;
int i, ret; int i;
if (s->state_valid) {
return;
}
/* /*
* Measure the delta between CLOCK_MONOTONIC, the base used for * Measure the delta between CLOCK_MONOTONIC, the base used for
...@@ -72,6 +65,21 @@ static void kvm_pit_get(PITCommonState *pit) ...@@ -72,6 +65,21 @@ static void kvm_pit_get(PITCommonState *pit)
clock_offset = offset; clock_offset = offset;
} }
} }
s->kernel_clock_offset = clock_offset;
}
static void kvm_pit_get(PITCommonState *pit)
{
KVMPITState *s = DO_UPCAST(KVMPITState, pit, pit);
struct kvm_pit_state2 kpit;
struct kvm_pit_channel_state *kchan;
struct PITChannelState *sc;
int i, ret;
/* No need to re-read the state if VM is stopped. */
if (s->vm_stopped) {
return;
}
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);
...@@ -106,7 +114,7 @@ static void kvm_pit_get(PITCommonState *pit) ...@@ -106,7 +114,7 @@ static void kvm_pit_get(PITCommonState *pit)
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 + clock_offset; sc->count_load_time = kchan->count_load_time + s->kernel_clock_offset;
} }
sc = &pit->channels[0]; sc = &pit->channels[0];
...@@ -114,17 +122,23 @@ static void kvm_pit_get(PITCommonState *pit) ...@@ -114,17 +122,23 @@ static void kvm_pit_get(PITCommonState *pit)
pit_get_next_transition_time(sc, sc->count_load_time); pit_get_next_transition_time(sc, sc->count_load_time);
} }
static void kvm_pit_put(PITCommonState *s) static void kvm_pit_put(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;
int i, ret; int i, ret;
kpit.flags = s->channels[0].irq_disabled ? KVM_PIT_FLAGS_HPET_LEGACY : 0; /* The offset keeps changing as long as the VM is stopped. */
if (s->vm_stopped) {
kvm_pit_update_clock_offset(s);
}
kpit.flags = pit->channels[0].irq_disabled ? KVM_PIT_FLAGS_HPET_LEGACY : 0;
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];
kchan->count = sc->count; kchan->count = sc->count;
kchan->latched_count = sc->latched_count; kchan->latched_count = sc->latched_count;
kchan->count_latched = sc->count_latched; kchan->count_latched = sc->count_latched;
...@@ -137,7 +151,7 @@ static void kvm_pit_put(PITCommonState *s) ...@@ -137,7 +151,7 @@ static void kvm_pit_put(PITCommonState *s)
kchan->mode = sc->mode; kchan->mode = sc->mode;
kchan->bcd = sc->bcd; kchan->bcd = sc->bcd;
kchan->gate = sc->gate; kchan->gate = sc->gate;
kchan->count_load_time = sc->count_load_time; kchan->count_load_time = sc->count_load_time - s->kernel_clock_offset;
} }
ret = kvm_vm_ioctl(kvm_state, ret = kvm_vm_ioctl(kvm_state,
...@@ -211,10 +225,12 @@ static void kvm_pit_vm_state_change(void *opaque, int running, ...@@ -211,10 +225,12 @@ static void kvm_pit_vm_state_change(void *opaque, int running,
KVMPITState *s = opaque; KVMPITState *s = opaque;
if (running) { if (running) {
s->state_valid = false; kvm_pit_update_clock_offset(s);
s->vm_stopped = false;
} else { } else {
kvm_pit_update_clock_offset(s);
kvm_pit_get(&s->pit); kvm_pit_get(&s->pit);
s->state_valid = true; s->vm_stopped = true;
} }
} }
......
...@@ -46,6 +46,11 @@ mkdir -p "$output/linux-headers/linux" ...@@ -46,6 +46,11 @@ mkdir -p "$output/linux-headers/linux"
for header in kvm.h kvm_para.h vhost.h virtio_config.h virtio_ring.h; do for header in kvm.h kvm_para.h vhost.h virtio_config.h virtio_ring.h; do
cp "$tmpdir/include/linux/$header" "$output/linux-headers/linux" cp "$tmpdir/include/linux/$header" "$output/linux-headers/linux"
done done
rm -rf "$output/linux-headers/asm-generic"
mkdir -p "$output/linux-headers/asm-generic"
for header in kvm_para.h; do
cp "$tmpdir/include/asm-generic/$header" "$output/linux-headers/asm-generic"
done
if [ -L "$linux/source" ]; then if [ -L "$linux/source" ]; then
cp "$linux/source/COPYING" "$output/linux-headers" cp "$linux/source/COPYING" "$output/linux-headers"
else else
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册