blob: d1d38daa93fbf723490f09a38b2e5e6eca2dcfd1 [file] [log] [blame]
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -05001/*
2 * This program is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License, version 2, as
4 * published by the Free Software Foundation.
5 *
6 * This program is distributed in the hope that it will be useful,
7 * but WITHOUT ANY WARRANTY; without even the implied warranty of
8 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
9 * GNU General Public License for more details.
10 *
11 * You should have received a copy of the GNU General Public License
12 * along with this program; if not, write to the Free Software
13 * Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
14 *
15 * Copyright IBM Corp. 2007
16 *
17 * Authors: Hollis Blanchard <hollisb@us.ibm.com>
18 */
19
20#include <linux/jiffies.h>
21#include <linux/timer.h>
22#include <linux/types.h>
23#include <linux/string.h>
24#include <linux/kvm_host.h>
25
Hollis Blanchard75f74f02008-11-05 09:36:16 -060026#include <asm/reg.h>
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050027#include <asm/time.h>
28#include <asm/byteorder.h>
29#include <asm/kvm_ppc.h>
Hollis Blanchardc381a042008-11-05 09:36:15 -060030#include <asm/disassemble.h>
Hollis Blanchard73e75b42008-12-02 15:51:57 -060031#include "timing.h"
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050032
Hollis Blanchard75f74f02008-11-05 09:36:16 -060033void kvmppc_emulate_dec(struct kvm_vcpu *vcpu)
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050034{
35 if (vcpu->arch.tcr & TCR_DIE) {
36 /* The decrementer ticks at the same rate as the timebase, so
37 * that's how we convert the guest DEC value to the number of
38 * host ticks. */
39 unsigned long nr_jiffies;
40
41 nr_jiffies = vcpu->arch.dec / tb_ticks_per_jiffy;
42 mod_timer(&vcpu->arch.dec_timer,
43 get_jiffies_64() + nr_jiffies);
44 } else {
45 del_timer(&vcpu->arch.dec_timer);
46 }
47}
48
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050049/* XXX to do:
50 * lhax
51 * lhaux
52 * lswx
53 * lswi
54 * stswx
55 * stswi
56 * lha
57 * lhau
58 * lmw
59 * stmw
60 *
61 * XXX is_bigendian should depend on MMU mapping or MSR[LE]
62 */
Hollis Blanchard75f74f02008-11-05 09:36:16 -060063/* XXX Should probably auto-generate instruction decoding for a particular core
64 * from opcode tables in the future. */
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050065int kvmppc_emulate_instruction(struct kvm_run *run, struct kvm_vcpu *vcpu)
66{
67 u32 inst = vcpu->arch.last_inst;
68 u32 ea;
69 int ra;
70 int rb;
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050071 int rs;
72 int rt;
73 int sprn;
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050074 enum emulation_result emulated = EMULATE_DONE;
75 int advance = 1;
76
Hollis Blanchard73e75b42008-12-02 15:51:57 -060077 /* this default type might be overwritten by subcategories */
78 kvmppc_set_exit_type(vcpu, EMULATED_INST_EXITS);
79
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050080 switch (get_op(inst)) {
Hollis Blanchardfcfdbd22008-11-05 09:36:24 -060081 case 3: /* trap */
82 vcpu->arch.esr |= ESR_PTR;
Hollis Blanchard9dd921c2008-11-05 09:36:14 -060083 kvmppc_core_queue_program(vcpu);
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050084 advance = 0;
85 break;
86
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050087 case 31:
88 switch (get_xop(inst)) {
89
Hollis Blanchardac3cd342008-05-21 18:22:52 -050090 case 23: /* lwzx */
91 rt = get_rt(inst);
92 emulated = kvmppc_handle_load(run, vcpu, rt, 4, 1);
93 break;
94
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -050095 case 87: /* lbzx */
96 rt = get_rt(inst);
97 emulated = kvmppc_handle_load(run, vcpu, rt, 1, 1);
98 break;
99
Hollis Blanchardac3cd342008-05-21 18:22:52 -0500100 case 151: /* stwx */
101 rs = get_rs(inst);
102 emulated = kvmppc_handle_store(run, vcpu,
103 vcpu->arch.gpr[rs],
104 4, 1);
105 break;
106
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500107 case 215: /* stbx */
108 rs = get_rs(inst);
109 emulated = kvmppc_handle_store(run, vcpu,
110 vcpu->arch.gpr[rs],
111 1, 1);
112 break;
113
114 case 247: /* stbux */
115 rs = get_rs(inst);
116 ra = get_ra(inst);
117 rb = get_rb(inst);
118
119 ea = vcpu->arch.gpr[rb];
120 if (ra)
121 ea += vcpu->arch.gpr[ra];
122
123 emulated = kvmppc_handle_store(run, vcpu,
124 vcpu->arch.gpr[rs],
125 1, 1);
126 vcpu->arch.gpr[rs] = ea;
127 break;
128
129 case 279: /* lhzx */
130 rt = get_rt(inst);
131 emulated = kvmppc_handle_load(run, vcpu, rt, 2, 1);
132 break;
133
134 case 311: /* lhzux */
135 rt = get_rt(inst);
136 ra = get_ra(inst);
137 rb = get_rb(inst);
138
139 ea = vcpu->arch.gpr[rb];
140 if (ra)
141 ea += vcpu->arch.gpr[ra];
142
143 emulated = kvmppc_handle_load(run, vcpu, rt, 2, 1);
144 vcpu->arch.gpr[ra] = ea;
145 break;
146
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500147 case 339: /* mfspr */
148 sprn = get_sprn(inst);
149 rt = get_rt(inst);
150
151 switch (sprn) {
152 case SPRN_SRR0:
153 vcpu->arch.gpr[rt] = vcpu->arch.srr0; break;
154 case SPRN_SRR1:
155 vcpu->arch.gpr[rt] = vcpu->arch.srr1; break;
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500156 case SPRN_PVR:
157 vcpu->arch.gpr[rt] = vcpu->arch.pvr; break;
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500158
159 /* Note: mftb and TBRL/TBWL are user-accessible, so
160 * the guest can always access the real TB anyways.
161 * In fact, we probably will never see these traps. */
162 case SPRN_TBWL:
163 vcpu->arch.gpr[rt] = mftbl(); break;
164 case SPRN_TBWU:
165 vcpu->arch.gpr[rt] = mftbu(); break;
166
167 case SPRN_SPRG0:
168 vcpu->arch.gpr[rt] = vcpu->arch.sprg0; break;
169 case SPRN_SPRG1:
170 vcpu->arch.gpr[rt] = vcpu->arch.sprg1; break;
171 case SPRN_SPRG2:
172 vcpu->arch.gpr[rt] = vcpu->arch.sprg2; break;
173 case SPRN_SPRG3:
174 vcpu->arch.gpr[rt] = vcpu->arch.sprg3; break;
175 /* Note: SPRG4-7 are user-readable, so we don't get
176 * a trap. */
177
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500178 default:
Hollis Blanchard75f74f02008-11-05 09:36:16 -0600179 emulated = kvmppc_core_emulate_mfspr(vcpu, sprn, rt);
180 if (emulated == EMULATE_FAIL) {
181 printk("mfspr: unknown spr %x\n", sprn);
182 vcpu->arch.gpr[rt] = 0;
183 }
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500184 break;
185 }
186 break;
187
188 case 407: /* sthx */
189 rs = get_rs(inst);
190 ra = get_ra(inst);
191 rb = get_rb(inst);
192
193 emulated = kvmppc_handle_store(run, vcpu,
194 vcpu->arch.gpr[rs],
195 2, 1);
196 break;
197
198 case 439: /* sthux */
199 rs = get_rs(inst);
200 ra = get_ra(inst);
201 rb = get_rb(inst);
202
203 ea = vcpu->arch.gpr[rb];
204 if (ra)
205 ea += vcpu->arch.gpr[ra];
206
207 emulated = kvmppc_handle_store(run, vcpu,
208 vcpu->arch.gpr[rs],
209 2, 1);
210 vcpu->arch.gpr[ra] = ea;
211 break;
212
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500213 case 467: /* mtspr */
214 sprn = get_sprn(inst);
215 rs = get_rs(inst);
216 switch (sprn) {
217 case SPRN_SRR0:
218 vcpu->arch.srr0 = vcpu->arch.gpr[rs]; break;
219 case SPRN_SRR1:
220 vcpu->arch.srr1 = vcpu->arch.gpr[rs]; break;
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500221
222 /* XXX We need to context-switch the timebase for
223 * watchdog and FIT. */
224 case SPRN_TBWL: break;
225 case SPRN_TBWU: break;
226
227 case SPRN_DEC:
228 vcpu->arch.dec = vcpu->arch.gpr[rs];
229 kvmppc_emulate_dec(vcpu);
230 break;
231
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500232 case SPRN_SPRG0:
233 vcpu->arch.sprg0 = vcpu->arch.gpr[rs]; break;
234 case SPRN_SPRG1:
235 vcpu->arch.sprg1 = vcpu->arch.gpr[rs]; break;
236 case SPRN_SPRG2:
237 vcpu->arch.sprg2 = vcpu->arch.gpr[rs]; break;
238 case SPRN_SPRG3:
239 vcpu->arch.sprg3 = vcpu->arch.gpr[rs]; break;
240
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500241 default:
Hollis Blanchard75f74f02008-11-05 09:36:16 -0600242 emulated = kvmppc_core_emulate_mtspr(vcpu, sprn, rs);
243 if (emulated == EMULATE_FAIL)
244 printk("mtspr: unknown spr %x\n", sprn);
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500245 break;
246 }
247 break;
248
249 case 470: /* dcbi */
250 /* Do nothing. The guest is performing dcbi because
251 * hardware DMA is not snooped by the dcache, but
252 * emulated DMA either goes through the dcache as
253 * normal writes, or the host kernel has handled dcache
254 * coherence. */
255 break;
256
257 case 534: /* lwbrx */
258 rt = get_rt(inst);
259 emulated = kvmppc_handle_load(run, vcpu, rt, 4, 0);
260 break;
261
262 case 566: /* tlbsync */
263 break;
264
265 case 662: /* stwbrx */
266 rs = get_rs(inst);
267 ra = get_ra(inst);
268 rb = get_rb(inst);
269
270 emulated = kvmppc_handle_store(run, vcpu,
271 vcpu->arch.gpr[rs],
272 4, 0);
273 break;
274
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500275 case 790: /* lhbrx */
276 rt = get_rt(inst);
277 emulated = kvmppc_handle_load(run, vcpu, rt, 2, 0);
278 break;
279
280 case 918: /* sthbrx */
281 rs = get_rs(inst);
282 ra = get_ra(inst);
283 rb = get_rb(inst);
284
285 emulated = kvmppc_handle_store(run, vcpu,
286 vcpu->arch.gpr[rs],
287 2, 0);
288 break;
289
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500290 default:
Hollis Blanchard75f74f02008-11-05 09:36:16 -0600291 /* Attempt core-specific emulation below. */
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500292 emulated = EMULATE_FAIL;
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500293 }
294 break;
295
296 case 32: /* lwz */
297 rt = get_rt(inst);
298 emulated = kvmppc_handle_load(run, vcpu, rt, 4, 1);
299 break;
300
301 case 33: /* lwzu */
302 ra = get_ra(inst);
303 rt = get_rt(inst);
304 emulated = kvmppc_handle_load(run, vcpu, rt, 4, 1);
305 vcpu->arch.gpr[ra] = vcpu->arch.paddr_accessed;
306 break;
307
308 case 34: /* lbz */
309 rt = get_rt(inst);
310 emulated = kvmppc_handle_load(run, vcpu, rt, 1, 1);
311 break;
312
313 case 35: /* lbzu */
314 ra = get_ra(inst);
315 rt = get_rt(inst);
316 emulated = kvmppc_handle_load(run, vcpu, rt, 1, 1);
317 vcpu->arch.gpr[ra] = vcpu->arch.paddr_accessed;
318 break;
319
320 case 36: /* stw */
321 rs = get_rs(inst);
322 emulated = kvmppc_handle_store(run, vcpu, vcpu->arch.gpr[rs],
323 4, 1);
324 break;
325
326 case 37: /* stwu */
327 ra = get_ra(inst);
328 rs = get_rs(inst);
329 emulated = kvmppc_handle_store(run, vcpu, vcpu->arch.gpr[rs],
330 4, 1);
331 vcpu->arch.gpr[ra] = vcpu->arch.paddr_accessed;
332 break;
333
334 case 38: /* stb */
335 rs = get_rs(inst);
336 emulated = kvmppc_handle_store(run, vcpu, vcpu->arch.gpr[rs],
337 1, 1);
338 break;
339
340 case 39: /* stbu */
341 ra = get_ra(inst);
342 rs = get_rs(inst);
343 emulated = kvmppc_handle_store(run, vcpu, vcpu->arch.gpr[rs],
344 1, 1);
345 vcpu->arch.gpr[ra] = vcpu->arch.paddr_accessed;
346 break;
347
348 case 40: /* lhz */
349 rt = get_rt(inst);
350 emulated = kvmppc_handle_load(run, vcpu, rt, 2, 1);
351 break;
352
353 case 41: /* lhzu */
354 ra = get_ra(inst);
355 rt = get_rt(inst);
356 emulated = kvmppc_handle_load(run, vcpu, rt, 2, 1);
357 vcpu->arch.gpr[ra] = vcpu->arch.paddr_accessed;
358 break;
359
360 case 44: /* sth */
361 rs = get_rs(inst);
362 emulated = kvmppc_handle_store(run, vcpu, vcpu->arch.gpr[rs],
363 2, 1);
364 break;
365
366 case 45: /* sthu */
367 ra = get_ra(inst);
368 rs = get_rs(inst);
369 emulated = kvmppc_handle_store(run, vcpu, vcpu->arch.gpr[rs],
370 2, 1);
371 vcpu->arch.gpr[ra] = vcpu->arch.paddr_accessed;
372 break;
373
374 default:
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500375 emulated = EMULATE_FAIL;
Hollis Blanchard75f74f02008-11-05 09:36:16 -0600376 }
377
378 if (emulated == EMULATE_FAIL) {
379 emulated = kvmppc_core_emulate_op(run, vcpu, inst, &advance);
380 if (emulated == EMULATE_FAIL) {
381 advance = 0;
382 printk(KERN_ERR "Couldn't emulate instruction 0x%08x "
383 "(op %d xop %d)\n", inst, get_op(inst), get_xop(inst));
384 }
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500385 }
386
Hollis Blanchard5cf8ca22008-11-05 09:36:19 -0600387 KVMTRACE_3D(PPC_INSTR, vcpu, inst, (int)vcpu->arch.pc, emulated, entryexit);
Christian Ehrhardt3b4bd792008-07-14 14:00:04 +0200388
Hollis Blanchardbbf45ba2008-04-16 23:28:09 -0500389 if (advance)
390 vcpu->arch.pc += 4; /* Advance past emulated instruction. */
391
392 return emulated;
393}