Update
[qemu-kvm/fedora.git] / target-ppc / op.c
blob82750c6ef3025ad11876a45a30ce150bc3071cd3
1 /*
2 * PowerPC emulation micro-operations for qemu.
4 * Copyright (c) 2003-2007 Jocelyn Mayer
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, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 //#define DEBUG_OP
23 #include "config.h"
24 #include "exec.h"
25 #include "host-utils.h"
26 #include "helper_regs.h"
27 #include "op_helper.h"
29 #if !defined(CONFIG_USER_ONLY)
30 /* Segment registers load and store */
31 void OPPROTO op_load_sr (void)
33 T0 = env->sr[T1];
34 RETURN();
37 void OPPROTO op_store_sr (void)
39 do_store_sr(env, T1, T0);
40 RETURN();
43 #if defined(TARGET_PPC64)
44 void OPPROTO op_load_slb (void)
46 T0 = ppc_load_slb(env, T1);
47 RETURN();
50 void OPPROTO op_store_slb (void)
52 ppc_store_slb(env, T1, T0);
53 RETURN();
55 #endif /* defined(TARGET_PPC64) */
57 void OPPROTO op_load_sdr1 (void)
59 T0 = env->sdr1;
60 RETURN();
63 void OPPROTO op_store_sdr1 (void)
65 do_store_sdr1(env, T0);
66 RETURN();
69 #if defined (TARGET_PPC64)
70 void OPPROTO op_load_asr (void)
72 T0 = env->asr;
73 RETURN();
76 void OPPROTO op_store_asr (void)
78 ppc_store_asr(env, T0);
79 RETURN();
81 #endif
83 void OPPROTO op_load_msr (void)
85 T0 = env->msr;
86 RETURN();
89 void OPPROTO op_store_msr (void)
91 do_store_msr();
92 RETURN();
95 #if defined (TARGET_PPC64)
96 void OPPROTO op_store_msr_32 (void)
98 T0 = (env->msr & ~0xFFFFFFFFULL) | (T0 & 0xFFFFFFFF);
99 do_store_msr();
100 RETURN();
102 #endif
104 void OPPROTO op_update_riee (void)
106 /* We don't call do_store_msr here as we won't trigger
107 * any special case nor change hflags
109 T0 &= (1 << MSR_RI) | (1 << MSR_EE);
110 env->msr &= ~(1 << MSR_RI) | (1 << MSR_EE);
111 env->msr |= T0;
112 RETURN();
114 #endif
116 /* SPR */
117 void OPPROTO op_load_spr (void)
119 T0 = env->spr[PARAM1];
120 RETURN();
123 void OPPROTO op_store_spr (void)
125 env->spr[PARAM1] = T0;
126 RETURN();
129 void OPPROTO op_load_dump_spr (void)
131 T0 = ppc_load_dump_spr(PARAM1);
132 RETURN();
135 void OPPROTO op_store_dump_spr (void)
137 ppc_store_dump_spr(PARAM1, T0);
138 RETURN();
141 void OPPROTO op_mask_spr (void)
143 env->spr[PARAM1] &= ~T0;
144 RETURN();
147 void OPPROTO op_load_tbl (void)
149 T0 = cpu_ppc_load_tbl(env);
150 RETURN();
153 void OPPROTO op_load_tbu (void)
155 T0 = cpu_ppc_load_tbu(env);
156 RETURN();
159 void OPPROTO op_load_atbl (void)
161 T0 = cpu_ppc_load_atbl(env);
162 RETURN();
165 void OPPROTO op_load_atbu (void)
167 T0 = cpu_ppc_load_atbu(env);
168 RETURN();
171 #if !defined(CONFIG_USER_ONLY)
172 void OPPROTO op_store_tbl (void)
174 cpu_ppc_store_tbl(env, T0);
175 RETURN();
178 void OPPROTO op_store_tbu (void)
180 cpu_ppc_store_tbu(env, T0);
181 RETURN();
184 void OPPROTO op_store_atbl (void)
186 cpu_ppc_store_atbl(env, T0);
187 RETURN();
190 void OPPROTO op_store_atbu (void)
192 cpu_ppc_store_atbu(env, T0);
193 RETURN();
196 void OPPROTO op_load_decr (void)
198 T0 = cpu_ppc_load_decr(env);
199 RETURN();
202 void OPPROTO op_store_decr (void)
204 cpu_ppc_store_decr(env, T0);
205 RETURN();
208 void OPPROTO op_load_ibat (void)
210 T0 = env->IBAT[PARAM1][PARAM2];
211 RETURN();
214 void OPPROTO op_store_ibatu (void)
216 do_store_ibatu(env, PARAM1, T0);
217 RETURN();
220 void OPPROTO op_store_ibatl (void)
222 #if 1
223 env->IBAT[1][PARAM1] = T0;
224 #else
225 do_store_ibatl(env, PARAM1, T0);
226 #endif
227 RETURN();
230 void OPPROTO op_load_dbat (void)
232 T0 = env->DBAT[PARAM1][PARAM2];
233 RETURN();
236 void OPPROTO op_store_dbatu (void)
238 do_store_dbatu(env, PARAM1, T0);
239 RETURN();
242 void OPPROTO op_store_dbatl (void)
244 #if 1
245 env->DBAT[1][PARAM1] = T0;
246 #else
247 do_store_dbatl(env, PARAM1, T0);
248 #endif
249 RETURN();
251 #endif /* !defined(CONFIG_USER_ONLY) */
253 /*** Integer shift ***/
254 void OPPROTO op_srli_T1 (void)
256 T1 = (uint32_t)T1 >> PARAM1;
257 RETURN();
260 /* Return from interrupt */
261 #if !defined(CONFIG_USER_ONLY)
262 /* Exception vectors */
263 void OPPROTO op_store_excp_prefix (void)
265 T0 &= env->ivpr_mask;
266 env->excp_prefix = T0;
267 RETURN();
270 void OPPROTO op_store_excp_vector (void)
272 T0 &= env->ivor_mask;
273 env->excp_vectors[PARAM1] = T0;
274 RETURN();
276 #endif
278 #if !defined(CONFIG_USER_ONLY)
279 /* tlbia */
280 void OPPROTO op_tlbia (void)
282 ppc_tlb_invalidate_all(env);
283 RETURN();
286 /* tlbie */
287 void OPPROTO op_tlbie (void)
289 ppc_tlb_invalidate_one(env, (uint32_t)T0);
290 RETURN();
293 #if defined(TARGET_PPC64)
294 void OPPROTO op_tlbie_64 (void)
296 ppc_tlb_invalidate_one(env, T0);
297 RETURN();
299 #endif
301 #if defined(TARGET_PPC64)
302 void OPPROTO op_slbia (void)
304 ppc_slb_invalidate_all(env);
305 RETURN();
308 void OPPROTO op_slbie (void)
310 ppc_slb_invalidate_one(env, (uint32_t)T0);
311 RETURN();
314 void OPPROTO op_slbie_64 (void)
316 ppc_slb_invalidate_one(env, T0);
317 RETURN();
319 #endif
320 #endif
322 /* 601 specific */
323 void OPPROTO op_load_601_rtcl (void)
325 T0 = cpu_ppc601_load_rtcl(env);
326 RETURN();
329 void OPPROTO op_load_601_rtcu (void)
331 T0 = cpu_ppc601_load_rtcu(env);
332 RETURN();
335 #if !defined(CONFIG_USER_ONLY)
336 void OPPROTO op_store_601_rtcl (void)
338 cpu_ppc601_store_rtcl(env, T0);
339 RETURN();
342 void OPPROTO op_store_601_rtcu (void)
344 cpu_ppc601_store_rtcu(env, T0);
345 RETURN();
348 void OPPROTO op_store_hid0_601 (void)
350 do_store_hid0_601();
351 RETURN();
354 void OPPROTO op_load_601_bat (void)
356 T0 = env->IBAT[PARAM1][PARAM2];
357 RETURN();
360 void OPPROTO op_store_601_batl (void)
362 do_store_ibatl_601(env, PARAM1, T0);
363 RETURN();
366 void OPPROTO op_store_601_batu (void)
368 do_store_ibatu_601(env, PARAM1, T0);
369 RETURN();
371 #endif /* !defined(CONFIG_USER_ONLY) */
373 /* PowerPC 601 specific instructions (POWER bridge) */
374 /* XXX: those micro-ops need tests ! */
375 void OPPROTO op_POWER_abs (void)
377 if ((int32_t)T0 == INT32_MIN)
378 T0 = INT32_MAX;
379 else if ((int32_t)T0 < 0)
380 T0 = -T0;
381 RETURN();
384 void OPPROTO op_POWER_abso (void)
386 do_POWER_abso();
387 RETURN();
390 void OPPROTO op_POWER_clcs (void)
392 do_POWER_clcs();
393 RETURN();
396 void OPPROTO op_POWER_div (void)
398 do_POWER_div();
399 RETURN();
402 void OPPROTO op_POWER_divo (void)
404 do_POWER_divo();
405 RETURN();
408 void OPPROTO op_POWER_divs (void)
410 do_POWER_divs();
411 RETURN();
414 void OPPROTO op_POWER_divso (void)
416 do_POWER_divso();
417 RETURN();
420 void OPPROTO op_POWER_doz (void)
422 if ((int32_t)T1 > (int32_t)T0)
423 T0 = T1 - T0;
424 else
425 T0 = 0;
426 RETURN();
429 void OPPROTO op_POWER_dozo (void)
431 do_POWER_dozo();
432 RETURN();
435 void OPPROTO op_POWER_maskg (void)
437 do_POWER_maskg();
438 RETURN();
441 void OPPROTO op_POWER_maskir (void)
443 T0 = (T0 & ~T2) | (T1 & T2);
444 RETURN();
447 void OPPROTO op_POWER_mul (void)
449 uint64_t tmp;
451 tmp = (uint64_t)T0 * (uint64_t)T1;
452 env->spr[SPR_MQ] = tmp >> 32;
453 T0 = tmp;
454 RETURN();
457 void OPPROTO op_POWER_mulo (void)
459 do_POWER_mulo();
460 RETURN();
463 void OPPROTO op_POWER_nabs (void)
465 if (T0 > 0)
466 T0 = -T0;
467 RETURN();
470 void OPPROTO op_POWER_nabso (void)
472 /* nabs never overflows */
473 if (T0 > 0)
474 T0 = -T0;
475 env->xer &= ~(1 << XER_OV);
476 RETURN();
479 /* XXX: factorise POWER rotates... */
480 void OPPROTO op_POWER_rlmi (void)
482 T0 = rotl32(T0, T2) & PARAM1;
483 T0 |= T1 & (uint32_t)PARAM2;
484 RETURN();
487 void OPPROTO op_POWER_rrib (void)
489 T2 &= 0x1FUL;
490 T0 = rotl32(T0 & INT32_MIN, T2);
491 T0 |= T1 & ~rotl32(INT32_MIN, T2);
492 RETURN();
495 void OPPROTO op_POWER_sle (void)
497 T1 &= 0x1FUL;
498 env->spr[SPR_MQ] = rotl32(T0, T1);
499 T0 = T0 << T1;
500 RETURN();
503 void OPPROTO op_POWER_sleq (void)
505 uint32_t tmp = env->spr[SPR_MQ];
507 T1 &= 0x1FUL;
508 env->spr[SPR_MQ] = rotl32(T0, T1);
509 T0 = T0 << T1;
510 T0 |= tmp >> (32 - T1);
511 RETURN();
514 void OPPROTO op_POWER_sllq (void)
516 uint32_t msk = UINT32_MAX;
518 msk = msk << (T1 & 0x1FUL);
519 if (T1 & 0x20UL)
520 msk = ~msk;
521 T1 &= 0x1FUL;
522 T0 = (T0 << T1) & msk;
523 T0 |= env->spr[SPR_MQ] & ~msk;
524 RETURN();
527 void OPPROTO op_POWER_slq (void)
529 uint32_t msk = UINT32_MAX, tmp;
531 msk = msk << (T1 & 0x1FUL);
532 if (T1 & 0x20UL)
533 msk = ~msk;
534 T1 &= 0x1FUL;
535 tmp = rotl32(T0, T1);
536 T0 = tmp & msk;
537 env->spr[SPR_MQ] = tmp;
538 RETURN();
541 void OPPROTO op_POWER_sraq (void)
543 env->spr[SPR_MQ] = rotl32(T0, 32 - (T1 & 0x1FUL));
544 if (T1 & 0x20UL)
545 T0 = UINT32_MAX;
546 else
547 T0 = (int32_t)T0 >> T1;
548 RETURN();
551 void OPPROTO op_POWER_sre (void)
553 T1 &= 0x1FUL;
554 env->spr[SPR_MQ] = rotl32(T0, 32 - T1);
555 T0 = (int32_t)T0 >> T1;
556 RETURN();
559 void OPPROTO op_POWER_srea (void)
561 T1 &= 0x1FUL;
562 env->spr[SPR_MQ] = T0 >> T1;
563 T0 = (int32_t)T0 >> T1;
564 RETURN();
567 void OPPROTO op_POWER_sreq (void)
569 uint32_t tmp;
570 int32_t msk;
572 T1 &= 0x1FUL;
573 msk = INT32_MIN >> T1;
574 tmp = env->spr[SPR_MQ];
575 env->spr[SPR_MQ] = rotl32(T0, 32 - T1);
576 T0 = T0 >> T1;
577 T0 |= tmp & msk;
578 RETURN();
581 void OPPROTO op_POWER_srlq (void)
583 uint32_t tmp;
584 int32_t msk;
586 msk = INT32_MIN >> (T1 & 0x1FUL);
587 if (T1 & 0x20UL)
588 msk = ~msk;
589 T1 &= 0x1FUL;
590 tmp = env->spr[SPR_MQ];
591 env->spr[SPR_MQ] = rotl32(T0, 32 - T1);
592 T0 = T0 >> T1;
593 T0 &= msk;
594 T0 |= tmp & ~msk;
595 RETURN();
598 void OPPROTO op_POWER_srq (void)
600 T1 &= 0x1FUL;
601 env->spr[SPR_MQ] = rotl32(T0, 32 - T1);
602 T0 = T0 >> T1;
603 RETURN();
606 /* POWER instructions not implemented in PowerPC 601 */
607 #if !defined(CONFIG_USER_ONLY)
608 void OPPROTO op_POWER_mfsri (void)
610 T1 = T0 >> 28;
611 T0 = env->sr[T1];
612 RETURN();
615 void OPPROTO op_POWER_rac (void)
617 do_POWER_rac();
618 RETURN();
620 #endif
622 /* PowerPC 4xx specific micro-ops */
623 void OPPROTO op_load_dcr (void)
625 do_load_dcr();
626 RETURN();
629 void OPPROTO op_store_dcr (void)
631 do_store_dcr();
632 RETURN();
635 #if !defined(CONFIG_USER_ONLY)
636 void OPPROTO op_wrte (void)
638 /* We don't call do_store_msr here as we won't trigger
639 * any special case nor change hflags
641 T0 &= 1 << MSR_EE;
642 env->msr &= ~(1 << MSR_EE);
643 env->msr |= T0;
644 RETURN();
647 void OPPROTO op_440_tlbre (void)
649 do_440_tlbre(PARAM1);
650 RETURN();
653 void OPPROTO op_440_tlbsx (void)
655 T0 = ppcemb_tlb_search(env, T0, env->spr[SPR_440_MMUCR] & 0xFF);
656 RETURN();
659 void OPPROTO op_4xx_tlbsx_check (void)
661 int tmp;
663 tmp = xer_so;
664 if ((int)T0 != -1)
665 tmp |= 0x02;
666 env->crf[0] = tmp;
667 RETURN();
670 void OPPROTO op_440_tlbwe (void)
672 do_440_tlbwe(PARAM1);
673 RETURN();
676 void OPPROTO op_4xx_tlbre_lo (void)
678 do_4xx_tlbre_lo();
679 RETURN();
682 void OPPROTO op_4xx_tlbre_hi (void)
684 do_4xx_tlbre_hi();
685 RETURN();
688 void OPPROTO op_4xx_tlbsx (void)
690 T0 = ppcemb_tlb_search(env, T0, env->spr[SPR_40x_PID]);
691 RETURN();
694 void OPPROTO op_4xx_tlbwe_lo (void)
696 do_4xx_tlbwe_lo();
697 RETURN();
700 void OPPROTO op_4xx_tlbwe_hi (void)
702 do_4xx_tlbwe_hi();
703 RETURN();
705 #endif
707 /* SPR micro-ops */
708 /* 440 specific */
709 #if !defined(CONFIG_USER_ONLY)
710 void OPPROTO op_store_pir (void)
712 env->spr[SPR_PIR] = T0 & 0x0000000FUL;
713 RETURN();
716 void OPPROTO op_load_403_pb (void)
718 do_load_403_pb(PARAM1);
719 RETURN();
722 void OPPROTO op_store_403_pb (void)
724 do_store_403_pb(PARAM1);
725 RETURN();
728 void OPPROTO op_load_40x_pit (void)
730 T0 = load_40x_pit(env);
731 RETURN();
734 void OPPROTO op_store_40x_pit (void)
736 store_40x_pit(env, T0);
737 RETURN();
740 void OPPROTO op_store_40x_dbcr0 (void)
742 store_40x_dbcr0(env, T0);
743 RETURN();
746 void OPPROTO op_store_40x_sler (void)
748 store_40x_sler(env, T0);
749 RETURN();
752 void OPPROTO op_store_booke_tcr (void)
754 store_booke_tcr(env, T0);
755 RETURN();
758 void OPPROTO op_store_booke_tsr (void)
760 store_booke_tsr(env, T0);
761 RETURN();
763 #endif /* !defined(CONFIG_USER_ONLY) */