]> git.ipfire.org Git - thirdparty/qemu.git/blame - target-sparc/translate.c
Merge remote-tracking branch 'pmaydell/arm-devs.for-upstream' into staging
[thirdparty/qemu.git] / target-sparc / translate.c
CommitLineData
7a3f1944
FB
1/*
2 SPARC translation
3
4 Copyright (C) 2003 Thomas M. Ogrisegg <tom@fnord.at>
3475187d 5 Copyright (C) 2003-2005 Fabrice Bellard
7a3f1944
FB
6
7 This library is free software; you can redistribute it and/or
8 modify it under the terms of the GNU Lesser General Public
9 License as published by the Free Software Foundation; either
10 version 2 of the License, or (at your option) any later version.
11
12 This library is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
16
17 You should have received a copy of the GNU Lesser General Public
8167ee88 18 License along with this library; if not, see <http://www.gnu.org/licenses/>.
7a3f1944
FB
19 */
20
7a3f1944
FB
21#include <stdarg.h>
22#include <stdlib.h>
23#include <stdio.h>
24#include <string.h>
25#include <inttypes.h>
26
27#include "cpu.h"
7a3f1944 28#include "disas.h"
1a2fb1c0 29#include "helper.h"
57fec1fe 30#include "tcg-op.h"
7a3f1944 31
a7812ae4
PB
32#define GEN_HELPER 1
33#include "helper.h"
34
7a3f1944
FB
35#define DEBUG_DISAS
36
72cbca10
FB
37#define DYNAMIC_PC 1 /* dynamic pc value */
38#define JUMP_PC 2 /* dynamic pc value which takes only two values
39 according to jump_pc[T2] */
40
1a2fb1c0 41/* global register indexes */
a7812ae4 42static TCGv_ptr cpu_env, cpu_regwptr;
25517f99
PB
43static TCGv cpu_cc_src, cpu_cc_src2, cpu_cc_dst;
44static TCGv_i32 cpu_cc_op;
a7812ae4
PB
45static TCGv_i32 cpu_psr;
46static TCGv cpu_fsr, cpu_pc, cpu_npc, cpu_gregs[8];
255e1fcb
BS
47static TCGv cpu_y;
48#ifndef CONFIG_USER_ONLY
49static TCGv cpu_tbr;
50#endif
42a8aa83 51static TCGv cpu_cond, cpu_dst, cpu_addr, cpu_val;
dc99a3f2 52#ifdef TARGET_SPARC64
a7812ae4
PB
53static TCGv_i32 cpu_xcc, cpu_asi, cpu_fprs;
54static TCGv cpu_gsr;
255e1fcb 55static TCGv cpu_tick_cmpr, cpu_stick_cmpr, cpu_hstick_cmpr;
a7812ae4
PB
56static TCGv cpu_hintp, cpu_htba, cpu_hver, cpu_ssr, cpu_ver;
57static TCGv_i32 cpu_softint;
255e1fcb
BS
58#else
59static TCGv cpu_wim;
dc99a3f2 60#endif
1a2fb1c0 61/* local register indexes (only used inside old micro ops) */
a7812ae4
PB
62static TCGv cpu_tmp0;
63static TCGv_i32 cpu_tmp32;
64static TCGv_i64 cpu_tmp64;
714547bb 65/* Floating point registers */
30038fd8 66static TCGv_i64 cpu_fpr[TARGET_DPREGS];
1a2fb1c0 67
1a7ff922
PB
68static target_ulong gen_opc_npc[OPC_BUF_SIZE];
69static target_ulong gen_opc_jump_pc[2];
70
2e70f6ef
PB
71#include "gen-icount.h"
72
7a3f1944 73typedef struct DisasContext {
0f8a249a
BS
74 target_ulong pc; /* current Program Counter: integer or DYNAMIC_PC */
75 target_ulong npc; /* next PC: integer or DYNAMIC_PC or JUMP_PC */
72cbca10 76 target_ulong jump_pc[2]; /* used when JUMP_PC pc value is used */
cf495bcf 77 int is_br;
e8af50a3 78 int mem_idx;
a80dde08 79 int fpu_enabled;
2cade6a3 80 int address_mask_32bit;
060718c1 81 int singlestep;
8393617c 82 uint32_t cc_op; /* current CC operation */
cf495bcf 83 struct TranslationBlock *tb;
5578ceab 84 sparc_def_t *def;
30038fd8
RH
85 TCGv_i32 t32[3];
86 int n_t32;
7a3f1944
FB
87} DisasContext;
88
416fcaea
RH
89typedef struct {
90 TCGCond cond;
91 bool is_bool;
92 bool g1, g2;
93 TCGv c1, c2;
94} DisasCompare;
95
3475187d 96// This function uses non-native bit order
dc1a6971
BS
97#define GET_FIELD(X, FROM, TO) \
98 ((X) >> (31 - (TO)) & ((1 << ((TO) - (FROM) + 1)) - 1))
7a3f1944 99
3475187d 100// This function uses the order in the manuals, i.e. bit 0 is 2^0
dc1a6971 101#define GET_FIELD_SP(X, FROM, TO) \
3475187d
FB
102 GET_FIELD(X, 31 - (TO), 31 - (FROM))
103
104#define GET_FIELDs(x,a,b) sign_extend (GET_FIELD(x,a,b), (b) - (a) + 1)
46d38ba8 105#define GET_FIELD_SPs(x,a,b) sign_extend (GET_FIELD_SP(x,a,b), ((b) - (a) + 1))
3475187d
FB
106
107#ifdef TARGET_SPARC64
0387d928 108#define DFPREG(r) (((r & 1) << 5) | (r & 0x1e))
1f587329 109#define QFPREG(r) (((r & 1) << 5) | (r & 0x1c))
3475187d 110#else
c185970a 111#define DFPREG(r) (r & 0x1e)
1f587329 112#define QFPREG(r) (r & 0x1c)
3475187d
FB
113#endif
114
b158a785
BS
115#define UA2005_HTRAP_MASK 0xff
116#define V8_TRAP_MASK 0x7f
117
3475187d
FB
118static int sign_extend(int x, int len)
119{
120 len = 32 - len;
121 return (x << len) >> len;
122}
123
7a3f1944
FB
124#define IS_IMM (insn & (1<<13))
125
141ae5c1
RH
126static inline void gen_update_fprs_dirty(int rd)
127{
128#if defined(TARGET_SPARC64)
129 tcg_gen_ori_i32(cpu_fprs, cpu_fprs, (rd < 32) ? 1 : 2);
130#endif
131}
132
ff07ec83 133/* floating point registers moves */
208ae657
RH
134static TCGv_i32 gen_load_fpr_F(DisasContext *dc, unsigned int src)
135{
30038fd8
RH
136#if TCG_TARGET_REG_BITS == 32
137 if (src & 1) {
138 return TCGV_LOW(cpu_fpr[src / 2]);
139 } else {
140 return TCGV_HIGH(cpu_fpr[src / 2]);
141 }
142#else
143 if (src & 1) {
144 return MAKE_TCGV_I32(GET_TCGV_I64(cpu_fpr[src / 2]));
145 } else {
146 TCGv_i32 ret = tcg_temp_local_new_i32();
147 TCGv_i64 t = tcg_temp_new_i64();
148
149 tcg_gen_shri_i64(t, cpu_fpr[src / 2], 32);
150 tcg_gen_trunc_i64_i32(ret, t);
151 tcg_temp_free_i64(t);
152
153 dc->t32[dc->n_t32++] = ret;
154 assert(dc->n_t32 <= ARRAY_SIZE(dc->t32));
155
156 return ret;
157 }
158#endif
208ae657
RH
159}
160
161static void gen_store_fpr_F(DisasContext *dc, unsigned int dst, TCGv_i32 v)
162{
30038fd8
RH
163#if TCG_TARGET_REG_BITS == 32
164 if (dst & 1) {
165 tcg_gen_mov_i32(TCGV_LOW(cpu_fpr[dst / 2]), v);
166 } else {
167 tcg_gen_mov_i32(TCGV_HIGH(cpu_fpr[dst / 2]), v);
168 }
169#else
170 TCGv_i64 t = MAKE_TCGV_I64(GET_TCGV_I32(v));
171 tcg_gen_deposit_i64(cpu_fpr[dst / 2], cpu_fpr[dst / 2], t,
172 (dst & 1 ? 0 : 32), 32);
173#endif
141ae5c1 174 gen_update_fprs_dirty(dst);
208ae657
RH
175}
176
177static TCGv_i32 gen_dest_fpr_F(void)
178{
179 return cpu_tmp32;
180}
181
96eda024
RH
182static TCGv_i64 gen_load_fpr_D(DisasContext *dc, unsigned int src)
183{
96eda024 184 src = DFPREG(src);
30038fd8 185 return cpu_fpr[src / 2];
96eda024
RH
186}
187
188static void gen_store_fpr_D(DisasContext *dc, unsigned int dst, TCGv_i64 v)
189{
190 dst = DFPREG(dst);
30038fd8 191 tcg_gen_mov_i64(cpu_fpr[dst / 2], v);
96eda024
RH
192 gen_update_fprs_dirty(dst);
193}
194
195static TCGv_i64 gen_dest_fpr_D(void)
196{
197 return cpu_tmp64;
198}
199
ff07ec83
BS
200static void gen_op_load_fpr_QT0(unsigned int src)
201{
30038fd8
RH
202 tcg_gen_st_i64(cpu_fpr[src / 2], cpu_env, offsetof(CPUSPARCState, qt0) +
203 offsetof(CPU_QuadU, ll.upper));
204 tcg_gen_st_i64(cpu_fpr[src/2 + 1], cpu_env, offsetof(CPUSPARCState, qt0) +
205 offsetof(CPU_QuadU, ll.lower));
ff07ec83
BS
206}
207
208static void gen_op_load_fpr_QT1(unsigned int src)
209{
30038fd8
RH
210 tcg_gen_st_i64(cpu_fpr[src / 2], cpu_env, offsetof(CPUSPARCState, qt1) +
211 offsetof(CPU_QuadU, ll.upper));
212 tcg_gen_st_i64(cpu_fpr[src/2 + 1], cpu_env, offsetof(CPUSPARCState, qt1) +
213 offsetof(CPU_QuadU, ll.lower));
ff07ec83
BS
214}
215
216static void gen_op_store_QT0_fpr(unsigned int dst)
217{
30038fd8
RH
218 tcg_gen_ld_i64(cpu_fpr[dst / 2], cpu_env, offsetof(CPUSPARCState, qt0) +
219 offsetof(CPU_QuadU, ll.upper));
220 tcg_gen_ld_i64(cpu_fpr[dst/2 + 1], cpu_env, offsetof(CPUSPARCState, qt0) +
221 offsetof(CPU_QuadU, ll.lower));
ff07ec83 222}
1f587329 223
ac11f776 224#ifdef TARGET_SPARC64
30038fd8 225static void gen_move_Q(unsigned int rd, unsigned int rs)
ac11f776
RH
226{
227 rd = QFPREG(rd);
228 rs = QFPREG(rs);
229
30038fd8
RH
230 tcg_gen_mov_i64(cpu_fpr[rd / 2], cpu_fpr[rs / 2]);
231 tcg_gen_mov_i64(cpu_fpr[rd / 2 + 1], cpu_fpr[rs / 2 + 1]);
ac11f776
RH
232 gen_update_fprs_dirty(rd);
233}
234#endif
235
81ad8ba2
BS
236/* moves */
237#ifdef CONFIG_USER_ONLY
3475187d 238#define supervisor(dc) 0
81ad8ba2 239#ifdef TARGET_SPARC64
e9ebed4d 240#define hypervisor(dc) 0
81ad8ba2 241#endif
3475187d 242#else
2aae2b8e 243#define supervisor(dc) (dc->mem_idx >= MMU_KERNEL_IDX)
81ad8ba2 244#ifdef TARGET_SPARC64
2aae2b8e 245#define hypervisor(dc) (dc->mem_idx == MMU_HYPV_IDX)
6f27aba6 246#else
3475187d 247#endif
81ad8ba2
BS
248#endif
249
2cade6a3
BS
250#ifdef TARGET_SPARC64
251#ifndef TARGET_ABI32
252#define AM_CHECK(dc) ((dc)->address_mask_32bit)
1a2fb1c0 253#else
2cade6a3
BS
254#define AM_CHECK(dc) (1)
255#endif
1a2fb1c0 256#endif
3391c818 257
2cade6a3
BS
258static inline void gen_address_mask(DisasContext *dc, TCGv addr)
259{
260#ifdef TARGET_SPARC64
261 if (AM_CHECK(dc))
262 tcg_gen_andi_tl(addr, addr, 0xffffffffULL);
263#endif
264}
265
1a2fb1c0 266static inline void gen_movl_reg_TN(int reg, TCGv tn)
81ad8ba2 267{
1a2fb1c0
BS
268 if (reg == 0)
269 tcg_gen_movi_tl(tn, 0);
270 else if (reg < 8)
f5069b26 271 tcg_gen_mov_tl(tn, cpu_gregs[reg]);
1a2fb1c0 272 else {
1a2fb1c0 273 tcg_gen_ld_tl(tn, cpu_regwptr, (reg - 8) * sizeof(target_ulong));
81ad8ba2
BS
274 }
275}
276
1a2fb1c0 277static inline void gen_movl_TN_reg(int reg, TCGv tn)
81ad8ba2 278{
1a2fb1c0
BS
279 if (reg == 0)
280 return;
281 else if (reg < 8)
f5069b26 282 tcg_gen_mov_tl(cpu_gregs[reg], tn);
1a2fb1c0 283 else {
1a2fb1c0 284 tcg_gen_st_tl(tn, cpu_regwptr, (reg - 8) * sizeof(target_ulong));
81ad8ba2
BS
285 }
286}
287
5fafdf24 288static inline void gen_goto_tb(DisasContext *s, int tb_num,
6e256c93
FB
289 target_ulong pc, target_ulong npc)
290{
291 TranslationBlock *tb;
292
293 tb = s->tb;
294 if ((pc & TARGET_PAGE_MASK) == (tb->pc & TARGET_PAGE_MASK) &&
060718c1
RH
295 (npc & TARGET_PAGE_MASK) == (tb->pc & TARGET_PAGE_MASK) &&
296 !s->singlestep) {
6e256c93 297 /* jump to same page: we can use a direct jump */
57fec1fe 298 tcg_gen_goto_tb(tb_num);
2f5680ee
BS
299 tcg_gen_movi_tl(cpu_pc, pc);
300 tcg_gen_movi_tl(cpu_npc, npc);
4b4a72e5 301 tcg_gen_exit_tb((tcg_target_long)tb + tb_num);
6e256c93
FB
302 } else {
303 /* jump to another page: currently not optimized */
2f5680ee
BS
304 tcg_gen_movi_tl(cpu_pc, pc);
305 tcg_gen_movi_tl(cpu_npc, npc);
57fec1fe 306 tcg_gen_exit_tb(0);
6e256c93
FB
307 }
308}
309
19f329ad 310// XXX suboptimal
a7812ae4 311static inline void gen_mov_reg_N(TCGv reg, TCGv_i32 src)
19f329ad 312{
8911f501 313 tcg_gen_extu_i32_tl(reg, src);
4b8b8b76 314 tcg_gen_shri_tl(reg, reg, PSR_NEG_SHIFT);
19f329ad
BS
315 tcg_gen_andi_tl(reg, reg, 0x1);
316}
317
a7812ae4 318static inline void gen_mov_reg_Z(TCGv reg, TCGv_i32 src)
19f329ad 319{
8911f501 320 tcg_gen_extu_i32_tl(reg, src);
4b8b8b76 321 tcg_gen_shri_tl(reg, reg, PSR_ZERO_SHIFT);
19f329ad
BS
322 tcg_gen_andi_tl(reg, reg, 0x1);
323}
324
a7812ae4 325static inline void gen_mov_reg_V(TCGv reg, TCGv_i32 src)
19f329ad 326{
8911f501 327 tcg_gen_extu_i32_tl(reg, src);
4b8b8b76 328 tcg_gen_shri_tl(reg, reg, PSR_OVF_SHIFT);
19f329ad
BS
329 tcg_gen_andi_tl(reg, reg, 0x1);
330}
331
a7812ae4 332static inline void gen_mov_reg_C(TCGv reg, TCGv_i32 src)
19f329ad 333{
8911f501 334 tcg_gen_extu_i32_tl(reg, src);
4b8b8b76 335 tcg_gen_shri_tl(reg, reg, PSR_CARRY_SHIFT);
19f329ad
BS
336 tcg_gen_andi_tl(reg, reg, 0x1);
337}
338
41d72852
BS
339static inline void gen_op_addi_cc(TCGv dst, TCGv src1, target_long src2)
340{
341 tcg_gen_mov_tl(cpu_cc_src, src1);
342 tcg_gen_movi_tl(cpu_cc_src2, src2);
343 tcg_gen_addi_tl(cpu_cc_dst, cpu_cc_src, src2);
bdf9f35d 344 tcg_gen_mov_tl(dst, cpu_cc_dst);
41d72852
BS
345}
346
4af984a7 347static inline void gen_op_add_cc(TCGv dst, TCGv src1, TCGv src2)
dc99a3f2 348{
4af984a7 349 tcg_gen_mov_tl(cpu_cc_src, src1);
6f551262 350 tcg_gen_mov_tl(cpu_cc_src2, src2);
5c6a0628 351 tcg_gen_add_tl(cpu_cc_dst, cpu_cc_src, cpu_cc_src2);
bdf9f35d 352 tcg_gen_mov_tl(dst, cpu_cc_dst);
41d72852
BS
353}
354
70c48285 355static TCGv_i32 gen_add32_carry32(void)
dc99a3f2 356{
70c48285
RH
357 TCGv_i32 carry_32, cc_src1_32, cc_src2_32;
358
359 /* Carry is computed from a previous add: (dst < src) */
360#if TARGET_LONG_BITS == 64
361 cc_src1_32 = tcg_temp_new_i32();
362 cc_src2_32 = tcg_temp_new_i32();
363 tcg_gen_trunc_i64_i32(cc_src1_32, cpu_cc_dst);
364 tcg_gen_trunc_i64_i32(cc_src2_32, cpu_cc_src);
365#else
366 cc_src1_32 = cpu_cc_dst;
367 cc_src2_32 = cpu_cc_src;
368#endif
369
370 carry_32 = tcg_temp_new_i32();
371 tcg_gen_setcond_i32(TCG_COND_LTU, carry_32, cc_src1_32, cc_src2_32);
372
373#if TARGET_LONG_BITS == 64
374 tcg_temp_free_i32(cc_src1_32);
375 tcg_temp_free_i32(cc_src2_32);
376#endif
377
378 return carry_32;
41d72852
BS
379}
380
70c48285 381static TCGv_i32 gen_sub32_carry32(void)
41d72852 382{
70c48285
RH
383 TCGv_i32 carry_32, cc_src1_32, cc_src2_32;
384
385 /* Carry is computed from a previous borrow: (src1 < src2) */
386#if TARGET_LONG_BITS == 64
387 cc_src1_32 = tcg_temp_new_i32();
388 cc_src2_32 = tcg_temp_new_i32();
389 tcg_gen_trunc_i64_i32(cc_src1_32, cpu_cc_src);
390 tcg_gen_trunc_i64_i32(cc_src2_32, cpu_cc_src2);
391#else
392 cc_src1_32 = cpu_cc_src;
393 cc_src2_32 = cpu_cc_src2;
394#endif
395
396 carry_32 = tcg_temp_new_i32();
397 tcg_gen_setcond_i32(TCG_COND_LTU, carry_32, cc_src1_32, cc_src2_32);
398
399#if TARGET_LONG_BITS == 64
400 tcg_temp_free_i32(cc_src1_32);
401 tcg_temp_free_i32(cc_src2_32);
402#endif
403
404 return carry_32;
405}
406
407static void gen_op_addx_int(DisasContext *dc, TCGv dst, TCGv src1,
408 TCGv src2, int update_cc)
409{
410 TCGv_i32 carry_32;
411 TCGv carry;
412
413 switch (dc->cc_op) {
414 case CC_OP_DIV:
415 case CC_OP_LOGIC:
416 /* Carry is known to be zero. Fall back to plain ADD. */
417 if (update_cc) {
418 gen_op_add_cc(dst, src1, src2);
419 } else {
420 tcg_gen_add_tl(dst, src1, src2);
421 }
422 return;
423
424 case CC_OP_ADD:
425 case CC_OP_TADD:
426 case CC_OP_TADDTV:
427#if TCG_TARGET_REG_BITS == 32 && TARGET_LONG_BITS == 32
428 {
429 /* For 32-bit hosts, we can re-use the host's hardware carry
430 generation by using an ADD2 opcode. We discard the low
431 part of the output. Ideally we'd combine this operation
432 with the add that generated the carry in the first place. */
433 TCGv dst_low = tcg_temp_new();
434 tcg_gen_op6_i32(INDEX_op_add2_i32, dst_low, dst,
435 cpu_cc_src, src1, cpu_cc_src2, src2);
436 tcg_temp_free(dst_low);
437 goto add_done;
438 }
439#endif
440 carry_32 = gen_add32_carry32();
441 break;
442
443 case CC_OP_SUB:
444 case CC_OP_TSUB:
445 case CC_OP_TSUBTV:
446 carry_32 = gen_sub32_carry32();
447 break;
448
449 default:
450 /* We need external help to produce the carry. */
451 carry_32 = tcg_temp_new_i32();
2ffd9176 452 gen_helper_compute_C_icc(carry_32, cpu_env);
70c48285
RH
453 break;
454 }
455
456#if TARGET_LONG_BITS == 64
457 carry = tcg_temp_new();
458 tcg_gen_extu_i32_i64(carry, carry_32);
459#else
460 carry = carry_32;
461#endif
462
463 tcg_gen_add_tl(dst, src1, src2);
464 tcg_gen_add_tl(dst, dst, carry);
465
466 tcg_temp_free_i32(carry_32);
467#if TARGET_LONG_BITS == 64
468 tcg_temp_free(carry);
469#endif
470
471#if TCG_TARGET_REG_BITS == 32 && TARGET_LONG_BITS == 32
472 add_done:
473#endif
474 if (update_cc) {
475 tcg_gen_mov_tl(cpu_cc_src, src1);
476 tcg_gen_mov_tl(cpu_cc_src2, src2);
477 tcg_gen_mov_tl(cpu_cc_dst, dst);
478 tcg_gen_movi_i32(cpu_cc_op, CC_OP_ADDX);
479 dc->cc_op = CC_OP_ADDX;
480 }
dc99a3f2
BS
481}
482
d4b0d468 483static inline void gen_op_subi_cc(TCGv dst, TCGv src1, target_long src2, DisasContext *dc)
41d72852
BS
484{
485 tcg_gen_mov_tl(cpu_cc_src, src1);
486 tcg_gen_movi_tl(cpu_cc_src2, src2);
719f66a7 487 if (src2 == 0) {
d4b0d468
BS
488 tcg_gen_mov_tl(cpu_cc_dst, src1);
489 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
490 dc->cc_op = CC_OP_LOGIC;
719f66a7
BS
491 } else {
492 tcg_gen_subi_tl(cpu_cc_dst, cpu_cc_src, src2);
d4b0d468
BS
493 tcg_gen_movi_i32(cpu_cc_op, CC_OP_SUB);
494 dc->cc_op = CC_OP_SUB;
719f66a7 495 }
d4b0d468 496 tcg_gen_mov_tl(dst, cpu_cc_dst);
41d72852
BS
497}
498
499static inline void gen_op_sub_cc(TCGv dst, TCGv src1, TCGv src2)
dc99a3f2 500{
4af984a7 501 tcg_gen_mov_tl(cpu_cc_src, src1);
6f551262 502 tcg_gen_mov_tl(cpu_cc_src2, src2);
41d72852 503 tcg_gen_sub_tl(cpu_cc_dst, cpu_cc_src, cpu_cc_src2);
d4b0d468 504 tcg_gen_mov_tl(dst, cpu_cc_dst);
41d72852
BS
505}
506
70c48285
RH
507static void gen_op_subx_int(DisasContext *dc, TCGv dst, TCGv src1,
508 TCGv src2, int update_cc)
41d72852 509{
70c48285
RH
510 TCGv_i32 carry_32;
511 TCGv carry;
41d72852 512
70c48285
RH
513 switch (dc->cc_op) {
514 case CC_OP_DIV:
515 case CC_OP_LOGIC:
516 /* Carry is known to be zero. Fall back to plain SUB. */
517 if (update_cc) {
518 gen_op_sub_cc(dst, src1, src2);
519 } else {
520 tcg_gen_sub_tl(dst, src1, src2);
521 }
522 return;
523
524 case CC_OP_ADD:
525 case CC_OP_TADD:
526 case CC_OP_TADDTV:
527 carry_32 = gen_add32_carry32();
528 break;
529
530 case CC_OP_SUB:
531 case CC_OP_TSUB:
532 case CC_OP_TSUBTV:
533#if TCG_TARGET_REG_BITS == 32 && TARGET_LONG_BITS == 32
534 {
535 /* For 32-bit hosts, we can re-use the host's hardware carry
536 generation by using a SUB2 opcode. We discard the low
537 part of the output. Ideally we'd combine this operation
538 with the add that generated the carry in the first place. */
539 TCGv dst_low = tcg_temp_new();
540 tcg_gen_op6_i32(INDEX_op_sub2_i32, dst_low, dst,
541 cpu_cc_src, src1, cpu_cc_src2, src2);
542 tcg_temp_free(dst_low);
543 goto sub_done;
544 }
545#endif
546 carry_32 = gen_sub32_carry32();
547 break;
548
549 default:
550 /* We need external help to produce the carry. */
551 carry_32 = tcg_temp_new_i32();
2ffd9176 552 gen_helper_compute_C_icc(carry_32, cpu_env);
70c48285
RH
553 break;
554 }
555
556#if TARGET_LONG_BITS == 64
557 carry = tcg_temp_new();
558 tcg_gen_extu_i32_i64(carry, carry_32);
559#else
560 carry = carry_32;
561#endif
562
563 tcg_gen_sub_tl(dst, src1, src2);
564 tcg_gen_sub_tl(dst, dst, carry);
565
566 tcg_temp_free_i32(carry_32);
567#if TARGET_LONG_BITS == 64
568 tcg_temp_free(carry);
569#endif
570
571#if TCG_TARGET_REG_BITS == 32 && TARGET_LONG_BITS == 32
572 sub_done:
573#endif
574 if (update_cc) {
575 tcg_gen_mov_tl(cpu_cc_src, src1);
576 tcg_gen_mov_tl(cpu_cc_src2, src2);
577 tcg_gen_mov_tl(cpu_cc_dst, dst);
578 tcg_gen_movi_i32(cpu_cc_op, CC_OP_SUBX);
579 dc->cc_op = CC_OP_SUBX;
580 }
dc99a3f2
BS
581}
582
4af984a7 583static inline void gen_op_mulscc(TCGv dst, TCGv src1, TCGv src2)
d9bdab86 584{
6cb675b0 585 TCGv r_temp, zero;
d9bdab86 586
a7812ae4 587 r_temp = tcg_temp_new();
d9bdab86
BS
588
589 /* old op:
590 if (!(env->y & 1))
591 T1 = 0;
592 */
6cb675b0 593 zero = tcg_const_tl(0);
72ccba79 594 tcg_gen_andi_tl(cpu_cc_src, src1, 0xffffffff);
255e1fcb 595 tcg_gen_andi_tl(r_temp, cpu_y, 0x1);
72ccba79 596 tcg_gen_andi_tl(cpu_cc_src2, src2, 0xffffffff);
6cb675b0
RH
597 tcg_gen_movcond_tl(TCG_COND_EQ, cpu_cc_src2, r_temp, zero,
598 zero, cpu_cc_src2);
599 tcg_temp_free(zero);
d9bdab86
BS
600
601 // b2 = T0 & 1;
602 // env->y = (b2 << 31) | (env->y >> 1);
105a1f04
BS
603 tcg_gen_andi_tl(r_temp, cpu_cc_src, 0x1);
604 tcg_gen_shli_tl(r_temp, r_temp, 31);
255e1fcb 605 tcg_gen_shri_tl(cpu_tmp0, cpu_y, 1);
72ccba79 606 tcg_gen_andi_tl(cpu_tmp0, cpu_tmp0, 0x7fffffff);
5068cbd9
BS
607 tcg_gen_or_tl(cpu_tmp0, cpu_tmp0, r_temp);
608 tcg_gen_andi_tl(cpu_y, cpu_tmp0, 0xffffffff);
d9bdab86
BS
609
610 // b1 = N ^ V;
611 gen_mov_reg_N(cpu_tmp0, cpu_psr);
612 gen_mov_reg_V(r_temp, cpu_psr);
613 tcg_gen_xor_tl(cpu_tmp0, cpu_tmp0, r_temp);
2ea815ca 614 tcg_temp_free(r_temp);
d9bdab86
BS
615
616 // T0 = (b1 << 31) | (T0 >> 1);
617 // src1 = T0;
618 tcg_gen_shli_tl(cpu_tmp0, cpu_tmp0, 31);
6f551262 619 tcg_gen_shri_tl(cpu_cc_src, cpu_cc_src, 1);
d9bdab86
BS
620 tcg_gen_or_tl(cpu_cc_src, cpu_cc_src, cpu_tmp0);
621
5c6a0628 622 tcg_gen_add_tl(cpu_cc_dst, cpu_cc_src, cpu_cc_src2);
d9bdab86 623
5c6a0628 624 tcg_gen_mov_tl(dst, cpu_cc_dst);
d9bdab86
BS
625}
626
fb170183 627static inline void gen_op_multiply(TCGv dst, TCGv src1, TCGv src2, int sign_ext)
8879d139 628{
fb170183 629 TCGv_i32 r_src1, r_src2;
a7812ae4 630 TCGv_i64 r_temp, r_temp2;
8879d139 631
fb170183
IK
632 r_src1 = tcg_temp_new_i32();
633 r_src2 = tcg_temp_new_i32();
634
635 tcg_gen_trunc_tl_i32(r_src1, src1);
636 tcg_gen_trunc_tl_i32(r_src2, src2);
637
a7812ae4
PB
638 r_temp = tcg_temp_new_i64();
639 r_temp2 = tcg_temp_new_i64();
8879d139 640
fb170183
IK
641 if (sign_ext) {
642 tcg_gen_ext_i32_i64(r_temp, r_src2);
643 tcg_gen_ext_i32_i64(r_temp2, r_src1);
644 } else {
645 tcg_gen_extu_i32_i64(r_temp, r_src2);
646 tcg_gen_extu_i32_i64(r_temp2, r_src1);
647 }
648
8879d139
BS
649 tcg_gen_mul_i64(r_temp2, r_temp, r_temp2);
650
651 tcg_gen_shri_i64(r_temp, r_temp2, 32);
105a1f04 652 tcg_gen_trunc_i64_tl(cpu_tmp0, r_temp);
a7812ae4 653 tcg_temp_free_i64(r_temp);
255e1fcb 654 tcg_gen_andi_tl(cpu_y, cpu_tmp0, 0xffffffff);
fb170183 655
4af984a7 656 tcg_gen_trunc_i64_tl(dst, r_temp2);
fb170183 657
a7812ae4 658 tcg_temp_free_i64(r_temp2);
fb170183
IK
659
660 tcg_temp_free_i32(r_src1);
661 tcg_temp_free_i32(r_src2);
8879d139
BS
662}
663
fb170183 664static inline void gen_op_umul(TCGv dst, TCGv src1, TCGv src2)
8879d139 665{
fb170183
IK
666 /* zero-extend truncated operands before multiplication */
667 gen_op_multiply(dst, src1, src2, 0);
668}
8879d139 669
fb170183
IK
670static inline void gen_op_smul(TCGv dst, TCGv src1, TCGv src2)
671{
672 /* sign-extend truncated operands before multiplication */
673 gen_op_multiply(dst, src1, src2, 1);
8879d139
BS
674}
675
19f329ad
BS
676// 1
677static inline void gen_op_eval_ba(TCGv dst)
678{
679 tcg_gen_movi_tl(dst, 1);
680}
681
682// Z
a7812ae4 683static inline void gen_op_eval_be(TCGv dst, TCGv_i32 src)
19f329ad
BS
684{
685 gen_mov_reg_Z(dst, src);
686}
687
688// Z | (N ^ V)
a7812ae4 689static inline void gen_op_eval_ble(TCGv dst, TCGv_i32 src)
19f329ad 690{
0425bee5 691 gen_mov_reg_N(cpu_tmp0, src);
19f329ad 692 gen_mov_reg_V(dst, src);
0425bee5
BS
693 tcg_gen_xor_tl(dst, dst, cpu_tmp0);
694 gen_mov_reg_Z(cpu_tmp0, src);
695 tcg_gen_or_tl(dst, dst, cpu_tmp0);
19f329ad
BS
696}
697
698// N ^ V
a7812ae4 699static inline void gen_op_eval_bl(TCGv dst, TCGv_i32 src)
19f329ad 700{
0425bee5 701 gen_mov_reg_V(cpu_tmp0, src);
19f329ad 702 gen_mov_reg_N(dst, src);
0425bee5 703 tcg_gen_xor_tl(dst, dst, cpu_tmp0);
19f329ad
BS
704}
705
706// C | Z
a7812ae4 707static inline void gen_op_eval_bleu(TCGv dst, TCGv_i32 src)
19f329ad 708{
0425bee5 709 gen_mov_reg_Z(cpu_tmp0, src);
19f329ad 710 gen_mov_reg_C(dst, src);
0425bee5 711 tcg_gen_or_tl(dst, dst, cpu_tmp0);
19f329ad
BS
712}
713
714// C
a7812ae4 715static inline void gen_op_eval_bcs(TCGv dst, TCGv_i32 src)
19f329ad
BS
716{
717 gen_mov_reg_C(dst, src);
718}
719
720// V
a7812ae4 721static inline void gen_op_eval_bvs(TCGv dst, TCGv_i32 src)
19f329ad
BS
722{
723 gen_mov_reg_V(dst, src);
724}
725
726// 0
727static inline void gen_op_eval_bn(TCGv dst)
728{
729 tcg_gen_movi_tl(dst, 0);
730}
731
732// N
a7812ae4 733static inline void gen_op_eval_bneg(TCGv dst, TCGv_i32 src)
19f329ad
BS
734{
735 gen_mov_reg_N(dst, src);
736}
737
738// !Z
a7812ae4 739static inline void gen_op_eval_bne(TCGv dst, TCGv_i32 src)
19f329ad
BS
740{
741 gen_mov_reg_Z(dst, src);
742 tcg_gen_xori_tl(dst, dst, 0x1);
743}
744
745// !(Z | (N ^ V))
a7812ae4 746static inline void gen_op_eval_bg(TCGv dst, TCGv_i32 src)
19f329ad 747{
0425bee5 748 gen_mov_reg_N(cpu_tmp0, src);
19f329ad 749 gen_mov_reg_V(dst, src);
0425bee5
BS
750 tcg_gen_xor_tl(dst, dst, cpu_tmp0);
751 gen_mov_reg_Z(cpu_tmp0, src);
752 tcg_gen_or_tl(dst, dst, cpu_tmp0);
19f329ad
BS
753 tcg_gen_xori_tl(dst, dst, 0x1);
754}
755
756// !(N ^ V)
a7812ae4 757static inline void gen_op_eval_bge(TCGv dst, TCGv_i32 src)
19f329ad 758{
0425bee5 759 gen_mov_reg_V(cpu_tmp0, src);
19f329ad 760 gen_mov_reg_N(dst, src);
0425bee5 761 tcg_gen_xor_tl(dst, dst, cpu_tmp0);
19f329ad
BS
762 tcg_gen_xori_tl(dst, dst, 0x1);
763}
764
765// !(C | Z)
a7812ae4 766static inline void gen_op_eval_bgu(TCGv dst, TCGv_i32 src)
19f329ad 767{
0425bee5 768 gen_mov_reg_Z(cpu_tmp0, src);
19f329ad 769 gen_mov_reg_C(dst, src);
0425bee5 770 tcg_gen_or_tl(dst, dst, cpu_tmp0);
19f329ad
BS
771 tcg_gen_xori_tl(dst, dst, 0x1);
772}
773
774// !C
a7812ae4 775static inline void gen_op_eval_bcc(TCGv dst, TCGv_i32 src)
19f329ad
BS
776{
777 gen_mov_reg_C(dst, src);
778 tcg_gen_xori_tl(dst, dst, 0x1);
779}
780
781// !N
a7812ae4 782static inline void gen_op_eval_bpos(TCGv dst, TCGv_i32 src)
19f329ad
BS
783{
784 gen_mov_reg_N(dst, src);
785 tcg_gen_xori_tl(dst, dst, 0x1);
786}
787
788// !V
a7812ae4 789static inline void gen_op_eval_bvc(TCGv dst, TCGv_i32 src)
19f329ad
BS
790{
791 gen_mov_reg_V(dst, src);
792 tcg_gen_xori_tl(dst, dst, 0x1);
793}
794
795/*
796 FPSR bit field FCC1 | FCC0:
797 0 =
798 1 <
799 2 >
800 3 unordered
801*/
802static inline void gen_mov_reg_FCC0(TCGv reg, TCGv src,
803 unsigned int fcc_offset)
804{
ba6a9d8c 805 tcg_gen_shri_tl(reg, src, FSR_FCC0_SHIFT + fcc_offset);
19f329ad
BS
806 tcg_gen_andi_tl(reg, reg, 0x1);
807}
808
809static inline void gen_mov_reg_FCC1(TCGv reg, TCGv src,
810 unsigned int fcc_offset)
811{
ba6a9d8c 812 tcg_gen_shri_tl(reg, src, FSR_FCC1_SHIFT + fcc_offset);
19f329ad
BS
813 tcg_gen_andi_tl(reg, reg, 0x1);
814}
815
816// !0: FCC0 | FCC1
817static inline void gen_op_eval_fbne(TCGv dst, TCGv src,
818 unsigned int fcc_offset)
819{
19f329ad 820 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
821 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
822 tcg_gen_or_tl(dst, dst, cpu_tmp0);
19f329ad
BS
823}
824
825// 1 or 2: FCC0 ^ FCC1
826static inline void gen_op_eval_fblg(TCGv dst, TCGv src,
827 unsigned int fcc_offset)
828{
19f329ad 829 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
830 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
831 tcg_gen_xor_tl(dst, dst, cpu_tmp0);
19f329ad
BS
832}
833
834// 1 or 3: FCC0
835static inline void gen_op_eval_fbul(TCGv dst, TCGv src,
836 unsigned int fcc_offset)
837{
838 gen_mov_reg_FCC0(dst, src, fcc_offset);
839}
840
841// 1: FCC0 & !FCC1
842static inline void gen_op_eval_fbl(TCGv dst, TCGv src,
843 unsigned int fcc_offset)
844{
19f329ad 845 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
846 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
847 tcg_gen_xori_tl(cpu_tmp0, cpu_tmp0, 0x1);
848 tcg_gen_and_tl(dst, dst, cpu_tmp0);
19f329ad
BS
849}
850
851// 2 or 3: FCC1
852static inline void gen_op_eval_fbug(TCGv dst, TCGv src,
853 unsigned int fcc_offset)
854{
855 gen_mov_reg_FCC1(dst, src, fcc_offset);
856}
857
858// 2: !FCC0 & FCC1
859static inline void gen_op_eval_fbg(TCGv dst, TCGv src,
860 unsigned int fcc_offset)
861{
19f329ad
BS
862 gen_mov_reg_FCC0(dst, src, fcc_offset);
863 tcg_gen_xori_tl(dst, dst, 0x1);
0425bee5
BS
864 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
865 tcg_gen_and_tl(dst, dst, cpu_tmp0);
19f329ad
BS
866}
867
868// 3: FCC0 & FCC1
869static inline void gen_op_eval_fbu(TCGv dst, TCGv src,
870 unsigned int fcc_offset)
871{
19f329ad 872 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
873 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
874 tcg_gen_and_tl(dst, dst, cpu_tmp0);
19f329ad
BS
875}
876
877// 0: !(FCC0 | FCC1)
878static inline void gen_op_eval_fbe(TCGv dst, TCGv src,
879 unsigned int fcc_offset)
880{
19f329ad 881 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
882 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
883 tcg_gen_or_tl(dst, dst, cpu_tmp0);
19f329ad
BS
884 tcg_gen_xori_tl(dst, dst, 0x1);
885}
886
887// 0 or 3: !(FCC0 ^ FCC1)
888static inline void gen_op_eval_fbue(TCGv dst, TCGv src,
889 unsigned int fcc_offset)
890{
19f329ad 891 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
892 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
893 tcg_gen_xor_tl(dst, dst, cpu_tmp0);
19f329ad
BS
894 tcg_gen_xori_tl(dst, dst, 0x1);
895}
896
897// 0 or 2: !FCC0
898static inline void gen_op_eval_fbge(TCGv dst, TCGv src,
899 unsigned int fcc_offset)
900{
901 gen_mov_reg_FCC0(dst, src, fcc_offset);
902 tcg_gen_xori_tl(dst, dst, 0x1);
903}
904
905// !1: !(FCC0 & !FCC1)
906static inline void gen_op_eval_fbuge(TCGv dst, TCGv src,
907 unsigned int fcc_offset)
908{
19f329ad 909 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
910 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
911 tcg_gen_xori_tl(cpu_tmp0, cpu_tmp0, 0x1);
912 tcg_gen_and_tl(dst, dst, cpu_tmp0);
19f329ad
BS
913 tcg_gen_xori_tl(dst, dst, 0x1);
914}
915
916// 0 or 1: !FCC1
917static inline void gen_op_eval_fble(TCGv dst, TCGv src,
918 unsigned int fcc_offset)
919{
920 gen_mov_reg_FCC1(dst, src, fcc_offset);
921 tcg_gen_xori_tl(dst, dst, 0x1);
922}
923
924// !2: !(!FCC0 & FCC1)
925static inline void gen_op_eval_fbule(TCGv dst, TCGv src,
926 unsigned int fcc_offset)
927{
19f329ad
BS
928 gen_mov_reg_FCC0(dst, src, fcc_offset);
929 tcg_gen_xori_tl(dst, dst, 0x1);
0425bee5
BS
930 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
931 tcg_gen_and_tl(dst, dst, cpu_tmp0);
19f329ad
BS
932 tcg_gen_xori_tl(dst, dst, 0x1);
933}
934
935// !3: !(FCC0 & FCC1)
936static inline void gen_op_eval_fbo(TCGv dst, TCGv src,
937 unsigned int fcc_offset)
938{
19f329ad 939 gen_mov_reg_FCC0(dst, src, fcc_offset);
0425bee5
BS
940 gen_mov_reg_FCC1(cpu_tmp0, src, fcc_offset);
941 tcg_gen_and_tl(dst, dst, cpu_tmp0);
19f329ad
BS
942 tcg_gen_xori_tl(dst, dst, 0x1);
943}
944
46525e1f 945static inline void gen_branch2(DisasContext *dc, target_ulong pc1,
19f329ad 946 target_ulong pc2, TCGv r_cond)
83469015
FB
947{
948 int l1;
949
950 l1 = gen_new_label();
951
cb63669a 952 tcg_gen_brcondi_tl(TCG_COND_EQ, r_cond, 0, l1);
83469015 953
6e256c93 954 gen_goto_tb(dc, 0, pc1, pc1 + 4);
83469015
FB
955
956 gen_set_label(l1);
6e256c93 957 gen_goto_tb(dc, 1, pc2, pc2 + 4);
83469015
FB
958}
959
46525e1f 960static inline void gen_branch_a(DisasContext *dc, target_ulong pc1,
19f329ad 961 target_ulong pc2, TCGv r_cond)
83469015
FB
962{
963 int l1;
964
965 l1 = gen_new_label();
966
cb63669a 967 tcg_gen_brcondi_tl(TCG_COND_EQ, r_cond, 0, l1);
83469015 968
6e256c93 969 gen_goto_tb(dc, 0, pc2, pc1);
83469015
FB
970
971 gen_set_label(l1);
6e256c93 972 gen_goto_tb(dc, 1, pc2 + 4, pc2 + 8);
83469015
FB
973}
974
2e655fe7 975static inline void gen_generic_branch(DisasContext *dc)
83469015 976{
61316742
RH
977 TCGv npc0 = tcg_const_tl(dc->jump_pc[0]);
978 TCGv npc1 = tcg_const_tl(dc->jump_pc[1]);
979 TCGv zero = tcg_const_tl(0);
19f329ad 980
61316742 981 tcg_gen_movcond_tl(TCG_COND_NE, cpu_npc, cpu_cond, zero, npc0, npc1);
83469015 982
61316742
RH
983 tcg_temp_free(npc0);
984 tcg_temp_free(npc1);
985 tcg_temp_free(zero);
83469015
FB
986}
987
4af984a7
BS
988/* call this function before using the condition register as it may
989 have been set for a jump */
dee8913c 990static inline void flush_cond(DisasContext *dc)
83469015
FB
991{
992 if (dc->npc == JUMP_PC) {
2e655fe7 993 gen_generic_branch(dc);
83469015
FB
994 dc->npc = DYNAMIC_PC;
995 }
996}
997
934da7ee 998static inline void save_npc(DisasContext *dc)
72cbca10
FB
999{
1000 if (dc->npc == JUMP_PC) {
2e655fe7 1001 gen_generic_branch(dc);
72cbca10
FB
1002 dc->npc = DYNAMIC_PC;
1003 } else if (dc->npc != DYNAMIC_PC) {
2f5680ee 1004 tcg_gen_movi_tl(cpu_npc, dc->npc);
72cbca10
FB
1005 }
1006}
1007
66442b07 1008static inline void save_state(DisasContext *dc)
72cbca10 1009{
2f5680ee 1010 tcg_gen_movi_tl(cpu_pc, dc->pc);
cfa90513
BS
1011 /* flush pending conditional evaluations before exposing cpu state */
1012 if (dc->cc_op != CC_OP_FLAGS) {
1013 dc->cc_op = CC_OP_FLAGS;
2ffd9176 1014 gen_helper_compute_psr(cpu_env);
cfa90513 1015 }
934da7ee 1016 save_npc(dc);
72cbca10
FB
1017}
1018
13a6dd00 1019static inline void gen_mov_pc_npc(DisasContext *dc)
0bee699e
FB
1020{
1021 if (dc->npc == JUMP_PC) {
2e655fe7 1022 gen_generic_branch(dc);
48d5c82b 1023 tcg_gen_mov_tl(cpu_pc, cpu_npc);
0bee699e
FB
1024 dc->pc = DYNAMIC_PC;
1025 } else if (dc->npc == DYNAMIC_PC) {
48d5c82b 1026 tcg_gen_mov_tl(cpu_pc, cpu_npc);
0bee699e
FB
1027 dc->pc = DYNAMIC_PC;
1028 } else {
1029 dc->pc = dc->npc;
1030 }
1031}
1032
38bc628b
BS
1033static inline void gen_op_next_insn(void)
1034{
48d5c82b
BS
1035 tcg_gen_mov_tl(cpu_pc, cpu_npc);
1036 tcg_gen_addi_tl(cpu_npc, cpu_npc, 4);
38bc628b
BS
1037}
1038
416fcaea
RH
1039static void free_compare(DisasCompare *cmp)
1040{
1041 if (!cmp->g1) {
1042 tcg_temp_free(cmp->c1);
1043 }
1044 if (!cmp->g2) {
1045 tcg_temp_free(cmp->c2);
1046 }
1047}
1048
2a484ecf 1049static void gen_compare(DisasCompare *cmp, bool xcc, unsigned int cond,
416fcaea 1050 DisasContext *dc)
19f329ad 1051{
2a484ecf
RH
1052 static int subcc_cond[16] = {
1053 -1, /* never */
1054 TCG_COND_EQ,
1055 TCG_COND_LE,
1056 TCG_COND_LT,
1057 TCG_COND_LEU,
1058 TCG_COND_LTU,
1059 -1, /* neg */
1060 -1, /* overflow */
1061 -1, /* always */
1062 TCG_COND_NE,
1063 TCG_COND_GT,
1064 TCG_COND_GE,
1065 TCG_COND_GTU,
1066 TCG_COND_GEU,
1067 -1, /* pos */
1068 -1, /* no overflow */
1069 };
1070
a7812ae4 1071 TCGv_i32 r_src;
416fcaea
RH
1072 TCGv r_dst;
1073
3475187d 1074#ifdef TARGET_SPARC64
2a484ecf 1075 if (xcc) {
dc99a3f2 1076 r_src = cpu_xcc;
2a484ecf 1077 } else {
dc99a3f2 1078 r_src = cpu_psr;
2a484ecf 1079 }
3475187d 1080#else
dc99a3f2 1081 r_src = cpu_psr;
3475187d 1082#endif
2a484ecf 1083
8393617c 1084 switch (dc->cc_op) {
2a484ecf
RH
1085 case CC_OP_SUB:
1086 switch (cond) {
1087 case 6: /* neg */
1088 case 14: /* pos */
1089 cmp->cond = (cond == 6 ? TCG_COND_LT : TCG_COND_GE);
1090 cmp->is_bool = false;
1091 cmp->g2 = false;
1092 cmp->c2 = tcg_const_tl(0);
1093#ifdef TARGET_SPARC64
1094 if (!xcc) {
1095 cmp->g1 = false;
1096 cmp->c1 = tcg_temp_new();
1097 tcg_gen_ext32s_tl(cmp->c1, cpu_cc_dst);
1098 break;
1099 }
1100#endif
1101 cmp->g1 = true;
1102 cmp->c1 = cpu_cc_dst;
1103 break;
1104
1105 case 0: /* never */
1106 case 8: /* always */
1107 case 7: /* overflow */
1108 case 15: /* !overflow */
1109 goto do_dynamic;
1110
1111 default:
1112 cmp->cond = subcc_cond[cond];
1113 cmp->is_bool = false;
1114#ifdef TARGET_SPARC64
1115 if (!xcc) {
1116 /* Note that sign-extension works for unsigned compares as
1117 long as both operands are sign-extended. */
1118 cmp->g1 = cmp->g2 = false;
1119 cmp->c1 = tcg_temp_new();
1120 cmp->c2 = tcg_temp_new();
1121 tcg_gen_ext32s_tl(cmp->c1, cpu_cc_src);
1122 tcg_gen_ext32s_tl(cmp->c2, cpu_cc_src2);
1123 }
1124#endif
1125 cmp->g1 = cmp->g2 = true;
1126 cmp->c1 = cpu_cc_src;
1127 cmp->c2 = cpu_cc_src2;
1128 break;
1129 }
8393617c 1130 break;
2a484ecf 1131
8393617c 1132 default:
2a484ecf 1133 do_dynamic:
2ffd9176 1134 gen_helper_compute_psr(cpu_env);
8393617c 1135 dc->cc_op = CC_OP_FLAGS;
2a484ecf
RH
1136 /* FALLTHRU */
1137
1138 case CC_OP_FLAGS:
1139 /* We're going to generate a boolean result. */
1140 cmp->cond = TCG_COND_NE;
1141 cmp->is_bool = true;
1142 cmp->g1 = cmp->g2 = false;
1143 cmp->c1 = r_dst = tcg_temp_new();
1144 cmp->c2 = tcg_const_tl(0);
1145
1146 switch (cond) {
1147 case 0x0:
1148 gen_op_eval_bn(r_dst);
1149 break;
1150 case 0x1:
1151 gen_op_eval_be(r_dst, r_src);
1152 break;
1153 case 0x2:
1154 gen_op_eval_ble(r_dst, r_src);
1155 break;
1156 case 0x3:
1157 gen_op_eval_bl(r_dst, r_src);
1158 break;
1159 case 0x4:
1160 gen_op_eval_bleu(r_dst, r_src);
1161 break;
1162 case 0x5:
1163 gen_op_eval_bcs(r_dst, r_src);
1164 break;
1165 case 0x6:
1166 gen_op_eval_bneg(r_dst, r_src);
1167 break;
1168 case 0x7:
1169 gen_op_eval_bvs(r_dst, r_src);
1170 break;
1171 case 0x8:
1172 gen_op_eval_ba(r_dst);
1173 break;
1174 case 0x9:
1175 gen_op_eval_bne(r_dst, r_src);
1176 break;
1177 case 0xa:
1178 gen_op_eval_bg(r_dst, r_src);
1179 break;
1180 case 0xb:
1181 gen_op_eval_bge(r_dst, r_src);
1182 break;
1183 case 0xc:
1184 gen_op_eval_bgu(r_dst, r_src);
1185 break;
1186 case 0xd:
1187 gen_op_eval_bcc(r_dst, r_src);
1188 break;
1189 case 0xe:
1190 gen_op_eval_bpos(r_dst, r_src);
1191 break;
1192 case 0xf:
1193 gen_op_eval_bvc(r_dst, r_src);
1194 break;
1195 }
19f329ad
BS
1196 break;
1197 }
1198}
7a3f1944 1199
416fcaea 1200static void gen_fcompare(DisasCompare *cmp, unsigned int cc, unsigned int cond)
e8af50a3 1201{
19f329ad 1202 unsigned int offset;
416fcaea
RH
1203 TCGv r_dst;
1204
1205 /* For now we still generate a straight boolean result. */
1206 cmp->cond = TCG_COND_NE;
1207 cmp->is_bool = true;
1208 cmp->g1 = cmp->g2 = false;
1209 cmp->c1 = r_dst = tcg_temp_new();
1210 cmp->c2 = tcg_const_tl(0);
19f329ad 1211
19f329ad
BS
1212 switch (cc) {
1213 default:
1214 case 0x0:
1215 offset = 0;
1216 break;
1217 case 0x1:
1218 offset = 32 - 10;
1219 break;
1220 case 0x2:
1221 offset = 34 - 10;
1222 break;
1223 case 0x3:
1224 offset = 36 - 10;
1225 break;
1226 }
1227
1228 switch (cond) {
1229 case 0x0:
1230 gen_op_eval_bn(r_dst);
1231 break;
1232 case 0x1:
87e92502 1233 gen_op_eval_fbne(r_dst, cpu_fsr, offset);
19f329ad
BS
1234 break;
1235 case 0x2:
87e92502 1236 gen_op_eval_fblg(r_dst, cpu_fsr, offset);
19f329ad
BS
1237 break;
1238 case 0x3:
87e92502 1239 gen_op_eval_fbul(r_dst, cpu_fsr, offset);
19f329ad
BS
1240 break;
1241 case 0x4:
87e92502 1242 gen_op_eval_fbl(r_dst, cpu_fsr, offset);
19f329ad
BS
1243 break;
1244 case 0x5:
87e92502 1245 gen_op_eval_fbug(r_dst, cpu_fsr, offset);
19f329ad
BS
1246 break;
1247 case 0x6:
87e92502 1248 gen_op_eval_fbg(r_dst, cpu_fsr, offset);
19f329ad
BS
1249 break;
1250 case 0x7:
87e92502 1251 gen_op_eval_fbu(r_dst, cpu_fsr, offset);
19f329ad
BS
1252 break;
1253 case 0x8:
1254 gen_op_eval_ba(r_dst);
1255 break;
1256 case 0x9:
87e92502 1257 gen_op_eval_fbe(r_dst, cpu_fsr, offset);
19f329ad
BS
1258 break;
1259 case 0xa:
87e92502 1260 gen_op_eval_fbue(r_dst, cpu_fsr, offset);
19f329ad
BS
1261 break;
1262 case 0xb:
87e92502 1263 gen_op_eval_fbge(r_dst, cpu_fsr, offset);
19f329ad
BS
1264 break;
1265 case 0xc:
87e92502 1266 gen_op_eval_fbuge(r_dst, cpu_fsr, offset);
19f329ad
BS
1267 break;
1268 case 0xd:
87e92502 1269 gen_op_eval_fble(r_dst, cpu_fsr, offset);
19f329ad
BS
1270 break;
1271 case 0xe:
87e92502 1272 gen_op_eval_fbule(r_dst, cpu_fsr, offset);
19f329ad
BS
1273 break;
1274 case 0xf:
87e92502 1275 gen_op_eval_fbo(r_dst, cpu_fsr, offset);
19f329ad
BS
1276 break;
1277 }
e8af50a3 1278}
00f219bf 1279
416fcaea
RH
1280static void gen_cond(TCGv r_dst, unsigned int cc, unsigned int cond,
1281 DisasContext *dc)
1282{
1283 DisasCompare cmp;
1284 gen_compare(&cmp, cc, cond, dc);
1285
1286 /* The interface is to return a boolean in r_dst. */
1287 if (cmp.is_bool) {
1288 tcg_gen_mov_tl(r_dst, cmp.c1);
1289 } else {
1290 tcg_gen_setcond_tl(cmp.cond, r_dst, cmp.c1, cmp.c2);
1291 }
1292
1293 free_compare(&cmp);
1294}
1295
1296static void gen_fcond(TCGv r_dst, unsigned int cc, unsigned int cond)
1297{
1298 DisasCompare cmp;
1299 gen_fcompare(&cmp, cc, cond);
1300
1301 /* The interface is to return a boolean in r_dst. */
1302 if (cmp.is_bool) {
1303 tcg_gen_mov_tl(r_dst, cmp.c1);
1304 } else {
1305 tcg_gen_setcond_tl(cmp.cond, r_dst, cmp.c1, cmp.c2);
1306 }
1307
1308 free_compare(&cmp);
1309}
1310
19f329ad 1311#ifdef TARGET_SPARC64
00f219bf
BS
1312// Inverted logic
1313static const int gen_tcg_cond_reg[8] = {
1314 -1,
1315 TCG_COND_NE,
1316 TCG_COND_GT,
1317 TCG_COND_GE,
1318 -1,
1319 TCG_COND_EQ,
1320 TCG_COND_LE,
1321 TCG_COND_LT,
1322};
19f329ad 1323
416fcaea
RH
1324static void gen_compare_reg(DisasCompare *cmp, int cond, TCGv r_src)
1325{
1326 cmp->cond = tcg_invert_cond(gen_tcg_cond_reg[cond]);
1327 cmp->is_bool = false;
1328 cmp->g1 = true;
1329 cmp->g2 = false;
1330 cmp->c1 = r_src;
1331 cmp->c2 = tcg_const_tl(0);
1332}
1333
4af984a7 1334static inline void gen_cond_reg(TCGv r_dst, int cond, TCGv r_src)
19f329ad 1335{
416fcaea
RH
1336 DisasCompare cmp;
1337 gen_compare_reg(&cmp, cond, r_src);
19f329ad 1338
416fcaea
RH
1339 /* The interface is to return a boolean in r_dst. */
1340 tcg_gen_setcond_tl(cmp.cond, r_dst, cmp.c1, cmp.c2);
1341
1342 free_compare(&cmp);
19f329ad 1343}
3475187d 1344#endif
cf495bcf 1345
d4a288ef 1346static void do_branch(DisasContext *dc, int32_t offset, uint32_t insn, int cc)
7a3f1944 1347{
cf495bcf 1348 unsigned int cond = GET_FIELD(insn, 3, 6), a = (insn & (1 << 29));
af7bf89b 1349 target_ulong target = dc->pc + offset;
5fafdf24 1350
22036a49
AT
1351#ifdef TARGET_SPARC64
1352 if (unlikely(AM_CHECK(dc))) {
1353 target &= 0xffffffffULL;
1354 }
1355#endif
cf495bcf 1356 if (cond == 0x0) {
0f8a249a
BS
1357 /* unconditional not taken */
1358 if (a) {
1359 dc->pc = dc->npc + 4;
1360 dc->npc = dc->pc + 4;
1361 } else {
1362 dc->pc = dc->npc;
1363 dc->npc = dc->pc + 4;
1364 }
cf495bcf 1365 } else if (cond == 0x8) {
0f8a249a
BS
1366 /* unconditional taken */
1367 if (a) {
1368 dc->pc = target;
1369 dc->npc = dc->pc + 4;
1370 } else {
1371 dc->pc = dc->npc;
1372 dc->npc = target;
c27e2752 1373 tcg_gen_mov_tl(cpu_pc, cpu_npc);
0f8a249a 1374 }
cf495bcf 1375 } else {
dee8913c 1376 flush_cond(dc);
d4a288ef 1377 gen_cond(cpu_cond, cc, cond, dc);
0f8a249a 1378 if (a) {
d4a288ef 1379 gen_branch_a(dc, target, dc->npc, cpu_cond);
cf495bcf 1380 dc->is_br = 1;
0f8a249a 1381 } else {
cf495bcf 1382 dc->pc = dc->npc;
72cbca10 1383 dc->jump_pc[0] = target;
548f66db
AT
1384 if (unlikely(dc->npc == DYNAMIC_PC)) {
1385 dc->jump_pc[1] = DYNAMIC_PC;
1386 tcg_gen_addi_tl(cpu_pc, cpu_npc, 4);
1387 } else {
1388 dc->jump_pc[1] = dc->npc + 4;
1389 dc->npc = JUMP_PC;
1390 }
0f8a249a 1391 }
cf495bcf 1392 }
7a3f1944
FB
1393}
1394
d4a288ef 1395static void do_fbranch(DisasContext *dc, int32_t offset, uint32_t insn, int cc)
e8af50a3
FB
1396{
1397 unsigned int cond = GET_FIELD(insn, 3, 6), a = (insn & (1 << 29));
af7bf89b
FB
1398 target_ulong target = dc->pc + offset;
1399
22036a49
AT
1400#ifdef TARGET_SPARC64
1401 if (unlikely(AM_CHECK(dc))) {
1402 target &= 0xffffffffULL;
1403 }
1404#endif
e8af50a3 1405 if (cond == 0x0) {
0f8a249a
BS
1406 /* unconditional not taken */
1407 if (a) {
1408 dc->pc = dc->npc + 4;
1409 dc->npc = dc->pc + 4;
1410 } else {
1411 dc->pc = dc->npc;
1412 dc->npc = dc->pc + 4;
1413 }
e8af50a3 1414 } else if (cond == 0x8) {
0f8a249a
BS
1415 /* unconditional taken */
1416 if (a) {
1417 dc->pc = target;
1418 dc->npc = dc->pc + 4;
1419 } else {
1420 dc->pc = dc->npc;
1421 dc->npc = target;
c27e2752 1422 tcg_gen_mov_tl(cpu_pc, cpu_npc);
0f8a249a 1423 }
e8af50a3 1424 } else {
dee8913c 1425 flush_cond(dc);
d4a288ef 1426 gen_fcond(cpu_cond, cc, cond);
0f8a249a 1427 if (a) {
d4a288ef 1428 gen_branch_a(dc, target, dc->npc, cpu_cond);
e8af50a3 1429 dc->is_br = 1;
0f8a249a 1430 } else {
e8af50a3
FB
1431 dc->pc = dc->npc;
1432 dc->jump_pc[0] = target;
548f66db
AT
1433 if (unlikely(dc->npc == DYNAMIC_PC)) {
1434 dc->jump_pc[1] = DYNAMIC_PC;
1435 tcg_gen_addi_tl(cpu_pc, cpu_npc, 4);
1436 } else {
1437 dc->jump_pc[1] = dc->npc + 4;
1438 dc->npc = JUMP_PC;
1439 }
0f8a249a 1440 }
e8af50a3
FB
1441 }
1442}
1443
3475187d 1444#ifdef TARGET_SPARC64
4af984a7 1445static void do_branch_reg(DisasContext *dc, int32_t offset, uint32_t insn,
d4a288ef 1446 TCGv r_reg)
7a3f1944 1447{
3475187d
FB
1448 unsigned int cond = GET_FIELD_SP(insn, 25, 27), a = (insn & (1 << 29));
1449 target_ulong target = dc->pc + offset;
1450
22036a49
AT
1451 if (unlikely(AM_CHECK(dc))) {
1452 target &= 0xffffffffULL;
1453 }
dee8913c 1454 flush_cond(dc);
d4a288ef 1455 gen_cond_reg(cpu_cond, cond, r_reg);
3475187d 1456 if (a) {
d4a288ef 1457 gen_branch_a(dc, target, dc->npc, cpu_cond);
0f8a249a 1458 dc->is_br = 1;
3475187d 1459 } else {
0f8a249a
BS
1460 dc->pc = dc->npc;
1461 dc->jump_pc[0] = target;
548f66db
AT
1462 if (unlikely(dc->npc == DYNAMIC_PC)) {
1463 dc->jump_pc[1] = DYNAMIC_PC;
1464 tcg_gen_addi_tl(cpu_pc, cpu_npc, 4);
1465 } else {
1466 dc->jump_pc[1] = dc->npc + 4;
1467 dc->npc = JUMP_PC;
1468 }
3475187d 1469 }
7a3f1944
FB
1470}
1471
a7812ae4 1472static inline void gen_op_fcmps(int fccno, TCGv_i32 r_rs1, TCGv_i32 r_rs2)
7e8c2b6c 1473{
714547bb
BS
1474 switch (fccno) {
1475 case 0:
2e2f4ade 1476 gen_helper_fcmps(cpu_env, r_rs1, r_rs2);
714547bb
BS
1477 break;
1478 case 1:
2e2f4ade 1479 gen_helper_fcmps_fcc1(cpu_env, r_rs1, r_rs2);
714547bb
BS
1480 break;
1481 case 2:
2e2f4ade 1482 gen_helper_fcmps_fcc2(cpu_env, r_rs1, r_rs2);
714547bb
BS
1483 break;
1484 case 3:
2e2f4ade 1485 gen_helper_fcmps_fcc3(cpu_env, r_rs1, r_rs2);
714547bb
BS
1486 break;
1487 }
7e8c2b6c
BS
1488}
1489
03fb8cfc 1490static inline void gen_op_fcmpd(int fccno, TCGv_i64 r_rs1, TCGv_i64 r_rs2)
7e8c2b6c 1491{
a7812ae4
PB
1492 switch (fccno) {
1493 case 0:
03fb8cfc 1494 gen_helper_fcmpd(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1495 break;
1496 case 1:
03fb8cfc 1497 gen_helper_fcmpd_fcc1(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1498 break;
1499 case 2:
03fb8cfc 1500 gen_helper_fcmpd_fcc2(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1501 break;
1502 case 3:
03fb8cfc 1503 gen_helper_fcmpd_fcc3(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1504 break;
1505 }
7e8c2b6c
BS
1506}
1507
7e8c2b6c
BS
1508static inline void gen_op_fcmpq(int fccno)
1509{
a7812ae4
PB
1510 switch (fccno) {
1511 case 0:
2e2f4ade 1512 gen_helper_fcmpq(cpu_env);
a7812ae4
PB
1513 break;
1514 case 1:
2e2f4ade 1515 gen_helper_fcmpq_fcc1(cpu_env);
a7812ae4
PB
1516 break;
1517 case 2:
2e2f4ade 1518 gen_helper_fcmpq_fcc2(cpu_env);
a7812ae4
PB
1519 break;
1520 case 3:
2e2f4ade 1521 gen_helper_fcmpq_fcc3(cpu_env);
a7812ae4
PB
1522 break;
1523 }
7e8c2b6c 1524}
7e8c2b6c 1525
a7812ae4 1526static inline void gen_op_fcmpes(int fccno, TCGv_i32 r_rs1, TCGv_i32 r_rs2)
7e8c2b6c 1527{
714547bb
BS
1528 switch (fccno) {
1529 case 0:
2e2f4ade 1530 gen_helper_fcmpes(cpu_env, r_rs1, r_rs2);
714547bb
BS
1531 break;
1532 case 1:
2e2f4ade 1533 gen_helper_fcmpes_fcc1(cpu_env, r_rs1, r_rs2);
714547bb
BS
1534 break;
1535 case 2:
2e2f4ade 1536 gen_helper_fcmpes_fcc2(cpu_env, r_rs1, r_rs2);
714547bb
BS
1537 break;
1538 case 3:
2e2f4ade 1539 gen_helper_fcmpes_fcc3(cpu_env, r_rs1, r_rs2);
714547bb
BS
1540 break;
1541 }
7e8c2b6c
BS
1542}
1543
03fb8cfc 1544static inline void gen_op_fcmped(int fccno, TCGv_i64 r_rs1, TCGv_i64 r_rs2)
7e8c2b6c 1545{
a7812ae4
PB
1546 switch (fccno) {
1547 case 0:
03fb8cfc 1548 gen_helper_fcmped(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1549 break;
1550 case 1:
03fb8cfc 1551 gen_helper_fcmped_fcc1(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1552 break;
1553 case 2:
03fb8cfc 1554 gen_helper_fcmped_fcc2(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1555 break;
1556 case 3:
03fb8cfc 1557 gen_helper_fcmped_fcc3(cpu_env, r_rs1, r_rs2);
a7812ae4
PB
1558 break;
1559 }
7e8c2b6c
BS
1560}
1561
7e8c2b6c
BS
1562static inline void gen_op_fcmpeq(int fccno)
1563{
a7812ae4
PB
1564 switch (fccno) {
1565 case 0:
2e2f4ade 1566 gen_helper_fcmpeq(cpu_env);
a7812ae4
PB
1567 break;
1568 case 1:
2e2f4ade 1569 gen_helper_fcmpeq_fcc1(cpu_env);
a7812ae4
PB
1570 break;
1571 case 2:
2e2f4ade 1572 gen_helper_fcmpeq_fcc2(cpu_env);
a7812ae4
PB
1573 break;
1574 case 3:
2e2f4ade 1575 gen_helper_fcmpeq_fcc3(cpu_env);
a7812ae4
PB
1576 break;
1577 }
7e8c2b6c 1578}
7e8c2b6c
BS
1579
1580#else
1581
714547bb 1582static inline void gen_op_fcmps(int fccno, TCGv r_rs1, TCGv r_rs2)
7e8c2b6c 1583{
2e2f4ade 1584 gen_helper_fcmps(cpu_env, r_rs1, r_rs2);
7e8c2b6c
BS
1585}
1586
03fb8cfc 1587static inline void gen_op_fcmpd(int fccno, TCGv_i64 r_rs1, TCGv_i64 r_rs2)
7e8c2b6c 1588{
03fb8cfc 1589 gen_helper_fcmpd(cpu_env, r_rs1, r_rs2);
7e8c2b6c
BS
1590}
1591
7e8c2b6c
BS
1592static inline void gen_op_fcmpq(int fccno)
1593{
2e2f4ade 1594 gen_helper_fcmpq(cpu_env);
7e8c2b6c 1595}
7e8c2b6c 1596
714547bb 1597static inline void gen_op_fcmpes(int fccno, TCGv r_rs1, TCGv r_rs2)
7e8c2b6c 1598{
2e2f4ade 1599 gen_helper_fcmpes(cpu_env, r_rs1, r_rs2);
7e8c2b6c
BS
1600}
1601
03fb8cfc 1602static inline void gen_op_fcmped(int fccno, TCGv_i64 r_rs1, TCGv_i64 r_rs2)
7e8c2b6c 1603{
03fb8cfc 1604 gen_helper_fcmped(cpu_env, r_rs1, r_rs2);
7e8c2b6c
BS
1605}
1606
7e8c2b6c
BS
1607static inline void gen_op_fcmpeq(int fccno)
1608{
2e2f4ade 1609 gen_helper_fcmpeq(cpu_env);
7e8c2b6c
BS
1610}
1611#endif
1612
134d77a1
BS
1613static inline void gen_op_fpexception_im(int fsr_flags)
1614{
a7812ae4 1615 TCGv_i32 r_const;
2ea815ca 1616
47ad35f1 1617 tcg_gen_andi_tl(cpu_fsr, cpu_fsr, FSR_FTT_NMASK);
87e92502 1618 tcg_gen_ori_tl(cpu_fsr, cpu_fsr, fsr_flags);
2ea815ca 1619 r_const = tcg_const_i32(TT_FP_EXCP);
bc265319 1620 gen_helper_raise_exception(cpu_env, r_const);
a7812ae4 1621 tcg_temp_free_i32(r_const);
134d77a1
BS
1622}
1623
5b12f1e8 1624static int gen_trap_ifnofpu(DisasContext *dc)
a80dde08
FB
1625{
1626#if !defined(CONFIG_USER_ONLY)
1627 if (!dc->fpu_enabled) {
a7812ae4 1628 TCGv_i32 r_const;
2ea815ca 1629
66442b07 1630 save_state(dc);
2ea815ca 1631 r_const = tcg_const_i32(TT_NFPU_INSN);
bc265319 1632 gen_helper_raise_exception(cpu_env, r_const);
a7812ae4 1633 tcg_temp_free_i32(r_const);
a80dde08
FB
1634 dc->is_br = 1;
1635 return 1;
1636 }
1637#endif
1638 return 0;
1639}
1640
7e8c2b6c
BS
1641static inline void gen_op_clear_ieee_excp_and_FTT(void)
1642{
47ad35f1 1643 tcg_gen_andi_tl(cpu_fsr, cpu_fsr, FSR_FTT_CEXC_NMASK);
7e8c2b6c
BS
1644}
1645
61f17f6e
RH
1646static inline void gen_fop_FF(DisasContext *dc, int rd, int rs,
1647 void (*gen)(TCGv_i32, TCGv_ptr, TCGv_i32))
1648{
1649 TCGv_i32 dst, src;
1650
61f17f6e
RH
1651 src = gen_load_fpr_F(dc, rs);
1652 dst = gen_dest_fpr_F();
1653
1654 gen(dst, cpu_env, src);
1655
61f17f6e
RH
1656 gen_store_fpr_F(dc, rd, dst);
1657}
1658
1659static inline void gen_ne_fop_FF(DisasContext *dc, int rd, int rs,
1660 void (*gen)(TCGv_i32, TCGv_i32))
1661{
1662 TCGv_i32 dst, src;
1663
1664 src = gen_load_fpr_F(dc, rs);
1665 dst = gen_dest_fpr_F();
1666
1667 gen(dst, src);
1668
1669 gen_store_fpr_F(dc, rd, dst);
1670}
1671
1672static inline void gen_fop_FFF(DisasContext *dc, int rd, int rs1, int rs2,
1673 void (*gen)(TCGv_i32, TCGv_ptr, TCGv_i32, TCGv_i32))
1674{
1675 TCGv_i32 dst, src1, src2;
1676
61f17f6e
RH
1677 src1 = gen_load_fpr_F(dc, rs1);
1678 src2 = gen_load_fpr_F(dc, rs2);
1679 dst = gen_dest_fpr_F();
1680
1681 gen(dst, cpu_env, src1, src2);
1682
61f17f6e
RH
1683 gen_store_fpr_F(dc, rd, dst);
1684}
1685
1686#ifdef TARGET_SPARC64
1687static inline void gen_ne_fop_FFF(DisasContext *dc, int rd, int rs1, int rs2,
1688 void (*gen)(TCGv_i32, TCGv_i32, TCGv_i32))
1689{
1690 TCGv_i32 dst, src1, src2;
1691
1692 src1 = gen_load_fpr_F(dc, rs1);
1693 src2 = gen_load_fpr_F(dc, rs2);
1694 dst = gen_dest_fpr_F();
1695
1696 gen(dst, src1, src2);
1697
1698 gen_store_fpr_F(dc, rd, dst);
1699}
1700#endif
1701
1702static inline void gen_fop_DD(DisasContext *dc, int rd, int rs,
1703 void (*gen)(TCGv_i64, TCGv_ptr, TCGv_i64))
1704{
1705 TCGv_i64 dst, src;
1706
61f17f6e
RH
1707 src = gen_load_fpr_D(dc, rs);
1708 dst = gen_dest_fpr_D();
1709
1710 gen(dst, cpu_env, src);
1711
61f17f6e
RH
1712 gen_store_fpr_D(dc, rd, dst);
1713}
1714
1715#ifdef TARGET_SPARC64
1716static inline void gen_ne_fop_DD(DisasContext *dc, int rd, int rs,
1717 void (*gen)(TCGv_i64, TCGv_i64))
1718{
1719 TCGv_i64 dst, src;
1720
1721 src = gen_load_fpr_D(dc, rs);
1722 dst = gen_dest_fpr_D();
1723
1724 gen(dst, src);
1725
1726 gen_store_fpr_D(dc, rd, dst);
1727}
1728#endif
1729
1730static inline void gen_fop_DDD(DisasContext *dc, int rd, int rs1, int rs2,
1731 void (*gen)(TCGv_i64, TCGv_ptr, TCGv_i64, TCGv_i64))
1732{
1733 TCGv_i64 dst, src1, src2;
1734
61f17f6e
RH
1735 src1 = gen_load_fpr_D(dc, rs1);
1736 src2 = gen_load_fpr_D(dc, rs2);
1737 dst = gen_dest_fpr_D();
1738
1739 gen(dst, cpu_env, src1, src2);
1740
61f17f6e
RH
1741 gen_store_fpr_D(dc, rd, dst);
1742}
1743
1744#ifdef TARGET_SPARC64
1745static inline void gen_ne_fop_DDD(DisasContext *dc, int rd, int rs1, int rs2,
1746 void (*gen)(TCGv_i64, TCGv_i64, TCGv_i64))
1747{
1748 TCGv_i64 dst, src1, src2;
1749
1750 src1 = gen_load_fpr_D(dc, rs1);
1751 src2 = gen_load_fpr_D(dc, rs2);
1752 dst = gen_dest_fpr_D();
1753
1754 gen(dst, src1, src2);
1755
1756 gen_store_fpr_D(dc, rd, dst);
1757}
f888300b 1758
2dedf314
RH
1759static inline void gen_gsr_fop_DDD(DisasContext *dc, int rd, int rs1, int rs2,
1760 void (*gen)(TCGv_i64, TCGv_i64, TCGv_i64, TCGv_i64))
1761{
1762 TCGv_i64 dst, src1, src2;
1763
1764 src1 = gen_load_fpr_D(dc, rs1);
1765 src2 = gen_load_fpr_D(dc, rs2);
1766 dst = gen_dest_fpr_D();
1767
1768 gen(dst, cpu_gsr, src1, src2);
1769
1770 gen_store_fpr_D(dc, rd, dst);
1771}
1772
f888300b
RH
1773static inline void gen_ne_fop_DDDD(DisasContext *dc, int rd, int rs1, int rs2,
1774 void (*gen)(TCGv_i64, TCGv_i64, TCGv_i64, TCGv_i64))
1775{
1776 TCGv_i64 dst, src0, src1, src2;
1777
1778 src1 = gen_load_fpr_D(dc, rs1);
1779 src2 = gen_load_fpr_D(dc, rs2);
1780 src0 = gen_load_fpr_D(dc, rd);
1781 dst = gen_dest_fpr_D();
1782
1783 gen(dst, src0, src1, src2);
1784
1785 gen_store_fpr_D(dc, rd, dst);
1786}
61f17f6e
RH
1787#endif
1788
1789static inline void gen_fop_QQ(DisasContext *dc, int rd, int rs,
1790 void (*gen)(TCGv_ptr))
1791{
61f17f6e
RH
1792 gen_op_load_fpr_QT1(QFPREG(rs));
1793
1794 gen(cpu_env);
1795
61f17f6e
RH
1796 gen_op_store_QT0_fpr(QFPREG(rd));
1797 gen_update_fprs_dirty(QFPREG(rd));
1798}
1799
1800#ifdef TARGET_SPARC64
1801static inline void gen_ne_fop_QQ(DisasContext *dc, int rd, int rs,
1802 void (*gen)(TCGv_ptr))
1803{
1804 gen_op_load_fpr_QT1(QFPREG(rs));
1805
1806 gen(cpu_env);
1807
1808 gen_op_store_QT0_fpr(QFPREG(rd));
1809 gen_update_fprs_dirty(QFPREG(rd));
1810}
1811#endif
1812
1813static inline void gen_fop_QQQ(DisasContext *dc, int rd, int rs1, int rs2,
1814 void (*gen)(TCGv_ptr))
1815{
61f17f6e
RH
1816 gen_op_load_fpr_QT0(QFPREG(rs1));
1817 gen_op_load_fpr_QT1(QFPREG(rs2));
1818
1819 gen(cpu_env);
1820
61f17f6e
RH
1821 gen_op_store_QT0_fpr(QFPREG(rd));
1822 gen_update_fprs_dirty(QFPREG(rd));
1823}
1824
1825static inline void gen_fop_DFF(DisasContext *dc, int rd, int rs1, int rs2,
1826 void (*gen)(TCGv_i64, TCGv_ptr, TCGv_i32, TCGv_i32))
1827{
1828 TCGv_i64 dst;
1829 TCGv_i32 src1, src2;
1830
61f17f6e
RH
1831 src1 = gen_load_fpr_F(dc, rs1);
1832 src2 = gen_load_fpr_F(dc, rs2);
1833 dst = gen_dest_fpr_D();
1834
1835 gen(dst, cpu_env, src1, src2);
1836
61f17f6e
RH
1837 gen_store_fpr_D(dc, rd, dst);
1838}
1839
1840static inline void gen_fop_QDD(DisasContext *dc, int rd, int rs1, int rs2,
1841 void (*gen)(TCGv_ptr, TCGv_i64, TCGv_i64))
1842{
1843 TCGv_i64 src1, src2;
1844
61f17f6e
RH
1845 src1 = gen_load_fpr_D(dc, rs1);
1846 src2 = gen_load_fpr_D(dc, rs2);
1847
1848 gen(cpu_env, src1, src2);
1849
61f17f6e
RH
1850 gen_op_store_QT0_fpr(QFPREG(rd));
1851 gen_update_fprs_dirty(QFPREG(rd));
1852}
1853
1854#ifdef TARGET_SPARC64
1855static inline void gen_fop_DF(DisasContext *dc, int rd, int rs,
1856 void (*gen)(TCGv_i64, TCGv_ptr, TCGv_i32))
1857{
1858 TCGv_i64 dst;
1859 TCGv_i32 src;
1860
61f17f6e
RH
1861 src = gen_load_fpr_F(dc, rs);
1862 dst = gen_dest_fpr_D();
1863
1864 gen(dst, cpu_env, src);
1865
61f17f6e
RH
1866 gen_store_fpr_D(dc, rd, dst);
1867}
1868#endif
1869
1870static inline void gen_ne_fop_DF(DisasContext *dc, int rd, int rs,
1871 void (*gen)(TCGv_i64, TCGv_ptr, TCGv_i32))
1872{
1873 TCGv_i64 dst;
1874 TCGv_i32 src;
1875
1876 src = gen_load_fpr_F(dc, rs);
1877 dst = gen_dest_fpr_D();
1878
1879 gen(dst, cpu_env, src);
1880
1881 gen_store_fpr_D(dc, rd, dst);
1882}
1883
1884static inline void gen_fop_FD(DisasContext *dc, int rd, int rs,
1885 void (*gen)(TCGv_i32, TCGv_ptr, TCGv_i64))
1886{
1887 TCGv_i32 dst;
1888 TCGv_i64 src;
1889
61f17f6e
RH
1890 src = gen_load_fpr_D(dc, rs);
1891 dst = gen_dest_fpr_F();
1892
1893 gen(dst, cpu_env, src);
1894
61f17f6e
RH
1895 gen_store_fpr_F(dc, rd, dst);
1896}
1897
1898static inline void gen_fop_FQ(DisasContext *dc, int rd, int rs,
1899 void (*gen)(TCGv_i32, TCGv_ptr))
1900{
1901 TCGv_i32 dst;
1902
61f17f6e
RH
1903 gen_op_load_fpr_QT1(QFPREG(rs));
1904 dst = gen_dest_fpr_F();
1905
1906 gen(dst, cpu_env);
1907
61f17f6e
RH
1908 gen_store_fpr_F(dc, rd, dst);
1909}
1910
1911static inline void gen_fop_DQ(DisasContext *dc, int rd, int rs,
1912 void (*gen)(TCGv_i64, TCGv_ptr))
1913{
1914 TCGv_i64 dst;
1915
61f17f6e
RH
1916 gen_op_load_fpr_QT1(QFPREG(rs));
1917 dst = gen_dest_fpr_D();
1918
1919 gen(dst, cpu_env);
1920
61f17f6e
RH
1921 gen_store_fpr_D(dc, rd, dst);
1922}
1923
1924static inline void gen_ne_fop_QF(DisasContext *dc, int rd, int rs,
1925 void (*gen)(TCGv_ptr, TCGv_i32))
1926{
1927 TCGv_i32 src;
1928
1929 src = gen_load_fpr_F(dc, rs);
1930
1931 gen(cpu_env, src);
1932
1933 gen_op_store_QT0_fpr(QFPREG(rd));
1934 gen_update_fprs_dirty(QFPREG(rd));
1935}
1936
1937static inline void gen_ne_fop_QD(DisasContext *dc, int rd, int rs,
1938 void (*gen)(TCGv_ptr, TCGv_i64))
1939{
1940 TCGv_i64 src;
1941
1942 src = gen_load_fpr_D(dc, rs);
1943
1944 gen(cpu_env, src);
1945
1946 gen_op_store_QT0_fpr(QFPREG(rd));
1947 gen_update_fprs_dirty(QFPREG(rd));
1948}
1949
1a2fb1c0
BS
1950/* asi moves */
1951#ifdef TARGET_SPARC64
a7812ae4 1952static inline TCGv_i32 gen_get_asi(int insn, TCGv r_addr)
1a2fb1c0 1953{
95f9397c 1954 int asi;
a7812ae4 1955 TCGv_i32 r_asi;
1a2fb1c0 1956
1a2fb1c0 1957 if (IS_IMM) {
a7812ae4 1958 r_asi = tcg_temp_new_i32();
255e1fcb 1959 tcg_gen_mov_i32(r_asi, cpu_asi);
1a2fb1c0
BS
1960 } else {
1961 asi = GET_FIELD(insn, 19, 26);
0425bee5 1962 r_asi = tcg_const_i32(asi);
1a2fb1c0 1963 }
0425bee5
BS
1964 return r_asi;
1965}
1966
77f193da
BS
1967static inline void gen_ld_asi(TCGv dst, TCGv addr, int insn, int size,
1968 int sign)
0425bee5 1969{
a7812ae4 1970 TCGv_i32 r_asi, r_size, r_sign;
0425bee5 1971
4af984a7 1972 r_asi = gen_get_asi(insn, addr);
2ea815ca
BS
1973 r_size = tcg_const_i32(size);
1974 r_sign = tcg_const_i32(sign);
fe8d8f0f 1975 gen_helper_ld_asi(dst, cpu_env, addr, r_asi, r_size, r_sign);
a7812ae4
PB
1976 tcg_temp_free_i32(r_sign);
1977 tcg_temp_free_i32(r_size);
1978 tcg_temp_free_i32(r_asi);
1a2fb1c0
BS
1979}
1980
4af984a7 1981static inline void gen_st_asi(TCGv src, TCGv addr, int insn, int size)
1a2fb1c0 1982{
a7812ae4 1983 TCGv_i32 r_asi, r_size;
1a2fb1c0 1984
4af984a7 1985 r_asi = gen_get_asi(insn, addr);
2ea815ca 1986 r_size = tcg_const_i32(size);
fe8d8f0f 1987 gen_helper_st_asi(cpu_env, addr, src, r_asi, r_size);
a7812ae4
PB
1988 tcg_temp_free_i32(r_size);
1989 tcg_temp_free_i32(r_asi);
1a2fb1c0
BS
1990}
1991
4af984a7 1992static inline void gen_ldf_asi(TCGv addr, int insn, int size, int rd)
1a2fb1c0 1993{
a7812ae4 1994 TCGv_i32 r_asi, r_size, r_rd;
1a2fb1c0 1995
4af984a7 1996 r_asi = gen_get_asi(insn, addr);
2ea815ca
BS
1997 r_size = tcg_const_i32(size);
1998 r_rd = tcg_const_i32(rd);
fe8d8f0f 1999 gen_helper_ldf_asi(cpu_env, addr, r_asi, r_size, r_rd);
a7812ae4
PB
2000 tcg_temp_free_i32(r_rd);
2001 tcg_temp_free_i32(r_size);
2002 tcg_temp_free_i32(r_asi);
1a2fb1c0
BS
2003}
2004
4af984a7 2005static inline void gen_stf_asi(TCGv addr, int insn, int size, int rd)
1a2fb1c0 2006{
a7812ae4 2007 TCGv_i32 r_asi, r_size, r_rd;
1a2fb1c0 2008
31741a27 2009 r_asi = gen_get_asi(insn, addr);
2ea815ca
BS
2010 r_size = tcg_const_i32(size);
2011 r_rd = tcg_const_i32(rd);
fe8d8f0f 2012 gen_helper_stf_asi(cpu_env, addr, r_asi, r_size, r_rd);
a7812ae4
PB
2013 tcg_temp_free_i32(r_rd);
2014 tcg_temp_free_i32(r_size);
2015 tcg_temp_free_i32(r_asi);
1a2fb1c0
BS
2016}
2017
4af984a7 2018static inline void gen_swap_asi(TCGv dst, TCGv addr, int insn)
1a2fb1c0 2019{
a7812ae4 2020 TCGv_i32 r_asi, r_size, r_sign;
1a2fb1c0 2021
4af984a7 2022 r_asi = gen_get_asi(insn, addr);
2ea815ca
BS
2023 r_size = tcg_const_i32(4);
2024 r_sign = tcg_const_i32(0);
fe8d8f0f 2025 gen_helper_ld_asi(cpu_tmp64, cpu_env, addr, r_asi, r_size, r_sign);
a7812ae4 2026 tcg_temp_free_i32(r_sign);
fe8d8f0f 2027 gen_helper_st_asi(cpu_env, addr, dst, r_asi, r_size);
a7812ae4
PB
2028 tcg_temp_free_i32(r_size);
2029 tcg_temp_free_i32(r_asi);
8d96d209 2030 tcg_gen_trunc_i64_tl(dst, cpu_tmp64);
1a2fb1c0
BS
2031}
2032
db166940 2033static inline void gen_ldda_asi(TCGv hi, TCGv addr, int insn, int rd)
1a2fb1c0 2034{
a7812ae4 2035 TCGv_i32 r_asi, r_rd;
1a2fb1c0 2036
4af984a7 2037 r_asi = gen_get_asi(insn, addr);
db166940 2038 r_rd = tcg_const_i32(rd);
fe8d8f0f 2039 gen_helper_ldda_asi(cpu_env, addr, r_asi, r_rd);
a7812ae4
PB
2040 tcg_temp_free_i32(r_rd);
2041 tcg_temp_free_i32(r_asi);
0425bee5
BS
2042}
2043
4af984a7 2044static inline void gen_stda_asi(TCGv hi, TCGv addr, int insn, int rd)
0425bee5 2045{
a7812ae4 2046 TCGv_i32 r_asi, r_size;
a7ec4229
BS
2047
2048 gen_movl_reg_TN(rd + 1, cpu_tmp0);
ab508019 2049 tcg_gen_concat_tl_i64(cpu_tmp64, cpu_tmp0, hi);
4af984a7 2050 r_asi = gen_get_asi(insn, addr);
2ea815ca 2051 r_size = tcg_const_i32(8);
fe8d8f0f 2052 gen_helper_st_asi(cpu_env, addr, cpu_tmp64, r_asi, r_size);
a7812ae4
PB
2053 tcg_temp_free_i32(r_size);
2054 tcg_temp_free_i32(r_asi);
1a2fb1c0
BS
2055}
2056
77f193da
BS
2057static inline void gen_cas_asi(TCGv dst, TCGv addr, TCGv val2, int insn,
2058 int rd)
1a2fb1c0 2059{
a7812ae4
PB
2060 TCGv r_val1;
2061 TCGv_i32 r_asi;
1a2fb1c0 2062
a7812ae4 2063 r_val1 = tcg_temp_new();
1a2fb1c0 2064 gen_movl_reg_TN(rd, r_val1);
4af984a7 2065 r_asi = gen_get_asi(insn, addr);
fe8d8f0f 2066 gen_helper_cas_asi(dst, cpu_env, addr, r_val1, val2, r_asi);
a7812ae4 2067 tcg_temp_free_i32(r_asi);
2ea815ca 2068 tcg_temp_free(r_val1);
1a2fb1c0
BS
2069}
2070
77f193da
BS
2071static inline void gen_casx_asi(TCGv dst, TCGv addr, TCGv val2, int insn,
2072 int rd)
1a2fb1c0 2073{
a7812ae4 2074 TCGv_i32 r_asi;
1a2fb1c0 2075
8911f501 2076 gen_movl_reg_TN(rd, cpu_tmp64);
4af984a7 2077 r_asi = gen_get_asi(insn, addr);
fe8d8f0f 2078 gen_helper_casx_asi(dst, cpu_env, addr, cpu_tmp64, val2, r_asi);
a7812ae4 2079 tcg_temp_free_i32(r_asi);
1a2fb1c0
BS
2080}
2081
2082#elif !defined(CONFIG_USER_ONLY)
2083
77f193da
BS
2084static inline void gen_ld_asi(TCGv dst, TCGv addr, int insn, int size,
2085 int sign)
1a2fb1c0 2086{
a7812ae4 2087 TCGv_i32 r_asi, r_size, r_sign;
1a2fb1c0 2088
2ea815ca
BS
2089 r_asi = tcg_const_i32(GET_FIELD(insn, 19, 26));
2090 r_size = tcg_const_i32(size);
2091 r_sign = tcg_const_i32(sign);
fe8d8f0f 2092 gen_helper_ld_asi(cpu_tmp64, cpu_env, addr, r_asi, r_size, r_sign);
2ea815ca
BS
2093 tcg_temp_free(r_sign);
2094 tcg_temp_free(r_size);
2095 tcg_temp_free(r_asi);
4af984a7 2096 tcg_gen_trunc_i64_tl(dst, cpu_tmp64);
1a2fb1c0
BS
2097}
2098
4af984a7 2099static inline void gen_st_asi(TCGv src, TCGv addr, int insn, int size)
1a2fb1c0 2100{
a7812ae4 2101 TCGv_i32 r_asi, r_size;
1a2fb1c0 2102
4af984a7 2103 tcg_gen_extu_tl_i64(cpu_tmp64, src);
2ea815ca
BS
2104 r_asi = tcg_const_i32(GET_FIELD(insn, 19, 26));
2105 r_size = tcg_const_i32(size);
fe8d8f0f 2106 gen_helper_st_asi(cpu_env, addr, cpu_tmp64, r_asi, r_size);
2ea815ca
BS
2107 tcg_temp_free(r_size);
2108 tcg_temp_free(r_asi);
1a2fb1c0
BS
2109}
2110
4af984a7 2111static inline void gen_swap_asi(TCGv dst, TCGv addr, int insn)
1a2fb1c0 2112{
a7812ae4
PB
2113 TCGv_i32 r_asi, r_size, r_sign;
2114 TCGv_i64 r_val;
1a2fb1c0 2115
2ea815ca
BS
2116 r_asi = tcg_const_i32(GET_FIELD(insn, 19, 26));
2117 r_size = tcg_const_i32(4);
2118 r_sign = tcg_const_i32(0);
fe8d8f0f 2119 gen_helper_ld_asi(cpu_tmp64, cpu_env, addr, r_asi, r_size, r_sign);
2ea815ca 2120 tcg_temp_free(r_sign);
a7812ae4
PB
2121 r_val = tcg_temp_new_i64();
2122 tcg_gen_extu_tl_i64(r_val, dst);
fe8d8f0f 2123 gen_helper_st_asi(cpu_env, addr, r_val, r_asi, r_size);
a7812ae4 2124 tcg_temp_free_i64(r_val);
2ea815ca
BS
2125 tcg_temp_free(r_size);
2126 tcg_temp_free(r_asi);
8d96d209 2127 tcg_gen_trunc_i64_tl(dst, cpu_tmp64);
1a2fb1c0
BS
2128}
2129
db166940 2130static inline void gen_ldda_asi(TCGv hi, TCGv addr, int insn, int rd)
1a2fb1c0 2131{
a7812ae4 2132 TCGv_i32 r_asi, r_size, r_sign;
1a2fb1c0 2133
2ea815ca
BS
2134 r_asi = tcg_const_i32(GET_FIELD(insn, 19, 26));
2135 r_size = tcg_const_i32(8);
2136 r_sign = tcg_const_i32(0);
fe8d8f0f 2137 gen_helper_ld_asi(cpu_tmp64, cpu_env, addr, r_asi, r_size, r_sign);
2ea815ca
BS
2138 tcg_temp_free(r_sign);
2139 tcg_temp_free(r_size);
2140 tcg_temp_free(r_asi);
db166940
BS
2141 tcg_gen_trunc_i64_tl(cpu_tmp0, cpu_tmp64);
2142 gen_movl_TN_reg(rd + 1, cpu_tmp0);
8911f501 2143 tcg_gen_shri_i64(cpu_tmp64, cpu_tmp64, 32);
4af984a7 2144 tcg_gen_trunc_i64_tl(hi, cpu_tmp64);
db166940 2145 gen_movl_TN_reg(rd, hi);
0425bee5
BS
2146}
2147
4af984a7 2148static inline void gen_stda_asi(TCGv hi, TCGv addr, int insn, int rd)
0425bee5 2149{
a7812ae4 2150 TCGv_i32 r_asi, r_size;
a7ec4229
BS
2151
2152 gen_movl_reg_TN(rd + 1, cpu_tmp0);
ab508019 2153 tcg_gen_concat_tl_i64(cpu_tmp64, cpu_tmp0, hi);
2ea815ca
BS
2154 r_asi = tcg_const_i32(GET_FIELD(insn, 19, 26));
2155 r_size = tcg_const_i32(8);
fe8d8f0f 2156 gen_helper_st_asi(cpu_env, addr, cpu_tmp64, r_asi, r_size);
2ea815ca
BS
2157 tcg_temp_free(r_size);
2158 tcg_temp_free(r_asi);
1a2fb1c0
BS
2159}
2160#endif
2161
2162#if !defined(CONFIG_USER_ONLY) || defined(TARGET_SPARC64)
4af984a7 2163static inline void gen_ldstub_asi(TCGv dst, TCGv addr, int insn)
1a2fb1c0 2164{
a7812ae4
PB
2165 TCGv_i64 r_val;
2166 TCGv_i32 r_asi, r_size;
1a2fb1c0 2167
4af984a7 2168 gen_ld_asi(dst, addr, insn, 1, 0);
1a2fb1c0 2169
2ea815ca
BS
2170 r_val = tcg_const_i64(0xffULL);
2171 r_asi = tcg_const_i32(GET_FIELD(insn, 19, 26));
2172 r_size = tcg_const_i32(1);
fe8d8f0f 2173 gen_helper_st_asi(cpu_env, addr, r_val, r_asi, r_size);
a7812ae4
PB
2174 tcg_temp_free_i32(r_size);
2175 tcg_temp_free_i32(r_asi);
2176 tcg_temp_free_i64(r_val);
1a2fb1c0
BS
2177}
2178#endif
2179
9322a4bf
BS
2180static inline TCGv get_src1(unsigned int insn, TCGv def)
2181{
2182 TCGv r_rs1 = def;
2183 unsigned int rs1;
2184
2185 rs1 = GET_FIELD(insn, 13, 17);
42a8aa83
RH
2186 if (rs1 == 0) {
2187 tcg_gen_movi_tl(def, 0);
2188 } else if (rs1 < 8) {
5c6a0628 2189 r_rs1 = cpu_gregs[rs1];
42a8aa83 2190 } else {
9322a4bf 2191 tcg_gen_ld_tl(def, cpu_regwptr, (rs1 - 8) * sizeof(target_ulong));
42a8aa83 2192 }
9322a4bf
BS
2193 return r_rs1;
2194}
2195
a49d9390
BS
2196static inline TCGv get_src2(unsigned int insn, TCGv def)
2197{
2198 TCGv r_rs2 = def;
a49d9390
BS
2199
2200 if (IS_IMM) { /* immediate */
42a8aa83
RH
2201 target_long simm = GET_FIELDs(insn, 19, 31);
2202 tcg_gen_movi_tl(def, simm);
a49d9390 2203 } else { /* register */
42a8aa83
RH
2204 unsigned int rs2 = GET_FIELD(insn, 27, 31);
2205 if (rs2 == 0) {
2206 tcg_gen_movi_tl(def, 0);
2207 } else if (rs2 < 8) {
a49d9390 2208 r_rs2 = cpu_gregs[rs2];
42a8aa83 2209 } else {
a49d9390 2210 tcg_gen_ld_tl(def, cpu_regwptr, (rs2 - 8) * sizeof(target_ulong));
42a8aa83 2211 }
a49d9390
BS
2212 }
2213 return r_rs2;
2214}
2215
8194f35a 2216#ifdef TARGET_SPARC64
7e480893
RH
2217static void gen_fmovs(DisasContext *dc, DisasCompare *cmp, int rd, int rs)
2218{
2219 TCGv_i32 c32, zero, dst, s1, s2;
2220
2221 /* We have two choices here: extend the 32 bit data and use movcond_i64,
2222 or fold the comparison down to 32 bits and use movcond_i32. Choose
2223 the later. */
2224 c32 = tcg_temp_new_i32();
2225 if (cmp->is_bool) {
2226 tcg_gen_trunc_i64_i32(c32, cmp->c1);
2227 } else {
2228 TCGv_i64 c64 = tcg_temp_new_i64();
2229 tcg_gen_setcond_i64(cmp->cond, c64, cmp->c1, cmp->c2);
2230 tcg_gen_trunc_i64_i32(c32, c64);
2231 tcg_temp_free_i64(c64);
2232 }
2233
2234 s1 = gen_load_fpr_F(dc, rs);
2235 s2 = gen_load_fpr_F(dc, rd);
2236 dst = gen_dest_fpr_F();
2237 zero = tcg_const_i32(0);
2238
2239 tcg_gen_movcond_i32(TCG_COND_NE, dst, c32, zero, s1, s2);
2240
2241 tcg_temp_free_i32(c32);
2242 tcg_temp_free_i32(zero);
2243 gen_store_fpr_F(dc, rd, dst);
2244}
2245
2246static void gen_fmovd(DisasContext *dc, DisasCompare *cmp, int rd, int rs)
2247{
2248 TCGv_i64 dst = gen_dest_fpr_D();
2249 tcg_gen_movcond_i64(cmp->cond, dst, cmp->c1, cmp->c2,
2250 gen_load_fpr_D(dc, rs),
2251 gen_load_fpr_D(dc, rd));
2252 gen_store_fpr_D(dc, rd, dst);
2253}
2254
2255static void gen_fmovq(DisasContext *dc, DisasCompare *cmp, int rd, int rs)
2256{
2257 int qd = QFPREG(rd);
2258 int qs = QFPREG(rs);
2259
2260 tcg_gen_movcond_i64(cmp->cond, cpu_fpr[qd / 2], cmp->c1, cmp->c2,
2261 cpu_fpr[qs / 2], cpu_fpr[qd / 2]);
2262 tcg_gen_movcond_i64(cmp->cond, cpu_fpr[qd / 2 + 1], cmp->c1, cmp->c2,
2263 cpu_fpr[qs / 2 + 1], cpu_fpr[qd / 2 + 1]);
2264
2265 gen_update_fprs_dirty(qd);
2266}
2267
8194f35a
IK
2268static inline void gen_load_trap_state_at_tl(TCGv_ptr r_tsptr, TCGv_ptr cpu_env)
2269{
b551ec04 2270 TCGv_i32 r_tl = tcg_temp_new_i32();
8194f35a
IK
2271
2272 /* load env->tl into r_tl */
b551ec04 2273 tcg_gen_ld_i32(r_tl, cpu_env, offsetof(CPUSPARCState, tl));
8194f35a
IK
2274
2275 /* tl = [0 ... MAXTL_MASK] where MAXTL_MASK must be power of 2 */
b551ec04 2276 tcg_gen_andi_i32(r_tl, r_tl, MAXTL_MASK);
8194f35a
IK
2277
2278 /* calculate offset to current trap state from env->ts, reuse r_tl */
b551ec04 2279 tcg_gen_muli_i32(r_tl, r_tl, sizeof (trap_state));
c5f9864e 2280 tcg_gen_addi_ptr(r_tsptr, cpu_env, offsetof(CPUSPARCState, ts));
8194f35a
IK
2281
2282 /* tsptr = env->ts[env->tl & MAXTL_MASK] */
b551ec04
JF
2283 {
2284 TCGv_ptr r_tl_tmp = tcg_temp_new_ptr();
2285 tcg_gen_ext_i32_ptr(r_tl_tmp, r_tl);
2286 tcg_gen_add_ptr(r_tsptr, r_tsptr, r_tl_tmp);
bc57c114 2287 tcg_temp_free_ptr(r_tl_tmp);
b551ec04 2288 }
8194f35a 2289
b551ec04 2290 tcg_temp_free_i32(r_tl);
8194f35a 2291}
6c073553
RH
2292
2293static void gen_edge(DisasContext *dc, TCGv dst, TCGv s1, TCGv s2,
2294 int width, bool cc, bool left)
2295{
2296 TCGv lo1, lo2, t1, t2;
2297 uint64_t amask, tabl, tabr;
2298 int shift, imask, omask;
2299
2300 if (cc) {
2301 tcg_gen_mov_tl(cpu_cc_src, s1);
2302 tcg_gen_mov_tl(cpu_cc_src2, s2);
2303 tcg_gen_sub_tl(cpu_cc_dst, s1, s2);
2304 tcg_gen_movi_i32(cpu_cc_op, CC_OP_SUB);
2305 dc->cc_op = CC_OP_SUB;
2306 }
2307
2308 /* Theory of operation: there are two tables, left and right (not to
2309 be confused with the left and right versions of the opcode). These
2310 are indexed by the low 3 bits of the inputs. To make things "easy",
2311 these tables are loaded into two constants, TABL and TABR below.
2312 The operation index = (input & imask) << shift calculates the index
2313 into the constant, while val = (table >> index) & omask calculates
2314 the value we're looking for. */
2315 switch (width) {
2316 case 8:
2317 imask = 0x7;
2318 shift = 3;
2319 omask = 0xff;
2320 if (left) {
2321 tabl = 0x80c0e0f0f8fcfeffULL;
2322 tabr = 0xff7f3f1f0f070301ULL;
2323 } else {
2324 tabl = 0x0103070f1f3f7fffULL;
2325 tabr = 0xfffefcf8f0e0c080ULL;
2326 }
2327 break;
2328 case 16:
2329 imask = 0x6;
2330 shift = 1;
2331 omask = 0xf;
2332 if (left) {
2333 tabl = 0x8cef;
2334 tabr = 0xf731;
2335 } else {
2336 tabl = 0x137f;
2337 tabr = 0xfec8;
2338 }
2339 break;
2340 case 32:
2341 imask = 0x4;
2342 shift = 0;
2343 omask = 0x3;
2344 if (left) {
2345 tabl = (2 << 2) | 3;
2346 tabr = (3 << 2) | 1;
2347 } else {
2348 tabl = (1 << 2) | 3;
2349 tabr = (3 << 2) | 2;
2350 }
2351 break;
2352 default:
2353 abort();
2354 }
2355
2356 lo1 = tcg_temp_new();
2357 lo2 = tcg_temp_new();
2358 tcg_gen_andi_tl(lo1, s1, imask);
2359 tcg_gen_andi_tl(lo2, s2, imask);
2360 tcg_gen_shli_tl(lo1, lo1, shift);
2361 tcg_gen_shli_tl(lo2, lo2, shift);
2362
2363 t1 = tcg_const_tl(tabl);
2364 t2 = tcg_const_tl(tabr);
2365 tcg_gen_shr_tl(lo1, t1, lo1);
2366 tcg_gen_shr_tl(lo2, t2, lo2);
2367 tcg_gen_andi_tl(dst, lo1, omask);
2368 tcg_gen_andi_tl(lo2, lo2, omask);
2369
2370 amask = -8;
2371 if (AM_CHECK(dc)) {
2372 amask &= 0xffffffffULL;
2373 }
2374 tcg_gen_andi_tl(s1, s1, amask);
2375 tcg_gen_andi_tl(s2, s2, amask);
2376
2377 /* We want to compute
2378 dst = (s1 == s2 ? lo1 : lo1 & lo2).
2379 We've already done dst = lo1, so this reduces to
2380 dst &= (s1 == s2 ? -1 : lo2)
2381 Which we perform by
2382 lo2 |= -(s1 == s2)
2383 dst &= lo2
2384 */
2385 tcg_gen_setcond_tl(TCG_COND_EQ, t1, s1, s2);
2386 tcg_gen_neg_tl(t1, t1);
2387 tcg_gen_or_tl(lo2, lo2, t1);
2388 tcg_gen_and_tl(dst, dst, lo2);
2389
2390 tcg_temp_free(lo1);
2391 tcg_temp_free(lo2);
2392 tcg_temp_free(t1);
2393 tcg_temp_free(t2);
2394}
add545ab
RH
2395
2396static void gen_alignaddr(TCGv dst, TCGv s1, TCGv s2, bool left)
2397{
2398 TCGv tmp = tcg_temp_new();
2399
2400 tcg_gen_add_tl(tmp, s1, s2);
2401 tcg_gen_andi_tl(dst, tmp, -8);
2402 if (left) {
2403 tcg_gen_neg_tl(tmp, tmp);
2404 }
2405 tcg_gen_deposit_tl(cpu_gsr, cpu_gsr, tmp, 0, 3);
2406
2407 tcg_temp_free(tmp);
2408}
50c796f9
RH
2409
2410static void gen_faligndata(TCGv dst, TCGv gsr, TCGv s1, TCGv s2)
2411{
2412 TCGv t1, t2, shift;
2413
2414 t1 = tcg_temp_new();
2415 t2 = tcg_temp_new();
2416 shift = tcg_temp_new();
2417
2418 tcg_gen_andi_tl(shift, gsr, 7);
2419 tcg_gen_shli_tl(shift, shift, 3);
2420 tcg_gen_shl_tl(t1, s1, shift);
2421
2422 /* A shift of 64 does not produce 0 in TCG. Divide this into a
2423 shift of (up to 63) followed by a constant shift of 1. */
2424 tcg_gen_xori_tl(shift, shift, 63);
2425 tcg_gen_shr_tl(t2, s2, shift);
2426 tcg_gen_shri_tl(t2, t2, 1);
2427
2428 tcg_gen_or_tl(dst, t1, t2);
2429
2430 tcg_temp_free(t1);
2431 tcg_temp_free(t2);
2432 tcg_temp_free(shift);
2433}
8194f35a
IK
2434#endif
2435
64a88d5d 2436#define CHECK_IU_FEATURE(dc, FEATURE) \
5578ceab 2437 if (!((dc)->def->features & CPU_FEATURE_ ## FEATURE)) \
64a88d5d
BS
2438 goto illegal_insn;
2439#define CHECK_FPU_FEATURE(dc, FEATURE) \
5578ceab 2440 if (!((dc)->def->features & CPU_FEATURE_ ## FEATURE)) \
64a88d5d
BS
2441 goto nfpu_insn;
2442
0bee699e 2443/* before an instruction, dc->pc must be static */
0184e266 2444static void disas_sparc_insn(DisasContext * dc, unsigned int insn)
cf495bcf 2445{
0184e266 2446 unsigned int opc, rs1, rs2, rd;
42a8aa83 2447 TCGv cpu_src1, cpu_src2, cpu_tmp1, cpu_tmp2;
208ae657 2448 TCGv_i32 cpu_src1_32, cpu_src2_32, cpu_dst_32;
96eda024 2449 TCGv_i64 cpu_src1_64, cpu_src2_64, cpu_dst_64;
67526b20 2450 target_long simm;
7a3f1944 2451
fdefe51c 2452 if (unlikely(qemu_loglevel_mask(CPU_LOG_TB_OP | CPU_LOG_TB_OP_OPT))) {
a8c768c0 2453 tcg_gen_debug_insn_start(dc->pc);
fdefe51c 2454 }
0184e266 2455
cf495bcf 2456 opc = GET_FIELD(insn, 0, 1);
7a3f1944 2457
cf495bcf 2458 rd = GET_FIELD(insn, 2, 6);
6ae20372 2459
42a8aa83
RH
2460 cpu_tmp1 = cpu_src1 = tcg_temp_new();
2461 cpu_tmp2 = cpu_src2 = tcg_temp_new();
6ae20372 2462
cf495bcf 2463 switch (opc) {
0f8a249a
BS
2464 case 0: /* branches/sethi */
2465 {
2466 unsigned int xop = GET_FIELD(insn, 7, 9);
2467 int32_t target;
2468 switch (xop) {
3475187d 2469#ifdef TARGET_SPARC64
0f8a249a
BS
2470 case 0x1: /* V9 BPcc */
2471 {
2472 int cc;
2473
2474 target = GET_FIELD_SP(insn, 0, 18);
86f1f2ae 2475 target = sign_extend(target, 19);
0f8a249a
BS
2476 target <<= 2;
2477 cc = GET_FIELD_SP(insn, 20, 21);
2478 if (cc == 0)
d4a288ef 2479 do_branch(dc, target, insn, 0);
0f8a249a 2480 else if (cc == 2)
d4a288ef 2481 do_branch(dc, target, insn, 1);
0f8a249a
BS
2482 else
2483 goto illegal_insn;
2484 goto jmp_insn;
2485 }
2486 case 0x3: /* V9 BPr */
2487 {
2488 target = GET_FIELD_SP(insn, 0, 13) |
13846e70 2489 (GET_FIELD_SP(insn, 20, 21) << 14);
0f8a249a
BS
2490 target = sign_extend(target, 16);
2491 target <<= 2;
9322a4bf 2492 cpu_src1 = get_src1(insn, cpu_src1);
d4a288ef 2493 do_branch_reg(dc, target, insn, cpu_src1);
0f8a249a
BS
2494 goto jmp_insn;
2495 }
2496 case 0x5: /* V9 FBPcc */
2497 {
2498 int cc = GET_FIELD_SP(insn, 20, 21);
5b12f1e8 2499 if (gen_trap_ifnofpu(dc)) {
a80dde08 2500 goto jmp_insn;
5b12f1e8 2501 }
0f8a249a
BS
2502 target = GET_FIELD_SP(insn, 0, 18);
2503 target = sign_extend(target, 19);
2504 target <<= 2;
d4a288ef 2505 do_fbranch(dc, target, insn, cc);
0f8a249a
BS
2506 goto jmp_insn;
2507 }
a4d17f19 2508#else
0f8a249a
BS
2509 case 0x7: /* CBN+x */
2510 {
2511 goto ncp_insn;
2512 }
2513#endif
2514 case 0x2: /* BN+x */
2515 {
2516 target = GET_FIELD(insn, 10, 31);
2517 target = sign_extend(target, 22);
2518 target <<= 2;
d4a288ef 2519 do_branch(dc, target, insn, 0);
0f8a249a
BS
2520 goto jmp_insn;
2521 }
2522 case 0x6: /* FBN+x */
2523 {
5b12f1e8 2524 if (gen_trap_ifnofpu(dc)) {
a80dde08 2525 goto jmp_insn;
5b12f1e8 2526 }
0f8a249a
BS
2527 target = GET_FIELD(insn, 10, 31);
2528 target = sign_extend(target, 22);
2529 target <<= 2;
d4a288ef 2530 do_fbranch(dc, target, insn, 0);
0f8a249a
BS
2531 goto jmp_insn;
2532 }
2533 case 0x4: /* SETHI */
0f8a249a 2534 if (rd) { // nop
0f8a249a 2535 uint32_t value = GET_FIELD(insn, 10, 31);
2ea815ca
BS
2536 TCGv r_const;
2537
2538 r_const = tcg_const_tl(value << 10);
2539 gen_movl_TN_reg(rd, r_const);
2540 tcg_temp_free(r_const);
0f8a249a 2541 }
0f8a249a
BS
2542 break;
2543 case 0x0: /* UNIMPL */
2544 default:
3475187d 2545 goto illegal_insn;
0f8a249a
BS
2546 }
2547 break;
2548 }
2549 break;
dc1a6971
BS
2550 case 1: /*CALL*/
2551 {
0f8a249a 2552 target_long target = GET_FIELDs(insn, 2, 31) << 2;
2ea815ca 2553 TCGv r_const;
cf495bcf 2554
2ea815ca
BS
2555 r_const = tcg_const_tl(dc->pc);
2556 gen_movl_TN_reg(15, r_const);
2557 tcg_temp_free(r_const);
0f8a249a 2558 target += dc->pc;
13a6dd00 2559 gen_mov_pc_npc(dc);
22036a49
AT
2560#ifdef TARGET_SPARC64
2561 if (unlikely(AM_CHECK(dc))) {
2562 target &= 0xffffffffULL;
2563 }
2564#endif
0f8a249a
BS
2565 dc->npc = target;
2566 }
2567 goto jmp_insn;
2568 case 2: /* FPU & Logical Operations */
2569 {
2570 unsigned int xop = GET_FIELD(insn, 7, 12);
2571 if (xop == 0x3a) { /* generate trap */
bd49ed41
RH
2572 int cond = GET_FIELD(insn, 3, 6);
2573 TCGv_i32 trap;
2574 int l1 = -1, mask;
3475187d 2575
bd49ed41
RH
2576 if (cond == 0) {
2577 /* Trap never. */
2578 break;
cf495bcf 2579 }
b04d9890 2580
bd49ed41 2581 save_state(dc);
b04d9890 2582
bd49ed41
RH
2583 if (cond != 8) {
2584 /* Conditional trap. */
3a49e759 2585 DisasCompare cmp;
3475187d 2586#ifdef TARGET_SPARC64
0f8a249a
BS
2587 /* V9 icc/xcc */
2588 int cc = GET_FIELD_SP(insn, 11, 12);
3a49e759
RH
2589 if (cc == 0) {
2590 gen_compare(&cmp, 0, cond, dc);
2591 } else if (cc == 2) {
2592 gen_compare(&cmp, 1, cond, dc);
2593 } else {
0f8a249a 2594 goto illegal_insn;
3a49e759 2595 }
3475187d 2596#else
3a49e759 2597 gen_compare(&cmp, 0, cond, dc);
3475187d 2598#endif
b158a785 2599 l1 = gen_new_label();
3a49e759
RH
2600 tcg_gen_brcond_tl(tcg_invert_cond(cmp.cond),
2601 cmp.c1, cmp.c2, l1);
2602 free_compare(&cmp);
bd49ed41 2603 }
b158a785 2604
bd49ed41
RH
2605 mask = ((dc->def->features & CPU_FEATURE_HYPV) && supervisor(dc)
2606 ? UA2005_HTRAP_MASK : V8_TRAP_MASK);
2607
2608 /* Don't use the normal temporaries, as they may well have
2609 gone out of scope with the branch above. While we're
2610 doing that we might as well pre-truncate to 32-bit. */
2611 trap = tcg_temp_new_i32();
2612
2613 rs1 = GET_FIELD_SP(insn, 14, 18);
2614 if (IS_IMM) {
2615 rs2 = GET_FIELD_SP(insn, 0, 6);
2616 if (rs1 == 0) {
2617 tcg_gen_movi_i32(trap, (rs2 & mask) + TT_TRAP);
2618 /* Signal that the trap value is fully constant. */
2619 mask = 0;
2620 } else {
2621 TCGv t1 = tcg_temp_new();
2622 gen_movl_reg_TN(rs1, t1);
2623 tcg_gen_trunc_tl_i32(trap, t1);
2624 tcg_temp_free(t1);
2625 tcg_gen_addi_i32(trap, trap, rs2);
2626 }
2627 } else {
2628 TCGv t1 = tcg_temp_new();
2629 TCGv t2 = tcg_temp_new();
2630 rs2 = GET_FIELD_SP(insn, 0, 4);
2631 gen_movl_reg_TN(rs1, t1);
2632 gen_movl_reg_TN(rs2, t2);
2633 tcg_gen_add_tl(t1, t1, t2);
2634 tcg_gen_trunc_tl_i32(trap, t1);
2635 tcg_temp_free(t1);
2636 tcg_temp_free(t2);
2637 }
2638 if (mask != 0) {
2639 tcg_gen_andi_i32(trap, trap, mask);
2640 tcg_gen_addi_i32(trap, trap, TT_TRAP);
2641 }
2642
2643 gen_helper_raise_exception(cpu_env, trap);
2644 tcg_temp_free_i32(trap);
b158a785 2645
fe1755cb
RH
2646 if (cond == 8) {
2647 /* An unconditional trap ends the TB. */
2648 dc->is_br = 1;
2649 goto jmp_insn;
2650 } else {
2651 /* A conditional trap falls through to the next insn. */
b158a785 2652 gen_set_label(l1);
fe1755cb 2653 break;
cf495bcf
FB
2654 }
2655 } else if (xop == 0x28) {
2656 rs1 = GET_FIELD(insn, 13, 17);
2657 switch(rs1) {
2658 case 0: /* rdy */
65fe7b09
BS
2659#ifndef TARGET_SPARC64
2660 case 0x01 ... 0x0e: /* undefined in the SPARCv8
2661 manual, rdy on the microSPARC
2662 II */
2663 case 0x0f: /* stbar in the SPARCv8 manual,
2664 rdy on the microSPARC II */
2665 case 0x10 ... 0x1f: /* implementation-dependent in the
2666 SPARCv8 manual, rdy on the
2667 microSPARC II */
4a2ba232
FC
2668 /* Read Asr17 */
2669 if (rs1 == 0x11 && dc->def->features & CPU_FEATURE_ASR17) {
2670 TCGv r_const;
2671
2672 /* Read Asr17 for a Leon3 monoprocessor */
2673 r_const = tcg_const_tl((1 << 8)
2674 | (dc->def->nwindows - 1));
2675 gen_movl_TN_reg(rd, r_const);
2676 tcg_temp_free(r_const);
2677 break;
2678 }
65fe7b09 2679#endif
255e1fcb 2680 gen_movl_TN_reg(rd, cpu_y);
cf495bcf 2681 break;
3475187d 2682#ifdef TARGET_SPARC64
0f8a249a 2683 case 0x2: /* V9 rdccr */
2ffd9176 2684 gen_helper_compute_psr(cpu_env);
063c3675 2685 gen_helper_rdccr(cpu_dst, cpu_env);
6ae20372 2686 gen_movl_TN_reg(rd, cpu_dst);
3475187d 2687 break;
0f8a249a 2688 case 0x3: /* V9 rdasi */
255e1fcb 2689 tcg_gen_ext_i32_tl(cpu_dst, cpu_asi);
6ae20372 2690 gen_movl_TN_reg(rd, cpu_dst);
3475187d 2691 break;
0f8a249a 2692 case 0x4: /* V9 rdtick */
ccd4a219 2693 {
a7812ae4 2694 TCGv_ptr r_tickptr;
ccd4a219 2695
a7812ae4 2696 r_tickptr = tcg_temp_new_ptr();
ccd4a219 2697 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 2698 offsetof(CPUSPARCState, tick));
a7812ae4
PB
2699 gen_helper_tick_get_count(cpu_dst, r_tickptr);
2700 tcg_temp_free_ptr(r_tickptr);
6ae20372 2701 gen_movl_TN_reg(rd, cpu_dst);
ccd4a219 2702 }
3475187d 2703 break;
0f8a249a 2704 case 0x5: /* V9 rdpc */
2ea815ca
BS
2705 {
2706 TCGv r_const;
2707
22036a49
AT
2708 if (unlikely(AM_CHECK(dc))) {
2709 r_const = tcg_const_tl(dc->pc & 0xffffffffULL);
2710 } else {
2711 r_const = tcg_const_tl(dc->pc);
2712 }
2ea815ca
BS
2713 gen_movl_TN_reg(rd, r_const);
2714 tcg_temp_free(r_const);
2715 }
0f8a249a
BS
2716 break;
2717 case 0x6: /* V9 rdfprs */
255e1fcb 2718 tcg_gen_ext_i32_tl(cpu_dst, cpu_fprs);
6ae20372 2719 gen_movl_TN_reg(rd, cpu_dst);
3475187d 2720 break;
65fe7b09
BS
2721 case 0xf: /* V9 membar */
2722 break; /* no effect */
0f8a249a 2723 case 0x13: /* Graphics Status */
5b12f1e8 2724 if (gen_trap_ifnofpu(dc)) {
725cb90b 2725 goto jmp_insn;
5b12f1e8 2726 }
255e1fcb 2727 gen_movl_TN_reg(rd, cpu_gsr);
725cb90b 2728 break;
9d926598
BS
2729 case 0x16: /* Softint */
2730 tcg_gen_ext_i32_tl(cpu_dst, cpu_softint);
2731 gen_movl_TN_reg(rd, cpu_dst);
2732 break;
0f8a249a 2733 case 0x17: /* Tick compare */
255e1fcb 2734 gen_movl_TN_reg(rd, cpu_tick_cmpr);
83469015 2735 break;
0f8a249a 2736 case 0x18: /* System tick */
ccd4a219 2737 {
a7812ae4 2738 TCGv_ptr r_tickptr;
ccd4a219 2739
a7812ae4 2740 r_tickptr = tcg_temp_new_ptr();
ccd4a219 2741 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 2742 offsetof(CPUSPARCState, stick));
a7812ae4
PB
2743 gen_helper_tick_get_count(cpu_dst, r_tickptr);
2744 tcg_temp_free_ptr(r_tickptr);
6ae20372 2745 gen_movl_TN_reg(rd, cpu_dst);
ccd4a219 2746 }
83469015 2747 break;
0f8a249a 2748 case 0x19: /* System tick compare */
255e1fcb 2749 gen_movl_TN_reg(rd, cpu_stick_cmpr);
83469015 2750 break;
0f8a249a
BS
2751 case 0x10: /* Performance Control */
2752 case 0x11: /* Performance Instrumentation Counter */
2753 case 0x12: /* Dispatch Control */
2754 case 0x14: /* Softint set, WO */
2755 case 0x15: /* Softint clear, WO */
3475187d
FB
2756#endif
2757 default:
cf495bcf
FB
2758 goto illegal_insn;
2759 }
e8af50a3 2760#if !defined(CONFIG_USER_ONLY)
e9ebed4d 2761 } else if (xop == 0x29) { /* rdpsr / UA2005 rdhpr */
3475187d 2762#ifndef TARGET_SPARC64
0f8a249a
BS
2763 if (!supervisor(dc))
2764 goto priv_insn;
2ffd9176 2765 gen_helper_compute_psr(cpu_env);
8393617c 2766 dc->cc_op = CC_OP_FLAGS;
063c3675 2767 gen_helper_rdpsr(cpu_dst, cpu_env);
e9ebed4d 2768#else
fb79ceb9 2769 CHECK_IU_FEATURE(dc, HYPV);
e9ebed4d
BS
2770 if (!hypervisor(dc))
2771 goto priv_insn;
2772 rs1 = GET_FIELD(insn, 13, 17);
2773 switch (rs1) {
2774 case 0: // hpstate
2775 // gen_op_rdhpstate();
2776 break;
2777 case 1: // htstate
2778 // gen_op_rdhtstate();
2779 break;
2780 case 3: // hintp
255e1fcb 2781 tcg_gen_mov_tl(cpu_dst, cpu_hintp);
e9ebed4d
BS
2782 break;
2783 case 5: // htba
255e1fcb 2784 tcg_gen_mov_tl(cpu_dst, cpu_htba);
e9ebed4d
BS
2785 break;
2786 case 6: // hver
255e1fcb 2787 tcg_gen_mov_tl(cpu_dst, cpu_hver);
e9ebed4d
BS
2788 break;
2789 case 31: // hstick_cmpr
255e1fcb 2790 tcg_gen_mov_tl(cpu_dst, cpu_hstick_cmpr);
e9ebed4d
BS
2791 break;
2792 default:
2793 goto illegal_insn;
2794 }
2795#endif
6ae20372 2796 gen_movl_TN_reg(rd, cpu_dst);
e8af50a3 2797 break;
3475187d 2798 } else if (xop == 0x2a) { /* rdwim / V9 rdpr */
0f8a249a
BS
2799 if (!supervisor(dc))
2800 goto priv_insn;
3475187d
FB
2801#ifdef TARGET_SPARC64
2802 rs1 = GET_FIELD(insn, 13, 17);
0f8a249a
BS
2803 switch (rs1) {
2804 case 0: // tpc
375ee38b 2805 {
a7812ae4 2806 TCGv_ptr r_tsptr;
375ee38b 2807
a7812ae4 2808 r_tsptr = tcg_temp_new_ptr();
8194f35a 2809 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
a7812ae4 2810 tcg_gen_ld_tl(cpu_tmp0, r_tsptr,
375ee38b 2811 offsetof(trap_state, tpc));
a7812ae4 2812 tcg_temp_free_ptr(r_tsptr);
375ee38b 2813 }
0f8a249a
BS
2814 break;
2815 case 1: // tnpc
375ee38b 2816 {
a7812ae4 2817 TCGv_ptr r_tsptr;
375ee38b 2818
a7812ae4 2819 r_tsptr = tcg_temp_new_ptr();
8194f35a 2820 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
ece43b8d 2821 tcg_gen_ld_tl(cpu_tmp0, r_tsptr,
375ee38b 2822 offsetof(trap_state, tnpc));
a7812ae4 2823 tcg_temp_free_ptr(r_tsptr);
375ee38b 2824 }
0f8a249a
BS
2825 break;
2826 case 2: // tstate
375ee38b 2827 {
a7812ae4 2828 TCGv_ptr r_tsptr;
375ee38b 2829
a7812ae4 2830 r_tsptr = tcg_temp_new_ptr();
8194f35a 2831 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
ece43b8d 2832 tcg_gen_ld_tl(cpu_tmp0, r_tsptr,
375ee38b 2833 offsetof(trap_state, tstate));
a7812ae4 2834 tcg_temp_free_ptr(r_tsptr);
375ee38b 2835 }
0f8a249a
BS
2836 break;
2837 case 3: // tt
375ee38b 2838 {
a7812ae4 2839 TCGv_ptr r_tsptr;
375ee38b 2840
a7812ae4 2841 r_tsptr = tcg_temp_new_ptr();
8194f35a 2842 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
a7812ae4 2843 tcg_gen_ld_i32(cpu_tmp32, r_tsptr,
375ee38b 2844 offsetof(trap_state, tt));
a7812ae4
PB
2845 tcg_temp_free_ptr(r_tsptr);
2846 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
375ee38b 2847 }
0f8a249a
BS
2848 break;
2849 case 4: // tick
ccd4a219 2850 {
a7812ae4 2851 TCGv_ptr r_tickptr;
ccd4a219 2852
a7812ae4 2853 r_tickptr = tcg_temp_new_ptr();
ccd4a219 2854 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 2855 offsetof(CPUSPARCState, tick));
a7812ae4 2856 gen_helper_tick_get_count(cpu_tmp0, r_tickptr);
ece43b8d 2857 gen_movl_TN_reg(rd, cpu_tmp0);
a7812ae4 2858 tcg_temp_free_ptr(r_tickptr);
ccd4a219 2859 }
0f8a249a
BS
2860 break;
2861 case 5: // tba
255e1fcb 2862 tcg_gen_mov_tl(cpu_tmp0, cpu_tbr);
0f8a249a
BS
2863 break;
2864 case 6: // pstate
77f193da
BS
2865 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2866 offsetof(CPUSPARCState, pstate));
ece43b8d 2867 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2868 break;
2869 case 7: // tl
77f193da
BS
2870 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2871 offsetof(CPUSPARCState, tl));
ece43b8d 2872 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2873 break;
2874 case 8: // pil
77f193da
BS
2875 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2876 offsetof(CPUSPARCState, psrpil));
ece43b8d 2877 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2878 break;
2879 case 9: // cwp
063c3675 2880 gen_helper_rdcwp(cpu_tmp0, cpu_env);
0f8a249a
BS
2881 break;
2882 case 10: // cansave
77f193da
BS
2883 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2884 offsetof(CPUSPARCState, cansave));
ece43b8d 2885 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2886 break;
2887 case 11: // canrestore
77f193da
BS
2888 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2889 offsetof(CPUSPARCState, canrestore));
ece43b8d 2890 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2891 break;
2892 case 12: // cleanwin
77f193da
BS
2893 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2894 offsetof(CPUSPARCState, cleanwin));
ece43b8d 2895 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2896 break;
2897 case 13: // otherwin
77f193da
BS
2898 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2899 offsetof(CPUSPARCState, otherwin));
ece43b8d 2900 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a
BS
2901 break;
2902 case 14: // wstate
77f193da
BS
2903 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2904 offsetof(CPUSPARCState, wstate));
ece43b8d 2905 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
0f8a249a 2906 break;
e9ebed4d 2907 case 16: // UA2005 gl
fb79ceb9 2908 CHECK_IU_FEATURE(dc, GL);
77f193da
BS
2909 tcg_gen_ld_i32(cpu_tmp32, cpu_env,
2910 offsetof(CPUSPARCState, gl));
ece43b8d 2911 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_tmp32);
e9ebed4d
BS
2912 break;
2913 case 26: // UA2005 strand status
fb79ceb9 2914 CHECK_IU_FEATURE(dc, HYPV);
e9ebed4d
BS
2915 if (!hypervisor(dc))
2916 goto priv_insn;
527067d8 2917 tcg_gen_mov_tl(cpu_tmp0, cpu_ssr);
e9ebed4d 2918 break;
0f8a249a 2919 case 31: // ver
255e1fcb 2920 tcg_gen_mov_tl(cpu_tmp0, cpu_ver);
0f8a249a
BS
2921 break;
2922 case 15: // fq
2923 default:
2924 goto illegal_insn;
2925 }
3475187d 2926#else
255e1fcb 2927 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_wim);
3475187d 2928#endif
ece43b8d 2929 gen_movl_TN_reg(rd, cpu_tmp0);
e8af50a3 2930 break;
3475187d
FB
2931 } else if (xop == 0x2b) { /* rdtbr / V9 flushw */
2932#ifdef TARGET_SPARC64
66442b07 2933 save_state(dc);
063c3675 2934 gen_helper_flushw(cpu_env);
3475187d 2935#else
0f8a249a
BS
2936 if (!supervisor(dc))
2937 goto priv_insn;
255e1fcb 2938 gen_movl_TN_reg(rd, cpu_tbr);
3475187d 2939#endif
e8af50a3
FB
2940 break;
2941#endif
0f8a249a 2942 } else if (xop == 0x34) { /* FPU Operations */
5b12f1e8 2943 if (gen_trap_ifnofpu(dc)) {
a80dde08 2944 goto jmp_insn;
5b12f1e8 2945 }
0f8a249a 2946 gen_op_clear_ieee_excp_and_FTT();
e8af50a3 2947 rs1 = GET_FIELD(insn, 13, 17);
0f8a249a
BS
2948 rs2 = GET_FIELD(insn, 27, 31);
2949 xop = GET_FIELD(insn, 18, 26);
66442b07 2950 save_state(dc);
0f8a249a 2951 switch (xop) {
dc1a6971 2952 case 0x1: /* fmovs */
208ae657
RH
2953 cpu_src1_32 = gen_load_fpr_F(dc, rs2);
2954 gen_store_fpr_F(dc, rd, cpu_src1_32);
dc1a6971
BS
2955 break;
2956 case 0x5: /* fnegs */
61f17f6e 2957 gen_ne_fop_FF(dc, rd, rs2, gen_helper_fnegs);
dc1a6971
BS
2958 break;
2959 case 0x9: /* fabss */
61f17f6e 2960 gen_ne_fop_FF(dc, rd, rs2, gen_helper_fabss);
dc1a6971
BS
2961 break;
2962 case 0x29: /* fsqrts */
2963 CHECK_FPU_FEATURE(dc, FSQRT);
61f17f6e 2964 gen_fop_FF(dc, rd, rs2, gen_helper_fsqrts);
dc1a6971
BS
2965 break;
2966 case 0x2a: /* fsqrtd */
2967 CHECK_FPU_FEATURE(dc, FSQRT);
61f17f6e 2968 gen_fop_DD(dc, rd, rs2, gen_helper_fsqrtd);
dc1a6971
BS
2969 break;
2970 case 0x2b: /* fsqrtq */
2971 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 2972 gen_fop_QQ(dc, rd, rs2, gen_helper_fsqrtq);
dc1a6971
BS
2973 break;
2974 case 0x41: /* fadds */
61f17f6e 2975 gen_fop_FFF(dc, rd, rs1, rs2, gen_helper_fadds);
dc1a6971
BS
2976 break;
2977 case 0x42: /* faddd */
61f17f6e 2978 gen_fop_DDD(dc, rd, rs1, rs2, gen_helper_faddd);
dc1a6971
BS
2979 break;
2980 case 0x43: /* faddq */
2981 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 2982 gen_fop_QQQ(dc, rd, rs1, rs2, gen_helper_faddq);
dc1a6971
BS
2983 break;
2984 case 0x45: /* fsubs */
61f17f6e 2985 gen_fop_FFF(dc, rd, rs1, rs2, gen_helper_fsubs);
dc1a6971
BS
2986 break;
2987 case 0x46: /* fsubd */
61f17f6e 2988 gen_fop_DDD(dc, rd, rs1, rs2, gen_helper_fsubd);
dc1a6971
BS
2989 break;
2990 case 0x47: /* fsubq */
2991 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 2992 gen_fop_QQQ(dc, rd, rs1, rs2, gen_helper_fsubq);
dc1a6971
BS
2993 break;
2994 case 0x49: /* fmuls */
2995 CHECK_FPU_FEATURE(dc, FMUL);
61f17f6e 2996 gen_fop_FFF(dc, rd, rs1, rs2, gen_helper_fmuls);
dc1a6971
BS
2997 break;
2998 case 0x4a: /* fmuld */
2999 CHECK_FPU_FEATURE(dc, FMUL);
61f17f6e 3000 gen_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmuld);
dc1a6971
BS
3001 break;
3002 case 0x4b: /* fmulq */
3003 CHECK_FPU_FEATURE(dc, FLOAT128);
3004 CHECK_FPU_FEATURE(dc, FMUL);
61f17f6e 3005 gen_fop_QQQ(dc, rd, rs1, rs2, gen_helper_fmulq);
dc1a6971
BS
3006 break;
3007 case 0x4d: /* fdivs */
61f17f6e 3008 gen_fop_FFF(dc, rd, rs1, rs2, gen_helper_fdivs);
dc1a6971
BS
3009 break;
3010 case 0x4e: /* fdivd */
61f17f6e 3011 gen_fop_DDD(dc, rd, rs1, rs2, gen_helper_fdivd);
dc1a6971
BS
3012 break;
3013 case 0x4f: /* fdivq */
3014 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3015 gen_fop_QQQ(dc, rd, rs1, rs2, gen_helper_fdivq);
dc1a6971
BS
3016 break;
3017 case 0x69: /* fsmuld */
3018 CHECK_FPU_FEATURE(dc, FSMULD);
61f17f6e 3019 gen_fop_DFF(dc, rd, rs1, rs2, gen_helper_fsmuld);
dc1a6971
BS
3020 break;
3021 case 0x6e: /* fdmulq */
3022 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3023 gen_fop_QDD(dc, rd, rs1, rs2, gen_helper_fdmulq);
dc1a6971
BS
3024 break;
3025 case 0xc4: /* fitos */
61f17f6e 3026 gen_fop_FF(dc, rd, rs2, gen_helper_fitos);
dc1a6971
BS
3027 break;
3028 case 0xc6: /* fdtos */
61f17f6e 3029 gen_fop_FD(dc, rd, rs2, gen_helper_fdtos);
dc1a6971
BS
3030 break;
3031 case 0xc7: /* fqtos */
3032 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3033 gen_fop_FQ(dc, rd, rs2, gen_helper_fqtos);
dc1a6971
BS
3034 break;
3035 case 0xc8: /* fitod */
61f17f6e 3036 gen_ne_fop_DF(dc, rd, rs2, gen_helper_fitod);
dc1a6971
BS
3037 break;
3038 case 0xc9: /* fstod */
61f17f6e 3039 gen_ne_fop_DF(dc, rd, rs2, gen_helper_fstod);
dc1a6971
BS
3040 break;
3041 case 0xcb: /* fqtod */
3042 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3043 gen_fop_DQ(dc, rd, rs2, gen_helper_fqtod);
dc1a6971
BS
3044 break;
3045 case 0xcc: /* fitoq */
3046 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3047 gen_ne_fop_QF(dc, rd, rs2, gen_helper_fitoq);
dc1a6971
BS
3048 break;
3049 case 0xcd: /* fstoq */
3050 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3051 gen_ne_fop_QF(dc, rd, rs2, gen_helper_fstoq);
dc1a6971
BS
3052 break;
3053 case 0xce: /* fdtoq */
3054 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3055 gen_ne_fop_QD(dc, rd, rs2, gen_helper_fdtoq);
dc1a6971
BS
3056 break;
3057 case 0xd1: /* fstoi */
61f17f6e 3058 gen_fop_FF(dc, rd, rs2, gen_helper_fstoi);
dc1a6971
BS
3059 break;
3060 case 0xd2: /* fdtoi */
61f17f6e 3061 gen_fop_FD(dc, rd, rs2, gen_helper_fdtoi);
dc1a6971
BS
3062 break;
3063 case 0xd3: /* fqtoi */
3064 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3065 gen_fop_FQ(dc, rd, rs2, gen_helper_fqtoi);
dc1a6971 3066 break;
3475187d 3067#ifdef TARGET_SPARC64
dc1a6971 3068 case 0x2: /* V9 fmovd */
96eda024
RH
3069 cpu_src1_64 = gen_load_fpr_D(dc, rs2);
3070 gen_store_fpr_D(dc, rd, cpu_src1_64);
dc1a6971
BS
3071 break;
3072 case 0x3: /* V9 fmovq */
3073 CHECK_FPU_FEATURE(dc, FLOAT128);
ac11f776 3074 gen_move_Q(rd, rs2);
dc1a6971
BS
3075 break;
3076 case 0x6: /* V9 fnegd */
61f17f6e 3077 gen_ne_fop_DD(dc, rd, rs2, gen_helper_fnegd);
dc1a6971
BS
3078 break;
3079 case 0x7: /* V9 fnegq */
3080 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3081 gen_ne_fop_QQ(dc, rd, rs2, gen_helper_fnegq);
dc1a6971
BS
3082 break;
3083 case 0xa: /* V9 fabsd */
61f17f6e 3084 gen_ne_fop_DD(dc, rd, rs2, gen_helper_fabsd);
dc1a6971
BS
3085 break;
3086 case 0xb: /* V9 fabsq */
3087 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3088 gen_ne_fop_QQ(dc, rd, rs2, gen_helper_fabsq);
dc1a6971
BS
3089 break;
3090 case 0x81: /* V9 fstox */
61f17f6e 3091 gen_fop_DF(dc, rd, rs2, gen_helper_fstox);
dc1a6971
BS
3092 break;
3093 case 0x82: /* V9 fdtox */
61f17f6e 3094 gen_fop_DD(dc, rd, rs2, gen_helper_fdtox);
dc1a6971
BS
3095 break;
3096 case 0x83: /* V9 fqtox */
3097 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3098 gen_fop_DQ(dc, rd, rs2, gen_helper_fqtox);
dc1a6971
BS
3099 break;
3100 case 0x84: /* V9 fxtos */
61f17f6e 3101 gen_fop_FD(dc, rd, rs2, gen_helper_fxtos);
dc1a6971
BS
3102 break;
3103 case 0x88: /* V9 fxtod */
61f17f6e 3104 gen_fop_DD(dc, rd, rs2, gen_helper_fxtod);
dc1a6971
BS
3105 break;
3106 case 0x8c: /* V9 fxtoq */
3107 CHECK_FPU_FEATURE(dc, FLOAT128);
61f17f6e 3108 gen_ne_fop_QD(dc, rd, rs2, gen_helper_fxtoq);
dc1a6971 3109 break;
0f8a249a 3110#endif
dc1a6971
BS
3111 default:
3112 goto illegal_insn;
0f8a249a
BS
3113 }
3114 } else if (xop == 0x35) { /* FPU Operations */
3475187d 3115#ifdef TARGET_SPARC64
0f8a249a 3116 int cond;
3475187d 3117#endif
5b12f1e8 3118 if (gen_trap_ifnofpu(dc)) {
a80dde08 3119 goto jmp_insn;
5b12f1e8 3120 }
0f8a249a 3121 gen_op_clear_ieee_excp_and_FTT();
cf495bcf 3122 rs1 = GET_FIELD(insn, 13, 17);
0f8a249a
BS
3123 rs2 = GET_FIELD(insn, 27, 31);
3124 xop = GET_FIELD(insn, 18, 26);
66442b07 3125 save_state(dc);
dcf24905 3126
690995a6
RH
3127#ifdef TARGET_SPARC64
3128#define FMOVR(sz) \
3129 do { \
3130 DisasCompare cmp; \
3131 cond = GET_FIELD_SP(insn, 14, 17); \
3132 cpu_src1 = get_src1(insn, cpu_src1); \
3133 gen_compare_reg(&cmp, cond, cpu_src1); \
3134 gen_fmov##sz(dc, &cmp, rd, rs2); \
3135 free_compare(&cmp); \
3136 } while (0)
3137
3138 if ((xop & 0x11f) == 0x005) { /* V9 fmovsr */
3139 FMOVR(s);
0f8a249a
BS
3140 break;
3141 } else if ((xop & 0x11f) == 0x006) { // V9 fmovdr
690995a6 3142 FMOVR(d);
0f8a249a
BS
3143 break;
3144 } else if ((xop & 0x11f) == 0x007) { // V9 fmovqr
64a88d5d 3145 CHECK_FPU_FEATURE(dc, FLOAT128);
690995a6 3146 FMOVR(q);
1f587329 3147 break;
0f8a249a 3148 }
690995a6 3149#undef FMOVR
0f8a249a
BS
3150#endif
3151 switch (xop) {
3475187d 3152#ifdef TARGET_SPARC64
7e480893
RH
3153#define FMOVCC(fcc, sz) \
3154 do { \
3155 DisasCompare cmp; \
714547bb 3156 cond = GET_FIELD_SP(insn, 14, 17); \
7e480893
RH
3157 gen_fcompare(&cmp, fcc, cond); \
3158 gen_fmov##sz(dc, &cmp, rd, rs2); \
3159 free_compare(&cmp); \
3160 } while (0)
3161
0f8a249a 3162 case 0x001: /* V9 fmovscc %fcc0 */
7e480893 3163 FMOVCC(0, s);
0f8a249a
BS
3164 break;
3165 case 0x002: /* V9 fmovdcc %fcc0 */
7e480893 3166 FMOVCC(0, d);
0f8a249a
BS
3167 break;
3168 case 0x003: /* V9 fmovqcc %fcc0 */
64a88d5d 3169 CHECK_FPU_FEATURE(dc, FLOAT128);
7e480893 3170 FMOVCC(0, q);
1f587329 3171 break;
0f8a249a 3172 case 0x041: /* V9 fmovscc %fcc1 */
7e480893 3173 FMOVCC(1, s);
0f8a249a
BS
3174 break;
3175 case 0x042: /* V9 fmovdcc %fcc1 */
7e480893 3176 FMOVCC(1, d);
0f8a249a
BS
3177 break;
3178 case 0x043: /* V9 fmovqcc %fcc1 */
64a88d5d 3179 CHECK_FPU_FEATURE(dc, FLOAT128);
7e480893 3180 FMOVCC(1, q);
1f587329 3181 break;
0f8a249a 3182 case 0x081: /* V9 fmovscc %fcc2 */
7e480893 3183 FMOVCC(2, s);
0f8a249a
BS
3184 break;
3185 case 0x082: /* V9 fmovdcc %fcc2 */
7e480893 3186 FMOVCC(2, d);
0f8a249a
BS
3187 break;
3188 case 0x083: /* V9 fmovqcc %fcc2 */
64a88d5d 3189 CHECK_FPU_FEATURE(dc, FLOAT128);
7e480893 3190 FMOVCC(2, q);
1f587329 3191 break;
0f8a249a 3192 case 0x0c1: /* V9 fmovscc %fcc3 */
7e480893 3193 FMOVCC(3, s);
0f8a249a
BS
3194 break;
3195 case 0x0c2: /* V9 fmovdcc %fcc3 */
7e480893 3196 FMOVCC(3, d);
0f8a249a
BS
3197 break;
3198 case 0x0c3: /* V9 fmovqcc %fcc3 */
64a88d5d 3199 CHECK_FPU_FEATURE(dc, FLOAT128);
7e480893 3200 FMOVCC(3, q);
1f587329 3201 break;
7e480893
RH
3202#undef FMOVCC
3203#define FMOVCC(xcc, sz) \
3204 do { \
3205 DisasCompare cmp; \
714547bb 3206 cond = GET_FIELD_SP(insn, 14, 17); \
7e480893
RH
3207 gen_compare(&cmp, xcc, cond, dc); \
3208 gen_fmov##sz(dc, &cmp, rd, rs2); \
3209 free_compare(&cmp); \
3210 } while (0)
19f329ad 3211
0f8a249a 3212 case 0x101: /* V9 fmovscc %icc */
7e480893 3213 FMOVCC(0, s);
0f8a249a
BS
3214 break;
3215 case 0x102: /* V9 fmovdcc %icc */
7e480893 3216 FMOVCC(0, d);
b7d69dc2 3217 break;
0f8a249a 3218 case 0x103: /* V9 fmovqcc %icc */
64a88d5d 3219 CHECK_FPU_FEATURE(dc, FLOAT128);
7e480893 3220 FMOVCC(0, q);
1f587329 3221 break;
0f8a249a 3222 case 0x181: /* V9 fmovscc %xcc */
7e480893 3223 FMOVCC(1, s);
0f8a249a
BS
3224 break;
3225 case 0x182: /* V9 fmovdcc %xcc */
7e480893 3226 FMOVCC(1, d);
0f8a249a
BS
3227 break;
3228 case 0x183: /* V9 fmovqcc %xcc */
64a88d5d 3229 CHECK_FPU_FEATURE(dc, FLOAT128);
7e480893 3230 FMOVCC(1, q);
1f587329 3231 break;
7e480893 3232#undef FMOVCC
1f587329
BS
3233#endif
3234 case 0x51: /* fcmps, V9 %fcc */
208ae657
RH
3235 cpu_src1_32 = gen_load_fpr_F(dc, rs1);
3236 cpu_src2_32 = gen_load_fpr_F(dc, rs2);
3237 gen_op_fcmps(rd & 3, cpu_src1_32, cpu_src2_32);
0f8a249a 3238 break;
1f587329 3239 case 0x52: /* fcmpd, V9 %fcc */
03fb8cfc
RH
3240 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
3241 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
3242 gen_op_fcmpd(rd & 3, cpu_src1_64, cpu_src2_64);
0f8a249a 3243 break;
1f587329 3244 case 0x53: /* fcmpq, V9 %fcc */
64a88d5d 3245 CHECK_FPU_FEATURE(dc, FLOAT128);
1f587329
BS
3246 gen_op_load_fpr_QT0(QFPREG(rs1));
3247 gen_op_load_fpr_QT1(QFPREG(rs2));
7e8c2b6c 3248 gen_op_fcmpq(rd & 3);
1f587329 3249 break;
0f8a249a 3250 case 0x55: /* fcmpes, V9 %fcc */
208ae657
RH
3251 cpu_src1_32 = gen_load_fpr_F(dc, rs1);
3252 cpu_src2_32 = gen_load_fpr_F(dc, rs2);
3253 gen_op_fcmpes(rd & 3, cpu_src1_32, cpu_src2_32);
0f8a249a
BS
3254 break;
3255 case 0x56: /* fcmped, V9 %fcc */
03fb8cfc
RH
3256 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
3257 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
3258 gen_op_fcmped(rd & 3, cpu_src1_64, cpu_src2_64);
0f8a249a 3259 break;
1f587329 3260 case 0x57: /* fcmpeq, V9 %fcc */
64a88d5d 3261 CHECK_FPU_FEATURE(dc, FLOAT128);
1f587329
BS
3262 gen_op_load_fpr_QT0(QFPREG(rs1));
3263 gen_op_load_fpr_QT1(QFPREG(rs2));
7e8c2b6c 3264 gen_op_fcmpeq(rd & 3);
1f587329 3265 break;
0f8a249a
BS
3266 default:
3267 goto illegal_insn;
3268 }
0f8a249a
BS
3269 } else if (xop == 0x2) {
3270 // clr/mov shortcut
e80cfcfc
FB
3271
3272 rs1 = GET_FIELD(insn, 13, 17);
0f8a249a 3273 if (rs1 == 0) {
1a2fb1c0 3274 // or %g0, x, y -> mov T0, x; mov y, T0
0f8a249a 3275 if (IS_IMM) { /* immediate */
2ea815ca
BS
3276 TCGv r_const;
3277
67526b20
BS
3278 simm = GET_FIELDs(insn, 19, 31);
3279 r_const = tcg_const_tl(simm);
2ea815ca
BS
3280 gen_movl_TN_reg(rd, r_const);
3281 tcg_temp_free(r_const);
0f8a249a
BS
3282 } else { /* register */
3283 rs2 = GET_FIELD(insn, 27, 31);
6ae20372 3284 gen_movl_reg_TN(rs2, cpu_dst);
9c6c6662 3285 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a 3286 }
0f8a249a 3287 } else {
9322a4bf 3288 cpu_src1 = get_src1(insn, cpu_src1);
0f8a249a 3289 if (IS_IMM) { /* immediate */
67526b20
BS
3290 simm = GET_FIELDs(insn, 19, 31);
3291 tcg_gen_ori_tl(cpu_dst, cpu_src1, simm);
9c6c6662 3292 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a
BS
3293 } else { /* register */
3294 // or x, %g0, y -> mov T1, x; mov y, T1
3295 rs2 = GET_FIELD(insn, 27, 31);
3296 if (rs2 != 0) {
6ae20372
BS
3297 gen_movl_reg_TN(rs2, cpu_src2);
3298 tcg_gen_or_tl(cpu_dst, cpu_src1, cpu_src2);
9c6c6662 3299 gen_movl_TN_reg(rd, cpu_dst);
6f551262 3300 } else
9c6c6662 3301 gen_movl_TN_reg(rd, cpu_src1);
0f8a249a 3302 }
0f8a249a 3303 }
83469015 3304#ifdef TARGET_SPARC64
0f8a249a 3305 } else if (xop == 0x25) { /* sll, V9 sllx */
9322a4bf 3306 cpu_src1 = get_src1(insn, cpu_src1);
0f8a249a 3307 if (IS_IMM) { /* immediate */
67526b20 3308 simm = GET_FIELDs(insn, 20, 31);
1a2fb1c0 3309 if (insn & (1 << 12)) {
67526b20 3310 tcg_gen_shli_i64(cpu_dst, cpu_src1, simm & 0x3f);
1a2fb1c0 3311 } else {
67526b20 3312 tcg_gen_shli_i64(cpu_dst, cpu_src1, simm & 0x1f);
1a2fb1c0 3313 }
0f8a249a 3314 } else { /* register */
83469015 3315 rs2 = GET_FIELD(insn, 27, 31);
6ae20372 3316 gen_movl_reg_TN(rs2, cpu_src2);
1a2fb1c0 3317 if (insn & (1 << 12)) {
6ae20372 3318 tcg_gen_andi_i64(cpu_tmp0, cpu_src2, 0x3f);
1a2fb1c0 3319 } else {
6ae20372 3320 tcg_gen_andi_i64(cpu_tmp0, cpu_src2, 0x1f);
1a2fb1c0 3321 }
01b1fa6d 3322 tcg_gen_shl_i64(cpu_dst, cpu_src1, cpu_tmp0);
83469015 3323 }
6ae20372 3324 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a 3325 } else if (xop == 0x26) { /* srl, V9 srlx */
9322a4bf 3326 cpu_src1 = get_src1(insn, cpu_src1);
0f8a249a 3327 if (IS_IMM) { /* immediate */
67526b20 3328 simm = GET_FIELDs(insn, 20, 31);
1a2fb1c0 3329 if (insn & (1 << 12)) {
67526b20 3330 tcg_gen_shri_i64(cpu_dst, cpu_src1, simm & 0x3f);
1a2fb1c0 3331 } else {
6ae20372 3332 tcg_gen_andi_i64(cpu_dst, cpu_src1, 0xffffffffULL);
67526b20 3333 tcg_gen_shri_i64(cpu_dst, cpu_dst, simm & 0x1f);
1a2fb1c0 3334 }
0f8a249a 3335 } else { /* register */
83469015 3336 rs2 = GET_FIELD(insn, 27, 31);
6ae20372 3337 gen_movl_reg_TN(rs2, cpu_src2);
1a2fb1c0 3338 if (insn & (1 << 12)) {
6ae20372
BS
3339 tcg_gen_andi_i64(cpu_tmp0, cpu_src2, 0x3f);
3340 tcg_gen_shr_i64(cpu_dst, cpu_src1, cpu_tmp0);
1a2fb1c0 3341 } else {
6ae20372
BS
3342 tcg_gen_andi_i64(cpu_tmp0, cpu_src2, 0x1f);
3343 tcg_gen_andi_i64(cpu_dst, cpu_src1, 0xffffffffULL);
3344 tcg_gen_shr_i64(cpu_dst, cpu_dst, cpu_tmp0);
1a2fb1c0 3345 }
83469015 3346 }
6ae20372 3347 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a 3348 } else if (xop == 0x27) { /* sra, V9 srax */
9322a4bf 3349 cpu_src1 = get_src1(insn, cpu_src1);
0f8a249a 3350 if (IS_IMM) { /* immediate */
67526b20 3351 simm = GET_FIELDs(insn, 20, 31);
1a2fb1c0 3352 if (insn & (1 << 12)) {
67526b20 3353 tcg_gen_sari_i64(cpu_dst, cpu_src1, simm & 0x3f);
1a2fb1c0 3354 } else {
6ae20372 3355 tcg_gen_andi_i64(cpu_dst, cpu_src1, 0xffffffffULL);
527067d8 3356 tcg_gen_ext32s_i64(cpu_dst, cpu_dst);
67526b20 3357 tcg_gen_sari_i64(cpu_dst, cpu_dst, simm & 0x1f);
1a2fb1c0 3358 }
0f8a249a 3359 } else { /* register */
83469015 3360 rs2 = GET_FIELD(insn, 27, 31);
6ae20372 3361 gen_movl_reg_TN(rs2, cpu_src2);
1a2fb1c0 3362 if (insn & (1 << 12)) {
6ae20372
BS
3363 tcg_gen_andi_i64(cpu_tmp0, cpu_src2, 0x3f);
3364 tcg_gen_sar_i64(cpu_dst, cpu_src1, cpu_tmp0);
1a2fb1c0 3365 } else {
6ae20372
BS
3366 tcg_gen_andi_i64(cpu_tmp0, cpu_src2, 0x1f);
3367 tcg_gen_andi_i64(cpu_dst, cpu_src1, 0xffffffffULL);
527067d8 3368 tcg_gen_ext32s_i64(cpu_dst, cpu_dst);
6ae20372 3369 tcg_gen_sar_i64(cpu_dst, cpu_dst, cpu_tmp0);
1a2fb1c0 3370 }
83469015 3371 }
6ae20372 3372 gen_movl_TN_reg(rd, cpu_dst);
e80cfcfc 3373#endif
fcc72045 3374 } else if (xop < 0x36) {
cf495bcf 3375 if (xop < 0x20) {
41d72852
BS
3376 cpu_src1 = get_src1(insn, cpu_src1);
3377 cpu_src2 = get_src2(insn, cpu_src2);
cf495bcf 3378 switch (xop & ~0x10) {
b89e94af 3379 case 0x0: /* add */
41d72852
BS
3380 if (IS_IMM) {
3381 simm = GET_FIELDs(insn, 19, 31);
3382 if (xop & 0x10) {
3383 gen_op_addi_cc(cpu_dst, cpu_src1, simm);
bdf9f35d
BS
3384 tcg_gen_movi_i32(cpu_cc_op, CC_OP_ADD);
3385 dc->cc_op = CC_OP_ADD;
41d72852
BS
3386 } else {
3387 tcg_gen_addi_tl(cpu_dst, cpu_src1, simm);
3388 }
3389 } else {
3390 if (xop & 0x10) {
3391 gen_op_add_cc(cpu_dst, cpu_src1, cpu_src2);
bdf9f35d
BS
3392 tcg_gen_movi_i32(cpu_cc_op, CC_OP_ADD);
3393 dc->cc_op = CC_OP_ADD;
41d72852
BS
3394 } else {
3395 tcg_gen_add_tl(cpu_dst, cpu_src1, cpu_src2);
3396 }
3397 }
cf495bcf 3398 break;
b89e94af 3399 case 0x1: /* and */
41d72852
BS
3400 if (IS_IMM) {
3401 simm = GET_FIELDs(insn, 19, 31);
3402 tcg_gen_andi_tl(cpu_dst, cpu_src1, simm);
3403 } else {
3404 tcg_gen_and_tl(cpu_dst, cpu_src1, cpu_src2);
3405 }
3406 if (xop & 0x10) {
38482a77
BS
3407 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3408 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3409 dc->cc_op = CC_OP_LOGIC;
41d72852 3410 }
cf495bcf 3411 break;
b89e94af 3412 case 0x2: /* or */
41d72852
BS
3413 if (IS_IMM) {
3414 simm = GET_FIELDs(insn, 19, 31);
3415 tcg_gen_ori_tl(cpu_dst, cpu_src1, simm);
3416 } else {
3417 tcg_gen_or_tl(cpu_dst, cpu_src1, cpu_src2);
3418 }
8393617c 3419 if (xop & 0x10) {
38482a77
BS
3420 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3421 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3422 dc->cc_op = CC_OP_LOGIC;
8393617c 3423 }
0f8a249a 3424 break;
b89e94af 3425 case 0x3: /* xor */
41d72852
BS
3426 if (IS_IMM) {
3427 simm = GET_FIELDs(insn, 19, 31);
3428 tcg_gen_xori_tl(cpu_dst, cpu_src1, simm);
3429 } else {
3430 tcg_gen_xor_tl(cpu_dst, cpu_src1, cpu_src2);
3431 }
8393617c 3432 if (xop & 0x10) {
38482a77
BS
3433 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3434 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3435 dc->cc_op = CC_OP_LOGIC;
8393617c 3436 }
cf495bcf 3437 break;
b89e94af 3438 case 0x4: /* sub */
41d72852
BS
3439 if (IS_IMM) {
3440 simm = GET_FIELDs(insn, 19, 31);
3441 if (xop & 0x10) {
d4b0d468 3442 gen_op_subi_cc(cpu_dst, cpu_src1, simm, dc);
41d72852
BS
3443 } else {
3444 tcg_gen_subi_tl(cpu_dst, cpu_src1, simm);
3445 }
3446 } else {
3447 if (xop & 0x10) {
3448 gen_op_sub_cc(cpu_dst, cpu_src1, cpu_src2);
d4b0d468
BS
3449 tcg_gen_movi_i32(cpu_cc_op, CC_OP_SUB);
3450 dc->cc_op = CC_OP_SUB;
41d72852
BS
3451 } else {
3452 tcg_gen_sub_tl(cpu_dst, cpu_src1, cpu_src2);
3453 }
3454 }
cf495bcf 3455 break;
b89e94af 3456 case 0x5: /* andn */
41d72852
BS
3457 if (IS_IMM) {
3458 simm = GET_FIELDs(insn, 19, 31);
3459 tcg_gen_andi_tl(cpu_dst, cpu_src1, ~simm);
3460 } else {
3461 tcg_gen_andc_tl(cpu_dst, cpu_src1, cpu_src2);
3462 }
8393617c 3463 if (xop & 0x10) {
38482a77
BS
3464 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3465 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3466 dc->cc_op = CC_OP_LOGIC;
8393617c 3467 }
cf495bcf 3468 break;
b89e94af 3469 case 0x6: /* orn */
41d72852
BS
3470 if (IS_IMM) {
3471 simm = GET_FIELDs(insn, 19, 31);
3472 tcg_gen_ori_tl(cpu_dst, cpu_src1, ~simm);
3473 } else {
3474 tcg_gen_orc_tl(cpu_dst, cpu_src1, cpu_src2);
3475 }
8393617c 3476 if (xop & 0x10) {
38482a77
BS
3477 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3478 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3479 dc->cc_op = CC_OP_LOGIC;
8393617c 3480 }
cf495bcf 3481 break;
b89e94af 3482 case 0x7: /* xorn */
41d72852
BS
3483 if (IS_IMM) {
3484 simm = GET_FIELDs(insn, 19, 31);
3485 tcg_gen_xori_tl(cpu_dst, cpu_src1, ~simm);
3486 } else {
3487 tcg_gen_not_tl(cpu_tmp0, cpu_src2);
3488 tcg_gen_xor_tl(cpu_dst, cpu_src1, cpu_tmp0);
3489 }
8393617c 3490 if (xop & 0x10) {
38482a77
BS
3491 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3492 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3493 dc->cc_op = CC_OP_LOGIC;
8393617c 3494 }
cf495bcf 3495 break;
b89e94af 3496 case 0x8: /* addx, V9 addc */
70c48285
RH
3497 gen_op_addx_int(dc, cpu_dst, cpu_src1, cpu_src2,
3498 (xop & 0x10));
cf495bcf 3499 break;
ded3ab80 3500#ifdef TARGET_SPARC64
0f8a249a 3501 case 0x9: /* V9 mulx */
41d72852
BS
3502 if (IS_IMM) {
3503 simm = GET_FIELDs(insn, 19, 31);
3504 tcg_gen_muli_i64(cpu_dst, cpu_src1, simm);
3505 } else {
3506 tcg_gen_mul_i64(cpu_dst, cpu_src1, cpu_src2);
3507 }
ded3ab80
PB
3508 break;
3509#endif
b89e94af 3510 case 0xa: /* umul */
64a88d5d 3511 CHECK_IU_FEATURE(dc, MUL);
6ae20372 3512 gen_op_umul(cpu_dst, cpu_src1, cpu_src2);
8393617c 3513 if (xop & 0x10) {
38482a77
BS
3514 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3515 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3516 dc->cc_op = CC_OP_LOGIC;
8393617c 3517 }
cf495bcf 3518 break;
b89e94af 3519 case 0xb: /* smul */
64a88d5d 3520 CHECK_IU_FEATURE(dc, MUL);
6ae20372 3521 gen_op_smul(cpu_dst, cpu_src1, cpu_src2);
8393617c 3522 if (xop & 0x10) {
38482a77
BS
3523 tcg_gen_mov_tl(cpu_cc_dst, cpu_dst);
3524 tcg_gen_movi_i32(cpu_cc_op, CC_OP_LOGIC);
3525 dc->cc_op = CC_OP_LOGIC;
8393617c 3526 }
cf495bcf 3527 break;
b89e94af 3528 case 0xc: /* subx, V9 subc */
70c48285
RH
3529 gen_op_subx_int(dc, cpu_dst, cpu_src1, cpu_src2,
3530 (xop & 0x10));
cf495bcf 3531 break;
ded3ab80 3532#ifdef TARGET_SPARC64
0f8a249a 3533 case 0xd: /* V9 udivx */
c28ae41e 3534 gen_helper_udivx(cpu_dst, cpu_env, cpu_src1, cpu_src2);
ded3ab80
PB
3535 break;
3536#endif
b89e94af 3537 case 0xe: /* udiv */
64a88d5d 3538 CHECK_IU_FEATURE(dc, DIV);
8393617c 3539 if (xop & 0x10) {
7a5e4488
BS
3540 gen_helper_udiv_cc(cpu_dst, cpu_env, cpu_src1,
3541 cpu_src2);
6c78ea32 3542 dc->cc_op = CC_OP_DIV;
0fcec41e 3543 } else {
7a5e4488
BS
3544 gen_helper_udiv(cpu_dst, cpu_env, cpu_src1,
3545 cpu_src2);
8393617c 3546 }
cf495bcf 3547 break;
b89e94af 3548 case 0xf: /* sdiv */
64a88d5d 3549 CHECK_IU_FEATURE(dc, DIV);
8393617c 3550 if (xop & 0x10) {
7a5e4488
BS
3551 gen_helper_sdiv_cc(cpu_dst, cpu_env, cpu_src1,
3552 cpu_src2);
6c78ea32 3553 dc->cc_op = CC_OP_DIV;
0fcec41e 3554 } else {
7a5e4488
BS
3555 gen_helper_sdiv(cpu_dst, cpu_env, cpu_src1,
3556 cpu_src2);
8393617c 3557 }
cf495bcf
FB
3558 break;
3559 default:
3560 goto illegal_insn;
3561 }
6ae20372 3562 gen_movl_TN_reg(rd, cpu_dst);
cf495bcf 3563 } else {
41d72852
BS
3564 cpu_src1 = get_src1(insn, cpu_src1);
3565 cpu_src2 = get_src2(insn, cpu_src2);
cf495bcf 3566 switch (xop) {
0f8a249a 3567 case 0x20: /* taddcc */
a2ea4aa9 3568 gen_op_add_cc(cpu_dst, cpu_src1, cpu_src2);
6ae20372 3569 gen_movl_TN_reg(rd, cpu_dst);
3b2d1e92
BS
3570 tcg_gen_movi_i32(cpu_cc_op, CC_OP_TADD);
3571 dc->cc_op = CC_OP_TADD;
0f8a249a
BS
3572 break;
3573 case 0x21: /* tsubcc */
a2ea4aa9 3574 gen_op_sub_cc(cpu_dst, cpu_src1, cpu_src2);
6ae20372 3575 gen_movl_TN_reg(rd, cpu_dst);
3b2d1e92
BS
3576 tcg_gen_movi_i32(cpu_cc_op, CC_OP_TSUB);
3577 dc->cc_op = CC_OP_TSUB;
0f8a249a
BS
3578 break;
3579 case 0x22: /* taddcctv */
a2ea4aa9
RH
3580 gen_helper_taddcctv(cpu_dst, cpu_env,
3581 cpu_src1, cpu_src2);
6ae20372 3582 gen_movl_TN_reg(rd, cpu_dst);
3b2d1e92 3583 dc->cc_op = CC_OP_TADDTV;
0f8a249a
BS
3584 break;
3585 case 0x23: /* tsubcctv */
a2ea4aa9
RH
3586 gen_helper_tsubcctv(cpu_dst, cpu_env,
3587 cpu_src1, cpu_src2);
6ae20372 3588 gen_movl_TN_reg(rd, cpu_dst);
3b2d1e92 3589 dc->cc_op = CC_OP_TSUBTV;
0f8a249a 3590 break;
cf495bcf 3591 case 0x24: /* mulscc */
2ffd9176 3592 gen_helper_compute_psr(cpu_env);
6ae20372
BS
3593 gen_op_mulscc(cpu_dst, cpu_src1, cpu_src2);
3594 gen_movl_TN_reg(rd, cpu_dst);
d084469c
BS
3595 tcg_gen_movi_i32(cpu_cc_op, CC_OP_ADD);
3596 dc->cc_op = CC_OP_ADD;
cf495bcf 3597 break;
83469015 3598#ifndef TARGET_SPARC64
0f8a249a 3599 case 0x25: /* sll */
e35298cd 3600 if (IS_IMM) { /* immediate */
67526b20
BS
3601 simm = GET_FIELDs(insn, 20, 31);
3602 tcg_gen_shli_tl(cpu_dst, cpu_src1, simm & 0x1f);
e35298cd
BS
3603 } else { /* register */
3604 tcg_gen_andi_tl(cpu_tmp0, cpu_src2, 0x1f);
3605 tcg_gen_shl_tl(cpu_dst, cpu_src1, cpu_tmp0);
3606 }
6ae20372 3607 gen_movl_TN_reg(rd, cpu_dst);
cf495bcf 3608 break;
83469015 3609 case 0x26: /* srl */
e35298cd 3610 if (IS_IMM) { /* immediate */
67526b20
BS
3611 simm = GET_FIELDs(insn, 20, 31);
3612 tcg_gen_shri_tl(cpu_dst, cpu_src1, simm & 0x1f);
e35298cd
BS
3613 } else { /* register */
3614 tcg_gen_andi_tl(cpu_tmp0, cpu_src2, 0x1f);
3615 tcg_gen_shr_tl(cpu_dst, cpu_src1, cpu_tmp0);
3616 }
6ae20372 3617 gen_movl_TN_reg(rd, cpu_dst);
cf495bcf 3618 break;
83469015 3619 case 0x27: /* sra */
e35298cd 3620 if (IS_IMM) { /* immediate */
67526b20
BS
3621 simm = GET_FIELDs(insn, 20, 31);
3622 tcg_gen_sari_tl(cpu_dst, cpu_src1, simm & 0x1f);
e35298cd
BS
3623 } else { /* register */
3624 tcg_gen_andi_tl(cpu_tmp0, cpu_src2, 0x1f);
3625 tcg_gen_sar_tl(cpu_dst, cpu_src1, cpu_tmp0);
3626 }
6ae20372 3627 gen_movl_TN_reg(rd, cpu_dst);
cf495bcf 3628 break;
83469015 3629#endif
cf495bcf
FB
3630 case 0x30:
3631 {
cf495bcf 3632 switch(rd) {
3475187d 3633 case 0: /* wry */
5068cbd9
BS
3634 tcg_gen_xor_tl(cpu_tmp0, cpu_src1, cpu_src2);
3635 tcg_gen_andi_tl(cpu_y, cpu_tmp0, 0xffffffff);
cf495bcf 3636 break;
65fe7b09
BS
3637#ifndef TARGET_SPARC64
3638 case 0x01 ... 0x0f: /* undefined in the
3639 SPARCv8 manual, nop
3640 on the microSPARC
3641 II */
3642 case 0x10 ... 0x1f: /* implementation-dependent
3643 in the SPARCv8
3644 manual, nop on the
3645 microSPARC II */
3646 break;
3647#else
0f8a249a 3648 case 0x2: /* V9 wrccr */
6ae20372 3649 tcg_gen_xor_tl(cpu_dst, cpu_src1, cpu_src2);
063c3675 3650 gen_helper_wrccr(cpu_env, cpu_dst);
8393617c
BS
3651 tcg_gen_movi_i32(cpu_cc_op, CC_OP_FLAGS);
3652 dc->cc_op = CC_OP_FLAGS;
0f8a249a
BS
3653 break;
3654 case 0x3: /* V9 wrasi */
6ae20372 3655 tcg_gen_xor_tl(cpu_dst, cpu_src1, cpu_src2);
01b5d4e5 3656 tcg_gen_andi_tl(cpu_dst, cpu_dst, 0xff);
255e1fcb 3657 tcg_gen_trunc_tl_i32(cpu_asi, cpu_dst);
0f8a249a
BS
3658 break;
3659 case 0x6: /* V9 wrfprs */
6ae20372 3660 tcg_gen_xor_tl(cpu_dst, cpu_src1, cpu_src2);
255e1fcb 3661 tcg_gen_trunc_tl_i32(cpu_fprs, cpu_dst);
66442b07 3662 save_state(dc);
3299908c 3663 gen_op_next_insn();
57fec1fe 3664 tcg_gen_exit_tb(0);
3299908c 3665 dc->is_br = 1;
0f8a249a
BS
3666 break;
3667 case 0xf: /* V9 sir, nop if user */
3475187d 3668#if !defined(CONFIG_USER_ONLY)
6ad6135d 3669 if (supervisor(dc)) {
1a2fb1c0 3670 ; // XXX
6ad6135d 3671 }
3475187d 3672#endif
0f8a249a
BS
3673 break;
3674 case 0x13: /* Graphics Status */
5b12f1e8 3675 if (gen_trap_ifnofpu(dc)) {
725cb90b 3676 goto jmp_insn;
5b12f1e8 3677 }
255e1fcb 3678 tcg_gen_xor_tl(cpu_gsr, cpu_src1, cpu_src2);
0f8a249a 3679 break;
9d926598
BS
3680 case 0x14: /* Softint set */
3681 if (!supervisor(dc))
3682 goto illegal_insn;
3683 tcg_gen_xor_tl(cpu_tmp64, cpu_src1, cpu_src2);
79227036 3684 gen_helper_set_softint(cpu_env, cpu_tmp64);
9d926598
BS
3685 break;
3686 case 0x15: /* Softint clear */
3687 if (!supervisor(dc))
3688 goto illegal_insn;
3689 tcg_gen_xor_tl(cpu_tmp64, cpu_src1, cpu_src2);
79227036 3690 gen_helper_clear_softint(cpu_env, cpu_tmp64);
9d926598
BS
3691 break;
3692 case 0x16: /* Softint write */
3693 if (!supervisor(dc))
3694 goto illegal_insn;
3695 tcg_gen_xor_tl(cpu_tmp64, cpu_src1, cpu_src2);
79227036 3696 gen_helper_write_softint(cpu_env, cpu_tmp64);
9d926598 3697 break;
0f8a249a 3698 case 0x17: /* Tick compare */
83469015 3699#if !defined(CONFIG_USER_ONLY)
0f8a249a
BS
3700 if (!supervisor(dc))
3701 goto illegal_insn;
83469015 3702#endif
ccd4a219 3703 {
a7812ae4 3704 TCGv_ptr r_tickptr;
ccd4a219 3705
255e1fcb 3706 tcg_gen_xor_tl(cpu_tick_cmpr, cpu_src1,
6ae20372 3707 cpu_src2);
a7812ae4 3708 r_tickptr = tcg_temp_new_ptr();
ccd4a219 3709 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 3710 offsetof(CPUSPARCState, tick));
a7812ae4
PB
3711 gen_helper_tick_set_limit(r_tickptr,
3712 cpu_tick_cmpr);
3713 tcg_temp_free_ptr(r_tickptr);
ccd4a219 3714 }
0f8a249a
BS
3715 break;
3716 case 0x18: /* System tick */
83469015 3717#if !defined(CONFIG_USER_ONLY)
0f8a249a
BS
3718 if (!supervisor(dc))
3719 goto illegal_insn;
83469015 3720#endif
ccd4a219 3721 {
a7812ae4 3722 TCGv_ptr r_tickptr;
ccd4a219 3723
6ae20372
BS
3724 tcg_gen_xor_tl(cpu_dst, cpu_src1,
3725 cpu_src2);
a7812ae4 3726 r_tickptr = tcg_temp_new_ptr();
ccd4a219 3727 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 3728 offsetof(CPUSPARCState, stick));
a7812ae4
PB
3729 gen_helper_tick_set_count(r_tickptr,
3730 cpu_dst);
3731 tcg_temp_free_ptr(r_tickptr);
ccd4a219 3732 }
0f8a249a
BS
3733 break;
3734 case 0x19: /* System tick compare */
83469015 3735#if !defined(CONFIG_USER_ONLY)
0f8a249a
BS
3736 if (!supervisor(dc))
3737 goto illegal_insn;
3475187d 3738#endif
ccd4a219 3739 {
a7812ae4 3740 TCGv_ptr r_tickptr;
ccd4a219 3741
255e1fcb 3742 tcg_gen_xor_tl(cpu_stick_cmpr, cpu_src1,
6ae20372 3743 cpu_src2);
a7812ae4 3744 r_tickptr = tcg_temp_new_ptr();
ccd4a219 3745 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 3746 offsetof(CPUSPARCState, stick));
a7812ae4
PB
3747 gen_helper_tick_set_limit(r_tickptr,
3748 cpu_stick_cmpr);
3749 tcg_temp_free_ptr(r_tickptr);
ccd4a219 3750 }
0f8a249a 3751 break;
83469015 3752
0f8a249a 3753 case 0x10: /* Performance Control */
77f193da
BS
3754 case 0x11: /* Performance Instrumentation
3755 Counter */
0f8a249a 3756 case 0x12: /* Dispatch Control */
83469015 3757#endif
3475187d 3758 default:
cf495bcf
FB
3759 goto illegal_insn;
3760 }
3761 }
3762 break;
e8af50a3 3763#if !defined(CONFIG_USER_ONLY)
af7bf89b 3764 case 0x31: /* wrpsr, V9 saved, restored */
e8af50a3 3765 {
0f8a249a
BS
3766 if (!supervisor(dc))
3767 goto priv_insn;
3475187d 3768#ifdef TARGET_SPARC64
0f8a249a
BS
3769 switch (rd) {
3770 case 0:
063c3675 3771 gen_helper_saved(cpu_env);
0f8a249a
BS
3772 break;
3773 case 1:
063c3675 3774 gen_helper_restored(cpu_env);
0f8a249a 3775 break;
e9ebed4d
BS
3776 case 2: /* UA2005 allclean */
3777 case 3: /* UA2005 otherw */
3778 case 4: /* UA2005 normalw */
3779 case 5: /* UA2005 invalw */
3780 // XXX
0f8a249a 3781 default:
3475187d
FB
3782 goto illegal_insn;
3783 }
3784#else
6ae20372 3785 tcg_gen_xor_tl(cpu_dst, cpu_src1, cpu_src2);
063c3675 3786 gen_helper_wrpsr(cpu_env, cpu_dst);
8393617c
BS
3787 tcg_gen_movi_i32(cpu_cc_op, CC_OP_FLAGS);
3788 dc->cc_op = CC_OP_FLAGS;
66442b07 3789 save_state(dc);
9e61bde5 3790 gen_op_next_insn();
57fec1fe 3791 tcg_gen_exit_tb(0);
0f8a249a 3792 dc->is_br = 1;
3475187d 3793#endif
e8af50a3
FB
3794 }
3795 break;
af7bf89b 3796 case 0x32: /* wrwim, V9 wrpr */
e8af50a3 3797 {
0f8a249a
BS
3798 if (!supervisor(dc))
3799 goto priv_insn;
ece43b8d 3800 tcg_gen_xor_tl(cpu_tmp0, cpu_src1, cpu_src2);
3475187d 3801#ifdef TARGET_SPARC64
0f8a249a
BS
3802 switch (rd) {
3803 case 0: // tpc
375ee38b 3804 {
a7812ae4 3805 TCGv_ptr r_tsptr;
375ee38b 3806
a7812ae4 3807 r_tsptr = tcg_temp_new_ptr();
8194f35a 3808 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
ece43b8d 3809 tcg_gen_st_tl(cpu_tmp0, r_tsptr,
375ee38b 3810 offsetof(trap_state, tpc));
a7812ae4 3811 tcg_temp_free_ptr(r_tsptr);
375ee38b 3812 }
0f8a249a
BS
3813 break;
3814 case 1: // tnpc
375ee38b 3815 {
a7812ae4 3816 TCGv_ptr r_tsptr;
375ee38b 3817
a7812ae4 3818 r_tsptr = tcg_temp_new_ptr();
8194f35a 3819 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
ece43b8d 3820 tcg_gen_st_tl(cpu_tmp0, r_tsptr,
375ee38b 3821 offsetof(trap_state, tnpc));
a7812ae4 3822 tcg_temp_free_ptr(r_tsptr);
375ee38b 3823 }
0f8a249a
BS
3824 break;
3825 case 2: // tstate
375ee38b 3826 {
a7812ae4 3827 TCGv_ptr r_tsptr;
375ee38b 3828
a7812ae4 3829 r_tsptr = tcg_temp_new_ptr();
8194f35a 3830 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
ece43b8d 3831 tcg_gen_st_tl(cpu_tmp0, r_tsptr,
77f193da
BS
3832 offsetof(trap_state,
3833 tstate));
a7812ae4 3834 tcg_temp_free_ptr(r_tsptr);
375ee38b 3835 }
0f8a249a
BS
3836 break;
3837 case 3: // tt
375ee38b 3838 {
a7812ae4 3839 TCGv_ptr r_tsptr;
375ee38b 3840
a7812ae4 3841 r_tsptr = tcg_temp_new_ptr();
8194f35a 3842 gen_load_trap_state_at_tl(r_tsptr, cpu_env);
527067d8
BS
3843 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
3844 tcg_gen_st_i32(cpu_tmp32, r_tsptr,
375ee38b 3845 offsetof(trap_state, tt));
a7812ae4 3846 tcg_temp_free_ptr(r_tsptr);
375ee38b 3847 }
0f8a249a
BS
3848 break;
3849 case 4: // tick
ccd4a219 3850 {
a7812ae4 3851 TCGv_ptr r_tickptr;
ccd4a219 3852
a7812ae4 3853 r_tickptr = tcg_temp_new_ptr();
ccd4a219 3854 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 3855 offsetof(CPUSPARCState, tick));
a7812ae4
PB
3856 gen_helper_tick_set_count(r_tickptr,
3857 cpu_tmp0);
3858 tcg_temp_free_ptr(r_tickptr);
ccd4a219 3859 }
0f8a249a
BS
3860 break;
3861 case 5: // tba
255e1fcb 3862 tcg_gen_mov_tl(cpu_tbr, cpu_tmp0);
0f8a249a
BS
3863 break;
3864 case 6: // pstate
a2589e5c
BS
3865 {
3866 TCGv r_tmp = tcg_temp_local_new();
3867
3868 tcg_gen_mov_tl(r_tmp, cpu_tmp0);
66442b07 3869 save_state(dc);
063c3675 3870 gen_helper_wrpstate(cpu_env, r_tmp);
a2589e5c
BS
3871 tcg_temp_free(r_tmp);
3872 dc->npc = DYNAMIC_PC;
3873 }
0f8a249a
BS
3874 break;
3875 case 7: // tl
a2589e5c
BS
3876 {
3877 TCGv r_tmp = tcg_temp_local_new();
3878
3879 tcg_gen_mov_tl(r_tmp, cpu_tmp0);
66442b07 3880 save_state(dc);
a2589e5c
BS
3881 tcg_gen_trunc_tl_i32(cpu_tmp32, r_tmp);
3882 tcg_temp_free(r_tmp);
3883 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3884 offsetof(CPUSPARCState, tl));
3885 dc->npc = DYNAMIC_PC;
3886 }
0f8a249a
BS
3887 break;
3888 case 8: // pil
063c3675 3889 gen_helper_wrpil(cpu_env, cpu_tmp0);
0f8a249a
BS
3890 break;
3891 case 9: // cwp
063c3675 3892 gen_helper_wrcwp(cpu_env, cpu_tmp0);
0f8a249a
BS
3893 break;
3894 case 10: // cansave
ece43b8d 3895 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
77f193da
BS
3896 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3897 offsetof(CPUSPARCState,
3898 cansave));
0f8a249a
BS
3899 break;
3900 case 11: // canrestore
ece43b8d 3901 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
77f193da
BS
3902 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3903 offsetof(CPUSPARCState,
3904 canrestore));
0f8a249a
BS
3905 break;
3906 case 12: // cleanwin
ece43b8d 3907 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
77f193da
BS
3908 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3909 offsetof(CPUSPARCState,
3910 cleanwin));
0f8a249a
BS
3911 break;
3912 case 13: // otherwin
ece43b8d 3913 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
77f193da
BS
3914 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3915 offsetof(CPUSPARCState,
3916 otherwin));
0f8a249a
BS
3917 break;
3918 case 14: // wstate
ece43b8d 3919 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
77f193da
BS
3920 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3921 offsetof(CPUSPARCState,
3922 wstate));
0f8a249a 3923 break;
e9ebed4d 3924 case 16: // UA2005 gl
fb79ceb9 3925 CHECK_IU_FEATURE(dc, GL);
ece43b8d 3926 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
77f193da
BS
3927 tcg_gen_st_i32(cpu_tmp32, cpu_env,
3928 offsetof(CPUSPARCState, gl));
e9ebed4d
BS
3929 break;
3930 case 26: // UA2005 strand status
fb79ceb9 3931 CHECK_IU_FEATURE(dc, HYPV);
e9ebed4d
BS
3932 if (!hypervisor(dc))
3933 goto priv_insn;
527067d8 3934 tcg_gen_mov_tl(cpu_ssr, cpu_tmp0);
e9ebed4d 3935 break;
0f8a249a
BS
3936 default:
3937 goto illegal_insn;
3938 }
3475187d 3939#else
ece43b8d 3940 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
c93e7817
BS
3941 if (dc->def->nwindows != 32)
3942 tcg_gen_andi_tl(cpu_tmp32, cpu_tmp32,
3943 (1 << dc->def->nwindows) - 1);
255e1fcb 3944 tcg_gen_mov_i32(cpu_wim, cpu_tmp32);
3475187d 3945#endif
e8af50a3
FB
3946 }
3947 break;
e9ebed4d 3948 case 0x33: /* wrtbr, UA2005 wrhpr */
e8af50a3 3949 {
e9ebed4d 3950#ifndef TARGET_SPARC64
0f8a249a
BS
3951 if (!supervisor(dc))
3952 goto priv_insn;
255e1fcb 3953 tcg_gen_xor_tl(cpu_tbr, cpu_src1, cpu_src2);
e9ebed4d 3954#else
fb79ceb9 3955 CHECK_IU_FEATURE(dc, HYPV);
e9ebed4d
BS
3956 if (!hypervisor(dc))
3957 goto priv_insn;
ece43b8d 3958 tcg_gen_xor_tl(cpu_tmp0, cpu_src1, cpu_src2);
e9ebed4d
BS
3959 switch (rd) {
3960 case 0: // hpstate
3961 // XXX gen_op_wrhpstate();
66442b07 3962 save_state(dc);
e9ebed4d 3963 gen_op_next_insn();
57fec1fe 3964 tcg_gen_exit_tb(0);
e9ebed4d
BS
3965 dc->is_br = 1;
3966 break;
3967 case 1: // htstate
3968 // XXX gen_op_wrhtstate();
3969 break;
3970 case 3: // hintp
255e1fcb 3971 tcg_gen_mov_tl(cpu_hintp, cpu_tmp0);
e9ebed4d
BS
3972 break;
3973 case 5: // htba
255e1fcb 3974 tcg_gen_mov_tl(cpu_htba, cpu_tmp0);
e9ebed4d
BS
3975 break;
3976 case 31: // hstick_cmpr
ccd4a219 3977 {
a7812ae4 3978 TCGv_ptr r_tickptr;
ccd4a219 3979
255e1fcb 3980 tcg_gen_mov_tl(cpu_hstick_cmpr, cpu_tmp0);
a7812ae4 3981 r_tickptr = tcg_temp_new_ptr();
ccd4a219 3982 tcg_gen_ld_ptr(r_tickptr, cpu_env,
c5f9864e 3983 offsetof(CPUSPARCState, hstick));
a7812ae4
PB
3984 gen_helper_tick_set_limit(r_tickptr,
3985 cpu_hstick_cmpr);
3986 tcg_temp_free_ptr(r_tickptr);
ccd4a219 3987 }
e9ebed4d
BS
3988 break;
3989 case 6: // hver readonly
3990 default:
3991 goto illegal_insn;
3992 }
3993#endif
e8af50a3
FB
3994 }
3995 break;
3996#endif
3475187d 3997#ifdef TARGET_SPARC64
0f8a249a
BS
3998 case 0x2c: /* V9 movcc */
3999 {
4000 int cc = GET_FIELD_SP(insn, 11, 12);
4001 int cond = GET_FIELD_SP(insn, 14, 17);
f52879b4 4002 DisasCompare cmp;
00f219bf 4003
0f8a249a 4004 if (insn & (1 << 18)) {
f52879b4
RH
4005 if (cc == 0) {
4006 gen_compare(&cmp, 0, cond, dc);
4007 } else if (cc == 2) {
4008 gen_compare(&cmp, 1, cond, dc);
4009 } else {
0f8a249a 4010 goto illegal_insn;
f52879b4 4011 }
0f8a249a 4012 } else {
f52879b4 4013 gen_fcompare(&cmp, cc, cond);
0f8a249a 4014 }
00f219bf 4015
f52879b4
RH
4016 /* The get_src2 above loaded the normal 13-bit
4017 immediate field, not the 11-bit field we have
4018 in movcc. But it did handle the reg case. */
4019 if (IS_IMM) {
67526b20 4020 simm = GET_FIELD_SPs(insn, 0, 10);
f52879b4 4021 tcg_gen_movi_tl(cpu_src2, simm);
00f219bf 4022 }
f52879b4
RH
4023
4024 gen_movl_reg_TN(rd, cpu_dst);
4025 tcg_gen_movcond_tl(cmp.cond, cpu_dst,
4026 cmp.c1, cmp.c2,
4027 cpu_src2, cpu_dst);
4028 free_compare(&cmp);
4029 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a
BS
4030 break;
4031 }
4032 case 0x2d: /* V9 sdivx */
c28ae41e 4033 gen_helper_sdivx(cpu_dst, cpu_env, cpu_src1, cpu_src2);
6ae20372 4034 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a
BS
4035 break;
4036 case 0x2e: /* V9 popc */
4037 {
a49d9390 4038 cpu_src2 = get_src2(insn, cpu_src2);
a7812ae4 4039 gen_helper_popc(cpu_dst, cpu_src2);
6ae20372 4040 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a
BS
4041 }
4042 case 0x2f: /* V9 movr */
4043 {
4044 int cond = GET_FIELD_SP(insn, 10, 12);
c33f80f5 4045 DisasCompare cmp;
00f219bf 4046
c33f80f5 4047 gen_compare_reg(&cmp, cond, cpu_src1);
2ea815ca 4048
c33f80f5
RH
4049 /* The get_src2 above loaded the normal 13-bit
4050 immediate field, not the 10-bit field we have
4051 in movr. But it did handle the reg case. */
4052 if (IS_IMM) {
67526b20 4053 simm = GET_FIELD_SPs(insn, 0, 9);
c33f80f5 4054 tcg_gen_movi_tl(cpu_src2, simm);
0f8a249a 4055 }
c33f80f5
RH
4056
4057 gen_movl_reg_TN(rd, cpu_dst);
4058 tcg_gen_movcond_tl(cmp.cond, cpu_dst,
4059 cmp.c1, cmp.c2,
4060 cpu_src2, cpu_dst);
4061 free_compare(&cmp);
4062 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a
BS
4063 break;
4064 }
4065#endif
4066 default:
4067 goto illegal_insn;
4068 }
4069 }
3299908c
BS
4070 } else if (xop == 0x36) { /* UltraSparc shutdown, VIS, V8 CPop1 */
4071#ifdef TARGET_SPARC64
4072 int opf = GET_FIELD_SP(insn, 5, 13);
4073 rs1 = GET_FIELD(insn, 13, 17);
4074 rs2 = GET_FIELD(insn, 27, 31);
5b12f1e8 4075 if (gen_trap_ifnofpu(dc)) {
e9ebed4d 4076 goto jmp_insn;
5b12f1e8 4077 }
3299908c
BS
4078
4079 switch (opf) {
e9ebed4d 4080 case 0x000: /* VIS I edge8cc */
6c073553
RH
4081 CHECK_FPU_FEATURE(dc, VIS1);
4082 gen_movl_reg_TN(rs1, cpu_src1);
4083 gen_movl_reg_TN(rs2, cpu_src2);
4084 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 8, 1, 0);
4085 gen_movl_TN_reg(rd, cpu_dst);
4086 break;
e9ebed4d 4087 case 0x001: /* VIS II edge8n */
6c073553
RH
4088 CHECK_FPU_FEATURE(dc, VIS2);
4089 gen_movl_reg_TN(rs1, cpu_src1);
4090 gen_movl_reg_TN(rs2, cpu_src2);
4091 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 8, 0, 0);
4092 gen_movl_TN_reg(rd, cpu_dst);
4093 break;
e9ebed4d 4094 case 0x002: /* VIS I edge8lcc */
6c073553
RH
4095 CHECK_FPU_FEATURE(dc, VIS1);
4096 gen_movl_reg_TN(rs1, cpu_src1);
4097 gen_movl_reg_TN(rs2, cpu_src2);
4098 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 8, 1, 1);
4099 gen_movl_TN_reg(rd, cpu_dst);
4100 break;
e9ebed4d 4101 case 0x003: /* VIS II edge8ln */
6c073553
RH
4102 CHECK_FPU_FEATURE(dc, VIS2);
4103 gen_movl_reg_TN(rs1, cpu_src1);
4104 gen_movl_reg_TN(rs2, cpu_src2);
4105 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 8, 0, 1);
4106 gen_movl_TN_reg(rd, cpu_dst);
4107 break;
e9ebed4d 4108 case 0x004: /* VIS I edge16cc */
6c073553
RH
4109 CHECK_FPU_FEATURE(dc, VIS1);
4110 gen_movl_reg_TN(rs1, cpu_src1);
4111 gen_movl_reg_TN(rs2, cpu_src2);
4112 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 16, 1, 0);
4113 gen_movl_TN_reg(rd, cpu_dst);
4114 break;
e9ebed4d 4115 case 0x005: /* VIS II edge16n */
6c073553
RH
4116 CHECK_FPU_FEATURE(dc, VIS2);
4117 gen_movl_reg_TN(rs1, cpu_src1);
4118 gen_movl_reg_TN(rs2, cpu_src2);
4119 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 16, 0, 0);
4120 gen_movl_TN_reg(rd, cpu_dst);
4121 break;
e9ebed4d 4122 case 0x006: /* VIS I edge16lcc */
6c073553
RH
4123 CHECK_FPU_FEATURE(dc, VIS1);
4124 gen_movl_reg_TN(rs1, cpu_src1);
4125 gen_movl_reg_TN(rs2, cpu_src2);
4126 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 16, 1, 1);
4127 gen_movl_TN_reg(rd, cpu_dst);
4128 break;
e9ebed4d 4129 case 0x007: /* VIS II edge16ln */
6c073553
RH
4130 CHECK_FPU_FEATURE(dc, VIS2);
4131 gen_movl_reg_TN(rs1, cpu_src1);
4132 gen_movl_reg_TN(rs2, cpu_src2);
4133 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 16, 0, 1);
4134 gen_movl_TN_reg(rd, cpu_dst);
4135 break;
e9ebed4d 4136 case 0x008: /* VIS I edge32cc */
6c073553
RH
4137 CHECK_FPU_FEATURE(dc, VIS1);
4138 gen_movl_reg_TN(rs1, cpu_src1);
4139 gen_movl_reg_TN(rs2, cpu_src2);
4140 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 32, 1, 0);
4141 gen_movl_TN_reg(rd, cpu_dst);
4142 break;
e9ebed4d 4143 case 0x009: /* VIS II edge32n */
6c073553
RH
4144 CHECK_FPU_FEATURE(dc, VIS2);
4145 gen_movl_reg_TN(rs1, cpu_src1);
4146 gen_movl_reg_TN(rs2, cpu_src2);
4147 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 32, 0, 0);
4148 gen_movl_TN_reg(rd, cpu_dst);
4149 break;
e9ebed4d 4150 case 0x00a: /* VIS I edge32lcc */
6c073553
RH
4151 CHECK_FPU_FEATURE(dc, VIS1);
4152 gen_movl_reg_TN(rs1, cpu_src1);
4153 gen_movl_reg_TN(rs2, cpu_src2);
4154 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 32, 1, 1);
4155 gen_movl_TN_reg(rd, cpu_dst);
4156 break;
e9ebed4d 4157 case 0x00b: /* VIS II edge32ln */
6c073553
RH
4158 CHECK_FPU_FEATURE(dc, VIS2);
4159 gen_movl_reg_TN(rs1, cpu_src1);
4160 gen_movl_reg_TN(rs2, cpu_src2);
4161 gen_edge(dc, cpu_dst, cpu_src1, cpu_src2, 32, 0, 1);
4162 gen_movl_TN_reg(rd, cpu_dst);
4163 break;
e9ebed4d 4164 case 0x010: /* VIS I array8 */
64a88d5d 4165 CHECK_FPU_FEATURE(dc, VIS1);
9322a4bf 4166 cpu_src1 = get_src1(insn, cpu_src1);
6ae20372 4167 gen_movl_reg_TN(rs2, cpu_src2);
f027c3b1 4168 gen_helper_array8(cpu_dst, cpu_src1, cpu_src2);
6ae20372 4169 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4170 break;
4171 case 0x012: /* VIS I array16 */
64a88d5d 4172 CHECK_FPU_FEATURE(dc, VIS1);
9322a4bf 4173 cpu_src1 = get_src1(insn, cpu_src1);
6ae20372 4174 gen_movl_reg_TN(rs2, cpu_src2);
f027c3b1 4175 gen_helper_array8(cpu_dst, cpu_src1, cpu_src2);
6ae20372
BS
4176 tcg_gen_shli_i64(cpu_dst, cpu_dst, 1);
4177 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4178 break;
4179 case 0x014: /* VIS I array32 */
64a88d5d 4180 CHECK_FPU_FEATURE(dc, VIS1);
9322a4bf 4181 cpu_src1 = get_src1(insn, cpu_src1);
6ae20372 4182 gen_movl_reg_TN(rs2, cpu_src2);
f027c3b1 4183 gen_helper_array8(cpu_dst, cpu_src1, cpu_src2);
6ae20372
BS
4184 tcg_gen_shli_i64(cpu_dst, cpu_dst, 2);
4185 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d 4186 break;
3299908c 4187 case 0x018: /* VIS I alignaddr */
64a88d5d 4188 CHECK_FPU_FEATURE(dc, VIS1);
9322a4bf 4189 cpu_src1 = get_src1(insn, cpu_src1);
6ae20372 4190 gen_movl_reg_TN(rs2, cpu_src2);
add545ab 4191 gen_alignaddr(cpu_dst, cpu_src1, cpu_src2, 0);
6ae20372 4192 gen_movl_TN_reg(rd, cpu_dst);
3299908c
BS
4193 break;
4194 case 0x01a: /* VIS I alignaddrl */
add545ab
RH
4195 CHECK_FPU_FEATURE(dc, VIS1);
4196 cpu_src1 = get_src1(insn, cpu_src1);
4197 gen_movl_reg_TN(rs2, cpu_src2);
4198 gen_alignaddr(cpu_dst, cpu_src1, cpu_src2, 1);
4199 gen_movl_TN_reg(rd, cpu_dst);
4200 break;
4201 case 0x019: /* VIS II bmask */
793a137a
RH
4202 CHECK_FPU_FEATURE(dc, VIS2);
4203 cpu_src1 = get_src1(insn, cpu_src1);
4204 cpu_src2 = get_src1(insn, cpu_src2);
4205 tcg_gen_add_tl(cpu_dst, cpu_src1, cpu_src2);
4206 tcg_gen_deposit_tl(cpu_gsr, cpu_gsr, cpu_dst, 32, 32);
4207 gen_movl_TN_reg(rd, cpu_dst);
4208 break;
e9ebed4d 4209 case 0x020: /* VIS I fcmple16 */
64a88d5d 4210 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4211 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4212 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4213 gen_helper_fcmple16(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4214 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4215 break;
4216 case 0x022: /* VIS I fcmpne16 */
64a88d5d 4217 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4218 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4219 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4220 gen_helper_fcmpne16(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4221 gen_movl_TN_reg(rd, cpu_dst);
3299908c 4222 break;
e9ebed4d 4223 case 0x024: /* VIS I fcmple32 */
64a88d5d 4224 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4225 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4226 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4227 gen_helper_fcmple32(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4228 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4229 break;
4230 case 0x026: /* VIS I fcmpne32 */
64a88d5d 4231 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4232 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4233 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4234 gen_helper_fcmpne32(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4235 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4236 break;
4237 case 0x028: /* VIS I fcmpgt16 */
64a88d5d 4238 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4239 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4240 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4241 gen_helper_fcmpgt16(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4242 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4243 break;
4244 case 0x02a: /* VIS I fcmpeq16 */
64a88d5d 4245 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4246 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4247 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4248 gen_helper_fcmpeq16(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4249 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4250 break;
4251 case 0x02c: /* VIS I fcmpgt32 */
64a88d5d 4252 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4253 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4254 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4255 gen_helper_fcmpgt32(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4256 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4257 break;
4258 case 0x02e: /* VIS I fcmpeq32 */
64a88d5d 4259 CHECK_FPU_FEATURE(dc, VIS1);
03fb8cfc
RH
4260 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4261 cpu_src2_64 = gen_load_fpr_D(dc, rs2);
f027c3b1 4262 gen_helper_fcmpeq32(cpu_dst, cpu_src1_64, cpu_src2_64);
afcb7375 4263 gen_movl_TN_reg(rd, cpu_dst);
e9ebed4d
BS
4264 break;
4265 case 0x031: /* VIS I fmul8x16 */
64a88d5d 4266 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4267 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmul8x16);
e9ebed4d
BS
4268 break;
4269 case 0x033: /* VIS I fmul8x16au */
64a88d5d 4270 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4271 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmul8x16au);
e9ebed4d
BS
4272 break;
4273 case 0x035: /* VIS I fmul8x16al */
64a88d5d 4274 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4275 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmul8x16al);
e9ebed4d
BS
4276 break;
4277 case 0x036: /* VIS I fmul8sux16 */
64a88d5d 4278 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4279 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmul8sux16);
e9ebed4d
BS
4280 break;
4281 case 0x037: /* VIS I fmul8ulx16 */
64a88d5d 4282 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4283 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmul8ulx16);
e9ebed4d
BS
4284 break;
4285 case 0x038: /* VIS I fmuld8sux16 */
64a88d5d 4286 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4287 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmuld8sux16);
e9ebed4d
BS
4288 break;
4289 case 0x039: /* VIS I fmuld8ulx16 */
64a88d5d 4290 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4291 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fmuld8ulx16);
e9ebed4d
BS
4292 break;
4293 case 0x03a: /* VIS I fpack32 */
2dedf314
RH
4294 CHECK_FPU_FEATURE(dc, VIS1);
4295 gen_gsr_fop_DDD(dc, rd, rs1, rs2, gen_helper_fpack32);
4296 break;
e9ebed4d 4297 case 0x03b: /* VIS I fpack16 */
2dedf314
RH
4298 CHECK_FPU_FEATURE(dc, VIS1);
4299 cpu_src1_64 = gen_load_fpr_D(dc, rs2);
4300 cpu_dst_32 = gen_dest_fpr_F();
4301 gen_helper_fpack16(cpu_dst_32, cpu_gsr, cpu_src1_64);
4302 gen_store_fpr_F(dc, rd, cpu_dst_32);
4303 break;
e9ebed4d 4304 case 0x03d: /* VIS I fpackfix */
2dedf314
RH
4305 CHECK_FPU_FEATURE(dc, VIS1);
4306 cpu_src1_64 = gen_load_fpr_D(dc, rs2);
4307 cpu_dst_32 = gen_dest_fpr_F();
4308 gen_helper_fpackfix(cpu_dst_32, cpu_gsr, cpu_src1_64);
4309 gen_store_fpr_F(dc, rd, cpu_dst_32);
4310 break;
f888300b
RH
4311 case 0x03e: /* VIS I pdist */
4312 CHECK_FPU_FEATURE(dc, VIS1);
4313 gen_ne_fop_DDDD(dc, rd, rs1, rs2, gen_helper_pdist);
4314 break;
3299908c 4315 case 0x048: /* VIS I faligndata */
64a88d5d 4316 CHECK_FPU_FEATURE(dc, VIS1);
50c796f9 4317 gen_gsr_fop_DDD(dc, rd, rs1, rs2, gen_faligndata);
3299908c 4318 break;
e9ebed4d 4319 case 0x04b: /* VIS I fpmerge */
64a88d5d 4320 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4321 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fpmerge);
e9ebed4d
BS
4322 break;
4323 case 0x04c: /* VIS II bshuffle */
793a137a
RH
4324 CHECK_FPU_FEATURE(dc, VIS2);
4325 gen_gsr_fop_DDD(dc, rd, rs1, rs2, gen_helper_bshuffle);
4326 break;
e9ebed4d 4327 case 0x04d: /* VIS I fexpand */
64a88d5d 4328 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4329 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fexpand);
e9ebed4d
BS
4330 break;
4331 case 0x050: /* VIS I fpadd16 */
64a88d5d 4332 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4333 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fpadd16);
e9ebed4d
BS
4334 break;
4335 case 0x051: /* VIS I fpadd16s */
64a88d5d 4336 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4337 gen_ne_fop_FFF(dc, rd, rs1, rs2, gen_helper_fpadd16s);
e9ebed4d
BS
4338 break;
4339 case 0x052: /* VIS I fpadd32 */
64a88d5d 4340 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4341 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fpadd32);
e9ebed4d
BS
4342 break;
4343 case 0x053: /* VIS I fpadd32s */
64a88d5d 4344 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4345 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_add_i32);
e9ebed4d
BS
4346 break;
4347 case 0x054: /* VIS I fpsub16 */
64a88d5d 4348 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4349 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fpsub16);
e9ebed4d
BS
4350 break;
4351 case 0x055: /* VIS I fpsub16s */
64a88d5d 4352 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4353 gen_ne_fop_FFF(dc, rd, rs1, rs2, gen_helper_fpsub16s);
e9ebed4d
BS
4354 break;
4355 case 0x056: /* VIS I fpsub32 */
64a88d5d 4356 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4357 gen_ne_fop_DDD(dc, rd, rs1, rs2, gen_helper_fpsub32);
e9ebed4d
BS
4358 break;
4359 case 0x057: /* VIS I fpsub32s */
64a88d5d 4360 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4361 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_sub_i32);
e9ebed4d 4362 break;
3299908c 4363 case 0x060: /* VIS I fzero */
64a88d5d 4364 CHECK_FPU_FEATURE(dc, VIS1);
96eda024
RH
4365 cpu_dst_64 = gen_dest_fpr_D();
4366 tcg_gen_movi_i64(cpu_dst_64, 0);
4367 gen_store_fpr_D(dc, rd, cpu_dst_64);
3299908c
BS
4368 break;
4369 case 0x061: /* VIS I fzeros */
64a88d5d 4370 CHECK_FPU_FEATURE(dc, VIS1);
208ae657
RH
4371 cpu_dst_32 = gen_dest_fpr_F();
4372 tcg_gen_movi_i32(cpu_dst_32, 0);
4373 gen_store_fpr_F(dc, rd, cpu_dst_32);
3299908c 4374 break;
e9ebed4d 4375 case 0x062: /* VIS I fnor */
64a88d5d 4376 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4377 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_nor_i64);
e9ebed4d
BS
4378 break;
4379 case 0x063: /* VIS I fnors */
64a88d5d 4380 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4381 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_nor_i32);
e9ebed4d
BS
4382 break;
4383 case 0x064: /* VIS I fandnot2 */
64a88d5d 4384 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4385 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_andc_i64);
e9ebed4d
BS
4386 break;
4387 case 0x065: /* VIS I fandnot2s */
64a88d5d 4388 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4389 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_andc_i32);
e9ebed4d
BS
4390 break;
4391 case 0x066: /* VIS I fnot2 */
64a88d5d 4392 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4393 gen_ne_fop_DD(dc, rd, rs2, tcg_gen_not_i64);
e9ebed4d
BS
4394 break;
4395 case 0x067: /* VIS I fnot2s */
64a88d5d 4396 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4397 gen_ne_fop_FF(dc, rd, rs2, tcg_gen_not_i32);
e9ebed4d
BS
4398 break;
4399 case 0x068: /* VIS I fandnot1 */
64a88d5d 4400 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4401 gen_ne_fop_DDD(dc, rd, rs2, rs1, tcg_gen_andc_i64);
e9ebed4d
BS
4402 break;
4403 case 0x069: /* VIS I fandnot1s */
64a88d5d 4404 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4405 gen_ne_fop_FFF(dc, rd, rs2, rs1, tcg_gen_andc_i32);
e9ebed4d
BS
4406 break;
4407 case 0x06a: /* VIS I fnot1 */
64a88d5d 4408 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4409 gen_ne_fop_DD(dc, rd, rs1, tcg_gen_not_i64);
e9ebed4d
BS
4410 break;
4411 case 0x06b: /* VIS I fnot1s */
64a88d5d 4412 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4413 gen_ne_fop_FF(dc, rd, rs1, tcg_gen_not_i32);
e9ebed4d
BS
4414 break;
4415 case 0x06c: /* VIS I fxor */
64a88d5d 4416 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4417 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_xor_i64);
e9ebed4d
BS
4418 break;
4419 case 0x06d: /* VIS I fxors */
64a88d5d 4420 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4421 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_xor_i32);
e9ebed4d
BS
4422 break;
4423 case 0x06e: /* VIS I fnand */
64a88d5d 4424 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4425 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_nand_i64);
e9ebed4d
BS
4426 break;
4427 case 0x06f: /* VIS I fnands */
64a88d5d 4428 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4429 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_nand_i32);
e9ebed4d
BS
4430 break;
4431 case 0x070: /* VIS I fand */
64a88d5d 4432 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4433 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_and_i64);
e9ebed4d
BS
4434 break;
4435 case 0x071: /* VIS I fands */
64a88d5d 4436 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4437 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_and_i32);
e9ebed4d
BS
4438 break;
4439 case 0x072: /* VIS I fxnor */
64a88d5d 4440 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4441 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_eqv_i64);
e9ebed4d
BS
4442 break;
4443 case 0x073: /* VIS I fxnors */
64a88d5d 4444 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4445 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_eqv_i32);
e9ebed4d 4446 break;
3299908c 4447 case 0x074: /* VIS I fsrc1 */
64a88d5d 4448 CHECK_FPU_FEATURE(dc, VIS1);
96eda024
RH
4449 cpu_src1_64 = gen_load_fpr_D(dc, rs1);
4450 gen_store_fpr_D(dc, rd, cpu_src1_64);
3299908c
BS
4451 break;
4452 case 0x075: /* VIS I fsrc1s */
64a88d5d 4453 CHECK_FPU_FEATURE(dc, VIS1);
208ae657
RH
4454 cpu_src1_32 = gen_load_fpr_F(dc, rs1);
4455 gen_store_fpr_F(dc, rd, cpu_src1_32);
3299908c 4456 break;
e9ebed4d 4457 case 0x076: /* VIS I fornot2 */
64a88d5d 4458 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4459 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_orc_i64);
e9ebed4d
BS
4460 break;
4461 case 0x077: /* VIS I fornot2s */
64a88d5d 4462 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4463 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_orc_i32);
e9ebed4d 4464 break;
3299908c 4465 case 0x078: /* VIS I fsrc2 */
64a88d5d 4466 CHECK_FPU_FEATURE(dc, VIS1);
96eda024
RH
4467 cpu_src1_64 = gen_load_fpr_D(dc, rs2);
4468 gen_store_fpr_D(dc, rd, cpu_src1_64);
3299908c
BS
4469 break;
4470 case 0x079: /* VIS I fsrc2s */
64a88d5d 4471 CHECK_FPU_FEATURE(dc, VIS1);
208ae657
RH
4472 cpu_src1_32 = gen_load_fpr_F(dc, rs2);
4473 gen_store_fpr_F(dc, rd, cpu_src1_32);
3299908c 4474 break;
e9ebed4d 4475 case 0x07a: /* VIS I fornot1 */
64a88d5d 4476 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4477 gen_ne_fop_DDD(dc, rd, rs2, rs1, tcg_gen_orc_i64);
e9ebed4d
BS
4478 break;
4479 case 0x07b: /* VIS I fornot1s */
64a88d5d 4480 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4481 gen_ne_fop_FFF(dc, rd, rs2, rs1, tcg_gen_orc_i32);
e9ebed4d
BS
4482 break;
4483 case 0x07c: /* VIS I for */
64a88d5d 4484 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4485 gen_ne_fop_DDD(dc, rd, rs1, rs2, tcg_gen_or_i64);
e9ebed4d
BS
4486 break;
4487 case 0x07d: /* VIS I fors */
64a88d5d 4488 CHECK_FPU_FEATURE(dc, VIS1);
61f17f6e 4489 gen_ne_fop_FFF(dc, rd, rs1, rs2, tcg_gen_or_i32);
e9ebed4d 4490 break;
3299908c 4491 case 0x07e: /* VIS I fone */
64a88d5d 4492 CHECK_FPU_FEATURE(dc, VIS1);
96eda024
RH
4493 cpu_dst_64 = gen_dest_fpr_D();
4494 tcg_gen_movi_i64(cpu_dst_64, -1);
4495 gen_store_fpr_D(dc, rd, cpu_dst_64);
3299908c
BS
4496 break;
4497 case 0x07f: /* VIS I fones */
64a88d5d 4498 CHECK_FPU_FEATURE(dc, VIS1);
208ae657
RH
4499 cpu_dst_32 = gen_dest_fpr_F();
4500 tcg_gen_movi_i32(cpu_dst_32, -1);
4501 gen_store_fpr_F(dc, rd, cpu_dst_32);
3299908c 4502 break;
e9ebed4d
BS
4503 case 0x080: /* VIS I shutdown */
4504 case 0x081: /* VIS II siam */
4505 // XXX
4506 goto illegal_insn;
3299908c
BS
4507 default:
4508 goto illegal_insn;
4509 }
4510#else
0f8a249a 4511 goto ncp_insn;
3299908c
BS
4512#endif
4513 } else if (xop == 0x37) { /* V8 CPop2, V9 impdep2 */
fcc72045 4514#ifdef TARGET_SPARC64
0f8a249a 4515 goto illegal_insn;
fcc72045 4516#else
0f8a249a 4517 goto ncp_insn;
fcc72045 4518#endif
3475187d 4519#ifdef TARGET_SPARC64
0f8a249a 4520 } else if (xop == 0x39) { /* V9 return */
a7812ae4 4521 TCGv_i32 r_const;
2ea815ca 4522
66442b07 4523 save_state(dc);
9322a4bf 4524 cpu_src1 = get_src1(insn, cpu_src1);
0f8a249a 4525 if (IS_IMM) { /* immediate */
67526b20
BS
4526 simm = GET_FIELDs(insn, 19, 31);
4527 tcg_gen_addi_tl(cpu_dst, cpu_src1, simm);
0f8a249a 4528 } else { /* register */
3475187d 4529 rs2 = GET_FIELD(insn, 27, 31);
0f8a249a 4530 if (rs2) {
6ae20372
BS
4531 gen_movl_reg_TN(rs2, cpu_src2);
4532 tcg_gen_add_tl(cpu_dst, cpu_src1, cpu_src2);
6f551262
BS
4533 } else
4534 tcg_gen_mov_tl(cpu_dst, cpu_src1);
3475187d 4535 }
063c3675 4536 gen_helper_restore(cpu_env);
13a6dd00 4537 gen_mov_pc_npc(dc);
2ea815ca 4538 r_const = tcg_const_i32(3);
fe8d8f0f 4539 gen_helper_check_align(cpu_env, cpu_dst, r_const);
a7812ae4 4540 tcg_temp_free_i32(r_const);
6ae20372 4541 tcg_gen_mov_tl(cpu_npc, cpu_dst);
0f8a249a
BS
4542 dc->npc = DYNAMIC_PC;
4543 goto jmp_insn;
3475187d 4544#endif
0f8a249a 4545 } else {
9322a4bf 4546 cpu_src1 = get_src1(insn, cpu_src1);
0f8a249a 4547 if (IS_IMM) { /* immediate */
67526b20
BS
4548 simm = GET_FIELDs(insn, 19, 31);
4549 tcg_gen_addi_tl(cpu_dst, cpu_src1, simm);
0f8a249a 4550 } else { /* register */
e80cfcfc 4551 rs2 = GET_FIELD(insn, 27, 31);
0f8a249a 4552 if (rs2) {
6ae20372
BS
4553 gen_movl_reg_TN(rs2, cpu_src2);
4554 tcg_gen_add_tl(cpu_dst, cpu_src1, cpu_src2);
6f551262
BS
4555 } else
4556 tcg_gen_mov_tl(cpu_dst, cpu_src1);
cf495bcf 4557 }
0f8a249a
BS
4558 switch (xop) {
4559 case 0x38: /* jmpl */
4560 {
a7812ae4
PB
4561 TCGv r_pc;
4562 TCGv_i32 r_const;
2ea815ca 4563
a7812ae4
PB
4564 r_pc = tcg_const_tl(dc->pc);
4565 gen_movl_TN_reg(rd, r_pc);
4566 tcg_temp_free(r_pc);
13a6dd00 4567 gen_mov_pc_npc(dc);
2ea815ca 4568 r_const = tcg_const_i32(3);
fe8d8f0f 4569 gen_helper_check_align(cpu_env, cpu_dst, r_const);
a7812ae4 4570 tcg_temp_free_i32(r_const);
22036a49 4571 gen_address_mask(dc, cpu_dst);
6ae20372 4572 tcg_gen_mov_tl(cpu_npc, cpu_dst);
0f8a249a
BS
4573 dc->npc = DYNAMIC_PC;
4574 }
4575 goto jmp_insn;
3475187d 4576#if !defined(CONFIG_USER_ONLY) && !defined(TARGET_SPARC64)
0f8a249a
BS
4577 case 0x39: /* rett, V9 return */
4578 {
a7812ae4 4579 TCGv_i32 r_const;
2ea815ca 4580
0f8a249a
BS
4581 if (!supervisor(dc))
4582 goto priv_insn;
13a6dd00 4583 gen_mov_pc_npc(dc);
2ea815ca 4584 r_const = tcg_const_i32(3);
fe8d8f0f 4585 gen_helper_check_align(cpu_env, cpu_dst, r_const);
a7812ae4 4586 tcg_temp_free_i32(r_const);
6ae20372 4587 tcg_gen_mov_tl(cpu_npc, cpu_dst);
0f8a249a 4588 dc->npc = DYNAMIC_PC;
063c3675 4589 gen_helper_rett(cpu_env);
0f8a249a
BS
4590 }
4591 goto jmp_insn;
4592#endif
4593 case 0x3b: /* flush */
5578ceab 4594 if (!((dc)->def->features & CPU_FEATURE_FLUSH))
64a88d5d 4595 goto unimp_flush;
dcfd14b3 4596 /* nop */
0f8a249a
BS
4597 break;
4598 case 0x3c: /* save */
66442b07 4599 save_state(dc);
063c3675 4600 gen_helper_save(cpu_env);
6ae20372 4601 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a
BS
4602 break;
4603 case 0x3d: /* restore */
66442b07 4604 save_state(dc);
063c3675 4605 gen_helper_restore(cpu_env);
6ae20372 4606 gen_movl_TN_reg(rd, cpu_dst);
0f8a249a 4607 break;
3475187d 4608#if !defined(CONFIG_USER_ONLY) && defined(TARGET_SPARC64)
0f8a249a
BS
4609 case 0x3e: /* V9 done/retry */
4610 {
4611 switch (rd) {
4612 case 0:
4613 if (!supervisor(dc))
4614 goto priv_insn;
4615 dc->npc = DYNAMIC_PC;
4616 dc->pc = DYNAMIC_PC;
063c3675 4617 gen_helper_done(cpu_env);
0f8a249a
BS
4618 goto jmp_insn;
4619 case 1:
4620 if (!supervisor(dc))
4621 goto priv_insn;
4622 dc->npc = DYNAMIC_PC;
4623 dc->pc = DYNAMIC_PC;
063c3675 4624 gen_helper_retry(cpu_env);
0f8a249a
BS
4625 goto jmp_insn;
4626 default:
4627 goto illegal_insn;
4628 }
4629 }
4630 break;
4631#endif
4632 default:
4633 goto illegal_insn;
4634 }
cf495bcf 4635 }
0f8a249a
BS
4636 break;
4637 }
4638 break;
4639 case 3: /* load/store instructions */
4640 {
4641 unsigned int xop = GET_FIELD(insn, 7, 12);
9322a4bf 4642
cfa90513
BS
4643 /* flush pending conditional evaluations before exposing
4644 cpu state */
4645 if (dc->cc_op != CC_OP_FLAGS) {
4646 dc->cc_op = CC_OP_FLAGS;
2ffd9176 4647 gen_helper_compute_psr(cpu_env);
cfa90513 4648 }
9322a4bf 4649 cpu_src1 = get_src1(insn, cpu_src1);
71817e48 4650 if (xop == 0x3c || xop == 0x3e) { // V9 casa/casxa
81ad8ba2 4651 rs2 = GET_FIELD(insn, 27, 31);
6ae20372 4652 gen_movl_reg_TN(rs2, cpu_src2);
71817e48
BS
4653 tcg_gen_mov_tl(cpu_addr, cpu_src1);
4654 } else if (IS_IMM) { /* immediate */
67526b20
BS
4655 simm = GET_FIELDs(insn, 19, 31);
4656 tcg_gen_addi_tl(cpu_addr, cpu_src1, simm);
0f8a249a
BS
4657 } else { /* register */
4658 rs2 = GET_FIELD(insn, 27, 31);
0f8a249a 4659 if (rs2 != 0) {
6ae20372
BS
4660 gen_movl_reg_TN(rs2, cpu_src2);
4661 tcg_gen_add_tl(cpu_addr, cpu_src1, cpu_src2);
6f551262
BS
4662 } else
4663 tcg_gen_mov_tl(cpu_addr, cpu_src1);
0f8a249a 4664 }
2f2ecb83
BS
4665 if (xop < 4 || (xop > 7 && xop < 0x14 && xop != 0x0e) ||
4666 (xop > 0x17 && xop <= 0x1d ) ||
4667 (xop > 0x2c && xop <= 0x33) || xop == 0x1f || xop == 0x3d) {
0f8a249a 4668 switch (xop) {
b89e94af 4669 case 0x0: /* ld, V9 lduw, load unsigned word */
2cade6a3 4670 gen_address_mask(dc, cpu_addr);
6ae20372 4671 tcg_gen_qemu_ld32u(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4672 break;
b89e94af 4673 case 0x1: /* ldub, load unsigned byte */
2cade6a3 4674 gen_address_mask(dc, cpu_addr);
6ae20372 4675 tcg_gen_qemu_ld8u(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4676 break;
b89e94af 4677 case 0x2: /* lduh, load unsigned halfword */
2cade6a3 4678 gen_address_mask(dc, cpu_addr);
6ae20372 4679 tcg_gen_qemu_ld16u(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4680 break;
b89e94af 4681 case 0x3: /* ldd, load double word */
0f8a249a 4682 if (rd & 1)
d4218d99 4683 goto illegal_insn;
1a2fb1c0 4684 else {
a7812ae4 4685 TCGv_i32 r_const;
2ea815ca 4686
66442b07 4687 save_state(dc);
2ea815ca 4688 r_const = tcg_const_i32(7);
fe8d8f0f
BS
4689 /* XXX remove alignment check */
4690 gen_helper_check_align(cpu_env, cpu_addr, r_const);
a7812ae4 4691 tcg_temp_free_i32(r_const);
2cade6a3 4692 gen_address_mask(dc, cpu_addr);
6ae20372 4693 tcg_gen_qemu_ld64(cpu_tmp64, cpu_addr, dc->mem_idx);
32b6c812
BS
4694 tcg_gen_trunc_i64_tl(cpu_tmp0, cpu_tmp64);
4695 tcg_gen_andi_tl(cpu_tmp0, cpu_tmp0, 0xffffffffULL);
4696 gen_movl_TN_reg(rd + 1, cpu_tmp0);
8911f501 4697 tcg_gen_shri_i64(cpu_tmp64, cpu_tmp64, 32);
6ae20372
BS
4698 tcg_gen_trunc_i64_tl(cpu_val, cpu_tmp64);
4699 tcg_gen_andi_tl(cpu_val, cpu_val, 0xffffffffULL);
1a2fb1c0 4700 }
0f8a249a 4701 break;
b89e94af 4702 case 0x9: /* ldsb, load signed byte */
2cade6a3 4703 gen_address_mask(dc, cpu_addr);
6ae20372 4704 tcg_gen_qemu_ld8s(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4705 break;
b89e94af 4706 case 0xa: /* ldsh, load signed halfword */
2cade6a3 4707 gen_address_mask(dc, cpu_addr);
6ae20372 4708 tcg_gen_qemu_ld16s(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a
BS
4709 break;
4710 case 0xd: /* ldstub -- XXX: should be atomically */
2ea815ca
BS
4711 {
4712 TCGv r_const;
4713
2cade6a3 4714 gen_address_mask(dc, cpu_addr);
2ea815ca
BS
4715 tcg_gen_qemu_ld8s(cpu_val, cpu_addr, dc->mem_idx);
4716 r_const = tcg_const_tl(0xff);
4717 tcg_gen_qemu_st8(r_const, cpu_addr, dc->mem_idx);
4718 tcg_temp_free(r_const);
4719 }
0f8a249a 4720 break;
b89e94af 4721 case 0x0f: /* swap, swap register with memory. Also
77f193da 4722 atomically */
64a88d5d 4723 CHECK_IU_FEATURE(dc, SWAP);
6ae20372 4724 gen_movl_reg_TN(rd, cpu_val);
2cade6a3 4725 gen_address_mask(dc, cpu_addr);
527067d8 4726 tcg_gen_qemu_ld32u(cpu_tmp0, cpu_addr, dc->mem_idx);
6ae20372 4727 tcg_gen_qemu_st32(cpu_val, cpu_addr, dc->mem_idx);
527067d8 4728 tcg_gen_mov_tl(cpu_val, cpu_tmp0);
0f8a249a 4729 break;
3475187d 4730#if !defined(CONFIG_USER_ONLY) || defined(TARGET_SPARC64)
b89e94af 4731 case 0x10: /* lda, V9 lduwa, load word alternate */
3475187d 4732#ifndef TARGET_SPARC64
0f8a249a
BS
4733 if (IS_IMM)
4734 goto illegal_insn;
4735 if (!supervisor(dc))
4736 goto priv_insn;
6ea4a6c8 4737#endif
66442b07 4738 save_state(dc);
6ae20372 4739 gen_ld_asi(cpu_val, cpu_addr, insn, 4, 0);
0f8a249a 4740 break;
b89e94af 4741 case 0x11: /* lduba, load unsigned byte alternate */
3475187d 4742#ifndef TARGET_SPARC64
0f8a249a
BS
4743 if (IS_IMM)
4744 goto illegal_insn;
4745 if (!supervisor(dc))
4746 goto priv_insn;
4747#endif
66442b07 4748 save_state(dc);
6ae20372 4749 gen_ld_asi(cpu_val, cpu_addr, insn, 1, 0);
0f8a249a 4750 break;
b89e94af 4751 case 0x12: /* lduha, load unsigned halfword alternate */
3475187d 4752#ifndef TARGET_SPARC64
0f8a249a
BS
4753 if (IS_IMM)
4754 goto illegal_insn;
4755 if (!supervisor(dc))
4756 goto priv_insn;
3475187d 4757#endif
66442b07 4758 save_state(dc);
6ae20372 4759 gen_ld_asi(cpu_val, cpu_addr, insn, 2, 0);
0f8a249a 4760 break;
b89e94af 4761 case 0x13: /* ldda, load double word alternate */
3475187d 4762#ifndef TARGET_SPARC64
0f8a249a
BS
4763 if (IS_IMM)
4764 goto illegal_insn;
4765 if (!supervisor(dc))
4766 goto priv_insn;
3475187d 4767#endif
0f8a249a 4768 if (rd & 1)
d4218d99 4769 goto illegal_insn;
66442b07 4770 save_state(dc);
db166940
BS
4771 gen_ldda_asi(cpu_val, cpu_addr, insn, rd);
4772 goto skip_move;
b89e94af 4773 case 0x19: /* ldsba, load signed byte alternate */
3475187d 4774#ifndef TARGET_SPARC64
0f8a249a
BS
4775 if (IS_IMM)
4776 goto illegal_insn;
4777 if (!supervisor(dc))
4778 goto priv_insn;
4779#endif
66442b07 4780 save_state(dc);
6ae20372 4781 gen_ld_asi(cpu_val, cpu_addr, insn, 1, 1);
0f8a249a 4782 break;
b89e94af 4783 case 0x1a: /* ldsha, load signed halfword alternate */
3475187d 4784#ifndef TARGET_SPARC64
0f8a249a
BS
4785 if (IS_IMM)
4786 goto illegal_insn;
4787 if (!supervisor(dc))
4788 goto priv_insn;
3475187d 4789#endif
66442b07 4790 save_state(dc);
6ae20372 4791 gen_ld_asi(cpu_val, cpu_addr, insn, 2, 1);
0f8a249a
BS
4792 break;
4793 case 0x1d: /* ldstuba -- XXX: should be atomically */
3475187d 4794#ifndef TARGET_SPARC64
0f8a249a
BS
4795 if (IS_IMM)
4796 goto illegal_insn;
4797 if (!supervisor(dc))
4798 goto priv_insn;
4799#endif
66442b07 4800 save_state(dc);
6ae20372 4801 gen_ldstub_asi(cpu_val, cpu_addr, insn);
0f8a249a 4802 break;
b89e94af 4803 case 0x1f: /* swapa, swap reg with alt. memory. Also
77f193da 4804 atomically */
64a88d5d 4805 CHECK_IU_FEATURE(dc, SWAP);
3475187d 4806#ifndef TARGET_SPARC64
0f8a249a
BS
4807 if (IS_IMM)
4808 goto illegal_insn;
4809 if (!supervisor(dc))
4810 goto priv_insn;
6ea4a6c8 4811#endif
66442b07 4812 save_state(dc);
6ae20372
BS
4813 gen_movl_reg_TN(rd, cpu_val);
4814 gen_swap_asi(cpu_val, cpu_addr, insn);
0f8a249a 4815 break;
3475187d
FB
4816
4817#ifndef TARGET_SPARC64
0f8a249a
BS
4818 case 0x30: /* ldc */
4819 case 0x31: /* ldcsr */
4820 case 0x33: /* lddc */
4821 goto ncp_insn;
3475187d
FB
4822#endif
4823#endif
4824#ifdef TARGET_SPARC64
0f8a249a 4825 case 0x08: /* V9 ldsw */
2cade6a3 4826 gen_address_mask(dc, cpu_addr);
6ae20372 4827 tcg_gen_qemu_ld32s(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a
BS
4828 break;
4829 case 0x0b: /* V9 ldx */
2cade6a3 4830 gen_address_mask(dc, cpu_addr);
6ae20372 4831 tcg_gen_qemu_ld64(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a
BS
4832 break;
4833 case 0x18: /* V9 ldswa */
66442b07 4834 save_state(dc);
6ae20372 4835 gen_ld_asi(cpu_val, cpu_addr, insn, 4, 1);
0f8a249a
BS
4836 break;
4837 case 0x1b: /* V9 ldxa */
66442b07 4838 save_state(dc);
6ae20372 4839 gen_ld_asi(cpu_val, cpu_addr, insn, 8, 0);
0f8a249a
BS
4840 break;
4841 case 0x2d: /* V9 prefetch, no effect */
4842 goto skip_move;
4843 case 0x30: /* V9 ldfa */
5b12f1e8 4844 if (gen_trap_ifnofpu(dc)) {
8872eb4f
TS
4845 goto jmp_insn;
4846 }
66442b07 4847 save_state(dc);
6ae20372 4848 gen_ldf_asi(cpu_addr, insn, 4, rd);
638737ad 4849 gen_update_fprs_dirty(rd);
81ad8ba2 4850 goto skip_move;
0f8a249a 4851 case 0x33: /* V9 lddfa */
5b12f1e8 4852 if (gen_trap_ifnofpu(dc)) {
8872eb4f
TS
4853 goto jmp_insn;
4854 }
66442b07 4855 save_state(dc);
6ae20372 4856 gen_ldf_asi(cpu_addr, insn, 8, DFPREG(rd));
638737ad 4857 gen_update_fprs_dirty(DFPREG(rd));
81ad8ba2 4858 goto skip_move;
0f8a249a
BS
4859 case 0x3d: /* V9 prefetcha, no effect */
4860 goto skip_move;
4861 case 0x32: /* V9 ldqfa */
64a88d5d 4862 CHECK_FPU_FEATURE(dc, FLOAT128);
5b12f1e8 4863 if (gen_trap_ifnofpu(dc)) {
8872eb4f
TS
4864 goto jmp_insn;
4865 }
66442b07 4866 save_state(dc);
6ae20372 4867 gen_ldf_asi(cpu_addr, insn, 16, QFPREG(rd));
638737ad 4868 gen_update_fprs_dirty(QFPREG(rd));
1f587329 4869 goto skip_move;
0f8a249a
BS
4870#endif
4871 default:
4872 goto illegal_insn;
4873 }
6ae20372 4874 gen_movl_TN_reg(rd, cpu_val);
db166940 4875#if !defined(CONFIG_USER_ONLY) || defined(TARGET_SPARC64)
0f8a249a 4876 skip_move: ;
3475187d 4877#endif
0f8a249a 4878 } else if (xop >= 0x20 && xop < 0x24) {
5b12f1e8 4879 if (gen_trap_ifnofpu(dc)) {
a80dde08 4880 goto jmp_insn;
5b12f1e8 4881 }
66442b07 4882 save_state(dc);
0f8a249a 4883 switch (xop) {
b89e94af 4884 case 0x20: /* ldf, load fpreg */
2cade6a3 4885 gen_address_mask(dc, cpu_addr);
527067d8 4886 tcg_gen_qemu_ld32u(cpu_tmp0, cpu_addr, dc->mem_idx);
208ae657
RH
4887 cpu_dst_32 = gen_dest_fpr_F();
4888 tcg_gen_trunc_tl_i32(cpu_dst_32, cpu_tmp0);
4889 gen_store_fpr_F(dc, rd, cpu_dst_32);
0f8a249a 4890 break;
3a3b925d
BS
4891 case 0x21: /* ldfsr, V9 ldxfsr */
4892#ifdef TARGET_SPARC64
2cade6a3 4893 gen_address_mask(dc, cpu_addr);
3a3b925d
BS
4894 if (rd == 1) {
4895 tcg_gen_qemu_ld64(cpu_tmp64, cpu_addr, dc->mem_idx);
2e2f4ade 4896 gen_helper_ldxfsr(cpu_env, cpu_tmp64);
fe987e23
IK
4897 } else {
4898 tcg_gen_qemu_ld32u(cpu_tmp0, cpu_addr, dc->mem_idx);
4899 tcg_gen_trunc_tl_i32(cpu_tmp32, cpu_tmp0);
2e2f4ade 4900 gen_helper_ldfsr(cpu_env, cpu_tmp32);
fe987e23 4901 }
3a3b925d
BS
4902#else
4903 {
4904 tcg_gen_qemu_ld32u(cpu_tmp32, cpu_addr, dc->mem_idx);
2e2f4ade 4905 gen_helper_ldfsr(cpu_env, cpu_tmp32);
3a3b925d
BS
4906 }
4907#endif
0f8a249a 4908 break;
b89e94af 4909 case 0x22: /* ldqf, load quad fpreg */
2ea815ca 4910 {
a7812ae4 4911 TCGv_i32 r_const;
2ea815ca
BS
4912
4913 CHECK_FPU_FEATURE(dc, FLOAT128);
4914 r_const = tcg_const_i32(dc->mem_idx);
1295001c 4915 gen_address_mask(dc, cpu_addr);
fe8d8f0f 4916 gen_helper_ldqf(cpu_env, cpu_addr, r_const);
a7812ae4 4917 tcg_temp_free_i32(r_const);
2ea815ca 4918 gen_op_store_QT0_fpr(QFPREG(rd));
638737ad 4919 gen_update_fprs_dirty(QFPREG(rd));
2ea815ca 4920 }
1f587329 4921 break;
b89e94af 4922 case 0x23: /* lddf, load double fpreg */
03fb8cfc
RH
4923 gen_address_mask(dc, cpu_addr);
4924 cpu_dst_64 = gen_dest_fpr_D();
4925 tcg_gen_qemu_ld64(cpu_dst_64, cpu_addr, dc->mem_idx);
4926 gen_store_fpr_D(dc, rd, cpu_dst_64);
0f8a249a
BS
4927 break;
4928 default:
4929 goto illegal_insn;
4930 }
dc1a6971 4931 } else if (xop < 8 || (xop >= 0x14 && xop < 0x18) ||
0f8a249a 4932 xop == 0xe || xop == 0x1e) {
6ae20372 4933 gen_movl_reg_TN(rd, cpu_val);
0f8a249a 4934 switch (xop) {
b89e94af 4935 case 0x4: /* st, store word */
2cade6a3 4936 gen_address_mask(dc, cpu_addr);
6ae20372 4937 tcg_gen_qemu_st32(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4938 break;
b89e94af 4939 case 0x5: /* stb, store byte */
2cade6a3 4940 gen_address_mask(dc, cpu_addr);
6ae20372 4941 tcg_gen_qemu_st8(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4942 break;
b89e94af 4943 case 0x6: /* sth, store halfword */
2cade6a3 4944 gen_address_mask(dc, cpu_addr);
6ae20372 4945 tcg_gen_qemu_st16(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a 4946 break;
b89e94af 4947 case 0x7: /* std, store double word */
0f8a249a 4948 if (rd & 1)
d4218d99 4949 goto illegal_insn;
1a2fb1c0 4950 else {
a7812ae4 4951 TCGv_i32 r_const;
1a2fb1c0 4952
66442b07 4953 save_state(dc);
2cade6a3 4954 gen_address_mask(dc, cpu_addr);
2ea815ca 4955 r_const = tcg_const_i32(7);
fe8d8f0f
BS
4956 /* XXX remove alignment check */
4957 gen_helper_check_align(cpu_env, cpu_addr, r_const);
a7812ae4 4958 tcg_temp_free_i32(r_const);
a7ec4229 4959 gen_movl_reg_TN(rd + 1, cpu_tmp0);
ab508019 4960 tcg_gen_concat_tl_i64(cpu_tmp64, cpu_tmp0, cpu_val);
6ae20372 4961 tcg_gen_qemu_st64(cpu_tmp64, cpu_addr, dc->mem_idx);
7fa76c0b 4962 }
0f8a249a 4963 break;
3475187d 4964#if !defined(CONFIG_USER_ONLY) || defined(TARGET_SPARC64)
b89e94af 4965 case 0x14: /* sta, V9 stwa, store word alternate */
3475187d 4966#ifndef TARGET_SPARC64
0f8a249a
BS
4967 if (IS_IMM)
4968 goto illegal_insn;
4969 if (!supervisor(dc))
4970 goto priv_insn;
6ea4a6c8 4971#endif
66442b07 4972 save_state(dc);
6ae20372 4973 gen_st_asi(cpu_val, cpu_addr, insn, 4);
9fd1ae3a 4974 dc->npc = DYNAMIC_PC;
d39c0b99 4975 break;
b89e94af 4976 case 0x15: /* stba, store byte alternate */
3475187d 4977#ifndef TARGET_SPARC64
0f8a249a
BS
4978 if (IS_IMM)
4979 goto illegal_insn;
4980 if (!supervisor(dc))
4981 goto priv_insn;
3475187d 4982#endif
66442b07 4983 save_state(dc);
6ae20372 4984 gen_st_asi(cpu_val, cpu_addr, insn, 1);
9fd1ae3a 4985 dc->npc = DYNAMIC_PC;
d39c0b99 4986 break;
b89e94af 4987 case 0x16: /* stha, store halfword alternate */
3475187d 4988#ifndef TARGET_SPARC64
0f8a249a
BS
4989 if (IS_IMM)
4990 goto illegal_insn;
4991 if (!supervisor(dc))
4992 goto priv_insn;
6ea4a6c8 4993#endif
66442b07 4994 save_state(dc);
6ae20372 4995 gen_st_asi(cpu_val, cpu_addr, insn, 2);
9fd1ae3a 4996 dc->npc = DYNAMIC_PC;
d39c0b99 4997 break;
b89e94af 4998 case 0x17: /* stda, store double word alternate */
3475187d 4999#ifndef TARGET_SPARC64
0f8a249a
BS
5000 if (IS_IMM)
5001 goto illegal_insn;
5002 if (!supervisor(dc))
5003 goto priv_insn;
3475187d 5004#endif
0f8a249a 5005 if (rd & 1)
d4218d99 5006 goto illegal_insn;
1a2fb1c0 5007 else {
66442b07 5008 save_state(dc);
6ae20372 5009 gen_stda_asi(cpu_val, cpu_addr, insn, rd);
1a2fb1c0 5010 }
d39c0b99 5011 break;
e80cfcfc 5012#endif
3475187d 5013#ifdef TARGET_SPARC64
0f8a249a 5014 case 0x0e: /* V9 stx */
2cade6a3 5015 gen_address_mask(dc, cpu_addr);
6ae20372 5016 tcg_gen_qemu_st64(cpu_val, cpu_addr, dc->mem_idx);
0f8a249a
BS
5017 break;
5018 case 0x1e: /* V9 stxa */
66442b07 5019 save_state(dc);
6ae20372 5020 gen_st_asi(cpu_val, cpu_addr, insn, 8);
9fd1ae3a 5021 dc->npc = DYNAMIC_PC;
0f8a249a 5022 break;
3475187d 5023#endif
0f8a249a
BS
5024 default:
5025 goto illegal_insn;
5026 }
5027 } else if (xop > 0x23 && xop < 0x28) {
5b12f1e8 5028 if (gen_trap_ifnofpu(dc)) {
a80dde08 5029 goto jmp_insn;
5b12f1e8 5030 }
66442b07 5031 save_state(dc);
0f8a249a 5032 switch (xop) {
b89e94af 5033 case 0x24: /* stf, store fpreg */
2cade6a3 5034 gen_address_mask(dc, cpu_addr);
208ae657
RH
5035 cpu_src1_32 = gen_load_fpr_F(dc, rd);
5036 tcg_gen_ext_i32_tl(cpu_tmp0, cpu_src1_32);
527067d8 5037 tcg_gen_qemu_st32(cpu_tmp0, cpu_addr, dc->mem_idx);
0f8a249a
BS
5038 break;
5039 case 0x25: /* stfsr, V9 stxfsr */
3a3b925d 5040#ifdef TARGET_SPARC64
2cade6a3 5041 gen_address_mask(dc, cpu_addr);
c5f9864e 5042 tcg_gen_ld_i64(cpu_tmp64, cpu_env, offsetof(CPUSPARCState, fsr));
3a3b925d
BS
5043 if (rd == 1)
5044 tcg_gen_qemu_st64(cpu_tmp64, cpu_addr, dc->mem_idx);
527067d8
BS
5045 else
5046 tcg_gen_qemu_st32(cpu_tmp64, cpu_addr, dc->mem_idx);
3a3b925d 5047#else
c5f9864e 5048 tcg_gen_ld_i32(cpu_tmp32, cpu_env, offsetof(CPUSPARCState, fsr));
6ae20372 5049 tcg_gen_qemu_st32(cpu_tmp32, cpu_addr, dc->mem_idx);
3a3b925d 5050#endif
0f8a249a 5051 break;
1f587329
BS
5052 case 0x26:
5053#ifdef TARGET_SPARC64
1f587329 5054 /* V9 stqf, store quad fpreg */
2ea815ca 5055 {
a7812ae4 5056 TCGv_i32 r_const;
2ea815ca
BS
5057
5058 CHECK_FPU_FEATURE(dc, FLOAT128);
5059 gen_op_load_fpr_QT0(QFPREG(rd));
5060 r_const = tcg_const_i32(dc->mem_idx);
1295001c 5061 gen_address_mask(dc, cpu_addr);
fe8d8f0f 5062 gen_helper_stqf(cpu_env, cpu_addr, r_const);
a7812ae4 5063 tcg_temp_free_i32(r_const);
2ea815ca 5064 }
1f587329 5065 break;
1f587329
BS
5066#else /* !TARGET_SPARC64 */
5067 /* stdfq, store floating point queue */
5068#if defined(CONFIG_USER_ONLY)
5069 goto illegal_insn;
5070#else
0f8a249a
BS
5071 if (!supervisor(dc))
5072 goto priv_insn;
5b12f1e8 5073 if (gen_trap_ifnofpu(dc)) {
0f8a249a 5074 goto jmp_insn;
5b12f1e8 5075 }
0f8a249a 5076 goto nfq_insn;
1f587329 5077#endif
0f8a249a 5078#endif
b89e94af 5079 case 0x27: /* stdf, store double fpreg */
03fb8cfc
RH
5080 gen_address_mask(dc, cpu_addr);
5081 cpu_src1_64 = gen_load_fpr_D(dc, rd);
5082 tcg_gen_qemu_st64(cpu_src1_64, cpu_addr, dc->mem_idx);
0f8a249a
BS
5083 break;
5084 default:
5085 goto illegal_insn;
5086 }
5087 } else if (xop > 0x33 && xop < 0x3f) {
66442b07 5088 save_state(dc);
0f8a249a 5089 switch (xop) {
a4d17f19 5090#ifdef TARGET_SPARC64
0f8a249a 5091 case 0x34: /* V9 stfa */
5b12f1e8 5092 if (gen_trap_ifnofpu(dc)) {
5f06b547
TS
5093 goto jmp_insn;
5094 }
6ae20372 5095 gen_stf_asi(cpu_addr, insn, 4, rd);
0f8a249a 5096 break;
1f587329 5097 case 0x36: /* V9 stqfa */
2ea815ca 5098 {
a7812ae4 5099 TCGv_i32 r_const;
2ea815ca
BS
5100
5101 CHECK_FPU_FEATURE(dc, FLOAT128);
5b12f1e8 5102 if (gen_trap_ifnofpu(dc)) {
5f06b547
TS
5103 goto jmp_insn;
5104 }
2ea815ca 5105 r_const = tcg_const_i32(7);
fe8d8f0f 5106 gen_helper_check_align(cpu_env, cpu_addr, r_const);
a7812ae4 5107 tcg_temp_free_i32(r_const);
2ea815ca
BS
5108 gen_stf_asi(cpu_addr, insn, 16, QFPREG(rd));
5109 }
1f587329 5110 break;
0f8a249a 5111 case 0x37: /* V9 stdfa */
5b12f1e8 5112 if (gen_trap_ifnofpu(dc)) {
5f06b547
TS
5113 goto jmp_insn;
5114 }
6ae20372 5115 gen_stf_asi(cpu_addr, insn, 8, DFPREG(rd));
0f8a249a
BS
5116 break;
5117 case 0x3c: /* V9 casa */
71817e48 5118 gen_cas_asi(cpu_val, cpu_addr, cpu_src2, insn, rd);
6ae20372 5119 gen_movl_TN_reg(rd, cpu_val);
0f8a249a
BS
5120 break;
5121 case 0x3e: /* V9 casxa */
71817e48 5122 gen_casx_asi(cpu_val, cpu_addr, cpu_src2, insn, rd);
6ae20372 5123 gen_movl_TN_reg(rd, cpu_val);
0f8a249a 5124 break;
a4d17f19 5125#else
0f8a249a
BS
5126 case 0x34: /* stc */
5127 case 0x35: /* stcsr */
5128 case 0x36: /* stdcq */
5129 case 0x37: /* stdc */
5130 goto ncp_insn;
5131#endif
5132 default:
5133 goto illegal_insn;
5134 }
dc1a6971 5135 } else
0f8a249a
BS
5136 goto illegal_insn;
5137 }
5138 break;
cf495bcf
FB
5139 }
5140 /* default case for non jump instructions */
72cbca10 5141 if (dc->npc == DYNAMIC_PC) {
0f8a249a
BS
5142 dc->pc = DYNAMIC_PC;
5143 gen_op_next_insn();
72cbca10
FB
5144 } else if (dc->npc == JUMP_PC) {
5145 /* we can do a static jump */
6ae20372 5146 gen_branch2(dc, dc->jump_pc[0], dc->jump_pc[1], cpu_cond);
72cbca10
FB
5147 dc->is_br = 1;
5148 } else {
0f8a249a
BS
5149 dc->pc = dc->npc;
5150 dc->npc = dc->npc + 4;
cf495bcf 5151 }
e80cfcfc 5152 jmp_insn:
42a8aa83 5153 goto egress;
cf495bcf 5154 illegal_insn:
2ea815ca 5155 {
a7812ae4 5156 TCGv_i32 r_const;
2ea815ca 5157
66442b07 5158 save_state(dc);
2ea815ca 5159 r_const = tcg_const_i32(TT_ILL_INSN);
bc265319 5160 gen_helper_raise_exception(cpu_env, r_const);
a7812ae4 5161 tcg_temp_free_i32(r_const);
2ea815ca
BS
5162 dc->is_br = 1;
5163 }
42a8aa83 5164 goto egress;
64a88d5d 5165 unimp_flush:
2ea815ca 5166 {
a7812ae4 5167 TCGv_i32 r_const;
2ea815ca 5168
66442b07 5169 save_state(dc);
2ea815ca 5170 r_const = tcg_const_i32(TT_UNIMP_FLUSH);
bc265319 5171 gen_helper_raise_exception(cpu_env, r_const);
a7812ae4 5172 tcg_temp_free_i32(r_const);
2ea815ca
BS
5173 dc->is_br = 1;
5174 }
42a8aa83 5175 goto egress;
e80cfcfc 5176#if !defined(CONFIG_USER_ONLY)
e8af50a3 5177 priv_insn:
2ea815ca 5178 {
a7812ae4 5179 TCGv_i32 r_const;
2ea815ca 5180
66442b07 5181 save_state(dc);
2ea815ca 5182 r_const = tcg_const_i32(TT_PRIV_INSN);
bc265319 5183 gen_helper_raise_exception(cpu_env, r_const);
a7812ae4 5184 tcg_temp_free_i32(r_const);
2ea815ca
BS
5185 dc->is_br = 1;
5186 }
42a8aa83 5187 goto egress;
64a88d5d 5188#endif
e80cfcfc 5189 nfpu_insn:
66442b07 5190 save_state(dc);
e80cfcfc
FB
5191 gen_op_fpexception_im(FSR_FTT_UNIMPFPOP);
5192 dc->is_br = 1;
42a8aa83 5193 goto egress;
64a88d5d 5194#if !defined(CONFIG_USER_ONLY) && !defined(TARGET_SPARC64)
9143e598 5195 nfq_insn:
66442b07 5196 save_state(dc);
9143e598
BS
5197 gen_op_fpexception_im(FSR_FTT_SEQ_ERROR);
5198 dc->is_br = 1;
42a8aa83 5199 goto egress;
9143e598 5200#endif
fcc72045
BS
5201#ifndef TARGET_SPARC64
5202 ncp_insn:
2ea815ca
BS
5203 {
5204 TCGv r_const;
5205
66442b07 5206 save_state(dc);
2ea815ca 5207 r_const = tcg_const_i32(TT_NCP_INSN);
bc265319 5208 gen_helper_raise_exception(cpu_env, r_const);
2ea815ca
BS
5209 tcg_temp_free(r_const);
5210 dc->is_br = 1;
5211 }
42a8aa83 5212 goto egress;
fcc72045 5213#endif
42a8aa83
RH
5214 egress:
5215 tcg_temp_free(cpu_tmp1);
5216 tcg_temp_free(cpu_tmp2);
30038fd8
RH
5217 if (dc->n_t32 != 0) {
5218 int i;
5219 for (i = dc->n_t32 - 1; i >= 0; --i) {
5220 tcg_temp_free_i32(dc->t32[i]);
5221 }
5222 dc->n_t32 = 0;
5223 }
7a3f1944
FB
5224}
5225
2cfc5f17
TS
5226static inline void gen_intermediate_code_internal(TranslationBlock * tb,
5227 int spc, CPUSPARCState *env)
7a3f1944 5228{
72cbca10 5229 target_ulong pc_start, last_pc;
cf495bcf
FB
5230 uint16_t *gen_opc_end;
5231 DisasContext dc1, *dc = &dc1;
a1d1bb31 5232 CPUBreakpoint *bp;
e8af50a3 5233 int j, lj = -1;
2e70f6ef
PB
5234 int num_insns;
5235 int max_insns;
0184e266 5236 unsigned int insn;
cf495bcf
FB
5237
5238 memset(dc, 0, sizeof(DisasContext));
cf495bcf 5239 dc->tb = tb;
72cbca10 5240 pc_start = tb->pc;
cf495bcf 5241 dc->pc = pc_start;
e80cfcfc 5242 last_pc = dc->pc;
72cbca10 5243 dc->npc = (target_ulong) tb->cs_base;
8393617c 5244 dc->cc_op = CC_OP_DYNAMIC;
6f27aba6 5245 dc->mem_idx = cpu_mmu_index(env);
5578ceab 5246 dc->def = env->def;
f838e2c5
BS
5247 dc->fpu_enabled = tb_fpu_enabled(tb->flags);
5248 dc->address_mask_32bit = tb_am_enabled(tb->flags);
060718c1 5249 dc->singlestep = (env->singlestep_enabled || singlestep);
cf495bcf 5250 gen_opc_end = gen_opc_buf + OPC_MAX_SIZE;
cf495bcf 5251
2e70f6ef
PB
5252 num_insns = 0;
5253 max_insns = tb->cflags & CF_COUNT_MASK;
5254 if (max_insns == 0)
5255 max_insns = CF_COUNT_MASK;
5256 gen_icount_start();
cf495bcf 5257 do {
72cf2d4f
BS
5258 if (unlikely(!QTAILQ_EMPTY(&env->breakpoints))) {
5259 QTAILQ_FOREACH(bp, &env->breakpoints, entry) {
a1d1bb31 5260 if (bp->pc == dc->pc) {
0f8a249a 5261 if (dc->pc != pc_start)
66442b07 5262 save_state(dc);
bc265319 5263 gen_helper_debug(cpu_env);
57fec1fe 5264 tcg_gen_exit_tb(0);
0f8a249a 5265 dc->is_br = 1;
e80cfcfc 5266 goto exit_gen_loop;
e8af50a3
FB
5267 }
5268 }
5269 }
5270 if (spc) {
93fcfe39 5271 qemu_log("Search PC...\n");
e8af50a3
FB
5272 j = gen_opc_ptr - gen_opc_buf;
5273 if (lj < j) {
5274 lj++;
5275 while (lj < j)
5276 gen_opc_instr_start[lj++] = 0;
5277 gen_opc_pc[lj] = dc->pc;
5278 gen_opc_npc[lj] = dc->npc;
5279 gen_opc_instr_start[lj] = 1;
2e70f6ef 5280 gen_opc_icount[lj] = num_insns;
e8af50a3
FB
5281 }
5282 }
2e70f6ef
PB
5283 if (num_insns + 1 == max_insns && (tb->cflags & CF_LAST_IO))
5284 gen_io_start();
0f8a249a 5285 last_pc = dc->pc;
0184e266 5286 insn = cpu_ldl_code(env, dc->pc);
b09b2fd3
RH
5287
5288 cpu_tmp0 = tcg_temp_new();
5289 cpu_tmp32 = tcg_temp_new_i32();
5290 cpu_tmp64 = tcg_temp_new_i64();
5291 cpu_dst = tcg_temp_new();
5292 cpu_val = tcg_temp_new();
5293 cpu_addr = tcg_temp_new();
5294
0184e266 5295 disas_sparc_insn(dc, insn);
2e70f6ef 5296 num_insns++;
0f8a249a 5297
b09b2fd3
RH
5298 tcg_temp_free(cpu_addr);
5299 tcg_temp_free(cpu_val);
5300 tcg_temp_free(cpu_dst);
5301 tcg_temp_free_i64(cpu_tmp64);
5302 tcg_temp_free_i32(cpu_tmp32);
5303 tcg_temp_free(cpu_tmp0);
5304
0f8a249a
BS
5305 if (dc->is_br)
5306 break;
5307 /* if the next PC is different, we abort now */
5308 if (dc->pc != (last_pc + 4))
5309 break;
d39c0b99
FB
5310 /* if we reach a page boundary, we stop generation so that the
5311 PC of a TT_TFAULT exception is always in the right page */
5312 if ((dc->pc & (TARGET_PAGE_SIZE - 1)) == 0)
5313 break;
e80cfcfc
FB
5314 /* if single step mode, we generate only one instruction and
5315 generate an exception */
060718c1 5316 if (dc->singlestep) {
e80cfcfc
FB
5317 break;
5318 }
cf495bcf 5319 } while ((gen_opc_ptr < gen_opc_end) &&
2e70f6ef
PB
5320 (dc->pc - pc_start) < (TARGET_PAGE_SIZE - 32) &&
5321 num_insns < max_insns);
e80cfcfc
FB
5322
5323 exit_gen_loop:
b09b2fd3 5324 if (tb->cflags & CF_LAST_IO) {
2e70f6ef 5325 gen_io_end();
b09b2fd3 5326 }
72cbca10 5327 if (!dc->is_br) {
5fafdf24 5328 if (dc->pc != DYNAMIC_PC &&
72cbca10
FB
5329 (dc->npc != DYNAMIC_PC && dc->npc != JUMP_PC)) {
5330 /* static PC and NPC: we can use direct chaining */
2f5680ee 5331 gen_goto_tb(dc, 0, dc->pc, dc->npc);
72cbca10 5332 } else {
b09b2fd3 5333 if (dc->pc != DYNAMIC_PC) {
2f5680ee 5334 tcg_gen_movi_tl(cpu_pc, dc->pc);
b09b2fd3 5335 }
934da7ee 5336 save_npc(dc);
57fec1fe 5337 tcg_gen_exit_tb(0);
72cbca10
FB
5338 }
5339 }
2e70f6ef 5340 gen_icount_end(tb, num_insns);
cf495bcf 5341 *gen_opc_ptr = INDEX_op_end;
e8af50a3
FB
5342 if (spc) {
5343 j = gen_opc_ptr - gen_opc_buf;
5344 lj++;
5345 while (lj <= j)
5346 gen_opc_instr_start[lj++] = 0;
e8af50a3 5347#if 0
93fcfe39 5348 log_page_dump();
e8af50a3 5349#endif
c3278b7b
FB
5350 gen_opc_jump_pc[0] = dc->jump_pc[0];
5351 gen_opc_jump_pc[1] = dc->jump_pc[1];
e8af50a3 5352 } else {
e80cfcfc 5353 tb->size = last_pc + 4 - pc_start;
2e70f6ef 5354 tb->icount = num_insns;
e8af50a3 5355 }
7a3f1944 5356#ifdef DEBUG_DISAS
8fec2b8c 5357 if (qemu_loglevel_mask(CPU_LOG_TB_IN_ASM)) {
93fcfe39
AL
5358 qemu_log("--------------\n");
5359 qemu_log("IN: %s\n", lookup_symbol(pc_start));
5360 log_target_disas(pc_start, last_pc + 4 - pc_start, 0);
5361 qemu_log("\n");
cf495bcf 5362 }
7a3f1944 5363#endif
7a3f1944
FB
5364}
5365
2cfc5f17 5366void gen_intermediate_code(CPUSPARCState * env, TranslationBlock * tb)
7a3f1944 5367{
2cfc5f17 5368 gen_intermediate_code_internal(tb, 0, env);
7a3f1944
FB
5369}
5370
2cfc5f17 5371void gen_intermediate_code_pc(CPUSPARCState * env, TranslationBlock * tb)
7a3f1944 5372{
2cfc5f17 5373 gen_intermediate_code_internal(tb, 1, env);
7a3f1944
FB
5374}
5375
c48fcb47 5376void gen_intermediate_code_init(CPUSPARCState *env)
e80cfcfc 5377{
f5069b26 5378 unsigned int i;
c48fcb47 5379 static int inited;
f5069b26
BS
5380 static const char * const gregnames[8] = {
5381 NULL, // g0 not used
5382 "g1",
5383 "g2",
5384 "g3",
5385 "g4",
5386 "g5",
5387 "g6",
5388 "g7",
5389 };
30038fd8
RH
5390 static const char * const fregnames[32] = {
5391 "f0", "f2", "f4", "f6", "f8", "f10", "f12", "f14",
5392 "f16", "f18", "f20", "f22", "f24", "f26", "f28", "f30",
5393 "f32", "f34", "f36", "f38", "f40", "f42", "f44", "f46",
5394 "f48", "f50", "f52", "f54", "f56", "f58", "f60", "f62",
714547bb 5395 };
aaed909a 5396
1a2fb1c0
BS
5397 /* init various static tables */
5398 if (!inited) {
5399 inited = 1;
5400
a7812ae4
PB
5401 cpu_env = tcg_global_reg_new_ptr(TCG_AREG0, "env");
5402 cpu_regwptr = tcg_global_mem_new_ptr(TCG_AREG0,
c5f9864e 5403 offsetof(CPUSPARCState, regwptr),
a7812ae4 5404 "regwptr");
1a2fb1c0 5405#ifdef TARGET_SPARC64
c5f9864e 5406 cpu_xcc = tcg_global_mem_new_i32(TCG_AREG0, offsetof(CPUSPARCState, xcc),
a7812ae4 5407 "xcc");
c5f9864e 5408 cpu_asi = tcg_global_mem_new_i32(TCG_AREG0, offsetof(CPUSPARCState, asi),
a7812ae4 5409 "asi");
c5f9864e 5410 cpu_fprs = tcg_global_mem_new_i32(TCG_AREG0, offsetof(CPUSPARCState, fprs),
a7812ae4 5411 "fprs");
c5f9864e 5412 cpu_gsr = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, gsr),
255e1fcb 5413 "gsr");
a7812ae4 5414 cpu_tick_cmpr = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5415 offsetof(CPUSPARCState, tick_cmpr),
255e1fcb 5416 "tick_cmpr");
a7812ae4 5417 cpu_stick_cmpr = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5418 offsetof(CPUSPARCState, stick_cmpr),
255e1fcb 5419 "stick_cmpr");
a7812ae4 5420 cpu_hstick_cmpr = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5421 offsetof(CPUSPARCState, hstick_cmpr),
255e1fcb 5422 "hstick_cmpr");
c5f9864e 5423 cpu_hintp = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, hintp),
255e1fcb 5424 "hintp");
c5f9864e 5425 cpu_htba = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, htba),
a7812ae4 5426 "htba");
c5f9864e 5427 cpu_hver = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, hver),
a7812ae4
PB
5428 "hver");
5429 cpu_ssr = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5430 offsetof(CPUSPARCState, ssr), "ssr");
a7812ae4 5431 cpu_ver = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5432 offsetof(CPUSPARCState, version), "ver");
a7812ae4 5433 cpu_softint = tcg_global_mem_new_i32(TCG_AREG0,
c5f9864e 5434 offsetof(CPUSPARCState, softint),
a7812ae4 5435 "softint");
255e1fcb 5436#else
c5f9864e 5437 cpu_wim = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, wim),
255e1fcb 5438 "wim");
1a2fb1c0 5439#endif
c5f9864e 5440 cpu_cond = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, cond),
77f193da 5441 "cond");
c5f9864e 5442 cpu_cc_src = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, cc_src),
dc99a3f2 5443 "cc_src");
a7812ae4 5444 cpu_cc_src2 = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5445 offsetof(CPUSPARCState, cc_src2),
d9bdab86 5446 "cc_src2");
c5f9864e 5447 cpu_cc_dst = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, cc_dst),
dc99a3f2 5448 "cc_dst");
c5f9864e 5449 cpu_cc_op = tcg_global_mem_new_i32(TCG_AREG0, offsetof(CPUSPARCState, cc_op),
8393617c 5450 "cc_op");
c5f9864e 5451 cpu_psr = tcg_global_mem_new_i32(TCG_AREG0, offsetof(CPUSPARCState, psr),
a7812ae4 5452 "psr");
c5f9864e 5453 cpu_fsr = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, fsr),
87e92502 5454 "fsr");
c5f9864e 5455 cpu_pc = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, pc),
48d5c82b 5456 "pc");
c5f9864e 5457 cpu_npc = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, npc),
a7812ae4 5458 "npc");
c5f9864e 5459 cpu_y = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, y), "y");
255e1fcb 5460#ifndef CONFIG_USER_ONLY
c5f9864e 5461 cpu_tbr = tcg_global_mem_new(TCG_AREG0, offsetof(CPUSPARCState, tbr),
255e1fcb
BS
5462 "tbr");
5463#endif
30038fd8 5464 for (i = 1; i < 8; i++) {
a7812ae4 5465 cpu_gregs[i] = tcg_global_mem_new(TCG_AREG0,
c5f9864e 5466 offsetof(CPUSPARCState, gregs[i]),
f5069b26 5467 gregnames[i]);
30038fd8
RH
5468 }
5469 for (i = 0; i < TARGET_DPREGS; i++) {
5470 cpu_fpr[i] = tcg_global_mem_new_i64(TCG_AREG0,
c5f9864e 5471 offsetof(CPUSPARCState, fpr[i]),
45c7b743 5472 fregnames[i]);
30038fd8 5473 }
714547bb 5474
c9e03d8f
BS
5475 /* register helpers */
5476
a7812ae4 5477#define GEN_HELPER 2
c9e03d8f 5478#include "helper.h"
1a2fb1c0 5479 }
658138bc 5480}
d2856f1a 5481
c5f9864e 5482void restore_state_to_opc(CPUSPARCState *env, TranslationBlock *tb, int pc_pos)
d2856f1a
AJ
5483{
5484 target_ulong npc;
5485 env->pc = gen_opc_pc[pc_pos];
5486 npc = gen_opc_npc[pc_pos];
5487 if (npc == 1) {
5488 /* dynamic NPC: already stored */
5489 } else if (npc == 2) {
d7da2a10
BS
5490 /* jump PC: use 'cond' and the jump targets of the translation */
5491 if (env->cond) {
d2856f1a 5492 env->npc = gen_opc_jump_pc[0];
d7da2a10 5493 } else {
d2856f1a 5494 env->npc = gen_opc_jump_pc[1];
d7da2a10 5495 }
d2856f1a
AJ
5496 } else {
5497 env->npc = npc;
5498 }
14ed7adc
IK
5499
5500 /* flush pending conditional evaluations before exposing cpu state */
5501 if (CC_OP != CC_OP_FLAGS) {
2ffd9176 5502 helper_compute_psr(env);
14ed7adc 5503 }
d2856f1a 5504}