xref: /openbmc/qemu/target/i386/tcg/sysemu/misc_helper.c (revision 06152b89db64bc5ccec1e54576706ba891654df9)
1 /*
2  *  x86 misc helpers - sysemu code
3  *
4  *  Copyright (c) 2003 Fabrice Bellard
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18  */
19 
20 #include "qemu/osdep.h"
21 #include "qemu/main-loop.h"
22 #include "cpu.h"
23 #include "exec/helper-proto.h"
24 #include "exec/cpu_ldst.h"
25 #include "exec/address-spaces.h"
26 #include "exec/exec-all.h"
27 #include "tcg/helper-tcg.h"
28 
29 void helper_outb(CPUX86State *env, uint32_t port, uint32_t data)
30 {
31     address_space_stb(&address_space_io, port, data,
32                       cpu_get_mem_attrs(env), NULL);
33 }
34 
35 target_ulong helper_inb(CPUX86State *env, uint32_t port)
36 {
37     return address_space_ldub(&address_space_io, port,
38                               cpu_get_mem_attrs(env), NULL);
39 }
40 
41 void helper_outw(CPUX86State *env, uint32_t port, uint32_t data)
42 {
43     address_space_stw(&address_space_io, port, data,
44                       cpu_get_mem_attrs(env), NULL);
45 }
46 
47 target_ulong helper_inw(CPUX86State *env, uint32_t port)
48 {
49     return address_space_lduw(&address_space_io, port,
50                               cpu_get_mem_attrs(env), NULL);
51 }
52 
53 void helper_outl(CPUX86State *env, uint32_t port, uint32_t data)
54 {
55     address_space_stl(&address_space_io, port, data,
56                       cpu_get_mem_attrs(env), NULL);
57 }
58 
59 target_ulong helper_inl(CPUX86State *env, uint32_t port)
60 {
61     return address_space_ldl(&address_space_io, port,
62                              cpu_get_mem_attrs(env), NULL);
63 }
64 
65 target_ulong helper_read_crN(CPUX86State *env, int reg)
66 {
67     target_ulong val;
68 
69     switch (reg) {
70     default:
71         val = env->cr[reg];
72         break;
73     case 8:
74         if (!(env->hflags2 & HF2_VINTR_MASK)) {
75             val = cpu_get_apic_tpr(env_archcpu(env)->apic_state);
76         } else {
77             val = env->int_ctl & V_TPR_MASK;
78         }
79         break;
80     }
81     return val;
82 }
83 
84 void helper_write_crN(CPUX86State *env, int reg, target_ulong t0)
85 {
86     switch (reg) {
87     case 0:
88         /*
89         * If we reach this point, the CR0 write intercept is disabled.
90         * But we could still exit if the hypervisor has requested the selective
91         * intercept for bits other than TS and MP
92         */
93         if (cpu_svm_has_intercept(env, SVM_EXIT_CR0_SEL_WRITE) &&
94             ((env->cr[0] ^ t0) & ~(CR0_TS_MASK | CR0_MP_MASK))) {
95             cpu_vmexit(env, SVM_EXIT_CR0_SEL_WRITE, 0, GETPC());
96         }
97         cpu_x86_update_cr0(env, t0);
98         break;
99     case 3:
100         if ((env->efer & MSR_EFER_LMA) &&
101                 (t0 & ((~0ULL) << env_archcpu(env)->phys_bits))) {
102             cpu_vmexit(env, SVM_EXIT_ERR, 0, GETPC());
103         }
104         if (!(env->efer & MSR_EFER_LMA)) {
105             t0 &= 0xffffffffUL;
106         }
107         cpu_x86_update_cr3(env, t0);
108         break;
109     case 4:
110         if (t0 & cr4_reserved_bits(env)) {
111             cpu_vmexit(env, SVM_EXIT_ERR, 0, GETPC());
112         }
113         if (((t0 ^ env->cr[4]) & CR4_LA57_MASK) &&
114             (env->hflags & HF_CS64_MASK)) {
115             raise_exception_ra(env, EXCP0D_GPF, GETPC());
116         }
117         cpu_x86_update_cr4(env, t0);
118         break;
119     case 8:
120         if (!(env->hflags2 & HF2_VINTR_MASK)) {
121             bql_lock();
122             cpu_set_apic_tpr(env_archcpu(env)->apic_state, t0);
123             bql_unlock();
124         }
125         env->int_ctl = (env->int_ctl & ~V_TPR_MASK) | (t0 & V_TPR_MASK);
126 
127         CPUState *cs = env_cpu(env);
128         if (ctl_has_irq(env)) {
129             cpu_interrupt(cs, CPU_INTERRUPT_VIRQ);
130         } else {
131             cpu_reset_interrupt(cs, CPU_INTERRUPT_VIRQ);
132         }
133         break;
134     default:
135         env->cr[reg] = t0;
136         break;
137     }
138 }
139 
140 void helper_wrmsr(CPUX86State *env)
141 {
142     uint64_t val;
143     CPUState *cs = env_cpu(env);
144 
145     cpu_svm_check_intercept_param(env, SVM_EXIT_MSR, 1, GETPC());
146 
147     val = ((uint32_t)env->regs[R_EAX]) |
148         ((uint64_t)((uint32_t)env->regs[R_EDX]) << 32);
149 
150     switch ((uint32_t)env->regs[R_ECX]) {
151     case MSR_IA32_SYSENTER_CS:
152         env->sysenter_cs = val & 0xffff;
153         break;
154     case MSR_IA32_SYSENTER_ESP:
155         env->sysenter_esp = val;
156         break;
157     case MSR_IA32_SYSENTER_EIP:
158         env->sysenter_eip = val;
159         break;
160     case MSR_IA32_APICBASE:
161         cpu_set_apic_base(env_archcpu(env)->apic_state, val);
162         break;
163     case MSR_EFER:
164         {
165             uint64_t update_mask;
166 
167             update_mask = 0;
168             if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_SYSCALL) {
169                 update_mask |= MSR_EFER_SCE;
170             }
171             if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_LM) {
172                 update_mask |= MSR_EFER_LME;
173             }
174             if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_FFXSR) {
175                 update_mask |= MSR_EFER_FFXSR;
176             }
177             if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_NX) {
178                 update_mask |= MSR_EFER_NXE;
179             }
180             if (env->features[FEAT_8000_0001_ECX] & CPUID_EXT3_SVM) {
181                 update_mask |= MSR_EFER_SVME;
182             }
183             if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_FFXSR) {
184                 update_mask |= MSR_EFER_FFXSR;
185             }
186             cpu_load_efer(env, (env->efer & ~update_mask) |
187                           (val & update_mask));
188         }
189         break;
190     case MSR_STAR:
191         env->star = val;
192         break;
193     case MSR_PAT:
194         env->pat = val;
195         break;
196     case MSR_IA32_PKRS:
197         if (val & 0xFFFFFFFF00000000ull) {
198             goto error;
199         }
200         env->pkrs = val;
201         tlb_flush(cs);
202         break;
203     case MSR_VM_HSAVE_PA:
204         env->vm_hsave = val;
205         break;
206 #ifdef TARGET_X86_64
207     case MSR_LSTAR:
208         env->lstar = val;
209         break;
210     case MSR_CSTAR:
211         env->cstar = val;
212         break;
213     case MSR_FMASK:
214         env->fmask = val;
215         break;
216     case MSR_FSBASE:
217         env->segs[R_FS].base = val;
218         break;
219     case MSR_GSBASE:
220         env->segs[R_GS].base = val;
221         break;
222     case MSR_KERNELGSBASE:
223         env->kernelgsbase = val;
224         break;
225 #endif
226     case MSR_MTRRphysBase(0):
227     case MSR_MTRRphysBase(1):
228     case MSR_MTRRphysBase(2):
229     case MSR_MTRRphysBase(3):
230     case MSR_MTRRphysBase(4):
231     case MSR_MTRRphysBase(5):
232     case MSR_MTRRphysBase(6):
233     case MSR_MTRRphysBase(7):
234         env->mtrr_var[((uint32_t)env->regs[R_ECX] -
235                        MSR_MTRRphysBase(0)) / 2].base = val;
236         break;
237     case MSR_MTRRphysMask(0):
238     case MSR_MTRRphysMask(1):
239     case MSR_MTRRphysMask(2):
240     case MSR_MTRRphysMask(3):
241     case MSR_MTRRphysMask(4):
242     case MSR_MTRRphysMask(5):
243     case MSR_MTRRphysMask(6):
244     case MSR_MTRRphysMask(7):
245         env->mtrr_var[((uint32_t)env->regs[R_ECX] -
246                        MSR_MTRRphysMask(0)) / 2].mask = val;
247         break;
248     case MSR_MTRRfix64K_00000:
249         env->mtrr_fixed[(uint32_t)env->regs[R_ECX] -
250                         MSR_MTRRfix64K_00000] = val;
251         break;
252     case MSR_MTRRfix16K_80000:
253     case MSR_MTRRfix16K_A0000:
254         env->mtrr_fixed[(uint32_t)env->regs[R_ECX] -
255                         MSR_MTRRfix16K_80000 + 1] = val;
256         break;
257     case MSR_MTRRfix4K_C0000:
258     case MSR_MTRRfix4K_C8000:
259     case MSR_MTRRfix4K_D0000:
260     case MSR_MTRRfix4K_D8000:
261     case MSR_MTRRfix4K_E0000:
262     case MSR_MTRRfix4K_E8000:
263     case MSR_MTRRfix4K_F0000:
264     case MSR_MTRRfix4K_F8000:
265         env->mtrr_fixed[(uint32_t)env->regs[R_ECX] -
266                         MSR_MTRRfix4K_C0000 + 3] = val;
267         break;
268     case MSR_MTRRdefType:
269         env->mtrr_deftype = val;
270         break;
271     case MSR_MCG_STATUS:
272         env->mcg_status = val;
273         break;
274     case MSR_MCG_CTL:
275         if ((env->mcg_cap & MCG_CTL_P)
276             && (val == 0 || val == ~(uint64_t)0)) {
277             env->mcg_ctl = val;
278         }
279         break;
280     case MSR_TSC_AUX:
281         env->tsc_aux = val;
282         break;
283     case MSR_IA32_MISC_ENABLE:
284         env->msr_ia32_misc_enable = val;
285         break;
286     case MSR_IA32_BNDCFGS:
287         /* FIXME: #GP if reserved bits are set.  */
288         /* FIXME: Extend highest implemented bit of linear address.  */
289         env->msr_bndcfgs = val;
290         cpu_sync_bndcs_hflags(env);
291         break;
292     default:
293         if ((uint32_t)env->regs[R_ECX] >= MSR_MC0_CTL
294             && (uint32_t)env->regs[R_ECX] < MSR_MC0_CTL +
295             (4 * env->mcg_cap & 0xff)) {
296             uint32_t offset = (uint32_t)env->regs[R_ECX] - MSR_MC0_CTL;
297             if ((offset & 0x3) != 0
298                 || (val == 0 || val == ~(uint64_t)0)) {
299                 env->mce_banks[offset] = val;
300             }
301             break;
302         }
303         /* XXX: exception? */
304         break;
305     }
306     return;
307 error:
308     raise_exception_err_ra(env, EXCP0D_GPF, 0, GETPC());
309 }
310 
311 void helper_rdmsr(CPUX86State *env)
312 {
313     X86CPU *x86_cpu = env_archcpu(env);
314     uint64_t val;
315 
316     cpu_svm_check_intercept_param(env, SVM_EXIT_MSR, 0, GETPC());
317 
318     switch ((uint32_t)env->regs[R_ECX]) {
319     case MSR_IA32_SYSENTER_CS:
320         val = env->sysenter_cs;
321         break;
322     case MSR_IA32_SYSENTER_ESP:
323         val = env->sysenter_esp;
324         break;
325     case MSR_IA32_SYSENTER_EIP:
326         val = env->sysenter_eip;
327         break;
328     case MSR_IA32_APICBASE:
329         val = cpu_get_apic_base(env_archcpu(env)->apic_state);
330         break;
331     case MSR_EFER:
332         val = env->efer;
333         break;
334     case MSR_STAR:
335         val = env->star;
336         break;
337     case MSR_PAT:
338         val = env->pat;
339         break;
340     case MSR_IA32_PKRS:
341         val = env->pkrs;
342         break;
343     case MSR_VM_HSAVE_PA:
344         val = env->vm_hsave;
345         break;
346     case MSR_IA32_PERF_STATUS:
347         /* tsc_increment_by_tick */
348         val = 1000ULL;
349         /* CPU multiplier */
350         val |= (((uint64_t)4ULL) << 40);
351         break;
352 #ifdef TARGET_X86_64
353     case MSR_LSTAR:
354         val = env->lstar;
355         break;
356     case MSR_CSTAR:
357         val = env->cstar;
358         break;
359     case MSR_FMASK:
360         val = env->fmask;
361         break;
362     case MSR_FSBASE:
363         val = env->segs[R_FS].base;
364         break;
365     case MSR_GSBASE:
366         val = env->segs[R_GS].base;
367         break;
368     case MSR_KERNELGSBASE:
369         val = env->kernelgsbase;
370         break;
371     case MSR_TSC_AUX:
372         val = env->tsc_aux;
373         break;
374 #endif
375     case MSR_SMI_COUNT:
376         val = env->msr_smi_count;
377         break;
378     case MSR_MTRRphysBase(0):
379     case MSR_MTRRphysBase(1):
380     case MSR_MTRRphysBase(2):
381     case MSR_MTRRphysBase(3):
382     case MSR_MTRRphysBase(4):
383     case MSR_MTRRphysBase(5):
384     case MSR_MTRRphysBase(6):
385     case MSR_MTRRphysBase(7):
386         val = env->mtrr_var[((uint32_t)env->regs[R_ECX] -
387                              MSR_MTRRphysBase(0)) / 2].base;
388         break;
389     case MSR_MTRRphysMask(0):
390     case MSR_MTRRphysMask(1):
391     case MSR_MTRRphysMask(2):
392     case MSR_MTRRphysMask(3):
393     case MSR_MTRRphysMask(4):
394     case MSR_MTRRphysMask(5):
395     case MSR_MTRRphysMask(6):
396     case MSR_MTRRphysMask(7):
397         val = env->mtrr_var[((uint32_t)env->regs[R_ECX] -
398                              MSR_MTRRphysMask(0)) / 2].mask;
399         break;
400     case MSR_MTRRfix64K_00000:
401         val = env->mtrr_fixed[0];
402         break;
403     case MSR_MTRRfix16K_80000:
404     case MSR_MTRRfix16K_A0000:
405         val = env->mtrr_fixed[(uint32_t)env->regs[R_ECX] -
406                               MSR_MTRRfix16K_80000 + 1];
407         break;
408     case MSR_MTRRfix4K_C0000:
409     case MSR_MTRRfix4K_C8000:
410     case MSR_MTRRfix4K_D0000:
411     case MSR_MTRRfix4K_D8000:
412     case MSR_MTRRfix4K_E0000:
413     case MSR_MTRRfix4K_E8000:
414     case MSR_MTRRfix4K_F0000:
415     case MSR_MTRRfix4K_F8000:
416         val = env->mtrr_fixed[(uint32_t)env->regs[R_ECX] -
417                               MSR_MTRRfix4K_C0000 + 3];
418         break;
419     case MSR_MTRRdefType:
420         val = env->mtrr_deftype;
421         break;
422     case MSR_MTRRcap:
423         if (env->features[FEAT_1_EDX] & CPUID_MTRR) {
424             val = MSR_MTRRcap_VCNT | MSR_MTRRcap_FIXRANGE_SUPPORT |
425                 MSR_MTRRcap_WC_SUPPORTED;
426         } else {
427             /* XXX: exception? */
428             val = 0;
429         }
430         break;
431     case MSR_MCG_CAP:
432         val = env->mcg_cap;
433         break;
434     case MSR_MCG_CTL:
435         if (env->mcg_cap & MCG_CTL_P) {
436             val = env->mcg_ctl;
437         } else {
438             val = 0;
439         }
440         break;
441     case MSR_MCG_STATUS:
442         val = env->mcg_status;
443         break;
444     case MSR_IA32_MISC_ENABLE:
445         val = env->msr_ia32_misc_enable;
446         break;
447     case MSR_IA32_BNDCFGS:
448         val = env->msr_bndcfgs;
449         break;
450      case MSR_IA32_UCODE_REV:
451         val = x86_cpu->ucode_rev;
452         break;
453     case MSR_CORE_THREAD_COUNT: {
454         CPUState *cs = CPU(x86_cpu);
455         val = (cs->nr_threads * cs->nr_cores) | (cs->nr_cores << 16);
456         break;
457     }
458     default:
459         if ((uint32_t)env->regs[R_ECX] >= MSR_MC0_CTL
460             && (uint32_t)env->regs[R_ECX] < MSR_MC0_CTL +
461             (4 * env->mcg_cap & 0xff)) {
462             uint32_t offset = (uint32_t)env->regs[R_ECX] - MSR_MC0_CTL;
463             val = env->mce_banks[offset];
464             break;
465         }
466         /* XXX: exception? */
467         val = 0;
468         break;
469     }
470     env->regs[R_EAX] = (uint32_t)(val);
471     env->regs[R_EDX] = (uint32_t)(val >> 32);
472 }
473 
474 void helper_flush_page(CPUX86State *env, target_ulong addr)
475 {
476     tlb_flush_page(env_cpu(env), addr);
477 }
478 
479 static G_NORETURN
480 void do_hlt(CPUX86State *env)
481 {
482     CPUState *cs = env_cpu(env);
483 
484     env->hflags &= ~HF_INHIBIT_IRQ_MASK; /* needed if sti is just before */
485     cs->halted = 1;
486     cs->exception_index = EXCP_HLT;
487     cpu_loop_exit(cs);
488 }
489 
490 G_NORETURN void helper_hlt(CPUX86State *env, int next_eip_addend)
491 {
492     cpu_svm_check_intercept_param(env, SVM_EXIT_HLT, 0, GETPC());
493     env->eip += next_eip_addend;
494 
495     do_hlt(env);
496 }
497 
498 void helper_monitor(CPUX86State *env, target_ulong ptr)
499 {
500     if ((uint32_t)env->regs[R_ECX] != 0) {
501         raise_exception_ra(env, EXCP0D_GPF, GETPC());
502     }
503     /* XXX: store address? */
504     cpu_svm_check_intercept_param(env, SVM_EXIT_MONITOR, 0, GETPC());
505 }
506 
507 G_NORETURN void helper_mwait(CPUX86State *env, int next_eip_addend)
508 {
509     CPUState *cs = env_cpu(env);
510 
511     if ((uint32_t)env->regs[R_ECX] != 0) {
512         raise_exception_ra(env, EXCP0D_GPF, GETPC());
513     }
514     cpu_svm_check_intercept_param(env, SVM_EXIT_MWAIT, 0, GETPC());
515     env->eip += next_eip_addend;
516 
517     /* XXX: not complete but not completely erroneous */
518     if (cs->cpu_index != 0 || CPU_NEXT(cs) != NULL) {
519         do_pause(env);
520     } else {
521         do_hlt(env);
522     }
523 }
524