Commit 1872a3f4 authored by Avi Kivity's avatar Avi Kivity

KVM: VMX: Fix guest state validity checks

The vmx guest state validity checks are full of bugs.  Make them
conform to the manual.
Signed-off-by: default avatarAvi Kivity <avi@redhat.com>
parent bb3a8a17
...@@ -1784,14 +1784,16 @@ static bool code_segment_valid(struct kvm_vcpu *vcpu) ...@@ -1784,14 +1784,16 @@ static bool code_segment_valid(struct kvm_vcpu *vcpu)
vmx_get_segment(vcpu, &cs, VCPU_SREG_CS); vmx_get_segment(vcpu, &cs, VCPU_SREG_CS);
cs_rpl = cs.selector & SELECTOR_RPL_MASK; cs_rpl = cs.selector & SELECTOR_RPL_MASK;
if (cs.unusable)
return false;
if (~cs.type & (AR_TYPE_CODE_MASK|AR_TYPE_ACCESSES_MASK)) if (~cs.type & (AR_TYPE_CODE_MASK|AR_TYPE_ACCESSES_MASK))
return false; return false;
if (!cs.s) if (!cs.s)
return false; return false;
if (!(~cs.type & (AR_TYPE_CODE_MASK|AR_TYPE_WRITEABLE_MASK))) { if (cs.type & AR_TYPE_WRITEABLE_MASK) {
if (cs.dpl > cs_rpl) if (cs.dpl > cs_rpl)
return false; return false;
} else if (cs.type & AR_TYPE_CODE_MASK) { } else {
if (cs.dpl != cs_rpl) if (cs.dpl != cs_rpl)
return false; return false;
} }
...@@ -1810,7 +1812,9 @@ static bool stack_segment_valid(struct kvm_vcpu *vcpu) ...@@ -1810,7 +1812,9 @@ static bool stack_segment_valid(struct kvm_vcpu *vcpu)
vmx_get_segment(vcpu, &ss, VCPU_SREG_SS); vmx_get_segment(vcpu, &ss, VCPU_SREG_SS);
ss_rpl = ss.selector & SELECTOR_RPL_MASK; ss_rpl = ss.selector & SELECTOR_RPL_MASK;
if ((ss.type != 3) || (ss.type != 7)) if (ss.unusable)
return true;
if (ss.type != 3 && ss.type != 7)
return false; return false;
if (!ss.s) if (!ss.s)
return false; return false;
...@@ -1830,6 +1834,8 @@ static bool data_segment_valid(struct kvm_vcpu *vcpu, int seg) ...@@ -1830,6 +1834,8 @@ static bool data_segment_valid(struct kvm_vcpu *vcpu, int seg)
vmx_get_segment(vcpu, &var, seg); vmx_get_segment(vcpu, &var, seg);
rpl = var.selector & SELECTOR_RPL_MASK; rpl = var.selector & SELECTOR_RPL_MASK;
if (var.unusable)
return true;
if (!var.s) if (!var.s)
return false; return false;
if (!var.present) if (!var.present)
...@@ -1851,9 +1857,11 @@ static bool tr_valid(struct kvm_vcpu *vcpu) ...@@ -1851,9 +1857,11 @@ static bool tr_valid(struct kvm_vcpu *vcpu)
vmx_get_segment(vcpu, &tr, VCPU_SREG_TR); vmx_get_segment(vcpu, &tr, VCPU_SREG_TR);
if (tr.unusable)
return false;
if (tr.selector & SELECTOR_TI_MASK) /* TI = 1 */ if (tr.selector & SELECTOR_TI_MASK) /* TI = 1 */
return false; return false;
if ((tr.type != 3) || (tr.type != 11)) /* TODO: Check if guest is in IA32e mode */ if (tr.type != 3 && tr.type != 11) /* TODO: Check if guest is in IA32e mode */
return false; return false;
if (!tr.present) if (!tr.present)
return false; return false;
...@@ -1867,6 +1875,8 @@ static bool ldtr_valid(struct kvm_vcpu *vcpu) ...@@ -1867,6 +1875,8 @@ static bool ldtr_valid(struct kvm_vcpu *vcpu)
vmx_get_segment(vcpu, &ldtr, VCPU_SREG_LDTR); vmx_get_segment(vcpu, &ldtr, VCPU_SREG_LDTR);
if (ldtr.unusable)
return true;
if (ldtr.selector & SELECTOR_TI_MASK) /* TI = 1 */ if (ldtr.selector & SELECTOR_TI_MASK) /* TI = 1 */
return false; return false;
if (ldtr.type != 2) if (ldtr.type != 2)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment