diff options
| author | Nir Weiner <nir.weiner@oracle.com> | 2019-01-27 05:17:16 -0500 |
|---|---|---|
| committer | Paolo Bonzini <pbonzini@redhat.com> | 2019-02-20 16:48:51 -0500 |
| commit | dee339b5c1da3e6fa139b97f74f99dc9f0b03ff6 (patch) | |
| tree | 6693b07487412d7ca10a17cf07d31dcdb9d43fe0 | |
| parent | 49113d360bdeb4dd916fb6bffbcc3e157422b6fd (diff) | |
KVM: Never start grow vCPU halt_poll_ns from value below halt_poll_ns_grow_start
grow_halt_poll_ns() have a strange behaviour in case
(vcpu->halt_poll_ns != 0) &&
(vcpu->halt_poll_ns < halt_poll_ns_grow_start).
In this case, vcpu->halt_poll_ns will be multiplied by grow factor
(halt_poll_ns_grow) which will require several grow iteration in order
to reach a value bigger than halt_poll_ns_grow_start.
This means that growing vcpu->halt_poll_ns from value of 0 is slower
than growing it from a positive value less than halt_poll_ns_grow_start.
Which is misleading and inaccurate.
Fix issue by changing grow_halt_poll_ns() to set vcpu->halt_poll_ns
to halt_poll_ns_grow_start in any case that
(vcpu->halt_poll_ns < halt_poll_ns_grow_start).
Regardless if vcpu->halt_poll_ns is 0.
use READ_ONCE to get a consistent number for all cases.
Reviewed-by: Boris Ostrovsky <boris.ostrovsky@oracle.com>
Reviewed-by: Liran Alon <liran.alon@oracle.com>
Signed-off-by: Nir Weiner <nir.weiner@oracle.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
| -rw-r--r-- | arch/powerpc/kvm/book3s_hv.c | 5 | ||||
| -rw-r--r-- | virt/kvm/kvm_main.c | 10 |
2 files changed, 7 insertions, 8 deletions
diff --git a/arch/powerpc/kvm/book3s_hv.c b/arch/powerpc/kvm/book3s_hv.c index 29ffc99bd79b..062f3c92c871 100644 --- a/arch/powerpc/kvm/book3s_hv.c +++ b/arch/powerpc/kvm/book3s_hv.c | |||
| @@ -3634,10 +3634,9 @@ static void grow_halt_poll_ns(struct kvmppc_vcore *vc) | |||
| 3634 | if (!halt_poll_ns_grow) | 3634 | if (!halt_poll_ns_grow) |
| 3635 | return; | 3635 | return; |
| 3636 | 3636 | ||
| 3637 | if (vc->halt_poll_ns == 0) | 3637 | vc->halt_poll_ns *= halt_poll_ns_grow; |
| 3638 | if (vc->halt_poll_ns < halt_poll_ns_grow_start) | ||
| 3638 | vc->halt_poll_ns = halt_poll_ns_grow_start; | 3639 | vc->halt_poll_ns = halt_poll_ns_grow_start; |
| 3639 | else | ||
| 3640 | vc->halt_poll_ns *= halt_poll_ns_grow; | ||
| 3641 | } | 3640 | } |
| 3642 | 3641 | ||
| 3643 | static void shrink_halt_poll_ns(struct kvmppc_vcore *vc) | 3642 | static void shrink_halt_poll_ns(struct kvmppc_vcore *vc) |
diff --git a/virt/kvm/kvm_main.c b/virt/kvm/kvm_main.c index ae818d27a1a4..5087cf703ed1 100644 --- a/virt/kvm/kvm_main.c +++ b/virt/kvm/kvm_main.c | |||
| @@ -2189,17 +2189,17 @@ void kvm_sigset_deactivate(struct kvm_vcpu *vcpu) | |||
| 2189 | 2189 | ||
| 2190 | static void grow_halt_poll_ns(struct kvm_vcpu *vcpu) | 2190 | static void grow_halt_poll_ns(struct kvm_vcpu *vcpu) |
| 2191 | { | 2191 | { |
| 2192 | unsigned int old, val, grow; | 2192 | unsigned int old, val, grow, grow_start; |
| 2193 | 2193 | ||
| 2194 | old = val = vcpu->halt_poll_ns; | 2194 | old = val = vcpu->halt_poll_ns; |
| 2195 | grow_start = READ_ONCE(halt_poll_ns_grow_start); | ||
| 2195 | grow = READ_ONCE(halt_poll_ns_grow); | 2196 | grow = READ_ONCE(halt_poll_ns_grow); |
| 2196 | if (!grow) | 2197 | if (!grow) |
| 2197 | goto out; | 2198 | goto out; |
| 2198 | 2199 | ||
| 2199 | if (val == 0) | 2200 | val *= grow; |
| 2200 | val = halt_poll_ns_grow_start; | 2201 | if (val < grow_start) |
| 2201 | else | 2202 | val = grow_start; |
| 2202 | val *= grow; | ||
| 2203 | 2203 | ||
| 2204 | if (val > halt_poll_ns) | 2204 | if (val > halt_poll_ns) |
| 2205 | val = halt_poll_ns; | 2205 | val = halt_poll_ns; |
