xref: /qemu/target/i386/tcg/tcg-cpu.c (revision b6aeb8d243c5ab8b914b55f0036e8289a99322c8)
1 /*
2  * i386 TCG cpu class initialization
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 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 "cpu.h"
22 #include "helper-tcg.h"
23 #include "qemu/accel.h"
24 #include "accel/accel-cpu-target.h"
25 #include "exec/translation-block.h"
26 #include "exec/target_page.h"
27 #include "accel/tcg/cpu-ops.h"
28 #include "tcg-cpu.h"
29 
30 /* Frob eflags into and out of the CPU temporary format.  */
31 
32 static void x86_cpu_exec_enter(CPUState *cs)
33 {
34     X86CPU *cpu = X86_CPU(cs);
35     CPUX86State *env = &cpu->env;
36 
37     CC_SRC = env->eflags & (CC_O | CC_S | CC_Z | CC_A | CC_P | CC_C);
38     env->df = 1 - (2 * ((env->eflags >> 10) & 1));
39     CC_OP = CC_OP_EFLAGS;
40     env->eflags &= ~(DF_MASK | CC_O | CC_S | CC_Z | CC_A | CC_P | CC_C);
41 }
42 
43 static void x86_cpu_exec_exit(CPUState *cs)
44 {
45     X86CPU *cpu = X86_CPU(cs);
46     CPUX86State *env = &cpu->env;
47 
48     env->eflags = cpu_compute_eflags(env);
49 }
50 
51 void cpu_get_tb_cpu_state(CPUX86State *env, vaddr *pc,
52                           uint64_t *cs_base, uint32_t *flags)
53 {
54     *flags = env->hflags |
55         (env->eflags & (IOPL_MASK | TF_MASK | RF_MASK | VM_MASK | AC_MASK));
56     if (env->hflags & HF_CS64_MASK) {
57         *cs_base = 0;
58         *pc = env->eip;
59     } else {
60         *cs_base = env->segs[R_CS].base;
61         *pc = (uint32_t)(*cs_base + env->eip);
62     }
63 }
64 
65 static void x86_cpu_synchronize_from_tb(CPUState *cs,
66                                         const TranslationBlock *tb)
67 {
68     /* The instruction pointer is always up to date with CF_PCREL. */
69     if (!(tb_cflags(tb) & CF_PCREL)) {
70         CPUX86State *env = cpu_env(cs);
71 
72         if (tb->flags & HF_CS64_MASK) {
73             env->eip = tb->pc;
74         } else {
75             env->eip = (uint32_t)(tb->pc - tb->cs_base);
76         }
77     }
78 }
79 
80 static void x86_restore_state_to_opc(CPUState *cs,
81                                      const TranslationBlock *tb,
82                                      const uint64_t *data)
83 {
84     X86CPU *cpu = X86_CPU(cs);
85     CPUX86State *env = &cpu->env;
86     int cc_op = data[1];
87     uint64_t new_pc;
88 
89     if (tb_cflags(tb) & CF_PCREL) {
90         /*
91          * data[0] in PC-relative TBs is also a linear address, i.e. an address with
92          * the CS base added, because it is not guaranteed that EIP bits 12 and higher
93          * stay the same across the translation block.  Add the CS base back before
94          * replacing the low bits, and subtract it below just like for !CF_PCREL.
95          */
96         uint64_t pc = env->eip + tb->cs_base;
97         new_pc = (pc & TARGET_PAGE_MASK) | data[0];
98     } else {
99         new_pc = data[0];
100     }
101     if (tb->flags & HF_CS64_MASK) {
102         env->eip = new_pc;
103     } else {
104         env->eip = (uint32_t)(new_pc - tb->cs_base);
105     }
106 
107     if (cc_op != CC_OP_DYNAMIC) {
108         env->cc_op = cc_op;
109     }
110 }
111 
112 int x86_mmu_index_pl(CPUX86State *env, unsigned pl)
113 {
114     int mmu_index_32 = (env->hflags & HF_CS64_MASK) ? 0 : 1;
115     int mmu_index_base =
116         pl == 3 ? MMU_USER64_IDX :
117         !(env->hflags & HF_SMAP_MASK) ? MMU_KNOSMAP64_IDX :
118         (env->eflags & AC_MASK) ? MMU_KNOSMAP64_IDX : MMU_KSMAP64_IDX;
119 
120     return mmu_index_base + mmu_index_32;
121 }
122 
123 static int x86_cpu_mmu_index(CPUState *cs, bool ifetch)
124 {
125     CPUX86State *env = cpu_env(cs);
126     return x86_mmu_index_pl(env, env->hflags & HF_CPL_MASK);
127 }
128 
129 #ifndef CONFIG_USER_ONLY
130 static bool x86_debug_check_breakpoint(CPUState *cs)
131 {
132     X86CPU *cpu = X86_CPU(cs);
133     CPUX86State *env = &cpu->env;
134 
135     /* RF disables all architectural breakpoints. */
136     return !(env->eflags & RF_MASK);
137 }
138 
139 static void x86_cpu_exec_reset(CPUState *cs)
140 {
141     CPUArchState *env = cpu_env(cs);
142 
143     cpu_svm_check_intercept_param(env, SVM_EXIT_INIT, 0, 0);
144     do_cpu_init(env_archcpu(env));
145     cs->exception_index = EXCP_HALTED;
146 }
147 #endif
148 
149 const TCGCPUOps x86_tcg_ops = {
150     .mttcg_supported = true,
151     .precise_smc = true,
152     /*
153      * The x86 has a strong memory model with some store-after-load re-ordering
154      */
155     .guest_default_memory_order = TCG_MO_ALL & ~TCG_MO_ST_LD,
156     .initialize = tcg_x86_init,
157     .translate_code = x86_translate_code,
158     .synchronize_from_tb = x86_cpu_synchronize_from_tb,
159     .restore_state_to_opc = x86_restore_state_to_opc,
160     .mmu_index = x86_cpu_mmu_index,
161     .cpu_exec_enter = x86_cpu_exec_enter,
162     .cpu_exec_exit = x86_cpu_exec_exit,
163 #ifdef CONFIG_USER_ONLY
164     .fake_user_interrupt = x86_cpu_do_interrupt,
165     .record_sigsegv = x86_cpu_record_sigsegv,
166     .record_sigbus = x86_cpu_record_sigbus,
167 #else
168     .tlb_fill = x86_cpu_tlb_fill,
169     .do_interrupt = x86_cpu_do_interrupt,
170     .cpu_exec_halt = x86_cpu_exec_halt,
171     .cpu_exec_interrupt = x86_cpu_exec_interrupt,
172     .cpu_exec_reset = x86_cpu_exec_reset,
173     .do_unaligned_access = x86_cpu_do_unaligned_access,
174     .debug_excp_handler = breakpoint_handler,
175     .debug_check_breakpoint = x86_debug_check_breakpoint,
176     .need_replay_interrupt = x86_need_replay_interrupt,
177 #endif /* !CONFIG_USER_ONLY */
178 };
179 
180 static void x86_tcg_cpu_xsave_init(void)
181 {
182 #define XO(bit, field) \
183     x86_ext_save_areas[bit].offset = offsetof(X86XSaveArea, field);
184 
185     XO(XSTATE_FP_BIT, legacy);
186     XO(XSTATE_SSE_BIT, legacy);
187     XO(XSTATE_YMM_BIT, avx_state);
188     XO(XSTATE_BNDREGS_BIT, bndreg_state);
189     XO(XSTATE_BNDCSR_BIT, bndcsr_state);
190     XO(XSTATE_OPMASK_BIT, opmask_state);
191     XO(XSTATE_ZMM_Hi256_BIT, zmm_hi256_state);
192     XO(XSTATE_Hi16_ZMM_BIT, hi16_zmm_state);
193     XO(XSTATE_PKRU_BIT, pkru_state);
194 
195 #undef XO
196 }
197 
198 /*
199  * TCG-specific defaults that override cpudef models when using TCG.
200  * Only for builtin_x86_defs models initialized with x86_register_cpudef_types.
201  */
202 static PropValue x86_tcg_default_props[] = {
203     { "vme", "off" },
204     { NULL, NULL },
205 };
206 
207 static void x86_tcg_cpu_instance_init(CPUState *cs)
208 {
209     X86CPU *cpu = X86_CPU(cs);
210     X86CPUClass *xcc = X86_CPU_GET_CLASS(cpu);
211 
212     if (xcc->model) {
213         /* Special cases not set in the X86CPUDefinition structs: */
214         x86_cpu_apply_props(cpu, x86_tcg_default_props);
215     }
216 
217     x86_tcg_cpu_xsave_init();
218 }
219 
220 static void x86_tcg_cpu_accel_class_init(ObjectClass *oc, const void *data)
221 {
222     AccelCPUClass *acc = ACCEL_CPU_CLASS(oc);
223 
224 #ifndef CONFIG_USER_ONLY
225     acc->cpu_target_realize = tcg_cpu_realizefn;
226 #endif /* CONFIG_USER_ONLY */
227 
228     acc->cpu_instance_init = x86_tcg_cpu_instance_init;
229 }
230 static const TypeInfo x86_tcg_cpu_accel_type_info = {
231     .name = ACCEL_CPU_NAME("tcg"),
232 
233     .parent = TYPE_ACCEL_CPU,
234     .class_init = x86_tcg_cpu_accel_class_init,
235     .abstract = true,
236 };
237 static void x86_tcg_cpu_accel_register_types(void)
238 {
239     type_register_static(&x86_tcg_cpu_accel_type_info);
240 }
241 type_init(x86_tcg_cpu_accel_register_types);
242