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
This commit is contained in:
commit
cc92186773
@ -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 @@
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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,
|
|||||||
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"
|
|||||||
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
|
||||||
|
Loading…
x
Reference in New Issue
Block a user