1 /*
2 * MIPS32 emulation for qemu: main translation routines.
3 *
4 * Copyright (c) 2004-2005 Jocelyn Mayer
5 * Copyright (c) 2006 Marius Groeger (FPU operations)
6 * Copyright (c) 2006 Thiemo Seufer (MIPS32R2 support)
7 *
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
12 *
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
20 */
21
22 #include <stdarg.h>
23 #include <stdlib.h>
24 #include <stdio.h>
25 #include <string.h>
26 #include <inttypes.h>
27
28 #include "cpu.h"
29 #include "exec-all.h"
30 #include "disas.h"
31 #include "tcg-op.h"
32 #include "qemu-common.h"
33
34 #include "helper.h"
35 #define GEN_HELPER 1
36 #include "helper.h"
37
38 //#define MIPS_DEBUG_DISAS
39 //#define MIPS_DEBUG_SIGN_EXTENSIONS
40
41 /* MIPS major opcodes */
42 #define MASK_OP_MAJOR(op) (op & (0x3F << 26))
43
44 enum {
45 /* indirect opcode tables */
46 OPC_SPECIAL = (0x00 << 26),
47 OPC_REGIMM = (0x01 << 26),
48 OPC_CP0 = (0x10 << 26),
49 OPC_CP1 = (0x11 << 26),
50 OPC_CP2 = (0x12 << 26),
51 OPC_CP3 = (0x13 << 26),
52 OPC_SPECIAL2 = (0x1C << 26),
53 OPC_SPECIAL3 = (0x1F << 26),
54 /* arithmetic with immediate */
55 OPC_ADDI = (0x08 << 26),
56 OPC_ADDIU = (0x09 << 26),
57 OPC_SLTI = (0x0A << 26),
58 OPC_SLTIU = (0x0B << 26),
59 /* logic with immediate */
60 OPC_ANDI = (0x0C << 26),
61 OPC_ORI = (0x0D << 26),
62 OPC_XORI = (0x0E << 26),
63 OPC_LUI = (0x0F << 26),
64 /* arithmetic with immediate */
65 OPC_DADDI = (0x18 << 26),
66 OPC_DADDIU = (0x19 << 26),
67 /* Jump and branches */
68 OPC_J = (0x02 << 26),
69 OPC_JAL = (0x03 << 26),
70 OPC_BEQ = (0x04 << 26), /* Unconditional if rs = rt = 0 (B) */
71 OPC_BEQL = (0x14 << 26),
72 OPC_BNE = (0x05 << 26),
73 OPC_BNEL = (0x15 << 26),
74 OPC_BLEZ = (0x06 << 26),
75 OPC_BLEZL = (0x16 << 26),
76 OPC_BGTZ = (0x07 << 26),
77 OPC_BGTZL = (0x17 << 26),
78 OPC_JALX = (0x1D << 26), /* MIPS 16 only */
79 /* Load and stores */
80 OPC_LDL = (0x1A << 26),
81 OPC_LDR = (0x1B << 26),
82 OPC_LB = (0x20 << 26),
83 OPC_LH = (0x21 << 26),
84 OPC_LWL = (0x22 << 26),
85 OPC_LW = (0x23 << 26),
86 OPC_LBU = (0x24 << 26),
87 OPC_LHU = (0x25 << 26),
88 OPC_LWR = (0x26 << 26),
89 OPC_LWU = (0x27 << 26),
90 OPC_SB = (0x28 << 26),
91 OPC_SH = (0x29 << 26),
92 OPC_SWL = (0x2A << 26),
93 OPC_SW = (0x2B << 26),
94 OPC_SDL = (0x2C << 26),
95 OPC_SDR = (0x2D << 26),
96 OPC_SWR = (0x2E << 26),
97 OPC_LL = (0x30 << 26),
98 OPC_LLD = (0x34 << 26),
99 OPC_LD = (0x37 << 26),
100 OPC_SC = (0x38 << 26),
101 OPC_SCD = (0x3C << 26),
102 OPC_SD = (0x3F << 26),
103 /* Floating point load/store */
104 OPC_LWC1 = (0x31 << 26),
105 OPC_LWC2 = (0x32 << 26),
106 OPC_LDC1 = (0x35 << 26),
107 OPC_LDC2 = (0x36 << 26),
108 OPC_SWC1 = (0x39 << 26),
109 OPC_SWC2 = (0x3A << 26),
110 OPC_SDC1 = (0x3D << 26),
111 OPC_SDC2 = (0x3E << 26),
112 /* MDMX ASE specific */
113 OPC_MDMX = (0x1E << 26),
114 /* Cache and prefetch */
115 OPC_CACHE = (0x2F << 26),
116 OPC_PREF = (0x33 << 26),
117 /* Reserved major opcode */
118 OPC_MAJOR3B_RESERVED = (0x3B << 26),
119 };
120
121 /* MIPS special opcodes */
122 #define MASK_SPECIAL(op) MASK_OP_MAJOR(op) | (op & 0x3F)
123
124 enum {
125 /* Shifts */
126 OPC_SLL = 0x00 | OPC_SPECIAL,
127 /* NOP is SLL r0, r0, 0 */
128 /* SSNOP is SLL r0, r0, 1 */
129 /* EHB is SLL r0, r0, 3 */
130 OPC_SRL = 0x02 | OPC_SPECIAL, /* also ROTR */
131 OPC_SRA = 0x03 | OPC_SPECIAL,
132 OPC_SLLV = 0x04 | OPC_SPECIAL,
133 OPC_SRLV = 0x06 | OPC_SPECIAL, /* also ROTRV */
134 OPC_SRAV = 0x07 | OPC_SPECIAL,
135 OPC_DSLLV = 0x14 | OPC_SPECIAL,
136 OPC_DSRLV = 0x16 | OPC_SPECIAL, /* also DROTRV */
137 OPC_DSRAV = 0x17 | OPC_SPECIAL,
138 OPC_DSLL = 0x38 | OPC_SPECIAL,
139 OPC_DSRL = 0x3A | OPC_SPECIAL, /* also DROTR */
140 OPC_DSRA = 0x3B | OPC_SPECIAL,
141 OPC_DSLL32 = 0x3C | OPC_SPECIAL,
142 OPC_DSRL32 = 0x3E | OPC_SPECIAL, /* also DROTR32 */
143 OPC_DSRA32 = 0x3F | OPC_SPECIAL,
144 /* Multiplication / division */
145 OPC_MULT = 0x18 | OPC_SPECIAL,
146 OPC_MULTU = 0x19 | OPC_SPECIAL,
147 OPC_DIV = 0x1A | OPC_SPECIAL,
148 OPC_DIVU = 0x1B | OPC_SPECIAL,
149 OPC_DMULT = 0x1C | OPC_SPECIAL,
150 OPC_DMULTU = 0x1D | OPC_SPECIAL,
151 OPC_DDIV = 0x1E | OPC_SPECIAL,
152 OPC_DDIVU = 0x1F | OPC_SPECIAL,
153 /* 2 registers arithmetic / logic */
154 OPC_ADD = 0x20 | OPC_SPECIAL,
155 OPC_ADDU = 0x21 | OPC_SPECIAL,
156 OPC_SUB = 0x22 | OPC_SPECIAL,
157 OPC_SUBU = 0x23 | OPC_SPECIAL,
158 OPC_AND = 0x24 | OPC_SPECIAL,
159 OPC_OR = 0x25 | OPC_SPECIAL,
160 OPC_XOR = 0x26 | OPC_SPECIAL,
161 OPC_NOR = 0x27 | OPC_SPECIAL,
162 OPC_SLT = 0x2A | OPC_SPECIAL,
163 OPC_SLTU = 0x2B | OPC_SPECIAL,
164 OPC_DADD = 0x2C | OPC_SPECIAL,
165 OPC_DADDU = 0x2D | OPC_SPECIAL,
166 OPC_DSUB = 0x2E | OPC_SPECIAL,
167 OPC_DSUBU = 0x2F | OPC_SPECIAL,
168 /* Jumps */
169 OPC_JR = 0x08 | OPC_SPECIAL, /* Also JR.HB */
170 OPC_JALR = 0x09 | OPC_SPECIAL, /* Also JALR.HB */
171 /* Traps */
172 OPC_TGE = 0x30 | OPC_SPECIAL,
173 OPC_TGEU = 0x31 | OPC_SPECIAL,
174 OPC_TLT = 0x32 | OPC_SPECIAL,
175 OPC_TLTU = 0x33 | OPC_SPECIAL,
176 OPC_TEQ = 0x34 | OPC_SPECIAL,
177 OPC_TNE = 0x36 | OPC_SPECIAL,
178 /* HI / LO registers load & stores */
179 OPC_MFHI = 0x10 | OPC_SPECIAL,
180 OPC_MTHI = 0x11 | OPC_SPECIAL,
181 OPC_MFLO = 0x12 | OPC_SPECIAL,
182 OPC_MTLO = 0x13 | OPC_SPECIAL,
183 /* Conditional moves */
184 OPC_MOVZ = 0x0A | OPC_SPECIAL,
185 OPC_MOVN = 0x0B | OPC_SPECIAL,
186
187 OPC_MOVCI = 0x01 | OPC_SPECIAL,
188
189 /* Special */
190 OPC_PMON = 0x05 | OPC_SPECIAL, /* unofficial */
191 OPC_SYSCALL = 0x0C | OPC_SPECIAL,
192 OPC_BREAK = 0x0D | OPC_SPECIAL,
193 OPC_SPIM = 0x0E | OPC_SPECIAL, /* unofficial */
194 OPC_SYNC = 0x0F | OPC_SPECIAL,
195
196 OPC_SPECIAL15_RESERVED = 0x15 | OPC_SPECIAL,
197 OPC_SPECIAL28_RESERVED = 0x28 | OPC_SPECIAL,
198 OPC_SPECIAL29_RESERVED = 0x29 | OPC_SPECIAL,
199 OPC_SPECIAL35_RESERVED = 0x35 | OPC_SPECIAL,
200 OPC_SPECIAL37_RESERVED = 0x37 | OPC_SPECIAL,
201 OPC_SPECIAL39_RESERVED = 0x39 | OPC_SPECIAL,
202 OPC_SPECIAL3D_RESERVED = 0x3D | OPC_SPECIAL,
203 };
204
205 /* Multiplication variants of the vr54xx. */
206 #define MASK_MUL_VR54XX(op) MASK_SPECIAL(op) | (op & (0x1F << 6))
207
208 enum {
209 OPC_VR54XX_MULS = (0x03 << 6) | OPC_MULT,
210 OPC_VR54XX_MULSU = (0x03 << 6) | OPC_MULTU,
211 OPC_VR54XX_MACC = (0x05 << 6) | OPC_MULT,
212 OPC_VR54XX_MACCU = (0x05 << 6) | OPC_MULTU,
213 OPC_VR54XX_MSAC = (0x07 << 6) | OPC_MULT,
214 OPC_VR54XX_MSACU = (0x07 << 6) | OPC_MULTU,
215 OPC_VR54XX_MULHI = (0x09 << 6) | OPC_MULT,
216 OPC_VR54XX_MULHIU = (0x09 << 6) | OPC_MULTU,
217 OPC_VR54XX_MULSHI = (0x0B << 6) | OPC_MULT,
218 OPC_VR54XX_MULSHIU = (0x0B << 6) | OPC_MULTU,
219 OPC_VR54XX_MACCHI = (0x0D << 6) | OPC_MULT,
220 OPC_VR54XX_MACCHIU = (0x0D << 6) | OPC_MULTU,
221 OPC_VR54XX_MSACHI = (0x0F << 6) | OPC_MULT,
222 OPC_VR54XX_MSACHIU = (0x0F << 6) | OPC_MULTU,
223 };
224
225 /* REGIMM (rt field) opcodes */
226 #define MASK_REGIMM(op) MASK_OP_MAJOR(op) | (op & (0x1F << 16))
227
228 enum {
229 OPC_BLTZ = (0x00 << 16) | OPC_REGIMM,
230 OPC_BLTZL = (0x02 << 16) | OPC_REGIMM,
231 OPC_BGEZ = (0x01 << 16) | OPC_REGIMM,
232 OPC_BGEZL = (0x03 << 16) | OPC_REGIMM,
233 OPC_BLTZAL = (0x10 << 16) | OPC_REGIMM,
234 OPC_BLTZALL = (0x12 << 16) | OPC_REGIMM,
235 OPC_BGEZAL = (0x11 << 16) | OPC_REGIMM,
236 OPC_BGEZALL = (0x13 << 16) | OPC_REGIMM,
237 OPC_TGEI = (0x08 << 16) | OPC_REGIMM,
238 OPC_TGEIU = (0x09 << 16) | OPC_REGIMM,
239 OPC_TLTI = (0x0A << 16) | OPC_REGIMM,
240 OPC_TLTIU = (0x0B << 16) | OPC_REGIMM,
241 OPC_TEQI = (0x0C << 16) | OPC_REGIMM,
242 OPC_TNEI = (0x0E << 16) | OPC_REGIMM,
243 OPC_SYNCI = (0x1F << 16) | OPC_REGIMM,
244 };
245
246 /* Special2 opcodes */
247 #define MASK_SPECIAL2(op) MASK_OP_MAJOR(op) | (op & 0x3F)
248
249 enum {
250 /* Multiply & xxx operations */
251 OPC_MADD = 0x00 | OPC_SPECIAL2,
252 OPC_MADDU = 0x01 | OPC_SPECIAL2,
253 OPC_MUL = 0x02 | OPC_SPECIAL2,
254 OPC_MSUB = 0x04 | OPC_SPECIAL2,
255 OPC_MSUBU = 0x05 | OPC_SPECIAL2,
256 /* Misc */
257 OPC_CLZ = 0x20 | OPC_SPECIAL2,
258 OPC_CLO = 0x21 | OPC_SPECIAL2,
259 OPC_DCLZ = 0x24 | OPC_SPECIAL2,
260 OPC_DCLO = 0x25 | OPC_SPECIAL2,
261 /* Special */
262 OPC_SDBBP = 0x3F | OPC_SPECIAL2,
263 };
264
265 /* Special3 opcodes */
266 #define MASK_SPECIAL3(op) MASK_OP_MAJOR(op) | (op & 0x3F)
267
268 enum {
269 OPC_EXT = 0x00 | OPC_SPECIAL3,
270 OPC_DEXTM = 0x01 | OPC_SPECIAL3,
271 OPC_DEXTU = 0x02 | OPC_SPECIAL3,
272 OPC_DEXT = 0x03 | OPC_SPECIAL3,
273 OPC_INS = 0x04 | OPC_SPECIAL3,
274 OPC_DINSM = 0x05 | OPC_SPECIAL3,
275 OPC_DINSU = 0x06 | OPC_SPECIAL3,
276 OPC_DINS = 0x07 | OPC_SPECIAL3,
277 OPC_FORK = 0x08 | OPC_SPECIAL3,
278 OPC_YIELD = 0x09 | OPC_SPECIAL3,
279 OPC_BSHFL = 0x20 | OPC_SPECIAL3,
280 OPC_DBSHFL = 0x24 | OPC_SPECIAL3,
281 OPC_RDHWR = 0x3B | OPC_SPECIAL3,
282 };
283
284 /* BSHFL opcodes */
285 #define MASK_BSHFL(op) MASK_SPECIAL3(op) | (op & (0x1F << 6))
286
287 enum {
288 OPC_WSBH = (0x02 << 6) | OPC_BSHFL,
289 OPC_SEB = (0x10 << 6) | OPC_BSHFL,
290 OPC_SEH = (0x18 << 6) | OPC_BSHFL,
291 };
292
293 /* DBSHFL opcodes */
294 #define MASK_DBSHFL(op) MASK_SPECIAL3(op) | (op & (0x1F << 6))
295
296 enum {
297 OPC_DSBH = (0x02 << 6) | OPC_DBSHFL,
298 OPC_DSHD = (0x05 << 6) | OPC_DBSHFL,
299 };
300
301 /* Coprocessor 0 (rs field) */
302 #define MASK_CP0(op) MASK_OP_MAJOR(op) | (op & (0x1F << 21))
303
304 enum {
305 OPC_MFC0 = (0x00 << 21) | OPC_CP0,
306 OPC_DMFC0 = (0x01 << 21) | OPC_CP0,
307 OPC_MTC0 = (0x04 << 21) | OPC_CP0,
308 OPC_DMTC0 = (0x05 << 21) | OPC_CP0,
309 OPC_MFTR = (0x08 << 21) | OPC_CP0,
310 OPC_RDPGPR = (0x0A << 21) | OPC_CP0,
311 OPC_MFMC0 = (0x0B << 21) | OPC_CP0,
312 OPC_MTTR = (0x0C << 21) | OPC_CP0,
313 OPC_WRPGPR = (0x0E << 21) | OPC_CP0,
314 OPC_C0 = (0x10 << 21) | OPC_CP0,
315 OPC_C0_FIRST = (0x10 << 21) | OPC_CP0,
316 OPC_C0_LAST = (0x1F << 21) | OPC_CP0,
317 };
318
319 /* MFMC0 opcodes */
320 #define MASK_MFMC0(op) MASK_CP0(op) | (op & 0xFFFF)
321
322 enum {
323 OPC_DMT = 0x01 | (0 << 5) | (0x0F << 6) | (0x01 << 11) | OPC_MFMC0,
324 OPC_EMT = 0x01 | (1 << 5) | (0x0F << 6) | (0x01 << 11) | OPC_MFMC0,
325 OPC_DVPE = 0x01 | (0 << 5) | OPC_MFMC0,
326 OPC_EVPE = 0x01 | (1 << 5) | OPC_MFMC0,
327 OPC_DI = (0 << 5) | (0x0C << 11) | OPC_MFMC0,
328 OPC_EI = (1 << 5) | (0x0C << 11) | OPC_MFMC0,
329 };
330
331 /* Coprocessor 0 (with rs == C0) */
332 #define MASK_C0(op) MASK_CP0(op) | (op & 0x3F)
333
334 enum {
335 OPC_TLBR = 0x01 | OPC_C0,
336 OPC_TLBWI = 0x02 | OPC_C0,
337 OPC_TLBWR = 0x06 | OPC_C0,
338 OPC_TLBP = 0x08 | OPC_C0,
339 OPC_RFE = 0x10 | OPC_C0,
340 OPC_ERET = 0x18 | OPC_C0,
341 OPC_DERET = 0x1F | OPC_C0,
342 OPC_WAIT = 0x20 | OPC_C0,
343 };
344
345 /* Coprocessor 1 (rs field) */
346 #define MASK_CP1(op) MASK_OP_MAJOR(op) | (op & (0x1F << 21))
347
348 enum {
349 OPC_MFC1 = (0x00 << 21) | OPC_CP1,
350 OPC_DMFC1 = (0x01 << 21) | OPC_CP1,
351 OPC_CFC1 = (0x02 << 21) | OPC_CP1,
352 OPC_MFHC1 = (0x03 << 21) | OPC_CP1,
353 OPC_MTC1 = (0x04 << 21) | OPC_CP1,
354 OPC_DMTC1 = (0x05 << 21) | OPC_CP1,
355 OPC_CTC1 = (0x06 << 21) | OPC_CP1,
356 OPC_MTHC1 = (0x07 << 21) | OPC_CP1,
357 OPC_BC1 = (0x08 << 21) | OPC_CP1, /* bc */
358 OPC_BC1ANY2 = (0x09 << 21) | OPC_CP1,
359 OPC_BC1ANY4 = (0x0A << 21) | OPC_CP1,
360 OPC_S_FMT = (0x10 << 21) | OPC_CP1, /* 16: fmt=single fp */
361 OPC_D_FMT = (0x11 << 21) | OPC_CP1, /* 17: fmt=double fp */
362 OPC_E_FMT = (0x12 << 21) | OPC_CP1, /* 18: fmt=extended fp */
363 OPC_Q_FMT = (0x13 << 21) | OPC_CP1, /* 19: fmt=quad fp */
364 OPC_W_FMT = (0x14 << 21) | OPC_CP1, /* 20: fmt=32bit fixed */
365 OPC_L_FMT = (0x15 << 21) | OPC_CP1, /* 21: fmt=64bit fixed */
366 OPC_PS_FMT = (0x16 << 21) | OPC_CP1, /* 22: fmt=paired single fp */
367 };
368
369 #define MASK_CP1_FUNC(op) MASK_CP1(op) | (op & 0x3F)
370 #define MASK_BC1(op) MASK_CP1(op) | (op & (0x3 << 16))
371
372 enum {
373 OPC_BC1F = (0x00 << 16) | OPC_BC1,
374 OPC_BC1T = (0x01 << 16) | OPC_BC1,
375 OPC_BC1FL = (0x02 << 16) | OPC_BC1,
376 OPC_BC1TL = (0x03 << 16) | OPC_BC1,
377 };
378
379 enum {
380 OPC_BC1FANY2 = (0x00 << 16) | OPC_BC1ANY2,
381 OPC_BC1TANY2 = (0x01 << 16) | OPC_BC1ANY2,
382 };
383
384 enum {
385 OPC_BC1FANY4 = (0x00 << 16) | OPC_BC1ANY4,
386 OPC_BC1TANY4 = (0x01 << 16) | OPC_BC1ANY4,
387 };
388
389 #define MASK_CP2(op) MASK_OP_MAJOR(op) | (op & (0x1F << 21))
390
391 enum {
392 OPC_MFC2 = (0x00 << 21) | OPC_CP2,
393 OPC_DMFC2 = (0x01 << 21) | OPC_CP2,
394 OPC_CFC2 = (0x02 << 21) | OPC_CP2,
395 OPC_MFHC2 = (0x03 << 21) | OPC_CP2,
396 OPC_MTC2 = (0x04 << 21) | OPC_CP2,
397 OPC_DMTC2 = (0x05 << 21) | OPC_CP2,
398 OPC_CTC2 = (0x06 << 21) | OPC_CP2,
399 OPC_MTHC2 = (0x07 << 21) | OPC_CP2,
400 OPC_BC2 = (0x08 << 21) | OPC_CP2,
401 };
402
403 #define MASK_CP3(op) MASK_OP_MAJOR(op) | (op & 0x3F)
404
405 enum {
406 OPC_LWXC1 = 0x00 | OPC_CP3,
407 OPC_LDXC1 = 0x01 | OPC_CP3,
408 OPC_LUXC1 = 0x05 | OPC_CP3,
409 OPC_SWXC1 = 0x08 | OPC_CP3,
410 OPC_SDXC1 = 0x09 | OPC_CP3,
411 OPC_SUXC1 = 0x0D | OPC_CP3,
412 OPC_PREFX = 0x0F | OPC_CP3,
413 OPC_ALNV_PS = 0x1E | OPC_CP3,
414 OPC_MADD_S = 0x20 | OPC_CP3,
415 OPC_MADD_D = 0x21 | OPC_CP3,
416 OPC_MADD_PS = 0x26 | OPC_CP3,
417 OPC_MSUB_S = 0x28 | OPC_CP3,
418 OPC_MSUB_D = 0x29 | OPC_CP3,
419 OPC_MSUB_PS = 0x2E | OPC_CP3,
420 OPC_NMADD_S = 0x30 | OPC_CP3,
421 OPC_NMADD_D = 0x31 | OPC_CP3,
422 OPC_NMADD_PS= 0x36 | OPC_CP3,
423 OPC_NMSUB_S = 0x38 | OPC_CP3,
424 OPC_NMSUB_D = 0x39 | OPC_CP3,
425 OPC_NMSUB_PS= 0x3E | OPC_CP3,
426 };
427
428 /* global register indices */
429 static TCGv_ptr cpu_env;
430 static TCGv cpu_gpr[32], cpu_PC;
431 static TCGv cpu_HI[MIPS_DSP_ACC], cpu_LO[MIPS_DSP_ACC], cpu_ACX[MIPS_DSP_ACC];
432 static TCGv cpu_dspctrl, btarget, bcond;
433 static TCGv_i32 hflags;
434 static TCGv_i32 fpu_fcr0, fpu_fcr31;
435
436 #include "gen-icount.h"
437
438 #define gen_helper_0i(name, arg) do { \
439 TCGv_i32 helper_tmp = tcg_const_i32(arg); \
440 gen_helper_##name(helper_tmp); \
441 tcg_temp_free_i32(helper_tmp); \
442 } while(0)
443
444 #define gen_helper_1i(name, arg1, arg2) do { \
445 TCGv_i32 helper_tmp = tcg_const_i32(arg2); \
446 gen_helper_##name(arg1, helper_tmp); \
447 tcg_temp_free_i32(helper_tmp); \
448 } while(0)
449
450 #define gen_helper_2i(name, arg1, arg2, arg3) do { \
451 TCGv_i32 helper_tmp = tcg_const_i32(arg3); \
452 gen_helper_##name(arg1, arg2, helper_tmp); \
453 tcg_temp_free_i32(helper_tmp); \
454 } while(0)
455
456 #define gen_helper_3i(name, arg1, arg2, arg3, arg4) do { \
457 TCGv_i32 helper_tmp = tcg_const_i32(arg4); \
458 gen_helper_##name(arg1, arg2, arg3, helper_tmp); \
459 tcg_temp_free_i32(helper_tmp); \
460 } while(0)
461
462 typedef struct DisasContext {
463 struct TranslationBlock *tb;
464 target_ulong pc, saved_pc;
465 uint32_t opcode;
466 int singlestep_enabled;
467 /* Routine used to access memory */
468 int mem_idx;
469 uint32_t hflags, saved_hflags;
470 int bstate;
471 target_ulong btarget;
472 } DisasContext;
473
474 enum {
475 BS_NONE = 0, /* We go out of the TB without reaching a branch or an
476 * exception condition */
477 BS_STOP = 1, /* We want to stop translation for any reason */
478 BS_BRANCH = 2, /* We reached a branch condition */
479 BS_EXCP = 3, /* We reached an exception condition */
480 };
481
482 static const char *regnames[] =
483 { "r0", "at", "v0", "v1", "a0", "a1", "a2", "a3",
484 "t0", "t1", "t2", "t3", "t4", "t5", "t6", "t7",
485 "s0", "s1", "s2", "s3", "s4", "s5", "s6", "s7",
486 "t8", "t9", "k0", "k1", "gp", "sp", "s8", "ra", };
487
488 static const char *regnames_HI[] =
489 { "HI0", "HI1", "HI2", "HI3", };
490
491 static const char *regnames_LO[] =
492 { "LO0", "LO1", "LO2", "LO3", };
493
494 static const char *regnames_ACX[] =
495 { "ACX0", "ACX1", "ACX2", "ACX3", };
496
497 static const char *fregnames[] =
498 { "f0", "f1", "f2", "f3", "f4", "f5", "f6", "f7",
499 "f8", "f9", "f10", "f11", "f12", "f13", "f14", "f15",
500 "f16", "f17", "f18", "f19", "f20", "f21", "f22", "f23",
501 "f24", "f25", "f26", "f27", "f28", "f29", "f30", "f31", };
502
503 #ifdef MIPS_DEBUG_DISAS
504 #define MIPS_DEBUG(fmt, ...) \
505 qemu_log_mask(CPU_LOG_TB_IN_ASM, \
506 TARGET_FMT_lx ": %08x " fmt "\n", \
507 ctx->pc, ctx->opcode , ## __VA_ARGS__)
508 #define LOG_DISAS(...) qemu_log_mask(CPU_LOG_TB_IN_ASM, ## __VA_ARGS__)
509 #else
510 #define MIPS_DEBUG(fmt, ...) do { } while(0)
511 #define LOG_DISAS(...) do { } while (0)
512 #endif
513
514 #define MIPS_INVAL(op) \
515 do { \
516 MIPS_DEBUG("Invalid %s %03x %03x %03x", op, ctx->opcode >> 26, \
517 ctx->opcode & 0x3F, ((ctx->opcode >> 16) & 0x1F)); \
518 } while (0)
519
520 /* General purpose registers moves. */
gen_load_gpr(TCGv t,int reg)521 static inline void gen_load_gpr (TCGv t, int reg)
522 {
523 if (reg == 0)
524 tcg_gen_movi_tl(t, 0);
525 else
526 tcg_gen_mov_tl(t, cpu_gpr[reg]);
527 }
528
gen_store_gpr(TCGv t,int reg)529 static inline void gen_store_gpr (TCGv t, int reg)
530 {
531 if (reg != 0)
532 tcg_gen_mov_tl(cpu_gpr[reg], t);
533 }
534
535 /* Moves to/from ACX register. */
gen_load_ACX(TCGv t,int reg)536 static inline void gen_load_ACX (TCGv t, int reg)
537 {
538 tcg_gen_mov_tl(t, cpu_ACX[reg]);
539 }
540
gen_store_ACX(TCGv t,int reg)541 static inline void gen_store_ACX (TCGv t, int reg)
542 {
543 tcg_gen_mov_tl(cpu_ACX[reg], t);
544 }
545
546 /* Moves to/from shadow registers. */
gen_load_srsgpr(int from,int to)547 static inline void gen_load_srsgpr (int from, int to)
548 {
549 TCGv t0 = tcg_temp_new();
550
551 if (from == 0)
552 tcg_gen_movi_tl(t0, 0);
553 else {
554 TCGv_i32 t2 = tcg_temp_new_i32();
555 TCGv_ptr addr = tcg_temp_new_ptr();
556
557 tcg_gen_ld_i32(t2, cpu_env, offsetof(CPUState, CP0_SRSCtl));
558 tcg_gen_shri_i32(t2, t2, CP0SRSCtl_PSS);
559 tcg_gen_andi_i32(t2, t2, 0xf);
560 tcg_gen_muli_i32(t2, t2, sizeof(target_ulong) * 32);
561 tcg_gen_ext_i32_ptr(addr, t2);
562 tcg_gen_add_ptr(addr, cpu_env, addr);
563
564 tcg_gen_ld_tl(t0, addr, sizeof(target_ulong) * from);
565 tcg_temp_free_ptr(addr);
566 tcg_temp_free_i32(t2);
567 }
568 gen_store_gpr(t0, to);
569 tcg_temp_free(t0);
570 }
571
gen_store_srsgpr(int from,int to)572 static inline void gen_store_srsgpr (int from, int to)
573 {
574 if (to != 0) {
575 TCGv t0 = tcg_temp_new();
576 TCGv_i32 t2 = tcg_temp_new_i32();
577 TCGv_ptr addr = tcg_temp_new_ptr();
578
579 gen_load_gpr(t0, from);
580 tcg_gen_ld_i32(t2, cpu_env, offsetof(CPUState, CP0_SRSCtl));
581 tcg_gen_shri_i32(t2, t2, CP0SRSCtl_PSS);
582 tcg_gen_andi_i32(t2, t2, 0xf);
583 tcg_gen_muli_i32(t2, t2, sizeof(target_ulong) * 32);
584 tcg_gen_ext_i32_ptr(addr, t2);
585 tcg_gen_add_ptr(addr, cpu_env, addr);
586
587 tcg_gen_st_tl(t0, addr, sizeof(target_ulong) * to);
588 tcg_temp_free_ptr(addr);
589 tcg_temp_free_i32(t2);
590 tcg_temp_free(t0);
591 }
592 }
593
594 /* Floating point register moves. */
gen_load_fpr32(TCGv_i32 t,int reg)595 static inline void gen_load_fpr32 (TCGv_i32 t, int reg)
596 {
597 tcg_gen_ld_i32(t, cpu_env, offsetof(CPUState, active_fpu.fpr[reg].w[FP_ENDIAN_IDX]));
598 }
599
gen_store_fpr32(TCGv_i32 t,int reg)600 static inline void gen_store_fpr32 (TCGv_i32 t, int reg)
601 {
602 tcg_gen_st_i32(t, cpu_env, offsetof(CPUState, active_fpu.fpr[reg].w[FP_ENDIAN_IDX]));
603 }
604
gen_load_fpr32h(TCGv_i32 t,int reg)605 static inline void gen_load_fpr32h (TCGv_i32 t, int reg)
606 {
607 tcg_gen_ld_i32(t, cpu_env, offsetof(CPUState, active_fpu.fpr[reg].w[!FP_ENDIAN_IDX]));
608 }
609
gen_store_fpr32h(TCGv_i32 t,int reg)610 static inline void gen_store_fpr32h (TCGv_i32 t, int reg)
611 {
612 tcg_gen_st_i32(t, cpu_env, offsetof(CPUState, active_fpu.fpr[reg].w[!FP_ENDIAN_IDX]));
613 }
614
gen_load_fpr64(DisasContext * ctx,TCGv_i64 t,int reg)615 static inline void gen_load_fpr64 (DisasContext *ctx, TCGv_i64 t, int reg)
616 {
617 if (ctx->hflags & MIPS_HFLAG_F64) {
618 tcg_gen_ld_i64(t, cpu_env, offsetof(CPUState, active_fpu.fpr[reg].d));
619 } else {
620 TCGv_i32 t0 = tcg_temp_new_i32();
621 TCGv_i32 t1 = tcg_temp_new_i32();
622 gen_load_fpr32(t0, reg & ~1);
623 gen_load_fpr32(t1, reg | 1);
624 tcg_gen_concat_i32_i64(t, t0, t1);
625 tcg_temp_free_i32(t0);
626 tcg_temp_free_i32(t1);
627 }
628 }
629
gen_store_fpr64(DisasContext * ctx,TCGv_i64 t,int reg)630 static inline void gen_store_fpr64 (DisasContext *ctx, TCGv_i64 t, int reg)
631 {
632 if (ctx->hflags & MIPS_HFLAG_F64) {
633 tcg_gen_st_i64(t, cpu_env, offsetof(CPUState, active_fpu.fpr[reg].d));
634 } else {
635 TCGv_i64 t0 = tcg_temp_new_i64();
636 TCGv_i32 t1 = tcg_temp_new_i32();
637 tcg_gen_trunc_i64_i32(t1, t);
638 gen_store_fpr32(t1, reg & ~1);
639 tcg_gen_shri_i64(t0, t, 32);
640 tcg_gen_trunc_i64_i32(t1, t0);
641 gen_store_fpr32(t1, reg | 1);
642 tcg_temp_free_i32(t1);
643 tcg_temp_free_i64(t0);
644 }
645 }
646
get_fp_bit(int cc)647 static inline int get_fp_bit (int cc)
648 {
649 if (cc)
650 return 24 + cc;
651 else
652 return 23;
653 }
654
655 #define FOP_CONDS(type, fmt, bits) \
656 static inline void gen_cmp ## type ## _ ## fmt(int n, TCGv_i##bits a, \
657 TCGv_i##bits b, int cc) \
658 { \
659 switch (n) { \
660 case 0: gen_helper_2i(cmp ## type ## _ ## fmt ## _f, a, b, cc); break;\
661 case 1: gen_helper_2i(cmp ## type ## _ ## fmt ## _un, a, b, cc); break;\
662 case 2: gen_helper_2i(cmp ## type ## _ ## fmt ## _eq, a, b, cc); break;\
663 case 3: gen_helper_2i(cmp ## type ## _ ## fmt ## _ueq, a, b, cc); break;\
664 case 4: gen_helper_2i(cmp ## type ## _ ## fmt ## _olt, a, b, cc); break;\
665 case 5: gen_helper_2i(cmp ## type ## _ ## fmt ## _ult, a, b, cc); break;\
666 case 6: gen_helper_2i(cmp ## type ## _ ## fmt ## _ole, a, b, cc); break;\
667 case 7: gen_helper_2i(cmp ## type ## _ ## fmt ## _ule, a, b, cc); break;\
668 case 8: gen_helper_2i(cmp ## type ## _ ## fmt ## _sf, a, b, cc); break;\
669 case 9: gen_helper_2i(cmp ## type ## _ ## fmt ## _ngle, a, b, cc); break;\
670 case 10: gen_helper_2i(cmp ## type ## _ ## fmt ## _seq, a, b, cc); break;\
671 case 11: gen_helper_2i(cmp ## type ## _ ## fmt ## _ngl, a, b, cc); break;\
672 case 12: gen_helper_2i(cmp ## type ## _ ## fmt ## _lt, a, b, cc); break;\
673 case 13: gen_helper_2i(cmp ## type ## _ ## fmt ## _nge, a, b, cc); break;\
674 case 14: gen_helper_2i(cmp ## type ## _ ## fmt ## _le, a, b, cc); break;\
675 case 15: gen_helper_2i(cmp ## type ## _ ## fmt ## _ngt, a, b, cc); break;\
676 default: abort(); \
677 } \
678 }
679
680 FOP_CONDS(, d, 64)
681 FOP_CONDS(abs, d, 64)
682 FOP_CONDS(, s, 32)
683 FOP_CONDS(abs, s, 32)
684 FOP_CONDS(, ps, 64)
685 FOP_CONDS(abs, ps, 64)
686 #undef FOP_CONDS
687
688 /* Tests */
689 #define OP_COND(name, cond) \
690 static inline void glue(gen_op_, name) (TCGv ret, TCGv t0, TCGv t1) \
691 { \
692 int l1 = gen_new_label(); \
693 int l2 = gen_new_label(); \
694 \
695 tcg_gen_brcond_tl(cond, t0, t1, l1); \
696 tcg_gen_movi_tl(ret, 0); \
697 tcg_gen_br(l2); \
698 gen_set_label(l1); \
699 tcg_gen_movi_tl(ret, 1); \
700 gen_set_label(l2); \
701 }
702 OP_COND(eq, TCG_COND_EQ);
703 OP_COND(ne, TCG_COND_NE);
704 OP_COND(ge, TCG_COND_GE);
705 OP_COND(geu, TCG_COND_GEU);
706 OP_COND(lt, TCG_COND_LT);
707 OP_COND(ltu, TCG_COND_LTU);
708 #undef OP_COND
709
710 #define OP_CONDI(name, cond) \
711 static inline void glue(gen_op_, name) (TCGv ret, TCGv t0, target_ulong val) \
712 { \
713 int l1 = gen_new_label(); \
714 int l2 = gen_new_label(); \
715 \
716 tcg_gen_brcondi_tl(cond, t0, val, l1); \
717 tcg_gen_movi_tl(ret, 0); \
718 tcg_gen_br(l2); \
719 gen_set_label(l1); \
720 tcg_gen_movi_tl(ret, 1); \
721 gen_set_label(l2); \
722 }
723 OP_CONDI(lti, TCG_COND_LT);
724 OP_CONDI(ltiu, TCG_COND_LTU);
725 #undef OP_CONDI
726
727 #define OP_CONDZ(name, cond) \
728 static inline void glue(gen_op_, name) (TCGv ret, TCGv t0) \
729 { \
730 int l1 = gen_new_label(); \
731 int l2 = gen_new_label(); \
732 \
733 tcg_gen_brcondi_tl(cond, t0, 0, l1); \
734 tcg_gen_movi_tl(ret, 0); \
735 tcg_gen_br(l2); \
736 gen_set_label(l1); \
737 tcg_gen_movi_tl(ret, 1); \
738 gen_set_label(l2); \
739 }
740 OP_CONDZ(gez, TCG_COND_GE);
741 OP_CONDZ(gtz, TCG_COND_GT);
742 OP_CONDZ(lez, TCG_COND_LE);
743 OP_CONDZ(ltz, TCG_COND_LT);
744 #undef OP_CONDZ
745
gen_save_pc(target_ulong pc)746 static inline void gen_save_pc(target_ulong pc)
747 {
748 tcg_gen_movi_tl(cpu_PC, pc);
749 }
750
save_cpu_state(DisasContext * ctx,int do_save_pc)751 static inline void save_cpu_state (DisasContext *ctx, int do_save_pc)
752 {
753 LOG_DISAS("hflags %08x saved %08x\n", ctx->hflags, ctx->saved_hflags);
754 if (do_save_pc && ctx->pc != ctx->saved_pc) {
755 gen_save_pc(ctx->pc);
756 ctx->saved_pc = ctx->pc;
757 }
758 if (ctx->hflags != ctx->saved_hflags) {
759 tcg_gen_movi_i32(hflags, ctx->hflags);
760 ctx->saved_hflags = ctx->hflags;
761 switch (ctx->hflags & MIPS_HFLAG_BMASK) {
762 case MIPS_HFLAG_BR:
763 break;
764 case MIPS_HFLAG_BC:
765 case MIPS_HFLAG_BL:
766 case MIPS_HFLAG_B:
767 tcg_gen_movi_tl(btarget, ctx->btarget);
768 break;
769 }
770 }
771 }
772
restore_cpu_state(CPUState * env,DisasContext * ctx)773 static inline void restore_cpu_state (CPUState *env, DisasContext *ctx)
774 {
775 ctx->saved_hflags = ctx->hflags;
776 switch (ctx->hflags & MIPS_HFLAG_BMASK) {
777 case MIPS_HFLAG_BR:
778 break;
779 case MIPS_HFLAG_BC:
780 case MIPS_HFLAG_BL:
781 case MIPS_HFLAG_B:
782 ctx->btarget = env->btarget;
783 break;
784 }
785 }
786
787 static inline void
generate_exception_err(DisasContext * ctx,int excp,int err)788 generate_exception_err (DisasContext *ctx, int excp, int err)
789 {
790 TCGv_i32 texcp = tcg_const_i32(excp);
791 TCGv_i32 terr = tcg_const_i32(err);
792 save_cpu_state(ctx, 1);
793 gen_helper_raise_exception_err(texcp, terr);
794 tcg_temp_free_i32(terr);
795 tcg_temp_free_i32(texcp);
796 }
797
798 static inline void
generate_exception(DisasContext * ctx,int excp)799 generate_exception (DisasContext *ctx, int excp)
800 {
801 save_cpu_state(ctx, 1);
802 gen_helper_0i(raise_exception, excp);
803 }
804
805 /* Addresses computation */
gen_op_addr_add(DisasContext * ctx,TCGv ret,TCGv arg0,TCGv arg1)806 static inline void gen_op_addr_add (DisasContext *ctx, TCGv ret, TCGv arg0, TCGv arg1)
807 {
808 tcg_gen_add_tl(ret, arg0, arg1);
809
810 #if defined(TARGET_MIPS64)
811 /* For compatibility with 32-bit code, data reference in user mode
812 with Status_UX = 0 should be casted to 32-bit and sign extended.
813 See the MIPS64 PRA manual, section 4.10. */
814 if (((ctx->hflags & MIPS_HFLAG_KSU) == MIPS_HFLAG_UM) &&
815 !(ctx->hflags & MIPS_HFLAG_UX)) {
816 tcg_gen_ext32s_i64(ret, ret);
817 }
818 #endif
819 }
820
check_cp0_enabled(DisasContext * ctx)821 static inline void check_cp0_enabled(DisasContext *ctx)
822 {
823 if (unlikely(!(ctx->hflags & MIPS_HFLAG_CP0)))
824 generate_exception_err(ctx, EXCP_CpU, 0);
825 }
826
check_cp1_enabled(DisasContext * ctx)827 static inline void check_cp1_enabled(DisasContext *ctx)
828 {
829 if (unlikely(!(ctx->hflags & MIPS_HFLAG_FPU)))
830 generate_exception_err(ctx, EXCP_CpU, 1);
831 }
832
833 /* Verify that the processor is running with COP1X instructions enabled.
834 This is associated with the nabla symbol in the MIPS32 and MIPS64
835 opcode tables. */
836
check_cop1x(DisasContext * ctx)837 static inline void check_cop1x(DisasContext *ctx)
838 {
839 if (unlikely(!(ctx->hflags & MIPS_HFLAG_COP1X)))
840 generate_exception(ctx, EXCP_RI);
841 }
842
843 /* Verify that the processor is running with 64-bit floating-point
844 operations enabled. */
845
check_cp1_64bitmode(DisasContext * ctx)846 static inline void check_cp1_64bitmode(DisasContext *ctx)
847 {
848 if (unlikely(~ctx->hflags & (MIPS_HFLAG_F64 | MIPS_HFLAG_COP1X)))
849 generate_exception(ctx, EXCP_RI);
850 }
851
852 /*
853 * Verify if floating point register is valid; an operation is not defined
854 * if bit 0 of any register specification is set and the FR bit in the
855 * Status register equals zero, since the register numbers specify an
856 * even-odd pair of adjacent coprocessor general registers. When the FR bit
857 * in the Status register equals one, both even and odd register numbers
858 * are valid. This limitation exists only for 64 bit wide (d,l,ps) registers.
859 *
860 * Multiple 64 bit wide registers can be checked by calling
861 * gen_op_cp1_registers(freg1 | freg2 | ... | fregN);
862 */
check_cp1_registers(DisasContext * ctx,int regs)863 static inline void check_cp1_registers(DisasContext *ctx, int regs)
864 {
865 if (unlikely(!(ctx->hflags & MIPS_HFLAG_F64) && (regs & 1)))
866 generate_exception(ctx, EXCP_RI);
867 }
868
869 /* This code generates a "reserved instruction" exception if the
870 CPU does not support the instruction set corresponding to flags. */
check_insn(CPUState * env,DisasContext * ctx,int flags)871 static inline void check_insn(CPUState *env, DisasContext *ctx, int flags)
872 {
873 if (unlikely(!(env->insn_flags & flags)))
874 generate_exception(ctx, EXCP_RI);
875 }
876
877 /* This code generates a "reserved instruction" exception if 64-bit
878 instructions are not enabled. */
check_mips_64(DisasContext * ctx)879 static inline void check_mips_64(DisasContext *ctx)
880 {
881 if (unlikely(!(ctx->hflags & MIPS_HFLAG_64)))
882 generate_exception(ctx, EXCP_RI);
883 }
884
885 /* load/store instructions. */
886 #define OP_LD(insn,fname) \
887 static inline void op_ldst_##insn(TCGv ret, TCGv arg1, DisasContext *ctx) \
888 { \
889 tcg_gen_qemu_##fname(ret, arg1, ctx->mem_idx); \
890 }
891 OP_LD(lb,ld8s);
892 OP_LD(lbu,ld8u);
893 OP_LD(lh,ld16s);
894 OP_LD(lhu,ld16u);
895 OP_LD(lw,ld32s);
896 #if defined(TARGET_MIPS64)
897 OP_LD(lwu,ld32u);
898 OP_LD(ld,ld64);
899 #endif
900 #undef OP_LD
901
902 #define OP_ST(insn,fname) \
903 static inline void op_ldst_##insn(TCGv arg1, TCGv arg2, DisasContext *ctx) \
904 { \
905 tcg_gen_qemu_##fname(arg1, arg2, ctx->mem_idx); \
906 }
907 OP_ST(sb,st8);
908 OP_ST(sh,st16);
909 OP_ST(sw,st32);
910 #if defined(TARGET_MIPS64)
911 OP_ST(sd,st64);
912 #endif
913 #undef OP_ST
914
915 #ifdef CONFIG_USER_ONLY
916 #define OP_LD_ATOMIC(insn,fname) \
917 static inline void op_ldst_##insn(TCGv ret, TCGv arg1, DisasContext *ctx) \
918 { \
919 TCGv t0 = tcg_temp_new(); \
920 tcg_gen_mov_tl(t0, arg1); \
921 tcg_gen_qemu_##fname(ret, arg1, ctx->mem_idx); \
922 tcg_gen_st_tl(t0, cpu_env, offsetof(CPUState, lladdr)); \
923 tcg_gen_st_tl(ret, cpu_env, offsetof(CPUState, llval)); \
924 tcg_temp_free(t0); \
925 }
926 #else
927 #define OP_LD_ATOMIC(insn,fname) \
928 static inline void op_ldst_##insn(TCGv ret, TCGv arg1, DisasContext *ctx) \
929 { \
930 gen_helper_2i(insn, ret, arg1, ctx->mem_idx); \
931 }
932 #endif
933 OP_LD_ATOMIC(ll,ld32s);
934 #if defined(TARGET_MIPS64)
935 OP_LD_ATOMIC(lld,ld64);
936 #endif
937 #undef OP_LD_ATOMIC
938
939 #ifdef CONFIG_USER_ONLY
940 #define OP_ST_ATOMIC(insn,fname,ldname,almask) \
941 static inline void op_ldst_##insn(TCGv arg1, TCGv arg2, int rt, DisasContext *ctx) \
942 { \
943 TCGv t0 = tcg_temp_new(); \
944 int l1 = gen_new_label(); \
945 int l2 = gen_new_label(); \
946 \
947 tcg_gen_andi_tl(t0, arg2, almask); \
948 tcg_gen_brcondi_tl(TCG_COND_EQ, t0, 0, l1); \
949 tcg_gen_st_tl(arg2, cpu_env, offsetof(CPUState, CP0_BadVAddr)); \
950 generate_exception(ctx, EXCP_AdES); \
951 gen_set_label(l1); \
952 tcg_gen_ld_tl(t0, cpu_env, offsetof(CPUState, lladdr)); \
953 tcg_gen_brcond_tl(TCG_COND_NE, arg2, t0, l2); \
954 tcg_gen_movi_tl(t0, rt | ((almask << 3) & 0x20)); \
955 tcg_gen_st_tl(t0, cpu_env, offsetof(CPUState, llreg)); \
956 tcg_gen_st_tl(arg1, cpu_env, offsetof(CPUState, llnewval)); \
957 gen_helper_0i(raise_exception, EXCP_SC); \
958 gen_set_label(l2); \
959 tcg_gen_movi_tl(t0, 0); \
960 gen_store_gpr(t0, rt); \
961 tcg_temp_free(t0); \
962 }
963 #else
964 #define OP_ST_ATOMIC(insn,fname,ldname,almask) \
965 static inline void op_ldst_##insn(TCGv arg1, TCGv arg2, int rt, DisasContext *ctx) \
966 { \
967 TCGv t0 = tcg_temp_new(); \
968 gen_helper_3i(insn, t0, arg1, arg2, ctx->mem_idx); \
969 gen_store_gpr(t0, rt); \
970 tcg_temp_free(t0); \
971 }
972 #endif
973 OP_ST_ATOMIC(sc,st32,ld32s,0x3);
974 #if defined(TARGET_MIPS64)
975 OP_ST_ATOMIC(scd,st64,ld64,0x7);
976 #endif
977 #undef OP_ST_ATOMIC
978
979 /* Load and store */
gen_ldst(DisasContext * ctx,uint32_t opc,int rt,int base,int16_t offset)980 static void gen_ldst (DisasContext *ctx, uint32_t opc, int rt,
981 int base, int16_t offset)
982 {
983 const char *opn = "ldst";
984 TCGv t0 = tcg_temp_new();
985 TCGv t1 = tcg_temp_new();
986
987 if (base == 0) {
988 tcg_gen_movi_tl(t0, offset);
989 } else if (offset == 0) {
990 gen_load_gpr(t0, base);
991 } else {
992 tcg_gen_movi_tl(t0, offset);
993 gen_op_addr_add(ctx, t0, cpu_gpr[base], t0);
994 }
995 /* Don't do NOP if destination is zero: we must perform the actual
996 memory access. */
997 switch (opc) {
998 #if defined(TARGET_MIPS64)
999 case OPC_LWU:
1000 save_cpu_state(ctx, 0);
1001 op_ldst_lwu(t0, t0, ctx);
1002 gen_store_gpr(t0, rt);
1003 opn = "lwu";
1004 break;
1005 case OPC_LD:
1006 save_cpu_state(ctx, 0);
1007 op_ldst_ld(t0, t0, ctx);
1008 gen_store_gpr(t0, rt);
1009 opn = "ld";
1010 break;
1011 case OPC_LLD:
1012 save_cpu_state(ctx, 0);
1013 op_ldst_lld(t0, t0, ctx);
1014 gen_store_gpr(t0, rt);
1015 opn = "lld";
1016 break;
1017 case OPC_SD:
1018 save_cpu_state(ctx, 0);
1019 gen_load_gpr(t1, rt);
1020 op_ldst_sd(t1, t0, ctx);
1021 opn = "sd";
1022 break;
1023 case OPC_LDL:
1024 save_cpu_state(ctx, 1);
1025 gen_load_gpr(t1, rt);
1026 gen_helper_3i(ldl, t1, t1, t0, ctx->mem_idx);
1027 gen_store_gpr(t1, rt);
1028 opn = "ldl";
1029 break;
1030 case OPC_SDL:
1031 save_cpu_state(ctx, 1);
1032 gen_load_gpr(t1, rt);
1033 gen_helper_2i(sdl, t1, t0, ctx->mem_idx);
1034 opn = "sdl";
1035 break;
1036 case OPC_LDR:
1037 save_cpu_state(ctx, 1);
1038 gen_load_gpr(t1, rt);
1039 gen_helper_3i(ldr, t1, t1, t0, ctx->mem_idx);
1040 gen_store_gpr(t1, rt);
1041 opn = "ldr";
1042 break;
1043 case OPC_SDR:
1044 save_cpu_state(ctx, 1);
1045 gen_load_gpr(t1, rt);
1046 gen_helper_2i(sdr, t1, t0, ctx->mem_idx);
1047 opn = "sdr";
1048 break;
1049 #endif
1050 case OPC_LW:
1051 save_cpu_state(ctx, 0);
1052 op_ldst_lw(t0, t0, ctx);
1053 gen_store_gpr(t0, rt);
1054 opn = "lw";
1055 break;
1056 case OPC_SW:
1057 save_cpu_state(ctx, 0);
1058 gen_load_gpr(t1, rt);
1059 op_ldst_sw(t1, t0, ctx);
1060 opn = "sw";
1061 break;
1062 case OPC_LH:
1063 save_cpu_state(ctx, 0);
1064 op_ldst_lh(t0, t0, ctx);
1065 gen_store_gpr(t0, rt);
1066 opn = "lh";
1067 break;
1068 case OPC_SH:
1069 save_cpu_state(ctx, 0);
1070 gen_load_gpr(t1, rt);
1071 op_ldst_sh(t1, t0, ctx);
1072 opn = "sh";
1073 break;
1074 case OPC_LHU:
1075 save_cpu_state(ctx, 0);
1076 op_ldst_lhu(t0, t0, ctx);
1077 gen_store_gpr(t0, rt);
1078 opn = "lhu";
1079 break;
1080 case OPC_LB:
1081 save_cpu_state(ctx, 0);
1082 op_ldst_lb(t0, t0, ctx);
1083 gen_store_gpr(t0, rt);
1084 opn = "lb";
1085 break;
1086 case OPC_SB:
1087 save_cpu_state(ctx, 0);
1088 gen_load_gpr(t1, rt);
1089 op_ldst_sb(t1, t0, ctx);
1090 opn = "sb";
1091 break;
1092 case OPC_LBU:
1093 save_cpu_state(ctx, 0);
1094 op_ldst_lbu(t0, t0, ctx);
1095 gen_store_gpr(t0, rt);
1096 opn = "lbu";
1097 break;
1098 case OPC_LWL:
1099 save_cpu_state(ctx, 1);
1100 gen_load_gpr(t1, rt);
1101 gen_helper_3i(lwl, t1, t1, t0, ctx->mem_idx);
1102 gen_store_gpr(t1, rt);
1103 opn = "lwl";
1104 break;
1105 case OPC_SWL:
1106 save_cpu_state(ctx, 1);
1107 gen_load_gpr(t1, rt);
1108 gen_helper_2i(swl, t1, t0, ctx->mem_idx);
1109 opn = "swr";
1110 break;
1111 case OPC_LWR:
1112 save_cpu_state(ctx, 1);
1113 gen_load_gpr(t1, rt);
1114 gen_helper_3i(lwr, t1, t1, t0, ctx->mem_idx);
1115 gen_store_gpr(t1, rt);
1116 opn = "lwr";
1117 break;
1118 case OPC_SWR:
1119 save_cpu_state(ctx, 1);
1120 gen_load_gpr(t1, rt);
1121 gen_helper_2i(swr, t1, t0, ctx->mem_idx);
1122 opn = "swr";
1123 break;
1124 case OPC_LL:
1125 save_cpu_state(ctx, 1);
1126 op_ldst_ll(t0, t0, ctx);
1127 gen_store_gpr(t0, rt);
1128 opn = "ll";
1129 break;
1130 }
1131 MIPS_DEBUG("%s %s, %d(%s)", opn, regnames[rt], offset, regnames[base]);
1132 tcg_temp_free(t0);
1133 tcg_temp_free(t1);
1134 }
1135
1136 /* Store conditional */
gen_st_cond(DisasContext * ctx,uint32_t opc,int rt,int base,int16_t offset)1137 static void gen_st_cond (DisasContext *ctx, uint32_t opc, int rt,
1138 int base, int16_t offset)
1139 {
1140 const char *opn = "st_cond";
1141 TCGv t0, t1;
1142
1143 t0 = tcg_temp_local_new();
1144
1145 if (base == 0) {
1146 tcg_gen_movi_tl(t0, offset);
1147 } else if (offset == 0) {
1148 gen_load_gpr(t0, base);
1149 } else {
1150 tcg_gen_movi_tl(t0, offset);
1151 gen_op_addr_add(ctx, t0, cpu_gpr[base], t0);
1152 }
1153 /* Don't do NOP if destination is zero: we must perform the actual
1154 memory access. */
1155
1156 t1 = tcg_temp_local_new();
1157 gen_load_gpr(t1, rt);
1158 switch (opc) {
1159 #if defined(TARGET_MIPS64)
1160 case OPC_SCD:
1161 save_cpu_state(ctx, 0);
1162 op_ldst_scd(t1, t0, rt, ctx);
1163 opn = "scd";
1164 break;
1165 #endif
1166 case OPC_SC:
1167 save_cpu_state(ctx, 1);
1168 op_ldst_sc(t1, t0, rt, ctx);
1169 opn = "sc";
1170 break;
1171 }
1172 MIPS_DEBUG("%s %s, %d(%s)", opn, regnames[rt], offset, regnames[base]);
1173 tcg_temp_free(t1);
1174 tcg_temp_free(t0);
1175 }
1176
1177 /* Load and store */
gen_flt_ldst(DisasContext * ctx,uint32_t opc,int ft,int base,int16_t offset)1178 static void gen_flt_ldst (DisasContext *ctx, uint32_t opc, int ft,
1179 int base, int16_t offset)
1180 {
1181 const char *opn = "flt_ldst";
1182 TCGv t0 = tcg_temp_new();
1183
1184 if (base == 0) {
1185 tcg_gen_movi_tl(t0, offset);
1186 } else if (offset == 0) {
1187 gen_load_gpr(t0, base);
1188 } else {
1189 tcg_gen_movi_tl(t0, offset);
1190 gen_op_addr_add(ctx, t0, cpu_gpr[base], t0);
1191 }
1192 /* Don't do NOP if destination is zero: we must perform the actual
1193 memory access. */
1194 switch (opc) {
1195 case OPC_LWC1:
1196 {
1197 TCGv_i32 fp0 = tcg_temp_new_i32();
1198
1199 tcg_gen_qemu_ld32s(t0, t0, ctx->mem_idx);
1200 tcg_gen_trunc_tl_i32(fp0, t0);
1201 gen_store_fpr32(fp0, ft);
1202 tcg_temp_free_i32(fp0);
1203 }
1204 opn = "lwc1";
1205 break;
1206 case OPC_SWC1:
1207 {
1208 TCGv_i32 fp0 = tcg_temp_new_i32();
1209 TCGv t1 = tcg_temp_new();
1210
1211 gen_load_fpr32(fp0, ft);
1212 tcg_gen_extu_i32_tl(t1, fp0);
1213 tcg_gen_qemu_st32(t1, t0, ctx->mem_idx);
1214 tcg_temp_free(t1);
1215 tcg_temp_free_i32(fp0);
1216 }
1217 opn = "swc1";
1218 break;
1219 case OPC_LDC1:
1220 {
1221 TCGv_i64 fp0 = tcg_temp_new_i64();
1222
1223 tcg_gen_qemu_ld64(fp0, t0, ctx->mem_idx);
1224 gen_store_fpr64(ctx, fp0, ft);
1225 tcg_temp_free_i64(fp0);
1226 }
1227 opn = "ldc1";
1228 break;
1229 case OPC_SDC1:
1230 {
1231 TCGv_i64 fp0 = tcg_temp_new_i64();
1232
1233 gen_load_fpr64(ctx, fp0, ft);
1234 tcg_gen_qemu_st64(fp0, t0, ctx->mem_idx);
1235 tcg_temp_free_i64(fp0);
1236 }
1237 opn = "sdc1";
1238 break;
1239 default:
1240 MIPS_INVAL(opn);
1241 generate_exception(ctx, EXCP_RI);
1242 goto out;
1243 }
1244 MIPS_DEBUG("%s %s, %d(%s)", opn, fregnames[ft], offset, regnames[base]);
1245 out:
1246 tcg_temp_free(t0);
1247 }
1248
1249 /* Arithmetic with immediate operand */
gen_arith_imm(CPUState * env,DisasContext * ctx,uint32_t opc,int rt,int rs,int16_t imm)1250 static void gen_arith_imm (CPUState *env, DisasContext *ctx, uint32_t opc,
1251 int rt, int rs, int16_t imm)
1252 {
1253 target_ulong uimm = (target_long)imm; /* Sign extend to 32/64 bits */
1254 const char *opn = "imm arith";
1255
1256 if (rt == 0 && opc != OPC_ADDI && opc != OPC_DADDI) {
1257 /* If no destination, treat it as a NOP.
1258 For addi, we must generate the overflow exception when needed. */
1259 MIPS_DEBUG("NOP");
1260 return;
1261 }
1262 switch (opc) {
1263 case OPC_ADDI:
1264 {
1265 TCGv t0 = tcg_temp_local_new();
1266 TCGv t1 = tcg_temp_new();
1267 TCGv t2 = tcg_temp_new();
1268 int l1 = gen_new_label();
1269
1270 gen_load_gpr(t1, rs);
1271 tcg_gen_addi_tl(t0, t1, uimm);
1272 tcg_gen_ext32s_tl(t0, t0);
1273
1274 tcg_gen_xori_tl(t1, t1, ~uimm);
1275 tcg_gen_xori_tl(t2, t0, uimm);
1276 tcg_gen_and_tl(t1, t1, t2);
1277 tcg_temp_free(t2);
1278 tcg_gen_brcondi_tl(TCG_COND_GE, t1, 0, l1);
1279 tcg_temp_free(t1);
1280 /* operands of same sign, result different sign */
1281 generate_exception(ctx, EXCP_OVERFLOW);
1282 gen_set_label(l1);
1283 tcg_gen_ext32s_tl(t0, t0);
1284 gen_store_gpr(t0, rt);
1285 tcg_temp_free(t0);
1286 }
1287 opn = "addi";
1288 break;
1289 case OPC_ADDIU:
1290 if (rs != 0) {
1291 tcg_gen_addi_tl(cpu_gpr[rt], cpu_gpr[rs], uimm);
1292 tcg_gen_ext32s_tl(cpu_gpr[rt], cpu_gpr[rt]);
1293 } else {
1294 tcg_gen_movi_tl(cpu_gpr[rt], uimm);
1295 }
1296 opn = "addiu";
1297 break;
1298 #if defined(TARGET_MIPS64)
1299 case OPC_DADDI:
1300 {
1301 TCGv t0 = tcg_temp_local_new();
1302 TCGv t1 = tcg_temp_new();
1303 TCGv t2 = tcg_temp_new();
1304 int l1 = gen_new_label();
1305
1306 gen_load_gpr(t1, rs);
1307 tcg_gen_addi_tl(t0, t1, uimm);
1308
1309 tcg_gen_xori_tl(t1, t1, ~uimm);
1310 tcg_gen_xori_tl(t2, t0, uimm);
1311 tcg_gen_and_tl(t1, t1, t2);
1312 tcg_temp_free(t2);
1313 tcg_gen_brcondi_tl(TCG_COND_GE, t1, 0, l1);
1314 tcg_temp_free(t1);
1315 /* operands of same sign, result different sign */
1316 generate_exception(ctx, EXCP_OVERFLOW);
1317 gen_set_label(l1);
1318 gen_store_gpr(t0, rt);
1319 tcg_temp_free(t0);
1320 }
1321 opn = "daddi";
1322 break;
1323 case OPC_DADDIU:
1324 if (rs != 0) {
1325 tcg_gen_addi_tl(cpu_gpr[rt], cpu_gpr[rs], uimm);
1326 } else {
1327 tcg_gen_movi_tl(cpu_gpr[rt], uimm);
1328 }
1329 opn = "daddiu";
1330 break;
1331 #endif
1332 }
1333 MIPS_DEBUG("%s %s, %s, " TARGET_FMT_lx, opn, regnames[rt], regnames[rs], uimm);
1334 }
1335
1336 /* Logic with immediate operand */
gen_logic_imm(CPUState * env,uint32_t opc,int rt,int rs,int16_t imm)1337 static void gen_logic_imm (CPUState *env, uint32_t opc, int rt, int rs, int16_t imm)
1338 {
1339 target_ulong uimm;
1340 const char *opn = "imm logic";
1341
1342 if (rt == 0) {
1343 /* If no destination, treat it as a NOP. */
1344 MIPS_DEBUG("NOP");
1345 return;
1346 }
1347 uimm = (uint16_t)imm;
1348 switch (opc) {
1349 case OPC_ANDI:
1350 if (likely(rs != 0))
1351 tcg_gen_andi_tl(cpu_gpr[rt], cpu_gpr[rs], uimm);
1352 else
1353 tcg_gen_movi_tl(cpu_gpr[rt], 0);
1354 opn = "andi";
1355 break;
1356 case OPC_ORI:
1357 if (rs != 0)
1358 tcg_gen_ori_tl(cpu_gpr[rt], cpu_gpr[rs], uimm);
1359 else
1360 tcg_gen_movi_tl(cpu_gpr[rt], uimm);
1361 opn = "ori";
1362 break;
1363 case OPC_XORI:
1364 if (likely(rs != 0))
1365 tcg_gen_xori_tl(cpu_gpr[rt], cpu_gpr[rs], uimm);
1366 else
1367 tcg_gen_movi_tl(cpu_gpr[rt], uimm);
1368 opn = "xori";
1369 break;
1370 case OPC_LUI:
1371 tcg_gen_movi_tl(cpu_gpr[rt], imm << 16);
1372 opn = "lui";
1373 break;
1374 }
1375 MIPS_DEBUG("%s %s, %s, " TARGET_FMT_lx, opn, regnames[rt], regnames[rs], uimm);
1376 }
1377
1378 /* Set on less than with immediate operand */
gen_slt_imm(CPUState * env,uint32_t opc,int rt,int rs,int16_t imm)1379 static void gen_slt_imm (CPUState *env, uint32_t opc, int rt, int rs, int16_t imm)
1380 {
1381 target_ulong uimm = (target_long)imm; /* Sign extend to 32/64 bits */
1382 const char *opn = "imm arith";
1383 TCGv t0;
1384
1385 if (rt == 0) {
1386 /* If no destination, treat it as a NOP. */
1387 MIPS_DEBUG("NOP");
1388 return;
1389 }
1390 t0 = tcg_temp_new();
1391 gen_load_gpr(t0, rs);
1392 switch (opc) {
1393 case OPC_SLTI:
1394 gen_op_lti(cpu_gpr[rt], t0, uimm);
1395 opn = "slti";
1396 break;
1397 case OPC_SLTIU:
1398 gen_op_ltiu(cpu_gpr[rt], t0, uimm);
1399 opn = "sltiu";
1400 break;
1401 }
1402 MIPS_DEBUG("%s %s, %s, " TARGET_FMT_lx, opn, regnames[rt], regnames[rs], uimm);
1403 tcg_temp_free(t0);
1404 }
1405
1406 /* Shifts with immediate operand */
gen_shift_imm(CPUState * env,DisasContext * ctx,uint32_t opc,int rt,int rs,int16_t imm)1407 static void gen_shift_imm(CPUState *env, DisasContext *ctx, uint32_t opc,
1408 int rt, int rs, int16_t imm)
1409 {
1410 target_ulong uimm = ((uint16_t)imm) & 0x1f;
1411 const char *opn = "imm shift";
1412 TCGv t0;
1413
1414 if (rt == 0) {
1415 /* If no destination, treat it as a NOP. */
1416 MIPS_DEBUG("NOP");
1417 return;
1418 }
1419
1420 t0 = tcg_temp_new();
1421 gen_load_gpr(t0, rs);
1422 switch (opc) {
1423 case OPC_SLL:
1424 tcg_gen_shli_tl(t0, t0, uimm);
1425 tcg_gen_ext32s_tl(cpu_gpr[rt], t0);
1426 opn = "sll";
1427 break;
1428 case OPC_SRA:
1429 tcg_gen_ext32s_tl(t0, t0);
1430 tcg_gen_sari_tl(cpu_gpr[rt], t0, uimm);
1431 opn = "sra";
1432 break;
1433 case OPC_SRL:
1434 switch ((ctx->opcode >> 21) & 0x1f) {
1435 case 0:
1436 if (uimm != 0) {
1437 tcg_gen_ext32u_tl(t0, t0);
1438 tcg_gen_shri_tl(cpu_gpr[rt], t0, uimm);
1439 } else {
1440 tcg_gen_ext32s_tl(cpu_gpr[rt], t0);
1441 }
1442 opn = "srl";
1443 break;
1444 case 1:
1445 /* rotr is decoded as srl on non-R2 CPUs */
1446 if (env->insn_flags & ISA_MIPS32R2) {
1447 if (uimm != 0) {
1448 TCGv_i32 t1 = tcg_temp_new_i32();
1449
1450 tcg_gen_trunc_tl_i32(t1, t0);
1451 tcg_gen_rotri_i32(t1, t1, uimm);
1452 tcg_gen_ext_i32_tl(cpu_gpr[rt], t1);
1453 tcg_temp_free_i32(t1);
1454 } else {
1455 tcg_gen_ext32s_tl(cpu_gpr[rt], t0);
1456 }
1457 opn = "rotr";
1458 } else {
1459 if (uimm != 0) {
1460 tcg_gen_ext32u_tl(t0, t0);
1461 tcg_gen_shri_tl(cpu_gpr[rt], t0, uimm);
1462 } else {
1463 tcg_gen_ext32s_tl(cpu_gpr[rt], t0);
1464 }
1465 opn = "srl";
1466 }
1467 break;
1468 default:
1469 MIPS_INVAL("invalid srl flag");
1470 generate_exception(ctx, EXCP_RI);
1471 break;
1472 }
1473 break;
1474 #if defined(TARGET_MIPS64)
1475 case OPC_DSLL:
1476 tcg_gen_shli_tl(cpu_gpr[rt], t0, uimm);
1477 opn = "dsll";
1478 break;
1479 case OPC_DSRA:
1480 tcg_gen_sari_tl(cpu_gpr[rt], t0, uimm);
1481 opn = "dsra";
1482 break;
1483 case OPC_DSRL:
1484 switch ((ctx->opcode >> 21) & 0x1f) {
1485 case 0:
1486 tcg_gen_shri_tl(cpu_gpr[rt], t0, uimm);
1487 opn = "dsrl";
1488 break;
1489 case 1:
1490 /* drotr is decoded as dsrl on non-R2 CPUs */
1491 if (env->insn_flags & ISA_MIPS32R2) {
1492 if (uimm != 0) {
1493 tcg_gen_rotri_tl(cpu_gpr[rt], t0, uimm);
1494 } else {
1495 tcg_gen_mov_tl(cpu_gpr[rt], t0);
1496 }
1497 opn = "drotr";
1498 } else {
1499 tcg_gen_shri_tl(cpu_gpr[rt], t0, uimm);
1500 opn = "dsrl";
1501 }
1502 break;
1503 default:
1504 MIPS_INVAL("invalid dsrl flag");
1505 generate_exception(ctx, EXCP_RI);
1506 break;
1507 }
1508 break;
1509 case OPC_DSLL32:
1510 tcg_gen_shli_tl(cpu_gpr[rt], t0, uimm + 32);
1511 opn = "dsll32";
1512 break;
1513 case OPC_DSRA32:
1514 tcg_gen_sari_tl(cpu_gpr[rt], t0, uimm + 32);
1515 opn = "dsra32";
1516 break;
1517 case OPC_DSRL32:
1518 switch ((ctx->opcode >> 21) & 0x1f) {
1519 case 0:
1520 tcg_gen_shri_tl(cpu_gpr[rt], t0, uimm + 32);
1521 opn = "dsrl32";
1522 break;
1523 case 1:
1524 /* drotr32 is decoded as dsrl32 on non-R2 CPUs */
1525 if (env->insn_flags & ISA_MIPS32R2) {
1526 tcg_gen_rotri_tl(cpu_gpr[rt], t0, uimm + 32);
1527 opn = "drotr32";
1528 } else {
1529 tcg_gen_shri_tl(cpu_gpr[rt], t0, uimm + 32);
1530 opn = "dsrl32";
1531 }
1532 break;
1533 default:
1534 MIPS_INVAL("invalid dsrl32 flag");
1535 generate_exception(ctx, EXCP_RI);
1536 break;
1537 }
1538 break;
1539 #endif
1540 }
1541 MIPS_DEBUG("%s %s, %s, " TARGET_FMT_lx, opn, regnames[rt], regnames[rs], uimm);
1542 tcg_temp_free(t0);
1543 }
1544
1545 /* Arithmetic */
gen_arith(CPUState * env,DisasContext * ctx,uint32_t opc,int rd,int rs,int rt)1546 static void gen_arith (CPUState *env, DisasContext *ctx, uint32_t opc,
1547 int rd, int rs, int rt)
1548 {
1549 const char *opn = "arith";
1550
1551 if (rd == 0 && opc != OPC_ADD && opc != OPC_SUB
1552 && opc != OPC_DADD && opc != OPC_DSUB) {
1553 /* If no destination, treat it as a NOP.
1554 For add & sub, we must generate the overflow exception when needed. */
1555 MIPS_DEBUG("NOP");
1556 return;
1557 }
1558
1559 switch (opc) {
1560 case OPC_ADD:
1561 {
1562 TCGv t0 = tcg_temp_local_new();
1563 TCGv t1 = tcg_temp_new();
1564 TCGv t2 = tcg_temp_new();
1565 int l1 = gen_new_label();
1566
1567 gen_load_gpr(t1, rs);
1568 gen_load_gpr(t2, rt);
1569 tcg_gen_add_tl(t0, t1, t2);
1570 tcg_gen_ext32s_tl(t0, t0);
1571 tcg_gen_xor_tl(t1, t1, t2);
1572 tcg_gen_not_tl(t1, t1);
1573 tcg_gen_xor_tl(t2, t0, t2);
1574 tcg_gen_and_tl(t1, t1, t2);
1575 tcg_temp_free(t2);
1576 tcg_gen_brcondi_tl(TCG_COND_GE, t1, 0, l1);
1577 tcg_temp_free(t1);
1578 /* operands of same sign, result different sign */
1579 generate_exception(ctx, EXCP_OVERFLOW);
1580 gen_set_label(l1);
1581 gen_store_gpr(t0, rd);
1582 tcg_temp_free(t0);
1583 }
1584 opn = "add";
1585 break;
1586 case OPC_ADDU:
1587 if (rs != 0 && rt != 0) {
1588 tcg_gen_add_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1589 tcg_gen_ext32s_tl(cpu_gpr[rd], cpu_gpr[rd]);
1590 } else if (rs == 0 && rt != 0) {
1591 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rt]);
1592 } else if (rs != 0 && rt == 0) {
1593 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1594 } else {
1595 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1596 }
1597 opn = "addu";
1598 break;
1599 case OPC_SUB:
1600 {
1601 TCGv t0 = tcg_temp_local_new();
1602 TCGv t1 = tcg_temp_new();
1603 TCGv t2 = tcg_temp_new();
1604 int l1 = gen_new_label();
1605
1606 gen_load_gpr(t1, rs);
1607 gen_load_gpr(t2, rt);
1608 tcg_gen_sub_tl(t0, t1, t2);
1609 tcg_gen_ext32s_tl(t0, t0);
1610 tcg_gen_xor_tl(t2, t1, t2);
1611 tcg_gen_xor_tl(t1, t0, t1);
1612 tcg_gen_and_tl(t1, t1, t2);
1613 tcg_temp_free(t2);
1614 tcg_gen_brcondi_tl(TCG_COND_GE, t1, 0, l1);
1615 tcg_temp_free(t1);
1616 /* operands of different sign, first operand and result different sign */
1617 generate_exception(ctx, EXCP_OVERFLOW);
1618 gen_set_label(l1);
1619 gen_store_gpr(t0, rd);
1620 tcg_temp_free(t0);
1621 }
1622 opn = "sub";
1623 break;
1624 case OPC_SUBU:
1625 if (rs != 0 && rt != 0) {
1626 tcg_gen_sub_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1627 tcg_gen_ext32s_tl(cpu_gpr[rd], cpu_gpr[rd]);
1628 } else if (rs == 0 && rt != 0) {
1629 tcg_gen_neg_tl(cpu_gpr[rd], cpu_gpr[rt]);
1630 tcg_gen_ext32s_tl(cpu_gpr[rd], cpu_gpr[rd]);
1631 } else if (rs != 0 && rt == 0) {
1632 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1633 } else {
1634 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1635 }
1636 opn = "subu";
1637 break;
1638 #if defined(TARGET_MIPS64)
1639 case OPC_DADD:
1640 {
1641 TCGv t0 = tcg_temp_local_new();
1642 TCGv t1 = tcg_temp_new();
1643 TCGv t2 = tcg_temp_new();
1644 int l1 = gen_new_label();
1645
1646 gen_load_gpr(t1, rs);
1647 gen_load_gpr(t2, rt);
1648 tcg_gen_add_tl(t0, t1, t2);
1649 tcg_gen_xor_tl(t1, t1, t2);
1650 tcg_gen_not_tl(t1, t1);
1651 tcg_gen_xor_tl(t2, t0, t2);
1652 tcg_gen_and_tl(t1, t1, t2);
1653 tcg_temp_free(t2);
1654 tcg_gen_brcondi_tl(TCG_COND_GE, t1, 0, l1);
1655 tcg_temp_free(t1);
1656 /* operands of same sign, result different sign */
1657 generate_exception(ctx, EXCP_OVERFLOW);
1658 gen_set_label(l1);
1659 gen_store_gpr(t0, rd);
1660 tcg_temp_free(t0);
1661 }
1662 opn = "dadd";
1663 break;
1664 case OPC_DADDU:
1665 if (rs != 0 && rt != 0) {
1666 tcg_gen_add_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1667 } else if (rs == 0 && rt != 0) {
1668 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rt]);
1669 } else if (rs != 0 && rt == 0) {
1670 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1671 } else {
1672 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1673 }
1674 opn = "daddu";
1675 break;
1676 case OPC_DSUB:
1677 {
1678 TCGv t0 = tcg_temp_local_new();
1679 TCGv t1 = tcg_temp_new();
1680 TCGv t2 = tcg_temp_new();
1681 int l1 = gen_new_label();
1682
1683 gen_load_gpr(t1, rs);
1684 gen_load_gpr(t2, rt);
1685 tcg_gen_sub_tl(t0, t1, t2);
1686 tcg_gen_xor_tl(t2, t1, t2);
1687 tcg_gen_xor_tl(t1, t0, t1);
1688 tcg_gen_and_tl(t1, t1, t2);
1689 tcg_temp_free(t2);
1690 tcg_gen_brcondi_tl(TCG_COND_GE, t1, 0, l1);
1691 tcg_temp_free(t1);
1692 /* operands of different sign, first operand and result different sign */
1693 generate_exception(ctx, EXCP_OVERFLOW);
1694 gen_set_label(l1);
1695 gen_store_gpr(t0, rd);
1696 tcg_temp_free(t0);
1697 }
1698 opn = "dsub";
1699 break;
1700 case OPC_DSUBU:
1701 if (rs != 0 && rt != 0) {
1702 tcg_gen_sub_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1703 } else if (rs == 0 && rt != 0) {
1704 tcg_gen_neg_tl(cpu_gpr[rd], cpu_gpr[rt]);
1705 } else if (rs != 0 && rt == 0) {
1706 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1707 } else {
1708 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1709 }
1710 opn = "dsubu";
1711 break;
1712 #endif
1713 case OPC_MUL:
1714 if (likely(rs != 0 && rt != 0)) {
1715 tcg_gen_mul_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1716 tcg_gen_ext32s_tl(cpu_gpr[rd], cpu_gpr[rd]);
1717 } else {
1718 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1719 }
1720 opn = "mul";
1721 break;
1722 }
1723 MIPS_DEBUG("%s %s, %s, %s", opn, regnames[rd], regnames[rs], regnames[rt]);
1724 }
1725
1726 /* Conditional move */
gen_cond_move(CPUState * env,uint32_t opc,int rd,int rs,int rt)1727 static void gen_cond_move (CPUState *env, uint32_t opc, int rd, int rs, int rt)
1728 {
1729 const char *opn = "cond move";
1730 int l1;
1731
1732 if (rd == 0) {
1733 /* If no destination, treat it as a NOP.
1734 For add & sub, we must generate the overflow exception when needed. */
1735 MIPS_DEBUG("NOP");
1736 return;
1737 }
1738
1739 l1 = gen_new_label();
1740 switch (opc) {
1741 case OPC_MOVN:
1742 if (likely(rt != 0))
1743 tcg_gen_brcondi_tl(TCG_COND_EQ, cpu_gpr[rt], 0, l1);
1744 else
1745 tcg_gen_br(l1);
1746 opn = "movn";
1747 break;
1748 case OPC_MOVZ:
1749 if (likely(rt != 0))
1750 tcg_gen_brcondi_tl(TCG_COND_NE, cpu_gpr[rt], 0, l1);
1751 opn = "movz";
1752 break;
1753 }
1754 if (rs != 0)
1755 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1756 else
1757 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1758 gen_set_label(l1);
1759
1760 MIPS_DEBUG("%s %s, %s, %s", opn, regnames[rd], regnames[rs], regnames[rt]);
1761 }
1762
1763 /* Logic */
gen_logic(CPUState * env,uint32_t opc,int rd,int rs,int rt)1764 static void gen_logic (CPUState *env, uint32_t opc, int rd, int rs, int rt)
1765 {
1766 const char *opn = "logic";
1767
1768 if (rd == 0) {
1769 /* If no destination, treat it as a NOP. */
1770 MIPS_DEBUG("NOP");
1771 return;
1772 }
1773
1774 switch (opc) {
1775 case OPC_AND:
1776 if (likely(rs != 0 && rt != 0)) {
1777 tcg_gen_and_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1778 } else {
1779 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1780 }
1781 opn = "and";
1782 break;
1783 case OPC_NOR:
1784 if (rs != 0 && rt != 0) {
1785 tcg_gen_nor_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1786 } else if (rs == 0 && rt != 0) {
1787 tcg_gen_not_tl(cpu_gpr[rd], cpu_gpr[rt]);
1788 } else if (rs != 0 && rt == 0) {
1789 tcg_gen_not_tl(cpu_gpr[rd], cpu_gpr[rs]);
1790 } else {
1791 tcg_gen_movi_tl(cpu_gpr[rd], ~((target_ulong)0));
1792 }
1793 opn = "nor";
1794 break;
1795 case OPC_OR:
1796 if (likely(rs != 0 && rt != 0)) {
1797 tcg_gen_or_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1798 } else if (rs == 0 && rt != 0) {
1799 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rt]);
1800 } else if (rs != 0 && rt == 0) {
1801 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1802 } else {
1803 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1804 }
1805 opn = "or";
1806 break;
1807 case OPC_XOR:
1808 if (likely(rs != 0 && rt != 0)) {
1809 tcg_gen_xor_tl(cpu_gpr[rd], cpu_gpr[rs], cpu_gpr[rt]);
1810 } else if (rs == 0 && rt != 0) {
1811 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rt]);
1812 } else if (rs != 0 && rt == 0) {
1813 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
1814 } else {
1815 tcg_gen_movi_tl(cpu_gpr[rd], 0);
1816 }
1817 opn = "xor";
1818 break;
1819 }
1820 MIPS_DEBUG("%s %s, %s, %s", opn, regnames[rd], regnames[rs], regnames[rt]);
1821 }
1822
1823 /* Set on lower than */
gen_slt(CPUState * env,uint32_t opc,int rd,int rs,int rt)1824 static void gen_slt (CPUState *env, uint32_t opc, int rd, int rs, int rt)
1825 {
1826 const char *opn = "slt";
1827 TCGv t0, t1;
1828
1829 if (rd == 0) {
1830 /* If no destination, treat it as a NOP. */
1831 MIPS_DEBUG("NOP");
1832 return;
1833 }
1834
1835 t0 = tcg_temp_new();
1836 t1 = tcg_temp_new();
1837 gen_load_gpr(t0, rs);
1838 gen_load_gpr(t1, rt);
1839 switch (opc) {
1840 case OPC_SLT:
1841 gen_op_lt(cpu_gpr[rd], t0, t1);
1842 opn = "slt";
1843 break;
1844 case OPC_SLTU:
1845 gen_op_ltu(cpu_gpr[rd], t0, t1);
1846 opn = "sltu";
1847 break;
1848 }
1849 MIPS_DEBUG("%s %s, %s, %s", opn, regnames[rd], regnames[rs], regnames[rt]);
1850 tcg_temp_free(t0);
1851 tcg_temp_free(t1);
1852 }
1853
1854 /* Shifts */
gen_shift(CPUState * env,DisasContext * ctx,uint32_t opc,int rd,int rs,int rt)1855 static void gen_shift (CPUState *env, DisasContext *ctx, uint32_t opc,
1856 int rd, int rs, int rt)
1857 {
1858 const char *opn = "shifts";
1859 TCGv t0, t1;
1860
1861 if (rd == 0) {
1862 /* If no destination, treat it as a NOP.
1863 For add & sub, we must generate the overflow exception when needed. */
1864 MIPS_DEBUG("NOP");
1865 return;
1866 }
1867
1868 t0 = tcg_temp_new();
1869 t1 = tcg_temp_new();
1870 gen_load_gpr(t0, rs);
1871 gen_load_gpr(t1, rt);
1872 switch (opc) {
1873 case OPC_SLLV:
1874 tcg_gen_andi_tl(t0, t0, 0x1f);
1875 tcg_gen_shl_tl(t0, t1, t0);
1876 tcg_gen_ext32s_tl(cpu_gpr[rd], t0);
1877 opn = "sllv";
1878 break;
1879 case OPC_SRAV:
1880 tcg_gen_ext32s_tl(t1, t1);
1881 tcg_gen_andi_tl(t0, t0, 0x1f);
1882 tcg_gen_sar_tl(cpu_gpr[rd], t1, t0);
1883 opn = "srav";
1884 break;
1885 case OPC_SRLV:
1886 switch ((ctx->opcode >> 6) & 0x1f) {
1887 case 0:
1888 tcg_gen_ext32u_tl(t1, t1);
1889 tcg_gen_andi_tl(t0, t0, 0x1f);
1890 tcg_gen_shr_tl(t0, t1, t0);
1891 tcg_gen_ext32s_tl(cpu_gpr[rd], t0);
1892 opn = "srlv";
1893 break;
1894 case 1:
1895 /* rotrv is decoded as srlv on non-R2 CPUs */
1896 if (env->insn_flags & ISA_MIPS32R2) {
1897 TCGv_i32 t2 = tcg_temp_new_i32();
1898 TCGv_i32 t3 = tcg_temp_new_i32();
1899
1900 tcg_gen_trunc_tl_i32(t2, t0);
1901 tcg_gen_trunc_tl_i32(t3, t1);
1902 tcg_gen_andi_i32(t2, t2, 0x1f);
1903 tcg_gen_rotr_i32(t2, t3, t2);
1904 tcg_gen_ext_i32_tl(cpu_gpr[rd], t2);
1905 tcg_temp_free_i32(t2);
1906 tcg_temp_free_i32(t3);
1907 opn = "rotrv";
1908 } else {
1909 tcg_gen_ext32u_tl(t1, t1);
1910 tcg_gen_andi_tl(t0, t0, 0x1f);
1911 tcg_gen_shr_tl(t0, t1, t0);
1912 tcg_gen_ext32s_tl(cpu_gpr[rd], t0);
1913 opn = "srlv";
1914 }
1915 break;
1916 default:
1917 MIPS_INVAL("invalid srlv flag");
1918 generate_exception(ctx, EXCP_RI);
1919 break;
1920 }
1921 break;
1922 #if defined(TARGET_MIPS64)
1923 case OPC_DSLLV:
1924 tcg_gen_andi_tl(t0, t0, 0x3f);
1925 tcg_gen_shl_tl(cpu_gpr[rd], t1, t0);
1926 opn = "dsllv";
1927 break;
1928 case OPC_DSRAV:
1929 tcg_gen_andi_tl(t0, t0, 0x3f);
1930 tcg_gen_sar_tl(cpu_gpr[rd], t1, t0);
1931 opn = "dsrav";
1932 break;
1933 case OPC_DSRLV:
1934 switch ((ctx->opcode >> 6) & 0x1f) {
1935 case 0:
1936 tcg_gen_andi_tl(t0, t0, 0x3f);
1937 tcg_gen_shr_tl(cpu_gpr[rd], t1, t0);
1938 opn = "dsrlv";
1939 break;
1940 case 1:
1941 /* drotrv is decoded as dsrlv on non-R2 CPUs */
1942 if (env->insn_flags & ISA_MIPS32R2) {
1943 tcg_gen_andi_tl(t0, t0, 0x3f);
1944 tcg_gen_rotr_tl(cpu_gpr[rd], t1, t0);
1945 opn = "drotrv";
1946 } else {
1947 tcg_gen_andi_tl(t0, t0, 0x3f);
1948 tcg_gen_shr_tl(t0, t1, t0);
1949 opn = "dsrlv";
1950 }
1951 break;
1952 default:
1953 MIPS_INVAL("invalid dsrlv flag");
1954 generate_exception(ctx, EXCP_RI);
1955 break;
1956 }
1957 break;
1958 #endif
1959 }
1960 MIPS_DEBUG("%s %s, %s, %s", opn, regnames[rd], regnames[rs], regnames[rt]);
1961 tcg_temp_free(t0);
1962 tcg_temp_free(t1);
1963 }
1964
1965 /* Arithmetic on HI/LO registers */
gen_HILO(DisasContext * ctx,uint32_t opc,int reg)1966 static void gen_HILO (DisasContext *ctx, uint32_t opc, int reg)
1967 {
1968 const char *opn = "hilo";
1969
1970 if (reg == 0 && (opc == OPC_MFHI || opc == OPC_MFLO)) {
1971 /* Treat as NOP. */
1972 MIPS_DEBUG("NOP");
1973 return;
1974 }
1975 switch (opc) {
1976 case OPC_MFHI:
1977 tcg_gen_mov_tl(cpu_gpr[reg], cpu_HI[0]);
1978 opn = "mfhi";
1979 break;
1980 case OPC_MFLO:
1981 tcg_gen_mov_tl(cpu_gpr[reg], cpu_LO[0]);
1982 opn = "mflo";
1983 break;
1984 case OPC_MTHI:
1985 if (reg != 0)
1986 tcg_gen_mov_tl(cpu_HI[0], cpu_gpr[reg]);
1987 else
1988 tcg_gen_movi_tl(cpu_HI[0], 0);
1989 opn = "mthi";
1990 break;
1991 case OPC_MTLO:
1992 if (reg != 0)
1993 tcg_gen_mov_tl(cpu_LO[0], cpu_gpr[reg]);
1994 else
1995 tcg_gen_movi_tl(cpu_LO[0], 0);
1996 opn = "mtlo";
1997 break;
1998 }
1999 MIPS_DEBUG("%s %s", opn, regnames[reg]);
2000 }
2001
gen_muldiv(DisasContext * ctx,uint32_t opc,int rs,int rt)2002 static void gen_muldiv (DisasContext *ctx, uint32_t opc,
2003 int rs, int rt)
2004 {
2005 const char *opn = "mul/div";
2006 TCGv t0, t1;
2007
2008 switch (opc) {
2009 case OPC_DIV:
2010 case OPC_DIVU:
2011 #if defined(TARGET_MIPS64)
2012 case OPC_DDIV:
2013 case OPC_DDIVU:
2014 #endif
2015 t0 = tcg_temp_local_new();
2016 t1 = tcg_temp_local_new();
2017 break;
2018 default:
2019 t0 = tcg_temp_new();
2020 t1 = tcg_temp_new();
2021 break;
2022 }
2023
2024 gen_load_gpr(t0, rs);
2025 gen_load_gpr(t1, rt);
2026 switch (opc) {
2027 case OPC_DIV:
2028 {
2029 int l1 = gen_new_label();
2030 int l2 = gen_new_label();
2031
2032 tcg_gen_ext32s_tl(t0, t0);
2033 tcg_gen_ext32s_tl(t1, t1);
2034 tcg_gen_brcondi_tl(TCG_COND_EQ, t1, 0, l1);
2035 tcg_gen_brcondi_tl(TCG_COND_NE, t0, INT_MIN, l2);
2036 tcg_gen_brcondi_tl(TCG_COND_NE, t1, -1, l2);
2037
2038 tcg_gen_mov_tl(cpu_LO[0], t0);
2039 tcg_gen_movi_tl(cpu_HI[0], 0);
2040 tcg_gen_br(l1);
2041 gen_set_label(l2);
2042 tcg_gen_div_tl(cpu_LO[0], t0, t1);
2043 tcg_gen_rem_tl(cpu_HI[0], t0, t1);
2044 tcg_gen_ext32s_tl(cpu_LO[0], cpu_LO[0]);
2045 tcg_gen_ext32s_tl(cpu_HI[0], cpu_HI[0]);
2046 gen_set_label(l1);
2047 }
2048 opn = "div";
2049 break;
2050 case OPC_DIVU:
2051 {
2052 int l1 = gen_new_label();
2053
2054 tcg_gen_ext32u_tl(t0, t0);
2055 tcg_gen_ext32u_tl(t1, t1);
2056 tcg_gen_brcondi_tl(TCG_COND_EQ, t1, 0, l1);
2057 tcg_gen_divu_tl(cpu_LO[0], t0, t1);
2058 tcg_gen_remu_tl(cpu_HI[0], t0, t1);
2059 tcg_gen_ext32s_tl(cpu_LO[0], cpu_LO[0]);
2060 tcg_gen_ext32s_tl(cpu_HI[0], cpu_HI[0]);
2061 gen_set_label(l1);
2062 }
2063 opn = "divu";
2064 break;
2065 case OPC_MULT:
2066 {
2067 TCGv_i64 t2 = tcg_temp_new_i64();
2068 TCGv_i64 t3 = tcg_temp_new_i64();
2069
2070 tcg_gen_ext_tl_i64(t2, t0);
2071 tcg_gen_ext_tl_i64(t3, t1);
2072 tcg_gen_mul_i64(t2, t2, t3);
2073 tcg_temp_free_i64(t3);
2074 tcg_gen_trunc_i64_tl(t0, t2);
2075 tcg_gen_shri_i64(t2, t2, 32);
2076 tcg_gen_trunc_i64_tl(t1, t2);
2077 tcg_temp_free_i64(t2);
2078 tcg_gen_ext32s_tl(cpu_LO[0], t0);
2079 tcg_gen_ext32s_tl(cpu_HI[0], t1);
2080 }
2081 opn = "mult";
2082 break;
2083 case OPC_MULTU:
2084 {
2085 TCGv_i64 t2 = tcg_temp_new_i64();
2086 TCGv_i64 t3 = tcg_temp_new_i64();
2087
2088 tcg_gen_ext32u_tl(t0, t0);
2089 tcg_gen_ext32u_tl(t1, t1);
2090 tcg_gen_extu_tl_i64(t2, t0);
2091 tcg_gen_extu_tl_i64(t3, t1);
2092 tcg_gen_mul_i64(t2, t2, t3);
2093 tcg_temp_free_i64(t3);
2094 tcg_gen_trunc_i64_tl(t0, t2);
2095 tcg_gen_shri_i64(t2, t2, 32);
2096 tcg_gen_trunc_i64_tl(t1, t2);
2097 tcg_temp_free_i64(t2);
2098 tcg_gen_ext32s_tl(cpu_LO[0], t0);
2099 tcg_gen_ext32s_tl(cpu_HI[0], t1);
2100 }
2101 opn = "multu";
2102 break;
2103 #if defined(TARGET_MIPS64)
2104 case OPC_DDIV:
2105 {
2106 int l1 = gen_new_label();
2107 int l2 = gen_new_label();
2108
2109 tcg_gen_brcondi_tl(TCG_COND_EQ, t1, 0, l1);
2110 tcg_gen_brcondi_tl(TCG_COND_NE, t0, -1LL << 63, l2);
2111 tcg_gen_brcondi_tl(TCG_COND_NE, t1, -1LL, l2);
2112 tcg_gen_mov_tl(cpu_LO[0], t0);
2113 tcg_gen_movi_tl(cpu_HI[0], 0);
2114 tcg_gen_br(l1);
2115 gen_set_label(l2);
2116 tcg_gen_div_i64(cpu_LO[0], t0, t1);
2117 tcg_gen_rem_i64(cpu_HI[0], t0, t1);
2118 gen_set_label(l1);
2119 }
2120 opn = "ddiv";
2121 break;
2122 case OPC_DDIVU:
2123 {
2124 int l1 = gen_new_label();
2125
2126 tcg_gen_brcondi_tl(TCG_COND_EQ, t1, 0, l1);
2127 tcg_gen_divu_i64(cpu_LO[0], t0, t1);
2128 tcg_gen_remu_i64(cpu_HI[0], t0, t1);
2129 gen_set_label(l1);
2130 }
2131 opn = "ddivu";
2132 break;
2133 case OPC_DMULT:
2134 gen_helper_dmult(t0, t1);
2135 opn = "dmult";
2136 break;
2137 case OPC_DMULTU:
2138 gen_helper_dmultu(t0, t1);
2139 opn = "dmultu";
2140 break;
2141 #endif
2142 case OPC_MADD:
2143 {
2144 TCGv_i64 t2 = tcg_temp_new_i64();
2145 TCGv_i64 t3 = tcg_temp_new_i64();
2146
2147 tcg_gen_ext_tl_i64(t2, t0);
2148 tcg_gen_ext_tl_i64(t3, t1);
2149 tcg_gen_mul_i64(t2, t2, t3);
2150 tcg_gen_concat_tl_i64(t3, cpu_LO[0], cpu_HI[0]);
2151 tcg_gen_add_i64(t2, t2, t3);
2152 tcg_temp_free_i64(t3);
2153 tcg_gen_trunc_i64_tl(t0, t2);
2154 tcg_gen_shri_i64(t2, t2, 32);
2155 tcg_gen_trunc_i64_tl(t1, t2);
2156 tcg_temp_free_i64(t2);
2157 tcg_gen_ext32s_tl(cpu_LO[0], t0);
2158 tcg_gen_ext32s_tl(cpu_HI[0], t1);
2159 }
2160 opn = "madd";
2161 break;
2162 case OPC_MADDU:
2163 {
2164 TCGv_i64 t2 = tcg_temp_new_i64();
2165 TCGv_i64 t3 = tcg_temp_new_i64();
2166
2167 tcg_gen_ext32u_tl(t0, t0);
2168 tcg_gen_ext32u_tl(t1, t1);
2169 tcg_gen_extu_tl_i64(t2, t0);
2170 tcg_gen_extu_tl_i64(t3, t1);
2171 tcg_gen_mul_i64(t2, t2, t3);
2172 tcg_gen_concat_tl_i64(t3, cpu_LO[0], cpu_HI[0]);
2173 tcg_gen_add_i64(t2, t2, t3);
2174 tcg_temp_free_i64(t3);
2175 tcg_gen_trunc_i64_tl(t0, t2);
2176 tcg_gen_shri_i64(t2, t2, 32);
2177 tcg_gen_trunc_i64_tl(t1, t2);
2178 tcg_temp_free_i64(t2);
2179 tcg_gen_ext32s_tl(cpu_LO[0], t0);
2180 tcg_gen_ext32s_tl(cpu_HI[0], t1);
2181 }
2182 opn = "maddu";
2183 break;
2184 case OPC_MSUB:
2185 {
2186 TCGv_i64 t2 = tcg_temp_new_i64();
2187 TCGv_i64 t3 = tcg_temp_new_i64();
2188
2189 tcg_gen_ext_tl_i64(t2, t0);
2190 tcg_gen_ext_tl_i64(t3, t1);
2191 tcg_gen_mul_i64(t2, t2, t3);
2192 tcg_gen_concat_tl_i64(t3, cpu_LO[0], cpu_HI[0]);
2193 tcg_gen_sub_i64(t2, t3, t2);
2194 tcg_temp_free_i64(t3);
2195 tcg_gen_trunc_i64_tl(t0, t2);
2196 tcg_gen_shri_i64(t2, t2, 32);
2197 tcg_gen_trunc_i64_tl(t1, t2);
2198 tcg_temp_free_i64(t2);
2199 tcg_gen_ext32s_tl(cpu_LO[0], t0);
2200 tcg_gen_ext32s_tl(cpu_HI[0], t1);
2201 }
2202 opn = "msub";
2203 break;
2204 case OPC_MSUBU:
2205 {
2206 TCGv_i64 t2 = tcg_temp_new_i64();
2207 TCGv_i64 t3 = tcg_temp_new_i64();
2208
2209 tcg_gen_ext32u_tl(t0, t0);
2210 tcg_gen_ext32u_tl(t1, t1);
2211 tcg_gen_extu_tl_i64(t2, t0);
2212 tcg_gen_extu_tl_i64(t3, t1);
2213 tcg_gen_mul_i64(t2, t2, t3);
2214 tcg_gen_concat_tl_i64(t3, cpu_LO[0], cpu_HI[0]);
2215 tcg_gen_sub_i64(t2, t3, t2);
2216 tcg_temp_free_i64(t3);
2217 tcg_gen_trunc_i64_tl(t0, t2);
2218 tcg_gen_shri_i64(t2, t2, 32);
2219 tcg_gen_trunc_i64_tl(t1, t2);
2220 tcg_temp_free_i64(t2);
2221 tcg_gen_ext32s_tl(cpu_LO[0], t0);
2222 tcg_gen_ext32s_tl(cpu_HI[0], t1);
2223 }
2224 opn = "msubu";
2225 break;
2226 default:
2227 MIPS_INVAL(opn);
2228 generate_exception(ctx, EXCP_RI);
2229 goto out;
2230 }
2231 MIPS_DEBUG("%s %s %s", opn, regnames[rs], regnames[rt]);
2232 out:
2233 tcg_temp_free(t0);
2234 tcg_temp_free(t1);
2235 }
2236
gen_mul_vr54xx(DisasContext * ctx,uint32_t opc,int rd,int rs,int rt)2237 static void gen_mul_vr54xx (DisasContext *ctx, uint32_t opc,
2238 int rd, int rs, int rt)
2239 {
2240 const char *opn = "mul vr54xx";
2241 TCGv t0 = tcg_temp_new();
2242 TCGv t1 = tcg_temp_new();
2243
2244 gen_load_gpr(t0, rs);
2245 gen_load_gpr(t1, rt);
2246
2247 switch (opc) {
2248 case OPC_VR54XX_MULS:
2249 gen_helper_muls(t0, t0, t1);
2250 opn = "muls";
2251 break;
2252 case OPC_VR54XX_MULSU:
2253 gen_helper_mulsu(t0, t0, t1);
2254 opn = "mulsu";
2255 break;
2256 case OPC_VR54XX_MACC:
2257 gen_helper_macc(t0, t0, t1);
2258 opn = "macc";
2259 break;
2260 case OPC_VR54XX_MACCU:
2261 gen_helper_maccu(t0, t0, t1);
2262 opn = "maccu";
2263 break;
2264 case OPC_VR54XX_MSAC:
2265 gen_helper_msac(t0, t0, t1);
2266 opn = "msac";
2267 break;
2268 case OPC_VR54XX_MSACU:
2269 gen_helper_msacu(t0, t0, t1);
2270 opn = "msacu";
2271 break;
2272 case OPC_VR54XX_MULHI:
2273 gen_helper_mulhi(t0, t0, t1);
2274 opn = "mulhi";
2275 break;
2276 case OPC_VR54XX_MULHIU:
2277 gen_helper_mulhiu(t0, t0, t1);
2278 opn = "mulhiu";
2279 break;
2280 case OPC_VR54XX_MULSHI:
2281 gen_helper_mulshi(t0, t0, t1);
2282 opn = "mulshi";
2283 break;
2284 case OPC_VR54XX_MULSHIU:
2285 gen_helper_mulshiu(t0, t0, t1);
2286 opn = "mulshiu";
2287 break;
2288 case OPC_VR54XX_MACCHI:
2289 gen_helper_macchi(t0, t0, t1);
2290 opn = "macchi";
2291 break;
2292 case OPC_VR54XX_MACCHIU:
2293 gen_helper_macchiu(t0, t0, t1);
2294 opn = "macchiu";
2295 break;
2296 case OPC_VR54XX_MSACHI:
2297 gen_helper_msachi(t0, t0, t1);
2298 opn = "msachi";
2299 break;
2300 case OPC_VR54XX_MSACHIU:
2301 gen_helper_msachiu(t0, t0, t1);
2302 opn = "msachiu";
2303 break;
2304 default:
2305 MIPS_INVAL("mul vr54xx");
2306 generate_exception(ctx, EXCP_RI);
2307 goto out;
2308 }
2309 gen_store_gpr(t0, rd);
2310 MIPS_DEBUG("%s %s, %s, %s", opn, regnames[rd], regnames[rs], regnames[rt]);
2311
2312 out:
2313 tcg_temp_free(t0);
2314 tcg_temp_free(t1);
2315 }
2316
gen_cl(DisasContext * ctx,uint32_t opc,int rd,int rs)2317 static void gen_cl (DisasContext *ctx, uint32_t opc,
2318 int rd, int rs)
2319 {
2320 const char *opn = "CLx";
2321 TCGv t0;
2322
2323 if (rd == 0) {
2324 /* Treat as NOP. */
2325 MIPS_DEBUG("NOP");
2326 return;
2327 }
2328 t0 = tcg_temp_new();
2329 gen_load_gpr(t0, rs);
2330 switch (opc) {
2331 case OPC_CLO:
2332 gen_helper_clo(cpu_gpr[rd], t0);
2333 opn = "clo";
2334 break;
2335 case OPC_CLZ:
2336 gen_helper_clz(cpu_gpr[rd], t0);
2337 opn = "clz";
2338 break;
2339 #if defined(TARGET_MIPS64)
2340 case OPC_DCLO:
2341 gen_helper_dclo(cpu_gpr[rd], t0);
2342 opn = "dclo";
2343 break;
2344 case OPC_DCLZ:
2345 gen_helper_dclz(cpu_gpr[rd], t0);
2346 opn = "dclz";
2347 break;
2348 #endif
2349 }
2350 MIPS_DEBUG("%s %s, %s", opn, regnames[rd], regnames[rs]);
2351 tcg_temp_free(t0);
2352 }
2353
2354 /* Traps */
gen_trap(DisasContext * ctx,uint32_t opc,int rs,int rt,int16_t imm)2355 static void gen_trap (DisasContext *ctx, uint32_t opc,
2356 int rs, int rt, int16_t imm)
2357 {
2358 int cond;
2359 TCGv t0 = tcg_temp_new();
2360 TCGv t1 = tcg_temp_new();
2361
2362 cond = 0;
2363 /* Load needed operands */
2364 switch (opc) {
2365 case OPC_TEQ:
2366 case OPC_TGE:
2367 case OPC_TGEU:
2368 case OPC_TLT:
2369 case OPC_TLTU:
2370 case OPC_TNE:
2371 /* Compare two registers */
2372 if (rs != rt) {
2373 gen_load_gpr(t0, rs);
2374 gen_load_gpr(t1, rt);
2375 cond = 1;
2376 }
2377 break;
2378 case OPC_TEQI:
2379 case OPC_TGEI:
2380 case OPC_TGEIU:
2381 case OPC_TLTI:
2382 case OPC_TLTIU:
2383 case OPC_TNEI:
2384 /* Compare register to immediate */
2385 if (rs != 0 || imm != 0) {
2386 gen_load_gpr(t0, rs);
2387 tcg_gen_movi_tl(t1, (int32_t)imm);
2388 cond = 1;
2389 }
2390 break;
2391 }
2392 if (cond == 0) {
2393 switch (opc) {
2394 case OPC_TEQ: /* rs == rs */
2395 case OPC_TEQI: /* r0 == 0 */
2396 case OPC_TGE: /* rs >= rs */
2397 case OPC_TGEI: /* r0 >= 0 */
2398 case OPC_TGEU: /* rs >= rs unsigned */
2399 case OPC_TGEIU: /* r0 >= 0 unsigned */
2400 /* Always trap */
2401 generate_exception(ctx, EXCP_TRAP);
2402 break;
2403 case OPC_TLT: /* rs < rs */
2404 case OPC_TLTI: /* r0 < 0 */
2405 case OPC_TLTU: /* rs < rs unsigned */
2406 case OPC_TLTIU: /* r0 < 0 unsigned */
2407 case OPC_TNE: /* rs != rs */
2408 case OPC_TNEI: /* r0 != 0 */
2409 /* Never trap: treat as NOP. */
2410 break;
2411 }
2412 } else {
2413 int l1 = gen_new_label();
2414
2415 switch (opc) {
2416 case OPC_TEQ:
2417 case OPC_TEQI:
2418 tcg_gen_brcond_tl(TCG_COND_NE, t0, t1, l1);
2419 break;
2420 case OPC_TGE:
2421 case OPC_TGEI:
2422 tcg_gen_brcond_tl(TCG_COND_LT, t0, t1, l1);
2423 break;
2424 case OPC_TGEU:
2425 case OPC_TGEIU:
2426 tcg_gen_brcond_tl(TCG_COND_LTU, t0, t1, l1);
2427 break;
2428 case OPC_TLT:
2429 case OPC_TLTI:
2430 tcg_gen_brcond_tl(TCG_COND_GE, t0, t1, l1);
2431 break;
2432 case OPC_TLTU:
2433 case OPC_TLTIU:
2434 tcg_gen_brcond_tl(TCG_COND_GEU, t0, t1, l1);
2435 break;
2436 case OPC_TNE:
2437 case OPC_TNEI:
2438 tcg_gen_brcond_tl(TCG_COND_EQ, t0, t1, l1);
2439 break;
2440 }
2441 generate_exception(ctx, EXCP_TRAP);
2442 gen_set_label(l1);
2443 }
2444 tcg_temp_free(t0);
2445 tcg_temp_free(t1);
2446 }
2447
gen_goto_tb(DisasContext * ctx,int n,target_ulong dest)2448 static inline void gen_goto_tb(DisasContext *ctx, int n, target_ulong dest)
2449 {
2450 TranslationBlock *tb;
2451 tb = ctx->tb;
2452 if ((tb->pc & TARGET_PAGE_MASK) == (dest & TARGET_PAGE_MASK) &&
2453 likely(!ctx->singlestep_enabled)) {
2454 tcg_gen_goto_tb(n);
2455 gen_save_pc(dest);
2456 tcg_gen_exit_tb((long)tb + n);
2457 } else {
2458 gen_save_pc(dest);
2459 if (ctx->singlestep_enabled) {
2460 save_cpu_state(ctx, 0);
2461 gen_helper_0i(raise_exception, EXCP_DEBUG);
2462 }
2463 tcg_gen_exit_tb(0);
2464 }
2465 }
2466
2467 /* Branches (before delay slot) */
gen_compute_branch(DisasContext * ctx,uint32_t opc,int rs,int rt,int32_t offset)2468 static void gen_compute_branch (DisasContext *ctx, uint32_t opc,
2469 int rs, int rt, int32_t offset)
2470 {
2471 target_ulong btgt = -1;
2472 int blink = 0;
2473 int bcond_compute = 0;
2474 TCGv t0 = tcg_temp_new();
2475 TCGv t1 = tcg_temp_new();
2476
2477 if (ctx->hflags & MIPS_HFLAG_BMASK) {
2478 #ifdef MIPS_DEBUG_DISAS
2479 LOG_DISAS("Branch in delay slot at PC 0x" TARGET_FMT_lx "\n", ctx->pc);
2480 #endif
2481 generate_exception(ctx, EXCP_RI);
2482 goto out;
2483 }
2484
2485 /* Load needed operands */
2486 switch (opc) {
2487 case OPC_BEQ:
2488 case OPC_BEQL:
2489 case OPC_BNE:
2490 case OPC_BNEL:
2491 /* Compare two registers */
2492 if (rs != rt) {
2493 gen_load_gpr(t0, rs);
2494 gen_load_gpr(t1, rt);
2495 bcond_compute = 1;
2496 }
2497 btgt = ctx->pc + 4 + offset;
2498 break;
2499 case OPC_BGEZ:
2500 case OPC_BGEZAL:
2501 case OPC_BGEZALL:
2502 case OPC_BGEZL:
2503 case OPC_BGTZ:
2504 case OPC_BGTZL:
2505 case OPC_BLEZ:
2506 case OPC_BLEZL:
2507 case OPC_BLTZ:
2508 case OPC_BLTZAL:
2509 case OPC_BLTZALL:
2510 case OPC_BLTZL:
2511 /* Compare to zero */
2512 if (rs != 0) {
2513 gen_load_gpr(t0, rs);
2514 bcond_compute = 1;
2515 }
2516 btgt = ctx->pc + 4 + offset;
2517 break;
2518 case OPC_J:
2519 case OPC_JAL:
2520 /* Jump to immediate */
2521 btgt = ((ctx->pc + 4) & (int32_t)0xF0000000) | (uint32_t)offset;
2522 break;
2523 case OPC_JR:
2524 case OPC_JALR:
2525 /* Jump to register */
2526 if (offset != 0 && offset != 16) {
2527 /* Hint = 0 is JR/JALR, hint 16 is JR.HB/JALR.HB, the
2528 others are reserved. */
2529 MIPS_INVAL("jump hint");
2530 generate_exception(ctx, EXCP_RI);
2531 goto out;
2532 }
2533 gen_load_gpr(btarget, rs);
2534 break;
2535 default:
2536 MIPS_INVAL("branch/jump");
2537 generate_exception(ctx, EXCP_RI);
2538 goto out;
2539 }
2540 if (bcond_compute == 0) {
2541 /* No condition to be computed */
2542 switch (opc) {
2543 case OPC_BEQ: /* rx == rx */
2544 case OPC_BEQL: /* rx == rx likely */
2545 case OPC_BGEZ: /* 0 >= 0 */
2546 case OPC_BGEZL: /* 0 >= 0 likely */
2547 case OPC_BLEZ: /* 0 <= 0 */
2548 case OPC_BLEZL: /* 0 <= 0 likely */
2549 /* Always take */
2550 ctx->hflags |= MIPS_HFLAG_B;
2551 MIPS_DEBUG("balways");
2552 break;
2553 case OPC_BGEZAL: /* 0 >= 0 */
2554 case OPC_BGEZALL: /* 0 >= 0 likely */
2555 /* Always take and link */
2556 blink = 31;
2557 ctx->hflags |= MIPS_HFLAG_B;
2558 MIPS_DEBUG("balways and link");
2559 break;
2560 case OPC_BNE: /* rx != rx */
2561 case OPC_BGTZ: /* 0 > 0 */
2562 case OPC_BLTZ: /* 0 < 0 */
2563 /* Treat as NOP. */
2564 MIPS_DEBUG("bnever (NOP)");
2565 goto out;
2566 case OPC_BLTZAL: /* 0 < 0 */
2567 tcg_gen_movi_tl(cpu_gpr[31], ctx->pc + 8);
2568 MIPS_DEBUG("bnever and link");
2569 goto out;
2570 case OPC_BLTZALL: /* 0 < 0 likely */
2571 tcg_gen_movi_tl(cpu_gpr[31], ctx->pc + 8);
2572 /* Skip the instruction in the delay slot */
2573 MIPS_DEBUG("bnever, link and skip");
2574 ctx->pc += 4;
2575 goto out;
2576 case OPC_BNEL: /* rx != rx likely */
2577 case OPC_BGTZL: /* 0 > 0 likely */
2578 case OPC_BLTZL: /* 0 < 0 likely */
2579 /* Skip the instruction in the delay slot */
2580 MIPS_DEBUG("bnever and skip");
2581 ctx->pc += 4;
2582 goto out;
2583 case OPC_J:
2584 ctx->hflags |= MIPS_HFLAG_B;
2585 MIPS_DEBUG("j " TARGET_FMT_lx, btgt);
2586 break;
2587 case OPC_JAL:
2588 blink = 31;
2589 ctx->hflags |= MIPS_HFLAG_B;
2590 MIPS_DEBUG("jal " TARGET_FMT_lx, btgt);
2591 break;
2592 case OPC_JR:
2593 ctx->hflags |= MIPS_HFLAG_BR;
2594 MIPS_DEBUG("jr %s", regnames[rs]);
2595 break;
2596 case OPC_JALR:
2597 blink = rt;
2598 ctx->hflags |= MIPS_HFLAG_BR;
2599 MIPS_DEBUG("jalr %s, %s", regnames[rt], regnames[rs]);
2600 break;
2601 default:
2602 MIPS_INVAL("branch/jump");
2603 generate_exception(ctx, EXCP_RI);
2604 goto out;
2605 }
2606 } else {
2607 switch (opc) {
2608 case OPC_BEQ:
2609 gen_op_eq(bcond, t0, t1);
2610 MIPS_DEBUG("beq %s, %s, " TARGET_FMT_lx,
2611 regnames[rs], regnames[rt], btgt);
2612 goto not_likely;
2613 case OPC_BEQL:
2614 gen_op_eq(bcond, t0, t1);
2615 MIPS_DEBUG("beql %s, %s, " TARGET_FMT_lx,
2616 regnames[rs], regnames[rt], btgt);
2617 goto likely;
2618 case OPC_BNE:
2619 gen_op_ne(bcond, t0, t1);
2620 MIPS_DEBUG("bne %s, %s, " TARGET_FMT_lx,
2621 regnames[rs], regnames[rt], btgt);
2622 goto not_likely;
2623 case OPC_BNEL:
2624 gen_op_ne(bcond, t0, t1);
2625 MIPS_DEBUG("bnel %s, %s, " TARGET_FMT_lx,
2626 regnames[rs], regnames[rt], btgt);
2627 goto likely;
2628 case OPC_BGEZ:
2629 gen_op_gez(bcond, t0);
2630 MIPS_DEBUG("bgez %s, " TARGET_FMT_lx, regnames[rs], btgt);
2631 goto not_likely;
2632 case OPC_BGEZL:
2633 gen_op_gez(bcond, t0);
2634 MIPS_DEBUG("bgezl %s, " TARGET_FMT_lx, regnames[rs], btgt);
2635 goto likely;
2636 case OPC_BGEZAL:
2637 gen_op_gez(bcond, t0);
2638 MIPS_DEBUG("bgezal %s, " TARGET_FMT_lx, regnames[rs], btgt);
2639 blink = 31;
2640 goto not_likely;
2641 case OPC_BGEZALL:
2642 gen_op_gez(bcond, t0);
2643 blink = 31;
2644 MIPS_DEBUG("bgezall %s, " TARGET_FMT_lx, regnames[rs], btgt);
2645 goto likely;
2646 case OPC_BGTZ:
2647 gen_op_gtz(bcond, t0);
2648 MIPS_DEBUG("bgtz %s, " TARGET_FMT_lx, regnames[rs], btgt);
2649 goto not_likely;
2650 case OPC_BGTZL:
2651 gen_op_gtz(bcond, t0);
2652 MIPS_DEBUG("bgtzl %s, " TARGET_FMT_lx, regnames[rs], btgt);
2653 goto likely;
2654 case OPC_BLEZ:
2655 gen_op_lez(bcond, t0);
2656 MIPS_DEBUG("blez %s, " TARGET_FMT_lx, regnames[rs], btgt);
2657 goto not_likely;
2658 case OPC_BLEZL:
2659 gen_op_lez(bcond, t0);
2660 MIPS_DEBUG("blezl %s, " TARGET_FMT_lx, regnames[rs], btgt);
2661 goto likely;
2662 case OPC_BLTZ:
2663 gen_op_ltz(bcond, t0);
2664 MIPS_DEBUG("bltz %s, " TARGET_FMT_lx, regnames[rs], btgt);
2665 goto not_likely;
2666 case OPC_BLTZL:
2667 gen_op_ltz(bcond, t0);
2668 MIPS_DEBUG("bltzl %s, " TARGET_FMT_lx, regnames[rs], btgt);
2669 goto likely;
2670 case OPC_BLTZAL:
2671 gen_op_ltz(bcond, t0);
2672 blink = 31;
2673 MIPS_DEBUG("bltzal %s, " TARGET_FMT_lx, regnames[rs], btgt);
2674 not_likely:
2675 ctx->hflags |= MIPS_HFLAG_BC;
2676 break;
2677 case OPC_BLTZALL:
2678 gen_op_ltz(bcond, t0);
2679 blink = 31;
2680 MIPS_DEBUG("bltzall %s, " TARGET_FMT_lx, regnames[rs], btgt);
2681 likely:
2682 ctx->hflags |= MIPS_HFLAG_BL;
2683 break;
2684 default:
2685 MIPS_INVAL("conditional branch/jump");
2686 generate_exception(ctx, EXCP_RI);
2687 goto out;
2688 }
2689 }
2690 MIPS_DEBUG("enter ds: link %d cond %02x target " TARGET_FMT_lx,
2691 blink, ctx->hflags, btgt);
2692
2693 ctx->btarget = btgt;
2694 if (blink > 0) {
2695 tcg_gen_movi_tl(cpu_gpr[blink], ctx->pc + 8);
2696 }
2697
2698 out:
2699 tcg_temp_free(t0);
2700 tcg_temp_free(t1);
2701 }
2702
2703 /* special3 bitfield operations */
gen_bitops(DisasContext * ctx,uint32_t opc,int rt,int rs,int lsb,int msb)2704 static void gen_bitops (DisasContext *ctx, uint32_t opc, int rt,
2705 int rs, int lsb, int msb)
2706 {
2707 TCGv t0 = tcg_temp_new();
2708 TCGv t1 = tcg_temp_new();
2709 target_ulong mask;
2710
2711 gen_load_gpr(t1, rs);
2712 switch (opc) {
2713 case OPC_EXT:
2714 if (lsb + msb > 31)
2715 goto fail;
2716 tcg_gen_shri_tl(t0, t1, lsb);
2717 if (msb != 31) {
2718 tcg_gen_andi_tl(t0, t0, (1 << (msb + 1)) - 1);
2719 } else {
2720 tcg_gen_ext32s_tl(t0, t0);
2721 }
2722 break;
2723 #if defined(TARGET_MIPS64)
2724 case OPC_DEXTM:
2725 tcg_gen_shri_tl(t0, t1, lsb);
2726 if (msb != 31) {
2727 tcg_gen_andi_tl(t0, t0, (1ULL << (msb + 1 + 32)) - 1);
2728 }
2729 break;
2730 case OPC_DEXTU:
2731 tcg_gen_shri_tl(t0, t1, lsb + 32);
2732 tcg_gen_andi_tl(t0, t0, (1ULL << (msb + 1)) - 1);
2733 break;
2734 case OPC_DEXT:
2735 tcg_gen_shri_tl(t0, t1, lsb);
2736 tcg_gen_andi_tl(t0, t0, (1ULL << (msb + 1)) - 1);
2737 break;
2738 #endif
2739 case OPC_INS:
2740 if (lsb > msb)
2741 goto fail;
2742 mask = ((msb - lsb + 1 < 32) ? ((1 << (msb - lsb + 1)) - 1) : ~0) << lsb;
2743 gen_load_gpr(t0, rt);
2744 tcg_gen_andi_tl(t0, t0, ~mask);
2745 tcg_gen_shli_tl(t1, t1, lsb);
2746 tcg_gen_andi_tl(t1, t1, mask);
2747 tcg_gen_or_tl(t0, t0, t1);
2748 tcg_gen_ext32s_tl(t0, t0);
2749 break;
2750 #if defined(TARGET_MIPS64)
2751 case OPC_DINSM:
2752 if (lsb > msb)
2753 goto fail;
2754 mask = ((msb - lsb + 1 + 32 < 64) ? ((1ULL << (msb - lsb + 1 + 32)) - 1) : ~0ULL) << lsb;
2755 gen_load_gpr(t0, rt);
2756 tcg_gen_andi_tl(t0, t0, ~mask);
2757 tcg_gen_shli_tl(t1, t1, lsb);
2758 tcg_gen_andi_tl(t1, t1, mask);
2759 tcg_gen_or_tl(t0, t0, t1);
2760 break;
2761 case OPC_DINSU:
2762 if (lsb > msb)
2763 goto fail;
2764 mask = ((1ULL << (msb - lsb + 1)) - 1) << (lsb + 32);
2765 gen_load_gpr(t0, rt);
2766 tcg_gen_andi_tl(t0, t0, ~mask);
2767 tcg_gen_shli_tl(t1, t1, lsb + 32);
2768 tcg_gen_andi_tl(t1, t1, mask);
2769 tcg_gen_or_tl(t0, t0, t1);
2770 break;
2771 case OPC_DINS:
2772 if (lsb > msb)
2773 goto fail;
2774 gen_load_gpr(t0, rt);
2775 mask = ((1ULL << (msb - lsb + 1)) - 1) << lsb;
2776 gen_load_gpr(t0, rt);
2777 tcg_gen_andi_tl(t0, t0, ~mask);
2778 tcg_gen_shli_tl(t1, t1, lsb);
2779 tcg_gen_andi_tl(t1, t1, mask);
2780 tcg_gen_or_tl(t0, t0, t1);
2781 break;
2782 #endif
2783 default:
2784 fail:
2785 MIPS_INVAL("bitops");
2786 generate_exception(ctx, EXCP_RI);
2787 tcg_temp_free(t0);
2788 tcg_temp_free(t1);
2789 return;
2790 }
2791 gen_store_gpr(t0, rt);
2792 tcg_temp_free(t0);
2793 tcg_temp_free(t1);
2794 }
2795
gen_bshfl(DisasContext * ctx,uint32_t op2,int rt,int rd)2796 static void gen_bshfl (DisasContext *ctx, uint32_t op2, int rt, int rd)
2797 {
2798 TCGv t0;
2799
2800 if (rd == 0) {
2801 /* If no destination, treat it as a NOP. */
2802 MIPS_DEBUG("NOP");
2803 return;
2804 }
2805
2806 t0 = tcg_temp_new();
2807 gen_load_gpr(t0, rt);
2808 switch (op2) {
2809 case OPC_WSBH:
2810 {
2811 TCGv t1 = tcg_temp_new();
2812
2813 tcg_gen_shri_tl(t1, t0, 8);
2814 tcg_gen_andi_tl(t1, t1, 0x00FF00FF);
2815 tcg_gen_shli_tl(t0, t0, 8);
2816 tcg_gen_andi_tl(t0, t0, ~0x00FF00FF);
2817 tcg_gen_or_tl(t0, t0, t1);
2818 tcg_temp_free(t1);
2819 tcg_gen_ext32s_tl(cpu_gpr[rd], t0);
2820 }
2821 break;
2822 case OPC_SEB:
2823 tcg_gen_ext8s_tl(cpu_gpr[rd], t0);
2824 break;
2825 case OPC_SEH:
2826 tcg_gen_ext16s_tl(cpu_gpr[rd], t0);
2827 break;
2828 #if defined(TARGET_MIPS64)
2829 case OPC_DSBH:
2830 {
2831 TCGv t1 = tcg_temp_new();
2832
2833 tcg_gen_shri_tl(t1, t0, 8);
2834 tcg_gen_andi_tl(t1, t1, 0x00FF00FF00FF00FFULL);
2835 tcg_gen_shli_tl(t0, t0, 8);
2836 tcg_gen_andi_tl(t0, t0, ~0x00FF00FF00FF00FFULL);
2837 tcg_gen_or_tl(cpu_gpr[rd], t0, t1);
2838 tcg_temp_free(t1);
2839 }
2840 break;
2841 case OPC_DSHD:
2842 {
2843 TCGv t1 = tcg_temp_new();
2844
2845 tcg_gen_shri_tl(t1, t0, 16);
2846 tcg_gen_andi_tl(t1, t1, 0x0000FFFF0000FFFFULL);
2847 tcg_gen_shli_tl(t0, t0, 16);
2848 tcg_gen_andi_tl(t0, t0, ~0x0000FFFF0000FFFFULL);
2849 tcg_gen_or_tl(t0, t0, t1);
2850 tcg_gen_shri_tl(t1, t0, 32);
2851 tcg_gen_shli_tl(t0, t0, 32);
2852 tcg_gen_or_tl(cpu_gpr[rd], t0, t1);
2853 tcg_temp_free(t1);
2854 }
2855 break;
2856 #endif
2857 default:
2858 MIPS_INVAL("bsfhl");
2859 generate_exception(ctx, EXCP_RI);
2860 tcg_temp_free(t0);
2861 return;
2862 }
2863 tcg_temp_free(t0);
2864 }
2865
2866 #ifndef CONFIG_USER_ONLY
2867 /* CP0 (MMU and control) */
gen_mfc0_load32(TCGv arg,target_ulong off)2868 static inline void gen_mfc0_load32 (TCGv arg, target_ulong off)
2869 {
2870 TCGv_i32 t0 = tcg_temp_new_i32();
2871
2872 tcg_gen_ld_i32(t0, cpu_env, off);
2873 tcg_gen_ext_i32_tl(arg, t0);
2874 tcg_temp_free_i32(t0);
2875 }
2876
gen_mfc0_load64(TCGv arg,target_ulong off)2877 static inline void gen_mfc0_load64 (TCGv arg, target_ulong off)
2878 {
2879 tcg_gen_ld_tl(arg, cpu_env, off);
2880 tcg_gen_ext32s_tl(arg, arg);
2881 }
2882
gen_mtc0_store32(TCGv arg,target_ulong off)2883 static inline void gen_mtc0_store32 (TCGv arg, target_ulong off)
2884 {
2885 TCGv_i32 t0 = tcg_temp_new_i32();
2886
2887 tcg_gen_trunc_tl_i32(t0, arg);
2888 tcg_gen_st_i32(t0, cpu_env, off);
2889 tcg_temp_free_i32(t0);
2890 }
2891
gen_mtc0_store64(TCGv arg,target_ulong off)2892 static inline void gen_mtc0_store64 (TCGv arg, target_ulong off)
2893 {
2894 tcg_gen_ext32s_tl(arg, arg);
2895 tcg_gen_st_tl(arg, cpu_env, off);
2896 }
2897
gen_mfc0(CPUState * env,DisasContext * ctx,TCGv arg,int reg,int sel)2898 static void gen_mfc0 (CPUState *env, DisasContext *ctx, TCGv arg, int reg, int sel)
2899 {
2900 const char *rn = "invalid";
2901
2902 if (sel != 0)
2903 check_insn(env, ctx, ISA_MIPS32);
2904
2905 switch (reg) {
2906 case 0:
2907 switch (sel) {
2908 case 0:
2909 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Index));
2910 rn = "Index";
2911 break;
2912 case 1:
2913 check_insn(env, ctx, ASE_MT);
2914 gen_helper_mfc0_mvpcontrol(arg);
2915 rn = "MVPControl";
2916 break;
2917 case 2:
2918 check_insn(env, ctx, ASE_MT);
2919 gen_helper_mfc0_mvpconf0(arg);
2920 rn = "MVPConf0";
2921 break;
2922 case 3:
2923 check_insn(env, ctx, ASE_MT);
2924 gen_helper_mfc0_mvpconf1(arg);
2925 rn = "MVPConf1";
2926 break;
2927 default:
2928 goto die;
2929 }
2930 break;
2931 case 1:
2932 switch (sel) {
2933 case 0:
2934 gen_helper_mfc0_random(arg);
2935 rn = "Random";
2936 break;
2937 case 1:
2938 check_insn(env, ctx, ASE_MT);
2939 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEControl));
2940 rn = "VPEControl";
2941 break;
2942 case 2:
2943 check_insn(env, ctx, ASE_MT);
2944 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEConf0));
2945 rn = "VPEConf0";
2946 break;
2947 case 3:
2948 check_insn(env, ctx, ASE_MT);
2949 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEConf1));
2950 rn = "VPEConf1";
2951 break;
2952 case 4:
2953 check_insn(env, ctx, ASE_MT);
2954 gen_mfc0_load64(arg, offsetof(CPUState, CP0_YQMask));
2955 rn = "YQMask";
2956 break;
2957 case 5:
2958 check_insn(env, ctx, ASE_MT);
2959 gen_mfc0_load64(arg, offsetof(CPUState, CP0_VPESchedule));
2960 rn = "VPESchedule";
2961 break;
2962 case 6:
2963 check_insn(env, ctx, ASE_MT);
2964 gen_mfc0_load64(arg, offsetof(CPUState, CP0_VPEScheFBack));
2965 rn = "VPEScheFBack";
2966 break;
2967 case 7:
2968 check_insn(env, ctx, ASE_MT);
2969 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEOpt));
2970 rn = "VPEOpt";
2971 break;
2972 default:
2973 goto die;
2974 }
2975 break;
2976 case 2:
2977 switch (sel) {
2978 case 0:
2979 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EntryLo0));
2980 tcg_gen_ext32s_tl(arg, arg);
2981 rn = "EntryLo0";
2982 break;
2983 case 1:
2984 check_insn(env, ctx, ASE_MT);
2985 gen_helper_mfc0_tcstatus(arg);
2986 rn = "TCStatus";
2987 break;
2988 case 2:
2989 check_insn(env, ctx, ASE_MT);
2990 gen_helper_mfc0_tcbind(arg);
2991 rn = "TCBind";
2992 break;
2993 case 3:
2994 check_insn(env, ctx, ASE_MT);
2995 gen_helper_mfc0_tcrestart(arg);
2996 rn = "TCRestart";
2997 break;
2998 case 4:
2999 check_insn(env, ctx, ASE_MT);
3000 gen_helper_mfc0_tchalt(arg);
3001 rn = "TCHalt";
3002 break;
3003 case 5:
3004 check_insn(env, ctx, ASE_MT);
3005 gen_helper_mfc0_tccontext(arg);
3006 rn = "TCContext";
3007 break;
3008 case 6:
3009 check_insn(env, ctx, ASE_MT);
3010 gen_helper_mfc0_tcschedule(arg);
3011 rn = "TCSchedule";
3012 break;
3013 case 7:
3014 check_insn(env, ctx, ASE_MT);
3015 gen_helper_mfc0_tcschefback(arg);
3016 rn = "TCScheFBack";
3017 break;
3018 default:
3019 goto die;
3020 }
3021 break;
3022 case 3:
3023 switch (sel) {
3024 case 0:
3025 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EntryLo1));
3026 tcg_gen_ext32s_tl(arg, arg);
3027 rn = "EntryLo1";
3028 break;
3029 default:
3030 goto die;
3031 }
3032 break;
3033 case 4:
3034 switch (sel) {
3035 case 0:
3036 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_Context));
3037 tcg_gen_ext32s_tl(arg, arg);
3038 rn = "Context";
3039 break;
3040 case 1:
3041 // gen_helper_mfc0_contextconfig(arg); /* SmartMIPS ASE */
3042 rn = "ContextConfig";
3043 // break;
3044 default:
3045 goto die;
3046 }
3047 break;
3048 case 5:
3049 switch (sel) {
3050 case 0:
3051 gen_mfc0_load32(arg, offsetof(CPUState, CP0_PageMask));
3052 rn = "PageMask";
3053 break;
3054 case 1:
3055 check_insn(env, ctx, ISA_MIPS32R2);
3056 gen_mfc0_load32(arg, offsetof(CPUState, CP0_PageGrain));
3057 rn = "PageGrain";
3058 break;
3059 default:
3060 goto die;
3061 }
3062 break;
3063 case 6:
3064 switch (sel) {
3065 case 0:
3066 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Wired));
3067 rn = "Wired";
3068 break;
3069 case 1:
3070 check_insn(env, ctx, ISA_MIPS32R2);
3071 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf0));
3072 rn = "SRSConf0";
3073 break;
3074 case 2:
3075 check_insn(env, ctx, ISA_MIPS32R2);
3076 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf1));
3077 rn = "SRSConf1";
3078 break;
3079 case 3:
3080 check_insn(env, ctx, ISA_MIPS32R2);
3081 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf2));
3082 rn = "SRSConf2";
3083 break;
3084 case 4:
3085 check_insn(env, ctx, ISA_MIPS32R2);
3086 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf3));
3087 rn = "SRSConf3";
3088 break;
3089 case 5:
3090 check_insn(env, ctx, ISA_MIPS32R2);
3091 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf4));
3092 rn = "SRSConf4";
3093 break;
3094 default:
3095 goto die;
3096 }
3097 break;
3098 case 7:
3099 switch (sel) {
3100 case 0:
3101 check_insn(env, ctx, ISA_MIPS32R2);
3102 gen_mfc0_load32(arg, offsetof(CPUState, CP0_HWREna));
3103 rn = "HWREna";
3104 break;
3105 default:
3106 goto die;
3107 }
3108 break;
3109 case 8:
3110 switch (sel) {
3111 case 0:
3112 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_BadVAddr));
3113 tcg_gen_ext32s_tl(arg, arg);
3114 rn = "BadVAddr";
3115 break;
3116 default:
3117 goto die;
3118 }
3119 break;
3120 case 9:
3121 switch (sel) {
3122 case 0:
3123 /* Mark as an IO operation because we read the time. */
3124 if (use_icount)
3125 gen_io_start();
3126 gen_helper_mfc0_count(arg);
3127 if (use_icount) {
3128 gen_io_end();
3129 ctx->bstate = BS_STOP;
3130 }
3131 rn = "Count";
3132 break;
3133 /* 6,7 are implementation dependent */
3134 default:
3135 goto die;
3136 }
3137 break;
3138 case 10:
3139 switch (sel) {
3140 case 0:
3141 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EntryHi));
3142 tcg_gen_ext32s_tl(arg, arg);
3143 rn = "EntryHi";
3144 break;
3145 default:
3146 goto die;
3147 }
3148 break;
3149 case 11:
3150 switch (sel) {
3151 case 0:
3152 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Compare));
3153 rn = "Compare";
3154 break;
3155 /* 6,7 are implementation dependent */
3156 default:
3157 goto die;
3158 }
3159 break;
3160 case 12:
3161 switch (sel) {
3162 case 0:
3163 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Status));
3164 rn = "Status";
3165 break;
3166 case 1:
3167 check_insn(env, ctx, ISA_MIPS32R2);
3168 gen_mfc0_load32(arg, offsetof(CPUState, CP0_IntCtl));
3169 rn = "IntCtl";
3170 break;
3171 case 2:
3172 check_insn(env, ctx, ISA_MIPS32R2);
3173 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSCtl));
3174 rn = "SRSCtl";
3175 break;
3176 case 3:
3177 check_insn(env, ctx, ISA_MIPS32R2);
3178 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSMap));
3179 rn = "SRSMap";
3180 break;
3181 default:
3182 goto die;
3183 }
3184 break;
3185 case 13:
3186 switch (sel) {
3187 case 0:
3188 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Cause));
3189 rn = "Cause";
3190 break;
3191 default:
3192 goto die;
3193 }
3194 break;
3195 case 14:
3196 switch (sel) {
3197 case 0:
3198 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EPC));
3199 tcg_gen_ext32s_tl(arg, arg);
3200 rn = "EPC";
3201 break;
3202 default:
3203 goto die;
3204 }
3205 break;
3206 case 15:
3207 switch (sel) {
3208 case 0:
3209 gen_mfc0_load32(arg, offsetof(CPUState, CP0_PRid));
3210 rn = "PRid";
3211 break;
3212 case 1:
3213 check_insn(env, ctx, ISA_MIPS32R2);
3214 gen_mfc0_load32(arg, offsetof(CPUState, CP0_EBase));
3215 rn = "EBase";
3216 break;
3217 default:
3218 goto die;
3219 }
3220 break;
3221 case 16:
3222 switch (sel) {
3223 case 0:
3224 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config0));
3225 rn = "Config";
3226 break;
3227 case 1:
3228 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config1));
3229 rn = "Config1";
3230 break;
3231 case 2:
3232 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config2));
3233 rn = "Config2";
3234 break;
3235 case 3:
3236 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config3));
3237 rn = "Config3";
3238 break;
3239 /* 4,5 are reserved */
3240 /* 6,7 are implementation dependent */
3241 case 6:
3242 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config6));
3243 rn = "Config6";
3244 break;
3245 case 7:
3246 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config7));
3247 rn = "Config7";
3248 break;
3249 default:
3250 goto die;
3251 }
3252 break;
3253 case 17:
3254 switch (sel) {
3255 case 0:
3256 gen_helper_mfc0_lladdr(arg);
3257 rn = "LLAddr";
3258 break;
3259 default:
3260 goto die;
3261 }
3262 break;
3263 case 18:
3264 switch (sel) {
3265 case 0 ... 7:
3266 gen_helper_1i(mfc0_watchlo, arg, sel);
3267 rn = "WatchLo";
3268 break;
3269 default:
3270 goto die;
3271 }
3272 break;
3273 case 19:
3274 switch (sel) {
3275 case 0 ...7:
3276 gen_helper_1i(mfc0_watchhi, arg, sel);
3277 rn = "WatchHi";
3278 break;
3279 default:
3280 goto die;
3281 }
3282 break;
3283 case 20:
3284 switch (sel) {
3285 case 0:
3286 #if defined(TARGET_MIPS64)
3287 check_insn(env, ctx, ISA_MIPS3);
3288 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_XContext));
3289 tcg_gen_ext32s_tl(arg, arg);
3290 rn = "XContext";
3291 break;
3292 #endif
3293 default:
3294 goto die;
3295 }
3296 break;
3297 case 21:
3298 /* Officially reserved, but sel 0 is used for R1x000 framemask */
3299 switch (sel) {
3300 case 0:
3301 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Framemask));
3302 rn = "Framemask";
3303 break;
3304 default:
3305 goto die;
3306 }
3307 break;
3308 case 22:
3309 tcg_gen_movi_tl(arg, 0); /* unimplemented */
3310 rn = "'Diagnostic"; /* implementation dependent */
3311 break;
3312 case 23:
3313 switch (sel) {
3314 case 0:
3315 gen_helper_mfc0_debug(arg); /* EJTAG support */
3316 rn = "Debug";
3317 break;
3318 case 1:
3319 // gen_helper_mfc0_tracecontrol(arg); /* PDtrace support */
3320 rn = "TraceControl";
3321 // break;
3322 case 2:
3323 // gen_helper_mfc0_tracecontrol2(arg); /* PDtrace support */
3324 rn = "TraceControl2";
3325 // break;
3326 case 3:
3327 // gen_helper_mfc0_usertracedata(arg); /* PDtrace support */
3328 rn = "UserTraceData";
3329 // break;
3330 case 4:
3331 // gen_helper_mfc0_tracebpc(arg); /* PDtrace support */
3332 rn = "TraceBPC";
3333 // break;
3334 default:
3335 goto die;
3336 }
3337 break;
3338 case 24:
3339 switch (sel) {
3340 case 0:
3341 /* EJTAG support */
3342 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_DEPC));
3343 tcg_gen_ext32s_tl(arg, arg);
3344 rn = "DEPC";
3345 break;
3346 default:
3347 goto die;
3348 }
3349 break;
3350 case 25:
3351 switch (sel) {
3352 case 0:
3353 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Performance0));
3354 rn = "Performance0";
3355 break;
3356 case 1:
3357 // gen_helper_mfc0_performance1(arg);
3358 rn = "Performance1";
3359 // break;
3360 case 2:
3361 // gen_helper_mfc0_performance2(arg);
3362 rn = "Performance2";
3363 // break;
3364 case 3:
3365 // gen_helper_mfc0_performance3(arg);
3366 rn = "Performance3";
3367 // break;
3368 case 4:
3369 // gen_helper_mfc0_performance4(arg);
3370 rn = "Performance4";
3371 // break;
3372 case 5:
3373 // gen_helper_mfc0_performance5(arg);
3374 rn = "Performance5";
3375 // break;
3376 case 6:
3377 // gen_helper_mfc0_performance6(arg);
3378 rn = "Performance6";
3379 // break;
3380 case 7:
3381 // gen_helper_mfc0_performance7(arg);
3382 rn = "Performance7";
3383 // break;
3384 default:
3385 goto die;
3386 }
3387 break;
3388 case 26:
3389 tcg_gen_movi_tl(arg, 0); /* unimplemented */
3390 rn = "ECC";
3391 break;
3392 case 27:
3393 switch (sel) {
3394 case 0 ... 3:
3395 tcg_gen_movi_tl(arg, 0); /* unimplemented */
3396 rn = "CacheErr";
3397 break;
3398 default:
3399 goto die;
3400 }
3401 break;
3402 case 28:
3403 switch (sel) {
3404 case 0:
3405 case 2:
3406 case 4:
3407 case 6:
3408 gen_mfc0_load32(arg, offsetof(CPUState, CP0_TagLo));
3409 rn = "TagLo";
3410 break;
3411 case 1:
3412 case 3:
3413 case 5:
3414 case 7:
3415 gen_mfc0_load32(arg, offsetof(CPUState, CP0_DataLo));
3416 rn = "DataLo";
3417 break;
3418 default:
3419 goto die;
3420 }
3421 break;
3422 case 29:
3423 switch (sel) {
3424 case 0:
3425 case 2:
3426 case 4:
3427 case 6:
3428 gen_mfc0_load32(arg, offsetof(CPUState, CP0_TagHi));
3429 rn = "TagHi";
3430 break;
3431 case 1:
3432 case 3:
3433 case 5:
3434 case 7:
3435 gen_mfc0_load32(arg, offsetof(CPUState, CP0_DataHi));
3436 rn = "DataHi";
3437 break;
3438 default:
3439 goto die;
3440 }
3441 break;
3442 case 30:
3443 switch (sel) {
3444 case 0:
3445 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_ErrorEPC));
3446 tcg_gen_ext32s_tl(arg, arg);
3447 rn = "ErrorEPC";
3448 break;
3449 default:
3450 goto die;
3451 }
3452 break;
3453 case 31:
3454 switch (sel) {
3455 case 0:
3456 /* EJTAG support */
3457 gen_mfc0_load32(arg, offsetof(CPUState, CP0_DESAVE));
3458 rn = "DESAVE";
3459 break;
3460 default:
3461 goto die;
3462 }
3463 break;
3464 default:
3465 goto die;
3466 }
3467 LOG_DISAS("mfc0 %s (reg %d sel %d)\n", rn, reg, sel);
3468 return;
3469
3470 die:
3471 LOG_DISAS("mfc0 %s (reg %d sel %d)\n", rn, reg, sel);
3472 generate_exception(ctx, EXCP_RI);
3473 }
3474
gen_mtc0(CPUState * env,DisasContext * ctx,TCGv arg,int reg,int sel)3475 static void gen_mtc0 (CPUState *env, DisasContext *ctx, TCGv arg, int reg, int sel)
3476 {
3477 const char *rn = "invalid";
3478
3479 if (sel != 0)
3480 check_insn(env, ctx, ISA_MIPS32);
3481
3482 if (use_icount)
3483 gen_io_start();
3484
3485 switch (reg) {
3486 case 0:
3487 switch (sel) {
3488 case 0:
3489 gen_helper_mtc0_index(arg);
3490 rn = "Index";
3491 break;
3492 case 1:
3493 check_insn(env, ctx, ASE_MT);
3494 gen_helper_mtc0_mvpcontrol(arg);
3495 rn = "MVPControl";
3496 break;
3497 case 2:
3498 check_insn(env, ctx, ASE_MT);
3499 /* ignored */
3500 rn = "MVPConf0";
3501 break;
3502 case 3:
3503 check_insn(env, ctx, ASE_MT);
3504 /* ignored */
3505 rn = "MVPConf1";
3506 break;
3507 default:
3508 goto die;
3509 }
3510 break;
3511 case 1:
3512 switch (sel) {
3513 case 0:
3514 /* ignored */
3515 rn = "Random";
3516 break;
3517 case 1:
3518 check_insn(env, ctx, ASE_MT);
3519 gen_helper_mtc0_vpecontrol(arg);
3520 rn = "VPEControl";
3521 break;
3522 case 2:
3523 check_insn(env, ctx, ASE_MT);
3524 gen_helper_mtc0_vpeconf0(arg);
3525 rn = "VPEConf0";
3526 break;
3527 case 3:
3528 check_insn(env, ctx, ASE_MT);
3529 gen_helper_mtc0_vpeconf1(arg);
3530 rn = "VPEConf1";
3531 break;
3532 case 4:
3533 check_insn(env, ctx, ASE_MT);
3534 gen_helper_mtc0_yqmask(arg);
3535 rn = "YQMask";
3536 break;
3537 case 5:
3538 check_insn(env, ctx, ASE_MT);
3539 gen_mtc0_store64(arg, offsetof(CPUState, CP0_VPESchedule));
3540 rn = "VPESchedule";
3541 break;
3542 case 6:
3543 check_insn(env, ctx, ASE_MT);
3544 gen_mtc0_store64(arg, offsetof(CPUState, CP0_VPEScheFBack));
3545 rn = "VPEScheFBack";
3546 break;
3547 case 7:
3548 check_insn(env, ctx, ASE_MT);
3549 gen_helper_mtc0_vpeopt(arg);
3550 rn = "VPEOpt";
3551 break;
3552 default:
3553 goto die;
3554 }
3555 break;
3556 case 2:
3557 switch (sel) {
3558 case 0:
3559 gen_helper_mtc0_entrylo0(arg);
3560 rn = "EntryLo0";
3561 break;
3562 case 1:
3563 check_insn(env, ctx, ASE_MT);
3564 gen_helper_mtc0_tcstatus(arg);
3565 rn = "TCStatus";
3566 break;
3567 case 2:
3568 check_insn(env, ctx, ASE_MT);
3569 gen_helper_mtc0_tcbind(arg);
3570 rn = "TCBind";
3571 break;
3572 case 3:
3573 check_insn(env, ctx, ASE_MT);
3574 gen_helper_mtc0_tcrestart(arg);
3575 rn = "TCRestart";
3576 break;
3577 case 4:
3578 check_insn(env, ctx, ASE_MT);
3579 gen_helper_mtc0_tchalt(arg);
3580 rn = "TCHalt";
3581 break;
3582 case 5:
3583 check_insn(env, ctx, ASE_MT);
3584 gen_helper_mtc0_tccontext(arg);
3585 rn = "TCContext";
3586 break;
3587 case 6:
3588 check_insn(env, ctx, ASE_MT);
3589 gen_helper_mtc0_tcschedule(arg);
3590 rn = "TCSchedule";
3591 break;
3592 case 7:
3593 check_insn(env, ctx, ASE_MT);
3594 gen_helper_mtc0_tcschefback(arg);
3595 rn = "TCScheFBack";
3596 break;
3597 default:
3598 goto die;
3599 }
3600 break;
3601 case 3:
3602 switch (sel) {
3603 case 0:
3604 gen_helper_mtc0_entrylo1(arg);
3605 rn = "EntryLo1";
3606 break;
3607 default:
3608 goto die;
3609 }
3610 break;
3611 case 4:
3612 switch (sel) {
3613 case 0:
3614 gen_helper_mtc0_context(arg);
3615 rn = "Context";
3616 break;
3617 case 1:
3618 // gen_helper_mtc0_contextconfig(arg); /* SmartMIPS ASE */
3619 rn = "ContextConfig";
3620 // break;
3621 default:
3622 goto die;
3623 }
3624 break;
3625 case 5:
3626 switch (sel) {
3627 case 0:
3628 gen_helper_mtc0_pagemask(arg);
3629 rn = "PageMask";
3630 break;
3631 case 1:
3632 check_insn(env, ctx, ISA_MIPS32R2);
3633 gen_helper_mtc0_pagegrain(arg);
3634 rn = "PageGrain";
3635 break;
3636 default:
3637 goto die;
3638 }
3639 break;
3640 case 6:
3641 switch (sel) {
3642 case 0:
3643 gen_helper_mtc0_wired(arg);
3644 rn = "Wired";
3645 break;
3646 case 1:
3647 check_insn(env, ctx, ISA_MIPS32R2);
3648 gen_helper_mtc0_srsconf0(arg);
3649 rn = "SRSConf0";
3650 break;
3651 case 2:
3652 check_insn(env, ctx, ISA_MIPS32R2);
3653 gen_helper_mtc0_srsconf1(arg);
3654 rn = "SRSConf1";
3655 break;
3656 case 3:
3657 check_insn(env, ctx, ISA_MIPS32R2);
3658 gen_helper_mtc0_srsconf2(arg);
3659 rn = "SRSConf2";
3660 break;
3661 case 4:
3662 check_insn(env, ctx, ISA_MIPS32R2);
3663 gen_helper_mtc0_srsconf3(arg);
3664 rn = "SRSConf3";
3665 break;
3666 case 5:
3667 check_insn(env, ctx, ISA_MIPS32R2);
3668 gen_helper_mtc0_srsconf4(arg);
3669 rn = "SRSConf4";
3670 break;
3671 default:
3672 goto die;
3673 }
3674 break;
3675 case 7:
3676 switch (sel) {
3677 case 0:
3678 check_insn(env, ctx, ISA_MIPS32R2);
3679 gen_helper_mtc0_hwrena(arg);
3680 rn = "HWREna";
3681 break;
3682 default:
3683 goto die;
3684 }
3685 break;
3686 case 8:
3687 /* ignored */
3688 rn = "BadVAddr";
3689 break;
3690 case 9:
3691 switch (sel) {
3692 case 0:
3693 gen_helper_mtc0_count(arg);
3694 rn = "Count";
3695 break;
3696 /* 6,7 are implementation dependent */
3697 default:
3698 goto die;
3699 }
3700 break;
3701 case 10:
3702 switch (sel) {
3703 case 0:
3704 gen_helper_mtc0_entryhi(arg);
3705 rn = "EntryHi";
3706 break;
3707 default:
3708 goto die;
3709 }
3710 break;
3711 case 11:
3712 switch (sel) {
3713 case 0:
3714 gen_helper_mtc0_compare(arg);
3715 rn = "Compare";
3716 break;
3717 /* 6,7 are implementation dependent */
3718 default:
3719 goto die;
3720 }
3721 break;
3722 case 12:
3723 switch (sel) {
3724 case 0:
3725 save_cpu_state(ctx, 1);
3726 gen_helper_mtc0_status(arg);
3727 /* BS_STOP isn't good enough here, hflags may have changed. */
3728 gen_save_pc(ctx->pc + 4);
3729 ctx->bstate = BS_EXCP;
3730 rn = "Status";
3731 break;
3732 case 1:
3733 check_insn(env, ctx, ISA_MIPS32R2);
3734 gen_helper_mtc0_intctl(arg);
3735 /* Stop translation as we may have switched the execution mode */
3736 ctx->bstate = BS_STOP;
3737 rn = "IntCtl";
3738 break;
3739 case 2:
3740 check_insn(env, ctx, ISA_MIPS32R2);
3741 gen_helper_mtc0_srsctl(arg);
3742 /* Stop translation as we may have switched the execution mode */
3743 ctx->bstate = BS_STOP;
3744 rn = "SRSCtl";
3745 break;
3746 case 3:
3747 check_insn(env, ctx, ISA_MIPS32R2);
3748 gen_mtc0_store32(arg, offsetof(CPUState, CP0_SRSMap));
3749 /* Stop translation as we may have switched the execution mode */
3750 ctx->bstate = BS_STOP;
3751 rn = "SRSMap";
3752 break;
3753 default:
3754 goto die;
3755 }
3756 break;
3757 case 13:
3758 switch (sel) {
3759 case 0:
3760 save_cpu_state(ctx, 1);
3761 gen_helper_mtc0_cause(arg);
3762 rn = "Cause";
3763 break;
3764 default:
3765 goto die;
3766 }
3767 break;
3768 case 14:
3769 switch (sel) {
3770 case 0:
3771 gen_mtc0_store64(arg, offsetof(CPUState, CP0_EPC));
3772 rn = "EPC";
3773 break;
3774 default:
3775 goto die;
3776 }
3777 break;
3778 case 15:
3779 switch (sel) {
3780 case 0:
3781 /* ignored */
3782 rn = "PRid";
3783 break;
3784 case 1:
3785 check_insn(env, ctx, ISA_MIPS32R2);
3786 gen_helper_mtc0_ebase(arg);
3787 rn = "EBase";
3788 break;
3789 default:
3790 goto die;
3791 }
3792 break;
3793 case 16:
3794 switch (sel) {
3795 case 0:
3796 gen_helper_mtc0_config0(arg);
3797 rn = "Config";
3798 /* Stop translation as we may have switched the execution mode */
3799 ctx->bstate = BS_STOP;
3800 break;
3801 case 1:
3802 /* ignored, read only */
3803 rn = "Config1";
3804 break;
3805 case 2:
3806 gen_helper_mtc0_config2(arg);
3807 rn = "Config2";
3808 /* Stop translation as we may have switched the execution mode */
3809 ctx->bstate = BS_STOP;
3810 break;
3811 case 3:
3812 /* ignored, read only */
3813 rn = "Config3";
3814 break;
3815 /* 4,5 are reserved */
3816 /* 6,7 are implementation dependent */
3817 case 6:
3818 /* ignored */
3819 rn = "Config6";
3820 break;
3821 case 7:
3822 /* ignored */
3823 rn = "Config7";
3824 break;
3825 default:
3826 rn = "Invalid config selector";
3827 goto die;
3828 }
3829 break;
3830 case 17:
3831 switch (sel) {
3832 case 0:
3833 gen_helper_mtc0_lladdr(arg);
3834 rn = "LLAddr";
3835 break;
3836 default:
3837 goto die;
3838 }
3839 break;
3840 case 18:
3841 switch (sel) {
3842 case 0 ... 7:
3843 gen_helper_1i(mtc0_watchlo, arg, sel);
3844 rn = "WatchLo";
3845 break;
3846 default:
3847 goto die;
3848 }
3849 break;
3850 case 19:
3851 switch (sel) {
3852 case 0 ... 7:
3853 gen_helper_1i(mtc0_watchhi, arg, sel);
3854 rn = "WatchHi";
3855 break;
3856 default:
3857 goto die;
3858 }
3859 break;
3860 case 20:
3861 switch (sel) {
3862 case 0:
3863 #if defined(TARGET_MIPS64)
3864 check_insn(env, ctx, ISA_MIPS3);
3865 gen_helper_mtc0_xcontext(arg);
3866 rn = "XContext";
3867 break;
3868 #endif
3869 default:
3870 goto die;
3871 }
3872 break;
3873 case 21:
3874 /* Officially reserved, but sel 0 is used for R1x000 framemask */
3875 switch (sel) {
3876 case 0:
3877 gen_helper_mtc0_framemask(arg);
3878 rn = "Framemask";
3879 break;
3880 default:
3881 goto die;
3882 }
3883 break;
3884 case 22:
3885 /* ignored */
3886 rn = "Diagnostic"; /* implementation dependent */
3887 break;
3888 case 23:
3889 switch (sel) {
3890 case 0:
3891 gen_helper_mtc0_debug(arg); /* EJTAG support */
3892 /* BS_STOP isn't good enough here, hflags may have changed. */
3893 gen_save_pc(ctx->pc + 4);
3894 ctx->bstate = BS_EXCP;
3895 rn = "Debug";
3896 break;
3897 case 1:
3898 // gen_helper_mtc0_tracecontrol(arg); /* PDtrace support */
3899 rn = "TraceControl";
3900 /* Stop translation as we may have switched the execution mode */
3901 ctx->bstate = BS_STOP;
3902 // break;
3903 case 2:
3904 // gen_helper_mtc0_tracecontrol2(arg); /* PDtrace support */
3905 rn = "TraceControl2";
3906 /* Stop translation as we may have switched the execution mode */
3907 ctx->bstate = BS_STOP;
3908 // break;
3909 case 3:
3910 /* Stop translation as we may have switched the execution mode */
3911 ctx->bstate = BS_STOP;
3912 // gen_helper_mtc0_usertracedata(arg); /* PDtrace support */
3913 rn = "UserTraceData";
3914 /* Stop translation as we may have switched the execution mode */
3915 ctx->bstate = BS_STOP;
3916 // break;
3917 case 4:
3918 // gen_helper_mtc0_tracebpc(arg); /* PDtrace support */
3919 /* Stop translation as we may have switched the execution mode */
3920 ctx->bstate = BS_STOP;
3921 rn = "TraceBPC";
3922 // break;
3923 default:
3924 goto die;
3925 }
3926 break;
3927 case 24:
3928 switch (sel) {
3929 case 0:
3930 /* EJTAG support */
3931 gen_mtc0_store64(arg, offsetof(CPUState, CP0_DEPC));
3932 rn = "DEPC";
3933 break;
3934 default:
3935 goto die;
3936 }
3937 break;
3938 case 25:
3939 switch (sel) {
3940 case 0:
3941 gen_helper_mtc0_performance0(arg);
3942 rn = "Performance0";
3943 break;
3944 case 1:
3945 // gen_helper_mtc0_performance1(arg);
3946 rn = "Performance1";
3947 // break;
3948 case 2:
3949 // gen_helper_mtc0_performance2(arg);
3950 rn = "Performance2";
3951 // break;
3952 case 3:
3953 // gen_helper_mtc0_performance3(arg);
3954 rn = "Performance3";
3955 // break;
3956 case 4:
3957 // gen_helper_mtc0_performance4(arg);
3958 rn = "Performance4";
3959 // break;
3960 case 5:
3961 // gen_helper_mtc0_performance5(arg);
3962 rn = "Performance5";
3963 // break;
3964 case 6:
3965 // gen_helper_mtc0_performance6(arg);
3966 rn = "Performance6";
3967 // break;
3968 case 7:
3969 // gen_helper_mtc0_performance7(arg);
3970 rn = "Performance7";
3971 // break;
3972 default:
3973 goto die;
3974 }
3975 break;
3976 case 26:
3977 /* ignored */
3978 rn = "ECC";
3979 break;
3980 case 27:
3981 switch (sel) {
3982 case 0 ... 3:
3983 /* ignored */
3984 rn = "CacheErr";
3985 break;
3986 default:
3987 goto die;
3988 }
3989 break;
3990 case 28:
3991 switch (sel) {
3992 case 0:
3993 case 2:
3994 case 4:
3995 case 6:
3996 gen_helper_mtc0_taglo(arg);
3997 rn = "TagLo";
3998 break;
3999 case 1:
4000 case 3:
4001 case 5:
4002 case 7:
4003 gen_helper_mtc0_datalo(arg);
4004 rn = "DataLo";
4005 break;
4006 default:
4007 goto die;
4008 }
4009 break;
4010 case 29:
4011 switch (sel) {
4012 case 0:
4013 case 2:
4014 case 4:
4015 case 6:
4016 gen_helper_mtc0_taghi(arg);
4017 rn = "TagHi";
4018 break;
4019 case 1:
4020 case 3:
4021 case 5:
4022 case 7:
4023 gen_helper_mtc0_datahi(arg);
4024 rn = "DataHi";
4025 break;
4026 default:
4027 rn = "invalid sel";
4028 goto die;
4029 }
4030 break;
4031 case 30:
4032 switch (sel) {
4033 case 0:
4034 gen_mtc0_store64(arg, offsetof(CPUState, CP0_ErrorEPC));
4035 rn = "ErrorEPC";
4036 break;
4037 default:
4038 goto die;
4039 }
4040 break;
4041 case 31:
4042 switch (sel) {
4043 case 0:
4044 /* EJTAG support */
4045 gen_mtc0_store32(arg, offsetof(CPUState, CP0_DESAVE));
4046 rn = "DESAVE";
4047 break;
4048 default:
4049 goto die;
4050 }
4051 /* Stop translation as we may have switched the execution mode */
4052 ctx->bstate = BS_STOP;
4053 break;
4054 default:
4055 goto die;
4056 }
4057 LOG_DISAS("mtc0 %s (reg %d sel %d)\n", rn, reg, sel);
4058 /* For simplicity assume that all writes can cause interrupts. */
4059 if (use_icount) {
4060 gen_io_end();
4061 ctx->bstate = BS_STOP;
4062 }
4063 return;
4064
4065 die:
4066 LOG_DISAS("mtc0 %s (reg %d sel %d)\n", rn, reg, sel);
4067 generate_exception(ctx, EXCP_RI);
4068 }
4069
4070 #if defined(TARGET_MIPS64)
gen_dmfc0(CPUState * env,DisasContext * ctx,TCGv arg,int reg,int sel)4071 static void gen_dmfc0 (CPUState *env, DisasContext *ctx, TCGv arg, int reg, int sel)
4072 {
4073 const char *rn = "invalid";
4074
4075 if (sel != 0)
4076 check_insn(env, ctx, ISA_MIPS64);
4077
4078 switch (reg) {
4079 case 0:
4080 switch (sel) {
4081 case 0:
4082 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Index));
4083 rn = "Index";
4084 break;
4085 case 1:
4086 check_insn(env, ctx, ASE_MT);
4087 gen_helper_mfc0_mvpcontrol(arg);
4088 rn = "MVPControl";
4089 break;
4090 case 2:
4091 check_insn(env, ctx, ASE_MT);
4092 gen_helper_mfc0_mvpconf0(arg);
4093 rn = "MVPConf0";
4094 break;
4095 case 3:
4096 check_insn(env, ctx, ASE_MT);
4097 gen_helper_mfc0_mvpconf1(arg);
4098 rn = "MVPConf1";
4099 break;
4100 default:
4101 goto die;
4102 }
4103 break;
4104 case 1:
4105 switch (sel) {
4106 case 0:
4107 gen_helper_mfc0_random(arg);
4108 rn = "Random";
4109 break;
4110 case 1:
4111 check_insn(env, ctx, ASE_MT);
4112 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEControl));
4113 rn = "VPEControl";
4114 break;
4115 case 2:
4116 check_insn(env, ctx, ASE_MT);
4117 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEConf0));
4118 rn = "VPEConf0";
4119 break;
4120 case 3:
4121 check_insn(env, ctx, ASE_MT);
4122 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEConf1));
4123 rn = "VPEConf1";
4124 break;
4125 case 4:
4126 check_insn(env, ctx, ASE_MT);
4127 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_YQMask));
4128 rn = "YQMask";
4129 break;
4130 case 5:
4131 check_insn(env, ctx, ASE_MT);
4132 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_VPESchedule));
4133 rn = "VPESchedule";
4134 break;
4135 case 6:
4136 check_insn(env, ctx, ASE_MT);
4137 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_VPEScheFBack));
4138 rn = "VPEScheFBack";
4139 break;
4140 case 7:
4141 check_insn(env, ctx, ASE_MT);
4142 gen_mfc0_load32(arg, offsetof(CPUState, CP0_VPEOpt));
4143 rn = "VPEOpt";
4144 break;
4145 default:
4146 goto die;
4147 }
4148 break;
4149 case 2:
4150 switch (sel) {
4151 case 0:
4152 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EntryLo0));
4153 rn = "EntryLo0";
4154 break;
4155 case 1:
4156 check_insn(env, ctx, ASE_MT);
4157 gen_helper_mfc0_tcstatus(arg);
4158 rn = "TCStatus";
4159 break;
4160 case 2:
4161 check_insn(env, ctx, ASE_MT);
4162 gen_helper_mfc0_tcbind(arg);
4163 rn = "TCBind";
4164 break;
4165 case 3:
4166 check_insn(env, ctx, ASE_MT);
4167 gen_helper_dmfc0_tcrestart(arg);
4168 rn = "TCRestart";
4169 break;
4170 case 4:
4171 check_insn(env, ctx, ASE_MT);
4172 gen_helper_dmfc0_tchalt(arg);
4173 rn = "TCHalt";
4174 break;
4175 case 5:
4176 check_insn(env, ctx, ASE_MT);
4177 gen_helper_dmfc0_tccontext(arg);
4178 rn = "TCContext";
4179 break;
4180 case 6:
4181 check_insn(env, ctx, ASE_MT);
4182 gen_helper_dmfc0_tcschedule(arg);
4183 rn = "TCSchedule";
4184 break;
4185 case 7:
4186 check_insn(env, ctx, ASE_MT);
4187 gen_helper_dmfc0_tcschefback(arg);
4188 rn = "TCScheFBack";
4189 break;
4190 default:
4191 goto die;
4192 }
4193 break;
4194 case 3:
4195 switch (sel) {
4196 case 0:
4197 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EntryLo1));
4198 rn = "EntryLo1";
4199 break;
4200 default:
4201 goto die;
4202 }
4203 break;
4204 case 4:
4205 switch (sel) {
4206 case 0:
4207 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_Context));
4208 rn = "Context";
4209 break;
4210 case 1:
4211 // gen_helper_dmfc0_contextconfig(arg); /* SmartMIPS ASE */
4212 rn = "ContextConfig";
4213 // break;
4214 default:
4215 goto die;
4216 }
4217 break;
4218 case 5:
4219 switch (sel) {
4220 case 0:
4221 gen_mfc0_load32(arg, offsetof(CPUState, CP0_PageMask));
4222 rn = "PageMask";
4223 break;
4224 case 1:
4225 check_insn(env, ctx, ISA_MIPS32R2);
4226 gen_mfc0_load32(arg, offsetof(CPUState, CP0_PageGrain));
4227 rn = "PageGrain";
4228 break;
4229 default:
4230 goto die;
4231 }
4232 break;
4233 case 6:
4234 switch (sel) {
4235 case 0:
4236 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Wired));
4237 rn = "Wired";
4238 break;
4239 case 1:
4240 check_insn(env, ctx, ISA_MIPS32R2);
4241 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf0));
4242 rn = "SRSConf0";
4243 break;
4244 case 2:
4245 check_insn(env, ctx, ISA_MIPS32R2);
4246 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf1));
4247 rn = "SRSConf1";
4248 break;
4249 case 3:
4250 check_insn(env, ctx, ISA_MIPS32R2);
4251 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf2));
4252 rn = "SRSConf2";
4253 break;
4254 case 4:
4255 check_insn(env, ctx, ISA_MIPS32R2);
4256 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf3));
4257 rn = "SRSConf3";
4258 break;
4259 case 5:
4260 check_insn(env, ctx, ISA_MIPS32R2);
4261 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSConf4));
4262 rn = "SRSConf4";
4263 break;
4264 default:
4265 goto die;
4266 }
4267 break;
4268 case 7:
4269 switch (sel) {
4270 case 0:
4271 check_insn(env, ctx, ISA_MIPS32R2);
4272 gen_mfc0_load32(arg, offsetof(CPUState, CP0_HWREna));
4273 rn = "HWREna";
4274 break;
4275 default:
4276 goto die;
4277 }
4278 break;
4279 case 8:
4280 switch (sel) {
4281 case 0:
4282 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_BadVAddr));
4283 rn = "BadVAddr";
4284 break;
4285 default:
4286 goto die;
4287 }
4288 break;
4289 case 9:
4290 switch (sel) {
4291 case 0:
4292 /* Mark as an IO operation because we read the time. */
4293 if (use_icount)
4294 gen_io_start();
4295 gen_helper_mfc0_count(arg);
4296 if (use_icount) {
4297 gen_io_end();
4298 ctx->bstate = BS_STOP;
4299 }
4300 rn = "Count";
4301 break;
4302 /* 6,7 are implementation dependent */
4303 default:
4304 goto die;
4305 }
4306 break;
4307 case 10:
4308 switch (sel) {
4309 case 0:
4310 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EntryHi));
4311 rn = "EntryHi";
4312 break;
4313 default:
4314 goto die;
4315 }
4316 break;
4317 case 11:
4318 switch (sel) {
4319 case 0:
4320 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Compare));
4321 rn = "Compare";
4322 break;
4323 /* 6,7 are implementation dependent */
4324 default:
4325 goto die;
4326 }
4327 break;
4328 case 12:
4329 switch (sel) {
4330 case 0:
4331 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Status));
4332 rn = "Status";
4333 break;
4334 case 1:
4335 check_insn(env, ctx, ISA_MIPS32R2);
4336 gen_mfc0_load32(arg, offsetof(CPUState, CP0_IntCtl));
4337 rn = "IntCtl";
4338 break;
4339 case 2:
4340 check_insn(env, ctx, ISA_MIPS32R2);
4341 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSCtl));
4342 rn = "SRSCtl";
4343 break;
4344 case 3:
4345 check_insn(env, ctx, ISA_MIPS32R2);
4346 gen_mfc0_load32(arg, offsetof(CPUState, CP0_SRSMap));
4347 rn = "SRSMap";
4348 break;
4349 default:
4350 goto die;
4351 }
4352 break;
4353 case 13:
4354 switch (sel) {
4355 case 0:
4356 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Cause));
4357 rn = "Cause";
4358 break;
4359 default:
4360 goto die;
4361 }
4362 break;
4363 case 14:
4364 switch (sel) {
4365 case 0:
4366 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_EPC));
4367 rn = "EPC";
4368 break;
4369 default:
4370 goto die;
4371 }
4372 break;
4373 case 15:
4374 switch (sel) {
4375 case 0:
4376 gen_mfc0_load32(arg, offsetof(CPUState, CP0_PRid));
4377 rn = "PRid";
4378 break;
4379 case 1:
4380 check_insn(env, ctx, ISA_MIPS32R2);
4381 gen_mfc0_load32(arg, offsetof(CPUState, CP0_EBase));
4382 rn = "EBase";
4383 break;
4384 default:
4385 goto die;
4386 }
4387 break;
4388 case 16:
4389 switch (sel) {
4390 case 0:
4391 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config0));
4392 rn = "Config";
4393 break;
4394 case 1:
4395 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config1));
4396 rn = "Config1";
4397 break;
4398 case 2:
4399 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config2));
4400 rn = "Config2";
4401 break;
4402 case 3:
4403 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config3));
4404 rn = "Config3";
4405 break;
4406 /* 6,7 are implementation dependent */
4407 case 6:
4408 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config6));
4409 rn = "Config6";
4410 break;
4411 case 7:
4412 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Config7));
4413 rn = "Config7";
4414 break;
4415 default:
4416 goto die;
4417 }
4418 break;
4419 case 17:
4420 switch (sel) {
4421 case 0:
4422 gen_helper_dmfc0_lladdr(arg);
4423 rn = "LLAddr";
4424 break;
4425 default:
4426 goto die;
4427 }
4428 break;
4429 case 18:
4430 switch (sel) {
4431 case 0 ... 7:
4432 gen_helper_1i(dmfc0_watchlo, arg, sel);
4433 rn = "WatchLo";
4434 break;
4435 default:
4436 goto die;
4437 }
4438 break;
4439 case 19:
4440 switch (sel) {
4441 case 0 ... 7:
4442 gen_helper_1i(mfc0_watchhi, arg, sel);
4443 rn = "WatchHi";
4444 break;
4445 default:
4446 goto die;
4447 }
4448 break;
4449 case 20:
4450 switch (sel) {
4451 case 0:
4452 check_insn(env, ctx, ISA_MIPS3);
4453 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_XContext));
4454 rn = "XContext";
4455 break;
4456 default:
4457 goto die;
4458 }
4459 break;
4460 case 21:
4461 /* Officially reserved, but sel 0 is used for R1x000 framemask */
4462 switch (sel) {
4463 case 0:
4464 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Framemask));
4465 rn = "Framemask";
4466 break;
4467 default:
4468 goto die;
4469 }
4470 break;
4471 case 22:
4472 tcg_gen_movi_tl(arg, 0); /* unimplemented */
4473 rn = "'Diagnostic"; /* implementation dependent */
4474 break;
4475 case 23:
4476 switch (sel) {
4477 case 0:
4478 gen_helper_mfc0_debug(arg); /* EJTAG support */
4479 rn = "Debug";
4480 break;
4481 case 1:
4482 // gen_helper_dmfc0_tracecontrol(arg); /* PDtrace support */
4483 rn = "TraceControl";
4484 // break;
4485 case 2:
4486 // gen_helper_dmfc0_tracecontrol2(arg); /* PDtrace support */
4487 rn = "TraceControl2";
4488 // break;
4489 case 3:
4490 // gen_helper_dmfc0_usertracedata(arg); /* PDtrace support */
4491 rn = "UserTraceData";
4492 // break;
4493 case 4:
4494 // gen_helper_dmfc0_tracebpc(arg); /* PDtrace support */
4495 rn = "TraceBPC";
4496 // break;
4497 default:
4498 goto die;
4499 }
4500 break;
4501 case 24:
4502 switch (sel) {
4503 case 0:
4504 /* EJTAG support */
4505 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_DEPC));
4506 rn = "DEPC";
4507 break;
4508 default:
4509 goto die;
4510 }
4511 break;
4512 case 25:
4513 switch (sel) {
4514 case 0:
4515 gen_mfc0_load32(arg, offsetof(CPUState, CP0_Performance0));
4516 rn = "Performance0";
4517 break;
4518 case 1:
4519 // gen_helper_dmfc0_performance1(arg);
4520 rn = "Performance1";
4521 // break;
4522 case 2:
4523 // gen_helper_dmfc0_performance2(arg);
4524 rn = "Performance2";
4525 // break;
4526 case 3:
4527 // gen_helper_dmfc0_performance3(arg);
4528 rn = "Performance3";
4529 // break;
4530 case 4:
4531 // gen_helper_dmfc0_performance4(arg);
4532 rn = "Performance4";
4533 // break;
4534 case 5:
4535 // gen_helper_dmfc0_performance5(arg);
4536 rn = "Performance5";
4537 // break;
4538 case 6:
4539 // gen_helper_dmfc0_performance6(arg);
4540 rn = "Performance6";
4541 // break;
4542 case 7:
4543 // gen_helper_dmfc0_performance7(arg);
4544 rn = "Performance7";
4545 // break;
4546 default:
4547 goto die;
4548 }
4549 break;
4550 case 26:
4551 tcg_gen_movi_tl(arg, 0); /* unimplemented */
4552 rn = "ECC";
4553 break;
4554 case 27:
4555 switch (sel) {
4556 /* ignored */
4557 case 0 ... 3:
4558 tcg_gen_movi_tl(arg, 0); /* unimplemented */
4559 rn = "CacheErr";
4560 break;
4561 default:
4562 goto die;
4563 }
4564 break;
4565 case 28:
4566 switch (sel) {
4567 case 0:
4568 case 2:
4569 case 4:
4570 case 6:
4571 gen_mfc0_load32(arg, offsetof(CPUState, CP0_TagLo));
4572 rn = "TagLo";
4573 break;
4574 case 1:
4575 case 3:
4576 case 5:
4577 case 7:
4578 gen_mfc0_load32(arg, offsetof(CPUState, CP0_DataLo));
4579 rn = "DataLo";
4580 break;
4581 default:
4582 goto die;
4583 }
4584 break;
4585 case 29:
4586 switch (sel) {
4587 case 0:
4588 case 2:
4589 case 4:
4590 case 6:
4591 gen_mfc0_load32(arg, offsetof(CPUState, CP0_TagHi));
4592 rn = "TagHi";
4593 break;
4594 case 1:
4595 case 3:
4596 case 5:
4597 case 7:
4598 gen_mfc0_load32(arg, offsetof(CPUState, CP0_DataHi));
4599 rn = "DataHi";
4600 break;
4601 default:
4602 goto die;
4603 }
4604 break;
4605 case 30:
4606 switch (sel) {
4607 case 0:
4608 tcg_gen_ld_tl(arg, cpu_env, offsetof(CPUState, CP0_ErrorEPC));
4609 rn = "ErrorEPC";
4610 break;
4611 default:
4612 goto die;
4613 }
4614 break;
4615 case 31:
4616 switch (sel) {
4617 case 0:
4618 /* EJTAG support */
4619 gen_mfc0_load32(arg, offsetof(CPUState, CP0_DESAVE));
4620 rn = "DESAVE";
4621 break;
4622 default:
4623 goto die;
4624 }
4625 break;
4626 default:
4627 goto die;
4628 }
4629 LOG_DISAS("dmfc0 %s (reg %d sel %d)\n", rn, reg, sel);
4630 return;
4631
4632 die:
4633 LOG_DISAS("dmfc0 %s (reg %d sel %d)\n", rn, reg, sel);
4634 generate_exception(ctx, EXCP_RI);
4635 }
4636
gen_dmtc0(CPUState * env,DisasContext * ctx,TCGv arg,int reg,int sel)4637 static void gen_dmtc0 (CPUState *env, DisasContext *ctx, TCGv arg, int reg, int sel)
4638 {
4639 const char *rn = "invalid";
4640
4641 if (sel != 0)
4642 check_insn(env, ctx, ISA_MIPS64);
4643
4644 if (use_icount)
4645 gen_io_start();
4646
4647 switch (reg) {
4648 case 0:
4649 switch (sel) {
4650 case 0:
4651 gen_helper_mtc0_index(arg);
4652 rn = "Index";
4653 break;
4654 case 1:
4655 check_insn(env, ctx, ASE_MT);
4656 gen_helper_mtc0_mvpcontrol(arg);
4657 rn = "MVPControl";
4658 break;
4659 case 2:
4660 check_insn(env, ctx, ASE_MT);
4661 /* ignored */
4662 rn = "MVPConf0";
4663 break;
4664 case 3:
4665 check_insn(env, ctx, ASE_MT);
4666 /* ignored */
4667 rn = "MVPConf1";
4668 break;
4669 default:
4670 goto die;
4671 }
4672 break;
4673 case 1:
4674 switch (sel) {
4675 case 0:
4676 /* ignored */
4677 rn = "Random";
4678 break;
4679 case 1:
4680 check_insn(env, ctx, ASE_MT);
4681 gen_helper_mtc0_vpecontrol(arg);
4682 rn = "VPEControl";
4683 break;
4684 case 2:
4685 check_insn(env, ctx, ASE_MT);
4686 gen_helper_mtc0_vpeconf0(arg);
4687 rn = "VPEConf0";
4688 break;
4689 case 3:
4690 check_insn(env, ctx, ASE_MT);
4691 gen_helper_mtc0_vpeconf1(arg);
4692 rn = "VPEConf1";
4693 break;
4694 case 4:
4695 check_insn(env, ctx, ASE_MT);
4696 gen_helper_mtc0_yqmask(arg);
4697 rn = "YQMask";
4698 break;
4699 case 5:
4700 check_insn(env, ctx, ASE_MT);
4701 tcg_gen_st_tl(arg, cpu_env, offsetof(CPUState, CP0_VPESchedule));
4702 rn = "VPESchedule";
4703 break;
4704 case 6:
4705 check_insn(env, ctx, ASE_MT);
4706 tcg_gen_st_tl(arg, cpu_env, offsetof(CPUState, CP0_VPEScheFBack));
4707 rn = "VPEScheFBack";
4708 break;
4709 case 7:
4710 check_insn(env, ctx, ASE_MT);
4711 gen_helper_mtc0_vpeopt(arg);
4712 rn = "VPEOpt";
4713 break;
4714 default:
4715 goto die;
4716 }
4717 break;
4718 case 2:
4719 switch (sel) {
4720 case 0:
4721 gen_helper_mtc0_entrylo0(arg);
4722 rn = "EntryLo0";
4723 break;
4724 case 1:
4725 check_insn(env, ctx, ASE_MT);
4726 gen_helper_mtc0_tcstatus(arg);
4727 rn = "TCStatus";
4728 break;
4729 case 2:
4730 check_insn(env, ctx, ASE_MT);
4731 gen_helper_mtc0_tcbind(arg);
4732 rn = "TCBind";
4733 break;
4734 case 3:
4735 check_insn(env, ctx, ASE_MT);
4736 gen_helper_mtc0_tcrestart(arg);
4737 rn = "TCRestart";
4738 break;
4739 case 4:
4740 check_insn(env, ctx, ASE_MT);
4741 gen_helper_mtc0_tchalt(arg);
4742 rn = "TCHalt";
4743 break;
4744 case 5:
4745 check_insn(env, ctx, ASE_MT);
4746 gen_helper_mtc0_tccontext(arg);
4747 rn = "TCContext";
4748 break;
4749 case 6:
4750 check_insn(env, ctx, ASE_MT);
4751 gen_helper_mtc0_tcschedule(arg);
4752 rn = "TCSchedule";
4753 break;
4754 case 7:
4755 check_insn(env, ctx, ASE_MT);
4756 gen_helper_mtc0_tcschefback(arg);
4757 rn = "TCScheFBack";
4758 break;
4759 default:
4760 goto die;
4761 }
4762 break;
4763 case 3:
4764 switch (sel) {
4765 case 0:
4766 gen_helper_mtc0_entrylo1(arg);
4767 rn = "EntryLo1";
4768 break;
4769 default:
4770 goto die;
4771 }
4772 break;
4773 case 4:
4774 switch (sel) {
4775 case 0:
4776 gen_helper_mtc0_context(arg);
4777 rn = "Context";
4778 break;
4779 case 1:
4780 // gen_helper_mtc0_contextconfig(arg); /* SmartMIPS ASE */
4781 rn = "ContextConfig";
4782 // break;
4783 default:
4784 goto die;
4785 }
4786 break;
4787 case 5:
4788 switch (sel) {
4789 case 0:
4790 gen_helper_mtc0_pagemask(arg);
4791 rn = "PageMask";
4792 break;
4793 case 1:
4794 check_insn(env, ctx, ISA_MIPS32R2);
4795 gen_helper_mtc0_pagegrain(arg);
4796 rn = "PageGrain";
4797 break;
4798 default:
4799 goto die;
4800 }
4801 break;
4802 case 6:
4803 switch (sel) {
4804 case 0:
4805 gen_helper_mtc0_wired(arg);
4806 rn = "Wired";
4807 break;
4808 case 1:
4809 check_insn(env, ctx, ISA_MIPS32R2);
4810 gen_helper_mtc0_srsconf0(arg);
4811 rn = "SRSConf0";
4812 break;
4813 case 2:
4814 check_insn(env, ctx, ISA_MIPS32R2);
4815 gen_helper_mtc0_srsconf1(arg);
4816 rn = "SRSConf1";
4817 break;
4818 case 3:
4819 check_insn(env, ctx, ISA_MIPS32R2);
4820 gen_helper_mtc0_srsconf2(arg);
4821 rn = "SRSConf2";
4822 break;
4823 case 4:
4824 check_insn(env, ctx, ISA_MIPS32R2);
4825 gen_helper_mtc0_srsconf3(arg);
4826 rn = "SRSConf3";
4827 break;
4828 case 5:
4829 check_insn(env, ctx, ISA_MIPS32R2);
4830 gen_helper_mtc0_srsconf4(arg);
4831 rn = "SRSConf4";
4832 break;
4833 default:
4834 goto die;
4835 }
4836 break;
4837 case 7:
4838 switch (sel) {
4839 case 0:
4840 check_insn(env, ctx, ISA_MIPS32R2);
4841 gen_helper_mtc0_hwrena(arg);
4842 rn = "HWREna";
4843 break;
4844 default:
4845 goto die;
4846 }
4847 break;
4848 case 8:
4849 /* ignored */
4850 rn = "BadVAddr";
4851 break;
4852 case 9:
4853 switch (sel) {
4854 case 0:
4855 gen_helper_mtc0_count(arg);
4856 rn = "Count";
4857 break;
4858 /* 6,7 are implementation dependent */
4859 default:
4860 goto die;
4861 }
4862 /* Stop translation as we may have switched the execution mode */
4863 ctx->bstate = BS_STOP;
4864 break;
4865 case 10:
4866 switch (sel) {
4867 case 0:
4868 gen_helper_mtc0_entryhi(arg);
4869 rn = "EntryHi";
4870 break;
4871 default:
4872 goto die;
4873 }
4874 break;
4875 case 11:
4876 switch (sel) {
4877 case 0:
4878 gen_helper_mtc0_compare(arg);
4879 rn = "Compare";
4880 break;
4881 /* 6,7 are implementation dependent */
4882 default:
4883 goto die;
4884 }
4885 /* Stop translation as we may have switched the execution mode */
4886 ctx->bstate = BS_STOP;
4887 break;
4888 case 12:
4889 switch (sel) {
4890 case 0:
4891 save_cpu_state(ctx, 1);
4892 gen_helper_mtc0_status(arg);
4893 /* BS_STOP isn't good enough here, hflags may have changed. */
4894 gen_save_pc(ctx->pc + 4);
4895 ctx->bstate = BS_EXCP;
4896 rn = "Status";
4897 break;
4898 case 1:
4899 check_insn(env, ctx, ISA_MIPS32R2);
4900 gen_helper_mtc0_intctl(arg);
4901 /* Stop translation as we may have switched the execution mode */
4902 ctx->bstate = BS_STOP;
4903 rn = "IntCtl";
4904 break;
4905 case 2:
4906 check_insn(env, ctx, ISA_MIPS32R2);
4907 gen_helper_mtc0_srsctl(arg);
4908 /* Stop translation as we may have switched the execution mode */
4909 ctx->bstate = BS_STOP;
4910 rn = "SRSCtl";
4911 break;
4912 case 3:
4913 check_insn(env, ctx, ISA_MIPS32R2);
4914 gen_mtc0_store32(arg, offsetof(CPUState, CP0_SRSMap));
4915 /* Stop translation as we may have switched the execution mode */
4916 ctx->bstate = BS_STOP;
4917 rn = "SRSMap";
4918 break;
4919 default:
4920 goto die;
4921 }
4922 break;
4923 case 13:
4924 switch (sel) {
4925 case 0:
4926 save_cpu_state(ctx, 1);
4927 gen_helper_mtc0_cause(arg);
4928 rn = "Cause";
4929 break;
4930 default:
4931 goto die;
4932 }
4933 break;
4934 case 14:
4935 switch (sel) {
4936 case 0:
4937 tcg_gen_st_tl(arg, cpu_env, offsetof(CPUState, CP0_EPC));
4938 rn = "EPC";
4939 break;
4940 default:
4941 goto die;
4942 }
4943 break;
4944 case 15:
4945 switch (sel) {
4946 case 0:
4947 /* ignored */
4948 rn = "PRid";
4949 break;
4950 case 1:
4951 check_insn(env, ctx, ISA_MIPS32R2);
4952 gen_helper_mtc0_ebase(arg);
4953 rn = "EBase";
4954 break;
4955 default:
4956 goto die;
4957 }
4958 break;
4959 case 16:
4960 switch (sel) {
4961 case 0:
4962 gen_helper_mtc0_config0(arg);
4963 rn = "Config";
4964 /* Stop translation as we may have switched the execution mode */
4965 ctx->bstate = BS_STOP;
4966 break;
4967 case 1:
4968 /* ignored, read only */
4969 rn = "Config1";
4970 break;
4971 case 2:
4972 gen_helper_mtc0_config2(arg);
4973 rn = "Config2";
4974 /* Stop translation as we may have switched the execution mode */
4975 ctx->bstate = BS_STOP;
4976 break;
4977 case 3:
4978 /* ignored */
4979 rn = "Config3";
4980 break;
4981 /* 6,7 are implementation dependent */
4982 default:
4983 rn = "Invalid config selector";
4984 goto die;
4985 }
4986 break;
4987 case 17:
4988 switch (sel) {
4989 case 0:
4990 gen_helper_mtc0_lladdr(arg);
4991 rn = "LLAddr";
4992 break;
4993 default:
4994 goto die;
4995 }
4996 break;
4997 case 18:
4998 switch (sel) {
4999 case 0 ... 7:
5000 gen_helper_1i(mtc0_watchlo, arg, sel);
5001 rn = "WatchLo";
5002 break;
5003 default:
5004 goto die;
5005 }
5006 break;
5007 case 19:
5008 switch (sel) {
5009 case 0 ... 7:
5010 gen_helper_1i(mtc0_watchhi, arg, sel);
5011 rn = "WatchHi";
5012 break;
5013 default:
5014 goto die;
5015 }
5016 break;
5017 case 20:
5018 switch (sel) {
5019 case 0:
5020 check_insn(env, ctx, ISA_MIPS3);
5021 gen_helper_mtc0_xcontext(arg);
5022 rn = "XContext";
5023 break;
5024 default:
5025 goto die;
5026 }
5027 break;
5028 case 21:
5029 /* Officially reserved, but sel 0 is used for R1x000 framemask */
5030 switch (sel) {
5031 case 0:
5032 gen_helper_mtc0_framemask(arg);
5033 rn = "Framemask";
5034 break;
5035 default:
5036 goto die;
5037 }
5038 break;
5039 case 22:
5040 /* ignored */
5041 rn = "Diagnostic"; /* implementation dependent */
5042 break;
5043 case 23:
5044 switch (sel) {
5045 case 0:
5046 gen_helper_mtc0_debug(arg); /* EJTAG support */
5047 /* BS_STOP isn't good enough here, hflags may have changed. */
5048 gen_save_pc(ctx->pc + 4);
5049 ctx->bstate = BS_EXCP;
5050 rn = "Debug";
5051 break;
5052 case 1:
5053 // gen_helper_mtc0_tracecontrol(arg); /* PDtrace support */
5054 /* Stop translation as we may have switched the execution mode */
5055 ctx->bstate = BS_STOP;
5056 rn = "TraceControl";
5057 // break;
5058 case 2:
5059 // gen_helper_mtc0_tracecontrol2(arg); /* PDtrace support */
5060 /* Stop translation as we may have switched the execution mode */
5061 ctx->bstate = BS_STOP;
5062 rn = "TraceControl2";
5063 // break;
5064 case 3:
5065 // gen_helper_mtc0_usertracedata(arg); /* PDtrace support */
5066 /* Stop translation as we may have switched the execution mode */
5067 ctx->bstate = BS_STOP;
5068 rn = "UserTraceData";
5069 // break;
5070 case 4:
5071 // gen_helper_mtc0_tracebpc(arg); /* PDtrace support */
5072 /* Stop translation as we may have switched the execution mode */
5073 ctx->bstate = BS_STOP;
5074 rn = "TraceBPC";
5075 // break;
5076 default:
5077 goto die;
5078 }
5079 break;
5080 case 24:
5081 switch (sel) {
5082 case 0:
5083 /* EJTAG support */
5084 tcg_gen_st_tl(arg, cpu_env, offsetof(CPUState, CP0_DEPC));
5085 rn = "DEPC";
5086 break;
5087 default:
5088 goto die;
5089 }
5090 break;
5091 case 25:
5092 switch (sel) {
5093 case 0:
5094 gen_helper_mtc0_performance0(arg);
5095 rn = "Performance0";
5096 break;
5097 case 1:
5098 // gen_helper_mtc0_performance1(arg);
5099 rn = "Performance1";
5100 // break;
5101 case 2:
5102 // gen_helper_mtc0_performance2(arg);
5103 rn = "Performance2";
5104 // break;
5105 case 3:
5106 // gen_helper_mtc0_performance3(arg);
5107 rn = "Performance3";
5108 // break;
5109 case 4:
5110 // gen_helper_mtc0_performance4(arg);
5111 rn = "Performance4";
5112 // break;
5113 case 5:
5114 // gen_helper_mtc0_performance5(arg);
5115 rn = "Performance5";
5116 // break;
5117 case 6:
5118 // gen_helper_mtc0_performance6(arg);
5119 rn = "Performance6";
5120 // break;
5121 case 7:
5122 // gen_helper_mtc0_performance7(arg);
5123 rn = "Performance7";
5124 // break;
5125 default:
5126 goto die;
5127 }
5128 break;
5129 case 26:
5130 /* ignored */
5131 rn = "ECC";
5132 break;
5133 case 27:
5134 switch (sel) {
5135 case 0 ... 3:
5136 /* ignored */
5137 rn = "CacheErr";
5138 break;
5139 default:
5140 goto die;
5141 }
5142 break;
5143 case 28:
5144 switch (sel) {
5145 case 0:
5146 case 2:
5147 case 4:
5148 case 6:
5149 gen_helper_mtc0_taglo(arg);
5150 rn = "TagLo";
5151 break;
5152 case 1:
5153 case 3:
5154 case 5:
5155 case 7:
5156 gen_helper_mtc0_datalo(arg);
5157 rn = "DataLo";
5158 break;
5159 default:
5160 goto die;
5161 }
5162 break;
5163 case 29:
5164 switch (sel) {
5165 case 0:
5166 case 2:
5167 case 4:
5168 case 6:
5169 gen_helper_mtc0_taghi(arg);
5170 rn = "TagHi";
5171 break;
5172 case 1:
5173 case 3:
5174 case 5:
5175 case 7:
5176 gen_helper_mtc0_datahi(arg);
5177 rn = "DataHi";
5178 break;
5179 default:
5180 rn = "invalid sel";
5181 goto die;
5182 }
5183 break;
5184 case 30:
5185 switch (sel) {
5186 case 0:
5187 tcg_gen_st_tl(arg, cpu_env, offsetof(CPUState, CP0_ErrorEPC));
5188 rn = "ErrorEPC";
5189 break;
5190 default:
5191 goto die;
5192 }
5193 break;
5194 case 31:
5195 switch (sel) {
5196 case 0:
5197 /* EJTAG support */
5198 gen_mtc0_store32(arg, offsetof(CPUState, CP0_DESAVE));
5199 rn = "DESAVE";
5200 break;
5201 default:
5202 goto die;
5203 }
5204 /* Stop translation as we may have switched the execution mode */
5205 ctx->bstate = BS_STOP;
5206 break;
5207 default:
5208 goto die;
5209 }
5210 LOG_DISAS("dmtc0 %s (reg %d sel %d)\n", rn, reg, sel);
5211 /* For simplicity assume that all writes can cause interrupts. */
5212 if (use_icount) {
5213 gen_io_end();
5214 ctx->bstate = BS_STOP;
5215 }
5216 return;
5217
5218 die:
5219 LOG_DISAS("dmtc0 %s (reg %d sel %d)\n", rn, reg, sel);
5220 generate_exception(ctx, EXCP_RI);
5221 }
5222 #endif /* TARGET_MIPS64 */
5223
gen_mftr(CPUState * env,DisasContext * ctx,int rt,int rd,int u,int sel,int h)5224 static void gen_mftr(CPUState *env, DisasContext *ctx, int rt, int rd,
5225 int u, int sel, int h)
5226 {
5227 int other_tc = env->CP0_VPEControl & (0xff << CP0VPECo_TargTC);
5228 TCGv t0 = tcg_temp_local_new();
5229
5230 if ((env->CP0_VPEConf0 & (1 << CP0VPEC0_MVP)) == 0 &&
5231 ((env->tcs[other_tc].CP0_TCBind & (0xf << CP0TCBd_CurVPE)) !=
5232 (env->active_tc.CP0_TCBind & (0xf << CP0TCBd_CurVPE))))
5233 tcg_gen_movi_tl(t0, -1);
5234 else if ((env->CP0_VPEControl & (0xff << CP0VPECo_TargTC)) >
5235 (env->mvp->CP0_MVPConf0 & (0xff << CP0MVPC0_PTC)))
5236 tcg_gen_movi_tl(t0, -1);
5237 else if (u == 0) {
5238 switch (rt) {
5239 case 2:
5240 switch (sel) {
5241 case 1:
5242 gen_helper_mftc0_tcstatus(t0);
5243 break;
5244 case 2:
5245 gen_helper_mftc0_tcbind(t0);
5246 break;
5247 case 3:
5248 gen_helper_mftc0_tcrestart(t0);
5249 break;
5250 case 4:
5251 gen_helper_mftc0_tchalt(t0);
5252 break;
5253 case 5:
5254 gen_helper_mftc0_tccontext(t0);
5255 break;
5256 case 6:
5257 gen_helper_mftc0_tcschedule(t0);
5258 break;
5259 case 7:
5260 gen_helper_mftc0_tcschefback(t0);
5261 break;
5262 default:
5263 gen_mfc0(env, ctx, t0, rt, sel);
5264 break;
5265 }
5266 break;
5267 case 10:
5268 switch (sel) {
5269 case 0:
5270 gen_helper_mftc0_entryhi(t0);
5271 break;
5272 default:
5273 gen_mfc0(env, ctx, t0, rt, sel);
5274 break;
5275 }
5276 case 12:
5277 switch (sel) {
5278 case 0:
5279 gen_helper_mftc0_status(t0);
5280 break;
5281 default:
5282 gen_mfc0(env, ctx, t0, rt, sel);
5283 break;
5284 }
5285 case 23:
5286 switch (sel) {
5287 case 0:
5288 gen_helper_mftc0_debug(t0);
5289 break;
5290 default:
5291 gen_mfc0(env, ctx, t0, rt, sel);
5292 break;
5293 }
5294 break;
5295 default:
5296 gen_mfc0(env, ctx, t0, rt, sel);
5297 }
5298 } else switch (sel) {
5299 /* GPR registers. */
5300 case 0:
5301 gen_helper_1i(mftgpr, t0, rt);
5302 break;
5303 /* Auxiliary CPU registers */
5304 case 1:
5305 switch (rt) {
5306 case 0:
5307 gen_helper_1i(mftlo, t0, 0);
5308 break;
5309 case 1:
5310 gen_helper_1i(mfthi, t0, 0);
5311 break;
5312 case 2:
5313 gen_helper_1i(mftacx, t0, 0);
5314 break;
5315 case 4:
5316 gen_helper_1i(mftlo, t0, 1);
5317 break;
5318 case 5:
5319 gen_helper_1i(mfthi, t0, 1);
5320 break;
5321 case 6:
5322 gen_helper_1i(mftacx, t0, 1);
5323 break;
5324 case 8:
5325 gen_helper_1i(mftlo, t0, 2);
5326 break;
5327 case 9:
5328 gen_helper_1i(mfthi, t0, 2);
5329 break;
5330 case 10:
5331 gen_helper_1i(mftacx, t0, 2);
5332 break;
5333 case 12:
5334 gen_helper_1i(mftlo, t0, 3);
5335 break;
5336 case 13:
5337 gen_helper_1i(mfthi, t0, 3);
5338 break;
5339 case 14:
5340 gen_helper_1i(mftacx, t0, 3);
5341 break;
5342 case 16:
5343 gen_helper_mftdsp(t0);
5344 break;
5345 default:
5346 goto die;
5347 }
5348 break;
5349 /* Floating point (COP1). */
5350 case 2:
5351 /* XXX: For now we support only a single FPU context. */
5352 if (h == 0) {
5353 TCGv_i32 fp0 = tcg_temp_new_i32();
5354
5355 gen_load_fpr32(fp0, rt);
5356 tcg_gen_ext_i32_tl(t0, fp0);
5357 tcg_temp_free_i32(fp0);
5358 } else {
5359 TCGv_i32 fp0 = tcg_temp_new_i32();
5360
5361 gen_load_fpr32h(fp0, rt);
5362 tcg_gen_ext_i32_tl(t0, fp0);
5363 tcg_temp_free_i32(fp0);
5364 }
5365 break;
5366 case 3:
5367 /* XXX: For now we support only a single FPU context. */
5368 gen_helper_1i(cfc1, t0, rt);
5369 break;
5370 /* COP2: Not implemented. */
5371 case 4:
5372 case 5:
5373 /* fall through */
5374 default:
5375 goto die;
5376 }
5377 LOG_DISAS("mftr (reg %d u %d sel %d h %d)\n", rt, u, sel, h);
5378 gen_store_gpr(t0, rd);
5379 tcg_temp_free(t0);
5380 return;
5381
5382 die:
5383 tcg_temp_free(t0);
5384 LOG_DISAS("mftr (reg %d u %d sel %d h %d)\n", rt, u, sel, h);
5385 generate_exception(ctx, EXCP_RI);
5386 }
5387
gen_mttr(CPUState * env,DisasContext * ctx,int rd,int rt,int u,int sel,int h)5388 static void gen_mttr(CPUState *env, DisasContext *ctx, int rd, int rt,
5389 int u, int sel, int h)
5390 {
5391 int other_tc = env->CP0_VPEControl & (0xff << CP0VPECo_TargTC);
5392 TCGv t0 = tcg_temp_local_new();
5393
5394 gen_load_gpr(t0, rt);
5395 if ((env->CP0_VPEConf0 & (1 << CP0VPEC0_MVP)) == 0 &&
5396 ((env->tcs[other_tc].CP0_TCBind & (0xf << CP0TCBd_CurVPE)) !=
5397 (env->active_tc.CP0_TCBind & (0xf << CP0TCBd_CurVPE))))
5398 /* NOP */ ;
5399 else if ((env->CP0_VPEControl & (0xff << CP0VPECo_TargTC)) >
5400 (env->mvp->CP0_MVPConf0 & (0xff << CP0MVPC0_PTC)))
5401 /* NOP */ ;
5402 else if (u == 0) {
5403 switch (rd) {
5404 case 2:
5405 switch (sel) {
5406 case 1:
5407 gen_helper_mttc0_tcstatus(t0);
5408 break;
5409 case 2:
5410 gen_helper_mttc0_tcbind(t0);
5411 break;
5412 case 3:
5413 gen_helper_mttc0_tcrestart(t0);
5414 break;
5415 case 4:
5416 gen_helper_mttc0_tchalt(t0);
5417 break;
5418 case 5:
5419 gen_helper_mttc0_tccontext(t0);
5420 break;
5421 case 6:
5422 gen_helper_mttc0_tcschedule(t0);
5423 break;
5424 case 7:
5425 gen_helper_mttc0_tcschefback(t0);
5426 break;
5427 default:
5428 gen_mtc0(env, ctx, t0, rd, sel);
5429 break;
5430 }
5431 break;
5432 case 10:
5433 switch (sel) {
5434 case 0:
5435 gen_helper_mttc0_entryhi(t0);
5436 break;
5437 default:
5438 gen_mtc0(env, ctx, t0, rd, sel);
5439 break;
5440 }
5441 case 12:
5442 switch (sel) {
5443 case 0:
5444 gen_helper_mttc0_status(t0);
5445 break;
5446 default:
5447 gen_mtc0(env, ctx, t0, rd, sel);
5448 break;
5449 }
5450 case 23:
5451 switch (sel) {
5452 case 0:
5453 gen_helper_mttc0_debug(t0);
5454 break;
5455 default:
5456 gen_mtc0(env, ctx, t0, rd, sel);
5457 break;
5458 }
5459 break;
5460 default:
5461 gen_mtc0(env, ctx, t0, rd, sel);
5462 }
5463 } else switch (sel) {
5464 /* GPR registers. */
5465 case 0:
5466 gen_helper_1i(mttgpr, t0, rd);
5467 break;
5468 /* Auxiliary CPU registers */
5469 case 1:
5470 switch (rd) {
5471 case 0:
5472 gen_helper_1i(mttlo, t0, 0);
5473 break;
5474 case 1:
5475 gen_helper_1i(mtthi, t0, 0);
5476 break;
5477 case 2:
5478 gen_helper_1i(mttacx, t0, 0);
5479 break;
5480 case 4:
5481 gen_helper_1i(mttlo, t0, 1);
5482 break;
5483 case 5:
5484 gen_helper_1i(mtthi, t0, 1);
5485 break;
5486 case 6:
5487 gen_helper_1i(mttacx, t0, 1);
5488 break;
5489 case 8:
5490 gen_helper_1i(mttlo, t0, 2);
5491 break;
5492 case 9:
5493 gen_helper_1i(mtthi, t0, 2);
5494 break;
5495 case 10:
5496 gen_helper_1i(mttacx, t0, 2);
5497 break;
5498 case 12:
5499 gen_helper_1i(mttlo, t0, 3);
5500 break;
5501 case 13:
5502 gen_helper_1i(mtthi, t0, 3);
5503 break;
5504 case 14:
5505 gen_helper_1i(mttacx, t0, 3);
5506 break;
5507 case 16:
5508 gen_helper_mttdsp(t0);
5509 break;
5510 default:
5511 goto die;
5512 }
5513 break;
5514 /* Floating point (COP1). */
5515 case 2:
5516 /* XXX: For now we support only a single FPU context. */
5517 if (h == 0) {
5518 TCGv_i32 fp0 = tcg_temp_new_i32();
5519
5520 tcg_gen_trunc_tl_i32(fp0, t0);
5521 gen_store_fpr32(fp0, rd);
5522 tcg_temp_free_i32(fp0);
5523 } else {
5524 TCGv_i32 fp0 = tcg_temp_new_i32();
5525
5526 tcg_gen_trunc_tl_i32(fp0, t0);
5527 gen_store_fpr32h(fp0, rd);
5528 tcg_temp_free_i32(fp0);
5529 }
5530 break;
5531 case 3:
5532 /* XXX: For now we support only a single FPU context. */
5533 gen_helper_1i(ctc1, t0, rd);
5534 break;
5535 /* COP2: Not implemented. */
5536 case 4:
5537 case 5:
5538 /* fall through */
5539 default:
5540 goto die;
5541 }
5542 LOG_DISAS("mttr (reg %d u %d sel %d h %d)\n", rd, u, sel, h);
5543 tcg_temp_free(t0);
5544 return;
5545
5546 die:
5547 tcg_temp_free(t0);
5548 LOG_DISAS("mttr (reg %d u %d sel %d h %d)\n", rd, u, sel, h);
5549 generate_exception(ctx, EXCP_RI);
5550 }
5551
gen_cp0(CPUState * env,DisasContext * ctx,uint32_t opc,int rt,int rd)5552 static void gen_cp0 (CPUState *env, DisasContext *ctx, uint32_t opc, int rt, int rd)
5553 {
5554 const char *opn = "ldst";
5555
5556 switch (opc) {
5557 case OPC_MFC0:
5558 if (rt == 0) {
5559 /* Treat as NOP. */
5560 return;
5561 }
5562 gen_mfc0(env, ctx, cpu_gpr[rt], rd, ctx->opcode & 0x7);
5563 opn = "mfc0";
5564 break;
5565 case OPC_MTC0:
5566 {
5567 TCGv t0 = tcg_temp_new();
5568
5569 gen_load_gpr(t0, rt);
5570 gen_mtc0(env, ctx, t0, rd, ctx->opcode & 0x7);
5571 tcg_temp_free(t0);
5572 }
5573 opn = "mtc0";
5574 break;
5575 #if defined(TARGET_MIPS64)
5576 case OPC_DMFC0:
5577 check_insn(env, ctx, ISA_MIPS3);
5578 if (rt == 0) {
5579 /* Treat as NOP. */
5580 return;
5581 }
5582 gen_dmfc0(env, ctx, cpu_gpr[rt], rd, ctx->opcode & 0x7);
5583 opn = "dmfc0";
5584 break;
5585 case OPC_DMTC0:
5586 check_insn(env, ctx, ISA_MIPS3);
5587 {
5588 TCGv t0 = tcg_temp_new();
5589
5590 gen_load_gpr(t0, rt);
5591 gen_dmtc0(env, ctx, t0, rd, ctx->opcode & 0x7);
5592 tcg_temp_free(t0);
5593 }
5594 opn = "dmtc0";
5595 break;
5596 #endif
5597 case OPC_MFTR:
5598 check_insn(env, ctx, ASE_MT);
5599 if (rd == 0) {
5600 /* Treat as NOP. */
5601 return;
5602 }
5603 gen_mftr(env, ctx, rt, rd, (ctx->opcode >> 5) & 1,
5604 ctx->opcode & 0x7, (ctx->opcode >> 4) & 1);
5605 opn = "mftr";
5606 break;
5607 case OPC_MTTR:
5608 check_insn(env, ctx, ASE_MT);
5609 gen_mttr(env, ctx, rd, rt, (ctx->opcode >> 5) & 1,
5610 ctx->opcode & 0x7, (ctx->opcode >> 4) & 1);
5611 opn = "mttr";
5612 break;
5613 case OPC_TLBWI:
5614 opn = "tlbwi";
5615 if (!env->tlb->helper_tlbwi)
5616 goto die;
5617 gen_helper_tlbwi();
5618 break;
5619 case OPC_TLBWR:
5620 opn = "tlbwr";
5621 if (!env->tlb->helper_tlbwr)
5622 goto die;
5623 gen_helper_tlbwr();
5624 break;
5625 case OPC_TLBP:
5626 opn = "tlbp";
5627 if (!env->tlb->helper_tlbp)
5628 goto die;
5629 gen_helper_tlbp();
5630 break;
5631 case OPC_TLBR:
5632 opn = "tlbr";
5633 if (!env->tlb->helper_tlbr)
5634 goto die;
5635 gen_helper_tlbr();
5636 break;
5637 case OPC_ERET:
5638 opn = "eret";
5639 check_insn(env, ctx, ISA_MIPS2);
5640 gen_helper_eret();
5641 ctx->bstate = BS_EXCP;
5642 break;
5643 case OPC_DERET:
5644 opn = "deret";
5645 check_insn(env, ctx, ISA_MIPS32);
5646 if (!(ctx->hflags & MIPS_HFLAG_DM)) {
5647 MIPS_INVAL(opn);
5648 generate_exception(ctx, EXCP_RI);
5649 } else {
5650 gen_helper_deret();
5651 ctx->bstate = BS_EXCP;
5652 }
5653 break;
5654 case OPC_WAIT:
5655 opn = "wait";
5656 check_insn(env, ctx, ISA_MIPS3 | ISA_MIPS32);
5657 /* If we get an exception, we want to restart at next instruction */
5658 ctx->pc += 4;
5659 save_cpu_state(ctx, 1);
5660 ctx->pc -= 4;
5661 gen_helper_wait();
5662 ctx->bstate = BS_EXCP;
5663 break;
5664 default:
5665 die:
5666 MIPS_INVAL(opn);
5667 generate_exception(ctx, EXCP_RI);
5668 return;
5669 }
5670 MIPS_DEBUG("%s %s %d", opn, regnames[rt], rd);
5671 }
5672 #endif /* !CONFIG_USER_ONLY */
5673
5674 /* CP1 Branches (before delay slot) */
gen_compute_branch1(CPUState * env,DisasContext * ctx,uint32_t op,int32_t cc,int32_t offset)5675 static void gen_compute_branch1 (CPUState *env, DisasContext *ctx, uint32_t op,
5676 int32_t cc, int32_t offset)
5677 {
5678 target_ulong btarget;
5679 const char *opn = "cp1 cond branch";
5680 TCGv_i32 t0 = tcg_temp_new_i32();
5681
5682 if (cc != 0)
5683 check_insn(env, ctx, ISA_MIPS4 | ISA_MIPS32);
5684
5685 btarget = ctx->pc + 4 + offset;
5686
5687 switch (op) {
5688 case OPC_BC1F:
5689 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5690 tcg_gen_not_i32(t0, t0);
5691 tcg_gen_andi_i32(t0, t0, 1);
5692 tcg_gen_extu_i32_tl(bcond, t0);
5693 opn = "bc1f";
5694 goto not_likely;
5695 case OPC_BC1FL:
5696 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5697 tcg_gen_not_i32(t0, t0);
5698 tcg_gen_andi_i32(t0, t0, 1);
5699 tcg_gen_extu_i32_tl(bcond, t0);
5700 opn = "bc1fl";
5701 goto likely;
5702 case OPC_BC1T:
5703 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5704 tcg_gen_andi_i32(t0, t0, 1);
5705 tcg_gen_extu_i32_tl(bcond, t0);
5706 opn = "bc1t";
5707 goto not_likely;
5708 case OPC_BC1TL:
5709 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5710 tcg_gen_andi_i32(t0, t0, 1);
5711 tcg_gen_extu_i32_tl(bcond, t0);
5712 opn = "bc1tl";
5713 likely:
5714 ctx->hflags |= MIPS_HFLAG_BL;
5715 break;
5716 case OPC_BC1FANY2:
5717 {
5718 TCGv_i32 t1 = tcg_temp_new_i32();
5719 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5720 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+1));
5721 tcg_gen_or_i32(t0, t0, t1);
5722 tcg_temp_free_i32(t1);
5723 tcg_gen_not_i32(t0, t0);
5724 tcg_gen_andi_i32(t0, t0, 1);
5725 tcg_gen_extu_i32_tl(bcond, t0);
5726 }
5727 opn = "bc1any2f";
5728 goto not_likely;
5729 case OPC_BC1TANY2:
5730 {
5731 TCGv_i32 t1 = tcg_temp_new_i32();
5732 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5733 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+1));
5734 tcg_gen_or_i32(t0, t0, t1);
5735 tcg_temp_free_i32(t1);
5736 tcg_gen_andi_i32(t0, t0, 1);
5737 tcg_gen_extu_i32_tl(bcond, t0);
5738 }
5739 opn = "bc1any2t";
5740 goto not_likely;
5741 case OPC_BC1FANY4:
5742 {
5743 TCGv_i32 t1 = tcg_temp_new_i32();
5744 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5745 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+1));
5746 tcg_gen_or_i32(t0, t0, t1);
5747 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+2));
5748 tcg_gen_or_i32(t0, t0, t1);
5749 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+3));
5750 tcg_gen_or_i32(t0, t0, t1);
5751 tcg_temp_free_i32(t1);
5752 tcg_gen_not_i32(t0, t0);
5753 tcg_gen_andi_i32(t0, t0, 1);
5754 tcg_gen_extu_i32_tl(bcond, t0);
5755 }
5756 opn = "bc1any4f";
5757 goto not_likely;
5758 case OPC_BC1TANY4:
5759 {
5760 TCGv_i32 t1 = tcg_temp_new_i32();
5761 tcg_gen_shri_i32(t0, fpu_fcr31, get_fp_bit(cc));
5762 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+1));
5763 tcg_gen_or_i32(t0, t0, t1);
5764 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+2));
5765 tcg_gen_or_i32(t0, t0, t1);
5766 tcg_gen_shri_i32(t1, fpu_fcr31, get_fp_bit(cc+3));
5767 tcg_gen_or_i32(t0, t0, t1);
5768 tcg_temp_free_i32(t1);
5769 tcg_gen_andi_i32(t0, t0, 1);
5770 tcg_gen_extu_i32_tl(bcond, t0);
5771 }
5772 opn = "bc1any4t";
5773 not_likely:
5774 ctx->hflags |= MIPS_HFLAG_BC;
5775 break;
5776 default:
5777 MIPS_INVAL(opn);
5778 generate_exception (ctx, EXCP_RI);
5779 goto out;
5780 }
5781 MIPS_DEBUG("%s: cond %02x target " TARGET_FMT_lx, opn,
5782 ctx->hflags, btarget);
5783 ctx->btarget = btarget;
5784
5785 out:
5786 tcg_temp_free_i32(t0);
5787 }
5788
5789 /* Coprocessor 1 (FPU) */
5790
5791 #define FOP(func, fmt) (((fmt) << 21) | (func))
5792
gen_cp1(DisasContext * ctx,uint32_t opc,int rt,int fs)5793 static void gen_cp1 (DisasContext *ctx, uint32_t opc, int rt, int fs)
5794 {
5795 const char *opn = "cp1 move";
5796 TCGv t0 = tcg_temp_new();
5797
5798 switch (opc) {
5799 case OPC_MFC1:
5800 {
5801 TCGv_i32 fp0 = tcg_temp_new_i32();
5802
5803 gen_load_fpr32(fp0, fs);
5804 tcg_gen_ext_i32_tl(t0, fp0);
5805 tcg_temp_free_i32(fp0);
5806 }
5807 gen_store_gpr(t0, rt);
5808 opn = "mfc1";
5809 break;
5810 case OPC_MTC1:
5811 gen_load_gpr(t0, rt);
5812 {
5813 TCGv_i32 fp0 = tcg_temp_new_i32();
5814
5815 tcg_gen_trunc_tl_i32(fp0, t0);
5816 gen_store_fpr32(fp0, fs);
5817 tcg_temp_free_i32(fp0);
5818 }
5819 opn = "mtc1";
5820 break;
5821 case OPC_CFC1:
5822 gen_helper_1i(cfc1, t0, fs);
5823 gen_store_gpr(t0, rt);
5824 opn = "cfc1";
5825 break;
5826 case OPC_CTC1:
5827 gen_load_gpr(t0, rt);
5828 gen_helper_1i(ctc1, t0, fs);
5829 opn = "ctc1";
5830 break;
5831 #if defined(TARGET_MIPS64)
5832 case OPC_DMFC1:
5833 gen_load_fpr64(ctx, t0, fs);
5834 gen_store_gpr(t0, rt);
5835 opn = "dmfc1";
5836 break;
5837 case OPC_DMTC1:
5838 gen_load_gpr(t0, rt);
5839 gen_store_fpr64(ctx, t0, fs);
5840 opn = "dmtc1";
5841 break;
5842 #endif
5843 case OPC_MFHC1:
5844 {
5845 TCGv_i32 fp0 = tcg_temp_new_i32();
5846
5847 gen_load_fpr32h(fp0, fs);
5848 tcg_gen_ext_i32_tl(t0, fp0);
5849 tcg_temp_free_i32(fp0);
5850 }
5851 gen_store_gpr(t0, rt);
5852 opn = "mfhc1";
5853 break;
5854 case OPC_MTHC1:
5855 gen_load_gpr(t0, rt);
5856 {
5857 TCGv_i32 fp0 = tcg_temp_new_i32();
5858
5859 tcg_gen_trunc_tl_i32(fp0, t0);
5860 gen_store_fpr32h(fp0, fs);
5861 tcg_temp_free_i32(fp0);
5862 }
5863 opn = "mthc1";
5864 break;
5865 default:
5866 MIPS_INVAL(opn);
5867 generate_exception (ctx, EXCP_RI);
5868 goto out;
5869 }
5870 MIPS_DEBUG("%s %s %s", opn, regnames[rt], fregnames[fs]);
5871
5872 out:
5873 tcg_temp_free(t0);
5874 }
5875
gen_movci(DisasContext * ctx,int rd,int rs,int cc,int tf)5876 static void gen_movci (DisasContext *ctx, int rd, int rs, int cc, int tf)
5877 {
5878 int l1;
5879 TCGCond cond;
5880 TCGv_i32 t0;
5881
5882 if (rd == 0) {
5883 /* Treat as NOP. */
5884 return;
5885 }
5886
5887 if (tf)
5888 cond = TCG_COND_EQ;
5889 else
5890 cond = TCG_COND_NE;
5891
5892 l1 = gen_new_label();
5893 t0 = tcg_temp_new_i32();
5894 tcg_gen_andi_i32(t0, fpu_fcr31, 1 << get_fp_bit(cc));
5895 tcg_gen_brcondi_i32(cond, t0, 0, l1);
5896 tcg_temp_free_i32(t0);
5897 if (rs == 0) {
5898 tcg_gen_movi_tl(cpu_gpr[rd], 0);
5899 } else {
5900 tcg_gen_mov_tl(cpu_gpr[rd], cpu_gpr[rs]);
5901 }
5902 gen_set_label(l1);
5903 }
5904
gen_movcf_s(int fs,int fd,int cc,int tf)5905 static inline void gen_movcf_s (int fs, int fd, int cc, int tf)
5906 {
5907 int cond;
5908 TCGv_i32 t0 = tcg_temp_new_i32();
5909 int l1 = gen_new_label();
5910
5911 if (tf)
5912 cond = TCG_COND_EQ;
5913 else
5914 cond = TCG_COND_NE;
5915
5916 tcg_gen_andi_i32(t0, fpu_fcr31, 1 << get_fp_bit(cc));
5917 tcg_gen_brcondi_i32(cond, t0, 0, l1);
5918 gen_load_fpr32(t0, fs);
5919 gen_store_fpr32(t0, fd);
5920 gen_set_label(l1);
5921 tcg_temp_free_i32(t0);
5922 }
5923
gen_movcf_d(DisasContext * ctx,int fs,int fd,int cc,int tf)5924 static inline void gen_movcf_d (DisasContext *ctx, int fs, int fd, int cc, int tf)
5925 {
5926 int cond;
5927 TCGv_i32 t0 = tcg_temp_new_i32();
5928 TCGv_i64 fp0;
5929 int l1 = gen_new_label();
5930
5931 if (tf)
5932 cond = TCG_COND_EQ;
5933 else
5934 cond = TCG_COND_NE;
5935
5936 tcg_gen_andi_i32(t0, fpu_fcr31, 1 << get_fp_bit(cc));
5937 tcg_gen_brcondi_i32(cond, t0, 0, l1);
5938 tcg_temp_free_i32(t0);
5939 fp0 = tcg_temp_new_i64();
5940 gen_load_fpr64(ctx, fp0, fs);
5941 gen_store_fpr64(ctx, fp0, fd);
5942 tcg_temp_free_i64(fp0);
5943 gen_set_label(l1);
5944 }
5945
gen_movcf_ps(int fs,int fd,int cc,int tf)5946 static inline void gen_movcf_ps (int fs, int fd, int cc, int tf)
5947 {
5948 int cond;
5949 TCGv_i32 t0 = tcg_temp_new_i32();
5950 int l1 = gen_new_label();
5951 int l2 = gen_new_label();
5952
5953 if (tf)
5954 cond = TCG_COND_EQ;
5955 else
5956 cond = TCG_COND_NE;
5957
5958 tcg_gen_andi_i32(t0, fpu_fcr31, 1 << get_fp_bit(cc));
5959 tcg_gen_brcondi_i32(cond, t0, 0, l1);
5960 gen_load_fpr32(t0, fs);
5961 gen_store_fpr32(t0, fd);
5962 gen_set_label(l1);
5963
5964 tcg_gen_andi_i32(t0, fpu_fcr31, 1 << get_fp_bit(cc+1));
5965 tcg_gen_brcondi_i32(cond, t0, 0, l2);
5966 gen_load_fpr32h(t0, fs);
5967 gen_store_fpr32h(t0, fd);
5968 tcg_temp_free_i32(t0);
5969 gen_set_label(l2);
5970 }
5971
5972
gen_farith(DisasContext * ctx,uint32_t op1,int ft,int fs,int fd,int cc)5973 static void gen_farith (DisasContext *ctx, uint32_t op1,
5974 int ft, int fs, int fd, int cc)
5975 {
5976 const char *opn = "farith";
5977 const char *condnames[] = {
5978 "c.f",
5979 "c.un",
5980 "c.eq",
5981 "c.ueq",
5982 "c.olt",
5983 "c.ult",
5984 "c.ole",
5985 "c.ule",
5986 "c.sf",
5987 "c.ngle",
5988 "c.seq",
5989 "c.ngl",
5990 "c.lt",
5991 "c.nge",
5992 "c.le",
5993 "c.ngt",
5994 };
5995 const char *condnames_abs[] = {
5996 "cabs.f",
5997 "cabs.un",
5998 "cabs.eq",
5999 "cabs.ueq",
6000 "cabs.olt",
6001 "cabs.ult",
6002 "cabs.ole",
6003 "cabs.ule",
6004 "cabs.sf",
6005 "cabs.ngle",
6006 "cabs.seq",
6007 "cabs.ngl",
6008 "cabs.lt",
6009 "cabs.nge",
6010 "cabs.le",
6011 "cabs.ngt",
6012 };
6013 enum { BINOP, CMPOP, OTHEROP } optype = OTHEROP;
6014 uint32_t func = ctx->opcode & 0x3f;
6015
6016 switch (ctx->opcode & FOP(0x3f, 0x1f)) {
6017 case FOP(0, 16):
6018 {
6019 TCGv_i32 fp0 = tcg_temp_new_i32();
6020 TCGv_i32 fp1 = tcg_temp_new_i32();
6021
6022 gen_load_fpr32(fp0, fs);
6023 gen_load_fpr32(fp1, ft);
6024 gen_helper_float_add_s(fp0, fp0, fp1);
6025 tcg_temp_free_i32(fp1);
6026 gen_store_fpr32(fp0, fd);
6027 tcg_temp_free_i32(fp0);
6028 }
6029 opn = "add.s";
6030 optype = BINOP;
6031 break;
6032 case FOP(1, 16):
6033 {
6034 TCGv_i32 fp0 = tcg_temp_new_i32();
6035 TCGv_i32 fp1 = tcg_temp_new_i32();
6036
6037 gen_load_fpr32(fp0, fs);
6038 gen_load_fpr32(fp1, ft);
6039 gen_helper_float_sub_s(fp0, fp0, fp1);
6040 tcg_temp_free_i32(fp1);
6041 gen_store_fpr32(fp0, fd);
6042 tcg_temp_free_i32(fp0);
6043 }
6044 opn = "sub.s";
6045 optype = BINOP;
6046 break;
6047 case FOP(2, 16):
6048 {
6049 TCGv_i32 fp0 = tcg_temp_new_i32();
6050 TCGv_i32 fp1 = tcg_temp_new_i32();
6051
6052 gen_load_fpr32(fp0, fs);
6053 gen_load_fpr32(fp1, ft);
6054 gen_helper_float_mul_s(fp0, fp0, fp1);
6055 tcg_temp_free_i32(fp1);
6056 gen_store_fpr32(fp0, fd);
6057 tcg_temp_free_i32(fp0);
6058 }
6059 opn = "mul.s";
6060 optype = BINOP;
6061 break;
6062 case FOP(3, 16):
6063 {
6064 TCGv_i32 fp0 = tcg_temp_new_i32();
6065 TCGv_i32 fp1 = tcg_temp_new_i32();
6066
6067 gen_load_fpr32(fp0, fs);
6068 gen_load_fpr32(fp1, ft);
6069 gen_helper_float_div_s(fp0, fp0, fp1);
6070 tcg_temp_free_i32(fp1);
6071 gen_store_fpr32(fp0, fd);
6072 tcg_temp_free_i32(fp0);
6073 }
6074 opn = "div.s";
6075 optype = BINOP;
6076 break;
6077 case FOP(4, 16):
6078 {
6079 TCGv_i32 fp0 = tcg_temp_new_i32();
6080
6081 gen_load_fpr32(fp0, fs);
6082 gen_helper_float_sqrt_s(fp0, fp0);
6083 gen_store_fpr32(fp0, fd);
6084 tcg_temp_free_i32(fp0);
6085 }
6086 opn = "sqrt.s";
6087 break;
6088 case FOP(5, 16):
6089 {
6090 TCGv_i32 fp0 = tcg_temp_new_i32();
6091
6092 gen_load_fpr32(fp0, fs);
6093 gen_helper_float_abs_s(fp0, fp0);
6094 gen_store_fpr32(fp0, fd);
6095 tcg_temp_free_i32(fp0);
6096 }
6097 opn = "abs.s";
6098 break;
6099 case FOP(6, 16):
6100 {
6101 TCGv_i32 fp0 = tcg_temp_new_i32();
6102
6103 gen_load_fpr32(fp0, fs);
6104 gen_store_fpr32(fp0, fd);
6105 tcg_temp_free_i32(fp0);
6106 }
6107 opn = "mov.s";
6108 break;
6109 case FOP(7, 16):
6110 {
6111 TCGv_i32 fp0 = tcg_temp_new_i32();
6112
6113 gen_load_fpr32(fp0, fs);
6114 gen_helper_float_chs_s(fp0, fp0);
6115 gen_store_fpr32(fp0, fd);
6116 tcg_temp_free_i32(fp0);
6117 }
6118 opn = "neg.s";
6119 break;
6120 case FOP(8, 16):
6121 check_cp1_64bitmode(ctx);
6122 {
6123 TCGv_i32 fp32 = tcg_temp_new_i32();
6124 TCGv_i64 fp64 = tcg_temp_new_i64();
6125
6126 gen_load_fpr32(fp32, fs);
6127 gen_helper_float_roundl_s(fp64, fp32);
6128 tcg_temp_free_i32(fp32);
6129 gen_store_fpr64(ctx, fp64, fd);
6130 tcg_temp_free_i64(fp64);
6131 }
6132 opn = "round.l.s";
6133 break;
6134 case FOP(9, 16):
6135 check_cp1_64bitmode(ctx);
6136 {
6137 TCGv_i32 fp32 = tcg_temp_new_i32();
6138 TCGv_i64 fp64 = tcg_temp_new_i64();
6139
6140 gen_load_fpr32(fp32, fs);
6141 gen_helper_float_truncl_s(fp64, fp32);
6142 tcg_temp_free_i32(fp32);
6143 gen_store_fpr64(ctx, fp64, fd);
6144 tcg_temp_free_i64(fp64);
6145 }
6146 opn = "trunc.l.s";
6147 break;
6148 case FOP(10, 16):
6149 check_cp1_64bitmode(ctx);
6150 {
6151 TCGv_i32 fp32 = tcg_temp_new_i32();
6152 TCGv_i64 fp64 = tcg_temp_new_i64();
6153
6154 gen_load_fpr32(fp32, fs);
6155 gen_helper_float_ceill_s(fp64, fp32);
6156 tcg_temp_free_i32(fp32);
6157 gen_store_fpr64(ctx, fp64, fd);
6158 tcg_temp_free_i64(fp64);
6159 }
6160 opn = "ceil.l.s";
6161 break;
6162 case FOP(11, 16):
6163 check_cp1_64bitmode(ctx);
6164 {
6165 TCGv_i32 fp32 = tcg_temp_new_i32();
6166 TCGv_i64 fp64 = tcg_temp_new_i64();
6167
6168 gen_load_fpr32(fp32, fs);
6169 gen_helper_float_floorl_s(fp64, fp32);
6170 tcg_temp_free_i32(fp32);
6171 gen_store_fpr64(ctx, fp64, fd);
6172 tcg_temp_free_i64(fp64);
6173 }
6174 opn = "floor.l.s";
6175 break;
6176 case FOP(12, 16):
6177 {
6178 TCGv_i32 fp0 = tcg_temp_new_i32();
6179
6180 gen_load_fpr32(fp0, fs);
6181 gen_helper_float_roundw_s(fp0, fp0);
6182 gen_store_fpr32(fp0, fd);
6183 tcg_temp_free_i32(fp0);
6184 }
6185 opn = "round.w.s";
6186 break;
6187 case FOP(13, 16):
6188 {
6189 TCGv_i32 fp0 = tcg_temp_new_i32();
6190
6191 gen_load_fpr32(fp0, fs);
6192 gen_helper_float_truncw_s(fp0, fp0);
6193 gen_store_fpr32(fp0, fd);
6194 tcg_temp_free_i32(fp0);
6195 }
6196 opn = "trunc.w.s";
6197 break;
6198 case FOP(14, 16):
6199 {
6200 TCGv_i32 fp0 = tcg_temp_new_i32();
6201
6202 gen_load_fpr32(fp0, fs);
6203 gen_helper_float_ceilw_s(fp0, fp0);
6204 gen_store_fpr32(fp0, fd);
6205 tcg_temp_free_i32(fp0);
6206 }
6207 opn = "ceil.w.s";
6208 break;
6209 case FOP(15, 16):
6210 {
6211 TCGv_i32 fp0 = tcg_temp_new_i32();
6212
6213 gen_load_fpr32(fp0, fs);
6214 gen_helper_float_floorw_s(fp0, fp0);
6215 gen_store_fpr32(fp0, fd);
6216 tcg_temp_free_i32(fp0);
6217 }
6218 opn = "floor.w.s";
6219 break;
6220 case FOP(17, 16):
6221 gen_movcf_s(fs, fd, (ft >> 2) & 0x7, ft & 0x1);
6222 opn = "movcf.s";
6223 break;
6224 case FOP(18, 16):
6225 {
6226 int l1 = gen_new_label();
6227 TCGv_i32 fp0;
6228
6229 if (ft != 0) {
6230 tcg_gen_brcondi_tl(TCG_COND_NE, cpu_gpr[ft], 0, l1);
6231 }
6232 fp0 = tcg_temp_new_i32();
6233 gen_load_fpr32(fp0, fs);
6234 gen_store_fpr32(fp0, fd);
6235 tcg_temp_free_i32(fp0);
6236 gen_set_label(l1);
6237 }
6238 opn = "movz.s";
6239 break;
6240 case FOP(19, 16):
6241 {
6242 int l1 = gen_new_label();
6243 TCGv_i32 fp0;
6244
6245 if (ft != 0) {
6246 tcg_gen_brcondi_tl(TCG_COND_EQ, cpu_gpr[ft], 0, l1);
6247 fp0 = tcg_temp_new_i32();
6248 gen_load_fpr32(fp0, fs);
6249 gen_store_fpr32(fp0, fd);
6250 tcg_temp_free_i32(fp0);
6251 gen_set_label(l1);
6252 }
6253 }
6254 opn = "movn.s";
6255 break;
6256 case FOP(21, 16):
6257 check_cop1x(ctx);
6258 {
6259 TCGv_i32 fp0 = tcg_temp_new_i32();
6260
6261 gen_load_fpr32(fp0, fs);
6262 gen_helper_float_recip_s(fp0, fp0);
6263 gen_store_fpr32(fp0, fd);
6264 tcg_temp_free_i32(fp0);
6265 }
6266 opn = "recip.s";
6267 break;
6268 case FOP(22, 16):
6269 check_cop1x(ctx);
6270 {
6271 TCGv_i32 fp0 = tcg_temp_new_i32();
6272
6273 gen_load_fpr32(fp0, fs);
6274 gen_helper_float_rsqrt_s(fp0, fp0);
6275 gen_store_fpr32(fp0, fd);
6276 tcg_temp_free_i32(fp0);
6277 }
6278 opn = "rsqrt.s";
6279 break;
6280 case FOP(28, 16):
6281 check_cp1_64bitmode(ctx);
6282 {
6283 TCGv_i32 fp0 = tcg_temp_new_i32();
6284 TCGv_i32 fp1 = tcg_temp_new_i32();
6285
6286 gen_load_fpr32(fp0, fs);
6287 gen_load_fpr32(fp1, fd);
6288 gen_helper_float_recip2_s(fp0, fp0, fp1);
6289 tcg_temp_free_i32(fp1);
6290 gen_store_fpr32(fp0, fd);
6291 tcg_temp_free_i32(fp0);
6292 }
6293 opn = "recip2.s";
6294 break;
6295 case FOP(29, 16):
6296 check_cp1_64bitmode(ctx);
6297 {
6298 TCGv_i32 fp0 = tcg_temp_new_i32();
6299
6300 gen_load_fpr32(fp0, fs);
6301 gen_helper_float_recip1_s(fp0, fp0);
6302 gen_store_fpr32(fp0, fd);
6303 tcg_temp_free_i32(fp0);
6304 }
6305 opn = "recip1.s";
6306 break;
6307 case FOP(30, 16):
6308 check_cp1_64bitmode(ctx);
6309 {
6310 TCGv_i32 fp0 = tcg_temp_new_i32();
6311
6312 gen_load_fpr32(fp0, fs);
6313 gen_helper_float_rsqrt1_s(fp0, fp0);
6314 gen_store_fpr32(fp0, fd);
6315 tcg_temp_free_i32(fp0);
6316 }
6317 opn = "rsqrt1.s";
6318 break;
6319 case FOP(31, 16):
6320 check_cp1_64bitmode(ctx);
6321 {
6322 TCGv_i32 fp0 = tcg_temp_new_i32();
6323 TCGv_i32 fp1 = tcg_temp_new_i32();
6324
6325 gen_load_fpr32(fp0, fs);
6326 gen_load_fpr32(fp1, ft);
6327 gen_helper_float_rsqrt2_s(fp0, fp0, fp1);
6328 tcg_temp_free_i32(fp1);
6329 gen_store_fpr32(fp0, fd);
6330 tcg_temp_free_i32(fp0);
6331 }
6332 opn = "rsqrt2.s";
6333 break;
6334 case FOP(33, 16):
6335 check_cp1_registers(ctx, fd);
6336 {
6337 TCGv_i32 fp32 = tcg_temp_new_i32();
6338 TCGv_i64 fp64 = tcg_temp_new_i64();
6339
6340 gen_load_fpr32(fp32, fs);
6341 gen_helper_float_cvtd_s(fp64, fp32);
6342 tcg_temp_free_i32(fp32);
6343 gen_store_fpr64(ctx, fp64, fd);
6344 tcg_temp_free_i64(fp64);
6345 }
6346 opn = "cvt.d.s";
6347 break;
6348 case FOP(36, 16):
6349 {
6350 TCGv_i32 fp0 = tcg_temp_new_i32();
6351
6352 gen_load_fpr32(fp0, fs);
6353 gen_helper_float_cvtw_s(fp0, fp0);
6354 gen_store_fpr32(fp0, fd);
6355 tcg_temp_free_i32(fp0);
6356 }
6357 opn = "cvt.w.s";
6358 break;
6359 case FOP(37, 16):
6360 check_cp1_64bitmode(ctx);
6361 {
6362 TCGv_i32 fp32 = tcg_temp_new_i32();
6363 TCGv_i64 fp64 = tcg_temp_new_i64();
6364
6365 gen_load_fpr32(fp32, fs);
6366 gen_helper_float_cvtl_s(fp64, fp32);
6367 tcg_temp_free_i32(fp32);
6368 gen_store_fpr64(ctx, fp64, fd);
6369 tcg_temp_free_i64(fp64);
6370 }
6371 opn = "cvt.l.s";
6372 break;
6373 case FOP(38, 16):
6374 check_cp1_64bitmode(ctx);
6375 {
6376 TCGv_i64 fp64 = tcg_temp_new_i64();
6377 TCGv_i32 fp32_0 = tcg_temp_new_i32();
6378 TCGv_i32 fp32_1 = tcg_temp_new_i32();
6379
6380 gen_load_fpr32(fp32_0, fs);
6381 gen_load_fpr32(fp32_1, ft);
6382 tcg_gen_concat_i32_i64(fp64, fp32_0, fp32_1);
6383 tcg_temp_free_i32(fp32_1);
6384 tcg_temp_free_i32(fp32_0);
6385 gen_store_fpr64(ctx, fp64, fd);
6386 tcg_temp_free_i64(fp64);
6387 }
6388 opn = "cvt.ps.s";
6389 break;
6390 case FOP(48, 16):
6391 case FOP(49, 16):
6392 case FOP(50, 16):
6393 case FOP(51, 16):
6394 case FOP(52, 16):
6395 case FOP(53, 16):
6396 case FOP(54, 16):
6397 case FOP(55, 16):
6398 case FOP(56, 16):
6399 case FOP(57, 16):
6400 case FOP(58, 16):
6401 case FOP(59, 16):
6402 case FOP(60, 16):
6403 case FOP(61, 16):
6404 case FOP(62, 16):
6405 case FOP(63, 16):
6406 {
6407 TCGv_i32 fp0 = tcg_temp_new_i32();
6408 TCGv_i32 fp1 = tcg_temp_new_i32();
6409
6410 gen_load_fpr32(fp0, fs);
6411 gen_load_fpr32(fp1, ft);
6412 if (ctx->opcode & (1 << 6)) {
6413 check_cop1x(ctx);
6414 gen_cmpabs_s(func-48, fp0, fp1, cc);
6415 opn = condnames_abs[func-48];
6416 } else {
6417 gen_cmp_s(func-48, fp0, fp1, cc);
6418 opn = condnames[func-48];
6419 }
6420 tcg_temp_free_i32(fp0);
6421 tcg_temp_free_i32(fp1);
6422 }
6423 break;
6424 case FOP(0, 17):
6425 check_cp1_registers(ctx, fs | ft | fd);
6426 {
6427 TCGv_i64 fp0 = tcg_temp_new_i64();
6428 TCGv_i64 fp1 = tcg_temp_new_i64();
6429
6430 gen_load_fpr64(ctx, fp0, fs);
6431 gen_load_fpr64(ctx, fp1, ft);
6432 gen_helper_float_add_d(fp0, fp0, fp1);
6433 tcg_temp_free_i64(fp1);
6434 gen_store_fpr64(ctx, fp0, fd);
6435 tcg_temp_free_i64(fp0);
6436 }
6437 opn = "add.d";
6438 optype = BINOP;
6439 break;
6440 case FOP(1, 17):
6441 check_cp1_registers(ctx, fs | ft | fd);
6442 {
6443 TCGv_i64 fp0 = tcg_temp_new_i64();
6444 TCGv_i64 fp1 = tcg_temp_new_i64();
6445
6446 gen_load_fpr64(ctx, fp0, fs);
6447 gen_load_fpr64(ctx, fp1, ft);
6448 gen_helper_float_sub_d(fp0, fp0, fp1);
6449 tcg_temp_free_i64(fp1);
6450 gen_store_fpr64(ctx, fp0, fd);
6451 tcg_temp_free_i64(fp0);
6452 }
6453 opn = "sub.d";
6454 optype = BINOP;
6455 break;
6456 case FOP(2, 17):
6457 check_cp1_registers(ctx, fs | ft | fd);
6458 {
6459 TCGv_i64 fp0 = tcg_temp_new_i64();
6460 TCGv_i64 fp1 = tcg_temp_new_i64();
6461
6462 gen_load_fpr64(ctx, fp0, fs);
6463 gen_load_fpr64(ctx, fp1, ft);
6464 gen_helper_float_mul_d(fp0, fp0, fp1);
6465 tcg_temp_free_i64(fp1);
6466 gen_store_fpr64(ctx, fp0, fd);
6467 tcg_temp_free_i64(fp0);
6468 }
6469 opn = "mul.d";
6470 optype = BINOP;
6471 break;
6472 case FOP(3, 17):
6473 check_cp1_registers(ctx, fs | ft | fd);
6474 {
6475 TCGv_i64 fp0 = tcg_temp_new_i64();
6476 TCGv_i64 fp1 = tcg_temp_new_i64();
6477
6478 gen_load_fpr64(ctx, fp0, fs);
6479 gen_load_fpr64(ctx, fp1, ft);
6480 gen_helper_float_div_d(fp0, fp0, fp1);
6481 tcg_temp_free_i64(fp1);
6482 gen_store_fpr64(ctx, fp0, fd);
6483 tcg_temp_free_i64(fp0);
6484 }
6485 opn = "div.d";
6486 optype = BINOP;
6487 break;
6488 case FOP(4, 17):
6489 check_cp1_registers(ctx, fs | fd);
6490 {
6491 TCGv_i64 fp0 = tcg_temp_new_i64();
6492
6493 gen_load_fpr64(ctx, fp0, fs);
6494 gen_helper_float_sqrt_d(fp0, fp0);
6495 gen_store_fpr64(ctx, fp0, fd);
6496 tcg_temp_free_i64(fp0);
6497 }
6498 opn = "sqrt.d";
6499 break;
6500 case FOP(5, 17):
6501 check_cp1_registers(ctx, fs | fd);
6502 {
6503 TCGv_i64 fp0 = tcg_temp_new_i64();
6504
6505 gen_load_fpr64(ctx, fp0, fs);
6506 gen_helper_float_abs_d(fp0, fp0);
6507 gen_store_fpr64(ctx, fp0, fd);
6508 tcg_temp_free_i64(fp0);
6509 }
6510 opn = "abs.d";
6511 break;
6512 case FOP(6, 17):
6513 check_cp1_registers(ctx, fs | fd);
6514 {
6515 TCGv_i64 fp0 = tcg_temp_new_i64();
6516
6517 gen_load_fpr64(ctx, fp0, fs);
6518 gen_store_fpr64(ctx, fp0, fd);
6519 tcg_temp_free_i64(fp0);
6520 }
6521 opn = "mov.d";
6522 break;
6523 case FOP(7, 17):
6524 check_cp1_registers(ctx, fs | fd);
6525 {
6526 TCGv_i64 fp0 = tcg_temp_new_i64();
6527
6528 gen_load_fpr64(ctx, fp0, fs);
6529 gen_helper_float_chs_d(fp0, fp0);
6530 gen_store_fpr64(ctx, fp0, fd);
6531 tcg_temp_free_i64(fp0);
6532 }
6533 opn = "neg.d";
6534 break;
6535 case FOP(8, 17):
6536 check_cp1_64bitmode(ctx);
6537 {
6538 TCGv_i64 fp0 = tcg_temp_new_i64();
6539
6540 gen_load_fpr64(ctx, fp0, fs);
6541 gen_helper_float_roundl_d(fp0, fp0);
6542 gen_store_fpr64(ctx, fp0, fd);
6543 tcg_temp_free_i64(fp0);
6544 }
6545 opn = "round.l.d";
6546 break;
6547 case FOP(9, 17):
6548 check_cp1_64bitmode(ctx);
6549 {
6550 TCGv_i64 fp0 = tcg_temp_new_i64();
6551
6552 gen_load_fpr64(ctx, fp0, fs);
6553 gen_helper_float_truncl_d(fp0, fp0);
6554 gen_store_fpr64(ctx, fp0, fd);
6555 tcg_temp_free_i64(fp0);
6556 }
6557 opn = "trunc.l.d";
6558 break;
6559 case FOP(10, 17):
6560 check_cp1_64bitmode(ctx);
6561 {
6562 TCGv_i64 fp0 = tcg_temp_new_i64();
6563
6564 gen_load_fpr64(ctx, fp0, fs);
6565 gen_helper_float_ceill_d(fp0, fp0);
6566 gen_store_fpr64(ctx, fp0, fd);
6567 tcg_temp_free_i64(fp0);
6568 }
6569 opn = "ceil.l.d";
6570 break;
6571 case FOP(11, 17):
6572 check_cp1_64bitmode(ctx);
6573 {
6574 TCGv_i64 fp0 = tcg_temp_new_i64();
6575
6576 gen_load_fpr64(ctx, fp0, fs);
6577 gen_helper_float_floorl_d(fp0, fp0);
6578 gen_store_fpr64(ctx, fp0, fd);
6579 tcg_temp_free_i64(fp0);
6580 }
6581 opn = "floor.l.d";
6582 break;
6583 case FOP(12, 17):
6584 check_cp1_registers(ctx, fs);
6585 {
6586 TCGv_i32 fp32 = tcg_temp_new_i32();
6587 TCGv_i64 fp64 = tcg_temp_new_i64();
6588
6589 gen_load_fpr64(ctx, fp64, fs);
6590 gen_helper_float_roundw_d(fp32, fp64);
6591 tcg_temp_free_i64(fp64);
6592 gen_store_fpr32(fp32, fd);
6593 tcg_temp_free_i32(fp32);
6594 }
6595 opn = "round.w.d";
6596 break;
6597 case FOP(13, 17):
6598 check_cp1_registers(ctx, fs);
6599 {
6600 TCGv_i32 fp32 = tcg_temp_new_i32();
6601 TCGv_i64 fp64 = tcg_temp_new_i64();
6602
6603 gen_load_fpr64(ctx, fp64, fs);
6604 gen_helper_float_truncw_d(fp32, fp64);
6605 tcg_temp_free_i64(fp64);
6606 gen_store_fpr32(fp32, fd);
6607 tcg_temp_free_i32(fp32);
6608 }
6609 opn = "trunc.w.d";
6610 break;
6611 case FOP(14, 17):
6612 check_cp1_registers(ctx, fs);
6613 {
6614 TCGv_i32 fp32 = tcg_temp_new_i32();
6615 TCGv_i64 fp64 = tcg_temp_new_i64();
6616
6617 gen_load_fpr64(ctx, fp64, fs);
6618 gen_helper_float_ceilw_d(fp32, fp64);
6619 tcg_temp_free_i64(fp64);
6620 gen_store_fpr32(fp32, fd);
6621 tcg_temp_free_i32(fp32);
6622 }
6623 opn = "ceil.w.d";
6624 break;
6625 case FOP(15, 17):
6626 check_cp1_registers(ctx, fs);
6627 {
6628 TCGv_i32 fp32 = tcg_temp_new_i32();
6629 TCGv_i64 fp64 = tcg_temp_new_i64();
6630
6631 gen_load_fpr64(ctx, fp64, fs);
6632 gen_helper_float_floorw_d(fp32, fp64);
6633 tcg_temp_free_i64(fp64);
6634 gen_store_fpr32(fp32, fd);
6635 tcg_temp_free_i32(fp32);
6636 }
6637 opn = "floor.w.d";
6638 break;
6639 case FOP(17, 17):
6640 gen_movcf_d(ctx, fs, fd, (ft >> 2) & 0x7, ft & 0x1);
6641 opn = "movcf.d";
6642 break;
6643 case FOP(18, 17):
6644 {
6645 int l1 = gen_new_label();
6646 TCGv_i64 fp0;
6647
6648 if (ft != 0) {
6649 tcg_gen_brcondi_tl(TCG_COND_NE, cpu_gpr[ft], 0, l1);
6650 }
6651 fp0 = tcg_temp_new_i64();
6652 gen_load_fpr64(ctx, fp0, fs);
6653 gen_store_fpr64(ctx, fp0, fd);
6654 tcg_temp_free_i64(fp0);
6655 gen_set_label(l1);
6656 }
6657 opn = "movz.d";
6658 break;
6659 case FOP(19, 17):
6660 {
6661 int l1 = gen_new_label();
6662 TCGv_i64 fp0;
6663
6664 if (ft != 0) {
6665 tcg_gen_brcondi_tl(TCG_COND_EQ, cpu_gpr[ft], 0, l1);
6666 fp0 = tcg_temp_new_i64();
6667 gen_load_fpr64(ctx, fp0, fs);
6668 gen_store_fpr64(ctx, fp0, fd);
6669 tcg_temp_free_i64(fp0);
6670 gen_set_label(l1);
6671 }
6672 }
6673 opn = "movn.d";
6674 break;
6675 case FOP(21, 17):
6676 check_cp1_64bitmode(ctx);
6677 {
6678 TCGv_i64 fp0 = tcg_temp_new_i64();
6679
6680 gen_load_fpr64(ctx, fp0, fs);
6681 gen_helper_float_recip_d(fp0, fp0);
6682 gen_store_fpr64(ctx, fp0, fd);
6683 tcg_temp_free_i64(fp0);
6684 }
6685 opn = "recip.d";
6686 break;
6687 case FOP(22, 17):
6688 check_cp1_64bitmode(ctx);
6689 {
6690 TCGv_i64 fp0 = tcg_temp_new_i64();
6691
6692 gen_load_fpr64(ctx, fp0, fs);
6693 gen_helper_float_rsqrt_d(fp0, fp0);
6694 gen_store_fpr64(ctx, fp0, fd);
6695 tcg_temp_free_i64(fp0);
6696 }
6697 opn = "rsqrt.d";
6698 break;
6699 case FOP(28, 17):
6700 check_cp1_64bitmode(ctx);
6701 {
6702 TCGv_i64 fp0 = tcg_temp_new_i64();
6703 TCGv_i64 fp1 = tcg_temp_new_i64();
6704
6705 gen_load_fpr64(ctx, fp0, fs);
6706 gen_load_fpr64(ctx, fp1, ft);
6707 gen_helper_float_recip2_d(fp0, fp0, fp1);
6708 tcg_temp_free_i64(fp1);
6709 gen_store_fpr64(ctx, fp0, fd);
6710 tcg_temp_free_i64(fp0);
6711 }
6712 opn = "recip2.d";
6713 break;
6714 case FOP(29, 17):
6715 check_cp1_64bitmode(ctx);
6716 {
6717 TCGv_i64 fp0 = tcg_temp_new_i64();
6718
6719 gen_load_fpr64(ctx, fp0, fs);
6720 gen_helper_float_recip1_d(fp0, fp0);
6721 gen_store_fpr64(ctx, fp0, fd);
6722 tcg_temp_free_i64(fp0);
6723 }
6724 opn = "recip1.d";
6725 break;
6726 case FOP(30, 17):
6727 check_cp1_64bitmode(ctx);
6728 {
6729 TCGv_i64 fp0 = tcg_temp_new_i64();
6730
6731 gen_load_fpr64(ctx, fp0, fs);
6732 gen_helper_float_rsqrt1_d(fp0, fp0);
6733 gen_store_fpr64(ctx, fp0, fd);
6734 tcg_temp_free_i64(fp0);
6735 }
6736 opn = "rsqrt1.d";
6737 break;
6738 case FOP(31, 17):
6739 check_cp1_64bitmode(ctx);
6740 {
6741 TCGv_i64 fp0 = tcg_temp_new_i64();
6742 TCGv_i64 fp1 = tcg_temp_new_i64();
6743
6744 gen_load_fpr64(ctx, fp0, fs);
6745 gen_load_fpr64(ctx, fp1, ft);
6746 gen_helper_float_rsqrt2_d(fp0, fp0, fp1);
6747 tcg_temp_free_i64(fp1);
6748 gen_store_fpr64(ctx, fp0, fd);
6749 tcg_temp_free_i64(fp0);
6750 }
6751 opn = "rsqrt2.d";
6752 break;
6753 case FOP(48, 17):
6754 case FOP(49, 17):
6755 case FOP(50, 17):
6756 case FOP(51, 17):
6757 case FOP(52, 17):
6758 case FOP(53, 17):
6759 case FOP(54, 17):
6760 case FOP(55, 17):
6761 case FOP(56, 17):
6762 case FOP(57, 17):
6763 case FOP(58, 17):
6764 case FOP(59, 17):
6765 case FOP(60, 17):
6766 case FOP(61, 17):
6767 case FOP(62, 17):
6768 case FOP(63, 17):
6769 {
6770 TCGv_i64 fp0 = tcg_temp_new_i64();
6771 TCGv_i64 fp1 = tcg_temp_new_i64();
6772
6773 gen_load_fpr64(ctx, fp0, fs);
6774 gen_load_fpr64(ctx, fp1, ft);
6775 if (ctx->opcode & (1 << 6)) {
6776 check_cop1x(ctx);
6777 check_cp1_registers(ctx, fs | ft);
6778 gen_cmpabs_d(func-48, fp0, fp1, cc);
6779 opn = condnames_abs[func-48];
6780 } else {
6781 check_cp1_registers(ctx, fs | ft);
6782 gen_cmp_d(func-48, fp0, fp1, cc);
6783 opn = condnames[func-48];
6784 }
6785 tcg_temp_free_i64(fp0);
6786 tcg_temp_free_i64(fp1);
6787 }
6788 break;
6789 case FOP(32, 17):
6790 check_cp1_registers(ctx, fs);
6791 {
6792 TCGv_i32 fp32 = tcg_temp_new_i32();
6793 TCGv_i64 fp64 = tcg_temp_new_i64();
6794
6795 gen_load_fpr64(ctx, fp64, fs);
6796 gen_helper_float_cvts_d(fp32, fp64);
6797 tcg_temp_free_i64(fp64);
6798 gen_store_fpr32(fp32, fd);
6799 tcg_temp_free_i32(fp32);
6800 }
6801 opn = "cvt.s.d";
6802 break;
6803 case FOP(36, 17):
6804 check_cp1_registers(ctx, fs);
6805 {
6806 TCGv_i32 fp32 = tcg_temp_new_i32();
6807 TCGv_i64 fp64 = tcg_temp_new_i64();
6808
6809 gen_load_fpr64(ctx, fp64, fs);
6810 gen_helper_float_cvtw_d(fp32, fp64);
6811 tcg_temp_free_i64(fp64);
6812 gen_store_fpr32(fp32, fd);
6813 tcg_temp_free_i32(fp32);
6814 }
6815 opn = "cvt.w.d";
6816 break;
6817 case FOP(37, 17):
6818 check_cp1_64bitmode(ctx);
6819 {
6820 TCGv_i64 fp0 = tcg_temp_new_i64();
6821
6822 gen_load_fpr64(ctx, fp0, fs);
6823 gen_helper_float_cvtl_d(fp0, fp0);
6824 gen_store_fpr64(ctx, fp0, fd);
6825 tcg_temp_free_i64(fp0);
6826 }
6827 opn = "cvt.l.d";
6828 break;
6829 case FOP(32, 20):
6830 {
6831 TCGv_i32 fp0 = tcg_temp_new_i32();
6832
6833 gen_load_fpr32(fp0, fs);
6834 gen_helper_float_cvts_w(fp0, fp0);
6835 gen_store_fpr32(fp0, fd);
6836 tcg_temp_free_i32(fp0);
6837 }
6838 opn = "cvt.s.w";
6839 break;
6840 case FOP(33, 20):
6841 check_cp1_registers(ctx, fd);
6842 {
6843 TCGv_i32 fp32 = tcg_temp_new_i32();
6844 TCGv_i64 fp64 = tcg_temp_new_i64();
6845
6846 gen_load_fpr32(fp32, fs);
6847 gen_helper_float_cvtd_w(fp64, fp32);
6848 tcg_temp_free_i32(fp32);
6849 gen_store_fpr64(ctx, fp64, fd);
6850 tcg_temp_free_i64(fp64);
6851 }
6852 opn = "cvt.d.w";
6853 break;
6854 case FOP(32, 21):
6855 check_cp1_64bitmode(ctx);
6856 {
6857 TCGv_i32 fp32 = tcg_temp_new_i32();
6858 TCGv_i64 fp64 = tcg_temp_new_i64();
6859
6860 gen_load_fpr64(ctx, fp64, fs);
6861 gen_helper_float_cvts_l(fp32, fp64);
6862 tcg_temp_free_i64(fp64);
6863 gen_store_fpr32(fp32, fd);
6864 tcg_temp_free_i32(fp32);
6865 }
6866 opn = "cvt.s.l";
6867 break;
6868 case FOP(33, 21):
6869 check_cp1_64bitmode(ctx);
6870 {
6871 TCGv_i64 fp0 = tcg_temp_new_i64();
6872
6873 gen_load_fpr64(ctx, fp0, fs);
6874 gen_helper_float_cvtd_l(fp0, fp0);
6875 gen_store_fpr64(ctx, fp0, fd);
6876 tcg_temp_free_i64(fp0);
6877 }
6878 opn = "cvt.d.l";
6879 break;
6880 case FOP(38, 20):
6881 check_cp1_64bitmode(ctx);
6882 {
6883 TCGv_i64 fp0 = tcg_temp_new_i64();
6884
6885 gen_load_fpr64(ctx, fp0, fs);
6886 gen_helper_float_cvtps_pw(fp0, fp0);
6887 gen_store_fpr64(ctx, fp0, fd);
6888 tcg_temp_free_i64(fp0);
6889 }
6890 opn = "cvt.ps.pw";
6891 break;
6892 case FOP(0, 22):
6893 check_cp1_64bitmode(ctx);
6894 {
6895 TCGv_i64 fp0 = tcg_temp_new_i64();
6896 TCGv_i64 fp1 = tcg_temp_new_i64();
6897
6898 gen_load_fpr64(ctx, fp0, fs);
6899 gen_load_fpr64(ctx, fp1, ft);
6900 gen_helper_float_add_ps(fp0, fp0, fp1);
6901 tcg_temp_free_i64(fp1);
6902 gen_store_fpr64(ctx, fp0, fd);
6903 tcg_temp_free_i64(fp0);
6904 }
6905 opn = "add.ps";
6906 break;
6907 case FOP(1, 22):
6908 check_cp1_64bitmode(ctx);
6909 {
6910 TCGv_i64 fp0 = tcg_temp_new_i64();
6911 TCGv_i64 fp1 = tcg_temp_new_i64();
6912
6913 gen_load_fpr64(ctx, fp0, fs);
6914 gen_load_fpr64(ctx, fp1, ft);
6915 gen_helper_float_sub_ps(fp0, fp0, fp1);
6916 tcg_temp_free_i64(fp1);
6917 gen_store_fpr64(ctx, fp0, fd);
6918 tcg_temp_free_i64(fp0);
6919 }
6920 opn = "sub.ps";
6921 break;
6922 case FOP(2, 22):
6923 check_cp1_64bitmode(ctx);
6924 {
6925 TCGv_i64 fp0 = tcg_temp_new_i64();
6926 TCGv_i64 fp1 = tcg_temp_new_i64();
6927
6928 gen_load_fpr64(ctx, fp0, fs);
6929 gen_load_fpr64(ctx, fp1, ft);
6930 gen_helper_float_mul_ps(fp0, fp0, fp1);
6931 tcg_temp_free_i64(fp1);
6932 gen_store_fpr64(ctx, fp0, fd);
6933 tcg_temp_free_i64(fp0);
6934 }
6935 opn = "mul.ps";
6936 break;
6937 case FOP(5, 22):
6938 check_cp1_64bitmode(ctx);
6939 {
6940 TCGv_i64 fp0 = tcg_temp_new_i64();
6941
6942 gen_load_fpr64(ctx, fp0, fs);
6943 gen_helper_float_abs_ps(fp0, fp0);
6944 gen_store_fpr64(ctx, fp0, fd);
6945 tcg_temp_free_i64(fp0);
6946 }
6947 opn = "abs.ps";
6948 break;
6949 case FOP(6, 22):
6950 check_cp1_64bitmode(ctx);
6951 {
6952 TCGv_i64 fp0 = tcg_temp_new_i64();
6953
6954 gen_load_fpr64(ctx, fp0, fs);
6955 gen_store_fpr64(ctx, fp0, fd);
6956 tcg_temp_free_i64(fp0);
6957 }
6958 opn = "mov.ps";
6959 break;
6960 case FOP(7, 22):
6961 check_cp1_64bitmode(ctx);
6962 {
6963 TCGv_i64 fp0 = tcg_temp_new_i64();
6964
6965 gen_load_fpr64(ctx, fp0, fs);
6966 gen_helper_float_chs_ps(fp0, fp0);
6967 gen_store_fpr64(ctx, fp0, fd);
6968 tcg_temp_free_i64(fp0);
6969 }
6970 opn = "neg.ps";
6971 break;
6972 case FOP(17, 22):
6973 check_cp1_64bitmode(ctx);
6974 gen_movcf_ps(fs, fd, (ft >> 2) & 0x7, ft & 0x1);
6975 opn = "movcf.ps";
6976 break;
6977 case FOP(18, 22):
6978 check_cp1_64bitmode(ctx);
6979 {
6980 int l1 = gen_new_label();
6981 TCGv_i64 fp0;
6982
6983 if (ft != 0)
6984 tcg_gen_brcondi_tl(TCG_COND_NE, cpu_gpr[ft], 0, l1);
6985 fp0 = tcg_temp_new_i64();
6986 gen_load_fpr64(ctx, fp0, fs);
6987 gen_store_fpr64(ctx, fp0, fd);
6988 tcg_temp_free_i64(fp0);
6989 gen_set_label(l1);
6990 }
6991 opn = "movz.ps";
6992 break;
6993 case FOP(19, 22):
6994 check_cp1_64bitmode(ctx);
6995 {
6996 int l1 = gen_new_label();
6997 TCGv_i64 fp0;
6998
6999 if (ft != 0) {
7000 tcg_gen_brcondi_tl(TCG_COND_EQ, cpu_gpr[ft], 0, l1);
7001 fp0 = tcg_temp_new_i64();
7002 gen_load_fpr64(ctx, fp0, fs);
7003 gen_store_fpr64(ctx, fp0, fd);
7004 tcg_temp_free_i64(fp0);
7005 gen_set_label(l1);
7006 }
7007 }
7008 opn = "movn.ps";
7009 break;
7010 case FOP(24, 22):
7011 check_cp1_64bitmode(ctx);
7012 {
7013 TCGv_i64 fp0 = tcg_temp_new_i64();
7014 TCGv_i64 fp1 = tcg_temp_new_i64();
7015
7016 gen_load_fpr64(ctx, fp0, ft);
7017 gen_load_fpr64(ctx, fp1, fs);
7018 gen_helper_float_addr_ps(fp0, fp0, fp1);
7019 tcg_temp_free_i64(fp1);
7020 gen_store_fpr64(ctx, fp0, fd);
7021 tcg_temp_free_i64(fp0);
7022 }
7023 opn = "addr.ps";
7024 break;
7025 case FOP(26, 22):
7026 check_cp1_64bitmode(ctx);
7027 {
7028 TCGv_i64 fp0 = tcg_temp_new_i64();
7029 TCGv_i64 fp1 = tcg_temp_new_i64();
7030
7031 gen_load_fpr64(ctx, fp0, ft);
7032 gen_load_fpr64(ctx, fp1, fs);
7033 gen_helper_float_mulr_ps(fp0, fp0, fp1);
7034 tcg_temp_free_i64(fp1);
7035 gen_store_fpr64(ctx, fp0, fd);
7036 tcg_temp_free_i64(fp0);
7037 }
7038 opn = "mulr.ps";
7039 break;
7040 case FOP(28, 22):
7041 check_cp1_64bitmode(ctx);
7042 {
7043 TCGv_i64 fp0 = tcg_temp_new_i64();
7044 TCGv_i64 fp1 = tcg_temp_new_i64();
7045
7046 gen_load_fpr64(ctx, fp0, fs);
7047 gen_load_fpr64(ctx, fp1, fd);
7048 gen_helper_float_recip2_ps(fp0, fp0, fp1);
7049 tcg_temp_free_i64(fp1);
7050 gen_store_fpr64(ctx, fp0, fd);
7051 tcg_temp_free_i64(fp0);
7052 }
7053 opn = "recip2.ps";
7054 break;
7055 case FOP(29, 22):
7056 check_cp1_64bitmode(ctx);
7057 {
7058 TCGv_i64 fp0 = tcg_temp_new_i64();
7059
7060 gen_load_fpr64(ctx, fp0, fs);
7061 gen_helper_float_recip1_ps(fp0, fp0);
7062 gen_store_fpr64(ctx, fp0, fd);
7063 tcg_temp_free_i64(fp0);
7064 }
7065 opn = "recip1.ps";
7066 break;
7067 case FOP(30, 22):
7068 check_cp1_64bitmode(ctx);
7069 {
7070 TCGv_i64 fp0 = tcg_temp_new_i64();
7071
7072 gen_load_fpr64(ctx, fp0, fs);
7073 gen_helper_float_rsqrt1_ps(fp0, fp0);
7074 gen_store_fpr64(ctx, fp0, fd);
7075 tcg_temp_free_i64(fp0);
7076 }
7077 opn = "rsqrt1.ps";
7078 break;
7079 case FOP(31, 22):
7080 check_cp1_64bitmode(ctx);
7081 {
7082 TCGv_i64 fp0 = tcg_temp_new_i64();
7083 TCGv_i64 fp1 = tcg_temp_new_i64();
7084
7085 gen_load_fpr64(ctx, fp0, fs);
7086 gen_load_fpr64(ctx, fp1, ft);
7087 gen_helper_float_rsqrt2_ps(fp0, fp0, fp1);
7088 tcg_temp_free_i64(fp1);
7089 gen_store_fpr64(ctx, fp0, fd);
7090 tcg_temp_free_i64(fp0);
7091 }
7092 opn = "rsqrt2.ps";
7093 break;
7094 case FOP(32, 22):
7095 check_cp1_64bitmode(ctx);
7096 {
7097 TCGv_i32 fp0 = tcg_temp_new_i32();
7098
7099 gen_load_fpr32h(fp0, fs);
7100 gen_helper_float_cvts_pu(fp0, fp0);
7101 gen_store_fpr32(fp0, fd);
7102 tcg_temp_free_i32(fp0);
7103 }
7104 opn = "cvt.s.pu";
7105 break;
7106 case FOP(36, 22):
7107 check_cp1_64bitmode(ctx);
7108 {
7109 TCGv_i64 fp0 = tcg_temp_new_i64();
7110
7111 gen_load_fpr64(ctx, fp0, fs);
7112 gen_helper_float_cvtpw_ps(fp0, fp0);
7113 gen_store_fpr64(ctx, fp0, fd);
7114 tcg_temp_free_i64(fp0);
7115 }
7116 opn = "cvt.pw.ps";
7117 break;
7118 case FOP(40, 22):
7119 check_cp1_64bitmode(ctx);
7120 {
7121 TCGv_i32 fp0 = tcg_temp_new_i32();
7122
7123 gen_load_fpr32(fp0, fs);
7124 gen_helper_float_cvts_pl(fp0, fp0);
7125 gen_store_fpr32(fp0, fd);
7126 tcg_temp_free_i32(fp0);
7127 }
7128 opn = "cvt.s.pl";
7129 break;
7130 case FOP(44, 22):
7131 check_cp1_64bitmode(ctx);
7132 {
7133 TCGv_i32 fp0 = tcg_temp_new_i32();
7134 TCGv_i32 fp1 = tcg_temp_new_i32();
7135
7136 gen_load_fpr32(fp0, fs);
7137 gen_load_fpr32(fp1, ft);
7138 gen_store_fpr32h(fp0, fd);
7139 gen_store_fpr32(fp1, fd);
7140 tcg_temp_free_i32(fp0);
7141 tcg_temp_free_i32(fp1);
7142 }
7143 opn = "pll.ps";
7144 break;
7145 case FOP(45, 22):
7146 check_cp1_64bitmode(ctx);
7147 {
7148 TCGv_i32 fp0 = tcg_temp_new_i32();
7149 TCGv_i32 fp1 = tcg_temp_new_i32();
7150
7151 gen_load_fpr32(fp0, fs);
7152 gen_load_fpr32h(fp1, ft);
7153 gen_store_fpr32(fp1, fd);
7154 gen_store_fpr32h(fp0, fd);
7155 tcg_temp_free_i32(fp0);
7156 tcg_temp_free_i32(fp1);
7157 }
7158 opn = "plu.ps";
7159 break;
7160 case FOP(46, 22):
7161 check_cp1_64bitmode(ctx);
7162 {
7163 TCGv_i32 fp0 = tcg_temp_new_i32();
7164 TCGv_i32 fp1 = tcg_temp_new_i32();
7165
7166 gen_load_fpr32h(fp0, fs);
7167 gen_load_fpr32(fp1, ft);
7168 gen_store_fpr32(fp1, fd);
7169 gen_store_fpr32h(fp0, fd);
7170 tcg_temp_free_i32(fp0);
7171 tcg_temp_free_i32(fp1);
7172 }
7173 opn = "pul.ps";
7174 break;
7175 case FOP(47, 22):
7176 check_cp1_64bitmode(ctx);
7177 {
7178 TCGv_i32 fp0 = tcg_temp_new_i32();
7179 TCGv_i32 fp1 = tcg_temp_new_i32();
7180
7181 gen_load_fpr32h(fp0, fs);
7182 gen_load_fpr32h(fp1, ft);
7183 gen_store_fpr32(fp1, fd);
7184 gen_store_fpr32h(fp0, fd);
7185 tcg_temp_free_i32(fp0);
7186 tcg_temp_free_i32(fp1);
7187 }
7188 opn = "puu.ps";
7189 break;
7190 case FOP(48, 22):
7191 case FOP(49, 22):
7192 case FOP(50, 22):
7193 case FOP(51, 22):
7194 case FOP(52, 22):
7195 case FOP(53, 22):
7196 case FOP(54, 22):
7197 case FOP(55, 22):
7198 case FOP(56, 22):
7199 case FOP(57, 22):
7200 case FOP(58, 22):
7201 case FOP(59, 22):
7202 case FOP(60, 22):
7203 case FOP(61, 22):
7204 case FOP(62, 22):
7205 case FOP(63, 22):
7206 check_cp1_64bitmode(ctx);
7207 {
7208 TCGv_i64 fp0 = tcg_temp_new_i64();
7209 TCGv_i64 fp1 = tcg_temp_new_i64();
7210
7211 gen_load_fpr64(ctx, fp0, fs);
7212 gen_load_fpr64(ctx, fp1, ft);
7213 if (ctx->opcode & (1 << 6)) {
7214 gen_cmpabs_ps(func-48, fp0, fp1, cc);
7215 opn = condnames_abs[func-48];
7216 } else {
7217 gen_cmp_ps(func-48, fp0, fp1, cc);
7218 opn = condnames[func-48];
7219 }
7220 tcg_temp_free_i64(fp0);
7221 tcg_temp_free_i64(fp1);
7222 }
7223 break;
7224 default:
7225 MIPS_INVAL(opn);
7226 generate_exception (ctx, EXCP_RI);
7227 return;
7228 }
7229 switch (optype) {
7230 case BINOP:
7231 MIPS_DEBUG("%s %s, %s, %s", opn, fregnames[fd], fregnames[fs], fregnames[ft]);
7232 break;
7233 case CMPOP:
7234 MIPS_DEBUG("%s %s,%s", opn, fregnames[fs], fregnames[ft]);
7235 break;
7236 default:
7237 MIPS_DEBUG("%s %s,%s", opn, fregnames[fd], fregnames[fs]);
7238 break;
7239 }
7240 }
7241
7242 /* Coprocessor 3 (FPU) */
gen_flt3_ldst(DisasContext * ctx,uint32_t opc,int fd,int fs,int base,int index)7243 static void gen_flt3_ldst (DisasContext *ctx, uint32_t opc,
7244 int fd, int fs, int base, int index)
7245 {
7246 const char *opn = "extended float load/store";
7247 int store = 0;
7248 TCGv t0 = tcg_temp_new();
7249
7250 if (base == 0) {
7251 gen_load_gpr(t0, index);
7252 } else if (index == 0) {
7253 gen_load_gpr(t0, base);
7254 } else {
7255 gen_load_gpr(t0, index);
7256 gen_op_addr_add(ctx, t0, cpu_gpr[base], t0);
7257 }
7258 /* Don't do NOP if destination is zero: we must perform the actual
7259 memory access. */
7260 save_cpu_state(ctx, 0);
7261 switch (opc) {
7262 case OPC_LWXC1:
7263 check_cop1x(ctx);
7264 {
7265 TCGv_i32 fp0 = tcg_temp_new_i32();
7266
7267 tcg_gen_qemu_ld32s(t0, t0, ctx->mem_idx);
7268 tcg_gen_trunc_tl_i32(fp0, t0);
7269 gen_store_fpr32(fp0, fd);
7270 tcg_temp_free_i32(fp0);
7271 }
7272 opn = "lwxc1";
7273 break;
7274 case OPC_LDXC1:
7275 check_cop1x(ctx);
7276 check_cp1_registers(ctx, fd);
7277 {
7278 TCGv_i64 fp0 = tcg_temp_new_i64();
7279
7280 tcg_gen_qemu_ld64(fp0, t0, ctx->mem_idx);
7281 gen_store_fpr64(ctx, fp0, fd);
7282 tcg_temp_free_i64(fp0);
7283 }
7284 opn = "ldxc1";
7285 break;
7286 case OPC_LUXC1:
7287 check_cp1_64bitmode(ctx);
7288 tcg_gen_andi_tl(t0, t0, ~0x7);
7289 {
7290 TCGv_i64 fp0 = tcg_temp_new_i64();
7291
7292 tcg_gen_qemu_ld64(fp0, t0, ctx->mem_idx);
7293 gen_store_fpr64(ctx, fp0, fd);
7294 tcg_temp_free_i64(fp0);
7295 }
7296 opn = "luxc1";
7297 break;
7298 case OPC_SWXC1:
7299 check_cop1x(ctx);
7300 {
7301 TCGv_i32 fp0 = tcg_temp_new_i32();
7302 TCGv t1 = tcg_temp_new();
7303
7304 gen_load_fpr32(fp0, fs);
7305 tcg_gen_extu_i32_tl(t1, fp0);
7306 tcg_gen_qemu_st32(t1, t0, ctx->mem_idx);
7307 tcg_temp_free_i32(fp0);
7308 tcg_temp_free(t1);
7309 }
7310 opn = "swxc1";
7311 store = 1;
7312 break;
7313 case OPC_SDXC1:
7314 check_cop1x(ctx);
7315 check_cp1_registers(ctx, fs);
7316 {
7317 TCGv_i64 fp0 = tcg_temp_new_i64();
7318
7319 gen_load_fpr64(ctx, fp0, fs);
7320 tcg_gen_qemu_st64(fp0, t0, ctx->mem_idx);
7321 tcg_temp_free_i64(fp0);
7322 }
7323 opn = "sdxc1";
7324 store = 1;
7325 break;
7326 case OPC_SUXC1:
7327 check_cp1_64bitmode(ctx);
7328 tcg_gen_andi_tl(t0, t0, ~0x7);
7329 {
7330 TCGv_i64 fp0 = tcg_temp_new_i64();
7331
7332 gen_load_fpr64(ctx, fp0, fs);
7333 tcg_gen_qemu_st64(fp0, t0, ctx->mem_idx);
7334 tcg_temp_free_i64(fp0);
7335 }
7336 opn = "suxc1";
7337 store = 1;
7338 break;
7339 }
7340 tcg_temp_free(t0);
7341 MIPS_DEBUG("%s %s, %s(%s)", opn, fregnames[store ? fs : fd],
7342 regnames[index], regnames[base]);
7343 }
7344
gen_flt3_arith(DisasContext * ctx,uint32_t opc,int fd,int fr,int fs,int ft)7345 static void gen_flt3_arith (DisasContext *ctx, uint32_t opc,
7346 int fd, int fr, int fs, int ft)
7347 {
7348 const char *opn = "flt3_arith";
7349
7350 switch (opc) {
7351 case OPC_ALNV_PS:
7352 check_cp1_64bitmode(ctx);
7353 {
7354 TCGv t0 = tcg_temp_local_new();
7355 TCGv_i32 fp = tcg_temp_new_i32();
7356 TCGv_i32 fph = tcg_temp_new_i32();
7357 int l1 = gen_new_label();
7358 int l2 = gen_new_label();
7359
7360 gen_load_gpr(t0, fr);
7361 tcg_gen_andi_tl(t0, t0, 0x7);
7362
7363 tcg_gen_brcondi_tl(TCG_COND_NE, t0, 0, l1);
7364 gen_load_fpr32(fp, fs);
7365 gen_load_fpr32h(fph, fs);
7366 gen_store_fpr32(fp, fd);
7367 gen_store_fpr32h(fph, fd);
7368 tcg_gen_br(l2);
7369 gen_set_label(l1);
7370 tcg_gen_brcondi_tl(TCG_COND_NE, t0, 4, l2);
7371 tcg_temp_free(t0);
7372 #ifdef TARGET_WORDS_BIGENDIAN
7373 gen_load_fpr32(fp, fs);
7374 gen_load_fpr32h(fph, ft);
7375 gen_store_fpr32h(fp, fd);
7376 gen_store_fpr32(fph, fd);
7377 #else
7378 gen_load_fpr32h(fph, fs);
7379 gen_load_fpr32(fp, ft);
7380 gen_store_fpr32(fph, fd);
7381 gen_store_fpr32h(fp, fd);
7382 #endif
7383 gen_set_label(l2);
7384 tcg_temp_free_i32(fp);
7385 tcg_temp_free_i32(fph);
7386 }
7387 opn = "alnv.ps";
7388 break;
7389 case OPC_MADD_S:
7390 check_cop1x(ctx);
7391 {
7392 TCGv_i32 fp0 = tcg_temp_new_i32();
7393 TCGv_i32 fp1 = tcg_temp_new_i32();
7394 TCGv_i32 fp2 = tcg_temp_new_i32();
7395
7396 gen_load_fpr32(fp0, fs);
7397 gen_load_fpr32(fp1, ft);
7398 gen_load_fpr32(fp2, fr);
7399 gen_helper_float_muladd_s(fp2, fp0, fp1, fp2);
7400 tcg_temp_free_i32(fp0);
7401 tcg_temp_free_i32(fp1);
7402 gen_store_fpr32(fp2, fd);
7403 tcg_temp_free_i32(fp2);
7404 }
7405 opn = "madd.s";
7406 break;
7407 case OPC_MADD_D:
7408 check_cop1x(ctx);
7409 check_cp1_registers(ctx, fd | fs | ft | fr);
7410 {
7411 TCGv_i64 fp0 = tcg_temp_new_i64();
7412 TCGv_i64 fp1 = tcg_temp_new_i64();
7413 TCGv_i64 fp2 = tcg_temp_new_i64();
7414
7415 gen_load_fpr64(ctx, fp0, fs);
7416 gen_load_fpr64(ctx, fp1, ft);
7417 gen_load_fpr64(ctx, fp2, fr);
7418 gen_helper_float_muladd_d(fp2, fp0, fp1, fp2);
7419 tcg_temp_free_i64(fp0);
7420 tcg_temp_free_i64(fp1);
7421 gen_store_fpr64(ctx, fp2, fd);
7422 tcg_temp_free_i64(fp2);
7423 }
7424 opn = "madd.d";
7425 break;
7426 case OPC_MADD_PS:
7427 check_cp1_64bitmode(ctx);
7428 {
7429 TCGv_i64 fp0 = tcg_temp_new_i64();
7430 TCGv_i64 fp1 = tcg_temp_new_i64();
7431 TCGv_i64 fp2 = tcg_temp_new_i64();
7432
7433 gen_load_fpr64(ctx, fp0, fs);
7434 gen_load_fpr64(ctx, fp1, ft);
7435 gen_load_fpr64(ctx, fp2, fr);
7436 gen_helper_float_muladd_ps(fp2, fp0, fp1, fp2);
7437 tcg_temp_free_i64(fp0);
7438 tcg_temp_free_i64(fp1);
7439 gen_store_fpr64(ctx, fp2, fd);
7440 tcg_temp_free_i64(fp2);
7441 }
7442 opn = "madd.ps";
7443 break;
7444 case OPC_MSUB_S:
7445 check_cop1x(ctx);
7446 {
7447 TCGv_i32 fp0 = tcg_temp_new_i32();
7448 TCGv_i32 fp1 = tcg_temp_new_i32();
7449 TCGv_i32 fp2 = tcg_temp_new_i32();
7450
7451 gen_load_fpr32(fp0, fs);
7452 gen_load_fpr32(fp1, ft);
7453 gen_load_fpr32(fp2, fr);
7454 gen_helper_float_mulsub_s(fp2, fp0, fp1, fp2);
7455 tcg_temp_free_i32(fp0);
7456 tcg_temp_free_i32(fp1);
7457 gen_store_fpr32(fp2, fd);
7458 tcg_temp_free_i32(fp2);
7459 }
7460 opn = "msub.s";
7461 break;
7462 case OPC_MSUB_D:
7463 check_cop1x(ctx);
7464 check_cp1_registers(ctx, fd | fs | ft | fr);
7465 {
7466 TCGv_i64 fp0 = tcg_temp_new_i64();
7467 TCGv_i64 fp1 = tcg_temp_new_i64();
7468 TCGv_i64 fp2 = tcg_temp_new_i64();
7469
7470 gen_load_fpr64(ctx, fp0, fs);
7471 gen_load_fpr64(ctx, fp1, ft);
7472 gen_load_fpr64(ctx, fp2, fr);
7473 gen_helper_float_mulsub_d(fp2, fp0, fp1, fp2);
7474 tcg_temp_free_i64(fp0);
7475 tcg_temp_free_i64(fp1);
7476 gen_store_fpr64(ctx, fp2, fd);
7477 tcg_temp_free_i64(fp2);
7478 }
7479 opn = "msub.d";
7480 break;
7481 case OPC_MSUB_PS:
7482 check_cp1_64bitmode(ctx);
7483 {
7484 TCGv_i64 fp0 = tcg_temp_new_i64();
7485 TCGv_i64 fp1 = tcg_temp_new_i64();
7486 TCGv_i64 fp2 = tcg_temp_new_i64();
7487
7488 gen_load_fpr64(ctx, fp0, fs);
7489 gen_load_fpr64(ctx, fp1, ft);
7490 gen_load_fpr64(ctx, fp2, fr);
7491 gen_helper_float_mulsub_ps(fp2, fp0, fp1, fp2);
7492 tcg_temp_free_i64(fp0);
7493 tcg_temp_free_i64(fp1);
7494 gen_store_fpr64(ctx, fp2, fd);
7495 tcg_temp_free_i64(fp2);
7496 }
7497 opn = "msub.ps";
7498 break;
7499 case OPC_NMADD_S:
7500 check_cop1x(ctx);
7501 {
7502 TCGv_i32 fp0 = tcg_temp_new_i32();
7503 TCGv_i32 fp1 = tcg_temp_new_i32();
7504 TCGv_i32 fp2 = tcg_temp_new_i32();
7505
7506 gen_load_fpr32(fp0, fs);
7507 gen_load_fpr32(fp1, ft);
7508 gen_load_fpr32(fp2, fr);
7509 gen_helper_float_nmuladd_s(fp2, fp0, fp1, fp2);
7510 tcg_temp_free_i32(fp0);
7511 tcg_temp_free_i32(fp1);
7512 gen_store_fpr32(fp2, fd);
7513 tcg_temp_free_i32(fp2);
7514 }
7515 opn = "nmadd.s";
7516 break;
7517 case OPC_NMADD_D:
7518 check_cop1x(ctx);
7519 check_cp1_registers(ctx, fd | fs | ft | fr);
7520 {
7521 TCGv_i64 fp0 = tcg_temp_new_i64();
7522 TCGv_i64 fp1 = tcg_temp_new_i64();
7523 TCGv_i64 fp2 = tcg_temp_new_i64();
7524
7525 gen_load_fpr64(ctx, fp0, fs);
7526 gen_load_fpr64(ctx, fp1, ft);
7527 gen_load_fpr64(ctx, fp2, fr);
7528 gen_helper_float_nmuladd_d(fp2, fp0, fp1, fp2);
7529 tcg_temp_free_i64(fp0);
7530 tcg_temp_free_i64(fp1);
7531 gen_store_fpr64(ctx, fp2, fd);
7532 tcg_temp_free_i64(fp2);
7533 }
7534 opn = "nmadd.d";
7535 break;
7536 case OPC_NMADD_PS:
7537 check_cp1_64bitmode(ctx);
7538 {
7539 TCGv_i64 fp0 = tcg_temp_new_i64();
7540 TCGv_i64 fp1 = tcg_temp_new_i64();
7541 TCGv_i64 fp2 = tcg_temp_new_i64();
7542
7543 gen_load_fpr64(ctx, fp0, fs);
7544 gen_load_fpr64(ctx, fp1, ft);
7545 gen_load_fpr64(ctx, fp2, fr);
7546 gen_helper_float_nmuladd_ps(fp2, fp0, fp1, fp2);
7547 tcg_temp_free_i64(fp0);
7548 tcg_temp_free_i64(fp1);
7549 gen_store_fpr64(ctx, fp2, fd);
7550 tcg_temp_free_i64(fp2);
7551 }
7552 opn = "nmadd.ps";
7553 break;
7554 case OPC_NMSUB_S:
7555 check_cop1x(ctx);
7556 {
7557 TCGv_i32 fp0 = tcg_temp_new_i32();
7558 TCGv_i32 fp1 = tcg_temp_new_i32();
7559 TCGv_i32 fp2 = tcg_temp_new_i32();
7560
7561 gen_load_fpr32(fp0, fs);
7562 gen_load_fpr32(fp1, ft);
7563 gen_load_fpr32(fp2, fr);
7564 gen_helper_float_nmulsub_s(fp2, fp0, fp1, fp2);
7565 tcg_temp_free_i32(fp0);
7566 tcg_temp_free_i32(fp1);
7567 gen_store_fpr32(fp2, fd);
7568 tcg_temp_free_i32(fp2);
7569 }
7570 opn = "nmsub.s";
7571 break;
7572 case OPC_NMSUB_D:
7573 check_cop1x(ctx);
7574 check_cp1_registers(ctx, fd | fs | ft | fr);
7575 {
7576 TCGv_i64 fp0 = tcg_temp_new_i64();
7577 TCGv_i64 fp1 = tcg_temp_new_i64();
7578 TCGv_i64 fp2 = tcg_temp_new_i64();
7579
7580 gen_load_fpr64(ctx, fp0, fs);
7581 gen_load_fpr64(ctx, fp1, ft);
7582 gen_load_fpr64(ctx, fp2, fr);
7583 gen_helper_float_nmulsub_d(fp2, fp0, fp1, fp2);
7584 tcg_temp_free_i64(fp0);
7585 tcg_temp_free_i64(fp1);
7586 gen_store_fpr64(ctx, fp2, fd);
7587 tcg_temp_free_i64(fp2);
7588 }
7589 opn = "nmsub.d";
7590 break;
7591 case OPC_NMSUB_PS:
7592 check_cp1_64bitmode(ctx);
7593 {
7594 TCGv_i64 fp0 = tcg_temp_new_i64();
7595 TCGv_i64 fp1 = tcg_temp_new_i64();
7596 TCGv_i64 fp2 = tcg_temp_new_i64();
7597
7598 gen_load_fpr64(ctx, fp0, fs);
7599 gen_load_fpr64(ctx, fp1, ft);
7600 gen_load_fpr64(ctx, fp2, fr);
7601 gen_helper_float_nmulsub_ps(fp2, fp0, fp1, fp2);
7602 tcg_temp_free_i64(fp0);
7603 tcg_temp_free_i64(fp1);
7604 gen_store_fpr64(ctx, fp2, fd);
7605 tcg_temp_free_i64(fp2);
7606 }
7607 opn = "nmsub.ps";
7608 break;
7609 default:
7610 MIPS_INVAL(opn);
7611 generate_exception (ctx, EXCP_RI);
7612 return;
7613 }
7614 MIPS_DEBUG("%s %s, %s, %s, %s", opn, fregnames[fd], fregnames[fr],
7615 fregnames[fs], fregnames[ft]);
7616 }
7617
7618 /* ISA extensions (ASEs) */
7619 /* MIPS16 extension to MIPS32 */
7620 /* SmartMIPS extension to MIPS32 */
7621
7622 #if defined(TARGET_MIPS64)
7623
7624 /* MDMX extension to MIPS64 */
7625
7626 #endif
7627
decode_opc(CPUState * env,DisasContext * ctx)7628 static void decode_opc (CPUState *env, DisasContext *ctx)
7629 {
7630 int32_t offset;
7631 int rs, rt, rd, sa;
7632 uint32_t op, op1, op2;
7633 int16_t imm;
7634
7635 /* make sure instructions are on a word boundary */
7636 if (ctx->pc & 0x3) {
7637 env->CP0_BadVAddr = ctx->pc;
7638 generate_exception(ctx, EXCP_AdEL);
7639 return;
7640 }
7641
7642 /* Handle blikely not taken case */
7643 if ((ctx->hflags & MIPS_HFLAG_BMASK) == MIPS_HFLAG_BL) {
7644 int l1 = gen_new_label();
7645
7646 MIPS_DEBUG("blikely condition (" TARGET_FMT_lx ")", ctx->pc + 4);
7647 tcg_gen_brcondi_tl(TCG_COND_NE, bcond, 0, l1);
7648 tcg_gen_movi_i32(hflags, ctx->hflags & ~MIPS_HFLAG_BMASK);
7649 gen_goto_tb(ctx, 1, ctx->pc + 4);
7650 gen_set_label(l1);
7651 }
7652
7653 if (unlikely(qemu_loglevel_mask(CPU_LOG_TB_OP)))
7654 tcg_gen_debug_insn_start(ctx->pc);
7655
7656 op = MASK_OP_MAJOR(ctx->opcode);
7657 rs = (ctx->opcode >> 21) & 0x1f;
7658 rt = (ctx->opcode >> 16) & 0x1f;
7659 rd = (ctx->opcode >> 11) & 0x1f;
7660 sa = (ctx->opcode >> 6) & 0x1f;
7661 imm = (int16_t)ctx->opcode;
7662 switch (op) {
7663 case OPC_SPECIAL:
7664 op1 = MASK_SPECIAL(ctx->opcode);
7665 switch (op1) {
7666 case OPC_SLL: /* Shift with immediate */
7667 case OPC_SRA:
7668 case OPC_SRL:
7669 gen_shift_imm(env, ctx, op1, rd, rt, sa);
7670 break;
7671 case OPC_MOVN: /* Conditional move */
7672 case OPC_MOVZ:
7673 check_insn(env, ctx, ISA_MIPS4 | ISA_MIPS32);
7674 gen_cond_move(env, op1, rd, rs, rt);
7675 break;
7676 case OPC_ADD ... OPC_SUBU:
7677 gen_arith(env, ctx, op1, rd, rs, rt);
7678 break;
7679 case OPC_SLLV: /* Shifts */
7680 case OPC_SRLV:
7681 case OPC_SRAV:
7682 gen_shift(env, ctx, op1, rd, rs, rt);
7683 break;
7684 case OPC_SLT: /* Set on less than */
7685 case OPC_SLTU:
7686 gen_slt(env, op1, rd, rs, rt);
7687 break;
7688 case OPC_AND: /* Logic*/
7689 case OPC_OR:
7690 case OPC_NOR:
7691 case OPC_XOR:
7692 gen_logic(env, op1, rd, rs, rt);
7693 break;
7694 case OPC_MULT ... OPC_DIVU:
7695 if (sa) {
7696 check_insn(env, ctx, INSN_VR54XX);
7697 op1 = MASK_MUL_VR54XX(ctx->opcode);
7698 gen_mul_vr54xx(ctx, op1, rd, rs, rt);
7699 } else
7700 gen_muldiv(ctx, op1, rs, rt);
7701 break;
7702 case OPC_JR ... OPC_JALR:
7703 gen_compute_branch(ctx, op1, rs, rd, sa);
7704 return;
7705 case OPC_TGE ... OPC_TEQ: /* Traps */
7706 case OPC_TNE:
7707 gen_trap(ctx, op1, rs, rt, -1);
7708 break;
7709 case OPC_MFHI: /* Move from HI/LO */
7710 case OPC_MFLO:
7711 gen_HILO(ctx, op1, rd);
7712 break;
7713 case OPC_MTHI:
7714 case OPC_MTLO: /* Move to HI/LO */
7715 gen_HILO(ctx, op1, rs);
7716 break;
7717 case OPC_PMON: /* Pmon entry point, also R4010 selsl */
7718 #ifdef MIPS_STRICT_STANDARD
7719 MIPS_INVAL("PMON / selsl");
7720 generate_exception(ctx, EXCP_RI);
7721 #else
7722 gen_helper_0i(pmon, sa);
7723 #endif
7724 break;
7725 case OPC_SYSCALL:
7726 generate_exception(ctx, EXCP_SYSCALL);
7727 ctx->bstate = BS_STOP;
7728 break;
7729 case OPC_BREAK:
7730 generate_exception(ctx, EXCP_BREAK);
7731 break;
7732 case OPC_SPIM:
7733 #ifdef MIPS_STRICT_STANDARD
7734 MIPS_INVAL("SPIM");
7735 generate_exception(ctx, EXCP_RI);
7736 #else
7737 /* Implemented as RI exception for now. */
7738 MIPS_INVAL("spim (unofficial)");
7739 generate_exception(ctx, EXCP_RI);
7740 #endif
7741 break;
7742 case OPC_SYNC:
7743 /* Treat as NOP. */
7744 break;
7745
7746 case OPC_MOVCI:
7747 check_insn(env, ctx, ISA_MIPS4 | ISA_MIPS32);
7748 if (env->CP0_Config1 & (1 << CP0C1_FP)) {
7749 check_cp1_enabled(ctx);
7750 gen_movci(ctx, rd, rs, (ctx->opcode >> 18) & 0x7,
7751 (ctx->opcode >> 16) & 1);
7752 } else {
7753 generate_exception_err(ctx, EXCP_CpU, 1);
7754 }
7755 break;
7756
7757 #if defined(TARGET_MIPS64)
7758 /* MIPS64 specific opcodes */
7759 case OPC_DSLL:
7760 case OPC_DSRA:
7761 case OPC_DSRL:
7762 case OPC_DSLL32:
7763 case OPC_DSRA32:
7764 case OPC_DSRL32:
7765 check_insn(env, ctx, ISA_MIPS3);
7766 check_mips_64(ctx);
7767 gen_shift_imm(env, ctx, op1, rd, rt, sa);
7768 break;
7769 case OPC_DADD ... OPC_DSUBU:
7770 check_insn(env, ctx, ISA_MIPS3);
7771 check_mips_64(ctx);
7772 gen_arith(env, ctx, op1, rd, rs, rt);
7773 break;
7774 case OPC_DSLLV:
7775 case OPC_DSRAV:
7776 case OPC_DSRLV:
7777 check_insn(env, ctx, ISA_MIPS3);
7778 check_mips_64(ctx);
7779 gen_shift(env, ctx, op1, rd, rs, rt);
7780 break;
7781 case OPC_DMULT ... OPC_DDIVU:
7782 check_insn(env, ctx, ISA_MIPS3);
7783 check_mips_64(ctx);
7784 gen_muldiv(ctx, op1, rs, rt);
7785 break;
7786 #endif
7787 default: /* Invalid */
7788 MIPS_INVAL("special");
7789 generate_exception(ctx, EXCP_RI);
7790 break;
7791 }
7792 break;
7793 case OPC_SPECIAL2:
7794 op1 = MASK_SPECIAL2(ctx->opcode);
7795 switch (op1) {
7796 case OPC_MADD ... OPC_MADDU: /* Multiply and add/sub */
7797 case OPC_MSUB ... OPC_MSUBU:
7798 check_insn(env, ctx, ISA_MIPS32);
7799 gen_muldiv(ctx, op1, rs, rt);
7800 break;
7801 case OPC_MUL:
7802 gen_arith(env, ctx, op1, rd, rs, rt);
7803 break;
7804 case OPC_CLO:
7805 case OPC_CLZ:
7806 check_insn(env, ctx, ISA_MIPS32);
7807 gen_cl(ctx, op1, rd, rs);
7808 break;
7809 case OPC_SDBBP:
7810 /* XXX: not clear which exception should be raised
7811 * when in debug mode...
7812 */
7813 check_insn(env, ctx, ISA_MIPS32);
7814 if (!(ctx->hflags & MIPS_HFLAG_DM)) {
7815 generate_exception(ctx, EXCP_DBp);
7816 } else {
7817 generate_exception(ctx, EXCP_DBp);
7818 }
7819 /* Treat as NOP. */
7820 break;
7821 #if defined(TARGET_MIPS64)
7822 case OPC_DCLO:
7823 case OPC_DCLZ:
7824 check_insn(env, ctx, ISA_MIPS64);
7825 check_mips_64(ctx);
7826 gen_cl(ctx, op1, rd, rs);
7827 break;
7828 #endif
7829 default: /* Invalid */
7830 MIPS_INVAL("special2");
7831 generate_exception(ctx, EXCP_RI);
7832 break;
7833 }
7834 break;
7835 case OPC_SPECIAL3:
7836 op1 = MASK_SPECIAL3(ctx->opcode);
7837 switch (op1) {
7838 case OPC_EXT:
7839 case OPC_INS:
7840 check_insn(env, ctx, ISA_MIPS32R2);
7841 gen_bitops(ctx, op1, rt, rs, sa, rd);
7842 break;
7843 case OPC_BSHFL:
7844 check_insn(env, ctx, ISA_MIPS32R2);
7845 op2 = MASK_BSHFL(ctx->opcode);
7846 gen_bshfl(ctx, op2, rt, rd);
7847 break;
7848 case OPC_RDHWR:
7849 check_insn(env, ctx, ISA_MIPS32R2);
7850 {
7851 TCGv t0 = tcg_temp_new();
7852
7853 switch (rd) {
7854 case 0:
7855 save_cpu_state(ctx, 1);
7856 gen_helper_rdhwr_cpunum(t0);
7857 gen_store_gpr(t0, rt);
7858 break;
7859 case 1:
7860 save_cpu_state(ctx, 1);
7861 gen_helper_rdhwr_synci_step(t0);
7862 gen_store_gpr(t0, rt);
7863 break;
7864 case 2:
7865 save_cpu_state(ctx, 1);
7866 gen_helper_rdhwr_cc(t0);
7867 gen_store_gpr(t0, rt);
7868 break;
7869 case 3:
7870 save_cpu_state(ctx, 1);
7871 gen_helper_rdhwr_ccres(t0);
7872 gen_store_gpr(t0, rt);
7873 break;
7874 case 29:
7875 #if defined(CONFIG_USER_ONLY)
7876 tcg_gen_ld_tl(t0, cpu_env, offsetof(CPUState, tls_value));
7877 gen_store_gpr(t0, rt);
7878 break;
7879 #else
7880 /* XXX: Some CPUs implement this in hardware.
7881 Not supported yet. */
7882 #endif
7883 default: /* Invalid */
7884 MIPS_INVAL("rdhwr");
7885 generate_exception(ctx, EXCP_RI);
7886 break;
7887 }
7888 tcg_temp_free(t0);
7889 }
7890 break;
7891 case OPC_FORK:
7892 check_insn(env, ctx, ASE_MT);
7893 {
7894 TCGv t0 = tcg_temp_new();
7895 TCGv t1 = tcg_temp_new();
7896
7897 gen_load_gpr(t0, rt);
7898 gen_load_gpr(t1, rs);
7899 gen_helper_fork(t0, t1);
7900 tcg_temp_free(t0);
7901 tcg_temp_free(t1);
7902 }
7903 break;
7904 case OPC_YIELD:
7905 check_insn(env, ctx, ASE_MT);
7906 {
7907 TCGv t0 = tcg_temp_new();
7908
7909 save_cpu_state(ctx, 1);
7910 gen_load_gpr(t0, rs);
7911 gen_helper_yield(t0, t0);
7912 gen_store_gpr(t0, rd);
7913 tcg_temp_free(t0);
7914 }
7915 break;
7916 #if defined(TARGET_MIPS64)
7917 case OPC_DEXTM ... OPC_DEXT:
7918 case OPC_DINSM ... OPC_DINS:
7919 check_insn(env, ctx, ISA_MIPS64R2);
7920 check_mips_64(ctx);
7921 gen_bitops(ctx, op1, rt, rs, sa, rd);
7922 break;
7923 case OPC_DBSHFL:
7924 check_insn(env, ctx, ISA_MIPS64R2);
7925 check_mips_64(ctx);
7926 op2 = MASK_DBSHFL(ctx->opcode);
7927 gen_bshfl(ctx, op2, rt, rd);
7928 break;
7929 #endif
7930 default: /* Invalid */
7931 MIPS_INVAL("special3");
7932 generate_exception(ctx, EXCP_RI);
7933 break;
7934 }
7935 break;
7936 case OPC_REGIMM:
7937 op1 = MASK_REGIMM(ctx->opcode);
7938 switch (op1) {
7939 case OPC_BLTZ ... OPC_BGEZL: /* REGIMM branches */
7940 case OPC_BLTZAL ... OPC_BGEZALL:
7941 gen_compute_branch(ctx, op1, rs, -1, imm << 2);
7942 return;
7943 case OPC_TGEI ... OPC_TEQI: /* REGIMM traps */
7944 case OPC_TNEI:
7945 gen_trap(ctx, op1, rs, -1, imm);
7946 break;
7947 case OPC_SYNCI:
7948 check_insn(env, ctx, ISA_MIPS32R2);
7949 /* Treat as NOP. */
7950 break;
7951 default: /* Invalid */
7952 MIPS_INVAL("regimm");
7953 generate_exception(ctx, EXCP_RI);
7954 break;
7955 }
7956 break;
7957 case OPC_CP0:
7958 check_cp0_enabled(ctx);
7959 op1 = MASK_CP0(ctx->opcode);
7960 switch (op1) {
7961 case OPC_MFC0:
7962 case OPC_MTC0:
7963 case OPC_MFTR:
7964 case OPC_MTTR:
7965 #if defined(TARGET_MIPS64)
7966 case OPC_DMFC0:
7967 case OPC_DMTC0:
7968 #endif
7969 #ifndef CONFIG_USER_ONLY
7970 gen_cp0(env, ctx, op1, rt, rd);
7971 #endif /* !CONFIG_USER_ONLY */
7972 break;
7973 case OPC_C0_FIRST ... OPC_C0_LAST:
7974 #ifndef CONFIG_USER_ONLY
7975 gen_cp0(env, ctx, MASK_C0(ctx->opcode), rt, rd);
7976 #endif /* !CONFIG_USER_ONLY */
7977 break;
7978 case OPC_MFMC0:
7979 #ifndef CONFIG_USER_ONLY
7980 {
7981 TCGv t0 = tcg_temp_new();
7982
7983 op2 = MASK_MFMC0(ctx->opcode);
7984 switch (op2) {
7985 case OPC_DMT:
7986 check_insn(env, ctx, ASE_MT);
7987 gen_helper_dmt(t0, t0);
7988 gen_store_gpr(t0, rt);
7989 break;
7990 case OPC_EMT:
7991 check_insn(env, ctx, ASE_MT);
7992 gen_helper_emt(t0, t0);
7993 gen_store_gpr(t0, rt);
7994 break;
7995 case OPC_DVPE:
7996 check_insn(env, ctx, ASE_MT);
7997 gen_helper_dvpe(t0, t0);
7998 gen_store_gpr(t0, rt);
7999 break;
8000 case OPC_EVPE:
8001 check_insn(env, ctx, ASE_MT);
8002 gen_helper_evpe(t0, t0);
8003 gen_store_gpr(t0, rt);
8004 break;
8005 case OPC_DI:
8006 check_insn(env, ctx, ISA_MIPS32R2);
8007 save_cpu_state(ctx, 1);
8008 gen_helper_di(t0);
8009 gen_store_gpr(t0, rt);
8010 /* Stop translation as we may have switched the execution mode */
8011 ctx->bstate = BS_STOP;
8012 break;
8013 case OPC_EI:
8014 check_insn(env, ctx, ISA_MIPS32R2);
8015 save_cpu_state(ctx, 1);
8016 gen_helper_ei(t0);
8017 gen_store_gpr(t0, rt);
8018 /* Stop translation as we may have switched the execution mode */
8019 ctx->bstate = BS_STOP;
8020 break;
8021 default: /* Invalid */
8022 MIPS_INVAL("mfmc0");
8023 generate_exception(ctx, EXCP_RI);
8024 break;
8025 }
8026 tcg_temp_free(t0);
8027 }
8028 #endif /* !CONFIG_USER_ONLY */
8029 break;
8030 case OPC_RDPGPR:
8031 check_insn(env, ctx, ISA_MIPS32R2);
8032 gen_load_srsgpr(rt, rd);
8033 break;
8034 case OPC_WRPGPR:
8035 check_insn(env, ctx, ISA_MIPS32R2);
8036 gen_store_srsgpr(rt, rd);
8037 break;
8038 default:
8039 MIPS_INVAL("cp0");
8040 generate_exception(ctx, EXCP_RI);
8041 break;
8042 }
8043 break;
8044 case OPC_ADDI: /* Arithmetic with immediate opcode */
8045 case OPC_ADDIU:
8046 gen_arith_imm(env, ctx, op, rt, rs, imm);
8047 break;
8048 case OPC_SLTI: /* Set on less than with immediate opcode */
8049 case OPC_SLTIU:
8050 gen_slt_imm(env, op, rt, rs, imm);
8051 break;
8052 case OPC_ANDI: /* Arithmetic with immediate opcode */
8053 case OPC_LUI:
8054 case OPC_ORI:
8055 case OPC_XORI:
8056 gen_logic_imm(env, op, rt, rs, imm);
8057 break;
8058 case OPC_J ... OPC_JAL: /* Jump */
8059 offset = (int32_t)(ctx->opcode & 0x3FFFFFF) << 2;
8060 gen_compute_branch(ctx, op, rs, rt, offset);
8061 return;
8062 case OPC_BEQ ... OPC_BGTZ: /* Branch */
8063 case OPC_BEQL ... OPC_BGTZL:
8064 gen_compute_branch(ctx, op, rs, rt, imm << 2);
8065 return;
8066 case OPC_LB ... OPC_LWR: /* Load and stores */
8067 case OPC_SB ... OPC_SW:
8068 case OPC_SWR:
8069 case OPC_LL:
8070 gen_ldst(ctx, op, rt, rs, imm);
8071 break;
8072 case OPC_SC:
8073 gen_st_cond(ctx, op, rt, rs, imm);
8074 break;
8075 case OPC_CACHE:
8076 check_insn(env, ctx, ISA_MIPS3 | ISA_MIPS32);
8077 /* Treat as NOP. */
8078 break;
8079 case OPC_PREF:
8080 check_insn(env, ctx, ISA_MIPS4 | ISA_MIPS32);
8081 /* Treat as NOP. */
8082 break;
8083
8084 /* Floating point (COP1). */
8085 case OPC_LWC1:
8086 case OPC_LDC1:
8087 case OPC_SWC1:
8088 case OPC_SDC1:
8089 if (env->CP0_Config1 & (1 << CP0C1_FP)) {
8090 check_cp1_enabled(ctx);
8091 gen_flt_ldst(ctx, op, rt, rs, imm);
8092 } else {
8093 generate_exception_err(ctx, EXCP_CpU, 1);
8094 }
8095 break;
8096
8097 case OPC_CP1:
8098 if (env->CP0_Config1 & (1 << CP0C1_FP)) {
8099 check_cp1_enabled(ctx);
8100 op1 = MASK_CP1(ctx->opcode);
8101 switch (op1) {
8102 case OPC_MFHC1:
8103 case OPC_MTHC1:
8104 check_insn(env, ctx, ISA_MIPS32R2);
8105 case OPC_MFC1:
8106 case OPC_CFC1:
8107 case OPC_MTC1:
8108 case OPC_CTC1:
8109 gen_cp1(ctx, op1, rt, rd);
8110 break;
8111 #if defined(TARGET_MIPS64)
8112 case OPC_DMFC1:
8113 case OPC_DMTC1:
8114 check_insn(env, ctx, ISA_MIPS3);
8115 gen_cp1(ctx, op1, rt, rd);
8116 break;
8117 #endif
8118 case OPC_BC1ANY2:
8119 case OPC_BC1ANY4:
8120 check_cop1x(ctx);
8121 check_insn(env, ctx, ASE_MIPS3D);
8122 /* fall through */
8123 case OPC_BC1:
8124 gen_compute_branch1(env, ctx, MASK_BC1(ctx->opcode),
8125 (rt >> 2) & 0x7, imm << 2);
8126 return;
8127 case OPC_S_FMT:
8128 case OPC_D_FMT:
8129 case OPC_W_FMT:
8130 case OPC_L_FMT:
8131 case OPC_PS_FMT:
8132 gen_farith(ctx, MASK_CP1_FUNC(ctx->opcode), rt, rd, sa,
8133 (imm >> 8) & 0x7);
8134 break;
8135 default:
8136 MIPS_INVAL("cp1");
8137 generate_exception (ctx, EXCP_RI);
8138 break;
8139 }
8140 } else {
8141 generate_exception_err(ctx, EXCP_CpU, 1);
8142 }
8143 break;
8144
8145 /* COP2. */
8146 case OPC_LWC2:
8147 case OPC_LDC2:
8148 case OPC_SWC2:
8149 case OPC_SDC2:
8150 case OPC_CP2:
8151 /* COP2: Not implemented. */
8152 generate_exception_err(ctx, EXCP_CpU, 2);
8153 break;
8154
8155 case OPC_CP3:
8156 if (env->CP0_Config1 & (1 << CP0C1_FP)) {
8157 check_cp1_enabled(ctx);
8158 op1 = MASK_CP3(ctx->opcode);
8159 switch (op1) {
8160 case OPC_LWXC1:
8161 case OPC_LDXC1:
8162 case OPC_LUXC1:
8163 case OPC_SWXC1:
8164 case OPC_SDXC1:
8165 case OPC_SUXC1:
8166 gen_flt3_ldst(ctx, op1, sa, rd, rs, rt);
8167 break;
8168 case OPC_PREFX:
8169 /* Treat as NOP. */
8170 break;
8171 case OPC_ALNV_PS:
8172 case OPC_MADD_S:
8173 case OPC_MADD_D:
8174 case OPC_MADD_PS:
8175 case OPC_MSUB_S:
8176 case OPC_MSUB_D:
8177 case OPC_MSUB_PS:
8178 case OPC_NMADD_S:
8179 case OPC_NMADD_D:
8180 case OPC_NMADD_PS:
8181 case OPC_NMSUB_S:
8182 case OPC_NMSUB_D:
8183 case OPC_NMSUB_PS:
8184 gen_flt3_arith(ctx, op1, sa, rs, rd, rt);
8185 break;
8186 default:
8187 MIPS_INVAL("cp3");
8188 generate_exception (ctx, EXCP_RI);
8189 break;
8190 }
8191 } else {
8192 generate_exception_err(ctx, EXCP_CpU, 1);
8193 }
8194 break;
8195
8196 #if defined(TARGET_MIPS64)
8197 /* MIPS64 opcodes */
8198 case OPC_LWU:
8199 case OPC_LDL ... OPC_LDR:
8200 case OPC_SDL ... OPC_SDR:
8201 case OPC_LLD:
8202 case OPC_LD:
8203 case OPC_SD:
8204 check_insn(env, ctx, ISA_MIPS3);
8205 check_mips_64(ctx);
8206 gen_ldst(ctx, op, rt, rs, imm);
8207 break;
8208 case OPC_SCD:
8209 check_insn(env, ctx, ISA_MIPS3);
8210 check_mips_64(ctx);
8211 gen_st_cond(ctx, op, rt, rs, imm);
8212 break;
8213 case OPC_DADDI:
8214 case OPC_DADDIU:
8215 check_insn(env, ctx, ISA_MIPS3);
8216 check_mips_64(ctx);
8217 gen_arith_imm(env, ctx, op, rt, rs, imm);
8218 break;
8219 #endif
8220 case OPC_JALX:
8221 check_insn(env, ctx, ASE_MIPS16);
8222 /* MIPS16: Not implemented. */
8223 case OPC_MDMX:
8224 check_insn(env, ctx, ASE_MDMX);
8225 /* MDMX: Not implemented. */
8226 default: /* Invalid */
8227 MIPS_INVAL("major opcode");
8228 generate_exception(ctx, EXCP_RI);
8229 break;
8230 }
8231 if (ctx->hflags & MIPS_HFLAG_BMASK) {
8232 int hflags = ctx->hflags & MIPS_HFLAG_BMASK;
8233 /* Branches completion */
8234 ctx->hflags &= ~MIPS_HFLAG_BMASK;
8235 ctx->bstate = BS_BRANCH;
8236 save_cpu_state(ctx, 0);
8237 /* FIXME: Need to clear can_do_io. */
8238 switch (hflags) {
8239 case MIPS_HFLAG_B:
8240 /* unconditional branch */
8241 MIPS_DEBUG("unconditional branch");
8242 gen_goto_tb(ctx, 0, ctx->btarget);
8243 break;
8244 case MIPS_HFLAG_BL:
8245 /* blikely taken case */
8246 MIPS_DEBUG("blikely branch taken");
8247 gen_goto_tb(ctx, 0, ctx->btarget);
8248 break;
8249 case MIPS_HFLAG_BC:
8250 /* Conditional branch */
8251 MIPS_DEBUG("conditional branch");
8252 {
8253 int l1 = gen_new_label();
8254
8255 tcg_gen_brcondi_tl(TCG_COND_NE, bcond, 0, l1);
8256 gen_goto_tb(ctx, 1, ctx->pc + 4);
8257 gen_set_label(l1);
8258 gen_goto_tb(ctx, 0, ctx->btarget);
8259 }
8260 break;
8261 case MIPS_HFLAG_BR:
8262 /* unconditional branch to register */
8263 MIPS_DEBUG("branch to register");
8264 tcg_gen_mov_tl(cpu_PC, btarget);
8265 if (ctx->singlestep_enabled) {
8266 save_cpu_state(ctx, 0);
8267 gen_helper_0i(raise_exception, EXCP_DEBUG);
8268 }
8269 tcg_gen_exit_tb(0);
8270 break;
8271 default:
8272 MIPS_DEBUG("unknown branch");
8273 break;
8274 }
8275 }
8276 }
8277
8278 static inline void
gen_intermediate_code_internal(CPUState * env,TranslationBlock * tb,int search_pc)8279 gen_intermediate_code_internal (CPUState *env, TranslationBlock *tb,
8280 int search_pc)
8281 {
8282 DisasContext ctx;
8283 target_ulong pc_start;
8284 uint16_t *gen_opc_end;
8285 CPUBreakpoint *bp;
8286 int j, lj = -1;
8287 int num_insns;
8288 int max_insns;
8289
8290 if (search_pc)
8291 qemu_log("search pc %d\n", search_pc);
8292
8293 pc_start = tb->pc;
8294 gen_opc_end = gen_opc_buf + OPC_MAX_SIZE;
8295 ctx.pc = pc_start;
8296 ctx.saved_pc = -1;
8297 ctx.singlestep_enabled = env->singlestep_enabled;
8298 ctx.tb = tb;
8299 ctx.bstate = BS_NONE;
8300 /* Restore delay slot state from the tb context. */
8301 ctx.hflags = (uint32_t)tb->flags; /* FIXME: maybe use 64 bits here? */
8302 restore_cpu_state(env, &ctx);
8303 #ifdef CONFIG_USER_ONLY
8304 ctx.mem_idx = MIPS_HFLAG_UM;
8305 #else
8306 ctx.mem_idx = ctx.hflags & MIPS_HFLAG_KSU;
8307 #endif
8308 num_insns = 0;
8309 max_insns = tb->cflags & CF_COUNT_MASK;
8310 if (max_insns == 0)
8311 max_insns = CF_COUNT_MASK;
8312 #ifdef DEBUG_DISAS
8313 qemu_log_mask(CPU_LOG_TB_CPU, "------------------------------------------------\n");
8314 /* FIXME: This may print out stale hflags from env... */
8315 log_cpu_state_mask(CPU_LOG_TB_CPU, env, 0);
8316 #endif
8317 LOG_DISAS("\ntb %p idx %d hflags %04x\n", tb, ctx.mem_idx, ctx.hflags);
8318 gen_icount_start();
8319 while (ctx.bstate == BS_NONE) {
8320 if (unlikely(!QTAILQ_EMPTY(&env->breakpoints))) {
8321 QTAILQ_FOREACH(bp, &env->breakpoints, entry) {
8322 if (bp->pc == ctx.pc) {
8323 save_cpu_state(&ctx, 1);
8324 ctx.bstate = BS_BRANCH;
8325 gen_helper_0i(raise_exception, EXCP_DEBUG);
8326 /* Include the breakpoint location or the tb won't
8327 * be flushed when it must be. */
8328 ctx.pc += 4;
8329 goto done_generating;
8330 }
8331 }
8332 }
8333
8334 if (search_pc) {
8335 j = gen_opc_ptr - gen_opc_buf;
8336 if (lj < j) {
8337 lj++;
8338 while (lj < j)
8339 gen_opc_instr_start[lj++] = 0;
8340 }
8341 gen_opc_pc[lj] = ctx.pc;
8342 gen_opc_hflags[lj] = ctx.hflags & MIPS_HFLAG_BMASK;
8343 gen_opc_instr_start[lj] = 1;
8344 gen_opc_icount[lj] = num_insns;
8345 }
8346 if (num_insns + 1 == max_insns && (tb->cflags & CF_LAST_IO))
8347 gen_io_start();
8348 ctx.opcode = ldl_code(ctx.pc);
8349 decode_opc(env, &ctx);
8350 ctx.pc += 4;
8351 num_insns++;
8352
8353 /* Execute a branch and its delay slot as a single instruction.
8354 This is what GDB expects and is consistent with what the
8355 hardware does (e.g. if a delay slot instruction faults, the
8356 reported PC is the PC of the branch). */
8357 if (env->singlestep_enabled && (ctx.hflags & MIPS_HFLAG_BMASK) == 0)
8358 break;
8359
8360 /* Do not split a branch instruction and its delay slot into two
8361 TB's when a page boundary is crossed. This causes TB's to be
8362 invalidated incorrectly if branch target is patched. */
8363 if ((ctx.pc & (TARGET_PAGE_SIZE - 1)) == 0 && (ctx.hflags & MIPS_HFLAG_BMASK) == 0)
8364 break;
8365
8366 if (gen_opc_ptr >= gen_opc_end)
8367 break;
8368
8369 if (num_insns >= max_insns)
8370 break;
8371
8372 if (singlestep)
8373 break;
8374 }
8375 if (tb->cflags & CF_LAST_IO)
8376 gen_io_end();
8377 if (env->singlestep_enabled && ctx.bstate != BS_BRANCH) {
8378 save_cpu_state(&ctx, ctx.bstate == BS_NONE);
8379 gen_helper_0i(raise_exception, EXCP_DEBUG);
8380 } else {
8381 switch (ctx.bstate) {
8382 case BS_STOP:
8383 gen_helper_interrupt_restart();
8384 gen_goto_tb(&ctx, 0, ctx.pc);
8385 break;
8386 case BS_NONE:
8387 save_cpu_state(&ctx, 0);
8388 gen_goto_tb(&ctx, 0, ctx.pc);
8389 break;
8390 case BS_EXCP:
8391 gen_helper_interrupt_restart();
8392 tcg_gen_exit_tb(0);
8393 break;
8394 case BS_BRANCH:
8395 default:
8396 break;
8397 }
8398 }
8399 done_generating:
8400 gen_icount_end(tb, num_insns);
8401 *gen_opc_ptr = INDEX_op_end;
8402 if (search_pc) {
8403 j = gen_opc_ptr - gen_opc_buf;
8404 lj++;
8405 while (lj <= j)
8406 gen_opc_instr_start[lj++] = 0;
8407 } else {
8408 tb->size = ctx.pc - pc_start;
8409 tb->icount = num_insns;
8410 }
8411 #ifdef DEBUG_DISAS
8412 LOG_DISAS("\n");
8413 if (qemu_loglevel_mask(CPU_LOG_TB_IN_ASM)) {
8414 qemu_log("IN: %s\n", lookup_symbol(pc_start));
8415 log_target_disas(pc_start, ctx.pc - pc_start, 0);
8416 qemu_log("\n");
8417 }
8418 qemu_log_mask(CPU_LOG_TB_CPU, "---------------- %d %08x\n", ctx.bstate, ctx.hflags);
8419 #endif
8420 }
8421
gen_intermediate_code(CPUState * env,struct TranslationBlock * tb)8422 void gen_intermediate_code (CPUState *env, struct TranslationBlock *tb)
8423 {
8424 gen_intermediate_code_internal(env, tb, 0);
8425 }
8426
gen_intermediate_code_pc(CPUState * env,struct TranslationBlock * tb)8427 void gen_intermediate_code_pc (CPUState *env, struct TranslationBlock *tb)
8428 {
8429 gen_intermediate_code_internal(env, tb, 1);
8430 }
8431
fpu_dump_state(CPUState * env,FILE * f,int (* fpu_fprintf)(FILE * f,const char * fmt,...),int flags)8432 static void fpu_dump_state(CPUState *env, FILE *f,
8433 int (*fpu_fprintf)(FILE *f, const char *fmt, ...),
8434 int flags)
8435 {
8436 int i;
8437 int is_fpu64 = !!(env->hflags & MIPS_HFLAG_F64);
8438
8439 #define printfpr(fp) \
8440 do { \
8441 if (is_fpu64) \
8442 fpu_fprintf(f, "w:%08x d:%016lx fd:%13g fs:%13g psu: %13g\n", \
8443 (fp)->w[FP_ENDIAN_IDX], (fp)->d, (fp)->fd, \
8444 (fp)->fs[FP_ENDIAN_IDX], (fp)->fs[!FP_ENDIAN_IDX]); \
8445 else { \
8446 fpr_t tmp; \
8447 tmp.w[FP_ENDIAN_IDX] = (fp)->w[FP_ENDIAN_IDX]; \
8448 tmp.w[!FP_ENDIAN_IDX] = ((fp) + 1)->w[FP_ENDIAN_IDX]; \
8449 fpu_fprintf(f, "w:%08x d:%016lx fd:%13g fs:%13g psu:%13g\n", \
8450 tmp.w[FP_ENDIAN_IDX], tmp.d, tmp.fd, \
8451 tmp.fs[FP_ENDIAN_IDX], tmp.fs[!FP_ENDIAN_IDX]); \
8452 } \
8453 } while(0)
8454
8455
8456 fpu_fprintf(f, "CP1 FCR0 0x%08x FCR31 0x%08x SR.FR %d fp_status 0x%08x(0x%02x)\n",
8457 env->active_fpu.fcr0, env->active_fpu.fcr31, is_fpu64, env->active_fpu.fp_status,
8458 get_float_exception_flags(&env->active_fpu.fp_status));
8459 for (i = 0; i < 32; (is_fpu64) ? i++ : (i += 2)) {
8460 fpu_fprintf(f, "%3s: ", fregnames[i]);
8461 printfpr(&env->active_fpu.fpr[i]);
8462 }
8463
8464 #undef printfpr
8465 }
8466
8467 #if defined(TARGET_MIPS64) && defined(MIPS_DEBUG_SIGN_EXTENSIONS)
8468 /* Debug help: The architecture requires 32bit code to maintain proper
8469 sign-extended values on 64bit machines. */
8470
8471 #define SIGN_EXT_P(val) ((((val) & ~0x7fffffff) == 0) || (((val) & ~0x7fffffff) == ~0x7fffffff))
8472
8473 static void
cpu_mips_check_sign_extensions(CPUState * env,FILE * f,int (* cpu_fprintf)(FILE * f,const char * fmt,...),int flags)8474 cpu_mips_check_sign_extensions (CPUState *env, FILE *f,
8475 int (*cpu_fprintf)(FILE *f, const char *fmt, ...),
8476 int flags)
8477 {
8478 int i;
8479
8480 if (!SIGN_EXT_P(env->active_tc.PC))
8481 cpu_fprintf(f, "BROKEN: pc=0x" TARGET_FMT_lx "\n", env->active_tc.PC);
8482 if (!SIGN_EXT_P(env->active_tc.HI[0]))
8483 cpu_fprintf(f, "BROKEN: HI=0x" TARGET_FMT_lx "\n", env->active_tc.HI[0]);
8484 if (!SIGN_EXT_P(env->active_tc.LO[0]))
8485 cpu_fprintf(f, "BROKEN: LO=0x" TARGET_FMT_lx "\n", env->active_tc.LO[0]);
8486 if (!SIGN_EXT_P(env->btarget))
8487 cpu_fprintf(f, "BROKEN: btarget=0x" TARGET_FMT_lx "\n", env->btarget);
8488
8489 for (i = 0; i < 32; i++) {
8490 if (!SIGN_EXT_P(env->active_tc.gpr[i]))
8491 cpu_fprintf(f, "BROKEN: %s=0x" TARGET_FMT_lx "\n", regnames[i], env->active_tc.gpr[i]);
8492 }
8493
8494 if (!SIGN_EXT_P(env->CP0_EPC))
8495 cpu_fprintf(f, "BROKEN: EPC=0x" TARGET_FMT_lx "\n", env->CP0_EPC);
8496 if (!SIGN_EXT_P(env->lladdr))
8497 cpu_fprintf(f, "BROKEN: LLAddr=0x" TARGET_FMT_lx "\n", env->lladdr);
8498 }
8499 #endif
8500
cpu_dump_state(CPUState * env,FILE * f,int (* cpu_fprintf)(FILE * f,const char * fmt,...),int flags)8501 void cpu_dump_state (CPUState *env, FILE *f,
8502 int (*cpu_fprintf)(FILE *f, const char *fmt, ...),
8503 int flags)
8504 {
8505 int i;
8506
8507 cpu_fprintf(f, "pc=0x" TARGET_FMT_lx " HI=0x" TARGET_FMT_lx " LO=0x" TARGET_FMT_lx " ds %04x " TARGET_FMT_lx " %d\n",
8508 env->active_tc.PC, env->active_tc.HI[0], env->active_tc.LO[0],
8509 env->hflags, env->btarget, env->bcond);
8510 for (i = 0; i < 32; i++) {
8511 if ((i & 3) == 0)
8512 cpu_fprintf(f, "GPR%02d:", i);
8513 cpu_fprintf(f, " %s " TARGET_FMT_lx, regnames[i], env->active_tc.gpr[i]);
8514 if ((i & 3) == 3)
8515 cpu_fprintf(f, "\n");
8516 }
8517
8518 cpu_fprintf(f, "CP0 Status 0x%08x Cause 0x%08x EPC 0x" TARGET_FMT_lx "\n",
8519 env->CP0_Status, env->CP0_Cause, env->CP0_EPC);
8520 cpu_fprintf(f, " Config0 0x%08x Config1 0x%08x LLAddr 0x" TARGET_FMT_lx "\n",
8521 env->CP0_Config0, env->CP0_Config1, env->lladdr);
8522 if (env->hflags & MIPS_HFLAG_FPU)
8523 fpu_dump_state(env, f, cpu_fprintf, flags);
8524 #if defined(TARGET_MIPS64) && defined(MIPS_DEBUG_SIGN_EXTENSIONS)
8525 cpu_mips_check_sign_extensions(env, f, cpu_fprintf, flags);
8526 #endif
8527 }
8528
mips_tcg_init(void)8529 static void mips_tcg_init(void)
8530 {
8531 int i;
8532 static int inited;
8533
8534 /* Initialize various static tables. */
8535 if (inited)
8536 return;
8537
8538 cpu_env = tcg_global_reg_new_ptr(TCG_AREG0, "env");
8539 TCGV_UNUSED(cpu_gpr[0]);
8540 for (i = 1; i < 32; i++)
8541 cpu_gpr[i] = tcg_global_mem_new(TCG_AREG0,
8542 offsetof(CPUState, active_tc.gpr[i]),
8543 regnames[i]);
8544 cpu_PC = tcg_global_mem_new(TCG_AREG0,
8545 offsetof(CPUState, active_tc.PC), "PC");
8546 for (i = 0; i < MIPS_DSP_ACC; i++) {
8547 cpu_HI[i] = tcg_global_mem_new(TCG_AREG0,
8548 offsetof(CPUState, active_tc.HI[i]),
8549 regnames_HI[i]);
8550 cpu_LO[i] = tcg_global_mem_new(TCG_AREG0,
8551 offsetof(CPUState, active_tc.LO[i]),
8552 regnames_LO[i]);
8553 cpu_ACX[i] = tcg_global_mem_new(TCG_AREG0,
8554 offsetof(CPUState, active_tc.ACX[i]),
8555 regnames_ACX[i]);
8556 }
8557 cpu_dspctrl = tcg_global_mem_new(TCG_AREG0,
8558 offsetof(CPUState, active_tc.DSPControl),
8559 "DSPControl");
8560 bcond = tcg_global_mem_new(TCG_AREG0,
8561 offsetof(CPUState, bcond), "bcond");
8562 btarget = tcg_global_mem_new(TCG_AREG0,
8563 offsetof(CPUState, btarget), "btarget");
8564 hflags = tcg_global_mem_new_i32(TCG_AREG0,
8565 offsetof(CPUState, hflags), "hflags");
8566
8567 fpu_fcr0 = tcg_global_mem_new_i32(TCG_AREG0,
8568 offsetof(CPUState, active_fpu.fcr0),
8569 "fcr0");
8570 fpu_fcr31 = tcg_global_mem_new_i32(TCG_AREG0,
8571 offsetof(CPUState, active_fpu.fcr31),
8572 "fcr31");
8573
8574 /* register helpers */
8575 #define GEN_HELPER 2
8576 #include "helper.h"
8577
8578 inited = 1;
8579 }
8580
8581 #include "translate_init.c"
8582
cpu_mips_init(const char * cpu_model)8583 CPUMIPSState *cpu_mips_init (const char *cpu_model)
8584 {
8585 CPUMIPSState *env;
8586 const mips_def_t *def;
8587
8588 def = cpu_mips_find_by_name(cpu_model);
8589 if (!def)
8590 return NULL;
8591 env = qemu_mallocz(sizeof(CPUMIPSState));
8592 env->cpu_model = def;
8593 env->cpu_model_str = cpu_model;
8594
8595 cpu_exec_init(env);
8596 #ifndef CONFIG_USER_ONLY
8597 mmu_init(env, def);
8598 #endif
8599 mvp_init(env, def);
8600 mips_tcg_init();
8601 cpu_reset(env);
8602 qemu_init_vcpu(env);
8603 return env;
8604 }
8605
cpu_reset(CPUMIPSState * env)8606 void cpu_reset (CPUMIPSState *env)
8607 {
8608 if (qemu_loglevel_mask(CPU_LOG_RESET)) {
8609 qemu_log("CPU Reset (CPU %d)\n", env->cpu_index);
8610 log_cpu_state(env, 0);
8611 }
8612
8613 memset(env, 0, offsetof(CPUMIPSState, breakpoints));
8614 tlb_flush(env, 1);
8615
8616 /* Reset registers to their default values */
8617 env->CP0_PRid = env->cpu_model->CP0_PRid;
8618 env->CP0_Config0 = env->cpu_model->CP0_Config0;
8619 #ifdef TARGET_WORDS_BIGENDIAN
8620 env->CP0_Config0 |= (1 << CP0C0_BE);
8621 #endif
8622 env->CP0_Config1 = env->cpu_model->CP0_Config1;
8623 env->CP0_Config2 = env->cpu_model->CP0_Config2;
8624 env->CP0_Config3 = env->cpu_model->CP0_Config3;
8625 env->CP0_Config6 = env->cpu_model->CP0_Config6;
8626 env->CP0_Config7 = env->cpu_model->CP0_Config7;
8627 env->CP0_LLAddr_rw_bitmask = env->cpu_model->CP0_LLAddr_rw_bitmask
8628 << env->cpu_model->CP0_LLAddr_shift;
8629 env->CP0_LLAddr_shift = env->cpu_model->CP0_LLAddr_shift;
8630 env->SYNCI_Step = env->cpu_model->SYNCI_Step;
8631 env->CCRes = env->cpu_model->CCRes;
8632 env->CP0_Status_rw_bitmask = env->cpu_model->CP0_Status_rw_bitmask;
8633 env->CP0_TCStatus_rw_bitmask = env->cpu_model->CP0_TCStatus_rw_bitmask;
8634 env->CP0_SRSCtl = env->cpu_model->CP0_SRSCtl;
8635 env->current_tc = 0;
8636 env->SEGBITS = env->cpu_model->SEGBITS;
8637 env->SEGMask = (target_ulong)((1ULL << env->cpu_model->SEGBITS) - 1);
8638 #if defined(TARGET_MIPS64)
8639 if (env->cpu_model->insn_flags & ISA_MIPS3) {
8640 env->SEGMask |= 3ULL << 62;
8641 }
8642 #endif
8643 env->PABITS = env->cpu_model->PABITS;
8644 env->PAMask = (target_ulong)((1ULL << env->cpu_model->PABITS) - 1);
8645 env->CP0_SRSConf0_rw_bitmask = env->cpu_model->CP0_SRSConf0_rw_bitmask;
8646 env->CP0_SRSConf0 = env->cpu_model->CP0_SRSConf0;
8647 env->CP0_SRSConf1_rw_bitmask = env->cpu_model->CP0_SRSConf1_rw_bitmask;
8648 env->CP0_SRSConf1 = env->cpu_model->CP0_SRSConf1;
8649 env->CP0_SRSConf2_rw_bitmask = env->cpu_model->CP0_SRSConf2_rw_bitmask;
8650 env->CP0_SRSConf2 = env->cpu_model->CP0_SRSConf2;
8651 env->CP0_SRSConf3_rw_bitmask = env->cpu_model->CP0_SRSConf3_rw_bitmask;
8652 env->CP0_SRSConf3 = env->cpu_model->CP0_SRSConf3;
8653 env->CP0_SRSConf4_rw_bitmask = env->cpu_model->CP0_SRSConf4_rw_bitmask;
8654 env->CP0_SRSConf4 = env->cpu_model->CP0_SRSConf4;
8655 env->insn_flags = env->cpu_model->insn_flags;
8656
8657 fpu_init(env, env->cpu_model);
8658
8659 #if defined(CONFIG_USER_ONLY)
8660 env->hflags = MIPS_HFLAG_UM;
8661 /* Enable access to the SYNCI_Step register. */
8662 env->CP0_HWREna |= (1 << 1);
8663 if (env->CP0_Config1 & (1 << CP0C1_FP)) {
8664 env->hflags |= MIPS_HFLAG_FPU;
8665 }
8666 #ifdef TARGET_MIPS64
8667 if (env->active_fpu.fcr0 & (1 << FCR0_F64)) {
8668 env->hflags |= MIPS_HFLAG_F64;
8669 }
8670 #endif
8671 #else
8672 if (env->hflags & MIPS_HFLAG_BMASK) {
8673 /* If the exception was raised from a delay slot,
8674 come back to the jump. */
8675 env->CP0_ErrorEPC = env->active_tc.PC - 4;
8676 } else {
8677 env->CP0_ErrorEPC = env->active_tc.PC;
8678 }
8679 env->active_tc.PC = (int32_t)0xBFC00000;
8680 env->CP0_Random = env->tlb->nb_tlb - 1;
8681 env->CP0_Wired = 0;
8682 /* SMP not implemented */
8683 env->CP0_EBase = 0x80000000;
8684 env->CP0_Status = (1 << CP0St_BEV) | (1 << CP0St_ERL);
8685 /* vectored interrupts not implemented, timer on int 7,
8686 no performance counters. */
8687 env->CP0_IntCtl = 0xe0000000;
8688 {
8689 int i;
8690
8691 for (i = 0; i < 7; i++) {
8692 env->CP0_WatchLo[i] = 0;
8693 env->CP0_WatchHi[i] = 0x80000000;
8694 }
8695 env->CP0_WatchLo[7] = 0;
8696 env->CP0_WatchHi[7] = 0;
8697 }
8698 /* Count register increments in debug mode, EJTAG version 1 */
8699 env->CP0_Debug = (1 << CP0DB_CNT) | (0x1 << CP0DB_VER);
8700 env->hflags = MIPS_HFLAG_CP0;
8701 #endif
8702 #if defined(TARGET_MIPS64)
8703 if (env->cpu_model->insn_flags & ISA_MIPS3) {
8704 env->hflags |= MIPS_HFLAG_64;
8705 }
8706 #endif
8707 env->exception_index = EXCP_NONE;
8708 }
8709
restore_state_to_opc(CPUState * env,TranslationBlock * tb,int pc_pos)8710 void restore_state_to_opc(CPUState *env, TranslationBlock *tb, int pc_pos)
8711 {
8712 env->active_tc.PC = gen_opc_pc[pc_pos];
8713 env->hflags &= ~MIPS_HFLAG_BMASK;
8714 env->hflags |= gen_opc_hflags[pc_pos];
8715 }
8716