target-i386: get/set/migrate XSAVES state

Add xsaves related definition, it also adds corresponding part
to kvm_get/put, and vmstate.

Signed-off-by: Wanpeng Li <wanpeng.li@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
Wanpeng Li 2014-12-03 10:36:23 +08:00 committed by Paolo Bonzini
parent 906b53a2de
commit 18cd2c17b5
4 changed files with 38 additions and 1 deletions

View File

@ -408,7 +408,6 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
.cpuid_needs_ecx = true, .cpuid_ecx = 1, .cpuid_needs_ecx = true, .cpuid_ecx = 1,
.cpuid_reg = R_EAX, .cpuid_reg = R_EAX,
.tcg_features = 0, .tcg_features = 0,
.unmigratable_flags = FEAT_XSAVES,
}, },
}; };

View File

@ -389,6 +389,7 @@
#define MSR_VM_HSAVE_PA 0xc0010117 #define MSR_VM_HSAVE_PA 0xc0010117
#define MSR_IA32_BNDCFGS 0x00000d90 #define MSR_IA32_BNDCFGS 0x00000d90
#define MSR_IA32_XSS 0x00000da0
#define XSTATE_FP (1ULL << 0) #define XSTATE_FP (1ULL << 0)
#define XSTATE_SSE (1ULL << 1) #define XSTATE_SSE (1ULL << 1)
@ -1025,6 +1026,7 @@ typedef struct CPUX86State {
uint64_t xstate_bv; uint64_t xstate_bv;
uint64_t xcr0; uint64_t xcr0;
uint64_t xss;
TPRAccess tpr_access_type; TPRAccess tpr_access_type;
} CPUX86State; } CPUX86State;

View File

@ -80,6 +80,7 @@ static bool has_msr_hv_hypercall;
static bool has_msr_hv_vapic; static bool has_msr_hv_vapic;
static bool has_msr_hv_tsc; static bool has_msr_hv_tsc;
static bool has_msr_mtrr; static bool has_msr_mtrr;
static bool has_msr_xss;
static bool has_msr_architectural_pmu; static bool has_msr_architectural_pmu;
static uint32_t num_architectural_pmu_counters; static uint32_t num_architectural_pmu_counters;
@ -826,6 +827,10 @@ static int kvm_get_supported_msrs(KVMState *s)
has_msr_bndcfgs = true; has_msr_bndcfgs = true;
continue; continue;
} }
if (kvm_msr_list->indices[i] == MSR_IA32_XSS) {
has_msr_xss = true;
continue;
}
} }
} }
@ -1231,6 +1236,9 @@ static int kvm_put_msrs(X86CPU *cpu, int level)
if (has_msr_bndcfgs) { if (has_msr_bndcfgs) {
kvm_msr_entry_set(&msrs[n++], MSR_IA32_BNDCFGS, env->msr_bndcfgs); kvm_msr_entry_set(&msrs[n++], MSR_IA32_BNDCFGS, env->msr_bndcfgs);
} }
if (has_msr_xss) {
kvm_msr_entry_set(&msrs[n++], MSR_IA32_XSS, env->xss);
}
#ifdef TARGET_X86_64 #ifdef TARGET_X86_64
if (lm_capable_kernel) { if (lm_capable_kernel) {
kvm_msr_entry_set(&msrs[n++], MSR_CSTAR, env->cstar); kvm_msr_entry_set(&msrs[n++], MSR_CSTAR, env->cstar);
@ -1579,6 +1587,10 @@ static int kvm_get_msrs(X86CPU *cpu)
if (has_msr_bndcfgs) { if (has_msr_bndcfgs) {
msrs[n++].index = MSR_IA32_BNDCFGS; msrs[n++].index = MSR_IA32_BNDCFGS;
} }
if (has_msr_xss) {
msrs[n++].index = MSR_IA32_XSS;
}
if (!env->tsc_valid) { if (!env->tsc_valid) {
msrs[n++].index = MSR_IA32_TSC; msrs[n++].index = MSR_IA32_TSC;
@ -1729,6 +1741,9 @@ static int kvm_get_msrs(X86CPU *cpu)
case MSR_IA32_BNDCFGS: case MSR_IA32_BNDCFGS:
env->msr_bndcfgs = msrs[i].data; env->msr_bndcfgs = msrs[i].data;
break; break;
case MSR_IA32_XSS:
env->xss = msrs[i].data;
break;
default: default:
if (msrs[i].index >= MSR_MC0_CTL && if (msrs[i].index >= MSR_MC0_CTL &&
msrs[i].index < MSR_MC0_CTL + (env->mcg_cap & 0xff) * 4) { msrs[i].index < MSR_MC0_CTL + (env->mcg_cap & 0xff) * 4) {

View File

@ -687,6 +687,24 @@ static const VMStateDescription vmstate_avx512 = {
} }
}; };
static bool xss_needed(void *opaque)
{
X86CPU *cpu = opaque;
CPUX86State *env = &cpu->env;
return env->xss != 0;
}
static const VMStateDescription vmstate_xss = {
.name = "cpu/xss",
.version_id = 1,
.minimum_version_id = 1,
.fields = (VMStateField[]) {
VMSTATE_UINT64(env.xss, X86CPU),
VMSTATE_END_OF_LIST()
}
};
VMStateDescription vmstate_x86_cpu = { VMStateDescription vmstate_x86_cpu = {
.name = "cpu", .name = "cpu",
.version_id = 12, .version_id = 12,
@ -832,6 +850,9 @@ VMStateDescription vmstate_x86_cpu = {
}, { }, {
.vmsd = &vmstate_avx512, .vmsd = &vmstate_avx512,
.needed = avx512_needed, .needed = avx512_needed,
}, {
.vmsd = &vmstate_xss,
.needed = xss_needed,
} , { } , {
/* empty */ /* empty */
} }