2 * MIPS emulation micro-operations for qemu.
4 * Copyright (c) 2004-2005 Jocelyn Mayer
5 * Copyright (c) 2006 Marius Groeger (FPU operations)
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26 #define CALL_FROM_TB0(func) func();
29 #define CALL_FROM_TB1(func, arg0) func(arg0);
31 #ifndef CALL_FROM_TB1_CONST16
32 #define CALL_FROM_TB1_CONST16(func, arg0) CALL_FROM_TB1(func, arg0);
35 #define CALL_FROM_TB2(func, arg0, arg1) func(arg0, arg1);
37 #ifndef CALL_FROM_TB2_CONST16
38 #define CALL_FROM_TB2_CONST16(func, arg0, arg1) \
39 CALL_FROM_TB2(func, arg0, arg1);
42 #define CALL_FROM_TB3(func, arg0, arg1, arg2) func(arg0, arg1, arg2);
45 #define CALL_FROM_TB4(func, arg0, arg1, arg2, arg3) \
46 func(arg0, arg1, arg2, arg3);
50 #include "op_template.c"
53 #include "op_template.c"
56 #include "op_template.c"
59 #include "op_template.c"
62 #include "op_template.c"
65 #include "op_template.c"
68 #include "op_template.c"
71 #include "op_template.c"
74 #include "op_template.c"
77 #include "op_template.c"
80 #include "op_template.c"
83 #include "op_template.c"
86 #include "op_template.c"
89 #include "op_template.c"
92 #include "op_template.c"
95 #include "op_template.c"
98 #include "op_template.c"
101 #include "op_template.c"
104 #include "op_template.c"
107 #include "op_template.c"
110 #include "op_template.c"
113 #include "op_template.c"
116 #include "op_template.c"
119 #include "op_template.c"
122 #include "op_template.c"
125 #include "op_template.c"
128 #include "op_template.c"
131 #include "op_template.c"
134 #include "op_template.c"
137 #include "op_template.c"
140 #include "op_template.c"
144 #include "op_template.c"
147 #include "op_template.c"
150 #include "op_template.c"
157 #include "fop_template.c"
161 #include "fop_template.c"
165 #include "fop_template.c"
169 #include "fop_template.c"
173 #include "fop_template.c"
177 #include "fop_template.c"
181 #include "fop_template.c"
185 #include "fop_template.c"
189 #include "fop_template.c"
193 #include "fop_template.c"
197 #include "fop_template.c"
201 #include "fop_template.c"
205 #include "fop_template.c"
209 #include "fop_template.c"
213 #include "fop_template.c"
217 #include "fop_template.c"
221 #include "fop_template.c"
225 #include "fop_template.c"
229 #include "fop_template.c"
233 #include "fop_template.c"
237 #include "fop_template.c"
241 #include "fop_template.c"
245 #include "fop_template.c"
249 #include "fop_template.c"
253 #include "fop_template.c"
257 #include "fop_template.c"
261 #include "fop_template.c"
265 #include "fop_template.c"
269 #include "fop_template.c"
273 #include "fop_template.c"
277 #include "fop_template.c"
281 #include "fop_template.c"
285 #include "fop_template.c"
290 void op_dup_T0 (void)
296 void op_load_HI (void)
302 void op_store_HI (void)
308 void op_load_LO (void)
314 void op_store_LO (void)
321 #define MEMSUFFIX _raw
324 #if !defined(CONFIG_USER_ONLY)
325 #define MEMSUFFIX _user
329 #define MEMSUFFIX _kernel
347 if (((tmp
^ T1
^ (-1)) & (T0
^ T1
)) >> 31) {
348 /* operands of same sign, result different sign */
349 CALL_FROM_TB1(do_raise_exception_direct
, EXCP_OVERFLOW
);
365 T0
= (int32_t)T0
- (int32_t)T1
;
366 if (((tmp
^ T1
) & (tmp
^ T0
)) >> 31) {
367 /* operands of different sign, first operand and result different sign */
368 CALL_FROM_TB1(do_raise_exception_direct
, EXCP_OVERFLOW
);
375 T0
= (int32_t)T0
* (int32_t)T1
;
382 env
->LO
= (int32_t)T0
/ (int32_t)T1
;
383 env
->HI
= (int32_t)T0
% (int32_t)T1
;
430 T0
= (int32_t)T0
>> T1
;
442 T0
= T1
<< (T0
& 0x1F);
448 T0
= (int32_t)T1
>> (T0
& 0x1F);
454 T0
= T1
>> (T0
& 0x1F);
462 if (T0
== (target_ulong
)-1) {
465 for (n
= 0; n
< 32; n
++) {
466 if (!(T0
& (1 << 31)))
482 for (n
= 0; n
< 32; n
++) {
492 /* 64 bits arithmetic */
493 #if (HOST_LONG_BITS == 64)
494 static inline uint64_t get_HILO (void)
496 return ((uint64_t)env
->HI
<< 32) | (uint64_t)env
->LO
;
499 static inline void set_HILO (uint64_t HILO
)
501 env
->LO
= HILO
& 0xFFFFFFFF;
502 env
->HI
= HILO
>> 32;
507 set_HILO((int64_t)(int32_t)T0
* (int64_t)(int32_t)T1
);
513 set_HILO((uint64_t)T0
* (uint64_t)T1
);
521 tmp
= ((int64_t)(int32_t)T0
* (int64_t)(int32_t)T1
);
522 set_HILO((int64_t)get_HILO() + tmp
);
530 tmp
= ((uint64_t)T0
* (uint64_t)T1
);
531 set_HILO(get_HILO() + tmp
);
539 tmp
= ((int64_t)(int32_t)T0
* (int64_t)(int32_t)T1
);
540 set_HILO((int64_t)get_HILO() - tmp
);
548 tmp
= ((uint64_t)T0
* (uint64_t)T1
);
549 set_HILO(get_HILO() - tmp
);
555 CALL_FROM_TB0(do_mult
);
561 CALL_FROM_TB0(do_multu
);
567 CALL_FROM_TB0(do_madd
);
573 CALL_FROM_TB0(do_maddu
);
579 CALL_FROM_TB0(do_msub
);
585 CALL_FROM_TB0(do_msubu
);
590 /* Conditional moves */
594 env
->gpr
[PARAM1
] = T0
;
601 env
->gpr
[PARAM1
] = T0
;
606 #define OP_COND(name, cond) \
607 void glue(op_, name) (void) \
617 OP_COND(eq
, T0
== T1
);
618 OP_COND(ne
, T0
!= T1
);
619 OP_COND(ge
, (int32_t)T0
>= (int32_t)T1
);
620 OP_COND(geu
, T0
>= T1
);
621 OP_COND(lt
, (int32_t)T0
< (int32_t)T1
);
622 OP_COND(ltu
, T0
< T1
);
623 OP_COND(gez
, (int32_t)T0
>= 0);
624 OP_COND(gtz
, (int32_t)T0
> 0);
625 OP_COND(lez
, (int32_t)T0
<= 0);
626 OP_COND(ltz
, (int32_t)T0
< 0);
629 //#undef USE_DIRECT_JUMP
631 void OPPROTO
op_goto_tb0(void)
633 GOTO_TB(op_goto_tb0
, PARAM1
, 0);
636 void OPPROTO
op_goto_tb1(void)
638 GOTO_TB(op_goto_tb1
, PARAM1
, 1);
641 /* Branch to register */
642 void op_save_breg_target (void)
647 void op_restore_breg_target (void)
658 void op_save_btarget (void)
660 env
->btarget
= PARAM1
;
664 /* Conditional branch */
665 void op_set_bcond (void)
671 void op_save_bcond (void)
677 void op_restore_bcond (void)
683 void op_jnz_T2 (void)
693 CALL_FROM_TB2(do_mfc0
, PARAM1
, PARAM2
);
699 CALL_FROM_TB2(do_mtc0
, PARAM1
, PARAM2
);
706 # define DEBUG_FPU_STATE() CALL_FROM_TB1(dump_fpu, env)
708 # define DEBUG_FPU_STATE() do { } while(0)
711 void op_cp1_enabled(void)
713 if (!(env
->CP0_Status
& (1 << CP0St_CU1
))) {
714 CALL_FROM_TB2(do_raise_exception_err
, EXCP_CpU
, 1);
726 /* fetch fcr31, masking unused bits */
727 T0
= env
->fcr31
& 0x0183FFFF;
733 /* convert MIPS rounding mode in FCR31 to IEEE library */
734 unsigned int ieee_rm
[] = {
735 float_round_nearest_even
,
741 #define RESTORE_ROUNDING_MODE \
742 set_float_rounding_mode(ieee_rm[env->fcr31 & 3], &env->fp_status)
747 /* XXX should this throw an exception?
748 * don't write to FCR0.
753 /* store new fcr31, masking unused bits */
754 env
->fcr31
= T0
& 0x0183FFFF;
756 /* set rounding mode */
757 RESTORE_ROUNDING_MODE
;
759 #ifndef CONFIG_SOFTFLOAT
760 /* no floating point exception for native float */
761 SET_FP_ENABLE(env
->fcr31
, 0);
783 Single precition routines have a "s" suffix, double precision a
786 #define FLOAT_OP(name, p) void OPPROTO op_float_##name##_##p(void)
790 FDT2
= int32_to_float64(WT0
, &env
->fp_status
);
796 FST2
= int32_to_float32(WT0
, &env
->fp_status
);
802 WT2
= float32_to_int32(FST0
, &env
->fp_status
);
808 WT2
= float64_to_int32(FDT0
, &env
->fp_status
);
815 set_float_rounding_mode(float_round_nearest_even
, &env
->fp_status
);
816 WT2
= float64_round_to_int(FDT0
, &env
->fp_status
);
817 RESTORE_ROUNDING_MODE
;
824 set_float_rounding_mode(float_round_nearest_even
, &env
->fp_status
);
825 WT2
= float32_round_to_int(FST0
, &env
->fp_status
);
826 RESTORE_ROUNDING_MODE
;
833 WT2
= float64_to_int32_round_to_zero(FDT0
, &env
->fp_status
);
839 WT2
= float32_to_int32_round_to_zero(FST0
, &env
->fp_status
);
846 set_float_rounding_mode(float_round_up
, &env
->fp_status
);
847 WT2
= float64_round_to_int(FDT0
, &env
->fp_status
);
848 RESTORE_ROUNDING_MODE
;
855 set_float_rounding_mode(float_round_up
, &env
->fp_status
);
856 WT2
= float32_round_to_int(FST0
, &env
->fp_status
);
857 RESTORE_ROUNDING_MODE
;
864 set_float_rounding_mode(float_round_down
, &env
->fp_status
);
865 WT2
= float64_round_to_int(FDT0
, &env
->fp_status
);
866 RESTORE_ROUNDING_MODE
;
873 set_float_rounding_mode(float_round_down
, &env
->fp_status
);
874 WT2
= float32_round_to_int(FST0
, &env
->fp_status
);
875 RESTORE_ROUNDING_MODE
;
880 /* binary operations */
881 #define FLOAT_BINOP(name) \
884 FDT2 = float64_ ## name (FDT0, FDT1, &env->fp_status); \
889 FST2 = float32_ ## name (FST0, FST1, &env->fp_status); \
898 /* unary operations, modifying fp status */
899 #define FLOAT_UNOP(name) \
902 FDT2 = float64_ ## name(FDT0, &env->fp_status); \
907 FST2 = float32_ ## name(FST0, &env->fp_status); \
913 /* unary operations, not modifying fp status */
914 #define FLOAT_UNOP(name) \
917 FDT2 = float64_ ## name(FDT0); \
922 FST2 = float32_ ## name(FST0); \
942 #ifdef CONFIG_SOFTFLOAT
943 #define clear_invalid() do { \
944 int flags = get_float_exception_flags(&env->fp_status); \
945 flags &= ~float_flag_invalid; \
946 set_float_exception_flags(flags, &env->fp_status); \
949 #define clear_invalid() do { } while(0)
952 extern void dump_fpu_s(CPUState
*env
);
954 #define FOP_COND(fmt, op, sig, cond) \
955 void op_cmp_ ## fmt ## _ ## op (void) \
958 SET_FP_COND(env->fcr31); \
960 CLEAR_FP_COND(env->fcr31); \
963 /*CALL_FROM_TB1(dump_fpu_s, env);*/ \
968 flag
float64_is_unordered(float64 a
, float64 b STATUS_PARAM
)
970 extern flag
float64_is_nan( float64 a
);
971 if (float64_is_nan(a
) || float64_is_nan(b
)) {
972 float_raise(float_flag_invalid
, status
);
981 FOP_COND(d
, un
, 0, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
))
982 FOP_COND(d
, eq
, 0, float64_eq(FDT0
, FDT1
, &env
->fp_status
))
983 FOP_COND(d
, ueq
, 0, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
) || float64_eq(FDT0
, FDT1
, &env
->fp_status
))
984 FOP_COND(d
, olt
, 0, float64_lt(FDT0
, FDT1
, &env
->fp_status
))
985 FOP_COND(d
, ult
, 0, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
) || float64_lt(FDT0
, FDT1
, &env
->fp_status
))
986 FOP_COND(d
, ole
, 0, float64_le(FDT0
, FDT1
, &env
->fp_status
))
987 FOP_COND(d
, ule
, 0, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
) || float64_le(FDT0
, FDT1
, &env
->fp_status
))
988 /* NOTE: the comma operator will make "cond" to eval to false,
989 * but float*_is_unordered() is still called
991 FOP_COND(d
, sf
, 1, (float64_is_unordered(FDT0
, FDT1
, &env
->fp_status
), 0))
992 FOP_COND(d
, ngle
,1, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
))
993 FOP_COND(d
, seq
, 1, float64_eq(FDT0
, FDT1
, &env
->fp_status
))
994 FOP_COND(d
, ngl
, 1, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
) || float64_eq(FDT0
, FDT1
, &env
->fp_status
))
995 FOP_COND(d
, lt
, 1, float64_lt(FDT0
, FDT1
, &env
->fp_status
))
996 FOP_COND(d
, nge
, 1, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
) || float64_lt(FDT0
, FDT1
, &env
->fp_status
))
997 FOP_COND(d
, le
, 1, float64_le(FDT0
, FDT1
, &env
->fp_status
))
998 FOP_COND(d
, ngt
, 1, float64_is_unordered(FDT1
, FDT0
, &env
->fp_status
) || float64_le(FDT0
, FDT1
, &env
->fp_status
))
1000 flag
float32_is_unordered(float32 a
, float32 b STATUS_PARAM
)
1002 extern flag
float32_is_nan( float32 a
);
1003 if (float32_is_nan(a
) || float32_is_nan(b
)) {
1004 float_raise(float_flag_invalid
, status
);
1012 /* NOTE: the comma operator will make "cond" to eval to false,
1013 * but float*_is_unordered() is still called
1015 FOP_COND(s
, f
, 0, 0)
1016 FOP_COND(s
, un
, 0, float32_is_unordered(FST1
, FST0
, &env
->fp_status
))
1017 FOP_COND(s
, eq
, 0, float32_eq(FST0
, FST1
, &env
->fp_status
))
1018 FOP_COND(s
, ueq
, 0, float32_is_unordered(FST1
, FST0
, &env
->fp_status
) || float32_eq(FST0
, FST1
, &env
->fp_status
))
1019 FOP_COND(s
, olt
, 0, float32_lt(FST0
, FST1
, &env
->fp_status
))
1020 FOP_COND(s
, ult
, 0, float32_is_unordered(FST1
, FST0
, &env
->fp_status
) || float32_lt(FST0
, FST1
, &env
->fp_status
))
1021 FOP_COND(s
, ole
, 0, float32_le(FST0
, FST1
, &env
->fp_status
))
1022 FOP_COND(s
, ule
, 0, float32_is_unordered(FST1
, FST0
, &env
->fp_status
) || float32_le(FST0
, FST1
, &env
->fp_status
))
1023 /* NOTE: the comma operator will make "cond" to eval to false,
1024 * but float*_is_unordered() is still called
1026 FOP_COND(s
, sf
, 1, (float32_is_unordered(FST0
, FST1
, &env
->fp_status
), 0))
1027 FOP_COND(s
, ngle
,1, float32_is_unordered(FST1
, FST0
, &env
->fp_status
))
1028 FOP_COND(s
, seq
, 1, float32_eq(FST0
, FST1
, &env
->fp_status
))
1029 FOP_COND(s
, ngl
, 1, float32_is_unordered(FST1
, FST0
, &env
->fp_status
) || float32_eq(FST0
, FST1
, &env
->fp_status
))
1030 FOP_COND(s
, lt
, 1, float32_lt(FST0
, FST1
, &env
->fp_status
))
1031 FOP_COND(s
, nge
, 1, float32_is_unordered(FST1
, FST0
, &env
->fp_status
) || float32_lt(FST0
, FST1
, &env
->fp_status
))
1032 FOP_COND(s
, le
, 1, float32_le(FST0
, FST1
, &env
->fp_status
))
1033 FOP_COND(s
, ngt
, 1, float32_is_unordered(FST1
, FST0
, &env
->fp_status
) || float32_le(FST0
, FST1
, &env
->fp_status
))
1037 T0
= ! IS_FP_COND_SET(env
->fcr31
);
1044 T0
= IS_FP_COND_SET(env
->fcr31
);
1048 #endif /* MIPS_USES_FPU */
1050 #if defined(MIPS_USES_R4K_TLB)
1051 void op_tlbwi (void)
1053 CALL_FROM_TB0(do_tlbwi
);
1057 void op_tlbwr (void)
1059 CALL_FROM_TB0(do_tlbwr
);
1065 CALL_FROM_TB0(do_tlbp
);
1071 CALL_FROM_TB0(do_tlbr
);
1079 CALL_FROM_TB1(do_pmon
, PARAM1
);
1085 CALL_FROM_TB1(do_raise_exception_direct
, EXCP_TRAP
);
1090 void op_debug (void)
1092 CALL_FROM_TB1(do_raise_exception
, EXCP_DEBUG
);
1095 void op_set_lladdr (void)
1097 env
->CP0_LLAddr
= T2
;
1100 void debug_eret (void);
1103 CALL_FROM_TB0(debug_eret
);
1104 if (env
->hflags
& MIPS_HFLAG_ERL
) {
1105 env
->PC
= env
->CP0_ErrorEPC
;
1106 env
->hflags
&= ~MIPS_HFLAG_ERL
;
1108 env
->PC
= env
->CP0_EPC
;
1109 env
->hflags
&= ~MIPS_HFLAG_EXL
;
1111 env
->CP0_LLAddr
= 1;
1114 void op_deret (void)
1116 CALL_FROM_TB0(debug_eret
);
1117 env
->PC
= env
->CP0_DEPC
;
1120 void op_save_state (void)
1122 env
->hflags
= PARAM1
;
1126 void op_save_pc (void)
1132 void op_raise_exception (void)
1134 CALL_FROM_TB1(do_raise_exception
, PARAM1
);
1138 void op_raise_exception_err (void)
1140 CALL_FROM_TB2(do_raise_exception_err
, PARAM1
, PARAM2
);
1144 void op_exit_tb (void)
1152 CALL_FROM_TB1(do_raise_exception
, EXCP_HLT
);