Statistics
| Branch: | Revision:

root / tcg / hppa / tcg-target.c @ 739734cb

History | View | Annotate | Download (53.1 kB)

1 f54b3f92 aurel32
/*
2 f54b3f92 aurel32
 * Tiny Code Generator for QEMU
3 f54b3f92 aurel32
 *
4 f54b3f92 aurel32
 * Copyright (c) 2008 Fabrice Bellard
5 f54b3f92 aurel32
 *
6 f54b3f92 aurel32
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 f54b3f92 aurel32
 * of this software and associated documentation files (the "Software"), to deal
8 f54b3f92 aurel32
 * in the Software without restriction, including without limitation the rights
9 f54b3f92 aurel32
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 f54b3f92 aurel32
 * copies of the Software, and to permit persons to whom the Software is
11 f54b3f92 aurel32
 * furnished to do so, subject to the following conditions:
12 f54b3f92 aurel32
 *
13 f54b3f92 aurel32
 * The above copyright notice and this permission notice shall be included in
14 f54b3f92 aurel32
 * all copies or substantial portions of the Software.
15 f54b3f92 aurel32
 *
16 f54b3f92 aurel32
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 f54b3f92 aurel32
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 f54b3f92 aurel32
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 f54b3f92 aurel32
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 f54b3f92 aurel32
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 f54b3f92 aurel32
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 f54b3f92 aurel32
 * THE SOFTWARE.
23 f54b3f92 aurel32
 */
24 f54b3f92 aurel32
25 d4a9eb1f blueswir1
#ifndef NDEBUG
26 f54b3f92 aurel32
static const char * const tcg_target_reg_names[TCG_TARGET_NB_REGS] = {
27 fd76e73a Richard Henderson
    "%r0", "%r1", "%rp", "%r3", "%r4", "%r5", "%r6", "%r7",
28 fd76e73a Richard Henderson
    "%r8", "%r9", "%r10", "%r11", "%r12", "%r13", "%r14", "%r15",
29 fd76e73a Richard Henderson
    "%r16", "%r17", "%r18", "%r19", "%r20", "%r21", "%r22", "%r23",
30 fd76e73a Richard Henderson
    "%r24", "%r25", "%r26", "%dp", "%ret0", "%ret1", "%sp", "%r31",
31 f54b3f92 aurel32
};
32 d4a9eb1f blueswir1
#endif
33 f54b3f92 aurel32
34 fd76e73a Richard Henderson
/* This is an 8 byte temp slot in the stack frame.  */
35 fd76e73a Richard Henderson
#define STACK_TEMP_OFS -16
36 fd76e73a Richard Henderson
37 fd76e73a Richard Henderson
#ifdef CONFIG_USE_GUEST_BASE
38 fd76e73a Richard Henderson
#define TCG_GUEST_BASE_REG TCG_REG_R16
39 fd76e73a Richard Henderson
#else
40 fd76e73a Richard Henderson
#define TCG_GUEST_BASE_REG TCG_REG_R0
41 fd76e73a Richard Henderson
#endif
42 fd76e73a Richard Henderson
43 f54b3f92 aurel32
static const int tcg_target_reg_alloc_order[] = {
44 f54b3f92 aurel32
    TCG_REG_R4,
45 f54b3f92 aurel32
    TCG_REG_R5,
46 f54b3f92 aurel32
    TCG_REG_R6,
47 f54b3f92 aurel32
    TCG_REG_R7,
48 f54b3f92 aurel32
    TCG_REG_R8,
49 f54b3f92 aurel32
    TCG_REG_R9,
50 f54b3f92 aurel32
    TCG_REG_R10,
51 f54b3f92 aurel32
    TCG_REG_R11,
52 f54b3f92 aurel32
    TCG_REG_R12,
53 f54b3f92 aurel32
    TCG_REG_R13,
54 f54b3f92 aurel32
55 f54b3f92 aurel32
    TCG_REG_R17,
56 f54b3f92 aurel32
    TCG_REG_R14,
57 f54b3f92 aurel32
    TCG_REG_R15,
58 f54b3f92 aurel32
    TCG_REG_R16,
59 fd76e73a Richard Henderson
60 fd76e73a Richard Henderson
    TCG_REG_R26,
61 fd76e73a Richard Henderson
    TCG_REG_R25,
62 fd76e73a Richard Henderson
    TCG_REG_R24,
63 fd76e73a Richard Henderson
    TCG_REG_R23,
64 fd76e73a Richard Henderson
65 fd76e73a Richard Henderson
    TCG_REG_RET0,
66 fd76e73a Richard Henderson
    TCG_REG_RET1,
67 f54b3f92 aurel32
};
68 f54b3f92 aurel32
69 f54b3f92 aurel32
static const int tcg_target_call_iarg_regs[4] = {
70 f54b3f92 aurel32
    TCG_REG_R26,
71 f54b3f92 aurel32
    TCG_REG_R25,
72 f54b3f92 aurel32
    TCG_REG_R24,
73 f54b3f92 aurel32
    TCG_REG_R23,
74 f54b3f92 aurel32
};
75 f54b3f92 aurel32
76 f54b3f92 aurel32
static const int tcg_target_call_oarg_regs[2] = {
77 f54b3f92 aurel32
    TCG_REG_RET0,
78 f54b3f92 aurel32
    TCG_REG_RET1,
79 f54b3f92 aurel32
};
80 f54b3f92 aurel32
81 fd76e73a Richard Henderson
/* True iff val fits a signed field of width BITS.  */
82 fd76e73a Richard Henderson
static inline int check_fit_tl(tcg_target_long val, unsigned int bits)
83 fd76e73a Richard Henderson
{
84 fd76e73a Richard Henderson
    return (val << ((sizeof(tcg_target_long) * 8 - bits))
85 fd76e73a Richard Henderson
            >> (sizeof(tcg_target_long) * 8 - bits)) == val;
86 fd76e73a Richard Henderson
}
87 fd76e73a Richard Henderson
88 fd76e73a Richard Henderson
/* True iff depi can be used to compute (reg | MASK).
89 fd76e73a Richard Henderson
   Accept a bit pattern like:
90 fd76e73a Richard Henderson
      0....01....1
91 fd76e73a Richard Henderson
      1....10....0
92 fd76e73a Richard Henderson
      0..01..10..0
93 fd76e73a Richard Henderson
   Copied from gcc sources.  */
94 fd76e73a Richard Henderson
static inline int or_mask_p(tcg_target_ulong mask)
95 fd76e73a Richard Henderson
{
96 0085bd51 Richard Henderson
    if (mask == 0 || mask == -1) {
97 0085bd51 Richard Henderson
        return 0;
98 0085bd51 Richard Henderson
    }
99 fd76e73a Richard Henderson
    mask += mask & -mask;
100 fd76e73a Richard Henderson
    return (mask & (mask - 1)) == 0;
101 fd76e73a Richard Henderson
}
102 fd76e73a Richard Henderson
103 fd76e73a Richard Henderson
/* True iff depi or extru can be used to compute (reg & mask).
104 fd76e73a Richard Henderson
   Accept a bit pattern like these:
105 fd76e73a Richard Henderson
      0....01....1
106 fd76e73a Richard Henderson
      1....10....0
107 fd76e73a Richard Henderson
      1..10..01..1
108 fd76e73a Richard Henderson
   Copied from gcc sources.  */
109 fd76e73a Richard Henderson
static inline int and_mask_p(tcg_target_ulong mask)
110 fd76e73a Richard Henderson
{
111 fd76e73a Richard Henderson
    return or_mask_p(~mask);
112 fd76e73a Richard Henderson
}
113 fd76e73a Richard Henderson
114 fd76e73a Richard Henderson
static int low_sign_ext(int val, int len)
115 fd76e73a Richard Henderson
{
116 fd76e73a Richard Henderson
    return (((val << 1) & ~(-1u << len)) | ((val >> (len - 1)) & 1));
117 fd76e73a Richard Henderson
}
118 fd76e73a Richard Henderson
119 fd76e73a Richard Henderson
static int reassemble_12(int as12)
120 fd76e73a Richard Henderson
{
121 fd76e73a Richard Henderson
    return (((as12 & 0x800) >> 11) |
122 fd76e73a Richard Henderson
            ((as12 & 0x400) >> 8) |
123 fd76e73a Richard Henderson
            ((as12 & 0x3ff) << 3));
124 fd76e73a Richard Henderson
}
125 fd76e73a Richard Henderson
126 fd76e73a Richard Henderson
static int reassemble_17(int as17)
127 fd76e73a Richard Henderson
{
128 fd76e73a Richard Henderson
    return (((as17 & 0x10000) >> 16) |
129 fd76e73a Richard Henderson
            ((as17 & 0x0f800) << 5) |
130 fd76e73a Richard Henderson
            ((as17 & 0x00400) >> 8) |
131 fd76e73a Richard Henderson
            ((as17 & 0x003ff) << 3));
132 fd76e73a Richard Henderson
}
133 fd76e73a Richard Henderson
134 fd76e73a Richard Henderson
static int reassemble_21(int as21)
135 fd76e73a Richard Henderson
{
136 fd76e73a Richard Henderson
    return (((as21 & 0x100000) >> 20) |
137 fd76e73a Richard Henderson
            ((as21 & 0x0ffe00) >> 8) |
138 fd76e73a Richard Henderson
            ((as21 & 0x000180) << 7) |
139 fd76e73a Richard Henderson
            ((as21 & 0x00007c) << 14) |
140 fd76e73a Richard Henderson
            ((as21 & 0x000003) << 12));
141 fd76e73a Richard Henderson
}
142 fd76e73a Richard Henderson
143 fd76e73a Richard Henderson
/* ??? Bizzarely, there is no PCREL12F relocation type.  I guess all
144 fd76e73a Richard Henderson
   such relocations are simply fully handled by the assembler.  */
145 fd76e73a Richard Henderson
#define R_PARISC_PCREL12F  R_PARISC_NONE
146 fd76e73a Richard Henderson
147 f54b3f92 aurel32
static void patch_reloc(uint8_t *code_ptr, int type,
148 f54b3f92 aurel32
                        tcg_target_long value, tcg_target_long addend)
149 f54b3f92 aurel32
{
150 fd76e73a Richard Henderson
    uint32_t *insn_ptr = (uint32_t *)code_ptr;
151 fd76e73a Richard Henderson
    uint32_t insn = *insn_ptr;
152 fd76e73a Richard Henderson
    tcg_target_long pcrel;
153 fd76e73a Richard Henderson
154 fd76e73a Richard Henderson
    value += addend;
155 fd76e73a Richard Henderson
    pcrel = (value - ((tcg_target_long)code_ptr + 8)) >> 2;
156 fd76e73a Richard Henderson
157 f54b3f92 aurel32
    switch (type) {
158 fd76e73a Richard Henderson
    case R_PARISC_PCREL12F:
159 fd76e73a Richard Henderson
        assert(check_fit_tl(pcrel, 12));
160 fd76e73a Richard Henderson
        /* ??? We assume all patches are forward.  See tcg_out_brcond
161 fd76e73a Richard Henderson
           re setting the NUL bit on the branch and eliding the nop.  */
162 fd76e73a Richard Henderson
        assert(pcrel >= 0);
163 fd76e73a Richard Henderson
        insn &= ~0x1ffdu;
164 fd76e73a Richard Henderson
        insn |= reassemble_12(pcrel);
165 fd76e73a Richard Henderson
        break;
166 f54b3f92 aurel32
    case R_PARISC_PCREL17F:
167 fd76e73a Richard Henderson
        assert(check_fit_tl(pcrel, 17));
168 fd76e73a Richard Henderson
        insn &= ~0x1f1ffdu;
169 fd76e73a Richard Henderson
        insn |= reassemble_17(pcrel);
170 f54b3f92 aurel32
        break;
171 f54b3f92 aurel32
    default:
172 f54b3f92 aurel32
        tcg_abort();
173 f54b3f92 aurel32
    }
174 fd76e73a Richard Henderson
175 fd76e73a Richard Henderson
    *insn_ptr = insn;
176 f54b3f92 aurel32
}
177 f54b3f92 aurel32
178 f54b3f92 aurel32
/* maximum number of register used for input function arguments */
179 f54b3f92 aurel32
static inline int tcg_target_get_call_iarg_regs_count(int flags)
180 f54b3f92 aurel32
{
181 f54b3f92 aurel32
    return 4;
182 f54b3f92 aurel32
}
183 f54b3f92 aurel32
184 f54b3f92 aurel32
/* parse target specific constraints */
185 d4a9eb1f blueswir1
static int target_parse_constraint(TCGArgConstraint *ct, const char **pct_str)
186 f54b3f92 aurel32
{
187 f54b3f92 aurel32
    const char *ct_str;
188 f54b3f92 aurel32
189 f54b3f92 aurel32
    ct_str = *pct_str;
190 f54b3f92 aurel32
    switch (ct_str[0]) {
191 f54b3f92 aurel32
    case 'r':
192 f54b3f92 aurel32
        ct->ct |= TCG_CT_REG;
193 f54b3f92 aurel32
        tcg_regset_set32(ct->u.regs, 0, 0xffffffff);
194 f54b3f92 aurel32
        break;
195 f54b3f92 aurel32
    case 'L': /* qemu_ld/st constraint */
196 f54b3f92 aurel32
        ct->ct |= TCG_CT_REG;
197 f54b3f92 aurel32
        tcg_regset_set32(ct->u.regs, 0, 0xffffffff);
198 f54b3f92 aurel32
        tcg_regset_reset_reg(ct->u.regs, TCG_REG_R26);
199 f54b3f92 aurel32
        tcg_regset_reset_reg(ct->u.regs, TCG_REG_R25);
200 f54b3f92 aurel32
        tcg_regset_reset_reg(ct->u.regs, TCG_REG_R24);
201 f54b3f92 aurel32
        tcg_regset_reset_reg(ct->u.regs, TCG_REG_R23);
202 f54b3f92 aurel32
        break;
203 fd76e73a Richard Henderson
    case 'Z':
204 fd76e73a Richard Henderson
        ct->ct |= TCG_CT_CONST_0;
205 fd76e73a Richard Henderson
        break;
206 fd76e73a Richard Henderson
    case 'I':
207 fd76e73a Richard Henderson
        ct->ct |= TCG_CT_CONST_S11;
208 fd76e73a Richard Henderson
        break;
209 fd76e73a Richard Henderson
    case 'J':
210 fd76e73a Richard Henderson
        ct->ct |= TCG_CT_CONST_S5;
211 fd76e73a Richard Henderson
        break;
212 91493631 Richard Henderson
    case 'K':
213 91493631 Richard Henderson
        ct->ct |= TCG_CT_CONST_MS11;
214 91493631 Richard Henderson
        break;
215 0085bd51 Richard Henderson
    case 'M':
216 0085bd51 Richard Henderson
        ct->ct |= TCG_CT_CONST_AND;
217 0085bd51 Richard Henderson
        break;
218 0085bd51 Richard Henderson
    case 'O':
219 0085bd51 Richard Henderson
        ct->ct |= TCG_CT_CONST_OR;
220 0085bd51 Richard Henderson
        break;
221 f54b3f92 aurel32
    default:
222 f54b3f92 aurel32
        return -1;
223 f54b3f92 aurel32
    }
224 f54b3f92 aurel32
    ct_str++;
225 f54b3f92 aurel32
    *pct_str = ct_str;
226 f54b3f92 aurel32
    return 0;
227 f54b3f92 aurel32
}
228 f54b3f92 aurel32
229 f54b3f92 aurel32
/* test if a constant matches the constraint */
230 fd76e73a Richard Henderson
static int tcg_target_const_match(tcg_target_long val,
231 fd76e73a Richard Henderson
                                  const TCGArgConstraint *arg_ct)
232 f54b3f92 aurel32
{
233 fd76e73a Richard Henderson
    int ct = arg_ct->ct;
234 fd76e73a Richard Henderson
    if (ct & TCG_CT_CONST) {
235 fd76e73a Richard Henderson
        return 1;
236 fd76e73a Richard Henderson
    } else if (ct & TCG_CT_CONST_0) {
237 fd76e73a Richard Henderson
        return val == 0;
238 fd76e73a Richard Henderson
    } else if (ct & TCG_CT_CONST_S5) {
239 fd76e73a Richard Henderson
        return check_fit_tl(val, 5);
240 fd76e73a Richard Henderson
    } else if (ct & TCG_CT_CONST_S11) {
241 fd76e73a Richard Henderson
        return check_fit_tl(val, 11);
242 91493631 Richard Henderson
    } else if (ct & TCG_CT_CONST_MS11) {
243 91493631 Richard Henderson
        return check_fit_tl(-val, 11);
244 0085bd51 Richard Henderson
    } else if (ct & TCG_CT_CONST_AND) {
245 0085bd51 Richard Henderson
        return and_mask_p(val);
246 0085bd51 Richard Henderson
    } else if (ct & TCG_CT_CONST_OR) {
247 0085bd51 Richard Henderson
        return or_mask_p(val);
248 fd76e73a Richard Henderson
    }
249 f54b3f92 aurel32
    return 0;
250 f54b3f92 aurel32
}
251 f54b3f92 aurel32
252 f54b3f92 aurel32
#define INSN_OP(x)       ((x) << 26)
253 f54b3f92 aurel32
#define INSN_EXT3BR(x)   ((x) << 13)
254 f54b3f92 aurel32
#define INSN_EXT3SH(x)   ((x) << 10)
255 f54b3f92 aurel32
#define INSN_EXT4(x)     ((x) << 6)
256 f54b3f92 aurel32
#define INSN_EXT5(x)     (x)
257 f54b3f92 aurel32
#define INSN_EXT6(x)     ((x) << 6)
258 f54b3f92 aurel32
#define INSN_EXT7(x)     ((x) << 6)
259 f54b3f92 aurel32
#define INSN_EXT8A(x)    ((x) << 6)
260 f54b3f92 aurel32
#define INSN_EXT8B(x)    ((x) << 5)
261 f54b3f92 aurel32
#define INSN_T(x)        (x)
262 f54b3f92 aurel32
#define INSN_R1(x)       ((x) << 16)
263 f54b3f92 aurel32
#define INSN_R2(x)       ((x) << 21)
264 f54b3f92 aurel32
#define INSN_DEP_LEN(x)  (32 - (x))
265 f54b3f92 aurel32
#define INSN_SHDEP_CP(x) ((31 - (x)) << 5)
266 f54b3f92 aurel32
#define INSN_SHDEP_P(x)  ((x) << 5)
267 f54b3f92 aurel32
#define INSN_COND(x)     ((x) << 13)
268 fd76e73a Richard Henderson
#define INSN_IM11(x)     low_sign_ext(x, 11)
269 fd76e73a Richard Henderson
#define INSN_IM14(x)     low_sign_ext(x, 14)
270 fd76e73a Richard Henderson
#define INSN_IM5(x)      (low_sign_ext(x, 5) << 16)
271 fd76e73a Richard Henderson
272 fd76e73a Richard Henderson
#define COND_NEVER   0
273 fd76e73a Richard Henderson
#define COND_EQ      1
274 fd76e73a Richard Henderson
#define COND_LT      2
275 fd76e73a Richard Henderson
#define COND_LE      3
276 fd76e73a Richard Henderson
#define COND_LTU     4
277 fd76e73a Richard Henderson
#define COND_LEU     5
278 fd76e73a Richard Henderson
#define COND_SV      6
279 fd76e73a Richard Henderson
#define COND_OD      7
280 fd76e73a Richard Henderson
#define COND_FALSE   8
281 fd76e73a Richard Henderson
282 fd76e73a Richard Henderson
#define INSN_ADD        (INSN_OP(0x02) | INSN_EXT6(0x18))
283 fd76e73a Richard Henderson
#define INSN_ADDC        (INSN_OP(0x02) | INSN_EXT6(0x1c))
284 fd76e73a Richard Henderson
#define INSN_ADDI        (INSN_OP(0x2d))
285 fd76e73a Richard Henderson
#define INSN_ADDIL        (INSN_OP(0x0a))
286 fd76e73a Richard Henderson
#define INSN_ADDL        (INSN_OP(0x02) | INSN_EXT6(0x28))
287 fd76e73a Richard Henderson
#define INSN_AND        (INSN_OP(0x02) | INSN_EXT6(0x08))
288 fd76e73a Richard Henderson
#define INSN_ANDCM        (INSN_OP(0x02) | INSN_EXT6(0x00))
289 fd76e73a Richard Henderson
#define INSN_COMCLR        (INSN_OP(0x02) | INSN_EXT6(0x22))
290 fd76e73a Richard Henderson
#define INSN_COMICLR        (INSN_OP(0x24))
291 fd76e73a Richard Henderson
#define INSN_DEP        (INSN_OP(0x35) | INSN_EXT3SH(3))
292 fd76e73a Richard Henderson
#define INSN_DEPI        (INSN_OP(0x35) | INSN_EXT3SH(7))
293 fd76e73a Richard Henderson
#define INSN_EXTRS        (INSN_OP(0x34) | INSN_EXT3SH(7))
294 fd76e73a Richard Henderson
#define INSN_EXTRU        (INSN_OP(0x34) | INSN_EXT3SH(6))
295 fd76e73a Richard Henderson
#define INSN_LDIL        (INSN_OP(0x08))
296 fd76e73a Richard Henderson
#define INSN_LDO        (INSN_OP(0x0d))
297 fd76e73a Richard Henderson
#define INSN_MTCTL        (INSN_OP(0x00) | INSN_EXT8B(0xc2))
298 fd76e73a Richard Henderson
#define INSN_OR                (INSN_OP(0x02) | INSN_EXT6(0x09))
299 fd76e73a Richard Henderson
#define INSN_SHD        (INSN_OP(0x34) | INSN_EXT3SH(2))
300 fd76e73a Richard Henderson
#define INSN_SUB        (INSN_OP(0x02) | INSN_EXT6(0x10))
301 fd76e73a Richard Henderson
#define INSN_SUBB        (INSN_OP(0x02) | INSN_EXT6(0x14))
302 fd76e73a Richard Henderson
#define INSN_SUBI        (INSN_OP(0x25))
303 fd76e73a Richard Henderson
#define INSN_VEXTRS        (INSN_OP(0x34) | INSN_EXT3SH(5))
304 fd76e73a Richard Henderson
#define INSN_VEXTRU        (INSN_OP(0x34) | INSN_EXT3SH(4))
305 fd76e73a Richard Henderson
#define INSN_VSHD        (INSN_OP(0x34) | INSN_EXT3SH(0))
306 fd76e73a Richard Henderson
#define INSN_XOR        (INSN_OP(0x02) | INSN_EXT6(0x0a))
307 fd76e73a Richard Henderson
#define INSN_ZDEP        (INSN_OP(0x35) | INSN_EXT3SH(2))
308 fd76e73a Richard Henderson
#define INSN_ZVDEP        (INSN_OP(0x35) | INSN_EXT3SH(0))
309 fd76e73a Richard Henderson
310 fd76e73a Richard Henderson
#define INSN_BL         (INSN_OP(0x3a) | INSN_EXT3BR(0))
311 fd76e73a Richard Henderson
#define INSN_BL_N       (INSN_OP(0x3a) | INSN_EXT3BR(0) | 2)
312 fd76e73a Richard Henderson
#define INSN_BLR        (INSN_OP(0x3a) | INSN_EXT3BR(2))
313 fd76e73a Richard Henderson
#define INSN_BV         (INSN_OP(0x3a) | INSN_EXT3BR(6))
314 fd76e73a Richard Henderson
#define INSN_BV_N       (INSN_OP(0x3a) | INSN_EXT3BR(6) | 2)
315 fd76e73a Richard Henderson
#define INSN_BLE_SR4    (INSN_OP(0x39) | (1 << 13))
316 fd76e73a Richard Henderson
317 fd76e73a Richard Henderson
#define INSN_LDB        (INSN_OP(0x10))
318 fd76e73a Richard Henderson
#define INSN_LDH        (INSN_OP(0x11))
319 fd76e73a Richard Henderson
#define INSN_LDW        (INSN_OP(0x12))
320 fd76e73a Richard Henderson
#define INSN_LDWM       (INSN_OP(0x13))
321 fd76e73a Richard Henderson
#define INSN_FLDDS      (INSN_OP(0x0b) | INSN_EXT4(0) | (1 << 12))
322 fd76e73a Richard Henderson
323 fd76e73a Richard Henderson
#define INSN_LDBX        (INSN_OP(0x03) | INSN_EXT4(0))
324 fd76e73a Richard Henderson
#define INSN_LDHX        (INSN_OP(0x03) | INSN_EXT4(1))
325 fd76e73a Richard Henderson
#define INSN_LDWX       (INSN_OP(0x03) | INSN_EXT4(2))
326 fd76e73a Richard Henderson
327 fd76e73a Richard Henderson
#define INSN_STB        (INSN_OP(0x18))
328 fd76e73a Richard Henderson
#define INSN_STH        (INSN_OP(0x19))
329 fd76e73a Richard Henderson
#define INSN_STW        (INSN_OP(0x1a))
330 fd76e73a Richard Henderson
#define INSN_STWM       (INSN_OP(0x1b))
331 fd76e73a Richard Henderson
#define INSN_FSTDS      (INSN_OP(0x0b) | INSN_EXT4(8) | (1 << 12))
332 fd76e73a Richard Henderson
333 fd76e73a Richard Henderson
#define INSN_COMBT      (INSN_OP(0x20))
334 fd76e73a Richard Henderson
#define INSN_COMBF      (INSN_OP(0x22))
335 fd76e73a Richard Henderson
#define INSN_COMIBT     (INSN_OP(0x21))
336 fd76e73a Richard Henderson
#define INSN_COMIBF     (INSN_OP(0x23))
337 fd76e73a Richard Henderson
338 fd76e73a Richard Henderson
/* supplied by libgcc */
339 fd76e73a Richard Henderson
extern void *__canonicalize_funcptr_for_compare(void *);
340 fd76e73a Richard Henderson
341 fd76e73a Richard Henderson
static void tcg_out_mov(TCGContext *s, int ret, int arg)
342 fd76e73a Richard Henderson
{
343 fd76e73a Richard Henderson
    /* PA1.1 defines COPY as OR r,0,t; PA2.0 defines COPY as LDO 0(r),t
344 fd76e73a Richard Henderson
       but hppa-dis.c is unaware of this definition */
345 fd76e73a Richard Henderson
    if (ret != arg) {
346 fd76e73a Richard Henderson
        tcg_out32(s, INSN_OR | INSN_T(ret) | INSN_R1(arg)
347 fd76e73a Richard Henderson
                  | INSN_R2(TCG_REG_R0));
348 fd76e73a Richard Henderson
    }
349 fd76e73a Richard Henderson
}
350 f54b3f92 aurel32
351 fd76e73a Richard Henderson
static void tcg_out_movi(TCGContext *s, TCGType type,
352 fd76e73a Richard Henderson
                         int ret, tcg_target_long arg)
353 fd76e73a Richard Henderson
{
354 fd76e73a Richard Henderson
    if (check_fit_tl(arg, 14)) {
355 fd76e73a Richard Henderson
        tcg_out32(s, INSN_LDO | INSN_R1(ret)
356 fd76e73a Richard Henderson
                  | INSN_R2(TCG_REG_R0) | INSN_IM14(arg));
357 fd76e73a Richard Henderson
    } else {
358 fd76e73a Richard Henderson
        uint32_t hi, lo;
359 fd76e73a Richard Henderson
        hi = arg >> 11;
360 fd76e73a Richard Henderson
        lo = arg & 0x7ff;
361 fd76e73a Richard Henderson
362 fd76e73a Richard Henderson
        tcg_out32(s, INSN_LDIL | INSN_R2(ret) | reassemble_21(hi));
363 fd76e73a Richard Henderson
        if (lo) {
364 fd76e73a Richard Henderson
            tcg_out32(s, INSN_LDO | INSN_R1(ret)
365 fd76e73a Richard Henderson
                      | INSN_R2(ret) | INSN_IM14(lo));
366 fd76e73a Richard Henderson
        }
367 fd76e73a Richard Henderson
    }
368 fd76e73a Richard Henderson
}
369 f54b3f92 aurel32
370 fd76e73a Richard Henderson
static void tcg_out_ldst(TCGContext *s, int ret, int addr,
371 fd76e73a Richard Henderson
                         tcg_target_long offset, int op)
372 fd76e73a Richard Henderson
{
373 fd76e73a Richard Henderson
    if (!check_fit_tl(offset, 14)) {
374 fd76e73a Richard Henderson
        uint32_t hi, lo, op;
375 f54b3f92 aurel32
376 fd76e73a Richard Henderson
        hi = offset >> 11;
377 fd76e73a Richard Henderson
        lo = offset & 0x7ff;
378 f54b3f92 aurel32
379 fd76e73a Richard Henderson
        if (addr == TCG_REG_R0) {
380 fd76e73a Richard Henderson
            op = INSN_LDIL | INSN_R2(TCG_REG_R1);
381 fd76e73a Richard Henderson
        } else {
382 fd76e73a Richard Henderson
            op = INSN_ADDIL | INSN_R2(addr);
383 fd76e73a Richard Henderson
        }
384 fd76e73a Richard Henderson
        tcg_out32(s, op | reassemble_21(hi));
385 f54b3f92 aurel32
386 fd76e73a Richard Henderson
        addr = TCG_REG_R1;
387 fd76e73a Richard Henderson
        offset = lo;
388 fd76e73a Richard Henderson
    }
389 f54b3f92 aurel32
390 fd76e73a Richard Henderson
    if (ret != addr || offset != 0 || op != INSN_LDO) {
391 fd76e73a Richard Henderson
        tcg_out32(s, op | INSN_R1(ret) | INSN_R2(addr) | INSN_IM14(offset));
392 fd76e73a Richard Henderson
    }
393 fd76e73a Richard Henderson
}
394 f54b3f92 aurel32
395 fd76e73a Richard Henderson
/* This function is required by tcg.c.  */
396 fd76e73a Richard Henderson
static inline void tcg_out_ld(TCGContext *s, TCGType type, int ret,
397 fd76e73a Richard Henderson
                              int arg1, tcg_target_long arg2)
398 fd76e73a Richard Henderson
{
399 fd76e73a Richard Henderson
    tcg_out_ldst(s, ret, arg1, arg2, INSN_LDW);
400 fd76e73a Richard Henderson
}
401 fd76e73a Richard Henderson
402 fd76e73a Richard Henderson
/* This function is required by tcg.c.  */
403 fd76e73a Richard Henderson
static inline void tcg_out_st(TCGContext *s, TCGType type, int ret,
404 fd76e73a Richard Henderson
                              int arg1, tcg_target_long arg2)
405 fd76e73a Richard Henderson
{
406 fd76e73a Richard Henderson
    tcg_out_ldst(s, ret, arg1, arg2, INSN_STW);
407 fd76e73a Richard Henderson
}
408 fd76e73a Richard Henderson
409 fd76e73a Richard Henderson
static void tcg_out_ldst_index(TCGContext *s, int data,
410 fd76e73a Richard Henderson
                               int base, int index, int op)
411 fd76e73a Richard Henderson
{
412 fd76e73a Richard Henderson
    tcg_out32(s, op | INSN_T(data) | INSN_R1(index) | INSN_R2(base));
413 fd76e73a Richard Henderson
}
414 fd76e73a Richard Henderson
415 fd76e73a Richard Henderson
static inline void tcg_out_addi2(TCGContext *s, int ret, int arg1,
416 fd76e73a Richard Henderson
                                 tcg_target_long val)
417 fd76e73a Richard Henderson
{
418 fd76e73a Richard Henderson
    tcg_out_ldst(s, ret, arg1, val, INSN_LDO);
419 fd76e73a Richard Henderson
}
420 f54b3f92 aurel32
421 fd76e73a Richard Henderson
/* This function is required by tcg.c.  */
422 fd76e73a Richard Henderson
static inline void tcg_out_addi(TCGContext *s, int reg, tcg_target_long val)
423 fd76e73a Richard Henderson
{
424 fd76e73a Richard Henderson
    tcg_out_addi2(s, reg, reg, val);
425 fd76e73a Richard Henderson
}
426 f54b3f92 aurel32
427 fd76e73a Richard Henderson
static inline void tcg_out_arith(TCGContext *s, int t, int r1, int r2, int op)
428 fd76e73a Richard Henderson
{
429 fd76e73a Richard Henderson
    tcg_out32(s, op | INSN_T(t) | INSN_R1(r1) | INSN_R2(r2));
430 fd76e73a Richard Henderson
}
431 f54b3f92 aurel32
432 fd76e73a Richard Henderson
static inline void tcg_out_arithi(TCGContext *s, int t, int r1,
433 fd76e73a Richard Henderson
                                  tcg_target_long val, int op)
434 f54b3f92 aurel32
{
435 fd76e73a Richard Henderson
    assert(check_fit_tl(val, 11));
436 fd76e73a Richard Henderson
    tcg_out32(s, op | INSN_R1(t) | INSN_R2(r1) | INSN_IM11(val));
437 f54b3f92 aurel32
}
438 f54b3f92 aurel32
439 fd76e73a Richard Henderson
static inline void tcg_out_nop(TCGContext *s)
440 f54b3f92 aurel32
{
441 fd76e73a Richard Henderson
    tcg_out_arith(s, TCG_REG_R0, TCG_REG_R0, TCG_REG_R0, INSN_OR);
442 fd76e73a Richard Henderson
}
443 f54b3f92 aurel32
444 fd76e73a Richard Henderson
static inline void tcg_out_mtctl_sar(TCGContext *s, int arg)
445 fd76e73a Richard Henderson
{
446 fd76e73a Richard Henderson
    tcg_out32(s, INSN_MTCTL | INSN_R2(11) | INSN_R1(arg));
447 fd76e73a Richard Henderson
}
448 fd76e73a Richard Henderson
449 fd76e73a Richard Henderson
/* Extract LEN bits at position OFS from ARG and place in RET.
450 fd76e73a Richard Henderson
   Note that here the bit ordering is reversed from the PA-RISC
451 fd76e73a Richard Henderson
   standard, such that the right-most bit is 0.  */
452 fd76e73a Richard Henderson
static inline void tcg_out_extr(TCGContext *s, int ret, int arg,
453 fd76e73a Richard Henderson
                                unsigned ofs, unsigned len, int sign)
454 fd76e73a Richard Henderson
{
455 fd76e73a Richard Henderson
    assert(ofs < 32 && len <= 32 - ofs);
456 fd76e73a Richard Henderson
    tcg_out32(s, (sign ? INSN_EXTRS : INSN_EXTRU)
457 fd76e73a Richard Henderson
              | INSN_R1(ret) | INSN_R2(arg)
458 fd76e73a Richard Henderson
              | INSN_SHDEP_P(31 - ofs) | INSN_DEP_LEN(len));
459 f54b3f92 aurel32
}
460 f54b3f92 aurel32
461 fd76e73a Richard Henderson
/* Likewise with OFS interpreted little-endian.  */
462 fd76e73a Richard Henderson
static inline void tcg_out_dep(TCGContext *s, int ret, int arg,
463 fd76e73a Richard Henderson
                               unsigned ofs, unsigned len)
464 f54b3f92 aurel32
{
465 fd76e73a Richard Henderson
    assert(ofs < 32 && len <= 32 - ofs);
466 fd76e73a Richard Henderson
    tcg_out32(s, INSN_DEP | INSN_R2(ret) | INSN_R1(arg)
467 fd76e73a Richard Henderson
              | INSN_SHDEP_CP(31 - ofs) | INSN_DEP_LEN(len));
468 fd76e73a Richard Henderson
}
469 fd76e73a Richard Henderson
470 fd76e73a Richard Henderson
static inline void tcg_out_shd(TCGContext *s, int ret, int hi, int lo,
471 fd76e73a Richard Henderson
                               unsigned count)
472 fd76e73a Richard Henderson
{
473 fd76e73a Richard Henderson
    assert(count < 32);
474 fd76e73a Richard Henderson
    tcg_out32(s, INSN_SHD | INSN_R1(hi) | INSN_R2(lo) | INSN_T(ret)
475 fd76e73a Richard Henderson
              | INSN_SHDEP_CP(count));
476 fd76e73a Richard Henderson
}
477 fd76e73a Richard Henderson
478 fd76e73a Richard Henderson
static void tcg_out_vshd(TCGContext *s, int ret, int hi, int lo, int creg)
479 fd76e73a Richard Henderson
{
480 fd76e73a Richard Henderson
    tcg_out_mtctl_sar(s, creg);
481 fd76e73a Richard Henderson
    tcg_out32(s, INSN_VSHD | INSN_T(ret) | INSN_R1(hi) | INSN_R2(lo));
482 fd76e73a Richard Henderson
}
483 fd76e73a Richard Henderson
484 fd76e73a Richard Henderson
static void tcg_out_ori(TCGContext *s, int ret, int arg, tcg_target_ulong m)
485 fd76e73a Richard Henderson
{
486 0085bd51 Richard Henderson
    int bs0, bs1;
487 0085bd51 Richard Henderson
488 0085bd51 Richard Henderson
    /* Note that the argument is constrained to match or_mask_p.  */
489 0085bd51 Richard Henderson
    for (bs0 = 0; bs0 < 32; bs0++) {
490 0085bd51 Richard Henderson
        if ((m & (1u << bs0)) != 0) {
491 0085bd51 Richard Henderson
            break;
492 fd76e73a Richard Henderson
        }
493 0085bd51 Richard Henderson
    }
494 0085bd51 Richard Henderson
    for (bs1 = bs0; bs1 < 32; bs1++) {
495 0085bd51 Richard Henderson
        if ((m & (1u << bs1)) == 0) {
496 0085bd51 Richard Henderson
            break;
497 fd76e73a Richard Henderson
        }
498 fd76e73a Richard Henderson
    }
499 0085bd51 Richard Henderson
    assert(bs1 == 32 || (1ul << bs1) > m);
500 0085bd51 Richard Henderson
501 0085bd51 Richard Henderson
    tcg_out_mov(s, ret, arg);
502 0085bd51 Richard Henderson
    tcg_out32(s, INSN_DEPI | INSN_R2(ret) | INSN_IM5(-1)
503 0085bd51 Richard Henderson
              | INSN_SHDEP_CP(31 - bs0) | INSN_DEP_LEN(bs1 - bs0));
504 fd76e73a Richard Henderson
}
505 fd76e73a Richard Henderson
506 fd76e73a Richard Henderson
static void tcg_out_andi(TCGContext *s, int ret, int arg, tcg_target_ulong m)
507 fd76e73a Richard Henderson
{
508 0085bd51 Richard Henderson
    int ls0, ls1, ms0;
509 fd76e73a Richard Henderson
510 0085bd51 Richard Henderson
    /* Note that the argument is constrained to match and_mask_p.  */
511 0085bd51 Richard Henderson
    for (ls0 = 0; ls0 < 32; ls0++) {
512 0085bd51 Richard Henderson
        if ((m & (1u << ls0)) == 0) {
513 0085bd51 Richard Henderson
            break;
514 fd76e73a Richard Henderson
        }
515 0085bd51 Richard Henderson
    }
516 0085bd51 Richard Henderson
    for (ls1 = ls0; ls1 < 32; ls1++) {
517 0085bd51 Richard Henderson
        if ((m & (1u << ls1)) != 0) {
518 0085bd51 Richard Henderson
            break;
519 fd76e73a Richard Henderson
        }
520 0085bd51 Richard Henderson
    }
521 0085bd51 Richard Henderson
    for (ms0 = ls1; ms0 < 32; ms0++) {
522 0085bd51 Richard Henderson
        if ((m & (1u << ms0)) == 0) {
523 0085bd51 Richard Henderson
            break;
524 fd76e73a Richard Henderson
        }
525 0085bd51 Richard Henderson
    }
526 0085bd51 Richard Henderson
    assert (ms0 == 32);
527 fd76e73a Richard Henderson
528 0085bd51 Richard Henderson
    if (ls1 == 32) {
529 0085bd51 Richard Henderson
        tcg_out_extr(s, ret, arg, 0, ls0, 0);
530 f54b3f92 aurel32
    } else {
531 0085bd51 Richard Henderson
        tcg_out_mov(s, ret, arg);
532 0085bd51 Richard Henderson
        tcg_out32(s, INSN_DEPI | INSN_R2(ret) | INSN_IM5(0)
533 0085bd51 Richard Henderson
                  | INSN_SHDEP_CP(31 - ls0) | INSN_DEP_LEN(ls1 - ls0));
534 f54b3f92 aurel32
    }
535 f54b3f92 aurel32
}
536 f54b3f92 aurel32
537 fd76e73a Richard Henderson
static inline void tcg_out_ext8s(TCGContext *s, int ret, int arg)
538 f54b3f92 aurel32
{
539 fd76e73a Richard Henderson
    tcg_out_extr(s, ret, arg, 0, 8, 1);
540 f54b3f92 aurel32
}
541 f54b3f92 aurel32
542 fd76e73a Richard Henderson
static inline void tcg_out_ext16s(TCGContext *s, int ret, int arg)
543 f54b3f92 aurel32
{
544 fd76e73a Richard Henderson
    tcg_out_extr(s, ret, arg, 0, 16, 1);
545 f54b3f92 aurel32
}
546 f54b3f92 aurel32
547 fd76e73a Richard Henderson
static void tcg_out_shli(TCGContext *s, int ret, int arg, int count)
548 f54b3f92 aurel32
{
549 fd76e73a Richard Henderson
    count &= 31;
550 fd76e73a Richard Henderson
    tcg_out32(s, INSN_ZDEP | INSN_R2(ret) | INSN_R1(arg)
551 fd76e73a Richard Henderson
              | INSN_SHDEP_CP(31 - count) | INSN_DEP_LEN(32 - count));
552 f54b3f92 aurel32
}
553 f54b3f92 aurel32
554 fd76e73a Richard Henderson
static void tcg_out_shl(TCGContext *s, int ret, int arg, int creg)
555 f54b3f92 aurel32
{
556 fd76e73a Richard Henderson
    tcg_out_arithi(s, TCG_REG_R20, creg, 31, INSN_SUBI);
557 fd76e73a Richard Henderson
    tcg_out_mtctl_sar(s, TCG_REG_R20);
558 fd76e73a Richard Henderson
    tcg_out32(s, INSN_ZVDEP | INSN_R2(ret) | INSN_R1(arg) | INSN_DEP_LEN(32));
559 f54b3f92 aurel32
}
560 f54b3f92 aurel32
561 fd76e73a Richard Henderson
static void tcg_out_shri(TCGContext *s, int ret, int arg, int count)
562 f54b3f92 aurel32
{
563 fd76e73a Richard Henderson
    count &= 31;
564 fd76e73a Richard Henderson
    tcg_out_extr(s, ret, arg, count, 32 - count, 0);
565 f54b3f92 aurel32
}
566 f54b3f92 aurel32
567 fd76e73a Richard Henderson
static void tcg_out_shr(TCGContext *s, int ret, int arg, int creg)
568 f54b3f92 aurel32
{
569 fd76e73a Richard Henderson
    tcg_out_vshd(s, ret, TCG_REG_R0, arg, creg);
570 f54b3f92 aurel32
}
571 f54b3f92 aurel32
572 fd76e73a Richard Henderson
static void tcg_out_sari(TCGContext *s, int ret, int arg, int count)
573 f54b3f92 aurel32
{
574 fd76e73a Richard Henderson
    count &= 31;
575 fd76e73a Richard Henderson
    tcg_out_extr(s, ret, arg, count, 32 - count, 1);
576 f54b3f92 aurel32
}
577 f54b3f92 aurel32
578 fd76e73a Richard Henderson
static void tcg_out_sar(TCGContext *s, int ret, int arg, int creg)
579 f54b3f92 aurel32
{
580 fd76e73a Richard Henderson
    tcg_out_arithi(s, TCG_REG_R20, creg, 31, INSN_SUBI);
581 fd76e73a Richard Henderson
    tcg_out_mtctl_sar(s, TCG_REG_R20);
582 fd76e73a Richard Henderson
    tcg_out32(s, INSN_VEXTRS | INSN_R1(ret) | INSN_R2(arg) | INSN_DEP_LEN(32));
583 f54b3f92 aurel32
}
584 f54b3f92 aurel32
585 fd76e73a Richard Henderson
static void tcg_out_rotli(TCGContext *s, int ret, int arg, int count)
586 f54b3f92 aurel32
{
587 fd76e73a Richard Henderson
    count &= 31;
588 fd76e73a Richard Henderson
    tcg_out_shd(s, ret, arg, arg, 32 - count);
589 f54b3f92 aurel32
}
590 f54b3f92 aurel32
591 fd76e73a Richard Henderson
static void tcg_out_rotl(TCGContext *s, int ret, int arg, int creg)
592 fd76e73a Richard Henderson
{
593 fd76e73a Richard Henderson
    tcg_out_arithi(s, TCG_REG_R20, creg, 32, INSN_SUBI);
594 fd76e73a Richard Henderson
    tcg_out_vshd(s, ret, arg, arg, TCG_REG_R20);
595 f54b3f92 aurel32
}
596 f54b3f92 aurel32
597 fd76e73a Richard Henderson
static void tcg_out_rotri(TCGContext *s, int ret, int arg, int count)
598 fd76e73a Richard Henderson
{
599 fd76e73a Richard Henderson
    count &= 31;
600 fd76e73a Richard Henderson
    tcg_out_shd(s, ret, arg, arg, count);
601 f54b3f92 aurel32
}
602 f54b3f92 aurel32
603 fd76e73a Richard Henderson
static void tcg_out_rotr(TCGContext *s, int ret, int arg, int creg)
604 fd76e73a Richard Henderson
{
605 fd76e73a Richard Henderson
    tcg_out_vshd(s, ret, arg, arg, creg);
606 f54b3f92 aurel32
}
607 f54b3f92 aurel32
608 fd76e73a Richard Henderson
static void tcg_out_bswap16(TCGContext *s, int ret, int arg, int sign)
609 fd76e73a Richard Henderson
{
610 fd76e73a Richard Henderson
    if (ret != arg) {
611 fd76e73a Richard Henderson
        tcg_out_mov(s, ret, arg);             /* arg =  xxAB */
612 fd76e73a Richard Henderson
    }
613 fd76e73a Richard Henderson
    tcg_out_dep(s, ret, ret, 16, 8);          /* ret =  xBAB */
614 fd76e73a Richard Henderson
    tcg_out_extr(s, ret, ret, 8, 16, sign);   /* ret =  ..BA */
615 f54b3f92 aurel32
}
616 f54b3f92 aurel32
617 fd76e73a Richard Henderson
static void tcg_out_bswap32(TCGContext *s, int ret, int arg, int temp)
618 f54b3f92 aurel32
{
619 fd76e73a Richard Henderson
                                          /* arg =  ABCD */
620 fd76e73a Richard Henderson
    tcg_out_rotri(s, temp, arg, 16);      /* temp = CDAB */
621 fd76e73a Richard Henderson
    tcg_out_dep(s, temp, temp, 16, 8);    /* temp = CBAB */
622 fd76e73a Richard Henderson
    tcg_out_shd(s, ret, arg, temp, 8);    /* ret =  DCBA */
623 f54b3f92 aurel32
}
624 f54b3f92 aurel32
625 fd76e73a Richard Henderson
static void tcg_out_call(TCGContext *s, void *func)
626 fd76e73a Richard Henderson
{
627 fd76e73a Richard Henderson
    tcg_target_long val, hi, lo, disp;
628 fd76e73a Richard Henderson
629 fd76e73a Richard Henderson
    val = (uint32_t)__canonicalize_funcptr_for_compare(func);
630 fd76e73a Richard Henderson
    disp = (val - ((tcg_target_long)s->code_ptr + 8)) >> 2;
631 fd76e73a Richard Henderson
632 fd76e73a Richard Henderson
    if (check_fit_tl(disp, 17)) {
633 fd76e73a Richard Henderson
        tcg_out32(s, INSN_BL_N | INSN_R2(TCG_REG_RP) | reassemble_17(disp));
634 fd76e73a Richard Henderson
    } else {
635 fd76e73a Richard Henderson
        hi = val >> 11;
636 fd76e73a Richard Henderson
        lo = val & 0x7ff;
637 fd76e73a Richard Henderson
638 fd76e73a Richard Henderson
        tcg_out32(s, INSN_LDIL | INSN_R2(TCG_REG_R20) | reassemble_21(hi));
639 fd76e73a Richard Henderson
        tcg_out32(s, INSN_BLE_SR4 | INSN_R2(TCG_REG_R20)
640 fd76e73a Richard Henderson
                  | reassemble_17(lo >> 2));
641 fd76e73a Richard Henderson
        tcg_out_mov(s, TCG_REG_RP, TCG_REG_R31);
642 fd76e73a Richard Henderson
    }
643 fd76e73a Richard Henderson
}
644 79383c9c blueswir1
645 fd76e73a Richard Henderson
static void tcg_out_xmpyu(TCGContext *s, int retl, int reth,
646 fd76e73a Richard Henderson
                          int arg1, int arg2)
647 fd76e73a Richard Henderson
{
648 fd76e73a Richard Henderson
    /* Store both words into the stack for copy to the FPU.  */
649 fd76e73a Richard Henderson
    tcg_out_ldst(s, arg1, TCG_REG_SP, STACK_TEMP_OFS, INSN_STW);
650 fd76e73a Richard Henderson
    tcg_out_ldst(s, arg2, TCG_REG_SP, STACK_TEMP_OFS + 4, INSN_STW);
651 fd76e73a Richard Henderson
652 fd76e73a Richard Henderson
    /* Load both words into the FPU at the same time.  We get away
653 fd76e73a Richard Henderson
       with this because we can address the left and right half of the
654 fd76e73a Richard Henderson
       FPU registers individually once loaded.  */
655 fd76e73a Richard Henderson
    /* fldds stack_temp(sp),fr22 */
656 fd76e73a Richard Henderson
    tcg_out32(s, INSN_FLDDS | INSN_R2(TCG_REG_SP)
657 fd76e73a Richard Henderson
              | INSN_IM5(STACK_TEMP_OFS) | INSN_T(22));
658 fd76e73a Richard Henderson
659 fd76e73a Richard Henderson
    /* xmpyu fr22r,fr22,fr22 */
660 fd76e73a Richard Henderson
    tcg_out32(s, 0x3ad64796);
661 fd76e73a Richard Henderson
662 fd76e73a Richard Henderson
    /* Store the 64-bit result back into the stack.  */
663 fd76e73a Richard Henderson
    /* fstds stack_temp(sp),fr22 */
664 fd76e73a Richard Henderson
    tcg_out32(s, INSN_FSTDS | INSN_R2(TCG_REG_SP)
665 fd76e73a Richard Henderson
              | INSN_IM5(STACK_TEMP_OFS) | INSN_T(22));
666 fd76e73a Richard Henderson
667 fd76e73a Richard Henderson
    /* Load the pieces of the result that the caller requested.  */
668 fd76e73a Richard Henderson
    if (reth) {
669 fd76e73a Richard Henderson
        tcg_out_ldst(s, reth, TCG_REG_SP, STACK_TEMP_OFS, INSN_LDW);
670 fd76e73a Richard Henderson
    }
671 fd76e73a Richard Henderson
    if (retl) {
672 fd76e73a Richard Henderson
        tcg_out_ldst(s, retl, TCG_REG_SP, STACK_TEMP_OFS + 4, INSN_LDW);
673 fd76e73a Richard Henderson
    }
674 fd76e73a Richard Henderson
}
675 fd76e73a Richard Henderson
676 91493631 Richard Henderson
static void tcg_out_add2(TCGContext *s, int destl, int desth,
677 91493631 Richard Henderson
                         int al, int ah, int bl, int bh, int blconst)
678 91493631 Richard Henderson
{
679 91493631 Richard Henderson
    int tmp = (destl == ah || destl == bh ? TCG_REG_R20 : destl);
680 91493631 Richard Henderson
681 91493631 Richard Henderson
    if (blconst) {
682 91493631 Richard Henderson
        tcg_out_arithi(s, tmp, al, bl, INSN_ADDI);
683 91493631 Richard Henderson
    } else {
684 91493631 Richard Henderson
        tcg_out_arith(s, tmp, al, bl, INSN_ADD);
685 91493631 Richard Henderson
    }
686 91493631 Richard Henderson
    tcg_out_arith(s, desth, ah, bh, INSN_ADDC);
687 91493631 Richard Henderson
688 91493631 Richard Henderson
    tcg_out_mov(s, destl, tmp);
689 91493631 Richard Henderson
}
690 91493631 Richard Henderson
691 91493631 Richard Henderson
static void tcg_out_sub2(TCGContext *s, int destl, int desth, int al, int ah,
692 91493631 Richard Henderson
                         int bl, int bh, int alconst, int blconst)
693 91493631 Richard Henderson
{
694 91493631 Richard Henderson
    int tmp = (destl == ah || destl == bh ? TCG_REG_R20 : destl);
695 91493631 Richard Henderson
696 91493631 Richard Henderson
    if (alconst) {
697 91493631 Richard Henderson
        if (blconst) {
698 91493631 Richard Henderson
            tcg_out_movi(s, TCG_TYPE_I32, TCG_REG_R20, bl);
699 91493631 Richard Henderson
            bl = TCG_REG_R20;
700 91493631 Richard Henderson
        }
701 91493631 Richard Henderson
        tcg_out_arithi(s, tmp, bl, al, INSN_SUBI);
702 91493631 Richard Henderson
    } else if (blconst) {
703 91493631 Richard Henderson
        tcg_out_arithi(s, tmp, al, -bl, INSN_ADDI);
704 91493631 Richard Henderson
    } else {
705 91493631 Richard Henderson
        tcg_out_arith(s, tmp, al, bl, INSN_SUB);
706 91493631 Richard Henderson
    }
707 91493631 Richard Henderson
    tcg_out_arith(s, desth, ah, bh, INSN_SUBB);
708 91493631 Richard Henderson
709 91493631 Richard Henderson
    tcg_out_mov(s, destl, tmp);
710 91493631 Richard Henderson
}
711 91493631 Richard Henderson
712 fd76e73a Richard Henderson
static void tcg_out_branch(TCGContext *s, int label_index, int nul)
713 fd76e73a Richard Henderson
{
714 fd76e73a Richard Henderson
    TCGLabel *l = &s->labels[label_index];
715 fd76e73a Richard Henderson
    uint32_t op = nul ? INSN_BL_N : INSN_BL;
716 fd76e73a Richard Henderson
717 fd76e73a Richard Henderson
    if (l->has_value) {
718 fd76e73a Richard Henderson
        tcg_target_long val = l->u.value;
719 fd76e73a Richard Henderson
720 fd76e73a Richard Henderson
        val -= (tcg_target_long)s->code_ptr + 8;
721 fd76e73a Richard Henderson
        val >>= 2;
722 fd76e73a Richard Henderson
        assert(check_fit_tl(val, 17));
723 fd76e73a Richard Henderson
724 fd76e73a Richard Henderson
        tcg_out32(s, op | reassemble_17(val));
725 fd76e73a Richard Henderson
    } else {
726 fd76e73a Richard Henderson
        tcg_out_reloc(s, s->code_ptr, R_PARISC_PCREL17F, label_index, 0);
727 fd76e73a Richard Henderson
        tcg_out32(s, op);
728 fd76e73a Richard Henderson
    }
729 fd76e73a Richard Henderson
}
730 fd76e73a Richard Henderson
731 fd76e73a Richard Henderson
static const uint8_t tcg_cond_to_cmp_cond[10] =
732 fd76e73a Richard Henderson
{
733 fd76e73a Richard Henderson
    [TCG_COND_EQ] = COND_EQ,
734 fd76e73a Richard Henderson
    [TCG_COND_NE] = COND_EQ | COND_FALSE,
735 fd76e73a Richard Henderson
    [TCG_COND_LT] = COND_LT,
736 fd76e73a Richard Henderson
    [TCG_COND_GE] = COND_LT | COND_FALSE,
737 fd76e73a Richard Henderson
    [TCG_COND_LE] = COND_LE,
738 fd76e73a Richard Henderson
    [TCG_COND_GT] = COND_LE | COND_FALSE,
739 fd76e73a Richard Henderson
    [TCG_COND_LTU] = COND_LTU,
740 fd76e73a Richard Henderson
    [TCG_COND_GEU] = COND_LTU | COND_FALSE,
741 fd76e73a Richard Henderson
    [TCG_COND_LEU] = COND_LEU,
742 fd76e73a Richard Henderson
    [TCG_COND_GTU] = COND_LEU | COND_FALSE,
743 fd76e73a Richard Henderson
};
744 fd76e73a Richard Henderson
745 fd76e73a Richard Henderson
static void tcg_out_brcond(TCGContext *s, int cond, TCGArg c1,
746 fd76e73a Richard Henderson
                           TCGArg c2, int c2const, int label_index)
747 fd76e73a Richard Henderson
{
748 fd76e73a Richard Henderson
    TCGLabel *l = &s->labels[label_index];
749 fd76e73a Richard Henderson
    int op, pacond;
750 fd76e73a Richard Henderson
751 fd76e73a Richard Henderson
    /* Note that COMIB operates as if the immediate is the first
752 fd76e73a Richard Henderson
       operand.  We model brcond with the immediate in the second
753 fd76e73a Richard Henderson
       to better match what targets are likely to give us.  For
754 fd76e73a Richard Henderson
       consistency, model COMB with reversed operands as well.  */
755 fd76e73a Richard Henderson
    pacond = tcg_cond_to_cmp_cond[tcg_swap_cond(cond)];
756 fd76e73a Richard Henderson
757 fd76e73a Richard Henderson
    if (c2const) {
758 fd76e73a Richard Henderson
        op = (pacond & COND_FALSE ? INSN_COMIBF : INSN_COMIBT);
759 fd76e73a Richard Henderson
        op |= INSN_IM5(c2);
760 fd76e73a Richard Henderson
    } else {
761 fd76e73a Richard Henderson
        op = (pacond & COND_FALSE ? INSN_COMBF : INSN_COMBT);
762 fd76e73a Richard Henderson
        op |= INSN_R1(c2);
763 fd76e73a Richard Henderson
    }
764 fd76e73a Richard Henderson
    op |= INSN_R2(c1);
765 fd76e73a Richard Henderson
    op |= INSN_COND(pacond & 7);
766 fd76e73a Richard Henderson
767 fd76e73a Richard Henderson
    if (l->has_value) {
768 fd76e73a Richard Henderson
        tcg_target_long val = l->u.value;
769 fd76e73a Richard Henderson
770 fd76e73a Richard Henderson
        val -= (tcg_target_long)s->code_ptr + 8;
771 fd76e73a Richard Henderson
        val >>= 2;
772 fd76e73a Richard Henderson
        assert(check_fit_tl(val, 12));
773 fd76e73a Richard Henderson
774 fd76e73a Richard Henderson
        /* ??? Assume that all branches to defined labels are backward.
775 fd76e73a Richard Henderson
           Which means that if the nul bit is set, the delay slot is
776 fd76e73a Richard Henderson
           executed if the branch is taken, and not executed in fallthru.  */
777 fd76e73a Richard Henderson
        tcg_out32(s, op | reassemble_12(val));
778 fd76e73a Richard Henderson
        tcg_out_nop(s);
779 fd76e73a Richard Henderson
    } else {
780 fd76e73a Richard Henderson
        tcg_out_reloc(s, s->code_ptr, R_PARISC_PCREL12F, label_index, 0);
781 fd76e73a Richard Henderson
        /* ??? Assume that all branches to undefined labels are forward.
782 fd76e73a Richard Henderson
           Which means that if the nul bit is set, the delay slot is
783 fd76e73a Richard Henderson
           not executed if the branch is taken, which is what we want.  */
784 fd76e73a Richard Henderson
        tcg_out32(s, op | 2);
785 fd76e73a Richard Henderson
    }
786 fd76e73a Richard Henderson
}
787 fd76e73a Richard Henderson
788 fd76e73a Richard Henderson
static void tcg_out_comclr(TCGContext *s, int cond, TCGArg ret,
789 fd76e73a Richard Henderson
                           TCGArg c1, TCGArg c2, int c2const)
790 fd76e73a Richard Henderson
{
791 fd76e73a Richard Henderson
    int op, pacond;
792 fd76e73a Richard Henderson
793 fd76e73a Richard Henderson
    /* Note that COMICLR operates as if the immediate is the first
794 fd76e73a Richard Henderson
       operand.  We model setcond with the immediate in the second
795 fd76e73a Richard Henderson
       to better match what targets are likely to give us.  For
796 fd76e73a Richard Henderson
       consistency, model COMCLR with reversed operands as well.  */
797 fd76e73a Richard Henderson
    pacond = tcg_cond_to_cmp_cond[tcg_swap_cond(cond)];
798 fd76e73a Richard Henderson
799 fd76e73a Richard Henderson
    if (c2const) {
800 fd76e73a Richard Henderson
        op = INSN_COMICLR | INSN_R2(c1) | INSN_R1(ret) | INSN_IM11(c2);
801 fd76e73a Richard Henderson
    } else {
802 fd76e73a Richard Henderson
        op = INSN_COMCLR | INSN_R2(c1) | INSN_R1(c2) | INSN_T(ret);
803 fd76e73a Richard Henderson
    }
804 fd76e73a Richard Henderson
    op |= INSN_COND(pacond & 7);
805 fd76e73a Richard Henderson
    op |= pacond & COND_FALSE ? 1 << 12 : 0;
806 fd76e73a Richard Henderson
807 fd76e73a Richard Henderson
    tcg_out32(s, op);
808 fd76e73a Richard Henderson
}
809 fd76e73a Richard Henderson
810 fd76e73a Richard Henderson
static void tcg_out_brcond2(TCGContext *s, int cond, TCGArg al, TCGArg ah,
811 fd76e73a Richard Henderson
                            TCGArg bl, int blconst, TCGArg bh, int bhconst,
812 fd76e73a Richard Henderson
                            int label_index)
813 fd76e73a Richard Henderson
{
814 fd76e73a Richard Henderson
    switch (cond) {
815 fd76e73a Richard Henderson
    case TCG_COND_EQ:
816 fd76e73a Richard Henderson
    case TCG_COND_NE:
817 fd76e73a Richard Henderson
        tcg_out_comclr(s, tcg_invert_cond(cond), TCG_REG_R0, al, bl, blconst);
818 fd76e73a Richard Henderson
        tcg_out_brcond(s, cond, ah, bh, bhconst, label_index);
819 fd76e73a Richard Henderson
        break;
820 fd76e73a Richard Henderson
821 fd76e73a Richard Henderson
    default:
822 fd76e73a Richard Henderson
        tcg_out_brcond(s, cond, ah, bh, bhconst, label_index);
823 fd76e73a Richard Henderson
        tcg_out_comclr(s, TCG_COND_NE, TCG_REG_R0, ah, bh, bhconst);
824 fd76e73a Richard Henderson
        tcg_out_brcond(s, tcg_unsigned_cond(cond),
825 fd76e73a Richard Henderson
                       al, bl, blconst, label_index);
826 fd76e73a Richard Henderson
        break;
827 fd76e73a Richard Henderson
    }
828 fd76e73a Richard Henderson
}
829 fd76e73a Richard Henderson
830 fd76e73a Richard Henderson
static void tcg_out_setcond(TCGContext *s, int cond, TCGArg ret,
831 fd76e73a Richard Henderson
                            TCGArg c1, TCGArg c2, int c2const)
832 fd76e73a Richard Henderson
{
833 fd76e73a Richard Henderson
    tcg_out_comclr(s, tcg_invert_cond(cond), ret, c1, c2, c2const);
834 fd76e73a Richard Henderson
    tcg_out_movi(s, TCG_TYPE_I32, ret, 1);
835 fd76e73a Richard Henderson
}
836 fd76e73a Richard Henderson
837 fd76e73a Richard Henderson
static void tcg_out_setcond2(TCGContext *s, int cond, TCGArg ret,
838 fd76e73a Richard Henderson
                             TCGArg al, TCGArg ah, TCGArg bl, int blconst,
839 fd76e73a Richard Henderson
                             TCGArg bh, int bhconst)
840 fd76e73a Richard Henderson
{
841 fd76e73a Richard Henderson
    int scratch = TCG_REG_R20;
842 fd76e73a Richard Henderson
843 fd76e73a Richard Henderson
    if (ret != al && ret != ah
844 fd76e73a Richard Henderson
        && (blconst || ret != bl)
845 fd76e73a Richard Henderson
        && (bhconst || ret != bh)) {
846 fd76e73a Richard Henderson
        scratch = ret;
847 fd76e73a Richard Henderson
    }
848 fd76e73a Richard Henderson
849 fd76e73a Richard Henderson
    switch (cond) {
850 fd76e73a Richard Henderson
    case TCG_COND_EQ:
851 fd76e73a Richard Henderson
    case TCG_COND_NE:
852 fd76e73a Richard Henderson
        tcg_out_setcond(s, cond, scratch, al, bl, blconst);
853 fd76e73a Richard Henderson
        tcg_out_comclr(s, TCG_COND_EQ, TCG_REG_R0, ah, bh, bhconst);
854 fd76e73a Richard Henderson
        tcg_out_movi(s, TCG_TYPE_I32, scratch, cond == TCG_COND_NE);
855 fd76e73a Richard Henderson
        break;
856 fd76e73a Richard Henderson
857 fd76e73a Richard Henderson
    default:
858 fd76e73a Richard Henderson
        tcg_out_setcond(s, tcg_unsigned_cond(cond), scratch, al, bl, blconst);
859 fd76e73a Richard Henderson
        tcg_out_comclr(s, TCG_COND_EQ, TCG_REG_R0, ah, bh, bhconst);
860 fd76e73a Richard Henderson
        tcg_out_movi(s, TCG_TYPE_I32, scratch, 0);
861 fd76e73a Richard Henderson
        tcg_out_comclr(s, cond, TCG_REG_R0, ah, bh, bhconst);
862 fd76e73a Richard Henderson
        tcg_out_movi(s, TCG_TYPE_I32, scratch, 1);
863 fd76e73a Richard Henderson
        break;
864 fd76e73a Richard Henderson
    }
865 fd76e73a Richard Henderson
866 fd76e73a Richard Henderson
    tcg_out_mov(s, ret, scratch);
867 fd76e73a Richard Henderson
}
868 fd76e73a Richard Henderson
869 fd76e73a Richard Henderson
#if defined(CONFIG_SOFTMMU)
870 79383c9c blueswir1
#include "../../softmmu_defs.h"
871 f54b3f92 aurel32
872 f54b3f92 aurel32
static void *qemu_ld_helpers[4] = {
873 f54b3f92 aurel32
    __ldb_mmu,
874 f54b3f92 aurel32
    __ldw_mmu,
875 f54b3f92 aurel32
    __ldl_mmu,
876 f54b3f92 aurel32
    __ldq_mmu,
877 f54b3f92 aurel32
};
878 f54b3f92 aurel32
879 f54b3f92 aurel32
static void *qemu_st_helpers[4] = {
880 f54b3f92 aurel32
    __stb_mmu,
881 f54b3f92 aurel32
    __stw_mmu,
882 f54b3f92 aurel32
    __stl_mmu,
883 f54b3f92 aurel32
    __stq_mmu,
884 f54b3f92 aurel32
};
885 fd76e73a Richard Henderson
886 fd76e73a Richard Henderson
/* Load and compare a TLB entry, and branch if TLB miss.  OFFSET is set to
887 fd76e73a Richard Henderson
   the offset of the first ADDR_READ or ADDR_WRITE member of the appropriate
888 fd76e73a Richard Henderson
   TLB for the memory index.  The return value is the offset from ENV
889 fd76e73a Richard Henderson
   contained in R1 afterward (to be used when loading ADDEND); if the
890 fd76e73a Richard Henderson
   return value is 0, R1 is not used.  */
891 fd76e73a Richard Henderson
892 fd76e73a Richard Henderson
static int tcg_out_tlb_read(TCGContext *s, int r0, int r1, int addrlo,
893 fd76e73a Richard Henderson
                            int addrhi, int s_bits, int lab_miss, int offset)
894 fd76e73a Richard Henderson
{
895 fd76e73a Richard Henderson
    int ret;
896 fd76e73a Richard Henderson
897 fd76e73a Richard Henderson
    /* Extracting the index into the TLB.  The "normal C operation" is
898 fd76e73a Richard Henderson
          r1 = addr_reg >> TARGET_PAGE_BITS;
899 fd76e73a Richard Henderson
          r1 &= CPU_TLB_SIZE - 1;
900 fd76e73a Richard Henderson
          r1 <<= CPU_TLB_ENTRY_BITS;
901 fd76e73a Richard Henderson
       What this does is extract CPU_TLB_BITS beginning at TARGET_PAGE_BITS
902 fd76e73a Richard Henderson
       and place them at CPU_TLB_ENTRY_BITS.  We can combine the first two
903 fd76e73a Richard Henderson
       operations with an EXTRU.  Unfortunately, the current value of
904 fd76e73a Richard Henderson
       CPU_TLB_ENTRY_BITS is > 3, so we can't merge that shift with the
905 fd76e73a Richard Henderson
       add that follows.  */
906 fd76e73a Richard Henderson
    tcg_out_extr(s, r1, addrlo, TARGET_PAGE_BITS, CPU_TLB_BITS, 0);
907 fd76e73a Richard Henderson
    tcg_out_shli(s, r1, r1, CPU_TLB_ENTRY_BITS);
908 fd76e73a Richard Henderson
    tcg_out_arith(s, r1, r1, TCG_AREG0, INSN_ADDL);
909 fd76e73a Richard Henderson
910 fd76e73a Richard Henderson
    /* Make sure that both the addr_{read,write} and addend can be
911 fd76e73a Richard Henderson
       read with a 14-bit offset from the same base register.  */
912 fd76e73a Richard Henderson
    if (check_fit_tl(offset + CPU_TLB_SIZE, 14)) {
913 fd76e73a Richard Henderson
        ret = 0;
914 fd76e73a Richard Henderson
    } else {
915 fd76e73a Richard Henderson
        ret = (offset + 0x400) & ~0x7ff;
916 fd76e73a Richard Henderson
        offset = ret - offset;
917 fd76e73a Richard Henderson
        tcg_out_addi2(s, TCG_REG_R1, r1, ret);
918 fd76e73a Richard Henderson
        r1 = TCG_REG_R1;
919 fd76e73a Richard Henderson
    }
920 fd76e73a Richard Henderson
921 fd76e73a Richard Henderson
    /* Load the entry from the computed slot.  */
922 fd76e73a Richard Henderson
    if (TARGET_LONG_BITS == 64) {
923 fd76e73a Richard Henderson
        tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_R23, r1, offset);
924 fd76e73a Richard Henderson
        tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_R20, r1, offset + 4);
925 fd76e73a Richard Henderson
    } else {
926 fd76e73a Richard Henderson
        tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_R20, r1, offset);
927 fd76e73a Richard Henderson
    }
928 fd76e73a Richard Henderson
929 739734cb Richard Henderson
    /* Compute the value that ought to appear in the TLB for a hit, namely, the page
930 739734cb Richard Henderson
       of the address.  We include the low N bits of the address to catch unaligned
931 739734cb Richard Henderson
       accesses and force them onto the slow path.  Do this computation after having
932 739734cb Richard Henderson
       issued the load from the TLB slot to give the load time to complete.  */
933 739734cb Richard Henderson
    tcg_out_andi(s, r0, addrlo, TARGET_PAGE_MASK | ((1 << s_bits) - 1));
934 739734cb Richard Henderson
935 fd76e73a Richard Henderson
    /* If not equal, jump to lab_miss. */
936 fd76e73a Richard Henderson
    if (TARGET_LONG_BITS == 64) {
937 fd76e73a Richard Henderson
        tcg_out_brcond2(s, TCG_COND_NE, TCG_REG_R20, TCG_REG_R23,
938 fd76e73a Richard Henderson
                        r0, 0, addrhi, 0, lab_miss);
939 fd76e73a Richard Henderson
    } else {
940 fd76e73a Richard Henderson
        tcg_out_brcond(s, TCG_COND_NE, TCG_REG_R20, r0, 0, lab_miss);
941 fd76e73a Richard Henderson
    }
942 fd76e73a Richard Henderson
943 fd76e73a Richard Henderson
    return ret;
944 fd76e73a Richard Henderson
}
945 f54b3f92 aurel32
#endif
946 f54b3f92 aurel32
947 f061b40e Richard Henderson
static void tcg_out_qemu_ld_direct(TCGContext *s, int datalo_reg, int datahi_reg,
948 f061b40e Richard Henderson
                                   int addr_reg, int addend_reg, int opc)
949 f54b3f92 aurel32
{
950 f54b3f92 aurel32
#ifdef TARGET_WORDS_BIGENDIAN
951 f061b40e Richard Henderson
    const int bswap = 0;
952 f54b3f92 aurel32
#else
953 f061b40e Richard Henderson
    const int bswap = 1;
954 f54b3f92 aurel32
#endif
955 f061b40e Richard Henderson
956 f54b3f92 aurel32
    switch (opc) {
957 fd76e73a Richard Henderson
    case 0:
958 f061b40e Richard Henderson
        tcg_out_ldst_index(s, datalo_reg, addr_reg, addend_reg, INSN_LDBX);
959 fd76e73a Richard Henderson
        break;
960 fd76e73a Richard Henderson
    case 0 | 4:
961 f061b40e Richard Henderson
        tcg_out_ldst_index(s, datalo_reg, addr_reg, addend_reg, INSN_LDBX);
962 f061b40e Richard Henderson
        tcg_out_ext8s(s, datalo_reg, datalo_reg);
963 fd76e73a Richard Henderson
        break;
964 fd76e73a Richard Henderson
    case 1:
965 f061b40e Richard Henderson
        tcg_out_ldst_index(s, datalo_reg, addr_reg, addend_reg, INSN_LDHX);
966 fd76e73a Richard Henderson
        if (bswap) {
967 f061b40e Richard Henderson
            tcg_out_bswap16(s, datalo_reg, datalo_reg, 0);
968 fd76e73a Richard Henderson
        }
969 fd76e73a Richard Henderson
        break;
970 fd76e73a Richard Henderson
    case 1 | 4:
971 f061b40e Richard Henderson
        tcg_out_ldst_index(s, datalo_reg, addr_reg, addend_reg, INSN_LDHX);
972 fd76e73a Richard Henderson
        if (bswap) {
973 f061b40e Richard Henderson
            tcg_out_bswap16(s, datalo_reg, datalo_reg, 1);
974 fd76e73a Richard Henderson
        } else {
975 f061b40e Richard Henderson
            tcg_out_ext16s(s, datalo_reg, datalo_reg);
976 fd76e73a Richard Henderson
        }
977 fd76e73a Richard Henderson
        break;
978 fd76e73a Richard Henderson
    case 2:
979 f061b40e Richard Henderson
        tcg_out_ldst_index(s, datalo_reg, addr_reg, addend_reg, INSN_LDWX);
980 fd76e73a Richard Henderson
        if (bswap) {
981 f061b40e Richard Henderson
            tcg_out_bswap32(s, datalo_reg, datalo_reg, TCG_REG_R20);
982 fd76e73a Richard Henderson
        }
983 fd76e73a Richard Henderson
        break;
984 fd76e73a Richard Henderson
    case 3:
985 fd76e73a Richard Henderson
        if (bswap) {
986 f061b40e Richard Henderson
            int t = datahi_reg;
987 f061b40e Richard Henderson
            datahi_reg = datalo_reg;
988 f061b40e Richard Henderson
            datalo_reg = t;
989 fd76e73a Richard Henderson
        }
990 f061b40e Richard Henderson
        /* We can't access the low-part with a reg+reg addressing mode,
991 f061b40e Richard Henderson
           so perform the addition now and use reg_ofs addressing mode.  */
992 f061b40e Richard Henderson
        if (addend_reg != TCG_REG_R0) {
993 f061b40e Richard Henderson
            tcg_out_arith(s, TCG_REG_R20, addr_reg, addend_reg, INSN_ADD);
994 f061b40e Richard Henderson
            addr_reg = TCG_REG_R20;
995 f061b40e Richard Henderson
        }
996 f061b40e Richard Henderson
        /* Make sure not to clobber the base register.  */
997 f061b40e Richard Henderson
        if (datahi_reg == addr_reg) {
998 f061b40e Richard Henderson
            tcg_out_ldst(s, datalo_reg, addr_reg, 4, INSN_LDW);
999 f061b40e Richard Henderson
            tcg_out_ldst(s, datahi_reg, addr_reg, 0, INSN_LDW);
1000 fd76e73a Richard Henderson
        } else {
1001 f061b40e Richard Henderson
            tcg_out_ldst(s, datahi_reg, addr_reg, 0, INSN_LDW);
1002 f061b40e Richard Henderson
            tcg_out_ldst(s, datalo_reg, addr_reg, 4, INSN_LDW);
1003 fd76e73a Richard Henderson
        }
1004 fd76e73a Richard Henderson
        if (bswap) {
1005 f061b40e Richard Henderson
            tcg_out_bswap32(s, datalo_reg, datalo_reg, TCG_REG_R20);
1006 f061b40e Richard Henderson
            tcg_out_bswap32(s, datahi_reg, datahi_reg, TCG_REG_R20);
1007 fd76e73a Richard Henderson
        }
1008 fd76e73a Richard Henderson
        break;
1009 fd76e73a Richard Henderson
    default:
1010 fd76e73a Richard Henderson
        tcg_abort();
1011 f54b3f92 aurel32
    }
1012 f061b40e Richard Henderson
}
1013 f061b40e Richard Henderson
1014 f061b40e Richard Henderson
static void tcg_out_qemu_ld(TCGContext *s, const TCGArg *args, int opc)
1015 f061b40e Richard Henderson
{
1016 f061b40e Richard Henderson
    int datalo_reg = *args++;
1017 f061b40e Richard Henderson
    /* Note that datahi_reg is only used for 64-bit loads.  */
1018 f061b40e Richard Henderson
    int datahi_reg = (opc == 3 ? *args++ : TCG_REG_R0);
1019 f061b40e Richard Henderson
    int addrlo_reg = *args++;
1020 f54b3f92 aurel32
1021 f54b3f92 aurel32
#if defined(CONFIG_SOFTMMU)
1022 f061b40e Richard Henderson
    /* Note that addrhi_reg is only used for 64-bit guests.  */
1023 f061b40e Richard Henderson
    int addrhi_reg = (TARGET_LONG_BITS == 64 ? *args++ : TCG_REG_R0);
1024 f061b40e Richard Henderson
    int mem_index = *args;
1025 f061b40e Richard Henderson
    int lab1, lab2, argreg, offset;
1026 f061b40e Richard Henderson
1027 f061b40e Richard Henderson
    lab1 = gen_new_label();
1028 f061b40e Richard Henderson
    lab2 = gen_new_label();
1029 f061b40e Richard Henderson
1030 f061b40e Richard Henderson
    offset = offsetof(CPUState, tlb_table[mem_index][0].addr_read);
1031 f061b40e Richard Henderson
    offset = tcg_out_tlb_read(s, TCG_REG_R26, TCG_REG_R25, addrlo_reg, addrhi_reg,
1032 f061b40e Richard Henderson
                              opc & 3, lab1, offset);
1033 f061b40e Richard Henderson
1034 f061b40e Richard Henderson
    /* TLB Hit.  */
1035 f061b40e Richard Henderson
    tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_R20, (offset ? TCG_REG_R1 : TCG_REG_R25),
1036 f061b40e Richard Henderson
               offsetof(CPUState, tlb_table[mem_index][0].addend) - offset);
1037 f061b40e Richard Henderson
    tcg_out_qemu_ld_direct(s, datalo_reg, datahi_reg, addrlo_reg, TCG_REG_R20, opc);
1038 fd76e73a Richard Henderson
    tcg_out_branch(s, lab2, 1);
1039 fd76e73a Richard Henderson
1040 fd76e73a Richard Henderson
    /* TLB Miss.  */
1041 fd76e73a Richard Henderson
    /* label1: */
1042 fd76e73a Richard Henderson
    tcg_out_label(s, lab1, (tcg_target_long)s->code_ptr);
1043 fd76e73a Richard Henderson
1044 fd76e73a Richard Henderson
    argreg = TCG_REG_R26;
1045 f061b40e Richard Henderson
    tcg_out_mov(s, argreg--, addrlo_reg);
1046 fd76e73a Richard Henderson
    if (TARGET_LONG_BITS == 64) {
1047 f061b40e Richard Henderson
        tcg_out_mov(s, argreg--, addrhi_reg);
1048 fd76e73a Richard Henderson
    }
1049 fd76e73a Richard Henderson
    tcg_out_movi(s, TCG_TYPE_I32, argreg, mem_index);
1050 fd76e73a Richard Henderson
1051 f061b40e Richard Henderson
    tcg_out_call(s, qemu_ld_helpers[opc & 3]);
1052 fd76e73a Richard Henderson
1053 fd76e73a Richard Henderson
    switch (opc) {
1054 fd76e73a Richard Henderson
    case 0:
1055 f061b40e Richard Henderson
        tcg_out_andi(s, datalo_reg, TCG_REG_RET0, 0xff);
1056 fd76e73a Richard Henderson
        break;
1057 fd76e73a Richard Henderson
    case 0 | 4:
1058 f061b40e Richard Henderson
        tcg_out_ext8s(s, datalo_reg, TCG_REG_RET0);
1059 fd76e73a Richard Henderson
        break;
1060 fd76e73a Richard Henderson
    case 1:
1061 f061b40e Richard Henderson
        tcg_out_andi(s, datalo_reg, TCG_REG_RET0, 0xffff);
1062 fd76e73a Richard Henderson
        break;
1063 fd76e73a Richard Henderson
    case 1 | 4:
1064 f061b40e Richard Henderson
        tcg_out_ext16s(s, datalo_reg, TCG_REG_RET0);
1065 fd76e73a Richard Henderson
        break;
1066 fd76e73a Richard Henderson
    case 2:
1067 fd76e73a Richard Henderson
    case 2 | 4:
1068 f061b40e Richard Henderson
        tcg_out_mov(s, datalo_reg, TCG_REG_RET0);
1069 fd76e73a Richard Henderson
        break;
1070 fd76e73a Richard Henderson
    case 3:
1071 f061b40e Richard Henderson
        tcg_out_mov(s, datahi_reg, TCG_REG_RET0);
1072 f061b40e Richard Henderson
        tcg_out_mov(s, datalo_reg, TCG_REG_RET1);
1073 fd76e73a Richard Henderson
        break;
1074 fd76e73a Richard Henderson
    default:
1075 fd76e73a Richard Henderson
        tcg_abort();
1076 fd76e73a Richard Henderson
    }
1077 fd76e73a Richard Henderson
1078 f54b3f92 aurel32
    /* label2: */
1079 fd76e73a Richard Henderson
    tcg_out_label(s, lab2, (tcg_target_long)s->code_ptr);
1080 f061b40e Richard Henderson
#else
1081 f061b40e Richard Henderson
    tcg_out_qemu_ld_direct(s, datalo_reg, datahi_reg, addrlo_reg,
1082 f061b40e Richard Henderson
                           (GUEST_BASE ? TCG_GUEST_BASE_REG : TCG_REG_R0), opc);
1083 f54b3f92 aurel32
#endif
1084 f54b3f92 aurel32
}
1085 f54b3f92 aurel32
1086 f061b40e Richard Henderson
static void tcg_out_qemu_st_direct(TCGContext *s, int datalo_reg, int datahi_reg,
1087 f061b40e Richard Henderson
                                   int addr_reg, int opc)
1088 f54b3f92 aurel32
{
1089 f54b3f92 aurel32
#ifdef TARGET_WORDS_BIGENDIAN
1090 f061b40e Richard Henderson
    const int bswap = 0;
1091 f54b3f92 aurel32
#else
1092 f061b40e Richard Henderson
    const int bswap = 1;
1093 f54b3f92 aurel32
#endif
1094 f061b40e Richard Henderson
1095 f54b3f92 aurel32
    switch (opc) {
1096 f54b3f92 aurel32
    case 0:
1097 f061b40e Richard Henderson
        tcg_out_ldst(s, datalo_reg, addr_reg, 0, INSN_STB);
1098 f54b3f92 aurel32
        break;
1099 f54b3f92 aurel32
    case 1:
1100 f54b3f92 aurel32
        if (bswap) {
1101 f061b40e Richard Henderson
            tcg_out_bswap16(s, TCG_REG_R20, datalo_reg, 0);
1102 f061b40e Richard Henderson
            datalo_reg = TCG_REG_R20;
1103 f54b3f92 aurel32
        }
1104 f061b40e Richard Henderson
        tcg_out_ldst(s, datalo_reg, addr_reg, 0, INSN_STH);
1105 f54b3f92 aurel32
        break;
1106 f54b3f92 aurel32
    case 2:
1107 f54b3f92 aurel32
        if (bswap) {
1108 f061b40e Richard Henderson
            tcg_out_bswap32(s, TCG_REG_R20, datalo_reg, TCG_REG_R20);
1109 f061b40e Richard Henderson
            datalo_reg = TCG_REG_R20;
1110 f54b3f92 aurel32
        }
1111 f061b40e Richard Henderson
        tcg_out_ldst(s, datalo_reg, addr_reg, 0, INSN_STW);
1112 f54b3f92 aurel32
        break;
1113 f54b3f92 aurel32
    case 3:
1114 fd76e73a Richard Henderson
        if (bswap) {
1115 f061b40e Richard Henderson
            tcg_out_bswap32(s, TCG_REG_R20, datalo_reg, TCG_REG_R20);
1116 f061b40e Richard Henderson
            tcg_out_bswap32(s, TCG_REG_R23, datahi_reg, TCG_REG_R23);
1117 f061b40e Richard Henderson
            datahi_reg = TCG_REG_R20;
1118 f061b40e Richard Henderson
            datalo_reg = TCG_REG_R23;
1119 f54b3f92 aurel32
        }
1120 f061b40e Richard Henderson
        tcg_out_ldst(s, datahi_reg, addr_reg, 0, INSN_STW);
1121 f061b40e Richard Henderson
        tcg_out_ldst(s, datalo_reg, addr_reg, 4, INSN_STW);
1122 f54b3f92 aurel32
        break;
1123 f54b3f92 aurel32
    default:
1124 f54b3f92 aurel32
        tcg_abort();
1125 f54b3f92 aurel32
    }
1126 f54b3f92 aurel32
1127 f061b40e Richard Henderson
}
1128 f061b40e Richard Henderson
1129 f061b40e Richard Henderson
static void tcg_out_qemu_st(TCGContext *s, const TCGArg *args, int opc)
1130 f061b40e Richard Henderson
{
1131 f061b40e Richard Henderson
    int datalo_reg = *args++;
1132 f061b40e Richard Henderson
    /* Note that datahi_reg is only used for 64-bit loads.  */
1133 f061b40e Richard Henderson
    int datahi_reg = (opc == 3 ? *args++ : TCG_REG_R0);
1134 f061b40e Richard Henderson
    int addrlo_reg = *args++;
1135 f061b40e Richard Henderson
1136 f54b3f92 aurel32
#if defined(CONFIG_SOFTMMU)
1137 f061b40e Richard Henderson
    /* Note that addrhi_reg is only used for 64-bit guests.  */
1138 f061b40e Richard Henderson
    int addrhi_reg = (TARGET_LONG_BITS == 64 ? *args++ : TCG_REG_R0);
1139 f061b40e Richard Henderson
    int mem_index = *args;
1140 f061b40e Richard Henderson
    int lab1, lab2, argreg, offset;
1141 f061b40e Richard Henderson
1142 f061b40e Richard Henderson
    lab1 = gen_new_label();
1143 f061b40e Richard Henderson
    lab2 = gen_new_label();
1144 f061b40e Richard Henderson
1145 f061b40e Richard Henderson
    offset = offsetof(CPUState, tlb_table[mem_index][0].addr_write);
1146 f061b40e Richard Henderson
    offset = tcg_out_tlb_read(s, TCG_REG_R26, TCG_REG_R25, addrlo_reg, addrhi_reg,
1147 f061b40e Richard Henderson
                              opc, lab1, offset);
1148 f061b40e Richard Henderson
1149 f061b40e Richard Henderson
    /* TLB Hit.  */
1150 f061b40e Richard Henderson
    tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_R20, (offset ? TCG_REG_R1 : TCG_REG_R25),
1151 f061b40e Richard Henderson
               offsetof(CPUState, tlb_table[mem_index][0].addend) - offset);
1152 f061b40e Richard Henderson
1153 f061b40e Richard Henderson
    /* There are no indexed stores, so we must do this addition explitly.
1154 f061b40e Richard Henderson
       Careful to avoid R20, which is used for the bswaps to follow.  */
1155 f061b40e Richard Henderson
    tcg_out_arith(s, TCG_REG_R31, addrlo_reg, TCG_REG_R20, INSN_ADDL);
1156 f061b40e Richard Henderson
    tcg_out_qemu_st_direct(s, datalo_reg, datahi_reg, TCG_REG_R31, opc);
1157 fd76e73a Richard Henderson
    tcg_out_branch(s, lab2, 1);
1158 fd76e73a Richard Henderson
1159 fd76e73a Richard Henderson
    /* TLB Miss.  */
1160 fd76e73a Richard Henderson
    /* label1: */
1161 fd76e73a Richard Henderson
    tcg_out_label(s, lab1, (tcg_target_long)s->code_ptr);
1162 fd76e73a Richard Henderson
1163 fd76e73a Richard Henderson
    argreg = TCG_REG_R26;
1164 f061b40e Richard Henderson
    tcg_out_mov(s, argreg--, addrlo_reg);
1165 fd76e73a Richard Henderson
    if (TARGET_LONG_BITS == 64) {
1166 f061b40e Richard Henderson
        tcg_out_mov(s, argreg--, addrhi_reg);
1167 fd76e73a Richard Henderson
    }
1168 fd76e73a Richard Henderson
1169 fd76e73a Richard Henderson
    switch(opc) {
1170 fd76e73a Richard Henderson
    case 0:
1171 f061b40e Richard Henderson
        tcg_out_andi(s, argreg--, datalo_reg, 0xff);
1172 fd76e73a Richard Henderson
        tcg_out_movi(s, TCG_TYPE_I32, argreg, mem_index);
1173 fd76e73a Richard Henderson
        break;
1174 fd76e73a Richard Henderson
    case 1:
1175 f061b40e Richard Henderson
        tcg_out_andi(s, argreg--, datalo_reg, 0xffff);
1176 fd76e73a Richard Henderson
        tcg_out_movi(s, TCG_TYPE_I32, argreg, mem_index);
1177 fd76e73a Richard Henderson
        break;
1178 fd76e73a Richard Henderson
    case 2:
1179 f061b40e Richard Henderson
        tcg_out_mov(s, argreg--, datalo_reg);
1180 fd76e73a Richard Henderson
        tcg_out_movi(s, TCG_TYPE_I32, argreg, mem_index);
1181 fd76e73a Richard Henderson
        break;
1182 fd76e73a Richard Henderson
    case 3:
1183 fd76e73a Richard Henderson
        /* Because of the alignment required by the 64-bit data argument,
1184 fd76e73a Richard Henderson
           we will always use R23/R24.  Also, we will always run out of
1185 fd76e73a Richard Henderson
           argument registers for storing mem_index, so that will have
1186 fd76e73a Richard Henderson
           to go on the stack.  */
1187 fd76e73a Richard Henderson
        if (mem_index == 0) {
1188 fd76e73a Richard Henderson
            argreg = TCG_REG_R0;
1189 fd76e73a Richard Henderson
        } else {
1190 fd76e73a Richard Henderson
            argreg = TCG_REG_R20;
1191 fd76e73a Richard Henderson
            tcg_out_movi(s, TCG_TYPE_I32, argreg, mem_index);
1192 fd76e73a Richard Henderson
        }
1193 f061b40e Richard Henderson
        tcg_out_mov(s, TCG_REG_R23, datahi_reg);
1194 f061b40e Richard Henderson
        tcg_out_mov(s, TCG_REG_R24, datalo_reg);
1195 fd76e73a Richard Henderson
        tcg_out_st(s, TCG_TYPE_I32, argreg, TCG_REG_SP,
1196 fd76e73a Richard Henderson
                   TCG_TARGET_CALL_STACK_OFFSET - 4);
1197 fd76e73a Richard Henderson
        break;
1198 fd76e73a Richard Henderson
    default:
1199 fd76e73a Richard Henderson
        tcg_abort();
1200 fd76e73a Richard Henderson
    }
1201 fd76e73a Richard Henderson
1202 f061b40e Richard Henderson
    tcg_out_call(s, qemu_st_helpers[opc]);
1203 fd76e73a Richard Henderson
1204 f54b3f92 aurel32
    /* label2: */
1205 fd76e73a Richard Henderson
    tcg_out_label(s, lab2, (tcg_target_long)s->code_ptr);
1206 f061b40e Richard Henderson
#else
1207 f061b40e Richard Henderson
    /* There are no indexed stores, so if GUEST_BASE is set we must do the add
1208 f061b40e Richard Henderson
       explicitly.  Careful to avoid R20, which is used for the bswaps to follow.  */
1209 f061b40e Richard Henderson
    if (GUEST_BASE != 0) {
1210 f061b40e Richard Henderson
        tcg_out_arith(s, TCG_REG_R31, addrlo_reg, TCG_GUEST_BASE_REG, INSN_ADDL);
1211 f061b40e Richard Henderson
        addrlo_reg = TCG_REG_R31;
1212 f061b40e Richard Henderson
    }
1213 f061b40e Richard Henderson
    tcg_out_qemu_st_direct(s, datalo_reg, datahi_reg, addrlo_reg, opc);
1214 f54b3f92 aurel32
#endif
1215 f54b3f92 aurel32
}
1216 f54b3f92 aurel32
1217 fd76e73a Richard Henderson
static void tcg_out_exit_tb(TCGContext *s, TCGArg arg)
1218 fd76e73a Richard Henderson
{
1219 fd76e73a Richard Henderson
    if (!check_fit_tl(arg, 14)) {
1220 fd76e73a Richard Henderson
        uint32_t hi, lo;
1221 fd76e73a Richard Henderson
        hi = arg & ~0x7ff;
1222 fd76e73a Richard Henderson
        lo = arg & 0x7ff;
1223 fd76e73a Richard Henderson
        if (lo) {
1224 fd76e73a Richard Henderson
            tcg_out_movi(s, TCG_TYPE_PTR, TCG_REG_RET0, hi);
1225 fd76e73a Richard Henderson
            tcg_out32(s, INSN_BV | INSN_R2(TCG_REG_R18));
1226 fd76e73a Richard Henderson
            tcg_out_addi(s, TCG_REG_RET0, lo);
1227 fd76e73a Richard Henderson
            return;
1228 fd76e73a Richard Henderson
        }
1229 fd76e73a Richard Henderson
        arg = hi;
1230 fd76e73a Richard Henderson
    }
1231 fd76e73a Richard Henderson
    tcg_out32(s, INSN_BV | INSN_R2(TCG_REG_R18));
1232 fd76e73a Richard Henderson
    tcg_out_movi(s, TCG_TYPE_PTR, TCG_REG_RET0, arg);
1233 fd76e73a Richard Henderson
}
1234 fd76e73a Richard Henderson
1235 fd76e73a Richard Henderson
static void tcg_out_goto_tb(TCGContext *s, TCGArg arg)
1236 fd76e73a Richard Henderson
{
1237 fd76e73a Richard Henderson
    if (s->tb_jmp_offset) {
1238 fd76e73a Richard Henderson
        /* direct jump method */
1239 fd76e73a Richard Henderson
        fprintf(stderr, "goto_tb direct\n");
1240 fd76e73a Richard Henderson
        tcg_abort();
1241 fd76e73a Richard Henderson
    } else {
1242 fd76e73a Richard Henderson
        /* indirect jump method */
1243 fd76e73a Richard Henderson
        tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_R20, TCG_REG_R0,
1244 fd76e73a Richard Henderson
                   (tcg_target_long)(s->tb_next + arg));
1245 fd76e73a Richard Henderson
        tcg_out32(s, INSN_BV_N | INSN_R2(TCG_REG_R20));
1246 fd76e73a Richard Henderson
    }
1247 fd76e73a Richard Henderson
    s->tb_next_offset[arg] = s->code_ptr - s->code_buf;
1248 fd76e73a Richard Henderson
}
1249 fd76e73a Richard Henderson
1250 a9751609 Richard Henderson
static inline void tcg_out_op(TCGContext *s, TCGOpcode opc, const TCGArg *args,
1251 f54b3f92 aurel32
                              const int *const_args)
1252 f54b3f92 aurel32
{
1253 f54b3f92 aurel32
    switch (opc) {
1254 f54b3f92 aurel32
    case INDEX_op_exit_tb:
1255 fd76e73a Richard Henderson
        tcg_out_exit_tb(s, args[0]);
1256 f54b3f92 aurel32
        break;
1257 f54b3f92 aurel32
    case INDEX_op_goto_tb:
1258 fd76e73a Richard Henderson
        tcg_out_goto_tb(s, args[0]);
1259 f54b3f92 aurel32
        break;
1260 fd76e73a Richard Henderson
1261 f54b3f92 aurel32
    case INDEX_op_call:
1262 fd76e73a Richard Henderson
        if (const_args[0]) {
1263 fd76e73a Richard Henderson
            tcg_out_call(s, (void *)args[0]);
1264 fd76e73a Richard Henderson
        } else {
1265 3e1f46ea Richard Henderson
            /* ??? FIXME: the value in the register in args[0] is almost
1266 3e1f46ea Richard Henderson
               certainly a procedure descriptor, not a code address.  We
1267 3e1f46ea Richard Henderson
               probably need to use the millicode $$dyncall routine.  */
1268 3e1f46ea Richard Henderson
            tcg_abort();
1269 fd76e73a Richard Henderson
        }
1270 f54b3f92 aurel32
        break;
1271 fd76e73a Richard Henderson
1272 f54b3f92 aurel32
    case INDEX_op_jmp:
1273 f54b3f92 aurel32
        fprintf(stderr, "unimplemented jmp\n");
1274 f54b3f92 aurel32
        tcg_abort();
1275 f54b3f92 aurel32
        break;
1276 fd76e73a Richard Henderson
1277 f54b3f92 aurel32
    case INDEX_op_br:
1278 fd76e73a Richard Henderson
        tcg_out_branch(s, args[0], 1);
1279 f54b3f92 aurel32
        break;
1280 fd76e73a Richard Henderson
1281 f54b3f92 aurel32
    case INDEX_op_movi_i32:
1282 f54b3f92 aurel32
        tcg_out_movi(s, TCG_TYPE_I32, args[0], (uint32_t)args[1]);
1283 f54b3f92 aurel32
        break;
1284 f54b3f92 aurel32
1285 f54b3f92 aurel32
    case INDEX_op_ld8u_i32:
1286 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_LDB);
1287 f54b3f92 aurel32
        break;
1288 f54b3f92 aurel32
    case INDEX_op_ld8s_i32:
1289 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_LDB);
1290 f54b3f92 aurel32
        tcg_out_ext8s(s, args[0], args[0]);
1291 f54b3f92 aurel32
        break;
1292 f54b3f92 aurel32
    case INDEX_op_ld16u_i32:
1293 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_LDH);
1294 f54b3f92 aurel32
        break;
1295 f54b3f92 aurel32
    case INDEX_op_ld16s_i32:
1296 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_LDH);
1297 f54b3f92 aurel32
        tcg_out_ext16s(s, args[0], args[0]);
1298 f54b3f92 aurel32
        break;
1299 f54b3f92 aurel32
    case INDEX_op_ld_i32:
1300 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_LDW);
1301 f54b3f92 aurel32
        break;
1302 f54b3f92 aurel32
1303 f54b3f92 aurel32
    case INDEX_op_st8_i32:
1304 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_STB);
1305 f54b3f92 aurel32
        break;
1306 f54b3f92 aurel32
    case INDEX_op_st16_i32:
1307 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_STH);
1308 f54b3f92 aurel32
        break;
1309 f54b3f92 aurel32
    case INDEX_op_st_i32:
1310 fd76e73a Richard Henderson
        tcg_out_ldst(s, args[0], args[1], args[2], INSN_STW);
1311 fd76e73a Richard Henderson
        break;
1312 fd76e73a Richard Henderson
1313 fd76e73a Richard Henderson
    case INDEX_op_add_i32:
1314 fd76e73a Richard Henderson
        if (const_args[2]) {
1315 fd76e73a Richard Henderson
            tcg_out_addi2(s, args[0], args[1], args[2]);
1316 fd76e73a Richard Henderson
        } else {
1317 fd76e73a Richard Henderson
            tcg_out_arith(s, args[0], args[1], args[2], INSN_ADDL);
1318 fd76e73a Richard Henderson
        }
1319 f54b3f92 aurel32
        break;
1320 f54b3f92 aurel32
1321 f54b3f92 aurel32
    case INDEX_op_sub_i32:
1322 fd76e73a Richard Henderson
        if (const_args[1]) {
1323 fd76e73a Richard Henderson
            if (const_args[2]) {
1324 fd76e73a Richard Henderson
                tcg_out_movi(s, TCG_TYPE_I32, args[0], args[1] - args[2]);
1325 fd76e73a Richard Henderson
            } else {
1326 fd76e73a Richard Henderson
                /* Recall that SUBI is a reversed subtract.  */
1327 fd76e73a Richard Henderson
                tcg_out_arithi(s, args[0], args[2], args[1], INSN_SUBI);
1328 fd76e73a Richard Henderson
            }
1329 fd76e73a Richard Henderson
        } else if (const_args[2]) {
1330 fd76e73a Richard Henderson
            tcg_out_addi2(s, args[0], args[1], -args[2]);
1331 fd76e73a Richard Henderson
        } else {
1332 fd76e73a Richard Henderson
            tcg_out_arith(s, args[0], args[1], args[2], INSN_SUB);
1333 fd76e73a Richard Henderson
        }
1334 fd76e73a Richard Henderson
        break;
1335 fd76e73a Richard Henderson
1336 f54b3f92 aurel32
    case INDEX_op_and_i32:
1337 fd76e73a Richard Henderson
        if (const_args[2]) {
1338 fd76e73a Richard Henderson
            tcg_out_andi(s, args[0], args[1], args[2]);
1339 fd76e73a Richard Henderson
        } else {
1340 fd76e73a Richard Henderson
            tcg_out_arith(s, args[0], args[1], args[2], INSN_AND);
1341 fd76e73a Richard Henderson
        }
1342 fd76e73a Richard Henderson
        break;
1343 fd76e73a Richard Henderson
1344 f54b3f92 aurel32
    case INDEX_op_or_i32:
1345 fd76e73a Richard Henderson
        if (const_args[2]) {
1346 fd76e73a Richard Henderson
            tcg_out_ori(s, args[0], args[1], args[2]);
1347 fd76e73a Richard Henderson
        } else {
1348 fd76e73a Richard Henderson
            tcg_out_arith(s, args[0], args[1], args[2], INSN_OR);
1349 fd76e73a Richard Henderson
        }
1350 fd76e73a Richard Henderson
        break;
1351 fd76e73a Richard Henderson
1352 f54b3f92 aurel32
    case INDEX_op_xor_i32:
1353 fd76e73a Richard Henderson
        tcg_out_arith(s, args[0], args[1], args[2], INSN_XOR);
1354 fd76e73a Richard Henderson
        break;
1355 fd76e73a Richard Henderson
1356 fd76e73a Richard Henderson
    case INDEX_op_andc_i32:
1357 fd76e73a Richard Henderson
        if (const_args[2]) {
1358 fd76e73a Richard Henderson
            tcg_out_andi(s, args[0], args[1], ~args[2]);
1359 fd76e73a Richard Henderson
        } else {
1360 fd76e73a Richard Henderson
            tcg_out_arith(s, args[0], args[1], args[2], INSN_ANDCM);
1361 fd76e73a Richard Henderson
        }
1362 fd76e73a Richard Henderson
        break;
1363 f54b3f92 aurel32
1364 f54b3f92 aurel32
    case INDEX_op_shl_i32:
1365 fd76e73a Richard Henderson
        if (const_args[2]) {
1366 fd76e73a Richard Henderson
            tcg_out_shli(s, args[0], args[1], args[2]);
1367 fd76e73a Richard Henderson
        } else {
1368 fd76e73a Richard Henderson
            tcg_out_shl(s, args[0], args[1], args[2]);
1369 fd76e73a Richard Henderson
        }
1370 f54b3f92 aurel32
        break;
1371 fd76e73a Richard Henderson
1372 f54b3f92 aurel32
    case INDEX_op_shr_i32:
1373 fd76e73a Richard Henderson
        if (const_args[2]) {
1374 fd76e73a Richard Henderson
            tcg_out_shri(s, args[0], args[1], args[2]);
1375 fd76e73a Richard Henderson
        } else {
1376 fd76e73a Richard Henderson
            tcg_out_shr(s, args[0], args[1], args[2]);
1377 fd76e73a Richard Henderson
        }
1378 f54b3f92 aurel32
        break;
1379 fd76e73a Richard Henderson
1380 f54b3f92 aurel32
    case INDEX_op_sar_i32:
1381 fd76e73a Richard Henderson
        if (const_args[2]) {
1382 fd76e73a Richard Henderson
            tcg_out_sari(s, args[0], args[1], args[2]);
1383 fd76e73a Richard Henderson
        } else {
1384 fd76e73a Richard Henderson
            tcg_out_sar(s, args[0], args[1], args[2]);
1385 fd76e73a Richard Henderson
        }
1386 fd76e73a Richard Henderson
        break;
1387 fd76e73a Richard Henderson
1388 fd76e73a Richard Henderson
    case INDEX_op_rotl_i32:
1389 fd76e73a Richard Henderson
        if (const_args[2]) {
1390 fd76e73a Richard Henderson
            tcg_out_rotli(s, args[0], args[1], args[2]);
1391 fd76e73a Richard Henderson
        } else {
1392 fd76e73a Richard Henderson
            tcg_out_rotl(s, args[0], args[1], args[2]);
1393 fd76e73a Richard Henderson
        }
1394 fd76e73a Richard Henderson
        break;
1395 fd76e73a Richard Henderson
1396 fd76e73a Richard Henderson
    case INDEX_op_rotr_i32:
1397 fd76e73a Richard Henderson
        if (const_args[2]) {
1398 fd76e73a Richard Henderson
            tcg_out_rotri(s, args[0], args[1], args[2]);
1399 fd76e73a Richard Henderson
        } else {
1400 fd76e73a Richard Henderson
            tcg_out_rotr(s, args[0], args[1], args[2]);
1401 fd76e73a Richard Henderson
        }
1402 f54b3f92 aurel32
        break;
1403 f54b3f92 aurel32
1404 f54b3f92 aurel32
    case INDEX_op_mul_i32:
1405 fd76e73a Richard Henderson
        tcg_out_xmpyu(s, args[0], TCG_REG_R0, args[1], args[2]);
1406 f54b3f92 aurel32
        break;
1407 f54b3f92 aurel32
    case INDEX_op_mulu2_i32:
1408 fd76e73a Richard Henderson
        tcg_out_xmpyu(s, args[0], args[1], args[2], args[3]);
1409 f54b3f92 aurel32
        break;
1410 fd76e73a Richard Henderson
1411 fd76e73a Richard Henderson
    case INDEX_op_bswap16_i32:
1412 fd76e73a Richard Henderson
        tcg_out_bswap16(s, args[0], args[1], 0);
1413 f54b3f92 aurel32
        break;
1414 fd76e73a Richard Henderson
    case INDEX_op_bswap32_i32:
1415 fd76e73a Richard Henderson
        tcg_out_bswap32(s, args[0], args[1], TCG_REG_R20);
1416 fd76e73a Richard Henderson
        break;
1417 fd76e73a Richard Henderson
1418 fd76e73a Richard Henderson
    case INDEX_op_not_i32:
1419 fd76e73a Richard Henderson
        tcg_out_arithi(s, args[0], args[1], -1, INSN_SUBI);
1420 fd76e73a Richard Henderson
        break;
1421 fd76e73a Richard Henderson
    case INDEX_op_ext8s_i32:
1422 fd76e73a Richard Henderson
        tcg_out_ext8s(s, args[0], args[1]);
1423 fd76e73a Richard Henderson
        break;
1424 fd76e73a Richard Henderson
    case INDEX_op_ext16s_i32:
1425 fd76e73a Richard Henderson
        tcg_out_ext16s(s, args[0], args[1]);
1426 fd76e73a Richard Henderson
        break;
1427 fd76e73a Richard Henderson
1428 fd76e73a Richard Henderson
    /* These three correspond exactly to the fallback implementation.
1429 fd76e73a Richard Henderson
       But by including them we reduce the number of TCG ops that
1430 fd76e73a Richard Henderson
       need to be generated, and these opcodes are fairly common.  */
1431 fd76e73a Richard Henderson
    case INDEX_op_neg_i32:
1432 fd76e73a Richard Henderson
        tcg_out_arith(s, args[0], TCG_REG_R0, args[1], INSN_SUB);
1433 fd76e73a Richard Henderson
        break;
1434 fd76e73a Richard Henderson
    case INDEX_op_ext8u_i32:
1435 fd76e73a Richard Henderson
        tcg_out_andi(s, args[0], args[1], 0xff);
1436 fd76e73a Richard Henderson
        break;
1437 fd76e73a Richard Henderson
    case INDEX_op_ext16u_i32:
1438 fd76e73a Richard Henderson
        tcg_out_andi(s, args[0], args[1], 0xffff);
1439 f54b3f92 aurel32
        break;
1440 f54b3f92 aurel32
1441 f54b3f92 aurel32
    case INDEX_op_brcond_i32:
1442 fd76e73a Richard Henderson
        tcg_out_brcond(s, args[2], args[0], args[1], const_args[1], args[3]);
1443 fd76e73a Richard Henderson
        break;
1444 fd76e73a Richard Henderson
    case INDEX_op_brcond2_i32:
1445 fd76e73a Richard Henderson
        tcg_out_brcond2(s, args[4], args[0], args[1],
1446 fd76e73a Richard Henderson
                        args[2], const_args[2],
1447 fd76e73a Richard Henderson
                        args[3], const_args[3], args[5]);
1448 fd76e73a Richard Henderson
        break;
1449 fd76e73a Richard Henderson
1450 fd76e73a Richard Henderson
    case INDEX_op_setcond_i32:
1451 fd76e73a Richard Henderson
        tcg_out_setcond(s, args[3], args[0], args[1], args[2], const_args[2]);
1452 fd76e73a Richard Henderson
        break;
1453 fd76e73a Richard Henderson
    case INDEX_op_setcond2_i32:
1454 fd76e73a Richard Henderson
        tcg_out_setcond2(s, args[5], args[0], args[1], args[2],
1455 fd76e73a Richard Henderson
                         args[3], const_args[3], args[4], const_args[4]);
1456 fd76e73a Richard Henderson
        break;
1457 fd76e73a Richard Henderson
1458 fd76e73a Richard Henderson
    case INDEX_op_add2_i32:
1459 91493631 Richard Henderson
        tcg_out_add2(s, args[0], args[1], args[2], args[3],
1460 91493631 Richard Henderson
                     args[4], args[5], const_args[4]);
1461 fd76e73a Richard Henderson
        break;
1462 fd76e73a Richard Henderson
1463 fd76e73a Richard Henderson
    case INDEX_op_sub2_i32:
1464 91493631 Richard Henderson
        tcg_out_sub2(s, args[0], args[1], args[2], args[3],
1465 91493631 Richard Henderson
                     args[4], args[5], const_args[2], const_args[4]);
1466 f54b3f92 aurel32
        break;
1467 f54b3f92 aurel32
1468 f54b3f92 aurel32
    case INDEX_op_qemu_ld8u:
1469 f54b3f92 aurel32
        tcg_out_qemu_ld(s, args, 0);
1470 f54b3f92 aurel32
        break;
1471 f54b3f92 aurel32
    case INDEX_op_qemu_ld8s:
1472 f54b3f92 aurel32
        tcg_out_qemu_ld(s, args, 0 | 4);
1473 f54b3f92 aurel32
        break;
1474 f54b3f92 aurel32
    case INDEX_op_qemu_ld16u:
1475 f54b3f92 aurel32
        tcg_out_qemu_ld(s, args, 1);
1476 f54b3f92 aurel32
        break;
1477 f54b3f92 aurel32
    case INDEX_op_qemu_ld16s:
1478 f54b3f92 aurel32
        tcg_out_qemu_ld(s, args, 1 | 4);
1479 f54b3f92 aurel32
        break;
1480 86feb1c8 Richard Henderson
    case INDEX_op_qemu_ld32:
1481 f54b3f92 aurel32
        tcg_out_qemu_ld(s, args, 2);
1482 f54b3f92 aurel32
        break;
1483 fd76e73a Richard Henderson
    case INDEX_op_qemu_ld64:
1484 fd76e73a Richard Henderson
        tcg_out_qemu_ld(s, args, 3);
1485 fd76e73a Richard Henderson
        break;
1486 f54b3f92 aurel32
1487 f54b3f92 aurel32
    case INDEX_op_qemu_st8:
1488 f54b3f92 aurel32
        tcg_out_qemu_st(s, args, 0);
1489 f54b3f92 aurel32
        break;
1490 f54b3f92 aurel32
    case INDEX_op_qemu_st16:
1491 f54b3f92 aurel32
        tcg_out_qemu_st(s, args, 1);
1492 f54b3f92 aurel32
        break;
1493 f54b3f92 aurel32
    case INDEX_op_qemu_st32:
1494 f54b3f92 aurel32
        tcg_out_qemu_st(s, args, 2);
1495 f54b3f92 aurel32
        break;
1496 fd76e73a Richard Henderson
    case INDEX_op_qemu_st64:
1497 fd76e73a Richard Henderson
        tcg_out_qemu_st(s, args, 3);
1498 fd76e73a Richard Henderson
        break;
1499 f54b3f92 aurel32
1500 f54b3f92 aurel32
    default:
1501 f54b3f92 aurel32
        fprintf(stderr, "unknown opcode 0x%x\n", opc);
1502 f54b3f92 aurel32
        tcg_abort();
1503 f54b3f92 aurel32
    }
1504 f54b3f92 aurel32
}
1505 f54b3f92 aurel32
1506 f54b3f92 aurel32
static const TCGTargetOpDef hppa_op_defs[] = {
1507 f54b3f92 aurel32
    { INDEX_op_exit_tb, { } },
1508 f54b3f92 aurel32
    { INDEX_op_goto_tb, { } },
1509 f54b3f92 aurel32
1510 fd76e73a Richard Henderson
    { INDEX_op_call, { "ri" } },
1511 f54b3f92 aurel32
    { INDEX_op_jmp, { "r" } },
1512 f54b3f92 aurel32
    { INDEX_op_br, { } },
1513 f54b3f92 aurel32
1514 f54b3f92 aurel32
    { INDEX_op_mov_i32, { "r", "r" } },
1515 f54b3f92 aurel32
    { INDEX_op_movi_i32, { "r" } },
1516 fd76e73a Richard Henderson
1517 f54b3f92 aurel32
    { INDEX_op_ld8u_i32, { "r", "r" } },
1518 f54b3f92 aurel32
    { INDEX_op_ld8s_i32, { "r", "r" } },
1519 f54b3f92 aurel32
    { INDEX_op_ld16u_i32, { "r", "r" } },
1520 f54b3f92 aurel32
    { INDEX_op_ld16s_i32, { "r", "r" } },
1521 f54b3f92 aurel32
    { INDEX_op_ld_i32, { "r", "r" } },
1522 fd76e73a Richard Henderson
    { INDEX_op_st8_i32, { "rZ", "r" } },
1523 fd76e73a Richard Henderson
    { INDEX_op_st16_i32, { "rZ", "r" } },
1524 fd76e73a Richard Henderson
    { INDEX_op_st_i32, { "rZ", "r" } },
1525 fd76e73a Richard Henderson
1526 fd76e73a Richard Henderson
    { INDEX_op_add_i32, { "r", "rZ", "ri" } },
1527 fd76e73a Richard Henderson
    { INDEX_op_sub_i32, { "r", "rI", "ri" } },
1528 0085bd51 Richard Henderson
    { INDEX_op_and_i32, { "r", "rZ", "rM" } },
1529 0085bd51 Richard Henderson
    { INDEX_op_or_i32, { "r", "rZ", "rO" } },
1530 fd76e73a Richard Henderson
    { INDEX_op_xor_i32, { "r", "rZ", "rZ" } },
1531 0085bd51 Richard Henderson
    /* Note that the second argument will be inverted, which means
1532 0085bd51 Richard Henderson
       we want a constant whose inversion matches M, and that O = ~M.
1533 0085bd51 Richard Henderson
       See the implementation of and_mask_p.  */
1534 0085bd51 Richard Henderson
    { INDEX_op_andc_i32, { "r", "rZ", "rO" } },
1535 fd76e73a Richard Henderson
1536 fd76e73a Richard Henderson
    { INDEX_op_mul_i32, { "r", "r", "r" } },
1537 fd76e73a Richard Henderson
    { INDEX_op_mulu2_i32, { "r", "r", "r", "r" } },
1538 f54b3f92 aurel32
1539 fd76e73a Richard Henderson
    { INDEX_op_shl_i32, { "r", "r", "ri" } },
1540 fd76e73a Richard Henderson
    { INDEX_op_shr_i32, { "r", "r", "ri" } },
1541 fd76e73a Richard Henderson
    { INDEX_op_sar_i32, { "r", "r", "ri" } },
1542 fd76e73a Richard Henderson
    { INDEX_op_rotl_i32, { "r", "r", "ri" } },
1543 fd76e73a Richard Henderson
    { INDEX_op_rotr_i32, { "r", "r", "ri" } },
1544 f54b3f92 aurel32
1545 fd76e73a Richard Henderson
    { INDEX_op_bswap16_i32, { "r", "r" } },
1546 fd76e73a Richard Henderson
    { INDEX_op_bswap32_i32, { "r", "r" } },
1547 fd76e73a Richard Henderson
    { INDEX_op_neg_i32, { "r", "r" } },
1548 fd76e73a Richard Henderson
    { INDEX_op_not_i32, { "r", "r" } },
1549 f54b3f92 aurel32
1550 fd76e73a Richard Henderson
    { INDEX_op_ext8s_i32, { "r", "r" } },
1551 fd76e73a Richard Henderson
    { INDEX_op_ext8u_i32, { "r", "r" } },
1552 fd76e73a Richard Henderson
    { INDEX_op_ext16s_i32, { "r", "r" } },
1553 fd76e73a Richard Henderson
    { INDEX_op_ext16u_i32, { "r", "r" } },
1554 fd76e73a Richard Henderson
1555 fd76e73a Richard Henderson
    { INDEX_op_brcond_i32, { "rZ", "rJ" } },
1556 fd76e73a Richard Henderson
    { INDEX_op_brcond2_i32,  { "rZ", "rZ", "rJ", "rJ" } },
1557 fd76e73a Richard Henderson
1558 fd76e73a Richard Henderson
    { INDEX_op_setcond_i32, { "r", "rZ", "rI" } },
1559 fd76e73a Richard Henderson
    { INDEX_op_setcond2_i32, { "r", "rZ", "rZ", "rI", "rI" } },
1560 fd76e73a Richard Henderson
1561 fd76e73a Richard Henderson
    { INDEX_op_add2_i32, { "r", "r", "rZ", "rZ", "rI", "rZ" } },
1562 91493631 Richard Henderson
    { INDEX_op_sub2_i32, { "r", "r", "rI", "rZ", "rK", "rZ" } },
1563 f54b3f92 aurel32
1564 f54b3f92 aurel32
#if TARGET_LONG_BITS == 32
1565 f54b3f92 aurel32
    { INDEX_op_qemu_ld8u, { "r", "L" } },
1566 f54b3f92 aurel32
    { INDEX_op_qemu_ld8s, { "r", "L" } },
1567 f54b3f92 aurel32
    { INDEX_op_qemu_ld16u, { "r", "L" } },
1568 f54b3f92 aurel32
    { INDEX_op_qemu_ld16s, { "r", "L" } },
1569 86feb1c8 Richard Henderson
    { INDEX_op_qemu_ld32, { "r", "L" } },
1570 f54b3f92 aurel32
    { INDEX_op_qemu_ld64, { "r", "r", "L" } },
1571 f54b3f92 aurel32
1572 fd76e73a Richard Henderson
    { INDEX_op_qemu_st8, { "LZ", "L" } },
1573 fd76e73a Richard Henderson
    { INDEX_op_qemu_st16, { "LZ", "L" } },
1574 fd76e73a Richard Henderson
    { INDEX_op_qemu_st32, { "LZ", "L" } },
1575 fd76e73a Richard Henderson
    { INDEX_op_qemu_st64, { "LZ", "LZ", "L" } },
1576 f54b3f92 aurel32
#else
1577 f54b3f92 aurel32
    { INDEX_op_qemu_ld8u, { "r", "L", "L" } },
1578 f54b3f92 aurel32
    { INDEX_op_qemu_ld8s, { "r", "L", "L" } },
1579 f54b3f92 aurel32
    { INDEX_op_qemu_ld16u, { "r", "L", "L" } },
1580 f54b3f92 aurel32
    { INDEX_op_qemu_ld16s, { "r", "L", "L" } },
1581 86feb1c8 Richard Henderson
    { INDEX_op_qemu_ld32, { "r", "L", "L" } },
1582 f54b3f92 aurel32
    { INDEX_op_qemu_ld64, { "r", "r", "L", "L" } },
1583 f54b3f92 aurel32
1584 fd76e73a Richard Henderson
    { INDEX_op_qemu_st8, { "LZ", "L", "L" } },
1585 fd76e73a Richard Henderson
    { INDEX_op_qemu_st16, { "LZ", "L", "L" } },
1586 fd76e73a Richard Henderson
    { INDEX_op_qemu_st32, { "LZ", "L", "L" } },
1587 fd76e73a Richard Henderson
    { INDEX_op_qemu_st64, { "LZ", "LZ", "L", "L" } },
1588 f54b3f92 aurel32
#endif
1589 f54b3f92 aurel32
    { -1 },
1590 f54b3f92 aurel32
};
1591 f54b3f92 aurel32
1592 fd76e73a Richard Henderson
static int tcg_target_callee_save_regs[] = {
1593 fd76e73a Richard Henderson
    /* R2, the return address register, is saved specially
1594 fd76e73a Richard Henderson
       in the caller's frame.  */
1595 fd76e73a Richard Henderson
    /* R3, the frame pointer, is not currently modified.  */
1596 fd76e73a Richard Henderson
    TCG_REG_R4,
1597 fd76e73a Richard Henderson
    TCG_REG_R5,
1598 fd76e73a Richard Henderson
    TCG_REG_R6,
1599 fd76e73a Richard Henderson
    TCG_REG_R7,
1600 fd76e73a Richard Henderson
    TCG_REG_R8,
1601 fd76e73a Richard Henderson
    TCG_REG_R9,
1602 fd76e73a Richard Henderson
    TCG_REG_R10,
1603 fd76e73a Richard Henderson
    TCG_REG_R11,
1604 fd76e73a Richard Henderson
    TCG_REG_R12,
1605 fd76e73a Richard Henderson
    TCG_REG_R13,
1606 fd76e73a Richard Henderson
    TCG_REG_R14,
1607 fd76e73a Richard Henderson
    TCG_REG_R15,
1608 fd76e73a Richard Henderson
    TCG_REG_R16,
1609 fd76e73a Richard Henderson
    /* R17 is the global env, so no need to save.  */
1610 fd76e73a Richard Henderson
    TCG_REG_R18
1611 fd76e73a Richard Henderson
};
1612 fd76e73a Richard Henderson
1613 fd76e73a Richard Henderson
void tcg_target_qemu_prologue(TCGContext *s)
1614 fd76e73a Richard Henderson
{
1615 fd76e73a Richard Henderson
    int frame_size, i;
1616 fd76e73a Richard Henderson
1617 fd76e73a Richard Henderson
    /* Allocate space for the fixed frame marker.  */
1618 fd76e73a Richard Henderson
    frame_size = -TCG_TARGET_CALL_STACK_OFFSET;
1619 fd76e73a Richard Henderson
    frame_size += TCG_TARGET_STATIC_CALL_ARGS_SIZE;
1620 fd76e73a Richard Henderson
1621 fd76e73a Richard Henderson
    /* Allocate space for the saved registers.  */
1622 fd76e73a Richard Henderson
    frame_size += ARRAY_SIZE(tcg_target_callee_save_regs) * 4;
1623 fd76e73a Richard Henderson
1624 fd76e73a Richard Henderson
    /* Align the allocated space.  */
1625 fd76e73a Richard Henderson
    frame_size = ((frame_size + TCG_TARGET_STACK_ALIGN - 1)
1626 fd76e73a Richard Henderson
                  & -TCG_TARGET_STACK_ALIGN);
1627 fd76e73a Richard Henderson
1628 fd76e73a Richard Henderson
    /* The return address is stored in the caller's frame.  */
1629 fd76e73a Richard Henderson
    tcg_out_st(s, TCG_TYPE_PTR, TCG_REG_RP, TCG_REG_SP, -20);
1630 fd76e73a Richard Henderson
1631 fd76e73a Richard Henderson
    /* Allocate stack frame, saving the first register at the same time.  */
1632 fd76e73a Richard Henderson
    tcg_out_ldst(s, tcg_target_callee_save_regs[0],
1633 fd76e73a Richard Henderson
                 TCG_REG_SP, frame_size, INSN_STWM);
1634 fd76e73a Richard Henderson
1635 fd76e73a Richard Henderson
    /* Save all callee saved registers.  */
1636 fd76e73a Richard Henderson
    for (i = 1; i < ARRAY_SIZE(tcg_target_callee_save_regs); i++) {
1637 fd76e73a Richard Henderson
        tcg_out_st(s, TCG_TYPE_PTR, tcg_target_callee_save_regs[i],
1638 fd76e73a Richard Henderson
                   TCG_REG_SP, -frame_size + i * 4);
1639 fd76e73a Richard Henderson
    }
1640 fd76e73a Richard Henderson
1641 884d348b Richard Henderson
#ifdef CONFIG_USE_GUEST_BASE
1642 884d348b Richard Henderson
    /* Note that GUEST_BASE can change after the prologue is generated.
1643 884d348b Richard Henderson
       To combat that, load the value from the variable instead of
1644 884d348b Richard Henderson
       embedding a constant here.  */
1645 884d348b Richard Henderson
    tcg_out_ld(s, TCG_TYPE_PTR, TCG_GUEST_BASE_REG,
1646 884d348b Richard Henderson
               TCG_REG_R0, (tcg_target_long)&guest_base);
1647 884d348b Richard Henderson
#endif
1648 fd76e73a Richard Henderson
1649 fd76e73a Richard Henderson
    /* Jump to TB, and adjust R18 to be the return address.  */
1650 fd76e73a Richard Henderson
    tcg_out32(s, INSN_BLE_SR4 | INSN_R2(TCG_REG_R26));
1651 fd76e73a Richard Henderson
    tcg_out_mov(s, TCG_REG_R18, TCG_REG_R31);
1652 fd76e73a Richard Henderson
1653 fd76e73a Richard Henderson
    /* Restore callee saved registers.  */
1654 fd76e73a Richard Henderson
    tcg_out_ld(s, TCG_TYPE_PTR, TCG_REG_RP, TCG_REG_SP, -frame_size - 20);
1655 fd76e73a Richard Henderson
    for (i = 1; i < ARRAY_SIZE(tcg_target_callee_save_regs); i++) {
1656 fd76e73a Richard Henderson
        tcg_out_ld(s, TCG_TYPE_PTR, tcg_target_callee_save_regs[i],
1657 fd76e73a Richard Henderson
                   TCG_REG_SP, -frame_size + i * 4);
1658 fd76e73a Richard Henderson
    }
1659 fd76e73a Richard Henderson
1660 fd76e73a Richard Henderson
    /* Deallocate stack frame and return.  */
1661 fd76e73a Richard Henderson
    tcg_out32(s, INSN_BV | INSN_R2(TCG_REG_RP));
1662 fd76e73a Richard Henderson
    tcg_out_ldst(s, tcg_target_callee_save_regs[0],
1663 fd76e73a Richard Henderson
                 TCG_REG_SP, -frame_size, INSN_LDWM);
1664 fd76e73a Richard Henderson
}
1665 fd76e73a Richard Henderson
1666 f54b3f92 aurel32
void tcg_target_init(TCGContext *s)
1667 f54b3f92 aurel32
{
1668 f54b3f92 aurel32
    tcg_regset_set32(tcg_target_available_regs[TCG_TYPE_I32], 0, 0xffffffff);
1669 fd76e73a Richard Henderson
1670 fd76e73a Richard Henderson
    tcg_regset_clear(tcg_target_call_clobber_regs);
1671 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R20);
1672 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R21);
1673 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R22);
1674 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R23);
1675 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R24);
1676 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R25);
1677 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_R26);
1678 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_RET0);
1679 fd76e73a Richard Henderson
    tcg_regset_set_reg(tcg_target_call_clobber_regs, TCG_REG_RET1);
1680 f54b3f92 aurel32
1681 f54b3f92 aurel32
    tcg_regset_clear(s->reserved_regs);
1682 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R0);  /* hardwired to zero */
1683 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R1);  /* addil target */
1684 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_RP);  /* link register */
1685 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R3);  /* frame pointer */
1686 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R18); /* return pointer */
1687 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R19); /* clobbered w/o pic */
1688 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R20); /* reserved */
1689 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_DP);  /* data pointer */
1690 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_SP);  /* stack pointer */
1691 f54b3f92 aurel32
    tcg_regset_set_reg(s->reserved_regs, TCG_REG_R31); /* ble link reg */
1692 884d348b Richard Henderson
#ifdef CONFIG_USE_GUEST_BASE
1693 884d348b Richard Henderson
    tcg_regset_set_reg(s->reserved_regs, TCG_GUEST_BASE_REG);
1694 884d348b Richard Henderson
#endif
1695 f54b3f92 aurel32
1696 f54b3f92 aurel32
    tcg_add_target_add_op_defs(hppa_op_defs);
1697 f54b3f92 aurel32
}