aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kvm/vmx.c
diff options
context:
space:
mode:
authorGleb Natapov <gleb@redhat.com>2013-06-28 06:17:18 -0400
committerPaolo Bonzini <pbonzini@redhat.com>2013-07-04 08:40:36 -0400
commit03617c188f41eeeb4223c919ee7e66e5a114f2c6 (patch)
treedfc076e380e24be2299935f41fc9a7b1d08eddec /arch/x86/kvm/vmx.c
parentc3eb5b14449a0949e9764d39374a2ea63faae14f (diff)
KVM: VMX: mark unusable segment as nonpresent
Some userspaces do not preserve unusable property. Since usable segment has to be present according to VMX spec we can use present property to amend userspace bug by making unusable segment always nonpresent. vmx_segment_access_rights() already marks nonpresent segment as unusable. Cc: stable@vger.kernel.org # 3.9+ Reported-by: Stefan Pietsch <stefan.pietsch@lsexperts.de> Tested-by: Stefan Pietsch <stefan.pietsch@lsexperts.de> Signed-off-by: Gleb Natapov <gleb@redhat.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Diffstat (limited to 'arch/x86/kvm/vmx.c')
-rw-r--r--arch/x86/kvm/vmx.c11
1 files changed, 9 insertions, 2 deletions
diff --git a/arch/x86/kvm/vmx.c b/arch/x86/kvm/vmx.c
index a7e18551c968..064d0be67ecc 100644
--- a/arch/x86/kvm/vmx.c
+++ b/arch/x86/kvm/vmx.c
@@ -3404,15 +3404,22 @@ static void vmx_get_segment(struct kvm_vcpu *vcpu,
3404 var->limit = vmx_read_guest_seg_limit(vmx, seg); 3404 var->limit = vmx_read_guest_seg_limit(vmx, seg);
3405 var->selector = vmx_read_guest_seg_selector(vmx, seg); 3405 var->selector = vmx_read_guest_seg_selector(vmx, seg);
3406 ar = vmx_read_guest_seg_ar(vmx, seg); 3406 ar = vmx_read_guest_seg_ar(vmx, seg);
3407 var->unusable = (ar >> 16) & 1;
3407 var->type = ar & 15; 3408 var->type = ar & 15;
3408 var->s = (ar >> 4) & 1; 3409 var->s = (ar >> 4) & 1;
3409 var->dpl = (ar >> 5) & 3; 3410 var->dpl = (ar >> 5) & 3;
3410 var->present = (ar >> 7) & 1; 3411 /*
3412 * Some userspaces do not preserve unusable property. Since usable
3413 * segment has to be present according to VMX spec we can use present
3414 * property to amend userspace bug by making unusable segment always
3415 * nonpresent. vmx_segment_access_rights() already marks nonpresent
3416 * segment as unusable.
3417 */
3418 var->present = !var->unusable;
3411 var->avl = (ar >> 12) & 1; 3419 var->avl = (ar >> 12) & 1;
3412 var->l = (ar >> 13) & 1; 3420 var->l = (ar >> 13) & 1;
3413 var->db = (ar >> 14) & 1; 3421 var->db = (ar >> 14) & 1;
3414 var->g = (ar >> 15) & 1; 3422 var->g = (ar >> 15) & 1;
3415 var->unusable = (ar >> 16) & 1;
3416} 3423}
3417 3424
3418static u64 vmx_get_segment_base(struct kvm_vcpu *vcpu, int seg) 3425static u64 vmx_get_segment_base(struct kvm_vcpu *vcpu, int seg)