diff options
Diffstat (limited to 'arch/x86/kvm/vmx.c')
-rw-r--r-- | arch/x86/kvm/vmx.c | 11 |
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 | ||
3418 | static u64 vmx_get_segment_base(struct kvm_vcpu *vcpu, int seg) | 3425 | static u64 vmx_get_segment_base(struct kvm_vcpu *vcpu, int seg) |