4 * Copyright (c) 2003 Fabrice Bellard
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 of the License, or (at your option) any later version.
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.
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/>.
24 /* check if Port I/O is allowed in TSS */
25 static inline void check_io(CPUX86State
*env
, int addr
, int size
)
27 int io_offset
, val
, mask
;
29 /* TSS must be a valid 32 bit one */
30 if (!(env
->tr
.flags
& DESC_P_MASK
) ||
31 ((env
->tr
.flags
>> DESC_TYPE_SHIFT
) & 0xf) != 9 ||
32 env
->tr
.limit
< 103) {
35 io_offset
= cpu_lduw_kernel(env
, env
->tr
.base
+ 0x66);
36 io_offset
+= (addr
>> 3);
37 /* Note: the check needs two bytes */
38 if ((io_offset
+ 1) > env
->tr
.limit
) {
41 val
= cpu_lduw_kernel(env
, env
->tr
.base
+ io_offset
);
43 mask
= (1 << size
) - 1;
44 /* all bits must be zero to allow the I/O */
45 if ((val
& mask
) != 0) {
47 raise_exception_err(env
, EXCP0D_GPF
, 0);
51 void helper_check_iob(CPUX86State
*env
, uint32_t t0
)
56 void helper_check_iow(CPUX86State
*env
, uint32_t t0
)
61 void helper_check_iol(CPUX86State
*env
, uint32_t t0
)
66 void helper_outb(uint32_t port
, uint32_t data
)
68 cpu_outb(port
, data
& 0xff);
71 target_ulong
helper_inb(uint32_t port
)
76 void helper_outw(uint32_t port
, uint32_t data
)
78 cpu_outw(port
, data
& 0xffff);
81 target_ulong
helper_inw(uint32_t port
)
86 void helper_outl(uint32_t port
, uint32_t data
)
91 target_ulong
helper_inl(uint32_t port
)
96 void helper_into(CPUX86State
*env
, int next_eip_addend
)
100 eflags
= cpu_cc_compute_all(env
, CC_OP
);
102 raise_interrupt(env
, EXCP04_INTO
, 1, 0, next_eip_addend
);
106 void helper_single_step(CPUX86State
*env
)
108 #ifndef CONFIG_USER_ONLY
109 check_hw_breakpoints(env
, 1);
110 env
->dr
[6] |= DR6_BS
;
112 raise_exception(env
, EXCP01_DB
);
115 void helper_cpuid(CPUX86State
*env
)
117 uint32_t eax
, ebx
, ecx
, edx
;
119 cpu_svm_check_intercept_param(env
, SVM_EXIT_CPUID
, 0);
121 cpu_x86_cpuid(env
, (uint32_t)EAX
, (uint32_t)ECX
, &eax
, &ebx
, &ecx
, &edx
);
128 #if defined(CONFIG_USER_ONLY)
129 target_ulong
helper_read_crN(CPUX86State
*env
, int reg
)
134 void helper_write_crN(CPUX86State
*env
, int reg
, target_ulong t0
)
138 void helper_movl_drN_T0(CPUX86State
*env
, int reg
, target_ulong t0
)
142 target_ulong
helper_read_crN(CPUX86State
*env
, int reg
)
146 cpu_svm_check_intercept_param(env
, SVM_EXIT_READ_CR0
+ reg
, 0);
152 if (!(env
->hflags2
& HF2_VINTR_MASK
)) {
153 val
= cpu_get_apic_tpr(env
->apic_state
);
162 void helper_write_crN(CPUX86State
*env
, int reg
, target_ulong t0
)
164 cpu_svm_check_intercept_param(env
, SVM_EXIT_WRITE_CR0
+ reg
, 0);
167 cpu_x86_update_cr0(env
, t0
);
170 cpu_x86_update_cr3(env
, t0
);
173 cpu_x86_update_cr4(env
, t0
);
176 if (!(env
->hflags2
& HF2_VINTR_MASK
)) {
177 cpu_set_apic_tpr(env
->apic_state
, t0
);
179 env
->v_tpr
= t0
& 0x0f;
187 void helper_movl_drN_T0(CPUX86State
*env
, int reg
, target_ulong t0
)
192 hw_breakpoint_remove(env
, reg
);
194 hw_breakpoint_insert(env
, reg
);
195 } else if (reg
== 7) {
196 for (i
= 0; i
< 4; i
++) {
197 hw_breakpoint_remove(env
, i
);
200 for (i
= 0; i
< 4; i
++) {
201 hw_breakpoint_insert(env
, i
);
209 void helper_lmsw(CPUX86State
*env
, target_ulong t0
)
211 /* only 4 lower bits of CR0 are modified. PE cannot be set to zero
212 if already set to one. */
213 t0
= (env
->cr
[0] & ~0xe) | (t0
& 0xf);
214 helper_write_crN(env
, 0, t0
);
217 void helper_invlpg(CPUX86State
*env
, target_ulong addr
)
219 cpu_svm_check_intercept_param(env
, SVM_EXIT_INVLPG
, 0);
220 tlb_flush_page(env
, addr
);
223 void helper_rdtsc(CPUX86State
*env
)
227 if ((env
->cr
[4] & CR4_TSD_MASK
) && ((env
->hflags
& HF_CPL_MASK
) != 0)) {
228 raise_exception(env
, EXCP0D_GPF
);
230 cpu_svm_check_intercept_param(env
, SVM_EXIT_RDTSC
, 0);
232 val
= cpu_get_tsc(env
) + env
->tsc_offset
;
233 EAX
= (uint32_t)(val
);
234 EDX
= (uint32_t)(val
>> 32);
237 void helper_rdtscp(CPUX86State
*env
)
240 ECX
= (uint32_t)(env
->tsc_aux
);
243 void helper_rdpmc(CPUX86State
*env
)
245 if ((env
->cr
[4] & CR4_PCE_MASK
) && ((env
->hflags
& HF_CPL_MASK
) != 0)) {
246 raise_exception(env
, EXCP0D_GPF
);
248 cpu_svm_check_intercept_param(env
, SVM_EXIT_RDPMC
, 0);
250 /* currently unimplemented */
251 qemu_log_mask(LOG_UNIMP
, "x86: unimplemented rdpmc\n");
252 raise_exception_err(env
, EXCP06_ILLOP
, 0);
255 #if defined(CONFIG_USER_ONLY)
256 void helper_wrmsr(CPUX86State
*env
)
260 void helper_rdmsr(CPUX86State
*env
)
264 void helper_wrmsr(CPUX86State
*env
)
268 cpu_svm_check_intercept_param(env
, SVM_EXIT_MSR
, 1);
270 val
= ((uint32_t)EAX
) | ((uint64_t)((uint32_t)EDX
) << 32);
272 switch ((uint32_t)ECX
) {
273 case MSR_IA32_SYSENTER_CS
:
274 env
->sysenter_cs
= val
& 0xffff;
276 case MSR_IA32_SYSENTER_ESP
:
277 env
->sysenter_esp
= val
;
279 case MSR_IA32_SYSENTER_EIP
:
280 env
->sysenter_eip
= val
;
282 case MSR_IA32_APICBASE
:
283 cpu_set_apic_base(env
->apic_state
, val
);
287 uint64_t update_mask
;
290 if (env
->cpuid_ext2_features
& CPUID_EXT2_SYSCALL
) {
291 update_mask
|= MSR_EFER_SCE
;
293 if (env
->cpuid_ext2_features
& CPUID_EXT2_LM
) {
294 update_mask
|= MSR_EFER_LME
;
296 if (env
->cpuid_ext2_features
& CPUID_EXT2_FFXSR
) {
297 update_mask
|= MSR_EFER_FFXSR
;
299 if (env
->cpuid_ext2_features
& CPUID_EXT2_NX
) {
300 update_mask
|= MSR_EFER_NXE
;
302 if (env
->cpuid_ext3_features
& CPUID_EXT3_SVM
) {
303 update_mask
|= MSR_EFER_SVME
;
305 if (env
->cpuid_ext2_features
& CPUID_EXT2_FFXSR
) {
306 update_mask
|= MSR_EFER_FFXSR
;
308 cpu_load_efer(env
, (env
->efer
& ~update_mask
) |
309 (val
& update_mask
));
318 case MSR_VM_HSAVE_PA
:
332 env
->segs
[R_FS
].base
= val
;
335 env
->segs
[R_GS
].base
= val
;
337 case MSR_KERNELGSBASE
:
338 env
->kernelgsbase
= val
;
341 case MSR_MTRRphysBase(0):
342 case MSR_MTRRphysBase(1):
343 case MSR_MTRRphysBase(2):
344 case MSR_MTRRphysBase(3):
345 case MSR_MTRRphysBase(4):
346 case MSR_MTRRphysBase(5):
347 case MSR_MTRRphysBase(6):
348 case MSR_MTRRphysBase(7):
349 env
->mtrr_var
[((uint32_t)ECX
- MSR_MTRRphysBase(0)) / 2].base
= val
;
351 case MSR_MTRRphysMask(0):
352 case MSR_MTRRphysMask(1):
353 case MSR_MTRRphysMask(2):
354 case MSR_MTRRphysMask(3):
355 case MSR_MTRRphysMask(4):
356 case MSR_MTRRphysMask(5):
357 case MSR_MTRRphysMask(6):
358 case MSR_MTRRphysMask(7):
359 env
->mtrr_var
[((uint32_t)ECX
- MSR_MTRRphysMask(0)) / 2].mask
= val
;
361 case MSR_MTRRfix64K_00000
:
362 env
->mtrr_fixed
[(uint32_t)ECX
- MSR_MTRRfix64K_00000
] = val
;
364 case MSR_MTRRfix16K_80000
:
365 case MSR_MTRRfix16K_A0000
:
366 env
->mtrr_fixed
[(uint32_t)ECX
- MSR_MTRRfix16K_80000
+ 1] = val
;
368 case MSR_MTRRfix4K_C0000
:
369 case MSR_MTRRfix4K_C8000
:
370 case MSR_MTRRfix4K_D0000
:
371 case MSR_MTRRfix4K_D8000
:
372 case MSR_MTRRfix4K_E0000
:
373 case MSR_MTRRfix4K_E8000
:
374 case MSR_MTRRfix4K_F0000
:
375 case MSR_MTRRfix4K_F8000
:
376 env
->mtrr_fixed
[(uint32_t)ECX
- MSR_MTRRfix4K_C0000
+ 3] = val
;
378 case MSR_MTRRdefType
:
379 env
->mtrr_deftype
= val
;
382 env
->mcg_status
= val
;
385 if ((env
->mcg_cap
& MCG_CTL_P
)
386 && (val
== 0 || val
== ~(uint64_t)0)) {
393 case MSR_IA32_MISC_ENABLE
:
394 env
->msr_ia32_misc_enable
= val
;
397 if ((uint32_t)ECX
>= MSR_MC0_CTL
398 && (uint32_t)ECX
< MSR_MC0_CTL
+ (4 * env
->mcg_cap
& 0xff)) {
399 uint32_t offset
= (uint32_t)ECX
- MSR_MC0_CTL
;
400 if ((offset
& 0x3) != 0
401 || (val
== 0 || val
== ~(uint64_t)0)) {
402 env
->mce_banks
[offset
] = val
;
406 /* XXX: exception? */
411 void helper_rdmsr(CPUX86State
*env
)
415 cpu_svm_check_intercept_param(env
, SVM_EXIT_MSR
, 0);
417 switch ((uint32_t)ECX
) {
418 case MSR_IA32_SYSENTER_CS
:
419 val
= env
->sysenter_cs
;
421 case MSR_IA32_SYSENTER_ESP
:
422 val
= env
->sysenter_esp
;
424 case MSR_IA32_SYSENTER_EIP
:
425 val
= env
->sysenter_eip
;
427 case MSR_IA32_APICBASE
:
428 val
= cpu_get_apic_base(env
->apic_state
);
439 case MSR_VM_HSAVE_PA
:
442 case MSR_IA32_PERF_STATUS
:
443 /* tsc_increment_by_tick */
446 val
|= (((uint64_t)4ULL) << 40);
459 val
= env
->segs
[R_FS
].base
;
462 val
= env
->segs
[R_GS
].base
;
464 case MSR_KERNELGSBASE
:
465 val
= env
->kernelgsbase
;
471 case MSR_MTRRphysBase(0):
472 case MSR_MTRRphysBase(1):
473 case MSR_MTRRphysBase(2):
474 case MSR_MTRRphysBase(3):
475 case MSR_MTRRphysBase(4):
476 case MSR_MTRRphysBase(5):
477 case MSR_MTRRphysBase(6):
478 case MSR_MTRRphysBase(7):
479 val
= env
->mtrr_var
[((uint32_t)ECX
- MSR_MTRRphysBase(0)) / 2].base
;
481 case MSR_MTRRphysMask(0):
482 case MSR_MTRRphysMask(1):
483 case MSR_MTRRphysMask(2):
484 case MSR_MTRRphysMask(3):
485 case MSR_MTRRphysMask(4):
486 case MSR_MTRRphysMask(5):
487 case MSR_MTRRphysMask(6):
488 case MSR_MTRRphysMask(7):
489 val
= env
->mtrr_var
[((uint32_t)ECX
- MSR_MTRRphysMask(0)) / 2].mask
;
491 case MSR_MTRRfix64K_00000
:
492 val
= env
->mtrr_fixed
[0];
494 case MSR_MTRRfix16K_80000
:
495 case MSR_MTRRfix16K_A0000
:
496 val
= env
->mtrr_fixed
[(uint32_t)ECX
- MSR_MTRRfix16K_80000
+ 1];
498 case MSR_MTRRfix4K_C0000
:
499 case MSR_MTRRfix4K_C8000
:
500 case MSR_MTRRfix4K_D0000
:
501 case MSR_MTRRfix4K_D8000
:
502 case MSR_MTRRfix4K_E0000
:
503 case MSR_MTRRfix4K_E8000
:
504 case MSR_MTRRfix4K_F0000
:
505 case MSR_MTRRfix4K_F8000
:
506 val
= env
->mtrr_fixed
[(uint32_t)ECX
- MSR_MTRRfix4K_C0000
+ 3];
508 case MSR_MTRRdefType
:
509 val
= env
->mtrr_deftype
;
512 if (env
->cpuid_features
& CPUID_MTRR
) {
513 val
= MSR_MTRRcap_VCNT
| MSR_MTRRcap_FIXRANGE_SUPPORT
|
514 MSR_MTRRcap_WC_SUPPORTED
;
516 /* XXX: exception? */
524 if (env
->mcg_cap
& MCG_CTL_P
) {
531 val
= env
->mcg_status
;
533 case MSR_IA32_MISC_ENABLE
:
534 val
= env
->msr_ia32_misc_enable
;
537 if ((uint32_t)ECX
>= MSR_MC0_CTL
538 && (uint32_t)ECX
< MSR_MC0_CTL
+ (4 * env
->mcg_cap
& 0xff)) {
539 uint32_t offset
= (uint32_t)ECX
- MSR_MC0_CTL
;
540 val
= env
->mce_banks
[offset
];
543 /* XXX: exception? */
547 EAX
= (uint32_t)(val
);
548 EDX
= (uint32_t)(val
>> 32);
552 static void do_hlt(CPUX86State
*env
)
554 env
->hflags
&= ~HF_INHIBIT_IRQ_MASK
; /* needed if sti is just before */
556 env
->exception_index
= EXCP_HLT
;
560 void helper_hlt(CPUX86State
*env
, int next_eip_addend
)
562 cpu_svm_check_intercept_param(env
, SVM_EXIT_HLT
, 0);
563 EIP
+= next_eip_addend
;
568 void helper_monitor(CPUX86State
*env
, target_ulong ptr
)
570 if ((uint32_t)ECX
!= 0) {
571 raise_exception(env
, EXCP0D_GPF
);
573 /* XXX: store address? */
574 cpu_svm_check_intercept_param(env
, SVM_EXIT_MONITOR
, 0);
577 void helper_mwait(CPUX86State
*env
, int next_eip_addend
)
579 if ((uint32_t)ECX
!= 0) {
580 raise_exception(env
, EXCP0D_GPF
);
582 cpu_svm_check_intercept_param(env
, SVM_EXIT_MWAIT
, 0);
583 EIP
+= next_eip_addend
;
585 /* XXX: not complete but not completely erroneous */
586 if (env
->cpu_index
!= 0 || env
->next_cpu
!= NULL
) {
587 /* more than one CPU: do not sleep because another CPU may
594 void helper_debug(CPUX86State
*env
)
596 env
->exception_index
= EXCP_DEBUG
;