1a2b0a27dSPhilippe Mathieu-Daudé /*
2a2b0a27dSPhilippe Mathieu-Daudé * MIPS ASE DSP Instruction emulation helpers for QEMU.
3a2b0a27dSPhilippe Mathieu-Daudé *
4a2b0a27dSPhilippe Mathieu-Daudé * Copyright (c) 2012 Jia Liu <proljc@gmail.com>
5a2b0a27dSPhilippe Mathieu-Daudé * Dongxue Zhang <elta.era@gmail.com>
6a2b0a27dSPhilippe Mathieu-Daudé * This library is free software; you can redistribute it and/or
7a2b0a27dSPhilippe Mathieu-Daudé * modify it under the terms of the GNU Lesser General Public
8a2b0a27dSPhilippe Mathieu-Daudé * License as published by the Free Software Foundation; either
9a2b0a27dSPhilippe Mathieu-Daudé * version 2.1 of the License, or (at your option) any later version.
10a2b0a27dSPhilippe Mathieu-Daudé *
11a2b0a27dSPhilippe Mathieu-Daudé * This library is distributed in the hope that it will be useful,
12a2b0a27dSPhilippe Mathieu-Daudé * but WITHOUT ANY WARRANTY; without even the implied warranty of
13a2b0a27dSPhilippe Mathieu-Daudé * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14a2b0a27dSPhilippe Mathieu-Daudé * Lesser General Public License for more details.
15a2b0a27dSPhilippe Mathieu-Daudé *
16a2b0a27dSPhilippe Mathieu-Daudé * You should have received a copy of the GNU Lesser General Public
17a2b0a27dSPhilippe Mathieu-Daudé * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18a2b0a27dSPhilippe Mathieu-Daudé */
19a2b0a27dSPhilippe Mathieu-Daudé
20a2b0a27dSPhilippe Mathieu-Daudé #include "qemu/osdep.h"
21a2b0a27dSPhilippe Mathieu-Daudé #include "cpu.h"
22a2b0a27dSPhilippe Mathieu-Daudé #include "exec/helper-proto.h"
23a2b0a27dSPhilippe Mathieu-Daudé #include "qemu/bitops.h"
24a2b0a27dSPhilippe Mathieu-Daudé
25a2b0a27dSPhilippe Mathieu-Daudé /*
26a2b0a27dSPhilippe Mathieu-Daudé * As the byte ordering doesn't matter, i.e. all columns are treated
27a2b0a27dSPhilippe Mathieu-Daudé * identically, these unions can be used directly.
28a2b0a27dSPhilippe Mathieu-Daudé */
29a2b0a27dSPhilippe Mathieu-Daudé typedef union {
30a2b0a27dSPhilippe Mathieu-Daudé uint8_t ub[4];
31a2b0a27dSPhilippe Mathieu-Daudé int8_t sb[4];
32a2b0a27dSPhilippe Mathieu-Daudé uint16_t uh[2];
33a2b0a27dSPhilippe Mathieu-Daudé int16_t sh[2];
34a2b0a27dSPhilippe Mathieu-Daudé uint32_t uw[1];
35a2b0a27dSPhilippe Mathieu-Daudé int32_t sw[1];
36a2b0a27dSPhilippe Mathieu-Daudé } DSP32Value;
37a2b0a27dSPhilippe Mathieu-Daudé
38a2b0a27dSPhilippe Mathieu-Daudé typedef union {
39a2b0a27dSPhilippe Mathieu-Daudé uint8_t ub[8];
40a2b0a27dSPhilippe Mathieu-Daudé int8_t sb[8];
41a2b0a27dSPhilippe Mathieu-Daudé uint16_t uh[4];
42a2b0a27dSPhilippe Mathieu-Daudé int16_t sh[4];
43a2b0a27dSPhilippe Mathieu-Daudé uint32_t uw[2];
44a2b0a27dSPhilippe Mathieu-Daudé int32_t sw[2];
45a2b0a27dSPhilippe Mathieu-Daudé uint64_t ul[1];
46a2b0a27dSPhilippe Mathieu-Daudé int64_t sl[1];
47a2b0a27dSPhilippe Mathieu-Daudé } DSP64Value;
48a2b0a27dSPhilippe Mathieu-Daudé
49a2b0a27dSPhilippe Mathieu-Daudé /*** MIPS DSP internal functions begin ***/
50a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_ABS(x) (((x) >= 0) ? (x) : -(x))
51a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_OVERFLOW_ADD(a, b, c, d) (~((a) ^ (b)) & ((a) ^ (c)) & (d))
52a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_OVERFLOW_SUB(a, b, c, d) (((a) ^ (b)) & ((a) ^ (c)) & (d))
53a2b0a27dSPhilippe Mathieu-Daudé
set_DSPControl_overflow_flag(uint32_t flag,int position,CPUMIPSState * env)54a2b0a27dSPhilippe Mathieu-Daudé static inline void set_DSPControl_overflow_flag(uint32_t flag, int position,
55a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
56a2b0a27dSPhilippe Mathieu-Daudé {
57a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl |= (target_ulong)flag << position;
58a2b0a27dSPhilippe Mathieu-Daudé }
59a2b0a27dSPhilippe Mathieu-Daudé
set_DSPControl_carryflag(bool flag,CPUMIPSState * env)60a2b0a27dSPhilippe Mathieu-Daudé static inline void set_DSPControl_carryflag(bool flag, CPUMIPSState *env)
61a2b0a27dSPhilippe Mathieu-Daudé {
62a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl &= ~(1 << 13);
63a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl |= flag << 13;
64a2b0a27dSPhilippe Mathieu-Daudé }
65a2b0a27dSPhilippe Mathieu-Daudé
get_DSPControl_carryflag(CPUMIPSState * env)66a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t get_DSPControl_carryflag(CPUMIPSState *env)
67a2b0a27dSPhilippe Mathieu-Daudé {
68a2b0a27dSPhilippe Mathieu-Daudé return (env->active_tc.DSPControl >> 13) & 0x01;
69a2b0a27dSPhilippe Mathieu-Daudé }
70a2b0a27dSPhilippe Mathieu-Daudé
set_DSPControl_24(uint32_t flag,int len,CPUMIPSState * env)71a2b0a27dSPhilippe Mathieu-Daudé static inline void set_DSPControl_24(uint32_t flag, int len, CPUMIPSState *env)
72a2b0a27dSPhilippe Mathieu-Daudé {
73a2b0a27dSPhilippe Mathieu-Daudé uint32_t filter;
74a2b0a27dSPhilippe Mathieu-Daudé
75a2b0a27dSPhilippe Mathieu-Daudé filter = ((0x01 << len) - 1) << 24;
76a2b0a27dSPhilippe Mathieu-Daudé filter = ~filter;
77a2b0a27dSPhilippe Mathieu-Daudé
78a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl &= filter;
79a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl |= (target_ulong)flag << 24;
80a2b0a27dSPhilippe Mathieu-Daudé }
81a2b0a27dSPhilippe Mathieu-Daudé
set_DSPControl_pos(uint32_t pos,CPUMIPSState * env)82a2b0a27dSPhilippe Mathieu-Daudé static inline void set_DSPControl_pos(uint32_t pos, CPUMIPSState *env)
83a2b0a27dSPhilippe Mathieu-Daudé {
84a2b0a27dSPhilippe Mathieu-Daudé target_ulong dspc;
85a2b0a27dSPhilippe Mathieu-Daudé
86a2b0a27dSPhilippe Mathieu-Daudé dspc = env->active_tc.DSPControl;
87a2b0a27dSPhilippe Mathieu-Daudé #ifndef TARGET_MIPS64
88a2b0a27dSPhilippe Mathieu-Daudé dspc = dspc & 0xFFFFFFC0;
89a2b0a27dSPhilippe Mathieu-Daudé dspc |= (pos & 0x3F);
90a2b0a27dSPhilippe Mathieu-Daudé #else
91a2b0a27dSPhilippe Mathieu-Daudé dspc = dspc & 0xFFFFFF80;
92a2b0a27dSPhilippe Mathieu-Daudé dspc |= (pos & 0x7F);
93a2b0a27dSPhilippe Mathieu-Daudé #endif
94a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl = dspc;
95a2b0a27dSPhilippe Mathieu-Daudé }
96a2b0a27dSPhilippe Mathieu-Daudé
get_DSPControl_pos(CPUMIPSState * env)97a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t get_DSPControl_pos(CPUMIPSState *env)
98a2b0a27dSPhilippe Mathieu-Daudé {
99a2b0a27dSPhilippe Mathieu-Daudé target_ulong dspc;
100a2b0a27dSPhilippe Mathieu-Daudé uint32_t pos;
101a2b0a27dSPhilippe Mathieu-Daudé
102a2b0a27dSPhilippe Mathieu-Daudé dspc = env->active_tc.DSPControl;
103a2b0a27dSPhilippe Mathieu-Daudé
104a2b0a27dSPhilippe Mathieu-Daudé #ifndef TARGET_MIPS64
105a2b0a27dSPhilippe Mathieu-Daudé pos = dspc & 0x3F;
106a2b0a27dSPhilippe Mathieu-Daudé #else
107a2b0a27dSPhilippe Mathieu-Daudé pos = dspc & 0x7F;
108a2b0a27dSPhilippe Mathieu-Daudé #endif
109a2b0a27dSPhilippe Mathieu-Daudé
110a2b0a27dSPhilippe Mathieu-Daudé return pos;
111a2b0a27dSPhilippe Mathieu-Daudé }
112a2b0a27dSPhilippe Mathieu-Daudé
set_DSPControl_efi(uint32_t flag,CPUMIPSState * env)113a2b0a27dSPhilippe Mathieu-Daudé static inline void set_DSPControl_efi(uint32_t flag, CPUMIPSState *env)
114a2b0a27dSPhilippe Mathieu-Daudé {
115a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl &= 0xFFFFBFFF;
116a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl |= (target_ulong)flag << 14;
117a2b0a27dSPhilippe Mathieu-Daudé }
118a2b0a27dSPhilippe Mathieu-Daudé
119a2b0a27dSPhilippe Mathieu-Daudé #define DO_MIPS_SAT_ABS(size) \
120a2b0a27dSPhilippe Mathieu-Daudé static inline int##size##_t mipsdsp_sat_abs##size(int##size##_t a, \
121a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
122a2b0a27dSPhilippe Mathieu-Daudé { \
123a2b0a27dSPhilippe Mathieu-Daudé if (a == INT##size##_MIN) { \
124a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env); \
125a2b0a27dSPhilippe Mathieu-Daudé return INT##size##_MAX; \
126a2b0a27dSPhilippe Mathieu-Daudé } else { \
127a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_ABS(a); \
128a2b0a27dSPhilippe Mathieu-Daudé } \
129a2b0a27dSPhilippe Mathieu-Daudé }
130a2b0a27dSPhilippe Mathieu-Daudé DO_MIPS_SAT_ABS(8)
131a2b0a27dSPhilippe Mathieu-Daudé DO_MIPS_SAT_ABS(16)
132a2b0a27dSPhilippe Mathieu-Daudé DO_MIPS_SAT_ABS(32)
133a2b0a27dSPhilippe Mathieu-Daudé #undef DO_MIPS_SAT_ABS
134a2b0a27dSPhilippe Mathieu-Daudé
135a2b0a27dSPhilippe Mathieu-Daudé /* get sum value */
mipsdsp_add_i16(int16_t a,int16_t b,CPUMIPSState * env)136a2b0a27dSPhilippe Mathieu-Daudé static inline int16_t mipsdsp_add_i16(int16_t a, int16_t b, CPUMIPSState *env)
137a2b0a27dSPhilippe Mathieu-Daudé {
138a2b0a27dSPhilippe Mathieu-Daudé int16_t tempI;
139a2b0a27dSPhilippe Mathieu-Daudé
140a2b0a27dSPhilippe Mathieu-Daudé tempI = a + b;
141a2b0a27dSPhilippe Mathieu-Daudé
142a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_ADD(a, b, tempI, 0x8000)) {
143a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
144a2b0a27dSPhilippe Mathieu-Daudé }
145a2b0a27dSPhilippe Mathieu-Daudé
146a2b0a27dSPhilippe Mathieu-Daudé return tempI;
147a2b0a27dSPhilippe Mathieu-Daudé }
148a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat_add_i16(int16_t a,int16_t b,CPUMIPSState * env)149a2b0a27dSPhilippe Mathieu-Daudé static inline int16_t mipsdsp_sat_add_i16(int16_t a, int16_t b,
150a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
151a2b0a27dSPhilippe Mathieu-Daudé {
152a2b0a27dSPhilippe Mathieu-Daudé int16_t tempS;
153a2b0a27dSPhilippe Mathieu-Daudé
154a2b0a27dSPhilippe Mathieu-Daudé tempS = a + b;
155a2b0a27dSPhilippe Mathieu-Daudé
156a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_ADD(a, b, tempS, 0x8000)) {
157a2b0a27dSPhilippe Mathieu-Daudé if (a > 0) {
158a2b0a27dSPhilippe Mathieu-Daudé tempS = 0x7FFF;
159a2b0a27dSPhilippe Mathieu-Daudé } else {
160a2b0a27dSPhilippe Mathieu-Daudé tempS = 0x8000;
161a2b0a27dSPhilippe Mathieu-Daudé }
162a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
163a2b0a27dSPhilippe Mathieu-Daudé }
164a2b0a27dSPhilippe Mathieu-Daudé
165a2b0a27dSPhilippe Mathieu-Daudé return tempS;
166a2b0a27dSPhilippe Mathieu-Daudé }
167a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat_add_i32(int32_t a,int32_t b,CPUMIPSState * env)168a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_sat_add_i32(int32_t a, int32_t b,
169a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
170a2b0a27dSPhilippe Mathieu-Daudé {
171a2b0a27dSPhilippe Mathieu-Daudé int32_t tempI;
172a2b0a27dSPhilippe Mathieu-Daudé
173a2b0a27dSPhilippe Mathieu-Daudé tempI = a + b;
174a2b0a27dSPhilippe Mathieu-Daudé
175a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_ADD(a, b, tempI, 0x80000000)) {
176a2b0a27dSPhilippe Mathieu-Daudé if (a > 0) {
177a2b0a27dSPhilippe Mathieu-Daudé tempI = 0x7FFFFFFF;
178a2b0a27dSPhilippe Mathieu-Daudé } else {
179a2b0a27dSPhilippe Mathieu-Daudé tempI = 0x80000000;
180a2b0a27dSPhilippe Mathieu-Daudé }
181a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
182a2b0a27dSPhilippe Mathieu-Daudé }
183a2b0a27dSPhilippe Mathieu-Daudé
184a2b0a27dSPhilippe Mathieu-Daudé return tempI;
185a2b0a27dSPhilippe Mathieu-Daudé }
186a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_add_u8(uint8_t a,uint8_t b,CPUMIPSState * env)187a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_add_u8(uint8_t a, uint8_t b, CPUMIPSState *env)
188a2b0a27dSPhilippe Mathieu-Daudé {
189a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
190a2b0a27dSPhilippe Mathieu-Daudé
191a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a + (uint16_t)b;
192a2b0a27dSPhilippe Mathieu-Daudé
193a2b0a27dSPhilippe Mathieu-Daudé if (temp & 0x0100) {
194a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
195a2b0a27dSPhilippe Mathieu-Daudé }
196a2b0a27dSPhilippe Mathieu-Daudé
197a2b0a27dSPhilippe Mathieu-Daudé return temp & 0xFF;
198a2b0a27dSPhilippe Mathieu-Daudé }
199a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_add_u16(uint16_t a,uint16_t b,CPUMIPSState * env)200a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_add_u16(uint16_t a, uint16_t b,
201a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
202a2b0a27dSPhilippe Mathieu-Daudé {
203a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
204a2b0a27dSPhilippe Mathieu-Daudé
205a2b0a27dSPhilippe Mathieu-Daudé temp = (uint32_t)a + (uint32_t)b;
206a2b0a27dSPhilippe Mathieu-Daudé
207a2b0a27dSPhilippe Mathieu-Daudé if (temp & 0x00010000) {
208a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
209a2b0a27dSPhilippe Mathieu-Daudé }
210a2b0a27dSPhilippe Mathieu-Daudé
211a2b0a27dSPhilippe Mathieu-Daudé return temp & 0xFFFF;
212a2b0a27dSPhilippe Mathieu-Daudé }
213a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat_add_u8(uint8_t a,uint8_t b,CPUMIPSState * env)214a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_sat_add_u8(uint8_t a, uint8_t b,
215a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
216a2b0a27dSPhilippe Mathieu-Daudé {
217a2b0a27dSPhilippe Mathieu-Daudé uint8_t result;
218a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
219a2b0a27dSPhilippe Mathieu-Daudé
220a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a + (uint16_t)b;
221a2b0a27dSPhilippe Mathieu-Daudé result = temp & 0xFF;
222a2b0a27dSPhilippe Mathieu-Daudé
223a2b0a27dSPhilippe Mathieu-Daudé if (0x0100 & temp) {
224a2b0a27dSPhilippe Mathieu-Daudé result = 0xFF;
225a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
226a2b0a27dSPhilippe Mathieu-Daudé }
227a2b0a27dSPhilippe Mathieu-Daudé
228a2b0a27dSPhilippe Mathieu-Daudé return result;
229a2b0a27dSPhilippe Mathieu-Daudé }
230a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat_add_u16(uint16_t a,uint16_t b,CPUMIPSState * env)231a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_sat_add_u16(uint16_t a, uint16_t b,
232a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
233a2b0a27dSPhilippe Mathieu-Daudé {
234a2b0a27dSPhilippe Mathieu-Daudé uint16_t result;
235a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
236a2b0a27dSPhilippe Mathieu-Daudé
237a2b0a27dSPhilippe Mathieu-Daudé temp = (uint32_t)a + (uint32_t)b;
238a2b0a27dSPhilippe Mathieu-Daudé result = temp & 0xFFFF;
239a2b0a27dSPhilippe Mathieu-Daudé
240a2b0a27dSPhilippe Mathieu-Daudé if (0x00010000 & temp) {
241a2b0a27dSPhilippe Mathieu-Daudé result = 0xFFFF;
242a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
243a2b0a27dSPhilippe Mathieu-Daudé }
244a2b0a27dSPhilippe Mathieu-Daudé
245a2b0a27dSPhilippe Mathieu-Daudé return result;
246a2b0a27dSPhilippe Mathieu-Daudé }
247a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat32_acc_q31(int32_t acc,int32_t a,CPUMIPSState * env)248a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_sat32_acc_q31(int32_t acc, int32_t a,
249a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
250a2b0a27dSPhilippe Mathieu-Daudé {
251a2b0a27dSPhilippe Mathieu-Daudé int64_t temp;
252a2b0a27dSPhilippe Mathieu-Daudé int32_t temp32, temp31, result;
253a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum;
254a2b0a27dSPhilippe Mathieu-Daudé
255a2b0a27dSPhilippe Mathieu-Daudé #ifndef TARGET_MIPS64
256a2b0a27dSPhilippe Mathieu-Daudé temp = ((uint64_t)env->active_tc.HI[acc] << 32) |
257a2b0a27dSPhilippe Mathieu-Daudé (uint64_t)env->active_tc.LO[acc];
258a2b0a27dSPhilippe Mathieu-Daudé #else
259a2b0a27dSPhilippe Mathieu-Daudé temp = (uint64_t)env->active_tc.LO[acc];
260a2b0a27dSPhilippe Mathieu-Daudé #endif
261a2b0a27dSPhilippe Mathieu-Daudé
262a2b0a27dSPhilippe Mathieu-Daudé temp_sum = (int64_t)a + temp;
263a2b0a27dSPhilippe Mathieu-Daudé
264a2b0a27dSPhilippe Mathieu-Daudé temp32 = (temp_sum >> 32) & 0x01;
265a2b0a27dSPhilippe Mathieu-Daudé temp31 = (temp_sum >> 31) & 0x01;
266a2b0a27dSPhilippe Mathieu-Daudé result = temp_sum & 0xFFFFFFFF;
267a2b0a27dSPhilippe Mathieu-Daudé
268a2b0a27dSPhilippe Mathieu-Daudé if (temp32 != temp31) {
269a2b0a27dSPhilippe Mathieu-Daudé if (temp32 == 0) {
270a2b0a27dSPhilippe Mathieu-Daudé result = 0x7FFFFFFF;
271a2b0a27dSPhilippe Mathieu-Daudé } else {
272a2b0a27dSPhilippe Mathieu-Daudé result = 0x80000000;
273a2b0a27dSPhilippe Mathieu-Daudé }
274a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + acc, env);
275a2b0a27dSPhilippe Mathieu-Daudé }
276a2b0a27dSPhilippe Mathieu-Daudé
277a2b0a27dSPhilippe Mathieu-Daudé return result;
278a2b0a27dSPhilippe Mathieu-Daudé }
279a2b0a27dSPhilippe Mathieu-Daudé
280a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
281a2b0a27dSPhilippe Mathieu-Daudé /* a[0] is LO, a[1] is HI. */
mipsdsp_sat64_acc_add_q63(int64_t * ret,int32_t ac,int64_t * a,CPUMIPSState * env)282a2b0a27dSPhilippe Mathieu-Daudé static inline void mipsdsp_sat64_acc_add_q63(int64_t *ret,
283a2b0a27dSPhilippe Mathieu-Daudé int32_t ac,
284a2b0a27dSPhilippe Mathieu-Daudé int64_t *a,
285a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
286a2b0a27dSPhilippe Mathieu-Daudé {
287a2b0a27dSPhilippe Mathieu-Daudé bool temp64;
288a2b0a27dSPhilippe Mathieu-Daudé
289a2b0a27dSPhilippe Mathieu-Daudé ret[0] = env->active_tc.LO[ac] + a[0];
290a2b0a27dSPhilippe Mathieu-Daudé ret[1] = env->active_tc.HI[ac] + a[1];
291a2b0a27dSPhilippe Mathieu-Daudé
292a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)ret[0] < (uint64_t)env->active_tc.LO[ac]) &&
293a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)ret[0] < (uint64_t)a[0])) {
294a2b0a27dSPhilippe Mathieu-Daudé ret[1] += 1;
295a2b0a27dSPhilippe Mathieu-Daudé }
296a2b0a27dSPhilippe Mathieu-Daudé temp64 = ret[1] & 1;
297a2b0a27dSPhilippe Mathieu-Daudé if (temp64 != ((ret[0] >> 63) & 0x01)) {
298a2b0a27dSPhilippe Mathieu-Daudé if (temp64) {
299a2b0a27dSPhilippe Mathieu-Daudé ret[0] = (0x01ull << 63);
300a2b0a27dSPhilippe Mathieu-Daudé ret[1] = ~0ull;
301a2b0a27dSPhilippe Mathieu-Daudé } else {
302a2b0a27dSPhilippe Mathieu-Daudé ret[0] = (0x01ull << 63) - 1;
303a2b0a27dSPhilippe Mathieu-Daudé ret[1] = 0x00;
304a2b0a27dSPhilippe Mathieu-Daudé }
305a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env);
306a2b0a27dSPhilippe Mathieu-Daudé }
307a2b0a27dSPhilippe Mathieu-Daudé }
308a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat64_acc_sub_q63(int64_t * ret,int32_t ac,int64_t * a,CPUMIPSState * env)309a2b0a27dSPhilippe Mathieu-Daudé static inline void mipsdsp_sat64_acc_sub_q63(int64_t *ret,
310a2b0a27dSPhilippe Mathieu-Daudé int32_t ac,
311a2b0a27dSPhilippe Mathieu-Daudé int64_t *a,
312a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
313a2b0a27dSPhilippe Mathieu-Daudé {
314a2b0a27dSPhilippe Mathieu-Daudé bool temp64;
315a2b0a27dSPhilippe Mathieu-Daudé
316a2b0a27dSPhilippe Mathieu-Daudé ret[0] = env->active_tc.LO[ac] - a[0];
317a2b0a27dSPhilippe Mathieu-Daudé ret[1] = env->active_tc.HI[ac] - a[1];
318a2b0a27dSPhilippe Mathieu-Daudé
319a2b0a27dSPhilippe Mathieu-Daudé if ((uint64_t)ret[0] > (uint64_t)env->active_tc.LO[ac]) {
320a2b0a27dSPhilippe Mathieu-Daudé ret[1] -= 1;
321a2b0a27dSPhilippe Mathieu-Daudé }
322a2b0a27dSPhilippe Mathieu-Daudé temp64 = ret[1] & 1;
323a2b0a27dSPhilippe Mathieu-Daudé if (temp64 != ((ret[0] >> 63) & 0x01)) {
324a2b0a27dSPhilippe Mathieu-Daudé if (temp64) {
325a2b0a27dSPhilippe Mathieu-Daudé ret[0] = (0x01ull << 63);
326a2b0a27dSPhilippe Mathieu-Daudé ret[1] = ~0ull;
327a2b0a27dSPhilippe Mathieu-Daudé } else {
328a2b0a27dSPhilippe Mathieu-Daudé ret[0] = (0x01ull << 63) - 1;
329a2b0a27dSPhilippe Mathieu-Daudé ret[1] = 0x00;
330a2b0a27dSPhilippe Mathieu-Daudé }
331a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env);
332a2b0a27dSPhilippe Mathieu-Daudé }
333a2b0a27dSPhilippe Mathieu-Daudé }
334a2b0a27dSPhilippe Mathieu-Daudé #endif
335a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_i16_i16(int16_t a,int16_t b,CPUMIPSState * env)336a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_mul_i16_i16(int16_t a, int16_t b,
337a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
338a2b0a27dSPhilippe Mathieu-Daudé {
339a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
340a2b0a27dSPhilippe Mathieu-Daudé
341a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)a * (int32_t)b;
342a2b0a27dSPhilippe Mathieu-Daudé
343a2b0a27dSPhilippe Mathieu-Daudé if ((temp > (int)0x7FFF) || (temp < (int)0xFFFF8000)) {
344a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
345a2b0a27dSPhilippe Mathieu-Daudé }
346a2b0a27dSPhilippe Mathieu-Daudé temp &= 0x0000FFFF;
347a2b0a27dSPhilippe Mathieu-Daudé
348a2b0a27dSPhilippe Mathieu-Daudé return temp;
349a2b0a27dSPhilippe Mathieu-Daudé }
350a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_u16_u16(int32_t a,int32_t b)351a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_mul_u16_u16(int32_t a, int32_t b)
352a2b0a27dSPhilippe Mathieu-Daudé {
353a2b0a27dSPhilippe Mathieu-Daudé return a * b;
354a2b0a27dSPhilippe Mathieu-Daudé }
355a2b0a27dSPhilippe Mathieu-Daudé
356a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
mipsdsp_mul_i32_i32(int32_t a,int32_t b)357a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_mul_i32_i32(int32_t a, int32_t b)
358a2b0a27dSPhilippe Mathieu-Daudé {
359a2b0a27dSPhilippe Mathieu-Daudé return a * b;
360a2b0a27dSPhilippe Mathieu-Daudé }
361a2b0a27dSPhilippe Mathieu-Daudé #endif
362a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat16_mul_i16_i16(int16_t a,int16_t b,CPUMIPSState * env)363a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_sat16_mul_i16_i16(int16_t a, int16_t b,
364a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
365a2b0a27dSPhilippe Mathieu-Daudé {
366a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
367a2b0a27dSPhilippe Mathieu-Daudé
368a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)a * (int32_t)b;
369a2b0a27dSPhilippe Mathieu-Daudé
370a2b0a27dSPhilippe Mathieu-Daudé if (temp > (int)0x7FFF) {
371a2b0a27dSPhilippe Mathieu-Daudé temp = 0x00007FFF;
372a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
373a2b0a27dSPhilippe Mathieu-Daudé } else if (temp < (int)0xffff8000) {
374a2b0a27dSPhilippe Mathieu-Daudé temp = 0xFFFF8000;
375a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
376a2b0a27dSPhilippe Mathieu-Daudé }
377a2b0a27dSPhilippe Mathieu-Daudé temp &= 0x0000FFFF;
378a2b0a27dSPhilippe Mathieu-Daudé
379a2b0a27dSPhilippe Mathieu-Daudé return temp;
380a2b0a27dSPhilippe Mathieu-Daudé }
381a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_q15_q15_overflowflag21(uint16_t a,uint16_t b,CPUMIPSState * env)382a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_mul_q15_q15_overflowflag21(uint16_t a, uint16_t b,
383a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
384a2b0a27dSPhilippe Mathieu-Daudé {
385a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
386a2b0a27dSPhilippe Mathieu-Daudé
387a2b0a27dSPhilippe Mathieu-Daudé if ((a == 0x8000) && (b == 0x8000)) {
388a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFFFFFF;
389a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
390a2b0a27dSPhilippe Mathieu-Daudé } else {
391a2b0a27dSPhilippe Mathieu-Daudé temp = ((int16_t)a * (int16_t)b) << 1;
392a2b0a27dSPhilippe Mathieu-Daudé }
393a2b0a27dSPhilippe Mathieu-Daudé
394a2b0a27dSPhilippe Mathieu-Daudé return temp;
395a2b0a27dSPhilippe Mathieu-Daudé }
396a2b0a27dSPhilippe Mathieu-Daudé
397a2b0a27dSPhilippe Mathieu-Daudé /* right shift */
mipsdsp_rshift_u8(uint8_t a,target_ulong mov)398a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_rshift_u8(uint8_t a, target_ulong mov)
399a2b0a27dSPhilippe Mathieu-Daudé {
400a2b0a27dSPhilippe Mathieu-Daudé return a >> mov;
401a2b0a27dSPhilippe Mathieu-Daudé }
402a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rshift_u16(uint16_t a,target_ulong mov)403a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_rshift_u16(uint16_t a, target_ulong mov)
404a2b0a27dSPhilippe Mathieu-Daudé {
405a2b0a27dSPhilippe Mathieu-Daudé return a >> mov;
406a2b0a27dSPhilippe Mathieu-Daudé }
407a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rashift8(int8_t a,target_ulong mov)408a2b0a27dSPhilippe Mathieu-Daudé static inline int8_t mipsdsp_rashift8(int8_t a, target_ulong mov)
409a2b0a27dSPhilippe Mathieu-Daudé {
410a2b0a27dSPhilippe Mathieu-Daudé return a >> mov;
411a2b0a27dSPhilippe Mathieu-Daudé }
412a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rashift16(int16_t a,target_ulong mov)413a2b0a27dSPhilippe Mathieu-Daudé static inline int16_t mipsdsp_rashift16(int16_t a, target_ulong mov)
414a2b0a27dSPhilippe Mathieu-Daudé {
415a2b0a27dSPhilippe Mathieu-Daudé return a >> mov;
416a2b0a27dSPhilippe Mathieu-Daudé }
417a2b0a27dSPhilippe Mathieu-Daudé
418a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
mipsdsp_rashift32(int32_t a,target_ulong mov)419a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_rashift32(int32_t a, target_ulong mov)
420a2b0a27dSPhilippe Mathieu-Daudé {
421a2b0a27dSPhilippe Mathieu-Daudé return a >> mov;
422a2b0a27dSPhilippe Mathieu-Daudé }
423a2b0a27dSPhilippe Mathieu-Daudé #endif
424a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rshift1_add_q16(int16_t a,int16_t b)425a2b0a27dSPhilippe Mathieu-Daudé static inline int16_t mipsdsp_rshift1_add_q16(int16_t a, int16_t b)
426a2b0a27dSPhilippe Mathieu-Daudé {
427a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
428a2b0a27dSPhilippe Mathieu-Daudé
429a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)a + (int32_t)b;
430a2b0a27dSPhilippe Mathieu-Daudé
431a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFF;
432a2b0a27dSPhilippe Mathieu-Daudé }
433a2b0a27dSPhilippe Mathieu-Daudé
434a2b0a27dSPhilippe Mathieu-Daudé /* round right shift */
mipsdsp_rrshift1_add_q16(int16_t a,int16_t b)435a2b0a27dSPhilippe Mathieu-Daudé static inline int16_t mipsdsp_rrshift1_add_q16(int16_t a, int16_t b)
436a2b0a27dSPhilippe Mathieu-Daudé {
437a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
438a2b0a27dSPhilippe Mathieu-Daudé
439a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)a + (int32_t)b;
440a2b0a27dSPhilippe Mathieu-Daudé temp += 1;
441a2b0a27dSPhilippe Mathieu-Daudé
442a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFF;
443a2b0a27dSPhilippe Mathieu-Daudé }
444a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rshift1_add_q32(int32_t a,int32_t b)445a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_rshift1_add_q32(int32_t a, int32_t b)
446a2b0a27dSPhilippe Mathieu-Daudé {
447a2b0a27dSPhilippe Mathieu-Daudé int64_t temp;
448a2b0a27dSPhilippe Mathieu-Daudé
449a2b0a27dSPhilippe Mathieu-Daudé temp = (int64_t)a + (int64_t)b;
450a2b0a27dSPhilippe Mathieu-Daudé
451a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFFFFFF;
452a2b0a27dSPhilippe Mathieu-Daudé }
453a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rrshift1_add_q32(int32_t a,int32_t b)454a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_rrshift1_add_q32(int32_t a, int32_t b)
455a2b0a27dSPhilippe Mathieu-Daudé {
456a2b0a27dSPhilippe Mathieu-Daudé int64_t temp;
457a2b0a27dSPhilippe Mathieu-Daudé
458a2b0a27dSPhilippe Mathieu-Daudé temp = (int64_t)a + (int64_t)b;
459a2b0a27dSPhilippe Mathieu-Daudé temp += 1;
460a2b0a27dSPhilippe Mathieu-Daudé
461a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFFFFFF;
462a2b0a27dSPhilippe Mathieu-Daudé }
463a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rshift1_add_u8(uint8_t a,uint8_t b)464a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_rshift1_add_u8(uint8_t a, uint8_t b)
465a2b0a27dSPhilippe Mathieu-Daudé {
466a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
467a2b0a27dSPhilippe Mathieu-Daudé
468a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a + (uint16_t)b;
469a2b0a27dSPhilippe Mathieu-Daudé
470a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0x00FF;
471a2b0a27dSPhilippe Mathieu-Daudé }
472a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rrshift1_add_u8(uint8_t a,uint8_t b)473a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_rrshift1_add_u8(uint8_t a, uint8_t b)
474a2b0a27dSPhilippe Mathieu-Daudé {
475a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
476a2b0a27dSPhilippe Mathieu-Daudé
477a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a + (uint16_t)b + 1;
478a2b0a27dSPhilippe Mathieu-Daudé
479a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0x00FF;
480a2b0a27dSPhilippe Mathieu-Daudé }
481a2b0a27dSPhilippe Mathieu-Daudé
482a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
mipsdsp_rshift1_sub_u8(uint8_t a,uint8_t b)483a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_rshift1_sub_u8(uint8_t a, uint8_t b)
484a2b0a27dSPhilippe Mathieu-Daudé {
485a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
486a2b0a27dSPhilippe Mathieu-Daudé
487a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a - (uint16_t)b;
488a2b0a27dSPhilippe Mathieu-Daudé
489a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0x00FF;
490a2b0a27dSPhilippe Mathieu-Daudé }
491a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rrshift1_sub_u8(uint8_t a,uint8_t b)492a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_rrshift1_sub_u8(uint8_t a, uint8_t b)
493a2b0a27dSPhilippe Mathieu-Daudé {
494a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
495a2b0a27dSPhilippe Mathieu-Daudé
496a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a - (uint16_t)b + 1;
497a2b0a27dSPhilippe Mathieu-Daudé
498a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0x00FF;
499a2b0a27dSPhilippe Mathieu-Daudé }
500a2b0a27dSPhilippe Mathieu-Daudé #endif
501a2b0a27dSPhilippe Mathieu-Daudé
502a2b0a27dSPhilippe Mathieu-Daudé /* 128 bits long. p[0] is LO, p[1] is HI. */
mipsdsp_rndrashift_short_acc(int64_t * p,int32_t ac,int32_t shift,CPUMIPSState * env)503a2b0a27dSPhilippe Mathieu-Daudé static inline void mipsdsp_rndrashift_short_acc(int64_t *p,
504a2b0a27dSPhilippe Mathieu-Daudé int32_t ac,
505a2b0a27dSPhilippe Mathieu-Daudé int32_t shift,
506a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
507a2b0a27dSPhilippe Mathieu-Daudé {
508a2b0a27dSPhilippe Mathieu-Daudé int64_t acc;
509a2b0a27dSPhilippe Mathieu-Daudé
510a2b0a27dSPhilippe Mathieu-Daudé acc = ((int64_t)env->active_tc.HI[ac] << 32) |
511a2b0a27dSPhilippe Mathieu-Daudé ((int64_t)env->active_tc.LO[ac] & 0xFFFFFFFF);
512a2b0a27dSPhilippe Mathieu-Daudé p[0] = (shift == 0) ? (acc << 1) : (acc >> (shift - 1));
513a2b0a27dSPhilippe Mathieu-Daudé p[1] = (acc >> 63) & 0x01;
514a2b0a27dSPhilippe Mathieu-Daudé }
515a2b0a27dSPhilippe Mathieu-Daudé
516a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
517a2b0a27dSPhilippe Mathieu-Daudé /* 128 bits long. p[0] is LO, p[1] is HI */
mipsdsp_rashift_acc(uint64_t * p,uint32_t ac,uint32_t shift,CPUMIPSState * env)518a2b0a27dSPhilippe Mathieu-Daudé static inline void mipsdsp_rashift_acc(uint64_t *p,
519a2b0a27dSPhilippe Mathieu-Daudé uint32_t ac,
520a2b0a27dSPhilippe Mathieu-Daudé uint32_t shift,
521a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
522a2b0a27dSPhilippe Mathieu-Daudé {
523a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempB, tempA;
524a2b0a27dSPhilippe Mathieu-Daudé
525a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.HI[ac];
526a2b0a27dSPhilippe Mathieu-Daudé tempA = env->active_tc.LO[ac];
527a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x1F;
528a2b0a27dSPhilippe Mathieu-Daudé
529a2b0a27dSPhilippe Mathieu-Daudé if (shift == 0) {
530a2b0a27dSPhilippe Mathieu-Daudé p[1] = tempB;
531a2b0a27dSPhilippe Mathieu-Daudé p[0] = tempA;
532a2b0a27dSPhilippe Mathieu-Daudé } else {
533a2b0a27dSPhilippe Mathieu-Daudé p[0] = (tempB << (64 - shift)) | (tempA >> shift);
534a2b0a27dSPhilippe Mathieu-Daudé p[1] = (int64_t)tempB >> shift;
535a2b0a27dSPhilippe Mathieu-Daudé }
536a2b0a27dSPhilippe Mathieu-Daudé }
537a2b0a27dSPhilippe Mathieu-Daudé
538a2b0a27dSPhilippe Mathieu-Daudé /* 128 bits long. p[0] is LO, p[1] is HI , p[2] is sign of HI.*/
mipsdsp_rndrashift_acc(uint64_t * p,uint32_t ac,uint32_t shift,CPUMIPSState * env)539a2b0a27dSPhilippe Mathieu-Daudé static inline void mipsdsp_rndrashift_acc(uint64_t *p,
540a2b0a27dSPhilippe Mathieu-Daudé uint32_t ac,
541a2b0a27dSPhilippe Mathieu-Daudé uint32_t shift,
542a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
543a2b0a27dSPhilippe Mathieu-Daudé {
544a2b0a27dSPhilippe Mathieu-Daudé int64_t tempB, tempA;
545a2b0a27dSPhilippe Mathieu-Daudé
546a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.HI[ac];
547a2b0a27dSPhilippe Mathieu-Daudé tempA = env->active_tc.LO[ac];
548a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
549a2b0a27dSPhilippe Mathieu-Daudé
550a2b0a27dSPhilippe Mathieu-Daudé if (shift == 0) {
551a2b0a27dSPhilippe Mathieu-Daudé p[2] = tempB >> 63;
552a2b0a27dSPhilippe Mathieu-Daudé p[1] = (tempB << 1) | (tempA >> 63);
553a2b0a27dSPhilippe Mathieu-Daudé p[0] = tempA << 1;
554a2b0a27dSPhilippe Mathieu-Daudé } else {
555a2b0a27dSPhilippe Mathieu-Daudé p[0] = (tempB << (65 - shift)) | (tempA >> (shift - 1));
556a2b0a27dSPhilippe Mathieu-Daudé p[1] = (int64_t)tempB >> (shift - 1);
557a2b0a27dSPhilippe Mathieu-Daudé if (tempB >= 0) {
558a2b0a27dSPhilippe Mathieu-Daudé p[2] = 0x0;
559a2b0a27dSPhilippe Mathieu-Daudé } else {
560a2b0a27dSPhilippe Mathieu-Daudé p[2] = ~0ull;
561a2b0a27dSPhilippe Mathieu-Daudé }
562a2b0a27dSPhilippe Mathieu-Daudé }
563a2b0a27dSPhilippe Mathieu-Daudé }
564a2b0a27dSPhilippe Mathieu-Daudé #endif
565a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_q15_q15(int32_t ac,uint16_t a,uint16_t b,CPUMIPSState * env)566a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_mul_q15_q15(int32_t ac, uint16_t a, uint16_t b,
567a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
568a2b0a27dSPhilippe Mathieu-Daudé {
569a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
570a2b0a27dSPhilippe Mathieu-Daudé
571a2b0a27dSPhilippe Mathieu-Daudé if ((a == 0x8000) && (b == 0x8000)) {
572a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFFFFFF;
573a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env);
574a2b0a27dSPhilippe Mathieu-Daudé } else {
575a2b0a27dSPhilippe Mathieu-Daudé temp = ((int16_t)a * (int16_t)b) << 1;
576a2b0a27dSPhilippe Mathieu-Daudé }
577a2b0a27dSPhilippe Mathieu-Daudé
578a2b0a27dSPhilippe Mathieu-Daudé return temp;
579a2b0a27dSPhilippe Mathieu-Daudé }
580a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_q31_q31(int32_t ac,uint32_t a,uint32_t b,CPUMIPSState * env)581a2b0a27dSPhilippe Mathieu-Daudé static inline int64_t mipsdsp_mul_q31_q31(int32_t ac, uint32_t a, uint32_t b,
582a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
583a2b0a27dSPhilippe Mathieu-Daudé {
584a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp;
585a2b0a27dSPhilippe Mathieu-Daudé
586a2b0a27dSPhilippe Mathieu-Daudé if ((a == 0x80000000) && (b == 0x80000000)) {
587a2b0a27dSPhilippe Mathieu-Daudé temp = (0x01ull << 63) - 1;
588a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env);
589a2b0a27dSPhilippe Mathieu-Daudé } else {
590a2b0a27dSPhilippe Mathieu-Daudé temp = ((int64_t)(int32_t)a * (int32_t)b) << 1;
591a2b0a27dSPhilippe Mathieu-Daudé }
592a2b0a27dSPhilippe Mathieu-Daudé
593a2b0a27dSPhilippe Mathieu-Daudé return temp;
594a2b0a27dSPhilippe Mathieu-Daudé }
595a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_u8_u8(uint8_t a,uint8_t b)596a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_mul_u8_u8(uint8_t a, uint8_t b)
597a2b0a27dSPhilippe Mathieu-Daudé {
598a2b0a27dSPhilippe Mathieu-Daudé return (uint16_t)a * (uint16_t)b;
599a2b0a27dSPhilippe Mathieu-Daudé }
600a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_mul_u8_u16(uint8_t a,uint16_t b,CPUMIPSState * env)601a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_mul_u8_u16(uint8_t a, uint16_t b,
602a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
603a2b0a27dSPhilippe Mathieu-Daudé {
604a2b0a27dSPhilippe Mathieu-Daudé uint32_t tempI;
605a2b0a27dSPhilippe Mathieu-Daudé
606a2b0a27dSPhilippe Mathieu-Daudé tempI = (uint32_t)a * (uint32_t)b;
607a2b0a27dSPhilippe Mathieu-Daudé if (tempI > 0x0000FFFF) {
608a2b0a27dSPhilippe Mathieu-Daudé tempI = 0x0000FFFF;
609a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
610a2b0a27dSPhilippe Mathieu-Daudé }
611a2b0a27dSPhilippe Mathieu-Daudé
612a2b0a27dSPhilippe Mathieu-Daudé return tempI & 0x0000FFFF;
613a2b0a27dSPhilippe Mathieu-Daudé }
614a2b0a27dSPhilippe Mathieu-Daudé
615a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
mipsdsp_mul_u32_u32(uint32_t a,uint32_t b)616a2b0a27dSPhilippe Mathieu-Daudé static inline uint64_t mipsdsp_mul_u32_u32(uint32_t a, uint32_t b)
617a2b0a27dSPhilippe Mathieu-Daudé {
618a2b0a27dSPhilippe Mathieu-Daudé return (uint64_t)a * (uint64_t)b;
619a2b0a27dSPhilippe Mathieu-Daudé }
620a2b0a27dSPhilippe Mathieu-Daudé #endif
621a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rndq15_mul_q15_q15(uint16_t a,uint16_t b,CPUMIPSState * env)622a2b0a27dSPhilippe Mathieu-Daudé static inline int16_t mipsdsp_rndq15_mul_q15_q15(uint16_t a, uint16_t b,
623a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
624a2b0a27dSPhilippe Mathieu-Daudé {
625a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
626a2b0a27dSPhilippe Mathieu-Daudé
627a2b0a27dSPhilippe Mathieu-Daudé if ((a == 0x8000) && (b == 0x8000)) {
628a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFF0000;
629a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
630a2b0a27dSPhilippe Mathieu-Daudé } else {
631a2b0a27dSPhilippe Mathieu-Daudé temp = ((int16_t)a * (int16_t)b) << 1;
632a2b0a27dSPhilippe Mathieu-Daudé temp = temp + 0x00008000;
633a2b0a27dSPhilippe Mathieu-Daudé }
634a2b0a27dSPhilippe Mathieu-Daudé
635a2b0a27dSPhilippe Mathieu-Daudé return (temp & 0xFFFF0000) >> 16;
636a2b0a27dSPhilippe Mathieu-Daudé }
637a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat16_mul_q15_q15(uint16_t a,uint16_t b,CPUMIPSState * env)638a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_sat16_mul_q15_q15(uint16_t a, uint16_t b,
639a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
640a2b0a27dSPhilippe Mathieu-Daudé {
641a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
642a2b0a27dSPhilippe Mathieu-Daudé
643a2b0a27dSPhilippe Mathieu-Daudé if ((a == 0x8000) && (b == 0x8000)) {
644a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFF0000;
645a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env);
646a2b0a27dSPhilippe Mathieu-Daudé } else {
647a2b0a27dSPhilippe Mathieu-Daudé temp = (int16_t)a * (int16_t)b;
648a2b0a27dSPhilippe Mathieu-Daudé temp = temp << 1;
649a2b0a27dSPhilippe Mathieu-Daudé }
650a2b0a27dSPhilippe Mathieu-Daudé
651a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 16) & 0x0000FFFF;
652a2b0a27dSPhilippe Mathieu-Daudé }
653a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_trunc16_sat16_round(int32_t a,CPUMIPSState * env)654a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_trunc16_sat16_round(int32_t a,
655a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
656a2b0a27dSPhilippe Mathieu-Daudé {
657a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
658a2b0a27dSPhilippe Mathieu-Daudé
659a2b0a27dSPhilippe Mathieu-Daudé
660a2b0a27dSPhilippe Mathieu-Daudé /*
661a2b0a27dSPhilippe Mathieu-Daudé * The value 0x00008000 will be added to the input Q31 value, and the code
662a2b0a27dSPhilippe Mathieu-Daudé * needs to check if the addition causes an overflow. Since a positive value
663a2b0a27dSPhilippe Mathieu-Daudé * is added, overflow can happen in one direction only.
664a2b0a27dSPhilippe Mathieu-Daudé */
665a2b0a27dSPhilippe Mathieu-Daudé if (a > 0x7FFF7FFF) {
666a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFF;
667a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
668a2b0a27dSPhilippe Mathieu-Daudé } else {
669a2b0a27dSPhilippe Mathieu-Daudé temp = ((a + 0x8000) >> 16) & 0xFFFF;
670a2b0a27dSPhilippe Mathieu-Daudé }
671a2b0a27dSPhilippe Mathieu-Daudé
672a2b0a27dSPhilippe Mathieu-Daudé return temp;
673a2b0a27dSPhilippe Mathieu-Daudé }
674a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat8_reduce_precision(uint16_t a,CPUMIPSState * env)675a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_sat8_reduce_precision(uint16_t a,
676a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
677a2b0a27dSPhilippe Mathieu-Daudé {
678a2b0a27dSPhilippe Mathieu-Daudé uint16_t mag;
679a2b0a27dSPhilippe Mathieu-Daudé uint32_t sign;
680a2b0a27dSPhilippe Mathieu-Daudé
681a2b0a27dSPhilippe Mathieu-Daudé sign = (a >> 15) & 0x01;
682a2b0a27dSPhilippe Mathieu-Daudé mag = a & 0x7FFF;
683a2b0a27dSPhilippe Mathieu-Daudé
684a2b0a27dSPhilippe Mathieu-Daudé if (sign == 0) {
685a2b0a27dSPhilippe Mathieu-Daudé if (mag > 0x7F80) {
686a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
687a2b0a27dSPhilippe Mathieu-Daudé return 0xFF;
688a2b0a27dSPhilippe Mathieu-Daudé } else {
689a2b0a27dSPhilippe Mathieu-Daudé return (mag >> 7) & 0xFFFF;
690a2b0a27dSPhilippe Mathieu-Daudé }
691a2b0a27dSPhilippe Mathieu-Daudé } else {
692a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
693a2b0a27dSPhilippe Mathieu-Daudé return 0x00;
694a2b0a27dSPhilippe Mathieu-Daudé }
695a2b0a27dSPhilippe Mathieu-Daudé }
696a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_lshift8(uint8_t a,uint8_t s,CPUMIPSState * env)697a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_lshift8(uint8_t a, uint8_t s, CPUMIPSState *env)
698a2b0a27dSPhilippe Mathieu-Daudé {
699a2b0a27dSPhilippe Mathieu-Daudé uint8_t discard;
700a2b0a27dSPhilippe Mathieu-Daudé
701a2b0a27dSPhilippe Mathieu-Daudé if (s != 0) {
702a2b0a27dSPhilippe Mathieu-Daudé discard = a >> (8 - s);
703a2b0a27dSPhilippe Mathieu-Daudé
704a2b0a27dSPhilippe Mathieu-Daudé if (discard != 0x00) {
705a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
706a2b0a27dSPhilippe Mathieu-Daudé }
707a2b0a27dSPhilippe Mathieu-Daudé }
708a2b0a27dSPhilippe Mathieu-Daudé return a << s;
709a2b0a27dSPhilippe Mathieu-Daudé }
710a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_lshift16(uint16_t a,uint8_t s,CPUMIPSState * env)711a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_lshift16(uint16_t a, uint8_t s,
712a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
713a2b0a27dSPhilippe Mathieu-Daudé {
714a2b0a27dSPhilippe Mathieu-Daudé uint16_t discard;
715a2b0a27dSPhilippe Mathieu-Daudé
716a2b0a27dSPhilippe Mathieu-Daudé if (s != 0) {
717a2b0a27dSPhilippe Mathieu-Daudé discard = (int16_t)a >> (15 - s);
718a2b0a27dSPhilippe Mathieu-Daudé
719a2b0a27dSPhilippe Mathieu-Daudé if ((discard != 0x0000) && (discard != 0xFFFF)) {
720a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
721a2b0a27dSPhilippe Mathieu-Daudé }
722a2b0a27dSPhilippe Mathieu-Daudé }
723a2b0a27dSPhilippe Mathieu-Daudé return a << s;
724a2b0a27dSPhilippe Mathieu-Daudé }
725a2b0a27dSPhilippe Mathieu-Daudé
726a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
mipsdsp_lshift32(uint32_t a,uint8_t s,CPUMIPSState * env)727a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_lshift32(uint32_t a, uint8_t s,
728a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
729a2b0a27dSPhilippe Mathieu-Daudé {
730a2b0a27dSPhilippe Mathieu-Daudé uint32_t discard;
731a2b0a27dSPhilippe Mathieu-Daudé
732a2b0a27dSPhilippe Mathieu-Daudé if (s == 0) {
733a2b0a27dSPhilippe Mathieu-Daudé return a;
734a2b0a27dSPhilippe Mathieu-Daudé } else {
735a2b0a27dSPhilippe Mathieu-Daudé discard = (int32_t)a >> (31 - (s - 1));
736a2b0a27dSPhilippe Mathieu-Daudé
737a2b0a27dSPhilippe Mathieu-Daudé if ((discard != 0x00000000) && (discard != 0xFFFFFFFF)) {
738a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
739a2b0a27dSPhilippe Mathieu-Daudé }
740a2b0a27dSPhilippe Mathieu-Daudé return a << s;
741a2b0a27dSPhilippe Mathieu-Daudé }
742a2b0a27dSPhilippe Mathieu-Daudé }
743a2b0a27dSPhilippe Mathieu-Daudé #endif
744a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat16_lshift(uint16_t a,uint8_t s,CPUMIPSState * env)745a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_sat16_lshift(uint16_t a, uint8_t s,
746a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
747a2b0a27dSPhilippe Mathieu-Daudé {
748a2b0a27dSPhilippe Mathieu-Daudé uint8_t sign;
749a2b0a27dSPhilippe Mathieu-Daudé uint16_t discard;
750a2b0a27dSPhilippe Mathieu-Daudé
751a2b0a27dSPhilippe Mathieu-Daudé if (s == 0) {
752a2b0a27dSPhilippe Mathieu-Daudé return a;
753a2b0a27dSPhilippe Mathieu-Daudé } else {
754a2b0a27dSPhilippe Mathieu-Daudé sign = (a >> 15) & 0x01;
755a2b0a27dSPhilippe Mathieu-Daudé if (sign != 0) {
756a2b0a27dSPhilippe Mathieu-Daudé discard = (((0x01 << (16 - s)) - 1) << s) |
757a2b0a27dSPhilippe Mathieu-Daudé ((a >> (14 - (s - 1))) & ((0x01 << s) - 1));
758a2b0a27dSPhilippe Mathieu-Daudé } else {
759a2b0a27dSPhilippe Mathieu-Daudé discard = a >> (14 - (s - 1));
760a2b0a27dSPhilippe Mathieu-Daudé }
761a2b0a27dSPhilippe Mathieu-Daudé
762a2b0a27dSPhilippe Mathieu-Daudé if ((discard != 0x0000) && (discard != 0xFFFF)) {
763a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
764a2b0a27dSPhilippe Mathieu-Daudé return (sign == 0) ? 0x7FFF : 0x8000;
765a2b0a27dSPhilippe Mathieu-Daudé } else {
766a2b0a27dSPhilippe Mathieu-Daudé return a << s;
767a2b0a27dSPhilippe Mathieu-Daudé }
768a2b0a27dSPhilippe Mathieu-Daudé }
769a2b0a27dSPhilippe Mathieu-Daudé }
770a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat32_lshift(uint32_t a,uint8_t s,CPUMIPSState * env)771a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_sat32_lshift(uint32_t a, uint8_t s,
772a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
773a2b0a27dSPhilippe Mathieu-Daudé {
774a2b0a27dSPhilippe Mathieu-Daudé uint8_t sign;
775a2b0a27dSPhilippe Mathieu-Daudé uint32_t discard;
776a2b0a27dSPhilippe Mathieu-Daudé
777a2b0a27dSPhilippe Mathieu-Daudé if (s == 0) {
778a2b0a27dSPhilippe Mathieu-Daudé return a;
779a2b0a27dSPhilippe Mathieu-Daudé } else {
780a2b0a27dSPhilippe Mathieu-Daudé sign = (a >> 31) & 0x01;
781a2b0a27dSPhilippe Mathieu-Daudé if (sign != 0) {
782a2b0a27dSPhilippe Mathieu-Daudé discard = (((0x01 << (32 - s)) - 1) << s) |
783a2b0a27dSPhilippe Mathieu-Daudé ((a >> (30 - (s - 1))) & ((0x01 << s) - 1));
784a2b0a27dSPhilippe Mathieu-Daudé } else {
785a2b0a27dSPhilippe Mathieu-Daudé discard = a >> (30 - (s - 1));
786a2b0a27dSPhilippe Mathieu-Daudé }
787a2b0a27dSPhilippe Mathieu-Daudé
788a2b0a27dSPhilippe Mathieu-Daudé if ((discard != 0x00000000) && (discard != 0xFFFFFFFF)) {
789a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 22, env);
790a2b0a27dSPhilippe Mathieu-Daudé return (sign == 0) ? 0x7FFFFFFF : 0x80000000;
791a2b0a27dSPhilippe Mathieu-Daudé } else {
792a2b0a27dSPhilippe Mathieu-Daudé return a << s;
793a2b0a27dSPhilippe Mathieu-Daudé }
794a2b0a27dSPhilippe Mathieu-Daudé }
795a2b0a27dSPhilippe Mathieu-Daudé }
796a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rnd8_rashift(uint8_t a,uint8_t s)797a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_rnd8_rashift(uint8_t a, uint8_t s)
798a2b0a27dSPhilippe Mathieu-Daudé {
799a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
800a2b0a27dSPhilippe Mathieu-Daudé
801a2b0a27dSPhilippe Mathieu-Daudé if (s == 0) {
802a2b0a27dSPhilippe Mathieu-Daudé temp = (uint32_t)a << 1;
803a2b0a27dSPhilippe Mathieu-Daudé } else {
804a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)(int8_t)a >> (s - 1);
805a2b0a27dSPhilippe Mathieu-Daudé }
806a2b0a27dSPhilippe Mathieu-Daudé
807a2b0a27dSPhilippe Mathieu-Daudé return (temp + 1) >> 1;
808a2b0a27dSPhilippe Mathieu-Daudé }
809a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rnd16_rashift(uint16_t a,uint8_t s)810a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_rnd16_rashift(uint16_t a, uint8_t s)
811a2b0a27dSPhilippe Mathieu-Daudé {
812a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
813a2b0a27dSPhilippe Mathieu-Daudé
814a2b0a27dSPhilippe Mathieu-Daudé if (s == 0) {
815a2b0a27dSPhilippe Mathieu-Daudé temp = (uint32_t)a << 1;
816a2b0a27dSPhilippe Mathieu-Daudé } else {
817a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)(int16_t)a >> (s - 1);
818a2b0a27dSPhilippe Mathieu-Daudé }
819a2b0a27dSPhilippe Mathieu-Daudé
820a2b0a27dSPhilippe Mathieu-Daudé return (temp + 1) >> 1;
821a2b0a27dSPhilippe Mathieu-Daudé }
822a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rnd32_rashift(uint32_t a,uint8_t s)823a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_rnd32_rashift(uint32_t a, uint8_t s)
824a2b0a27dSPhilippe Mathieu-Daudé {
825a2b0a27dSPhilippe Mathieu-Daudé int64_t temp;
826a2b0a27dSPhilippe Mathieu-Daudé
827a2b0a27dSPhilippe Mathieu-Daudé if (s == 0) {
828a2b0a27dSPhilippe Mathieu-Daudé temp = (uint64_t)a << 1;
829a2b0a27dSPhilippe Mathieu-Daudé } else {
830a2b0a27dSPhilippe Mathieu-Daudé temp = (int64_t)(int32_t)a >> (s - 1);
831a2b0a27dSPhilippe Mathieu-Daudé }
832a2b0a27dSPhilippe Mathieu-Daudé temp += 1;
833a2b0a27dSPhilippe Mathieu-Daudé
834a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFFFFFFull;
835a2b0a27dSPhilippe Mathieu-Daudé }
836a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sub_i16(int16_t a,int16_t b,CPUMIPSState * env)837a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_sub_i16(int16_t a, int16_t b, CPUMIPSState *env)
838a2b0a27dSPhilippe Mathieu-Daudé {
839a2b0a27dSPhilippe Mathieu-Daudé int16_t temp;
840a2b0a27dSPhilippe Mathieu-Daudé
841a2b0a27dSPhilippe Mathieu-Daudé temp = a - b;
842a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_SUB(a, b, temp, 0x8000)) {
843a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
844a2b0a27dSPhilippe Mathieu-Daudé }
845a2b0a27dSPhilippe Mathieu-Daudé
846a2b0a27dSPhilippe Mathieu-Daudé return temp;
847a2b0a27dSPhilippe Mathieu-Daudé }
848a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat16_sub(int16_t a,int16_t b,CPUMIPSState * env)849a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_sat16_sub(int16_t a, int16_t b,
850a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
851a2b0a27dSPhilippe Mathieu-Daudé {
852a2b0a27dSPhilippe Mathieu-Daudé int16_t temp;
853a2b0a27dSPhilippe Mathieu-Daudé
854a2b0a27dSPhilippe Mathieu-Daudé temp = a - b;
855a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_SUB(a, b, temp, 0x8000)) {
856a2b0a27dSPhilippe Mathieu-Daudé if (a >= 0) {
857a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFF;
858a2b0a27dSPhilippe Mathieu-Daudé } else {
859a2b0a27dSPhilippe Mathieu-Daudé temp = 0x8000;
860a2b0a27dSPhilippe Mathieu-Daudé }
861a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
862a2b0a27dSPhilippe Mathieu-Daudé }
863a2b0a27dSPhilippe Mathieu-Daudé
864a2b0a27dSPhilippe Mathieu-Daudé return temp;
865a2b0a27dSPhilippe Mathieu-Daudé }
866a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sat32_sub(int32_t a,int32_t b,CPUMIPSState * env)867a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_sat32_sub(int32_t a, int32_t b,
868a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
869a2b0a27dSPhilippe Mathieu-Daudé {
870a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
871a2b0a27dSPhilippe Mathieu-Daudé
872a2b0a27dSPhilippe Mathieu-Daudé temp = a - b;
873a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_SUB(a, b, temp, 0x80000000)) {
874a2b0a27dSPhilippe Mathieu-Daudé if (a >= 0) {
875a2b0a27dSPhilippe Mathieu-Daudé temp = 0x7FFFFFFF;
876a2b0a27dSPhilippe Mathieu-Daudé } else {
877a2b0a27dSPhilippe Mathieu-Daudé temp = 0x80000000;
878a2b0a27dSPhilippe Mathieu-Daudé }
879a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
880a2b0a27dSPhilippe Mathieu-Daudé }
881a2b0a27dSPhilippe Mathieu-Daudé
882a2b0a27dSPhilippe Mathieu-Daudé return temp & 0xFFFFFFFFull;
883a2b0a27dSPhilippe Mathieu-Daudé }
884a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rshift1_sub_q16(int16_t a,int16_t b)885a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_rshift1_sub_q16(int16_t a, int16_t b)
886a2b0a27dSPhilippe Mathieu-Daudé {
887a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
888a2b0a27dSPhilippe Mathieu-Daudé
889a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)a - (int32_t)b;
890a2b0a27dSPhilippe Mathieu-Daudé
891a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0x0000FFFF;
892a2b0a27dSPhilippe Mathieu-Daudé }
893a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rrshift1_sub_q16(int16_t a,int16_t b)894a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_rrshift1_sub_q16(int16_t a, int16_t b)
895a2b0a27dSPhilippe Mathieu-Daudé {
896a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
897a2b0a27dSPhilippe Mathieu-Daudé
898a2b0a27dSPhilippe Mathieu-Daudé temp = (int32_t)a - (int32_t)b;
899a2b0a27dSPhilippe Mathieu-Daudé temp += 1;
900a2b0a27dSPhilippe Mathieu-Daudé
901a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0x0000FFFF;
902a2b0a27dSPhilippe Mathieu-Daudé }
903a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rshift1_sub_q32(int32_t a,int32_t b)904a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_rshift1_sub_q32(int32_t a, int32_t b)
905a2b0a27dSPhilippe Mathieu-Daudé {
906a2b0a27dSPhilippe Mathieu-Daudé int64_t temp;
907a2b0a27dSPhilippe Mathieu-Daudé
908a2b0a27dSPhilippe Mathieu-Daudé temp = (int64_t)a - (int64_t)b;
909a2b0a27dSPhilippe Mathieu-Daudé
910a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFFFFFFull;
911a2b0a27dSPhilippe Mathieu-Daudé }
912a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_rrshift1_sub_q32(int32_t a,int32_t b)913a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_rrshift1_sub_q32(int32_t a, int32_t b)
914a2b0a27dSPhilippe Mathieu-Daudé {
915a2b0a27dSPhilippe Mathieu-Daudé int64_t temp;
916a2b0a27dSPhilippe Mathieu-Daudé
917a2b0a27dSPhilippe Mathieu-Daudé temp = (int64_t)a - (int64_t)b;
918a2b0a27dSPhilippe Mathieu-Daudé temp += 1;
919a2b0a27dSPhilippe Mathieu-Daudé
920a2b0a27dSPhilippe Mathieu-Daudé return (temp >> 1) & 0xFFFFFFFFull;
921a2b0a27dSPhilippe Mathieu-Daudé }
922a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sub_u16_u16(uint16_t a,uint16_t b,CPUMIPSState * env)923a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_sub_u16_u16(uint16_t a, uint16_t b,
924a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
925a2b0a27dSPhilippe Mathieu-Daudé {
926a2b0a27dSPhilippe Mathieu-Daudé uint8_t temp16;
927a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
928a2b0a27dSPhilippe Mathieu-Daudé
929a2b0a27dSPhilippe Mathieu-Daudé temp = (uint32_t)a - (uint32_t)b;
930a2b0a27dSPhilippe Mathieu-Daudé temp16 = (temp >> 16) & 0x01;
931a2b0a27dSPhilippe Mathieu-Daudé if (temp16 == 1) {
932a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
933a2b0a27dSPhilippe Mathieu-Daudé }
934a2b0a27dSPhilippe Mathieu-Daudé return temp & 0x0000FFFF;
935a2b0a27dSPhilippe Mathieu-Daudé }
936a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_satu16_sub_u16_u16(uint16_t a,uint16_t b,CPUMIPSState * env)937a2b0a27dSPhilippe Mathieu-Daudé static inline uint16_t mipsdsp_satu16_sub_u16_u16(uint16_t a, uint16_t b,
938a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
939a2b0a27dSPhilippe Mathieu-Daudé {
940a2b0a27dSPhilippe Mathieu-Daudé uint8_t temp16;
941a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
942a2b0a27dSPhilippe Mathieu-Daudé
943a2b0a27dSPhilippe Mathieu-Daudé temp = (uint32_t)a - (uint32_t)b;
944a2b0a27dSPhilippe Mathieu-Daudé temp16 = (temp >> 16) & 0x01;
945a2b0a27dSPhilippe Mathieu-Daudé
946a2b0a27dSPhilippe Mathieu-Daudé if (temp16 == 1) {
947a2b0a27dSPhilippe Mathieu-Daudé temp = 0x0000;
948a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
949a2b0a27dSPhilippe Mathieu-Daudé }
950a2b0a27dSPhilippe Mathieu-Daudé
951a2b0a27dSPhilippe Mathieu-Daudé return temp & 0x0000FFFF;
952a2b0a27dSPhilippe Mathieu-Daudé }
953a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_sub_u8(uint8_t a,uint8_t b,CPUMIPSState * env)954a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_sub_u8(uint8_t a, uint8_t b, CPUMIPSState *env)
955a2b0a27dSPhilippe Mathieu-Daudé {
956a2b0a27dSPhilippe Mathieu-Daudé uint8_t temp8;
957a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
958a2b0a27dSPhilippe Mathieu-Daudé
959a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a - (uint16_t)b;
960a2b0a27dSPhilippe Mathieu-Daudé temp8 = (temp >> 8) & 0x01;
961a2b0a27dSPhilippe Mathieu-Daudé if (temp8 == 1) {
962a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
963a2b0a27dSPhilippe Mathieu-Daudé }
964a2b0a27dSPhilippe Mathieu-Daudé
965a2b0a27dSPhilippe Mathieu-Daudé return temp & 0x00FF;
966a2b0a27dSPhilippe Mathieu-Daudé }
967a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_satu8_sub(uint8_t a,uint8_t b,CPUMIPSState * env)968a2b0a27dSPhilippe Mathieu-Daudé static inline uint8_t mipsdsp_satu8_sub(uint8_t a, uint8_t b, CPUMIPSState *env)
969a2b0a27dSPhilippe Mathieu-Daudé {
970a2b0a27dSPhilippe Mathieu-Daudé uint8_t temp8;
971a2b0a27dSPhilippe Mathieu-Daudé uint16_t temp;
972a2b0a27dSPhilippe Mathieu-Daudé
973a2b0a27dSPhilippe Mathieu-Daudé temp = (uint16_t)a - (uint16_t)b;
974a2b0a27dSPhilippe Mathieu-Daudé temp8 = (temp >> 8) & 0x01;
975a2b0a27dSPhilippe Mathieu-Daudé if (temp8 == 1) {
976a2b0a27dSPhilippe Mathieu-Daudé temp = 0x00;
977a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
978a2b0a27dSPhilippe Mathieu-Daudé }
979a2b0a27dSPhilippe Mathieu-Daudé
980a2b0a27dSPhilippe Mathieu-Daudé return temp & 0x00FF;
981a2b0a27dSPhilippe Mathieu-Daudé }
982a2b0a27dSPhilippe Mathieu-Daudé
983a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
mipsdsp_sub32(int32_t a,int32_t b,CPUMIPSState * env)984a2b0a27dSPhilippe Mathieu-Daudé static inline uint32_t mipsdsp_sub32(int32_t a, int32_t b, CPUMIPSState *env)
985a2b0a27dSPhilippe Mathieu-Daudé {
986a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
987a2b0a27dSPhilippe Mathieu-Daudé
988a2b0a27dSPhilippe Mathieu-Daudé temp = a - b;
989a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_SUB(a, b, temp, 0x80000000)) {
990a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
991a2b0a27dSPhilippe Mathieu-Daudé }
992a2b0a27dSPhilippe Mathieu-Daudé
993a2b0a27dSPhilippe Mathieu-Daudé return temp;
994a2b0a27dSPhilippe Mathieu-Daudé }
995a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_add_i32(int32_t a,int32_t b,CPUMIPSState * env)996a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_add_i32(int32_t a, int32_t b, CPUMIPSState *env)
997a2b0a27dSPhilippe Mathieu-Daudé {
998a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
999a2b0a27dSPhilippe Mathieu-Daudé
1000a2b0a27dSPhilippe Mathieu-Daudé temp = a + b;
1001a2b0a27dSPhilippe Mathieu-Daudé
1002a2b0a27dSPhilippe Mathieu-Daudé if (MIPSDSP_OVERFLOW_ADD(a, b, temp, 0x80000000)) {
1003a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
1004a2b0a27dSPhilippe Mathieu-Daudé }
1005a2b0a27dSPhilippe Mathieu-Daudé
1006a2b0a27dSPhilippe Mathieu-Daudé return temp;
1007a2b0a27dSPhilippe Mathieu-Daudé }
1008a2b0a27dSPhilippe Mathieu-Daudé #endif
1009a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_cmp_eq(int32_t a,int32_t b)1010a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_cmp_eq(int32_t a, int32_t b)
1011a2b0a27dSPhilippe Mathieu-Daudé {
1012a2b0a27dSPhilippe Mathieu-Daudé return a == b;
1013a2b0a27dSPhilippe Mathieu-Daudé }
1014a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_cmp_le(int32_t a,int32_t b)1015a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_cmp_le(int32_t a, int32_t b)
1016a2b0a27dSPhilippe Mathieu-Daudé {
1017a2b0a27dSPhilippe Mathieu-Daudé return a <= b;
1018a2b0a27dSPhilippe Mathieu-Daudé }
1019a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_cmp_lt(int32_t a,int32_t b)1020a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_cmp_lt(int32_t a, int32_t b)
1021a2b0a27dSPhilippe Mathieu-Daudé {
1022a2b0a27dSPhilippe Mathieu-Daudé return a < b;
1023a2b0a27dSPhilippe Mathieu-Daudé }
1024a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_cmpu_eq(uint32_t a,uint32_t b)1025a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_cmpu_eq(uint32_t a, uint32_t b)
1026a2b0a27dSPhilippe Mathieu-Daudé {
1027a2b0a27dSPhilippe Mathieu-Daudé return a == b;
1028a2b0a27dSPhilippe Mathieu-Daudé }
1029a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_cmpu_le(uint32_t a,uint32_t b)1030a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_cmpu_le(uint32_t a, uint32_t b)
1031a2b0a27dSPhilippe Mathieu-Daudé {
1032a2b0a27dSPhilippe Mathieu-Daudé return a <= b;
1033a2b0a27dSPhilippe Mathieu-Daudé }
1034a2b0a27dSPhilippe Mathieu-Daudé
mipsdsp_cmpu_lt(uint32_t a,uint32_t b)1035a2b0a27dSPhilippe Mathieu-Daudé static inline int32_t mipsdsp_cmpu_lt(uint32_t a, uint32_t b)
1036a2b0a27dSPhilippe Mathieu-Daudé {
1037a2b0a27dSPhilippe Mathieu-Daudé return a < b;
1038a2b0a27dSPhilippe Mathieu-Daudé }
1039a2b0a27dSPhilippe Mathieu-Daudé /*** MIPS DSP internal functions end ***/
1040a2b0a27dSPhilippe Mathieu-Daudé
1041a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_LHI 0xFFFFFFFF00000000ull
1042a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_LLO 0x00000000FFFFFFFFull
1043a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_HI 0xFFFF0000
1044a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_LO 0x0000FFFF
1045a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_Q3 0xFF000000
1046a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_Q2 0x00FF0000
1047a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_Q1 0x0000FF00
1048a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_Q0 0x000000FF
1049a2b0a27dSPhilippe Mathieu-Daudé
1050a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_SPLIT32_8(num, a, b, c, d) \
1051a2b0a27dSPhilippe Mathieu-Daudé do { \
1052a2b0a27dSPhilippe Mathieu-Daudé a = ((num) >> 24) & MIPSDSP_Q0; \
1053a2b0a27dSPhilippe Mathieu-Daudé b = ((num) >> 16) & MIPSDSP_Q0; \
1054a2b0a27dSPhilippe Mathieu-Daudé c = ((num) >> 8) & MIPSDSP_Q0; \
1055a2b0a27dSPhilippe Mathieu-Daudé d = (num) & MIPSDSP_Q0; \
1056a2b0a27dSPhilippe Mathieu-Daudé } while (0)
1057a2b0a27dSPhilippe Mathieu-Daudé
1058a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_SPLIT32_16(num, a, b) \
1059a2b0a27dSPhilippe Mathieu-Daudé do { \
1060a2b0a27dSPhilippe Mathieu-Daudé a = ((num) >> 16) & MIPSDSP_LO; \
1061a2b0a27dSPhilippe Mathieu-Daudé b = (num) & MIPSDSP_LO; \
1062a2b0a27dSPhilippe Mathieu-Daudé } while (0)
1063a2b0a27dSPhilippe Mathieu-Daudé
1064a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_RETURN32_8(a, b, c, d) ((target_long)(int32_t) \
1065a2b0a27dSPhilippe Mathieu-Daudé (((uint32_t)(a) << 24) | \
1066a2b0a27dSPhilippe Mathieu-Daudé ((uint32_t)(b) << 16) | \
1067a2b0a27dSPhilippe Mathieu-Daudé ((uint32_t)(c) << 8) | \
1068a2b0a27dSPhilippe Mathieu-Daudé ((uint32_t)(d) & 0xFF)))
1069a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_RETURN32_16(a, b) ((target_long)(int32_t) \
1070a2b0a27dSPhilippe Mathieu-Daudé (((uint32_t)(a) << 16) | \
1071a2b0a27dSPhilippe Mathieu-Daudé ((uint32_t)(b) & 0xFFFF)))
1072a2b0a27dSPhilippe Mathieu-Daudé
1073a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
1074a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_SPLIT64_16(num, a, b, c, d) \
1075a2b0a27dSPhilippe Mathieu-Daudé do { \
1076a2b0a27dSPhilippe Mathieu-Daudé a = ((num) >> 48) & MIPSDSP_LO; \
1077a2b0a27dSPhilippe Mathieu-Daudé b = ((num) >> 32) & MIPSDSP_LO; \
1078a2b0a27dSPhilippe Mathieu-Daudé c = ((num) >> 16) & MIPSDSP_LO; \
1079a2b0a27dSPhilippe Mathieu-Daudé d = (num) & MIPSDSP_LO; \
1080a2b0a27dSPhilippe Mathieu-Daudé } while (0)
1081a2b0a27dSPhilippe Mathieu-Daudé
1082a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_SPLIT64_32(num, a, b) \
1083a2b0a27dSPhilippe Mathieu-Daudé do { \
1084a2b0a27dSPhilippe Mathieu-Daudé a = ((num) >> 32) & MIPSDSP_LLO; \
1085a2b0a27dSPhilippe Mathieu-Daudé b = (num) & MIPSDSP_LLO; \
1086a2b0a27dSPhilippe Mathieu-Daudé } while (0)
1087a2b0a27dSPhilippe Mathieu-Daudé
1088a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_RETURN64_16(a, b, c, d) (((uint64_t)(a) << 48) | \
1089a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)(b) << 32) | \
1090a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)(c) << 16) | \
1091a2b0a27dSPhilippe Mathieu-Daudé (uint64_t)(d))
1092a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP_RETURN64_32(a, b) (((uint64_t)(a) << 32) | (uint64_t)(b))
1093a2b0a27dSPhilippe Mathieu-Daudé #endif
1094a2b0a27dSPhilippe Mathieu-Daudé
1095a2b0a27dSPhilippe Mathieu-Daudé /** DSP Arithmetic Sub-class insns **/
1096a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP32_UNOP_ENV(name, func, element) \
1097a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rt, CPUMIPSState *env) \
1098a2b0a27dSPhilippe Mathieu-Daudé { \
1099a2b0a27dSPhilippe Mathieu-Daudé DSP32Value dt; \
1100a2b0a27dSPhilippe Mathieu-Daudé unsigned int i; \
1101a2b0a27dSPhilippe Mathieu-Daudé \
1102a2b0a27dSPhilippe Mathieu-Daudé dt.sw[0] = rt; \
1103a2b0a27dSPhilippe Mathieu-Daudé \
1104a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < ARRAY_SIZE(dt.element); i++) { \
1105a2b0a27dSPhilippe Mathieu-Daudé dt.element[i] = mipsdsp_##func(dt.element[i], env); \
1106a2b0a27dSPhilippe Mathieu-Daudé } \
1107a2b0a27dSPhilippe Mathieu-Daudé \
1108a2b0a27dSPhilippe Mathieu-Daudé return (target_long)dt.sw[0]; \
1109a2b0a27dSPhilippe Mathieu-Daudé }
1110a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_UNOP_ENV(absq_s_ph, sat_abs16, sh)
1111a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_UNOP_ENV(absq_s_qb, sat_abs8, sb)
1112a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_UNOP_ENV(absq_s_w, sat_abs32, sw)
1113a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP32_UNOP_ENV
1114a2b0a27dSPhilippe Mathieu-Daudé
1115a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
1116a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP64_UNOP_ENV(name, func, element) \
1117a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rt, CPUMIPSState *env) \
1118a2b0a27dSPhilippe Mathieu-Daudé { \
1119a2b0a27dSPhilippe Mathieu-Daudé DSP64Value dt; \
1120a2b0a27dSPhilippe Mathieu-Daudé unsigned int i; \
1121a2b0a27dSPhilippe Mathieu-Daudé \
1122a2b0a27dSPhilippe Mathieu-Daudé dt.sl[0] = rt; \
1123a2b0a27dSPhilippe Mathieu-Daudé \
1124a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < ARRAY_SIZE(dt.element); i++) { \
1125a2b0a27dSPhilippe Mathieu-Daudé dt.element[i] = mipsdsp_##func(dt.element[i], env); \
1126a2b0a27dSPhilippe Mathieu-Daudé } \
1127a2b0a27dSPhilippe Mathieu-Daudé \
1128a2b0a27dSPhilippe Mathieu-Daudé return dt.sl[0]; \
1129a2b0a27dSPhilippe Mathieu-Daudé }
1130a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_UNOP_ENV(absq_s_ob, sat_abs8, sb)
1131a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_UNOP_ENV(absq_s_qh, sat_abs16, sh)
1132a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_UNOP_ENV(absq_s_pw, sat_abs32, sw)
1133a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP64_UNOP_ENV
1134a2b0a27dSPhilippe Mathieu-Daudé #endif
1135a2b0a27dSPhilippe Mathieu-Daudé
1136a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP32_BINOP(name, func, element) \
1137a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt) \
1138a2b0a27dSPhilippe Mathieu-Daudé { \
1139a2b0a27dSPhilippe Mathieu-Daudé DSP32Value ds, dt; \
1140a2b0a27dSPhilippe Mathieu-Daudé unsigned int i; \
1141a2b0a27dSPhilippe Mathieu-Daudé \
1142a2b0a27dSPhilippe Mathieu-Daudé ds.sw[0] = rs; \
1143a2b0a27dSPhilippe Mathieu-Daudé dt.sw[0] = rt; \
1144a2b0a27dSPhilippe Mathieu-Daudé \
1145a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < ARRAY_SIZE(ds.element); i++) { \
1146a2b0a27dSPhilippe Mathieu-Daudé ds.element[i] = mipsdsp_##func(ds.element[i], dt.element[i]); \
1147a2b0a27dSPhilippe Mathieu-Daudé } \
1148a2b0a27dSPhilippe Mathieu-Daudé \
1149a2b0a27dSPhilippe Mathieu-Daudé return (target_long)ds.sw[0]; \
1150a2b0a27dSPhilippe Mathieu-Daudé }
1151a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(addqh_ph, rshift1_add_q16, sh);
1152a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(addqh_r_ph, rrshift1_add_q16, sh);
1153a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(addqh_r_w, rrshift1_add_q32, sw);
1154a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(addqh_w, rshift1_add_q32, sw);
1155a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(adduh_qb, rshift1_add_u8, ub);
1156a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(adduh_r_qb, rrshift1_add_u8, ub);
1157a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(subqh_ph, rshift1_sub_q16, sh);
1158a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(subqh_r_ph, rrshift1_sub_q16, sh);
1159a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(subqh_r_w, rrshift1_sub_q32, sw);
1160a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP(subqh_w, rshift1_sub_q32, sw);
1161a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP32_BINOP
1162a2b0a27dSPhilippe Mathieu-Daudé
1163a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP32_BINOP_ENV(name, func, element) \
1164a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
1165a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1166a2b0a27dSPhilippe Mathieu-Daudé { \
1167a2b0a27dSPhilippe Mathieu-Daudé DSP32Value ds, dt; \
1168a2b0a27dSPhilippe Mathieu-Daudé unsigned int i; \
1169a2b0a27dSPhilippe Mathieu-Daudé \
1170a2b0a27dSPhilippe Mathieu-Daudé ds.sw[0] = rs; \
1171a2b0a27dSPhilippe Mathieu-Daudé dt.sw[0] = rt; \
1172a2b0a27dSPhilippe Mathieu-Daudé \
1173a2b0a27dSPhilippe Mathieu-Daudé for (i = 0 ; i < ARRAY_SIZE(ds.element); i++) { \
1174a2b0a27dSPhilippe Mathieu-Daudé ds.element[i] = mipsdsp_##func(ds.element[i], dt.element[i], env); \
1175a2b0a27dSPhilippe Mathieu-Daudé } \
1176a2b0a27dSPhilippe Mathieu-Daudé \
1177a2b0a27dSPhilippe Mathieu-Daudé return (target_long)ds.sw[0]; \
1178a2b0a27dSPhilippe Mathieu-Daudé }
1179a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addq_ph, add_i16, sh)
1180a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addq_s_ph, sat_add_i16, sh)
1181a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addq_s_w, sat_add_i32, sw);
1182a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addu_ph, add_u16, sh)
1183a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addu_qb, add_u8, ub);
1184a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addu_s_ph, sat_add_u16, sh)
1185a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(addu_s_qb, sat_add_u8, ub);
1186a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subq_ph, sub_i16, sh);
1187a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subq_s_ph, sat16_sub, sh);
1188a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subq_s_w, sat32_sub, sw);
1189a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subu_ph, sub_u16_u16, sh);
1190a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subu_qb, sub_u8, ub);
1191a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subu_s_ph, satu16_sub_u16_u16, sh);
1192a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP32_BINOP_ENV(subu_s_qb, satu8_sub, ub);
1193a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP32_BINOP_ENV
1194a2b0a27dSPhilippe Mathieu-Daudé
1195a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
1196a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP64_BINOP(name, func, element) \
1197a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt) \
1198a2b0a27dSPhilippe Mathieu-Daudé { \
1199a2b0a27dSPhilippe Mathieu-Daudé DSP64Value ds, dt; \
1200a2b0a27dSPhilippe Mathieu-Daudé unsigned int i; \
1201a2b0a27dSPhilippe Mathieu-Daudé \
1202a2b0a27dSPhilippe Mathieu-Daudé ds.sl[0] = rs; \
1203a2b0a27dSPhilippe Mathieu-Daudé dt.sl[0] = rt; \
1204a2b0a27dSPhilippe Mathieu-Daudé \
1205a2b0a27dSPhilippe Mathieu-Daudé for (i = 0 ; i < ARRAY_SIZE(ds.element); i++) { \
1206a2b0a27dSPhilippe Mathieu-Daudé ds.element[i] = mipsdsp_##func(ds.element[i], dt.element[i]); \
1207a2b0a27dSPhilippe Mathieu-Daudé } \
1208a2b0a27dSPhilippe Mathieu-Daudé \
1209a2b0a27dSPhilippe Mathieu-Daudé return ds.sl[0]; \
1210a2b0a27dSPhilippe Mathieu-Daudé }
1211a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP(adduh_ob, rshift1_add_u8, ub);
1212a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP(adduh_r_ob, rrshift1_add_u8, ub);
1213a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP(subuh_ob, rshift1_sub_u8, ub);
1214a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP(subuh_r_ob, rrshift1_sub_u8, ub);
1215a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP64_BINOP
1216a2b0a27dSPhilippe Mathieu-Daudé
1217a2b0a27dSPhilippe Mathieu-Daudé #define MIPSDSP64_BINOP_ENV(name, func, element) \
1218a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
1219a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1220a2b0a27dSPhilippe Mathieu-Daudé { \
1221a2b0a27dSPhilippe Mathieu-Daudé DSP64Value ds, dt; \
1222a2b0a27dSPhilippe Mathieu-Daudé unsigned int i; \
1223a2b0a27dSPhilippe Mathieu-Daudé \
1224a2b0a27dSPhilippe Mathieu-Daudé ds.sl[0] = rs; \
1225a2b0a27dSPhilippe Mathieu-Daudé dt.sl[0] = rt; \
1226a2b0a27dSPhilippe Mathieu-Daudé \
1227a2b0a27dSPhilippe Mathieu-Daudé for (i = 0 ; i < ARRAY_SIZE(ds.element); i++) { \
1228a2b0a27dSPhilippe Mathieu-Daudé ds.element[i] = mipsdsp_##func(ds.element[i], dt.element[i], env); \
1229a2b0a27dSPhilippe Mathieu-Daudé } \
1230a2b0a27dSPhilippe Mathieu-Daudé \
1231a2b0a27dSPhilippe Mathieu-Daudé return ds.sl[0]; \
1232a2b0a27dSPhilippe Mathieu-Daudé }
1233a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addq_pw, add_i32, sw);
1234a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addq_qh, add_i16, sh);
1235a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addq_s_pw, sat_add_i32, sw);
1236a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addq_s_qh, sat_add_i16, sh);
1237a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addu_ob, add_u8, uh);
1238a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addu_qh, add_u16, uh);
1239a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addu_s_ob, sat_add_u8, uh);
1240a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(addu_s_qh, sat_add_u16, uh);
1241a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subq_pw, sub32, sw);
1242a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subq_qh, sub_i16, sh);
1243a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subq_s_pw, sat32_sub, sw);
1244a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subq_s_qh, sat16_sub, sh);
1245a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subu_ob, sub_u8, uh);
1246a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subu_qh, sub_u16_u16, uh);
1247a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subu_s_ob, satu8_sub, uh);
1248a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP64_BINOP_ENV(subu_s_qh, satu16_sub_u16_u16, uh);
1249a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP64_BINOP_ENV
1250a2b0a27dSPhilippe Mathieu-Daudé
1251a2b0a27dSPhilippe Mathieu-Daudé #endif
1252a2b0a27dSPhilippe Mathieu-Daudé
1253a2b0a27dSPhilippe Mathieu-Daudé #define SUBUH_QB(name, var) \
1254a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_qb(target_ulong rs, target_ulong rt) \
1255a2b0a27dSPhilippe Mathieu-Daudé { \
1256a2b0a27dSPhilippe Mathieu-Daudé uint8_t rs3, rs2, rs1, rs0; \
1257a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt3, rt2, rt1, rt0; \
1258a2b0a27dSPhilippe Mathieu-Daudé uint8_t tempD, tempC, tempB, tempA; \
1259a2b0a27dSPhilippe Mathieu-Daudé \
1260a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_8(rs, rs3, rs2, rs1, rs0); \
1261a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_8(rt, rt3, rt2, rt1, rt0); \
1262a2b0a27dSPhilippe Mathieu-Daudé \
1263a2b0a27dSPhilippe Mathieu-Daudé tempD = ((uint16_t)rs3 - (uint16_t)rt3 + var) >> 1; \
1264a2b0a27dSPhilippe Mathieu-Daudé tempC = ((uint16_t)rs2 - (uint16_t)rt2 + var) >> 1; \
1265a2b0a27dSPhilippe Mathieu-Daudé tempB = ((uint16_t)rs1 - (uint16_t)rt1 + var) >> 1; \
1266a2b0a27dSPhilippe Mathieu-Daudé tempA = ((uint16_t)rs0 - (uint16_t)rt0 + var) >> 1; \
1267a2b0a27dSPhilippe Mathieu-Daudé \
1268a2b0a27dSPhilippe Mathieu-Daudé return ((uint32_t)tempD << 24) | ((uint32_t)tempC << 16) | \
1269a2b0a27dSPhilippe Mathieu-Daudé ((uint32_t)tempB << 8) | ((uint32_t)tempA); \
1270a2b0a27dSPhilippe Mathieu-Daudé }
1271a2b0a27dSPhilippe Mathieu-Daudé
1272a2b0a27dSPhilippe Mathieu-Daudé SUBUH_QB(subuh, 0);
1273a2b0a27dSPhilippe Mathieu-Daudé SUBUH_QB(subuh_r, 1);
1274a2b0a27dSPhilippe Mathieu-Daudé
1275a2b0a27dSPhilippe Mathieu-Daudé #undef SUBUH_QB
1276a2b0a27dSPhilippe Mathieu-Daudé
helper_addsc(target_ulong rs,target_ulong rt,CPUMIPSState * env)1277a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_addsc(target_ulong rs, target_ulong rt, CPUMIPSState *env)
1278a2b0a27dSPhilippe Mathieu-Daudé {
1279a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp, tempRs, tempRt;
1280a2b0a27dSPhilippe Mathieu-Daudé bool flag;
1281a2b0a27dSPhilippe Mathieu-Daudé
1282a2b0a27dSPhilippe Mathieu-Daudé tempRs = (uint64_t)rs & MIPSDSP_LLO;
1283a2b0a27dSPhilippe Mathieu-Daudé tempRt = (uint64_t)rt & MIPSDSP_LLO;
1284a2b0a27dSPhilippe Mathieu-Daudé
1285a2b0a27dSPhilippe Mathieu-Daudé temp = tempRs + tempRt;
1286a2b0a27dSPhilippe Mathieu-Daudé flag = (temp & 0x0100000000ull) >> 32;
1287a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_carryflag(flag, env);
1288a2b0a27dSPhilippe Mathieu-Daudé
1289a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)(temp & MIPSDSP_LLO);
1290a2b0a27dSPhilippe Mathieu-Daudé }
1291a2b0a27dSPhilippe Mathieu-Daudé
helper_addwc(target_ulong rs,target_ulong rt,CPUMIPSState * env)1292a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_addwc(target_ulong rs, target_ulong rt, CPUMIPSState *env)
1293a2b0a27dSPhilippe Mathieu-Daudé {
1294a2b0a27dSPhilippe Mathieu-Daudé uint32_t rd;
1295a2b0a27dSPhilippe Mathieu-Daudé int32_t temp32, temp31;
1296a2b0a27dSPhilippe Mathieu-Daudé int64_t tempL;
1297a2b0a27dSPhilippe Mathieu-Daudé
1298a2b0a27dSPhilippe Mathieu-Daudé tempL = (int64_t)(int32_t)rs + (int64_t)(int32_t)rt +
1299a2b0a27dSPhilippe Mathieu-Daudé get_DSPControl_carryflag(env);
1300a2b0a27dSPhilippe Mathieu-Daudé temp31 = (tempL >> 31) & 0x01;
1301a2b0a27dSPhilippe Mathieu-Daudé temp32 = (tempL >> 32) & 0x01;
1302a2b0a27dSPhilippe Mathieu-Daudé
1303a2b0a27dSPhilippe Mathieu-Daudé if (temp31 != temp32) {
1304a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 20, env);
1305a2b0a27dSPhilippe Mathieu-Daudé }
1306a2b0a27dSPhilippe Mathieu-Daudé
1307a2b0a27dSPhilippe Mathieu-Daudé rd = tempL & MIPSDSP_LLO;
1308a2b0a27dSPhilippe Mathieu-Daudé
1309a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)rd;
1310a2b0a27dSPhilippe Mathieu-Daudé }
1311a2b0a27dSPhilippe Mathieu-Daudé
helper_modsub(target_ulong rs,target_ulong rt)1312a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_modsub(target_ulong rs, target_ulong rt)
1313a2b0a27dSPhilippe Mathieu-Daudé {
1314a2b0a27dSPhilippe Mathieu-Daudé int32_t decr;
1315a2b0a27dSPhilippe Mathieu-Daudé uint16_t lastindex;
1316a2b0a27dSPhilippe Mathieu-Daudé target_ulong rd;
1317a2b0a27dSPhilippe Mathieu-Daudé
1318a2b0a27dSPhilippe Mathieu-Daudé decr = rt & MIPSDSP_Q0;
1319a2b0a27dSPhilippe Mathieu-Daudé lastindex = (rt >> 8) & MIPSDSP_LO;
1320a2b0a27dSPhilippe Mathieu-Daudé
1321a2b0a27dSPhilippe Mathieu-Daudé if ((rs & MIPSDSP_LLO) == 0x00000000) {
1322a2b0a27dSPhilippe Mathieu-Daudé rd = (target_ulong)lastindex;
1323a2b0a27dSPhilippe Mathieu-Daudé } else {
1324a2b0a27dSPhilippe Mathieu-Daudé rd = rs - decr;
1325a2b0a27dSPhilippe Mathieu-Daudé }
1326a2b0a27dSPhilippe Mathieu-Daudé
1327a2b0a27dSPhilippe Mathieu-Daudé return rd;
1328a2b0a27dSPhilippe Mathieu-Daudé }
1329a2b0a27dSPhilippe Mathieu-Daudé
helper_raddu_w_qb(target_ulong rs)1330a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_raddu_w_qb(target_ulong rs)
1331a2b0a27dSPhilippe Mathieu-Daudé {
1332a2b0a27dSPhilippe Mathieu-Daudé target_ulong ret = 0;
1333a2b0a27dSPhilippe Mathieu-Daudé DSP32Value ds;
1334a2b0a27dSPhilippe Mathieu-Daudé unsigned int i;
1335a2b0a27dSPhilippe Mathieu-Daudé
1336a2b0a27dSPhilippe Mathieu-Daudé ds.uw[0] = rs;
1337a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 4; i++) {
1338a2b0a27dSPhilippe Mathieu-Daudé ret += ds.ub[i];
1339a2b0a27dSPhilippe Mathieu-Daudé }
1340a2b0a27dSPhilippe Mathieu-Daudé return ret;
1341a2b0a27dSPhilippe Mathieu-Daudé }
1342a2b0a27dSPhilippe Mathieu-Daudé
1343a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_raddu_l_ob(target_ulong rs)1344a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_raddu_l_ob(target_ulong rs)
1345a2b0a27dSPhilippe Mathieu-Daudé {
1346a2b0a27dSPhilippe Mathieu-Daudé target_ulong ret = 0;
1347a2b0a27dSPhilippe Mathieu-Daudé DSP64Value ds;
1348a2b0a27dSPhilippe Mathieu-Daudé unsigned int i;
1349a2b0a27dSPhilippe Mathieu-Daudé
1350a2b0a27dSPhilippe Mathieu-Daudé ds.ul[0] = rs;
1351a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 8; i++) {
1352a2b0a27dSPhilippe Mathieu-Daudé ret += ds.ub[i];
1353a2b0a27dSPhilippe Mathieu-Daudé }
1354a2b0a27dSPhilippe Mathieu-Daudé return ret;
1355a2b0a27dSPhilippe Mathieu-Daudé }
1356a2b0a27dSPhilippe Mathieu-Daudé #endif
1357a2b0a27dSPhilippe Mathieu-Daudé
1358a2b0a27dSPhilippe Mathieu-Daudé #define PRECR_QB_PH(name, a, b)\
1359a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_qb_ph(target_ulong rs, target_ulong rt) \
1360a2b0a27dSPhilippe Mathieu-Daudé { \
1361a2b0a27dSPhilippe Mathieu-Daudé uint8_t tempD, tempC, tempB, tempA; \
1362a2b0a27dSPhilippe Mathieu-Daudé \
1363a2b0a27dSPhilippe Mathieu-Daudé tempD = (rs >> a) & MIPSDSP_Q0; \
1364a2b0a27dSPhilippe Mathieu-Daudé tempC = (rs >> b) & MIPSDSP_Q0; \
1365a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> a) & MIPSDSP_Q0; \
1366a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> b) & MIPSDSP_Q0; \
1367a2b0a27dSPhilippe Mathieu-Daudé \
1368a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_8(tempD, tempC, tempB, tempA); \
1369a2b0a27dSPhilippe Mathieu-Daudé }
1370a2b0a27dSPhilippe Mathieu-Daudé
1371a2b0a27dSPhilippe Mathieu-Daudé PRECR_QB_PH(precr, 16, 0);
1372a2b0a27dSPhilippe Mathieu-Daudé PRECR_QB_PH(precrq, 24, 8);
1373a2b0a27dSPhilippe Mathieu-Daudé
1374a2b0a27dSPhilippe Mathieu-Daudé #undef PRECR_QB_OH
1375a2b0a27dSPhilippe Mathieu-Daudé
helper_precr_sra_ph_w(uint32_t sa,target_ulong rs,target_ulong rt)1376a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precr_sra_ph_w(uint32_t sa, target_ulong rs,
1377a2b0a27dSPhilippe Mathieu-Daudé target_ulong rt)
1378a2b0a27dSPhilippe Mathieu-Daudé {
1379a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA;
1380a2b0a27dSPhilippe Mathieu-Daudé
1381a2b0a27dSPhilippe Mathieu-Daudé tempB = ((int32_t)rt >> sa) & MIPSDSP_LO;
1382a2b0a27dSPhilippe Mathieu-Daudé tempA = ((int32_t)rs >> sa) & MIPSDSP_LO;
1383a2b0a27dSPhilippe Mathieu-Daudé
1384a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(tempB, tempA);
1385a2b0a27dSPhilippe Mathieu-Daudé }
1386a2b0a27dSPhilippe Mathieu-Daudé
helper_precr_sra_r_ph_w(uint32_t sa,target_ulong rs,target_ulong rt)1387a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precr_sra_r_ph_w(uint32_t sa,
1388a2b0a27dSPhilippe Mathieu-Daudé target_ulong rs, target_ulong rt)
1389a2b0a27dSPhilippe Mathieu-Daudé {
1390a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempB, tempA;
1391a2b0a27dSPhilippe Mathieu-Daudé
1392a2b0a27dSPhilippe Mathieu-Daudé /* If sa = 0, then (sa - 1) = -1 will case shift error, so we need else. */
1393a2b0a27dSPhilippe Mathieu-Daudé if (sa == 0) {
1394a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt & MIPSDSP_LO) << 1;
1395a2b0a27dSPhilippe Mathieu-Daudé tempA = (rs & MIPSDSP_LO) << 1;
1396a2b0a27dSPhilippe Mathieu-Daudé } else {
1397a2b0a27dSPhilippe Mathieu-Daudé tempB = ((int32_t)rt >> (sa - 1)) + 1;
1398a2b0a27dSPhilippe Mathieu-Daudé tempA = ((int32_t)rs >> (sa - 1)) + 1;
1399a2b0a27dSPhilippe Mathieu-Daudé }
1400a2b0a27dSPhilippe Mathieu-Daudé rt = (((tempB >> 1) & MIPSDSP_LO) << 16) | ((tempA >> 1) & MIPSDSP_LO);
1401a2b0a27dSPhilippe Mathieu-Daudé
1402a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)rt;
1403a2b0a27dSPhilippe Mathieu-Daudé }
1404a2b0a27dSPhilippe Mathieu-Daudé
helper_precrq_ph_w(target_ulong rs,target_ulong rt)1405a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrq_ph_w(target_ulong rs, target_ulong rt)
1406a2b0a27dSPhilippe Mathieu-Daudé {
1407a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA;
1408a2b0a27dSPhilippe Mathieu-Daudé
1409a2b0a27dSPhilippe Mathieu-Daudé tempB = (rs & MIPSDSP_HI) >> 16;
1410a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt & MIPSDSP_HI) >> 16;
1411a2b0a27dSPhilippe Mathieu-Daudé
1412a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(tempB, tempA);
1413a2b0a27dSPhilippe Mathieu-Daudé }
1414a2b0a27dSPhilippe Mathieu-Daudé
helper_precrq_rs_ph_w(target_ulong rs,target_ulong rt,CPUMIPSState * env)1415a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrq_rs_ph_w(target_ulong rs, target_ulong rt,
1416a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
1417a2b0a27dSPhilippe Mathieu-Daudé {
1418a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA;
1419a2b0a27dSPhilippe Mathieu-Daudé
1420a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_trunc16_sat16_round(rs, env);
1421a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_trunc16_sat16_round(rt, env);
1422a2b0a27dSPhilippe Mathieu-Daudé
1423a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(tempB, tempA);
1424a2b0a27dSPhilippe Mathieu-Daudé }
1425a2b0a27dSPhilippe Mathieu-Daudé
1426a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_precr_ob_qh(target_ulong rs,target_ulong rt)1427a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precr_ob_qh(target_ulong rs, target_ulong rt)
1428a2b0a27dSPhilippe Mathieu-Daudé {
1429a2b0a27dSPhilippe Mathieu-Daudé uint8_t rs6, rs4, rs2, rs0;
1430a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt6, rt4, rt2, rt0;
1431a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp;
1432a2b0a27dSPhilippe Mathieu-Daudé
1433a2b0a27dSPhilippe Mathieu-Daudé rs6 = (rs >> 48) & MIPSDSP_Q0;
1434a2b0a27dSPhilippe Mathieu-Daudé rs4 = (rs >> 32) & MIPSDSP_Q0;
1435a2b0a27dSPhilippe Mathieu-Daudé rs2 = (rs >> 16) & MIPSDSP_Q0;
1436a2b0a27dSPhilippe Mathieu-Daudé rs0 = rs & MIPSDSP_Q0;
1437a2b0a27dSPhilippe Mathieu-Daudé rt6 = (rt >> 48) & MIPSDSP_Q0;
1438a2b0a27dSPhilippe Mathieu-Daudé rt4 = (rt >> 32) & MIPSDSP_Q0;
1439a2b0a27dSPhilippe Mathieu-Daudé rt2 = (rt >> 16) & MIPSDSP_Q0;
1440a2b0a27dSPhilippe Mathieu-Daudé rt0 = rt & MIPSDSP_Q0;
1441a2b0a27dSPhilippe Mathieu-Daudé
1442a2b0a27dSPhilippe Mathieu-Daudé temp = ((uint64_t)rs6 << 56) | ((uint64_t)rs4 << 48) |
1443a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)rs2 << 40) | ((uint64_t)rs0 << 32) |
1444a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)rt6 << 24) | ((uint64_t)rt4 << 16) |
1445a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)rt2 << 8) | (uint64_t)rt0;
1446a2b0a27dSPhilippe Mathieu-Daudé
1447a2b0a27dSPhilippe Mathieu-Daudé return temp;
1448a2b0a27dSPhilippe Mathieu-Daudé }
1449a2b0a27dSPhilippe Mathieu-Daudé
1450a2b0a27dSPhilippe Mathieu-Daudé
1451a2b0a27dSPhilippe Mathieu-Daudé /*
1452a2b0a27dSPhilippe Mathieu-Daudé * In case sa == 0, use rt2, rt0, rs2, rs0.
1453a2b0a27dSPhilippe Mathieu-Daudé * In case sa != 0, use rt3, rt1, rs3, rs1.
1454a2b0a27dSPhilippe Mathieu-Daudé */
1455a2b0a27dSPhilippe Mathieu-Daudé #define PRECR_QH_PW(name, var) \
1456a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precr_##name##_qh_pw(target_ulong rs, \
1457a2b0a27dSPhilippe Mathieu-Daudé target_ulong rt, \
1458a2b0a27dSPhilippe Mathieu-Daudé uint32_t sa) \
1459a2b0a27dSPhilippe Mathieu-Daudé { \
1460a2b0a27dSPhilippe Mathieu-Daudé uint16_t rs3, rs2, rs1, rs0; \
1461a2b0a27dSPhilippe Mathieu-Daudé uint16_t rt3, rt2, rt1, rt0; \
1462a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA; \
1463a2b0a27dSPhilippe Mathieu-Daudé \
1464a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rs, rs3, rs2, rs1, rs0); \
1465a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rt, rt3, rt2, rt1, rt0); \
1466a2b0a27dSPhilippe Mathieu-Daudé \
1467a2b0a27dSPhilippe Mathieu-Daudé if (sa == 0) { \
1468a2b0a27dSPhilippe Mathieu-Daudé tempD = rt2 << var; \
1469a2b0a27dSPhilippe Mathieu-Daudé tempC = rt0 << var; \
1470a2b0a27dSPhilippe Mathieu-Daudé tempB = rs2 << var; \
1471a2b0a27dSPhilippe Mathieu-Daudé tempA = rs0 << var; \
1472a2b0a27dSPhilippe Mathieu-Daudé } else { \
1473a2b0a27dSPhilippe Mathieu-Daudé tempD = (((int16_t)rt3 >> sa) + var) >> var; \
1474a2b0a27dSPhilippe Mathieu-Daudé tempC = (((int16_t)rt1 >> sa) + var) >> var; \
1475a2b0a27dSPhilippe Mathieu-Daudé tempB = (((int16_t)rs3 >> sa) + var) >> var; \
1476a2b0a27dSPhilippe Mathieu-Daudé tempA = (((int16_t)rs1 >> sa) + var) >> var; \
1477a2b0a27dSPhilippe Mathieu-Daudé } \
1478a2b0a27dSPhilippe Mathieu-Daudé \
1479a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(tempD, tempC, tempB, tempA); \
1480a2b0a27dSPhilippe Mathieu-Daudé }
1481a2b0a27dSPhilippe Mathieu-Daudé
1482a2b0a27dSPhilippe Mathieu-Daudé PRECR_QH_PW(sra, 0);
1483a2b0a27dSPhilippe Mathieu-Daudé PRECR_QH_PW(sra_r, 1);
1484a2b0a27dSPhilippe Mathieu-Daudé
1485a2b0a27dSPhilippe Mathieu-Daudé #undef PRECR_QH_PW
1486a2b0a27dSPhilippe Mathieu-Daudé
helper_precrq_ob_qh(target_ulong rs,target_ulong rt)1487a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrq_ob_qh(target_ulong rs, target_ulong rt)
1488a2b0a27dSPhilippe Mathieu-Daudé {
1489a2b0a27dSPhilippe Mathieu-Daudé uint8_t rs6, rs4, rs2, rs0;
1490a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt6, rt4, rt2, rt0;
1491a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp;
1492a2b0a27dSPhilippe Mathieu-Daudé
1493a2b0a27dSPhilippe Mathieu-Daudé rs6 = (rs >> 56) & MIPSDSP_Q0;
1494a2b0a27dSPhilippe Mathieu-Daudé rs4 = (rs >> 40) & MIPSDSP_Q0;
1495a2b0a27dSPhilippe Mathieu-Daudé rs2 = (rs >> 24) & MIPSDSP_Q0;
1496a2b0a27dSPhilippe Mathieu-Daudé rs0 = (rs >> 8) & MIPSDSP_Q0;
1497a2b0a27dSPhilippe Mathieu-Daudé rt6 = (rt >> 56) & MIPSDSP_Q0;
1498a2b0a27dSPhilippe Mathieu-Daudé rt4 = (rt >> 40) & MIPSDSP_Q0;
1499a2b0a27dSPhilippe Mathieu-Daudé rt2 = (rt >> 24) & MIPSDSP_Q0;
1500a2b0a27dSPhilippe Mathieu-Daudé rt0 = (rt >> 8) & MIPSDSP_Q0;
1501a2b0a27dSPhilippe Mathieu-Daudé
1502a2b0a27dSPhilippe Mathieu-Daudé temp = ((uint64_t)rs6 << 56) | ((uint64_t)rs4 << 48) |
1503a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)rs2 << 40) | ((uint64_t)rs0 << 32) |
1504a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)rt6 << 24) | ((uint64_t)rt4 << 16) |
1505a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)rt2 << 8) | (uint64_t)rt0;
1506a2b0a27dSPhilippe Mathieu-Daudé
1507a2b0a27dSPhilippe Mathieu-Daudé return temp;
1508a2b0a27dSPhilippe Mathieu-Daudé }
1509a2b0a27dSPhilippe Mathieu-Daudé
helper_precrq_qh_pw(target_ulong rs,target_ulong rt)1510a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrq_qh_pw(target_ulong rs, target_ulong rt)
1511a2b0a27dSPhilippe Mathieu-Daudé {
1512a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA;
1513a2b0a27dSPhilippe Mathieu-Daudé
1514a2b0a27dSPhilippe Mathieu-Daudé tempD = (rs >> 48) & MIPSDSP_LO;
1515a2b0a27dSPhilippe Mathieu-Daudé tempC = (rs >> 16) & MIPSDSP_LO;
1516a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> 48) & MIPSDSP_LO;
1517a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> 16) & MIPSDSP_LO;
1518a2b0a27dSPhilippe Mathieu-Daudé
1519a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(tempD, tempC, tempB, tempA);
1520a2b0a27dSPhilippe Mathieu-Daudé }
1521a2b0a27dSPhilippe Mathieu-Daudé
helper_precrq_rs_qh_pw(target_ulong rs,target_ulong rt,CPUMIPSState * env)1522a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrq_rs_qh_pw(target_ulong rs, target_ulong rt,
1523a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
1524a2b0a27dSPhilippe Mathieu-Daudé {
1525a2b0a27dSPhilippe Mathieu-Daudé uint32_t rs2, rs0;
1526a2b0a27dSPhilippe Mathieu-Daudé uint32_t rt2, rt0;
1527a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA;
1528a2b0a27dSPhilippe Mathieu-Daudé
1529a2b0a27dSPhilippe Mathieu-Daudé rs2 = (rs >> 32) & MIPSDSP_LLO;
1530a2b0a27dSPhilippe Mathieu-Daudé rs0 = rs & MIPSDSP_LLO;
1531a2b0a27dSPhilippe Mathieu-Daudé rt2 = (rt >> 32) & MIPSDSP_LLO;
1532a2b0a27dSPhilippe Mathieu-Daudé rt0 = rt & MIPSDSP_LLO;
1533a2b0a27dSPhilippe Mathieu-Daudé
1534a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_trunc16_sat16_round(rs2, env);
1535a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_trunc16_sat16_round(rs0, env);
1536a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_trunc16_sat16_round(rt2, env);
1537a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_trunc16_sat16_round(rt0, env);
1538a2b0a27dSPhilippe Mathieu-Daudé
1539a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(tempD, tempC, tempB, tempA);
1540a2b0a27dSPhilippe Mathieu-Daudé }
1541a2b0a27dSPhilippe Mathieu-Daudé
helper_precrq_pw_l(target_ulong rs,target_ulong rt)1542a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrq_pw_l(target_ulong rs, target_ulong rt)
1543a2b0a27dSPhilippe Mathieu-Daudé {
1544a2b0a27dSPhilippe Mathieu-Daudé uint32_t tempB, tempA;
1545a2b0a27dSPhilippe Mathieu-Daudé
1546a2b0a27dSPhilippe Mathieu-Daudé tempB = (rs >> 32) & MIPSDSP_LLO;
1547a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> 32) & MIPSDSP_LLO;
1548a2b0a27dSPhilippe Mathieu-Daudé
1549a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_32(tempB, tempA);
1550a2b0a27dSPhilippe Mathieu-Daudé }
1551a2b0a27dSPhilippe Mathieu-Daudé #endif
1552a2b0a27dSPhilippe Mathieu-Daudé
helper_precrqu_s_qb_ph(target_ulong rs,target_ulong rt,CPUMIPSState * env)1553a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrqu_s_qb_ph(target_ulong rs, target_ulong rt,
1554a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
1555a2b0a27dSPhilippe Mathieu-Daudé {
1556a2b0a27dSPhilippe Mathieu-Daudé uint8_t tempD, tempC, tempB, tempA;
1557a2b0a27dSPhilippe Mathieu-Daudé uint16_t rsh, rsl, rth, rtl;
1558a2b0a27dSPhilippe Mathieu-Daudé
1559a2b0a27dSPhilippe Mathieu-Daudé rsh = (rs & MIPSDSP_HI) >> 16;
1560a2b0a27dSPhilippe Mathieu-Daudé rsl = rs & MIPSDSP_LO;
1561a2b0a27dSPhilippe Mathieu-Daudé rth = (rt & MIPSDSP_HI) >> 16;
1562a2b0a27dSPhilippe Mathieu-Daudé rtl = rt & MIPSDSP_LO;
1563a2b0a27dSPhilippe Mathieu-Daudé
1564a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_sat8_reduce_precision(rsh, env);
1565a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_sat8_reduce_precision(rsl, env);
1566a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_sat8_reduce_precision(rth, env);
1567a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_sat8_reduce_precision(rtl, env);
1568a2b0a27dSPhilippe Mathieu-Daudé
1569a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_8(tempD, tempC, tempB, tempA);
1570a2b0a27dSPhilippe Mathieu-Daudé }
1571a2b0a27dSPhilippe Mathieu-Daudé
1572a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_precrqu_s_ob_qh(target_ulong rs,target_ulong rt,CPUMIPSState * env)1573a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precrqu_s_ob_qh(target_ulong rs, target_ulong rt,
1574a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
1575a2b0a27dSPhilippe Mathieu-Daudé {
1576a2b0a27dSPhilippe Mathieu-Daudé int i;
1577a2b0a27dSPhilippe Mathieu-Daudé uint16_t rs3, rs2, rs1, rs0;
1578a2b0a27dSPhilippe Mathieu-Daudé uint16_t rt3, rt2, rt1, rt0;
1579a2b0a27dSPhilippe Mathieu-Daudé uint8_t temp[8];
1580a2b0a27dSPhilippe Mathieu-Daudé uint64_t result;
1581a2b0a27dSPhilippe Mathieu-Daudé
1582a2b0a27dSPhilippe Mathieu-Daudé result = 0;
1583a2b0a27dSPhilippe Mathieu-Daudé
1584a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rs, rs3, rs2, rs1, rs0);
1585a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rt, rt3, rt2, rt1, rt0);
1586a2b0a27dSPhilippe Mathieu-Daudé
1587a2b0a27dSPhilippe Mathieu-Daudé temp[7] = mipsdsp_sat8_reduce_precision(rs3, env);
1588a2b0a27dSPhilippe Mathieu-Daudé temp[6] = mipsdsp_sat8_reduce_precision(rs2, env);
1589a2b0a27dSPhilippe Mathieu-Daudé temp[5] = mipsdsp_sat8_reduce_precision(rs1, env);
1590a2b0a27dSPhilippe Mathieu-Daudé temp[4] = mipsdsp_sat8_reduce_precision(rs0, env);
1591a2b0a27dSPhilippe Mathieu-Daudé temp[3] = mipsdsp_sat8_reduce_precision(rt3, env);
1592a2b0a27dSPhilippe Mathieu-Daudé temp[2] = mipsdsp_sat8_reduce_precision(rt2, env);
1593a2b0a27dSPhilippe Mathieu-Daudé temp[1] = mipsdsp_sat8_reduce_precision(rt1, env);
1594a2b0a27dSPhilippe Mathieu-Daudé temp[0] = mipsdsp_sat8_reduce_precision(rt0, env);
1595a2b0a27dSPhilippe Mathieu-Daudé
1596a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 8; i++) {
1597a2b0a27dSPhilippe Mathieu-Daudé result |= (uint64_t)temp[i] << (8 * i);
1598a2b0a27dSPhilippe Mathieu-Daudé }
1599a2b0a27dSPhilippe Mathieu-Daudé
1600a2b0a27dSPhilippe Mathieu-Daudé return result;
1601a2b0a27dSPhilippe Mathieu-Daudé }
1602a2b0a27dSPhilippe Mathieu-Daudé
1603a2b0a27dSPhilippe Mathieu-Daudé #define PRECEQ_PW(name, a, b) \
1604a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_preceq_pw_##name(target_ulong rt) \
1605a2b0a27dSPhilippe Mathieu-Daudé { \
1606a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA; \
1607a2b0a27dSPhilippe Mathieu-Daudé uint32_t tempBI, tempAI; \
1608a2b0a27dSPhilippe Mathieu-Daudé \
1609a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> a) & MIPSDSP_LO; \
1610a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> b) & MIPSDSP_LO; \
1611a2b0a27dSPhilippe Mathieu-Daudé \
1612a2b0a27dSPhilippe Mathieu-Daudé tempBI = (uint32_t)tempB << 16; \
1613a2b0a27dSPhilippe Mathieu-Daudé tempAI = (uint32_t)tempA << 16; \
1614a2b0a27dSPhilippe Mathieu-Daudé \
1615a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_32(tempBI, tempAI); \
1616a2b0a27dSPhilippe Mathieu-Daudé }
1617a2b0a27dSPhilippe Mathieu-Daudé
1618a2b0a27dSPhilippe Mathieu-Daudé PRECEQ_PW(qhl, 48, 32);
1619a2b0a27dSPhilippe Mathieu-Daudé PRECEQ_PW(qhr, 16, 0);
1620a2b0a27dSPhilippe Mathieu-Daudé PRECEQ_PW(qhla, 48, 16);
1621a2b0a27dSPhilippe Mathieu-Daudé PRECEQ_PW(qhra, 32, 0);
1622a2b0a27dSPhilippe Mathieu-Daudé
1623a2b0a27dSPhilippe Mathieu-Daudé #undef PRECEQ_PW
1624a2b0a27dSPhilippe Mathieu-Daudé
1625a2b0a27dSPhilippe Mathieu-Daudé #endif
1626a2b0a27dSPhilippe Mathieu-Daudé
1627a2b0a27dSPhilippe Mathieu-Daudé #define PRECEQU_PH(name, a, b) \
1628a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precequ_ph_##name(target_ulong rt) \
1629a2b0a27dSPhilippe Mathieu-Daudé { \
1630a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA; \
1631a2b0a27dSPhilippe Mathieu-Daudé \
1632a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> a) & MIPSDSP_Q0; \
1633a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> b) & MIPSDSP_Q0; \
1634a2b0a27dSPhilippe Mathieu-Daudé \
1635a2b0a27dSPhilippe Mathieu-Daudé tempB = tempB << 7; \
1636a2b0a27dSPhilippe Mathieu-Daudé tempA = tempA << 7; \
1637a2b0a27dSPhilippe Mathieu-Daudé \
1638a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(tempB, tempA); \
1639a2b0a27dSPhilippe Mathieu-Daudé }
1640a2b0a27dSPhilippe Mathieu-Daudé
1641a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_PH(qbl, 24, 16);
1642a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_PH(qbr, 8, 0);
1643a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_PH(qbla, 24, 8);
1644a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_PH(qbra, 16, 0);
1645a2b0a27dSPhilippe Mathieu-Daudé
1646a2b0a27dSPhilippe Mathieu-Daudé #undef PRECEQU_PH
1647a2b0a27dSPhilippe Mathieu-Daudé
1648a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
1649a2b0a27dSPhilippe Mathieu-Daudé #define PRECEQU_QH(name, a, b, c, d) \
1650a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_precequ_qh_##name(target_ulong rt) \
1651a2b0a27dSPhilippe Mathieu-Daudé { \
1652a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA; \
1653a2b0a27dSPhilippe Mathieu-Daudé \
1654a2b0a27dSPhilippe Mathieu-Daudé tempD = (rt >> a) & MIPSDSP_Q0; \
1655a2b0a27dSPhilippe Mathieu-Daudé tempC = (rt >> b) & MIPSDSP_Q0; \
1656a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> c) & MIPSDSP_Q0; \
1657a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> d) & MIPSDSP_Q0; \
1658a2b0a27dSPhilippe Mathieu-Daudé \
1659a2b0a27dSPhilippe Mathieu-Daudé tempD = tempD << 7; \
1660a2b0a27dSPhilippe Mathieu-Daudé tempC = tempC << 7; \
1661a2b0a27dSPhilippe Mathieu-Daudé tempB = tempB << 7; \
1662a2b0a27dSPhilippe Mathieu-Daudé tempA = tempA << 7; \
1663a2b0a27dSPhilippe Mathieu-Daudé \
1664a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(tempD, tempC, tempB, tempA); \
1665a2b0a27dSPhilippe Mathieu-Daudé }
1666a2b0a27dSPhilippe Mathieu-Daudé
1667a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_QH(obl, 56, 48, 40, 32);
1668a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_QH(obr, 24, 16, 8, 0);
1669a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_QH(obla, 56, 40, 24, 8);
1670a2b0a27dSPhilippe Mathieu-Daudé PRECEQU_QH(obra, 48, 32, 16, 0);
1671a2b0a27dSPhilippe Mathieu-Daudé
1672a2b0a27dSPhilippe Mathieu-Daudé #undef PRECEQU_QH
1673a2b0a27dSPhilippe Mathieu-Daudé
1674a2b0a27dSPhilippe Mathieu-Daudé #endif
1675a2b0a27dSPhilippe Mathieu-Daudé
1676a2b0a27dSPhilippe Mathieu-Daudé #define PRECEU_PH(name, a, b) \
1677a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_preceu_ph_##name(target_ulong rt) \
1678a2b0a27dSPhilippe Mathieu-Daudé { \
1679a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA; \
1680a2b0a27dSPhilippe Mathieu-Daudé \
1681a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> a) & MIPSDSP_Q0; \
1682a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> b) & MIPSDSP_Q0; \
1683a2b0a27dSPhilippe Mathieu-Daudé \
1684a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(tempB, tempA); \
1685a2b0a27dSPhilippe Mathieu-Daudé }
1686a2b0a27dSPhilippe Mathieu-Daudé
1687a2b0a27dSPhilippe Mathieu-Daudé PRECEU_PH(qbl, 24, 16);
1688a2b0a27dSPhilippe Mathieu-Daudé PRECEU_PH(qbr, 8, 0);
1689a2b0a27dSPhilippe Mathieu-Daudé PRECEU_PH(qbla, 24, 8);
1690a2b0a27dSPhilippe Mathieu-Daudé PRECEU_PH(qbra, 16, 0);
1691a2b0a27dSPhilippe Mathieu-Daudé
1692a2b0a27dSPhilippe Mathieu-Daudé #undef PRECEU_PH
1693a2b0a27dSPhilippe Mathieu-Daudé
1694a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
1695a2b0a27dSPhilippe Mathieu-Daudé #define PRECEU_QH(name, a, b, c, d) \
1696a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_preceu_qh_##name(target_ulong rt) \
1697a2b0a27dSPhilippe Mathieu-Daudé { \
1698a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA; \
1699a2b0a27dSPhilippe Mathieu-Daudé \
1700a2b0a27dSPhilippe Mathieu-Daudé tempD = (rt >> a) & MIPSDSP_Q0; \
1701a2b0a27dSPhilippe Mathieu-Daudé tempC = (rt >> b) & MIPSDSP_Q0; \
1702a2b0a27dSPhilippe Mathieu-Daudé tempB = (rt >> c) & MIPSDSP_Q0; \
1703a2b0a27dSPhilippe Mathieu-Daudé tempA = (rt >> d) & MIPSDSP_Q0; \
1704a2b0a27dSPhilippe Mathieu-Daudé \
1705a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(tempD, tempC, tempB, tempA); \
1706a2b0a27dSPhilippe Mathieu-Daudé }
1707a2b0a27dSPhilippe Mathieu-Daudé
1708a2b0a27dSPhilippe Mathieu-Daudé PRECEU_QH(obl, 56, 48, 40, 32);
1709a2b0a27dSPhilippe Mathieu-Daudé PRECEU_QH(obr, 24, 16, 8, 0);
1710a2b0a27dSPhilippe Mathieu-Daudé PRECEU_QH(obla, 56, 40, 24, 8);
1711a2b0a27dSPhilippe Mathieu-Daudé PRECEU_QH(obra, 48, 32, 16, 0);
1712a2b0a27dSPhilippe Mathieu-Daudé
1713a2b0a27dSPhilippe Mathieu-Daudé #undef PRECEU_QH
1714a2b0a27dSPhilippe Mathieu-Daudé
1715a2b0a27dSPhilippe Mathieu-Daudé #endif
1716a2b0a27dSPhilippe Mathieu-Daudé
1717a2b0a27dSPhilippe Mathieu-Daudé /** DSP GPR-Based Shift Sub-class insns **/
1718a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_QB(name, func) \
1719a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_qb(target_ulong sa, target_ulong rt) \
1720a2b0a27dSPhilippe Mathieu-Daudé { \
1721a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt3, rt2, rt1, rt0; \
1722a2b0a27dSPhilippe Mathieu-Daudé \
1723a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x07; \
1724a2b0a27dSPhilippe Mathieu-Daudé \
1725a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_8(rt, rt3, rt2, rt1, rt0); \
1726a2b0a27dSPhilippe Mathieu-Daudé \
1727a2b0a27dSPhilippe Mathieu-Daudé rt3 = mipsdsp_##func(rt3, sa); \
1728a2b0a27dSPhilippe Mathieu-Daudé rt2 = mipsdsp_##func(rt2, sa); \
1729a2b0a27dSPhilippe Mathieu-Daudé rt1 = mipsdsp_##func(rt1, sa); \
1730a2b0a27dSPhilippe Mathieu-Daudé rt0 = mipsdsp_##func(rt0, sa); \
1731a2b0a27dSPhilippe Mathieu-Daudé \
1732a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_8(rt3, rt2, rt1, rt0); \
1733a2b0a27dSPhilippe Mathieu-Daudé }
1734a2b0a27dSPhilippe Mathieu-Daudé
1735a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_QB_ENV(name, func) \
1736a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_qb(target_ulong sa, target_ulong rt,\
1737a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1738a2b0a27dSPhilippe Mathieu-Daudé { \
1739a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt3, rt2, rt1, rt0; \
1740a2b0a27dSPhilippe Mathieu-Daudé \
1741a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x07; \
1742a2b0a27dSPhilippe Mathieu-Daudé \
1743a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_8(rt, rt3, rt2, rt1, rt0); \
1744a2b0a27dSPhilippe Mathieu-Daudé \
1745a2b0a27dSPhilippe Mathieu-Daudé rt3 = mipsdsp_##func(rt3, sa, env); \
1746a2b0a27dSPhilippe Mathieu-Daudé rt2 = mipsdsp_##func(rt2, sa, env); \
1747a2b0a27dSPhilippe Mathieu-Daudé rt1 = mipsdsp_##func(rt1, sa, env); \
1748a2b0a27dSPhilippe Mathieu-Daudé rt0 = mipsdsp_##func(rt0, sa, env); \
1749a2b0a27dSPhilippe Mathieu-Daudé \
1750a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_8(rt3, rt2, rt1, rt0); \
1751a2b0a27dSPhilippe Mathieu-Daudé }
1752a2b0a27dSPhilippe Mathieu-Daudé
1753a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QB_ENV(shll, lshift8);
1754a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QB(shrl, rshift_u8);
1755a2b0a27dSPhilippe Mathieu-Daudé
1756a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QB(shra, rashift8);
1757a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QB(shra_r, rnd8_rashift);
1758a2b0a27dSPhilippe Mathieu-Daudé
1759a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_QB
1760a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_QB_ENV
1761a2b0a27dSPhilippe Mathieu-Daudé
1762a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
1763a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_OB(name, func) \
1764a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_ob(target_ulong rt, target_ulong sa) \
1765a2b0a27dSPhilippe Mathieu-Daudé { \
1766a2b0a27dSPhilippe Mathieu-Daudé int i; \
1767a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt_t[8]; \
1768a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp; \
1769a2b0a27dSPhilippe Mathieu-Daudé \
1770a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x07; \
1771a2b0a27dSPhilippe Mathieu-Daudé temp = 0; \
1772a2b0a27dSPhilippe Mathieu-Daudé \
1773a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 8; i++) { \
1774a2b0a27dSPhilippe Mathieu-Daudé rt_t[i] = (rt >> (8 * i)) & MIPSDSP_Q0; \
1775a2b0a27dSPhilippe Mathieu-Daudé rt_t[i] = mipsdsp_##func(rt_t[i], sa); \
1776a2b0a27dSPhilippe Mathieu-Daudé temp |= (uint64_t)rt_t[i] << (8 * i); \
1777a2b0a27dSPhilippe Mathieu-Daudé } \
1778a2b0a27dSPhilippe Mathieu-Daudé \
1779a2b0a27dSPhilippe Mathieu-Daudé return temp; \
1780a2b0a27dSPhilippe Mathieu-Daudé }
1781a2b0a27dSPhilippe Mathieu-Daudé
1782a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_OB_ENV(name, func) \
1783a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_ob(target_ulong rt, target_ulong sa, \
1784a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1785a2b0a27dSPhilippe Mathieu-Daudé { \
1786a2b0a27dSPhilippe Mathieu-Daudé int i; \
1787a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt_t[8]; \
1788a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp; \
1789a2b0a27dSPhilippe Mathieu-Daudé \
1790a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x07; \
1791a2b0a27dSPhilippe Mathieu-Daudé temp = 0; \
1792a2b0a27dSPhilippe Mathieu-Daudé \
1793a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 8; i++) { \
1794a2b0a27dSPhilippe Mathieu-Daudé rt_t[i] = (rt >> (8 * i)) & MIPSDSP_Q0; \
1795a2b0a27dSPhilippe Mathieu-Daudé rt_t[i] = mipsdsp_##func(rt_t[i], sa, env); \
1796a2b0a27dSPhilippe Mathieu-Daudé temp |= (uint64_t)rt_t[i] << (8 * i); \
1797a2b0a27dSPhilippe Mathieu-Daudé } \
1798a2b0a27dSPhilippe Mathieu-Daudé \
1799a2b0a27dSPhilippe Mathieu-Daudé return temp; \
1800a2b0a27dSPhilippe Mathieu-Daudé }
1801a2b0a27dSPhilippe Mathieu-Daudé
1802a2b0a27dSPhilippe Mathieu-Daudé SHIFT_OB_ENV(shll, lshift8);
1803a2b0a27dSPhilippe Mathieu-Daudé SHIFT_OB(shrl, rshift_u8);
1804a2b0a27dSPhilippe Mathieu-Daudé
1805a2b0a27dSPhilippe Mathieu-Daudé SHIFT_OB(shra, rashift8);
1806a2b0a27dSPhilippe Mathieu-Daudé SHIFT_OB(shra_r, rnd8_rashift);
1807a2b0a27dSPhilippe Mathieu-Daudé
1808a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_OB
1809a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_OB_ENV
1810a2b0a27dSPhilippe Mathieu-Daudé
1811a2b0a27dSPhilippe Mathieu-Daudé #endif
1812a2b0a27dSPhilippe Mathieu-Daudé
1813a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_PH(name, func) \
1814a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_ph(target_ulong sa, target_ulong rt, \
1815a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1816a2b0a27dSPhilippe Mathieu-Daudé { \
1817a2b0a27dSPhilippe Mathieu-Daudé uint16_t rth, rtl; \
1818a2b0a27dSPhilippe Mathieu-Daudé \
1819a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x0F; \
1820a2b0a27dSPhilippe Mathieu-Daudé \
1821a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_16(rt, rth, rtl); \
1822a2b0a27dSPhilippe Mathieu-Daudé \
1823a2b0a27dSPhilippe Mathieu-Daudé rth = mipsdsp_##func(rth, sa, env); \
1824a2b0a27dSPhilippe Mathieu-Daudé rtl = mipsdsp_##func(rtl, sa, env); \
1825a2b0a27dSPhilippe Mathieu-Daudé \
1826a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(rth, rtl); \
1827a2b0a27dSPhilippe Mathieu-Daudé }
1828a2b0a27dSPhilippe Mathieu-Daudé
1829a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PH(shll, lshift16);
1830a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PH(shll_s, sat16_lshift);
1831a2b0a27dSPhilippe Mathieu-Daudé
1832a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_PH
1833a2b0a27dSPhilippe Mathieu-Daudé
1834a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
1835a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_QH(name, func) \
1836a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_qh(target_ulong rt, target_ulong sa) \
1837a2b0a27dSPhilippe Mathieu-Daudé { \
1838a2b0a27dSPhilippe Mathieu-Daudé uint16_t rt3, rt2, rt1, rt0; \
1839a2b0a27dSPhilippe Mathieu-Daudé \
1840a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x0F; \
1841a2b0a27dSPhilippe Mathieu-Daudé \
1842a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rt, rt3, rt2, rt1, rt0); \
1843a2b0a27dSPhilippe Mathieu-Daudé \
1844a2b0a27dSPhilippe Mathieu-Daudé rt3 = mipsdsp_##func(rt3, sa); \
1845a2b0a27dSPhilippe Mathieu-Daudé rt2 = mipsdsp_##func(rt2, sa); \
1846a2b0a27dSPhilippe Mathieu-Daudé rt1 = mipsdsp_##func(rt1, sa); \
1847a2b0a27dSPhilippe Mathieu-Daudé rt0 = mipsdsp_##func(rt0, sa); \
1848a2b0a27dSPhilippe Mathieu-Daudé \
1849a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(rt3, rt2, rt1, rt0); \
1850a2b0a27dSPhilippe Mathieu-Daudé }
1851a2b0a27dSPhilippe Mathieu-Daudé
1852a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_QH_ENV(name, func) \
1853a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_qh(target_ulong rt, target_ulong sa, \
1854a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1855a2b0a27dSPhilippe Mathieu-Daudé { \
1856a2b0a27dSPhilippe Mathieu-Daudé uint16_t rt3, rt2, rt1, rt0; \
1857a2b0a27dSPhilippe Mathieu-Daudé \
1858a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x0F; \
1859a2b0a27dSPhilippe Mathieu-Daudé \
1860a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rt, rt3, rt2, rt1, rt0); \
1861a2b0a27dSPhilippe Mathieu-Daudé \
1862a2b0a27dSPhilippe Mathieu-Daudé rt3 = mipsdsp_##func(rt3, sa, env); \
1863a2b0a27dSPhilippe Mathieu-Daudé rt2 = mipsdsp_##func(rt2, sa, env); \
1864a2b0a27dSPhilippe Mathieu-Daudé rt1 = mipsdsp_##func(rt1, sa, env); \
1865a2b0a27dSPhilippe Mathieu-Daudé rt0 = mipsdsp_##func(rt0, sa, env); \
1866a2b0a27dSPhilippe Mathieu-Daudé \
1867a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(rt3, rt2, rt1, rt0); \
1868a2b0a27dSPhilippe Mathieu-Daudé }
1869a2b0a27dSPhilippe Mathieu-Daudé
1870a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QH_ENV(shll, lshift16);
1871a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QH_ENV(shll_s, sat16_lshift);
1872a2b0a27dSPhilippe Mathieu-Daudé
1873a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QH(shrl, rshift_u16);
1874a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QH(shra, rashift16);
1875a2b0a27dSPhilippe Mathieu-Daudé SHIFT_QH(shra_r, rnd16_rashift);
1876a2b0a27dSPhilippe Mathieu-Daudé
1877a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_QH
1878a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_QH_ENV
1879a2b0a27dSPhilippe Mathieu-Daudé
1880a2b0a27dSPhilippe Mathieu-Daudé #endif
1881a2b0a27dSPhilippe Mathieu-Daudé
1882a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_W(name, func) \
1883a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_w(target_ulong sa, target_ulong rt) \
1884a2b0a27dSPhilippe Mathieu-Daudé { \
1885a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp; \
1886a2b0a27dSPhilippe Mathieu-Daudé \
1887a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x1F; \
1888a2b0a27dSPhilippe Mathieu-Daudé temp = mipsdsp_##func(rt, sa); \
1889a2b0a27dSPhilippe Mathieu-Daudé \
1890a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)temp; \
1891a2b0a27dSPhilippe Mathieu-Daudé }
1892a2b0a27dSPhilippe Mathieu-Daudé
1893a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_W_ENV(name, func) \
1894a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_w(target_ulong sa, target_ulong rt, \
1895a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1896a2b0a27dSPhilippe Mathieu-Daudé { \
1897a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp; \
1898a2b0a27dSPhilippe Mathieu-Daudé \
1899a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x1F; \
1900a2b0a27dSPhilippe Mathieu-Daudé temp = mipsdsp_##func(rt, sa, env); \
1901a2b0a27dSPhilippe Mathieu-Daudé \
1902a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)temp; \
1903a2b0a27dSPhilippe Mathieu-Daudé }
1904a2b0a27dSPhilippe Mathieu-Daudé
1905a2b0a27dSPhilippe Mathieu-Daudé SHIFT_W_ENV(shll_s, sat32_lshift);
1906a2b0a27dSPhilippe Mathieu-Daudé SHIFT_W(shra_r, rnd32_rashift);
1907a2b0a27dSPhilippe Mathieu-Daudé
1908a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_W
1909a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_W_ENV
1910a2b0a27dSPhilippe Mathieu-Daudé
1911a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
1912a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_PW(name, func) \
1913a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_pw(target_ulong rt, target_ulong sa) \
1914a2b0a27dSPhilippe Mathieu-Daudé { \
1915a2b0a27dSPhilippe Mathieu-Daudé uint32_t rt1, rt0; \
1916a2b0a27dSPhilippe Mathieu-Daudé \
1917a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x1F; \
1918a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_32(rt, rt1, rt0); \
1919a2b0a27dSPhilippe Mathieu-Daudé \
1920a2b0a27dSPhilippe Mathieu-Daudé rt1 = mipsdsp_##func(rt1, sa); \
1921a2b0a27dSPhilippe Mathieu-Daudé rt0 = mipsdsp_##func(rt0, sa); \
1922a2b0a27dSPhilippe Mathieu-Daudé \
1923a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_32(rt1, rt0); \
1924a2b0a27dSPhilippe Mathieu-Daudé }
1925a2b0a27dSPhilippe Mathieu-Daudé
1926a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_PW_ENV(name, func) \
1927a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_pw(target_ulong rt, target_ulong sa, \
1928a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1929a2b0a27dSPhilippe Mathieu-Daudé { \
1930a2b0a27dSPhilippe Mathieu-Daudé uint32_t rt1, rt0; \
1931a2b0a27dSPhilippe Mathieu-Daudé \
1932a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x1F; \
1933a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_32(rt, rt1, rt0); \
1934a2b0a27dSPhilippe Mathieu-Daudé \
1935a2b0a27dSPhilippe Mathieu-Daudé rt1 = mipsdsp_##func(rt1, sa, env); \
1936a2b0a27dSPhilippe Mathieu-Daudé rt0 = mipsdsp_##func(rt0, sa, env); \
1937a2b0a27dSPhilippe Mathieu-Daudé \
1938a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_32(rt1, rt0); \
1939a2b0a27dSPhilippe Mathieu-Daudé }
1940a2b0a27dSPhilippe Mathieu-Daudé
1941a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PW_ENV(shll, lshift32);
1942a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PW_ENV(shll_s, sat32_lshift);
1943a2b0a27dSPhilippe Mathieu-Daudé
1944a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PW(shra, rashift32);
1945a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PW(shra_r, rnd32_rashift);
1946a2b0a27dSPhilippe Mathieu-Daudé
1947a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_PW
1948a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_PW_ENV
1949a2b0a27dSPhilippe Mathieu-Daudé
1950a2b0a27dSPhilippe Mathieu-Daudé #endif
1951a2b0a27dSPhilippe Mathieu-Daudé
1952a2b0a27dSPhilippe Mathieu-Daudé #define SHIFT_PH(name, func) \
1953a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name##_ph(target_ulong sa, target_ulong rt) \
1954a2b0a27dSPhilippe Mathieu-Daudé { \
1955a2b0a27dSPhilippe Mathieu-Daudé uint16_t rth, rtl; \
1956a2b0a27dSPhilippe Mathieu-Daudé \
1957a2b0a27dSPhilippe Mathieu-Daudé sa = sa & 0x0F; \
1958a2b0a27dSPhilippe Mathieu-Daudé \
1959a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_16(rt, rth, rtl); \
1960a2b0a27dSPhilippe Mathieu-Daudé \
1961a2b0a27dSPhilippe Mathieu-Daudé rth = mipsdsp_##func(rth, sa); \
1962a2b0a27dSPhilippe Mathieu-Daudé rtl = mipsdsp_##func(rtl, sa); \
1963a2b0a27dSPhilippe Mathieu-Daudé \
1964a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(rth, rtl); \
1965a2b0a27dSPhilippe Mathieu-Daudé }
1966a2b0a27dSPhilippe Mathieu-Daudé
1967a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PH(shrl, rshift_u16);
1968a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PH(shra, rashift16);
1969a2b0a27dSPhilippe Mathieu-Daudé SHIFT_PH(shra_r, rnd16_rashift);
1970a2b0a27dSPhilippe Mathieu-Daudé
1971a2b0a27dSPhilippe Mathieu-Daudé #undef SHIFT_PH
1972a2b0a27dSPhilippe Mathieu-Daudé
1973a2b0a27dSPhilippe Mathieu-Daudé /** DSP Multiply Sub-class insns **/
1974a2b0a27dSPhilippe Mathieu-Daudé /*
1975a2b0a27dSPhilippe Mathieu-Daudé * Return value made up by two 16bits value.
1976a2b0a27dSPhilippe Mathieu-Daudé * FIXME give the macro a better name.
1977a2b0a27dSPhilippe Mathieu-Daudé */
1978a2b0a27dSPhilippe Mathieu-Daudé #define MUL_RETURN32_16_PH(name, func, \
1979a2b0a27dSPhilippe Mathieu-Daudé rsmov1, rsmov2, rsfilter, \
1980a2b0a27dSPhilippe Mathieu-Daudé rtmov1, rtmov2, rtfilter) \
1981a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
1982a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
1983a2b0a27dSPhilippe Mathieu-Daudé { \
1984a2b0a27dSPhilippe Mathieu-Daudé uint16_t rsB, rsA, rtB, rtA; \
1985a2b0a27dSPhilippe Mathieu-Daudé \
1986a2b0a27dSPhilippe Mathieu-Daudé rsB = (rs >> rsmov1) & rsfilter; \
1987a2b0a27dSPhilippe Mathieu-Daudé rsA = (rs >> rsmov2) & rsfilter; \
1988a2b0a27dSPhilippe Mathieu-Daudé rtB = (rt >> rtmov1) & rtfilter; \
1989a2b0a27dSPhilippe Mathieu-Daudé rtA = (rt >> rtmov2) & rtfilter; \
1990a2b0a27dSPhilippe Mathieu-Daudé \
1991a2b0a27dSPhilippe Mathieu-Daudé rsB = mipsdsp_##func(rsB, rtB, env); \
1992a2b0a27dSPhilippe Mathieu-Daudé rsA = mipsdsp_##func(rsA, rtA, env); \
1993a2b0a27dSPhilippe Mathieu-Daudé \
1994a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN32_16(rsB, rsA); \
1995a2b0a27dSPhilippe Mathieu-Daudé }
1996a2b0a27dSPhilippe Mathieu-Daudé
1997a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_16_PH(muleu_s_ph_qbl, mul_u8_u16, \
1998a2b0a27dSPhilippe Mathieu-Daudé 24, 16, MIPSDSP_Q0, \
1999a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO);
2000a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_16_PH(muleu_s_ph_qbr, mul_u8_u16, \
2001a2b0a27dSPhilippe Mathieu-Daudé 8, 0, MIPSDSP_Q0, \
2002a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO);
2003a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_16_PH(mulq_rs_ph, rndq15_mul_q15_q15, \
2004a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO, \
2005a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO);
2006a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_16_PH(mul_ph, mul_i16_i16, \
2007a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO, \
2008a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO);
2009a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_16_PH(mul_s_ph, sat16_mul_i16_i16, \
2010a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO, \
2011a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO);
2012a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_16_PH(mulq_s_ph, sat16_mul_q15_q15, \
2013a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO, \
2014a2b0a27dSPhilippe Mathieu-Daudé 16, 0, MIPSDSP_LO);
2015a2b0a27dSPhilippe Mathieu-Daudé
2016a2b0a27dSPhilippe Mathieu-Daudé #undef MUL_RETURN32_16_PH
2017a2b0a27dSPhilippe Mathieu-Daudé
2018a2b0a27dSPhilippe Mathieu-Daudé #define MUL_RETURN32_32_ph(name, func, movbits) \
2019a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
2020a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2021a2b0a27dSPhilippe Mathieu-Daudé { \
2022a2b0a27dSPhilippe Mathieu-Daudé int16_t rsh, rth; \
2023a2b0a27dSPhilippe Mathieu-Daudé int32_t temp; \
2024a2b0a27dSPhilippe Mathieu-Daudé \
2025a2b0a27dSPhilippe Mathieu-Daudé rsh = (rs >> movbits) & MIPSDSP_LO; \
2026a2b0a27dSPhilippe Mathieu-Daudé rth = (rt >> movbits) & MIPSDSP_LO; \
2027a2b0a27dSPhilippe Mathieu-Daudé temp = mipsdsp_##func(rsh, rth, env); \
2028a2b0a27dSPhilippe Mathieu-Daudé \
2029a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)temp; \
2030a2b0a27dSPhilippe Mathieu-Daudé }
2031a2b0a27dSPhilippe Mathieu-Daudé
2032a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_32_ph(muleq_s_w_phl, mul_q15_q15_overflowflag21, 16);
2033a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN32_32_ph(muleq_s_w_phr, mul_q15_q15_overflowflag21, 0);
2034a2b0a27dSPhilippe Mathieu-Daudé
2035a2b0a27dSPhilippe Mathieu-Daudé #undef MUL_RETURN32_32_ph
2036a2b0a27dSPhilippe Mathieu-Daudé
2037a2b0a27dSPhilippe Mathieu-Daudé #define MUL_VOID_PH(name, use_ac_env) \
2038a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2039a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2040a2b0a27dSPhilippe Mathieu-Daudé { \
2041a2b0a27dSPhilippe Mathieu-Daudé int16_t rsh, rsl, rth, rtl; \
2042a2b0a27dSPhilippe Mathieu-Daudé int32_t tempB, tempA; \
2043a2b0a27dSPhilippe Mathieu-Daudé int64_t acc, dotp; \
2044a2b0a27dSPhilippe Mathieu-Daudé \
2045a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_16(rs, rsh, rsl); \
2046a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_16(rt, rth, rtl); \
2047a2b0a27dSPhilippe Mathieu-Daudé \
2048a2b0a27dSPhilippe Mathieu-Daudé if (use_ac_env == 1) { \
2049a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_q15_q15(ac, rsh, rth, env); \
2050a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rsl, rtl, env); \
2051a2b0a27dSPhilippe Mathieu-Daudé } else { \
2052a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_u16_u16(rsh, rth); \
2053a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_u16_u16(rsl, rtl); \
2054a2b0a27dSPhilippe Mathieu-Daudé } \
2055a2b0a27dSPhilippe Mathieu-Daudé \
2056a2b0a27dSPhilippe Mathieu-Daudé dotp = (int64_t)tempB - (int64_t)tempA; \
2057a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) | \
2058a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO); \
2059a2b0a27dSPhilippe Mathieu-Daudé dotp = dotp + acc; \
2060a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t) \
2061a2b0a27dSPhilippe Mathieu-Daudé ((dotp & MIPSDSP_LHI) >> 32); \
2062a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t)(dotp & MIPSDSP_LLO); \
2063a2b0a27dSPhilippe Mathieu-Daudé }
2064a2b0a27dSPhilippe Mathieu-Daudé
2065a2b0a27dSPhilippe Mathieu-Daudé MUL_VOID_PH(mulsaq_s_w_ph, 1);
2066a2b0a27dSPhilippe Mathieu-Daudé MUL_VOID_PH(mulsa_w_ph, 0);
2067a2b0a27dSPhilippe Mathieu-Daudé
2068a2b0a27dSPhilippe Mathieu-Daudé #undef MUL_VOID_PH
2069a2b0a27dSPhilippe Mathieu-Daudé
2070a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
2071a2b0a27dSPhilippe Mathieu-Daudé #define MUL_RETURN64_16_QH(name, func, \
2072a2b0a27dSPhilippe Mathieu-Daudé rsmov1, rsmov2, rsmov3, rsmov4, rsfilter, \
2073a2b0a27dSPhilippe Mathieu-Daudé rtmov1, rtmov2, rtmov3, rtmov4, rtfilter) \
2074a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
2075a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2076a2b0a27dSPhilippe Mathieu-Daudé { \
2077a2b0a27dSPhilippe Mathieu-Daudé uint16_t rs3, rs2, rs1, rs0; \
2078a2b0a27dSPhilippe Mathieu-Daudé uint16_t rt3, rt2, rt1, rt0; \
2079a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA; \
2080a2b0a27dSPhilippe Mathieu-Daudé \
2081a2b0a27dSPhilippe Mathieu-Daudé rs3 = (rs >> rsmov1) & rsfilter; \
2082a2b0a27dSPhilippe Mathieu-Daudé rs2 = (rs >> rsmov2) & rsfilter; \
2083a2b0a27dSPhilippe Mathieu-Daudé rs1 = (rs >> rsmov3) & rsfilter; \
2084a2b0a27dSPhilippe Mathieu-Daudé rs0 = (rs >> rsmov4) & rsfilter; \
2085a2b0a27dSPhilippe Mathieu-Daudé rt3 = (rt >> rtmov1) & rtfilter; \
2086a2b0a27dSPhilippe Mathieu-Daudé rt2 = (rt >> rtmov2) & rtfilter; \
2087a2b0a27dSPhilippe Mathieu-Daudé rt1 = (rt >> rtmov3) & rtfilter; \
2088a2b0a27dSPhilippe Mathieu-Daudé rt0 = (rt >> rtmov4) & rtfilter; \
2089a2b0a27dSPhilippe Mathieu-Daudé \
2090a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_##func(rs3, rt3, env); \
2091a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_##func(rs2, rt2, env); \
2092a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_##func(rs1, rt1, env); \
2093a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_##func(rs0, rt0, env); \
2094a2b0a27dSPhilippe Mathieu-Daudé \
2095a2b0a27dSPhilippe Mathieu-Daudé return MIPSDSP_RETURN64_16(tempD, tempC, tempB, tempA); \
2096a2b0a27dSPhilippe Mathieu-Daudé }
2097a2b0a27dSPhilippe Mathieu-Daudé
2098a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN64_16_QH(muleu_s_qh_obl, mul_u8_u16, \
2099a2b0a27dSPhilippe Mathieu-Daudé 56, 48, 40, 32, MIPSDSP_Q0, \
2100a2b0a27dSPhilippe Mathieu-Daudé 48, 32, 16, 0, MIPSDSP_LO);
2101a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN64_16_QH(muleu_s_qh_obr, mul_u8_u16, \
2102a2b0a27dSPhilippe Mathieu-Daudé 24, 16, 8, 0, MIPSDSP_Q0, \
2103a2b0a27dSPhilippe Mathieu-Daudé 48, 32, 16, 0, MIPSDSP_LO);
2104a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN64_16_QH(mulq_rs_qh, rndq15_mul_q15_q15, \
2105a2b0a27dSPhilippe Mathieu-Daudé 48, 32, 16, 0, MIPSDSP_LO, \
2106a2b0a27dSPhilippe Mathieu-Daudé 48, 32, 16, 0, MIPSDSP_LO);
2107a2b0a27dSPhilippe Mathieu-Daudé
2108a2b0a27dSPhilippe Mathieu-Daudé #undef MUL_RETURN64_16_QH
2109a2b0a27dSPhilippe Mathieu-Daudé
2110a2b0a27dSPhilippe Mathieu-Daudé #define MUL_RETURN64_32_QH(name, \
2111a2b0a27dSPhilippe Mathieu-Daudé rsmov1, rsmov2, \
2112a2b0a27dSPhilippe Mathieu-Daudé rtmov1, rtmov2) \
2113a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
2114a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2115a2b0a27dSPhilippe Mathieu-Daudé { \
2116a2b0a27dSPhilippe Mathieu-Daudé uint16_t rsB, rsA; \
2117a2b0a27dSPhilippe Mathieu-Daudé uint16_t rtB, rtA; \
2118a2b0a27dSPhilippe Mathieu-Daudé uint32_t tempB, tempA; \
2119a2b0a27dSPhilippe Mathieu-Daudé \
2120a2b0a27dSPhilippe Mathieu-Daudé rsB = (rs >> rsmov1) & MIPSDSP_LO; \
2121a2b0a27dSPhilippe Mathieu-Daudé rsA = (rs >> rsmov2) & MIPSDSP_LO; \
2122a2b0a27dSPhilippe Mathieu-Daudé rtB = (rt >> rtmov1) & MIPSDSP_LO; \
2123a2b0a27dSPhilippe Mathieu-Daudé rtA = (rt >> rtmov2) & MIPSDSP_LO; \
2124a2b0a27dSPhilippe Mathieu-Daudé \
2125a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_q15_q15(5, rsB, rtB, env); \
2126a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(5, rsA, rtA, env); \
2127a2b0a27dSPhilippe Mathieu-Daudé \
2128a2b0a27dSPhilippe Mathieu-Daudé return ((uint64_t)tempB << 32) | (uint64_t)tempA; \
2129a2b0a27dSPhilippe Mathieu-Daudé }
2130a2b0a27dSPhilippe Mathieu-Daudé
2131a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN64_32_QH(muleq_s_pw_qhl, 48, 32, 48, 32);
2132a2b0a27dSPhilippe Mathieu-Daudé MUL_RETURN64_32_QH(muleq_s_pw_qhr, 16, 0, 16, 0);
2133a2b0a27dSPhilippe Mathieu-Daudé
2134a2b0a27dSPhilippe Mathieu-Daudé #undef MUL_RETURN64_32_QH
2135a2b0a27dSPhilippe Mathieu-Daudé
helper_mulsaq_s_w_qh(target_ulong rs,target_ulong rt,uint32_t ac,CPUMIPSState * env)2136a2b0a27dSPhilippe Mathieu-Daudé void helper_mulsaq_s_w_qh(target_ulong rs, target_ulong rt, uint32_t ac,
2137a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
2138a2b0a27dSPhilippe Mathieu-Daudé {
2139a2b0a27dSPhilippe Mathieu-Daudé int16_t rs3, rs2, rs1, rs0;
2140a2b0a27dSPhilippe Mathieu-Daudé int16_t rt3, rt2, rt1, rt0;
2141a2b0a27dSPhilippe Mathieu-Daudé int32_t tempD, tempC, tempB, tempA;
2142a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2];
2143a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2];
2144a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum;
2145a2b0a27dSPhilippe Mathieu-Daudé
2146a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rs, rs3, rs2, rs1, rs0);
2147a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rt, rt3, rt2, rt1, rt0);
2148a2b0a27dSPhilippe Mathieu-Daudé
2149a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_mul_q15_q15(ac, rs3, rt3, env);
2150a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_mul_q15_q15(ac, rs2, rt2, env);
2151a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_q15_q15(ac, rs1, rt1, env);
2152a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rs0, rt0, env);
2153a2b0a27dSPhilippe Mathieu-Daudé
2154a2b0a27dSPhilippe Mathieu-Daudé temp[0] = ((int32_t)tempD - (int32_t)tempC) +
2155a2b0a27dSPhilippe Mathieu-Daudé ((int32_t)tempB - (int32_t)tempA);
2156a2b0a27dSPhilippe Mathieu-Daudé temp[0] = (int64_t)(temp[0] << 30) >> 30;
2157a2b0a27dSPhilippe Mathieu-Daudé if (((temp[0] >> 33) & 0x01) == 0) {
2158a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0x00;
2159a2b0a27dSPhilippe Mathieu-Daudé } else {
2160a2b0a27dSPhilippe Mathieu-Daudé temp[1] = ~0ull;
2161a2b0a27dSPhilippe Mathieu-Daudé }
2162a2b0a27dSPhilippe Mathieu-Daudé
2163a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac];
2164a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac];
2165a2b0a27dSPhilippe Mathieu-Daudé
2166a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0];
2167a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) &&
2168a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) {
2169a2b0a27dSPhilippe Mathieu-Daudé acc[1] += 1;
2170a2b0a27dSPhilippe Mathieu-Daudé }
2171a2b0a27dSPhilippe Mathieu-Daudé acc[0] = temp_sum;
2172a2b0a27dSPhilippe Mathieu-Daudé acc[1] += temp[1];
2173a2b0a27dSPhilippe Mathieu-Daudé
2174a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = acc[1];
2175a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = acc[0];
2176a2b0a27dSPhilippe Mathieu-Daudé }
2177a2b0a27dSPhilippe Mathieu-Daudé #endif
2178a2b0a27dSPhilippe Mathieu-Daudé
2179a2b0a27dSPhilippe Mathieu-Daudé #define DP_QB(name, func, is_add, rsmov1, rsmov2, rtmov1, rtmov2) \
2180a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2181a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2182a2b0a27dSPhilippe Mathieu-Daudé { \
2183a2b0a27dSPhilippe Mathieu-Daudé uint8_t rs3, rs2; \
2184a2b0a27dSPhilippe Mathieu-Daudé uint8_t rt3, rt2; \
2185a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempB, tempA; \
2186a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempC, dotp; \
2187a2b0a27dSPhilippe Mathieu-Daudé \
2188a2b0a27dSPhilippe Mathieu-Daudé rs3 = (rs >> rsmov1) & MIPSDSP_Q0; \
2189a2b0a27dSPhilippe Mathieu-Daudé rs2 = (rs >> rsmov2) & MIPSDSP_Q0; \
2190a2b0a27dSPhilippe Mathieu-Daudé rt3 = (rt >> rtmov1) & MIPSDSP_Q0; \
2191a2b0a27dSPhilippe Mathieu-Daudé rt2 = (rt >> rtmov2) & MIPSDSP_Q0; \
2192a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_##func(rs3, rt3); \
2193a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_##func(rs2, rt2); \
2194a2b0a27dSPhilippe Mathieu-Daudé dotp = (int64_t)tempB + (int64_t)tempA; \
2195a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2196a2b0a27dSPhilippe Mathieu-Daudé tempC = (((uint64_t)env->active_tc.HI[ac] << 32) | \
2197a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO)) \
2198a2b0a27dSPhilippe Mathieu-Daudé + dotp; \
2199a2b0a27dSPhilippe Mathieu-Daudé } else { \
2200a2b0a27dSPhilippe Mathieu-Daudé tempC = (((uint64_t)env->active_tc.HI[ac] << 32) | \
2201a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO)) \
2202a2b0a27dSPhilippe Mathieu-Daudé - dotp; \
2203a2b0a27dSPhilippe Mathieu-Daudé } \
2204a2b0a27dSPhilippe Mathieu-Daudé \
2205a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t) \
2206a2b0a27dSPhilippe Mathieu-Daudé ((tempC & MIPSDSP_LHI) >> 32); \
2207a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t)(tempC & MIPSDSP_LLO); \
2208a2b0a27dSPhilippe Mathieu-Daudé }
2209a2b0a27dSPhilippe Mathieu-Daudé
2210a2b0a27dSPhilippe Mathieu-Daudé DP_QB(dpau_h_qbl, mul_u8_u8, 1, 24, 16, 24, 16);
2211a2b0a27dSPhilippe Mathieu-Daudé DP_QB(dpau_h_qbr, mul_u8_u8, 1, 8, 0, 8, 0);
2212a2b0a27dSPhilippe Mathieu-Daudé DP_QB(dpsu_h_qbl, mul_u8_u8, 0, 24, 16, 24, 16);
2213a2b0a27dSPhilippe Mathieu-Daudé DP_QB(dpsu_h_qbr, mul_u8_u8, 0, 8, 0, 8, 0);
2214a2b0a27dSPhilippe Mathieu-Daudé
2215a2b0a27dSPhilippe Mathieu-Daudé #undef DP_QB
2216a2b0a27dSPhilippe Mathieu-Daudé
2217a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
2218a2b0a27dSPhilippe Mathieu-Daudé #define DP_OB(name, add_sub, \
2219a2b0a27dSPhilippe Mathieu-Daudé rsmov1, rsmov2, rsmov3, rsmov4, \
2220a2b0a27dSPhilippe Mathieu-Daudé rtmov1, rtmov2, rtmov3, rtmov4) \
2221a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2222a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2223a2b0a27dSPhilippe Mathieu-Daudé { \
2224a2b0a27dSPhilippe Mathieu-Daudé uint8_t rsD, rsC, rsB, rsA; \
2225a2b0a27dSPhilippe Mathieu-Daudé uint8_t rtD, rtC, rtB, rtA; \
2226a2b0a27dSPhilippe Mathieu-Daudé uint16_t tempD, tempC, tempB, tempA; \
2227a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[2]; \
2228a2b0a27dSPhilippe Mathieu-Daudé uint64_t acc[2]; \
2229a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp_sum; \
2230a2b0a27dSPhilippe Mathieu-Daudé \
2231a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0; \
2232a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0; \
2233a2b0a27dSPhilippe Mathieu-Daudé \
2234a2b0a27dSPhilippe Mathieu-Daudé rsD = (rs >> rsmov1) & MIPSDSP_Q0; \
2235a2b0a27dSPhilippe Mathieu-Daudé rsC = (rs >> rsmov2) & MIPSDSP_Q0; \
2236a2b0a27dSPhilippe Mathieu-Daudé rsB = (rs >> rsmov3) & MIPSDSP_Q0; \
2237a2b0a27dSPhilippe Mathieu-Daudé rsA = (rs >> rsmov4) & MIPSDSP_Q0; \
2238a2b0a27dSPhilippe Mathieu-Daudé rtD = (rt >> rtmov1) & MIPSDSP_Q0; \
2239a2b0a27dSPhilippe Mathieu-Daudé rtC = (rt >> rtmov2) & MIPSDSP_Q0; \
2240a2b0a27dSPhilippe Mathieu-Daudé rtB = (rt >> rtmov3) & MIPSDSP_Q0; \
2241a2b0a27dSPhilippe Mathieu-Daudé rtA = (rt >> rtmov4) & MIPSDSP_Q0; \
2242a2b0a27dSPhilippe Mathieu-Daudé \
2243a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_mul_u8_u8(rsD, rtD); \
2244a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_mul_u8_u8(rsC, rtC); \
2245a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_u8_u8(rsB, rtB); \
2246a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_u8_u8(rsA, rtA); \
2247a2b0a27dSPhilippe Mathieu-Daudé \
2248a2b0a27dSPhilippe Mathieu-Daudé temp[0] = (uint64_t)tempD + (uint64_t)tempC + \
2249a2b0a27dSPhilippe Mathieu-Daudé (uint64_t)tempB + (uint64_t)tempA; \
2250a2b0a27dSPhilippe Mathieu-Daudé \
2251a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac]; \
2252a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac]; \
2253a2b0a27dSPhilippe Mathieu-Daudé \
2254a2b0a27dSPhilippe Mathieu-Daudé if (add_sub) { \
2255a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0]; \
2256a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) && \
2257a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) { \
2258a2b0a27dSPhilippe Mathieu-Daudé acc[1] += 1; \
2259a2b0a27dSPhilippe Mathieu-Daudé } \
2260a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2261a2b0a27dSPhilippe Mathieu-Daudé temp[1] = acc[1] + temp[1]; \
2262a2b0a27dSPhilippe Mathieu-Daudé } else { \
2263a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] - temp[0]; \
2264a2b0a27dSPhilippe Mathieu-Daudé if ((uint64_t)temp_sum > (uint64_t)acc[0]) { \
2265a2b0a27dSPhilippe Mathieu-Daudé acc[1] -= 1; \
2266a2b0a27dSPhilippe Mathieu-Daudé } \
2267a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2268a2b0a27dSPhilippe Mathieu-Daudé temp[1] = acc[1] - temp[1]; \
2269a2b0a27dSPhilippe Mathieu-Daudé } \
2270a2b0a27dSPhilippe Mathieu-Daudé \
2271a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = temp[1]; \
2272a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = temp[0]; \
2273a2b0a27dSPhilippe Mathieu-Daudé }
2274a2b0a27dSPhilippe Mathieu-Daudé
2275a2b0a27dSPhilippe Mathieu-Daudé DP_OB(dpau_h_obl, 1, 56, 48, 40, 32, 56, 48, 40, 32);
2276a2b0a27dSPhilippe Mathieu-Daudé DP_OB(dpau_h_obr, 1, 24, 16, 8, 0, 24, 16, 8, 0);
2277a2b0a27dSPhilippe Mathieu-Daudé DP_OB(dpsu_h_obl, 0, 56, 48, 40, 32, 56, 48, 40, 32);
2278a2b0a27dSPhilippe Mathieu-Daudé DP_OB(dpsu_h_obr, 0, 24, 16, 8, 0, 24, 16, 8, 0);
2279a2b0a27dSPhilippe Mathieu-Daudé
2280a2b0a27dSPhilippe Mathieu-Daudé #undef DP_OB
2281a2b0a27dSPhilippe Mathieu-Daudé #endif
2282a2b0a27dSPhilippe Mathieu-Daudé
2283a2b0a27dSPhilippe Mathieu-Daudé #define DP_NOFUNC_PH(name, is_add, rsmov1, rsmov2, rtmov1, rtmov2) \
2284a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2285a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2286a2b0a27dSPhilippe Mathieu-Daudé { \
2287a2b0a27dSPhilippe Mathieu-Daudé int16_t rsB, rsA, rtB, rtA; \
2288a2b0a27dSPhilippe Mathieu-Daudé int32_t tempA, tempB; \
2289a2b0a27dSPhilippe Mathieu-Daudé int64_t acc; \
2290a2b0a27dSPhilippe Mathieu-Daudé \
2291a2b0a27dSPhilippe Mathieu-Daudé rsB = (rs >> rsmov1) & MIPSDSP_LO; \
2292a2b0a27dSPhilippe Mathieu-Daudé rsA = (rs >> rsmov2) & MIPSDSP_LO; \
2293a2b0a27dSPhilippe Mathieu-Daudé rtB = (rt >> rtmov1) & MIPSDSP_LO; \
2294a2b0a27dSPhilippe Mathieu-Daudé rtA = (rt >> rtmov2) & MIPSDSP_LO; \
2295a2b0a27dSPhilippe Mathieu-Daudé \
2296a2b0a27dSPhilippe Mathieu-Daudé tempB = (int32_t)rsB * (int32_t)rtB; \
2297a2b0a27dSPhilippe Mathieu-Daudé tempA = (int32_t)rsA * (int32_t)rtA; \
2298a2b0a27dSPhilippe Mathieu-Daudé \
2299a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) | \
2300a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO); \
2301a2b0a27dSPhilippe Mathieu-Daudé \
2302a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2303a2b0a27dSPhilippe Mathieu-Daudé acc = acc + ((int64_t)tempB + (int64_t)tempA); \
2304a2b0a27dSPhilippe Mathieu-Daudé } else { \
2305a2b0a27dSPhilippe Mathieu-Daudé acc = acc - ((int64_t)tempB + (int64_t)tempA); \
2306a2b0a27dSPhilippe Mathieu-Daudé } \
2307a2b0a27dSPhilippe Mathieu-Daudé \
2308a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t)((acc & MIPSDSP_LHI) >> 32); \
2309a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t)(acc & MIPSDSP_LLO); \
2310a2b0a27dSPhilippe Mathieu-Daudé }
2311a2b0a27dSPhilippe Mathieu-Daudé
2312a2b0a27dSPhilippe Mathieu-Daudé DP_NOFUNC_PH(dpa_w_ph, 1, 16, 0, 16, 0);
2313a2b0a27dSPhilippe Mathieu-Daudé DP_NOFUNC_PH(dpax_w_ph, 1, 16, 0, 0, 16);
2314a2b0a27dSPhilippe Mathieu-Daudé DP_NOFUNC_PH(dps_w_ph, 0, 16, 0, 16, 0);
2315a2b0a27dSPhilippe Mathieu-Daudé DP_NOFUNC_PH(dpsx_w_ph, 0, 16, 0, 0, 16);
2316a2b0a27dSPhilippe Mathieu-Daudé #undef DP_NOFUNC_PH
2317a2b0a27dSPhilippe Mathieu-Daudé
2318a2b0a27dSPhilippe Mathieu-Daudé #define DP_HASFUNC_PH(name, is_add, rsmov1, rsmov2, rtmov1, rtmov2) \
2319a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2320a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2321a2b0a27dSPhilippe Mathieu-Daudé { \
2322a2b0a27dSPhilippe Mathieu-Daudé int16_t rsB, rsA, rtB, rtA; \
2323a2b0a27dSPhilippe Mathieu-Daudé int32_t tempB, tempA; \
2324a2b0a27dSPhilippe Mathieu-Daudé int64_t acc, dotp; \
2325a2b0a27dSPhilippe Mathieu-Daudé \
2326a2b0a27dSPhilippe Mathieu-Daudé rsB = (rs >> rsmov1) & MIPSDSP_LO; \
2327a2b0a27dSPhilippe Mathieu-Daudé rsA = (rs >> rsmov2) & MIPSDSP_LO; \
2328a2b0a27dSPhilippe Mathieu-Daudé rtB = (rt >> rtmov1) & MIPSDSP_LO; \
2329a2b0a27dSPhilippe Mathieu-Daudé rtA = (rt >> rtmov2) & MIPSDSP_LO; \
2330a2b0a27dSPhilippe Mathieu-Daudé \
2331a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_q15_q15(ac, rsB, rtB, env); \
2332a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rsA, rtA, env); \
2333a2b0a27dSPhilippe Mathieu-Daudé \
2334a2b0a27dSPhilippe Mathieu-Daudé dotp = (int64_t)tempB + (int64_t)tempA; \
2335a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) | \
2336a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO); \
2337a2b0a27dSPhilippe Mathieu-Daudé \
2338a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2339a2b0a27dSPhilippe Mathieu-Daudé acc = acc + dotp; \
2340a2b0a27dSPhilippe Mathieu-Daudé } else { \
2341a2b0a27dSPhilippe Mathieu-Daudé acc = acc - dotp; \
2342a2b0a27dSPhilippe Mathieu-Daudé } \
2343a2b0a27dSPhilippe Mathieu-Daudé \
2344a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t) \
2345a2b0a27dSPhilippe Mathieu-Daudé ((acc & MIPSDSP_LHI) >> 32); \
2346a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t) \
2347a2b0a27dSPhilippe Mathieu-Daudé (acc & MIPSDSP_LLO); \
2348a2b0a27dSPhilippe Mathieu-Daudé }
2349a2b0a27dSPhilippe Mathieu-Daudé
2350a2b0a27dSPhilippe Mathieu-Daudé DP_HASFUNC_PH(dpaq_s_w_ph, 1, 16, 0, 16, 0);
2351a2b0a27dSPhilippe Mathieu-Daudé DP_HASFUNC_PH(dpaqx_s_w_ph, 1, 16, 0, 0, 16);
2352a2b0a27dSPhilippe Mathieu-Daudé DP_HASFUNC_PH(dpsq_s_w_ph, 0, 16, 0, 16, 0);
2353a2b0a27dSPhilippe Mathieu-Daudé DP_HASFUNC_PH(dpsqx_s_w_ph, 0, 16, 0, 0, 16);
2354a2b0a27dSPhilippe Mathieu-Daudé
2355a2b0a27dSPhilippe Mathieu-Daudé #undef DP_HASFUNC_PH
2356a2b0a27dSPhilippe Mathieu-Daudé
2357a2b0a27dSPhilippe Mathieu-Daudé #define DP_128OPERATION_PH(name, is_add) \
2358a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2359a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2360a2b0a27dSPhilippe Mathieu-Daudé { \
2361a2b0a27dSPhilippe Mathieu-Daudé int16_t rsh, rsl, rth, rtl; \
2362a2b0a27dSPhilippe Mathieu-Daudé int32_t tempB, tempA, tempC62_31, tempC63; \
2363a2b0a27dSPhilippe Mathieu-Daudé int64_t acc, dotp, tempC; \
2364a2b0a27dSPhilippe Mathieu-Daudé \
2365a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_16(rs, rsh, rsl); \
2366a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT32_16(rt, rth, rtl); \
2367a2b0a27dSPhilippe Mathieu-Daudé \
2368a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_q15_q15(ac, rsh, rtl, env); \
2369a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rsl, rth, env); \
2370a2b0a27dSPhilippe Mathieu-Daudé \
2371a2b0a27dSPhilippe Mathieu-Daudé dotp = (int64_t)tempB + (int64_t)tempA; \
2372a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) | \
2373a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO); \
2374a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2375a2b0a27dSPhilippe Mathieu-Daudé tempC = acc + dotp; \
2376a2b0a27dSPhilippe Mathieu-Daudé } else { \
2377a2b0a27dSPhilippe Mathieu-Daudé tempC = acc - dotp; \
2378a2b0a27dSPhilippe Mathieu-Daudé } \
2379a2b0a27dSPhilippe Mathieu-Daudé tempC63 = (tempC >> 63) & 0x01; \
2380a2b0a27dSPhilippe Mathieu-Daudé tempC62_31 = (tempC >> 31) & 0xFFFFFFFF; \
2381a2b0a27dSPhilippe Mathieu-Daudé \
2382a2b0a27dSPhilippe Mathieu-Daudé if ((tempC63 == 0) && (tempC62_31 != 0x00000000)) { \
2383a2b0a27dSPhilippe Mathieu-Daudé tempC = 0x7FFFFFFF; \
2384a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env); \
2385a2b0a27dSPhilippe Mathieu-Daudé } \
2386a2b0a27dSPhilippe Mathieu-Daudé \
2387a2b0a27dSPhilippe Mathieu-Daudé if ((tempC63 == 1) && (tempC62_31 != 0xFFFFFFFF)) { \
2388a2b0a27dSPhilippe Mathieu-Daudé tempC = (int64_t)(int32_t)0x80000000; \
2389a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env); \
2390a2b0a27dSPhilippe Mathieu-Daudé } \
2391a2b0a27dSPhilippe Mathieu-Daudé \
2392a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t) \
2393a2b0a27dSPhilippe Mathieu-Daudé ((tempC & MIPSDSP_LHI) >> 32); \
2394a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t) \
2395a2b0a27dSPhilippe Mathieu-Daudé (tempC & MIPSDSP_LLO); \
2396a2b0a27dSPhilippe Mathieu-Daudé }
2397a2b0a27dSPhilippe Mathieu-Daudé
2398a2b0a27dSPhilippe Mathieu-Daudé DP_128OPERATION_PH(dpaqx_sa_w_ph, 1);
2399a2b0a27dSPhilippe Mathieu-Daudé DP_128OPERATION_PH(dpsqx_sa_w_ph, 0);
2400a2b0a27dSPhilippe Mathieu-Daudé
2401a2b0a27dSPhilippe Mathieu-Daudé #undef DP_128OPERATION_HP
2402a2b0a27dSPhilippe Mathieu-Daudé
2403a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
2404a2b0a27dSPhilippe Mathieu-Daudé #define DP_QH(name, is_add, use_ac_env) \
2405a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2406a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2407a2b0a27dSPhilippe Mathieu-Daudé { \
2408a2b0a27dSPhilippe Mathieu-Daudé int32_t rs3, rs2, rs1, rs0; \
2409a2b0a27dSPhilippe Mathieu-Daudé int32_t rt3, rt2, rt1, rt0; \
2410a2b0a27dSPhilippe Mathieu-Daudé int32_t tempD, tempC, tempB, tempA; \
2411a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2]; \
2412a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2]; \
2413a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum; \
2414a2b0a27dSPhilippe Mathieu-Daudé \
2415a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rs, rs3, rs2, rs1, rs0); \
2416a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_16(rt, rt3, rt2, rt1, rt0); \
2417a2b0a27dSPhilippe Mathieu-Daudé \
2418a2b0a27dSPhilippe Mathieu-Daudé if (use_ac_env) { \
2419a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_mul_q15_q15(ac, rs3, rt3, env); \
2420a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_mul_q15_q15(ac, rs2, rt2, env); \
2421a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_q15_q15(ac, rs1, rt1, env); \
2422a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rs0, rt0, env); \
2423a2b0a27dSPhilippe Mathieu-Daudé } else { \
2424a2b0a27dSPhilippe Mathieu-Daudé tempD = mipsdsp_mul_u16_u16(rs3, rt3); \
2425a2b0a27dSPhilippe Mathieu-Daudé tempC = mipsdsp_mul_u16_u16(rs2, rt2); \
2426a2b0a27dSPhilippe Mathieu-Daudé tempB = mipsdsp_mul_u16_u16(rs1, rt1); \
2427a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_u16_u16(rs0, rt0); \
2428a2b0a27dSPhilippe Mathieu-Daudé } \
2429a2b0a27dSPhilippe Mathieu-Daudé \
2430a2b0a27dSPhilippe Mathieu-Daudé temp[0] = (int64_t)tempD + (int64_t)tempC + \
2431a2b0a27dSPhilippe Mathieu-Daudé (int64_t)tempB + (int64_t)tempA; \
2432a2b0a27dSPhilippe Mathieu-Daudé \
2433a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] >= 0) { \
2434a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0; \
2435a2b0a27dSPhilippe Mathieu-Daudé } else { \
2436a2b0a27dSPhilippe Mathieu-Daudé temp[1] = ~0ull; \
2437a2b0a27dSPhilippe Mathieu-Daudé } \
2438a2b0a27dSPhilippe Mathieu-Daudé \
2439a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac]; \
2440a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac]; \
2441a2b0a27dSPhilippe Mathieu-Daudé \
2442a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2443a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0]; \
2444a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) && \
2445a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) { \
2446a2b0a27dSPhilippe Mathieu-Daudé acc[1] = acc[1] + 1; \
2447a2b0a27dSPhilippe Mathieu-Daudé } \
2448a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2449a2b0a27dSPhilippe Mathieu-Daudé temp[1] = acc[1] + temp[1]; \
2450a2b0a27dSPhilippe Mathieu-Daudé } else { \
2451a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] - temp[0]; \
2452a2b0a27dSPhilippe Mathieu-Daudé if ((uint64_t)temp_sum > (uint64_t)acc[0]) { \
2453a2b0a27dSPhilippe Mathieu-Daudé acc[1] = acc[1] - 1; \
2454a2b0a27dSPhilippe Mathieu-Daudé } \
2455a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2456a2b0a27dSPhilippe Mathieu-Daudé temp[1] = acc[1] - temp[1]; \
2457a2b0a27dSPhilippe Mathieu-Daudé } \
2458a2b0a27dSPhilippe Mathieu-Daudé \
2459a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = temp[1]; \
2460a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = temp[0]; \
2461a2b0a27dSPhilippe Mathieu-Daudé }
2462a2b0a27dSPhilippe Mathieu-Daudé
2463a2b0a27dSPhilippe Mathieu-Daudé DP_QH(dpa_w_qh, 1, 0);
2464a2b0a27dSPhilippe Mathieu-Daudé DP_QH(dpaq_s_w_qh, 1, 1);
2465a2b0a27dSPhilippe Mathieu-Daudé DP_QH(dps_w_qh, 0, 0);
2466a2b0a27dSPhilippe Mathieu-Daudé DP_QH(dpsq_s_w_qh, 0, 1);
2467a2b0a27dSPhilippe Mathieu-Daudé
2468a2b0a27dSPhilippe Mathieu-Daudé #undef DP_QH
2469a2b0a27dSPhilippe Mathieu-Daudé
2470a2b0a27dSPhilippe Mathieu-Daudé #endif
2471a2b0a27dSPhilippe Mathieu-Daudé
2472a2b0a27dSPhilippe Mathieu-Daudé #define DP_L_W(name, is_add) \
2473a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2474a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2475a2b0a27dSPhilippe Mathieu-Daudé { \
2476a2b0a27dSPhilippe Mathieu-Daudé int32_t temp63; \
2477a2b0a27dSPhilippe Mathieu-Daudé int64_t dotp, acc; \
2478a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp; \
2479a2b0a27dSPhilippe Mathieu-Daudé bool overflow; \
2480a2b0a27dSPhilippe Mathieu-Daudé \
2481a2b0a27dSPhilippe Mathieu-Daudé dotp = mipsdsp_mul_q31_q31(ac, rs, rt, env); \
2482a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) | \
2483a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO); \
2484a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2485a2b0a27dSPhilippe Mathieu-Daudé temp = acc + dotp; \
2486a2b0a27dSPhilippe Mathieu-Daudé overflow = MIPSDSP_OVERFLOW_ADD((uint64_t)acc, (uint64_t)dotp, \
2487a2b0a27dSPhilippe Mathieu-Daudé temp, (0x01ull << 63)); \
2488a2b0a27dSPhilippe Mathieu-Daudé } else { \
2489a2b0a27dSPhilippe Mathieu-Daudé temp = acc - dotp; \
2490a2b0a27dSPhilippe Mathieu-Daudé overflow = MIPSDSP_OVERFLOW_SUB((uint64_t)acc, (uint64_t)dotp, \
2491a2b0a27dSPhilippe Mathieu-Daudé temp, (0x01ull << 63)); \
2492a2b0a27dSPhilippe Mathieu-Daudé } \
2493a2b0a27dSPhilippe Mathieu-Daudé \
2494a2b0a27dSPhilippe Mathieu-Daudé if (overflow) { \
2495a2b0a27dSPhilippe Mathieu-Daudé temp63 = (temp >> 63) & 0x01; \
2496a2b0a27dSPhilippe Mathieu-Daudé if (temp63 == 1) { \
2497a2b0a27dSPhilippe Mathieu-Daudé temp = (0x01ull << 63) - 1; \
2498a2b0a27dSPhilippe Mathieu-Daudé } else { \
2499a2b0a27dSPhilippe Mathieu-Daudé temp = 0x01ull << 63; \
2500a2b0a27dSPhilippe Mathieu-Daudé } \
2501a2b0a27dSPhilippe Mathieu-Daudé \
2502a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 16 + ac, env); \
2503a2b0a27dSPhilippe Mathieu-Daudé } \
2504a2b0a27dSPhilippe Mathieu-Daudé \
2505a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t) \
2506a2b0a27dSPhilippe Mathieu-Daudé ((temp & MIPSDSP_LHI) >> 32); \
2507a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t) \
2508a2b0a27dSPhilippe Mathieu-Daudé (temp & MIPSDSP_LLO); \
2509a2b0a27dSPhilippe Mathieu-Daudé }
2510a2b0a27dSPhilippe Mathieu-Daudé
2511a2b0a27dSPhilippe Mathieu-Daudé DP_L_W(dpaq_sa_l_w, 1);
2512a2b0a27dSPhilippe Mathieu-Daudé DP_L_W(dpsq_sa_l_w, 0);
2513a2b0a27dSPhilippe Mathieu-Daudé
2514a2b0a27dSPhilippe Mathieu-Daudé #undef DP_L_W
2515a2b0a27dSPhilippe Mathieu-Daudé
2516a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
2517a2b0a27dSPhilippe Mathieu-Daudé #define DP_L_PW(name, func) \
2518a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2519a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2520a2b0a27dSPhilippe Mathieu-Daudé { \
2521a2b0a27dSPhilippe Mathieu-Daudé int32_t rs1, rs0; \
2522a2b0a27dSPhilippe Mathieu-Daudé int32_t rt1, rt0; \
2523a2b0a27dSPhilippe Mathieu-Daudé int64_t tempB[2], tempA[2]; \
2524a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2]; \
2525a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2]; \
2526a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum; \
2527a2b0a27dSPhilippe Mathieu-Daudé \
2528a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0; \
2529a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0; \
2530a2b0a27dSPhilippe Mathieu-Daudé \
2531a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_32(rs, rs1, rs0); \
2532a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_32(rt, rt1, rt0); \
2533a2b0a27dSPhilippe Mathieu-Daudé \
2534a2b0a27dSPhilippe Mathieu-Daudé tempB[0] = mipsdsp_mul_q31_q31(ac, rs1, rt1, env); \
2535a2b0a27dSPhilippe Mathieu-Daudé tempA[0] = mipsdsp_mul_q31_q31(ac, rs0, rt0, env); \
2536a2b0a27dSPhilippe Mathieu-Daudé \
2537a2b0a27dSPhilippe Mathieu-Daudé if (tempB[0] >= 0) { \
2538a2b0a27dSPhilippe Mathieu-Daudé tempB[1] = 0x00; \
2539a2b0a27dSPhilippe Mathieu-Daudé } else { \
2540a2b0a27dSPhilippe Mathieu-Daudé tempB[1] = ~0ull; \
2541a2b0a27dSPhilippe Mathieu-Daudé } \
2542a2b0a27dSPhilippe Mathieu-Daudé \
2543a2b0a27dSPhilippe Mathieu-Daudé if (tempA[0] >= 0) { \
2544a2b0a27dSPhilippe Mathieu-Daudé tempA[1] = 0x00; \
2545a2b0a27dSPhilippe Mathieu-Daudé } else { \
2546a2b0a27dSPhilippe Mathieu-Daudé tempA[1] = ~0ull; \
2547a2b0a27dSPhilippe Mathieu-Daudé } \
2548a2b0a27dSPhilippe Mathieu-Daudé \
2549a2b0a27dSPhilippe Mathieu-Daudé temp_sum = tempB[0] + tempA[0]; \
2550a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)tempB[0]) && \
2551a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)tempA[0])) { \
2552a2b0a27dSPhilippe Mathieu-Daudé temp[1] += 1; \
2553a2b0a27dSPhilippe Mathieu-Daudé } \
2554a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2555a2b0a27dSPhilippe Mathieu-Daudé temp[1] += tempB[1] + tempA[1]; \
2556a2b0a27dSPhilippe Mathieu-Daudé \
2557a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_##func(acc, ac, temp, env); \
2558a2b0a27dSPhilippe Mathieu-Daudé \
2559a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = acc[1]; \
2560a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = acc[0]; \
2561a2b0a27dSPhilippe Mathieu-Daudé }
2562a2b0a27dSPhilippe Mathieu-Daudé
2563a2b0a27dSPhilippe Mathieu-Daudé DP_L_PW(dpaq_sa_l_pw, sat64_acc_add_q63);
2564a2b0a27dSPhilippe Mathieu-Daudé DP_L_PW(dpsq_sa_l_pw, sat64_acc_sub_q63);
2565a2b0a27dSPhilippe Mathieu-Daudé
2566a2b0a27dSPhilippe Mathieu-Daudé #undef DP_L_PW
2567a2b0a27dSPhilippe Mathieu-Daudé
helper_mulsaq_s_l_pw(target_ulong rs,target_ulong rt,uint32_t ac,CPUMIPSState * env)2568a2b0a27dSPhilippe Mathieu-Daudé void helper_mulsaq_s_l_pw(target_ulong rs, target_ulong rt, uint32_t ac,
2569a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
2570a2b0a27dSPhilippe Mathieu-Daudé {
2571a2b0a27dSPhilippe Mathieu-Daudé int32_t rs1, rs0;
2572a2b0a27dSPhilippe Mathieu-Daudé int32_t rt1, rt0;
2573a2b0a27dSPhilippe Mathieu-Daudé int64_t tempB[2], tempA[2];
2574a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2];
2575a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2];
2576a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum;
2577a2b0a27dSPhilippe Mathieu-Daudé
2578a2b0a27dSPhilippe Mathieu-Daudé rs1 = (rs >> 32) & MIPSDSP_LLO;
2579a2b0a27dSPhilippe Mathieu-Daudé rs0 = rs & MIPSDSP_LLO;
2580a2b0a27dSPhilippe Mathieu-Daudé rt1 = (rt >> 32) & MIPSDSP_LLO;
2581a2b0a27dSPhilippe Mathieu-Daudé rt0 = rt & MIPSDSP_LLO;
2582a2b0a27dSPhilippe Mathieu-Daudé
2583a2b0a27dSPhilippe Mathieu-Daudé tempB[0] = mipsdsp_mul_q31_q31(ac, rs1, rt1, env);
2584a2b0a27dSPhilippe Mathieu-Daudé tempA[0] = mipsdsp_mul_q31_q31(ac, rs0, rt0, env);
2585a2b0a27dSPhilippe Mathieu-Daudé
2586a2b0a27dSPhilippe Mathieu-Daudé if (tempB[0] >= 0) {
2587a2b0a27dSPhilippe Mathieu-Daudé tempB[1] = 0x00;
2588a2b0a27dSPhilippe Mathieu-Daudé } else {
2589a2b0a27dSPhilippe Mathieu-Daudé tempB[1] = ~0ull;
2590a2b0a27dSPhilippe Mathieu-Daudé }
2591a2b0a27dSPhilippe Mathieu-Daudé
2592a2b0a27dSPhilippe Mathieu-Daudé if (tempA[0] >= 0) {
2593a2b0a27dSPhilippe Mathieu-Daudé tempA[1] = 0x00;
2594a2b0a27dSPhilippe Mathieu-Daudé } else {
2595a2b0a27dSPhilippe Mathieu-Daudé tempA[1] = ~0ull;
2596a2b0a27dSPhilippe Mathieu-Daudé }
2597a2b0a27dSPhilippe Mathieu-Daudé
2598a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac];
2599a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac];
2600a2b0a27dSPhilippe Mathieu-Daudé
2601a2b0a27dSPhilippe Mathieu-Daudé temp_sum = tempB[0] - tempA[0];
2602a2b0a27dSPhilippe Mathieu-Daudé if ((uint64_t)temp_sum > (uint64_t)tempB[0]) {
2603a2b0a27dSPhilippe Mathieu-Daudé tempB[1] -= 1;
2604a2b0a27dSPhilippe Mathieu-Daudé }
2605a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum;
2606a2b0a27dSPhilippe Mathieu-Daudé temp[1] = tempB[1] - tempA[1];
2607a2b0a27dSPhilippe Mathieu-Daudé
2608a2b0a27dSPhilippe Mathieu-Daudé if ((temp[1] & 0x01) == 0) {
2609a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0x00;
2610a2b0a27dSPhilippe Mathieu-Daudé } else {
2611a2b0a27dSPhilippe Mathieu-Daudé temp[1] = ~0ull;
2612a2b0a27dSPhilippe Mathieu-Daudé }
2613a2b0a27dSPhilippe Mathieu-Daudé
2614a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0];
2615a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) &&
2616a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) {
2617a2b0a27dSPhilippe Mathieu-Daudé acc[1] += 1;
2618a2b0a27dSPhilippe Mathieu-Daudé }
2619a2b0a27dSPhilippe Mathieu-Daudé acc[0] = temp_sum;
2620a2b0a27dSPhilippe Mathieu-Daudé acc[1] += temp[1];
2621a2b0a27dSPhilippe Mathieu-Daudé
2622a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = acc[1];
2623a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = acc[0];
2624a2b0a27dSPhilippe Mathieu-Daudé }
2625a2b0a27dSPhilippe Mathieu-Daudé #endif
2626a2b0a27dSPhilippe Mathieu-Daudé
2627a2b0a27dSPhilippe Mathieu-Daudé #define MAQ_S_W(name, mov) \
2628a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2629a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2630a2b0a27dSPhilippe Mathieu-Daudé { \
2631a2b0a27dSPhilippe Mathieu-Daudé int16_t rsh, rth; \
2632a2b0a27dSPhilippe Mathieu-Daudé int32_t tempA; \
2633a2b0a27dSPhilippe Mathieu-Daudé int64_t tempL, acc; \
2634a2b0a27dSPhilippe Mathieu-Daudé \
2635a2b0a27dSPhilippe Mathieu-Daudé rsh = (rs >> mov) & MIPSDSP_LO; \
2636a2b0a27dSPhilippe Mathieu-Daudé rth = (rt >> mov) & MIPSDSP_LO; \
2637a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rsh, rth, env); \
2638a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) | \
2639a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO); \
2640a2b0a27dSPhilippe Mathieu-Daudé tempL = (int64_t)tempA + acc; \
2641a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t) \
2642a2b0a27dSPhilippe Mathieu-Daudé ((tempL & MIPSDSP_LHI) >> 32); \
2643a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t) \
2644a2b0a27dSPhilippe Mathieu-Daudé (tempL & MIPSDSP_LLO); \
2645a2b0a27dSPhilippe Mathieu-Daudé }
2646a2b0a27dSPhilippe Mathieu-Daudé
2647a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_W(maq_s_w_phl, 16);
2648a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_W(maq_s_w_phr, 0);
2649a2b0a27dSPhilippe Mathieu-Daudé
2650a2b0a27dSPhilippe Mathieu-Daudé #undef MAQ_S_W
2651a2b0a27dSPhilippe Mathieu-Daudé
2652a2b0a27dSPhilippe Mathieu-Daudé #define MAQ_SA_W(name, mov) \
2653a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(uint32_t ac, target_ulong rs, target_ulong rt, \
2654a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2655a2b0a27dSPhilippe Mathieu-Daudé { \
2656a2b0a27dSPhilippe Mathieu-Daudé int16_t rsh, rth; \
2657a2b0a27dSPhilippe Mathieu-Daudé int32_t tempA; \
2658a2b0a27dSPhilippe Mathieu-Daudé \
2659a2b0a27dSPhilippe Mathieu-Daudé rsh = (rs >> mov) & MIPSDSP_LO; \
2660a2b0a27dSPhilippe Mathieu-Daudé rth = (rt >> mov) & MIPSDSP_LO; \
2661a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_mul_q15_q15(ac, rsh, rth, env); \
2662a2b0a27dSPhilippe Mathieu-Daudé tempA = mipsdsp_sat32_acc_q31(ac, tempA, env); \
2663a2b0a27dSPhilippe Mathieu-Daudé \
2664a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)(int32_t)(((int64_t)tempA & \
2665a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_LHI) >> 32); \
2666a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)(int32_t)((int64_t)tempA & \
2667a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_LLO); \
2668a2b0a27dSPhilippe Mathieu-Daudé }
2669a2b0a27dSPhilippe Mathieu-Daudé
2670a2b0a27dSPhilippe Mathieu-Daudé MAQ_SA_W(maq_sa_w_phl, 16);
2671a2b0a27dSPhilippe Mathieu-Daudé MAQ_SA_W(maq_sa_w_phr, 0);
2672a2b0a27dSPhilippe Mathieu-Daudé
2673a2b0a27dSPhilippe Mathieu-Daudé #undef MAQ_SA_W
2674a2b0a27dSPhilippe Mathieu-Daudé
2675a2b0a27dSPhilippe Mathieu-Daudé #define MULQ_W(name, addvar) \
2676a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
2677a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2678a2b0a27dSPhilippe Mathieu-Daudé { \
2679a2b0a27dSPhilippe Mathieu-Daudé int32_t rs_t, rt_t; \
2680a2b0a27dSPhilippe Mathieu-Daudé int32_t tempI; \
2681a2b0a27dSPhilippe Mathieu-Daudé int64_t tempL; \
2682a2b0a27dSPhilippe Mathieu-Daudé \
2683a2b0a27dSPhilippe Mathieu-Daudé rs_t = rs & MIPSDSP_LLO; \
2684a2b0a27dSPhilippe Mathieu-Daudé rt_t = rt & MIPSDSP_LLO; \
2685a2b0a27dSPhilippe Mathieu-Daudé \
2686a2b0a27dSPhilippe Mathieu-Daudé if ((rs_t == 0x80000000) && (rt_t == 0x80000000)) { \
2687a2b0a27dSPhilippe Mathieu-Daudé tempL = 0x7FFFFFFF00000000ull; \
2688a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 21, env); \
2689a2b0a27dSPhilippe Mathieu-Daudé } else { \
2690a2b0a27dSPhilippe Mathieu-Daudé tempL = ((int64_t)rs_t * (int64_t)rt_t) << 1; \
2691a2b0a27dSPhilippe Mathieu-Daudé tempL += addvar; \
2692a2b0a27dSPhilippe Mathieu-Daudé } \
2693a2b0a27dSPhilippe Mathieu-Daudé tempI = (tempL & MIPSDSP_LHI) >> 32; \
2694a2b0a27dSPhilippe Mathieu-Daudé \
2695a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)tempI; \
2696a2b0a27dSPhilippe Mathieu-Daudé }
2697a2b0a27dSPhilippe Mathieu-Daudé
2698a2b0a27dSPhilippe Mathieu-Daudé MULQ_W(mulq_s_w, 0);
2699a2b0a27dSPhilippe Mathieu-Daudé MULQ_W(mulq_rs_w, 0x80000000ull);
2700a2b0a27dSPhilippe Mathieu-Daudé
2701a2b0a27dSPhilippe Mathieu-Daudé #undef MULQ_W
2702a2b0a27dSPhilippe Mathieu-Daudé
2703a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
2704a2b0a27dSPhilippe Mathieu-Daudé
2705a2b0a27dSPhilippe Mathieu-Daudé #define MAQ_S_W_QH(name, mov) \
2706a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2707a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2708a2b0a27dSPhilippe Mathieu-Daudé { \
2709a2b0a27dSPhilippe Mathieu-Daudé int16_t rs_t, rt_t; \
2710a2b0a27dSPhilippe Mathieu-Daudé int32_t temp_mul; \
2711a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2]; \
2712a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2]; \
2713a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum; \
2714a2b0a27dSPhilippe Mathieu-Daudé \
2715a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0; \
2716a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0; \
2717a2b0a27dSPhilippe Mathieu-Daudé \
2718a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> mov) & MIPSDSP_LO; \
2719a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> mov) & MIPSDSP_LO; \
2720a2b0a27dSPhilippe Mathieu-Daudé temp_mul = mipsdsp_mul_q15_q15(ac, rs_t, rt_t, env); \
2721a2b0a27dSPhilippe Mathieu-Daudé \
2722a2b0a27dSPhilippe Mathieu-Daudé temp[0] = (int64_t)temp_mul; \
2723a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] >= 0) { \
2724a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0x00; \
2725a2b0a27dSPhilippe Mathieu-Daudé } else { \
2726a2b0a27dSPhilippe Mathieu-Daudé temp[1] = ~0ull; \
2727a2b0a27dSPhilippe Mathieu-Daudé } \
2728a2b0a27dSPhilippe Mathieu-Daudé \
2729a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac]; \
2730a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac]; \
2731a2b0a27dSPhilippe Mathieu-Daudé \
2732a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0]; \
2733a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) && \
2734a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) { \
2735a2b0a27dSPhilippe Mathieu-Daudé acc[1] += 1; \
2736a2b0a27dSPhilippe Mathieu-Daudé } \
2737a2b0a27dSPhilippe Mathieu-Daudé acc[0] = temp_sum; \
2738a2b0a27dSPhilippe Mathieu-Daudé acc[1] += temp[1]; \
2739a2b0a27dSPhilippe Mathieu-Daudé \
2740a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = acc[1]; \
2741a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = acc[0]; \
2742a2b0a27dSPhilippe Mathieu-Daudé }
2743a2b0a27dSPhilippe Mathieu-Daudé
2744a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_W_QH(maq_s_w_qhll, 48);
2745a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_W_QH(maq_s_w_qhlr, 32);
2746a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_W_QH(maq_s_w_qhrl, 16);
2747a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_W_QH(maq_s_w_qhrr, 0);
2748a2b0a27dSPhilippe Mathieu-Daudé
2749a2b0a27dSPhilippe Mathieu-Daudé #undef MAQ_S_W_QH
2750a2b0a27dSPhilippe Mathieu-Daudé
2751a2b0a27dSPhilippe Mathieu-Daudé #define MAQ_SA_W(name, mov) \
2752a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2753a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2754a2b0a27dSPhilippe Mathieu-Daudé { \
2755a2b0a27dSPhilippe Mathieu-Daudé int16_t rs_t, rt_t; \
2756a2b0a27dSPhilippe Mathieu-Daudé int32_t temp; \
2757a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2]; \
2758a2b0a27dSPhilippe Mathieu-Daudé \
2759a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> mov) & MIPSDSP_LO; \
2760a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> mov) & MIPSDSP_LO; \
2761a2b0a27dSPhilippe Mathieu-Daudé temp = mipsdsp_mul_q15_q15(ac, rs_t, rt_t, env); \
2762a2b0a27dSPhilippe Mathieu-Daudé temp = mipsdsp_sat32_acc_q31(ac, temp, env); \
2763a2b0a27dSPhilippe Mathieu-Daudé \
2764a2b0a27dSPhilippe Mathieu-Daudé acc[0] = (int64_t)(int32_t)temp; \
2765a2b0a27dSPhilippe Mathieu-Daudé if (acc[0] >= 0) { \
2766a2b0a27dSPhilippe Mathieu-Daudé acc[1] = 0x00; \
2767a2b0a27dSPhilippe Mathieu-Daudé } else { \
2768a2b0a27dSPhilippe Mathieu-Daudé acc[1] = ~0ull; \
2769a2b0a27dSPhilippe Mathieu-Daudé } \
2770a2b0a27dSPhilippe Mathieu-Daudé \
2771a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = acc[1]; \
2772a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = acc[0]; \
2773a2b0a27dSPhilippe Mathieu-Daudé }
2774a2b0a27dSPhilippe Mathieu-Daudé
2775a2b0a27dSPhilippe Mathieu-Daudé MAQ_SA_W(maq_sa_w_qhll, 48);
2776a2b0a27dSPhilippe Mathieu-Daudé MAQ_SA_W(maq_sa_w_qhlr, 32);
2777a2b0a27dSPhilippe Mathieu-Daudé MAQ_SA_W(maq_sa_w_qhrl, 16);
2778a2b0a27dSPhilippe Mathieu-Daudé MAQ_SA_W(maq_sa_w_qhrr, 0);
2779a2b0a27dSPhilippe Mathieu-Daudé
2780a2b0a27dSPhilippe Mathieu-Daudé #undef MAQ_SA_W
2781a2b0a27dSPhilippe Mathieu-Daudé
2782a2b0a27dSPhilippe Mathieu-Daudé #define MAQ_S_L_PW(name, mov) \
2783a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2784a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2785a2b0a27dSPhilippe Mathieu-Daudé { \
2786a2b0a27dSPhilippe Mathieu-Daudé int32_t rs_t, rt_t; \
2787a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2]; \
2788a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2]; \
2789a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum; \
2790a2b0a27dSPhilippe Mathieu-Daudé \
2791a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0; \
2792a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0; \
2793a2b0a27dSPhilippe Mathieu-Daudé \
2794a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> mov) & MIPSDSP_LLO; \
2795a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> mov) & MIPSDSP_LLO; \
2796a2b0a27dSPhilippe Mathieu-Daudé \
2797a2b0a27dSPhilippe Mathieu-Daudé temp[0] = mipsdsp_mul_q31_q31(ac, rs_t, rt_t, env); \
2798a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] >= 0) { \
2799a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0x00; \
2800a2b0a27dSPhilippe Mathieu-Daudé } else { \
2801a2b0a27dSPhilippe Mathieu-Daudé temp[1] = ~0ull; \
2802a2b0a27dSPhilippe Mathieu-Daudé } \
2803a2b0a27dSPhilippe Mathieu-Daudé \
2804a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac]; \
2805a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac]; \
2806a2b0a27dSPhilippe Mathieu-Daudé \
2807a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0]; \
2808a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) && \
2809a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) { \
2810a2b0a27dSPhilippe Mathieu-Daudé acc[1] += 1; \
2811a2b0a27dSPhilippe Mathieu-Daudé } \
2812a2b0a27dSPhilippe Mathieu-Daudé acc[0] = temp_sum; \
2813a2b0a27dSPhilippe Mathieu-Daudé acc[1] += temp[1]; \
2814a2b0a27dSPhilippe Mathieu-Daudé \
2815a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = acc[1]; \
2816a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = acc[0]; \
2817a2b0a27dSPhilippe Mathieu-Daudé }
2818a2b0a27dSPhilippe Mathieu-Daudé
2819a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_L_PW(maq_s_l_pwl, 32);
2820a2b0a27dSPhilippe Mathieu-Daudé MAQ_S_L_PW(maq_s_l_pwr, 0);
2821a2b0a27dSPhilippe Mathieu-Daudé
2822a2b0a27dSPhilippe Mathieu-Daudé #undef MAQ_S_L_PW
2823a2b0a27dSPhilippe Mathieu-Daudé
2824a2b0a27dSPhilippe Mathieu-Daudé #define DM_OPERATE(name, func, is_add, sigext) \
2825a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, uint32_t ac, \
2826a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2827a2b0a27dSPhilippe Mathieu-Daudé { \
2828a2b0a27dSPhilippe Mathieu-Daudé int32_t rs1, rs0; \
2829a2b0a27dSPhilippe Mathieu-Daudé int32_t rt1, rt0; \
2830a2b0a27dSPhilippe Mathieu-Daudé int64_t tempBL[2], tempAL[2]; \
2831a2b0a27dSPhilippe Mathieu-Daudé int64_t acc[2]; \
2832a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2]; \
2833a2b0a27dSPhilippe Mathieu-Daudé int64_t temp_sum; \
2834a2b0a27dSPhilippe Mathieu-Daudé \
2835a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0x00; \
2836a2b0a27dSPhilippe Mathieu-Daudé temp[1] = 0x00; \
2837a2b0a27dSPhilippe Mathieu-Daudé \
2838a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_32(rs, rs1, rs0); \
2839a2b0a27dSPhilippe Mathieu-Daudé MIPSDSP_SPLIT64_32(rt, rt1, rt0); \
2840a2b0a27dSPhilippe Mathieu-Daudé \
2841a2b0a27dSPhilippe Mathieu-Daudé if (sigext) { \
2842a2b0a27dSPhilippe Mathieu-Daudé tempBL[0] = (int64_t)mipsdsp_##func(rs1, rt1); \
2843a2b0a27dSPhilippe Mathieu-Daudé tempAL[0] = (int64_t)mipsdsp_##func(rs0, rt0); \
2844a2b0a27dSPhilippe Mathieu-Daudé \
2845a2b0a27dSPhilippe Mathieu-Daudé if (tempBL[0] >= 0) { \
2846a2b0a27dSPhilippe Mathieu-Daudé tempBL[1] = 0x0; \
2847a2b0a27dSPhilippe Mathieu-Daudé } else { \
2848a2b0a27dSPhilippe Mathieu-Daudé tempBL[1] = ~0ull; \
2849a2b0a27dSPhilippe Mathieu-Daudé } \
2850a2b0a27dSPhilippe Mathieu-Daudé \
2851a2b0a27dSPhilippe Mathieu-Daudé if (tempAL[0] >= 0) { \
2852a2b0a27dSPhilippe Mathieu-Daudé tempAL[1] = 0x0; \
2853a2b0a27dSPhilippe Mathieu-Daudé } else { \
2854a2b0a27dSPhilippe Mathieu-Daudé tempAL[1] = ~0ull; \
2855a2b0a27dSPhilippe Mathieu-Daudé } \
2856a2b0a27dSPhilippe Mathieu-Daudé } else { \
2857a2b0a27dSPhilippe Mathieu-Daudé tempBL[0] = mipsdsp_##func(rs1, rt1); \
2858a2b0a27dSPhilippe Mathieu-Daudé tempAL[0] = mipsdsp_##func(rs0, rt0); \
2859a2b0a27dSPhilippe Mathieu-Daudé tempBL[1] = 0; \
2860a2b0a27dSPhilippe Mathieu-Daudé tempAL[1] = 0; \
2861a2b0a27dSPhilippe Mathieu-Daudé } \
2862a2b0a27dSPhilippe Mathieu-Daudé \
2863a2b0a27dSPhilippe Mathieu-Daudé acc[1] = env->active_tc.HI[ac]; \
2864a2b0a27dSPhilippe Mathieu-Daudé acc[0] = env->active_tc.LO[ac]; \
2865a2b0a27dSPhilippe Mathieu-Daudé \
2866a2b0a27dSPhilippe Mathieu-Daudé temp_sum = tempBL[0] + tempAL[0]; \
2867a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)tempBL[0]) && \
2868a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)tempAL[0])) { \
2869a2b0a27dSPhilippe Mathieu-Daudé temp[1] += 1; \
2870a2b0a27dSPhilippe Mathieu-Daudé } \
2871a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2872a2b0a27dSPhilippe Mathieu-Daudé temp[1] += tempBL[1] + tempAL[1]; \
2873a2b0a27dSPhilippe Mathieu-Daudé \
2874a2b0a27dSPhilippe Mathieu-Daudé if (is_add) { \
2875a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] + temp[0]; \
2876a2b0a27dSPhilippe Mathieu-Daudé if (((uint64_t)temp_sum < (uint64_t)acc[0]) && \
2877a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)temp_sum < (uint64_t)temp[0])) { \
2878a2b0a27dSPhilippe Mathieu-Daudé acc[1] += 1; \
2879a2b0a27dSPhilippe Mathieu-Daudé } \
2880a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2881a2b0a27dSPhilippe Mathieu-Daudé temp[1] = acc[1] + temp[1]; \
2882a2b0a27dSPhilippe Mathieu-Daudé } else { \
2883a2b0a27dSPhilippe Mathieu-Daudé temp_sum = acc[0] - temp[0]; \
2884a2b0a27dSPhilippe Mathieu-Daudé if ((uint64_t)temp_sum > (uint64_t)acc[0]) { \
2885a2b0a27dSPhilippe Mathieu-Daudé acc[1] -= 1; \
2886a2b0a27dSPhilippe Mathieu-Daudé } \
2887a2b0a27dSPhilippe Mathieu-Daudé temp[0] = temp_sum; \
2888a2b0a27dSPhilippe Mathieu-Daudé temp[1] = acc[1] - temp[1]; \
2889a2b0a27dSPhilippe Mathieu-Daudé } \
2890a2b0a27dSPhilippe Mathieu-Daudé \
2891a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = temp[1]; \
2892a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = temp[0]; \
2893a2b0a27dSPhilippe Mathieu-Daudé }
2894a2b0a27dSPhilippe Mathieu-Daudé
2895a2b0a27dSPhilippe Mathieu-Daudé DM_OPERATE(dmadd, mul_i32_i32, 1, 1);
2896a2b0a27dSPhilippe Mathieu-Daudé DM_OPERATE(dmaddu, mul_u32_u32, 1, 0);
2897a2b0a27dSPhilippe Mathieu-Daudé DM_OPERATE(dmsub, mul_i32_i32, 0, 1);
2898a2b0a27dSPhilippe Mathieu-Daudé DM_OPERATE(dmsubu, mul_u32_u32, 0, 0);
2899a2b0a27dSPhilippe Mathieu-Daudé #undef DM_OPERATE
2900a2b0a27dSPhilippe Mathieu-Daudé #endif
2901a2b0a27dSPhilippe Mathieu-Daudé
2902a2b0a27dSPhilippe Mathieu-Daudé /** DSP Bit/Manipulation Sub-class insns **/
helper_bitrev(target_ulong rt)2903a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_bitrev(target_ulong rt)
2904a2b0a27dSPhilippe Mathieu-Daudé {
2905a2b0a27dSPhilippe Mathieu-Daudé int32_t temp;
2906a2b0a27dSPhilippe Mathieu-Daudé uint32_t rd;
2907a2b0a27dSPhilippe Mathieu-Daudé int i;
2908a2b0a27dSPhilippe Mathieu-Daudé
2909a2b0a27dSPhilippe Mathieu-Daudé temp = rt & MIPSDSP_LO;
2910a2b0a27dSPhilippe Mathieu-Daudé rd = 0;
2911a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 16; i++) {
2912a2b0a27dSPhilippe Mathieu-Daudé rd = (rd << 1) | (temp & 1);
2913a2b0a27dSPhilippe Mathieu-Daudé temp = temp >> 1;
2914a2b0a27dSPhilippe Mathieu-Daudé }
2915a2b0a27dSPhilippe Mathieu-Daudé
2916a2b0a27dSPhilippe Mathieu-Daudé return (target_ulong)rd;
2917a2b0a27dSPhilippe Mathieu-Daudé }
2918a2b0a27dSPhilippe Mathieu-Daudé
2919a2b0a27dSPhilippe Mathieu-Daudé #define BIT_INSV(name, posfilter, ret_type) \
2920a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(CPUMIPSState *env, target_ulong rs, \
2921a2b0a27dSPhilippe Mathieu-Daudé target_ulong rt) \
2922a2b0a27dSPhilippe Mathieu-Daudé { \
2923a2b0a27dSPhilippe Mathieu-Daudé uint32_t pos, size, msb, lsb; \
2924a2b0a27dSPhilippe Mathieu-Daudé uint32_t const sizefilter = 0x3F; \
2925a2b0a27dSPhilippe Mathieu-Daudé target_ulong temp; \
2926a2b0a27dSPhilippe Mathieu-Daudé target_ulong dspc; \
2927a2b0a27dSPhilippe Mathieu-Daudé \
2928a2b0a27dSPhilippe Mathieu-Daudé dspc = env->active_tc.DSPControl; \
2929a2b0a27dSPhilippe Mathieu-Daudé \
2930a2b0a27dSPhilippe Mathieu-Daudé pos = dspc & posfilter; \
2931a2b0a27dSPhilippe Mathieu-Daudé size = (dspc >> 7) & sizefilter; \
2932a2b0a27dSPhilippe Mathieu-Daudé \
2933a2b0a27dSPhilippe Mathieu-Daudé msb = pos + size - 1; \
2934a2b0a27dSPhilippe Mathieu-Daudé lsb = pos; \
2935a2b0a27dSPhilippe Mathieu-Daudé \
2936a2b0a27dSPhilippe Mathieu-Daudé if (lsb > msb || (msb > TARGET_LONG_BITS)) { \
2937a2b0a27dSPhilippe Mathieu-Daudé return rt; \
2938a2b0a27dSPhilippe Mathieu-Daudé } \
2939a2b0a27dSPhilippe Mathieu-Daudé \
2940a2b0a27dSPhilippe Mathieu-Daudé temp = deposit64(rt, pos, size, rs); \
2941a2b0a27dSPhilippe Mathieu-Daudé \
2942a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(ret_type)temp; \
2943a2b0a27dSPhilippe Mathieu-Daudé }
2944a2b0a27dSPhilippe Mathieu-Daudé
2945a2b0a27dSPhilippe Mathieu-Daudé BIT_INSV(insv, 0x1F, int32_t);
2946a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
2947a2b0a27dSPhilippe Mathieu-Daudé BIT_INSV(dinsv, 0x7F, target_long);
2948a2b0a27dSPhilippe Mathieu-Daudé #endif
2949a2b0a27dSPhilippe Mathieu-Daudé
2950a2b0a27dSPhilippe Mathieu-Daudé #undef BIT_INSV
2951a2b0a27dSPhilippe Mathieu-Daudé
2952a2b0a27dSPhilippe Mathieu-Daudé
2953a2b0a27dSPhilippe Mathieu-Daudé /** DSP Compare-Pick Sub-class insns **/
2954a2b0a27dSPhilippe Mathieu-Daudé #define CMP_HAS_RET(name, func, split_num, filter, bit_size) \
2955a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt) \
2956a2b0a27dSPhilippe Mathieu-Daudé { \
2957a2b0a27dSPhilippe Mathieu-Daudé uint32_t rs_t, rt_t; \
2958a2b0a27dSPhilippe Mathieu-Daudé uint8_t cc; \
2959a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp = 0; \
2960a2b0a27dSPhilippe Mathieu-Daudé int i; \
2961a2b0a27dSPhilippe Mathieu-Daudé \
2962a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < split_num; i++) { \
2963a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> (bit_size * i)) & filter; \
2964a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> (bit_size * i)) & filter; \
2965a2b0a27dSPhilippe Mathieu-Daudé cc = mipsdsp_##func(rs_t, rt_t); \
2966a2b0a27dSPhilippe Mathieu-Daudé temp |= cc << i; \
2967a2b0a27dSPhilippe Mathieu-Daudé } \
2968a2b0a27dSPhilippe Mathieu-Daudé \
2969a2b0a27dSPhilippe Mathieu-Daudé return (target_ulong)temp; \
2970a2b0a27dSPhilippe Mathieu-Daudé }
2971a2b0a27dSPhilippe Mathieu-Daudé
2972a2b0a27dSPhilippe Mathieu-Daudé CMP_HAS_RET(cmpgu_eq_qb, cmpu_eq, 4, MIPSDSP_Q0, 8);
2973a2b0a27dSPhilippe Mathieu-Daudé CMP_HAS_RET(cmpgu_lt_qb, cmpu_lt, 4, MIPSDSP_Q0, 8);
2974a2b0a27dSPhilippe Mathieu-Daudé CMP_HAS_RET(cmpgu_le_qb, cmpu_le, 4, MIPSDSP_Q0, 8);
2975a2b0a27dSPhilippe Mathieu-Daudé
2976a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
2977a2b0a27dSPhilippe Mathieu-Daudé CMP_HAS_RET(cmpgu_eq_ob, cmpu_eq, 8, MIPSDSP_Q0, 8);
2978a2b0a27dSPhilippe Mathieu-Daudé CMP_HAS_RET(cmpgu_lt_ob, cmpu_lt, 8, MIPSDSP_Q0, 8);
2979a2b0a27dSPhilippe Mathieu-Daudé CMP_HAS_RET(cmpgu_le_ob, cmpu_le, 8, MIPSDSP_Q0, 8);
2980a2b0a27dSPhilippe Mathieu-Daudé #endif
2981a2b0a27dSPhilippe Mathieu-Daudé
2982a2b0a27dSPhilippe Mathieu-Daudé #undef CMP_HAS_RET
2983a2b0a27dSPhilippe Mathieu-Daudé
2984a2b0a27dSPhilippe Mathieu-Daudé
2985a2b0a27dSPhilippe Mathieu-Daudé #define CMP_NO_RET(name, func, split_num, filter, bit_size) \
2986a2b0a27dSPhilippe Mathieu-Daudé void helper_##name(target_ulong rs, target_ulong rt, \
2987a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
2988a2b0a27dSPhilippe Mathieu-Daudé { \
2989a2b0a27dSPhilippe Mathieu-Daudé int##bit_size##_t rs_t, rt_t; \
2990a2b0a27dSPhilippe Mathieu-Daudé int##bit_size##_t flag = 0; \
2991a2b0a27dSPhilippe Mathieu-Daudé int##bit_size##_t cc; \
2992a2b0a27dSPhilippe Mathieu-Daudé int i; \
2993a2b0a27dSPhilippe Mathieu-Daudé \
2994a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < split_num; i++) { \
2995a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> (bit_size * i)) & filter; \
2996a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> (bit_size * i)) & filter; \
2997a2b0a27dSPhilippe Mathieu-Daudé \
2998a2b0a27dSPhilippe Mathieu-Daudé cc = mipsdsp_##func((int32_t)rs_t, (int32_t)rt_t); \
2999a2b0a27dSPhilippe Mathieu-Daudé flag |= cc << i; \
3000a2b0a27dSPhilippe Mathieu-Daudé } \
3001a2b0a27dSPhilippe Mathieu-Daudé \
3002a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_24(flag, split_num, env); \
3003a2b0a27dSPhilippe Mathieu-Daudé }
3004a2b0a27dSPhilippe Mathieu-Daudé
3005a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmpu_eq_qb, cmpu_eq, 4, MIPSDSP_Q0, 8);
3006a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmpu_lt_qb, cmpu_lt, 4, MIPSDSP_Q0, 8);
3007a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmpu_le_qb, cmpu_le, 4, MIPSDSP_Q0, 8);
3008a2b0a27dSPhilippe Mathieu-Daudé
3009a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_eq_ph, cmp_eq, 2, MIPSDSP_LO, 16);
3010a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_lt_ph, cmp_lt, 2, MIPSDSP_LO, 16);
3011a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_le_ph, cmp_le, 2, MIPSDSP_LO, 16);
3012a2b0a27dSPhilippe Mathieu-Daudé
3013a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
3014a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmpu_eq_ob, cmpu_eq, 8, MIPSDSP_Q0, 8);
3015a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmpu_lt_ob, cmpu_lt, 8, MIPSDSP_Q0, 8);
3016a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmpu_le_ob, cmpu_le, 8, MIPSDSP_Q0, 8);
3017a2b0a27dSPhilippe Mathieu-Daudé
3018a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_eq_qh, cmp_eq, 4, MIPSDSP_LO, 16);
3019a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_lt_qh, cmp_lt, 4, MIPSDSP_LO, 16);
3020a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_le_qh, cmp_le, 4, MIPSDSP_LO, 16);
3021a2b0a27dSPhilippe Mathieu-Daudé
3022a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_eq_pw, cmp_eq, 2, MIPSDSP_LLO, 32);
3023a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_lt_pw, cmp_lt, 2, MIPSDSP_LLO, 32);
3024a2b0a27dSPhilippe Mathieu-Daudé CMP_NO_RET(cmp_le_pw, cmp_le, 2, MIPSDSP_LLO, 32);
3025a2b0a27dSPhilippe Mathieu-Daudé #endif
3026a2b0a27dSPhilippe Mathieu-Daudé #undef CMP_NO_RET
3027a2b0a27dSPhilippe Mathieu-Daudé
3028a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
3029a2b0a27dSPhilippe Mathieu-Daudé
3030a2b0a27dSPhilippe Mathieu-Daudé #define CMPGDU_OB(name) \
3031a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_cmpgdu_##name##_ob(target_ulong rs, target_ulong rt, \
3032a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
3033a2b0a27dSPhilippe Mathieu-Daudé { \
3034a2b0a27dSPhilippe Mathieu-Daudé int i; \
3035a2b0a27dSPhilippe Mathieu-Daudé uint8_t rs_t, rt_t; \
3036a2b0a27dSPhilippe Mathieu-Daudé uint32_t cond; \
3037a2b0a27dSPhilippe Mathieu-Daudé \
3038a2b0a27dSPhilippe Mathieu-Daudé cond = 0; \
3039a2b0a27dSPhilippe Mathieu-Daudé \
3040a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 8; i++) { \
3041a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> (8 * i)) & MIPSDSP_Q0; \
3042a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> (8 * i)) & MIPSDSP_Q0; \
3043a2b0a27dSPhilippe Mathieu-Daudé \
3044a2b0a27dSPhilippe Mathieu-Daudé if (mipsdsp_cmpu_##name(rs_t, rt_t)) { \
3045a2b0a27dSPhilippe Mathieu-Daudé cond |= 0x01 << i; \
3046a2b0a27dSPhilippe Mathieu-Daudé } \
3047a2b0a27dSPhilippe Mathieu-Daudé } \
3048a2b0a27dSPhilippe Mathieu-Daudé \
3049a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_24(cond, 8, env); \
3050a2b0a27dSPhilippe Mathieu-Daudé \
3051a2b0a27dSPhilippe Mathieu-Daudé return (uint64_t)cond; \
3052a2b0a27dSPhilippe Mathieu-Daudé }
3053a2b0a27dSPhilippe Mathieu-Daudé
3054a2b0a27dSPhilippe Mathieu-Daudé CMPGDU_OB(eq)
3055a2b0a27dSPhilippe Mathieu-Daudé CMPGDU_OB(lt)
3056a2b0a27dSPhilippe Mathieu-Daudé CMPGDU_OB(le)
3057a2b0a27dSPhilippe Mathieu-Daudé #undef CMPGDU_OB
3058a2b0a27dSPhilippe Mathieu-Daudé #endif
3059a2b0a27dSPhilippe Mathieu-Daudé
3060a2b0a27dSPhilippe Mathieu-Daudé #define PICK_INSN(name, split_num, filter, bit_size, ret32bit) \
3061a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_##name(target_ulong rs, target_ulong rt, \
3062a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env) \
3063a2b0a27dSPhilippe Mathieu-Daudé { \
3064a2b0a27dSPhilippe Mathieu-Daudé uint32_t rs_t, rt_t; \
3065a2b0a27dSPhilippe Mathieu-Daudé uint32_t cc; \
3066a2b0a27dSPhilippe Mathieu-Daudé target_ulong dsp; \
3067a2b0a27dSPhilippe Mathieu-Daudé int i; \
3068a2b0a27dSPhilippe Mathieu-Daudé target_ulong result = 0; \
3069a2b0a27dSPhilippe Mathieu-Daudé \
3070a2b0a27dSPhilippe Mathieu-Daudé dsp = env->active_tc.DSPControl; \
3071a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < split_num; i++) { \
3072a2b0a27dSPhilippe Mathieu-Daudé rs_t = (rs >> (bit_size * i)) & filter; \
3073a2b0a27dSPhilippe Mathieu-Daudé rt_t = (rt >> (bit_size * i)) & filter; \
3074a2b0a27dSPhilippe Mathieu-Daudé cc = (dsp >> (24 + i)) & 0x01; \
3075a2b0a27dSPhilippe Mathieu-Daudé cc = cc == 1 ? rs_t : rt_t; \
3076a2b0a27dSPhilippe Mathieu-Daudé \
3077a2b0a27dSPhilippe Mathieu-Daudé result |= (target_ulong)cc << (bit_size * i); \
3078a2b0a27dSPhilippe Mathieu-Daudé } \
3079a2b0a27dSPhilippe Mathieu-Daudé \
3080a2b0a27dSPhilippe Mathieu-Daudé if (ret32bit) { \
3081a2b0a27dSPhilippe Mathieu-Daudé result = (target_long)(int32_t)(result & MIPSDSP_LLO); \
3082a2b0a27dSPhilippe Mathieu-Daudé } \
3083a2b0a27dSPhilippe Mathieu-Daudé \
3084a2b0a27dSPhilippe Mathieu-Daudé return result; \
3085a2b0a27dSPhilippe Mathieu-Daudé }
3086a2b0a27dSPhilippe Mathieu-Daudé
3087a2b0a27dSPhilippe Mathieu-Daudé PICK_INSN(pick_qb, 4, MIPSDSP_Q0, 8, 1);
3088a2b0a27dSPhilippe Mathieu-Daudé PICK_INSN(pick_ph, 2, MIPSDSP_LO, 16, 1);
3089a2b0a27dSPhilippe Mathieu-Daudé
3090a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
3091a2b0a27dSPhilippe Mathieu-Daudé PICK_INSN(pick_ob, 8, MIPSDSP_Q0, 8, 0);
3092a2b0a27dSPhilippe Mathieu-Daudé PICK_INSN(pick_qh, 4, MIPSDSP_LO, 16, 0);
3093a2b0a27dSPhilippe Mathieu-Daudé PICK_INSN(pick_pw, 2, MIPSDSP_LLO, 32, 0);
3094a2b0a27dSPhilippe Mathieu-Daudé #endif
3095a2b0a27dSPhilippe Mathieu-Daudé #undef PICK_INSN
3096a2b0a27dSPhilippe Mathieu-Daudé
helper_packrl_ph(target_ulong rs,target_ulong rt)3097a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_packrl_ph(target_ulong rs, target_ulong rt)
3098a2b0a27dSPhilippe Mathieu-Daudé {
3099a2b0a27dSPhilippe Mathieu-Daudé uint32_t rsl, rth;
3100a2b0a27dSPhilippe Mathieu-Daudé
3101a2b0a27dSPhilippe Mathieu-Daudé rsl = rs & MIPSDSP_LO;
3102a2b0a27dSPhilippe Mathieu-Daudé rth = (rt & MIPSDSP_HI) >> 16;
3103a2b0a27dSPhilippe Mathieu-Daudé
3104a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)((rsl << 16) | rth);
3105a2b0a27dSPhilippe Mathieu-Daudé }
3106a2b0a27dSPhilippe Mathieu-Daudé
3107a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_packrl_pw(target_ulong rs,target_ulong rt)3108a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_packrl_pw(target_ulong rs, target_ulong rt)
3109a2b0a27dSPhilippe Mathieu-Daudé {
3110a2b0a27dSPhilippe Mathieu-Daudé uint32_t rs0, rt1;
3111a2b0a27dSPhilippe Mathieu-Daudé
3112a2b0a27dSPhilippe Mathieu-Daudé rs0 = rs & MIPSDSP_LLO;
3113a2b0a27dSPhilippe Mathieu-Daudé rt1 = (rt >> 32) & MIPSDSP_LLO;
3114a2b0a27dSPhilippe Mathieu-Daudé
3115a2b0a27dSPhilippe Mathieu-Daudé return ((uint64_t)rs0 << 32) | (uint64_t)rt1;
3116a2b0a27dSPhilippe Mathieu-Daudé }
3117a2b0a27dSPhilippe Mathieu-Daudé #endif
3118a2b0a27dSPhilippe Mathieu-Daudé
3119a2b0a27dSPhilippe Mathieu-Daudé /** DSP Accumulator and DSPControl Access Sub-class insns **/
helper_extr_w(target_ulong ac,target_ulong shift,CPUMIPSState * env)3120a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_extr_w(target_ulong ac, target_ulong shift,
3121a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3122a2b0a27dSPhilippe Mathieu-Daudé {
3123a2b0a27dSPhilippe Mathieu-Daudé int32_t tempI;
3124a2b0a27dSPhilippe Mathieu-Daudé int64_t tempDL[2];
3125a2b0a27dSPhilippe Mathieu-Daudé
3126a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x1F;
3127a2b0a27dSPhilippe Mathieu-Daudé
3128a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_short_acc(tempDL, ac, shift, env);
3129a2b0a27dSPhilippe Mathieu-Daudé if ((tempDL[1] != 0 || (tempDL[0] & MIPSDSP_LHI) != 0) &&
3130a2b0a27dSPhilippe Mathieu-Daudé (tempDL[1] != 1 || (tempDL[0] & MIPSDSP_LHI) != MIPSDSP_LHI)) {
3131a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3132a2b0a27dSPhilippe Mathieu-Daudé }
3133a2b0a27dSPhilippe Mathieu-Daudé
3134a2b0a27dSPhilippe Mathieu-Daudé tempI = (tempDL[0] >> 1) & MIPSDSP_LLO;
3135a2b0a27dSPhilippe Mathieu-Daudé
3136a2b0a27dSPhilippe Mathieu-Daudé tempDL[0] += 1;
3137a2b0a27dSPhilippe Mathieu-Daudé if (tempDL[0] == 0) {
3138a2b0a27dSPhilippe Mathieu-Daudé tempDL[1] += 1;
3139a2b0a27dSPhilippe Mathieu-Daudé }
3140a2b0a27dSPhilippe Mathieu-Daudé
3141a2b0a27dSPhilippe Mathieu-Daudé if (((tempDL[1] & 0x01) != 0 || (tempDL[0] & MIPSDSP_LHI) != 0) &&
3142a2b0a27dSPhilippe Mathieu-Daudé ((tempDL[1] & 0x01) != 1 || (tempDL[0] & MIPSDSP_LHI) != MIPSDSP_LHI)) {
3143a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3144a2b0a27dSPhilippe Mathieu-Daudé }
3145a2b0a27dSPhilippe Mathieu-Daudé
3146a2b0a27dSPhilippe Mathieu-Daudé return (target_long)tempI;
3147a2b0a27dSPhilippe Mathieu-Daudé }
3148a2b0a27dSPhilippe Mathieu-Daudé
helper_extr_r_w(target_ulong ac,target_ulong shift,CPUMIPSState * env)3149a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_extr_r_w(target_ulong ac, target_ulong shift,
3150a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3151a2b0a27dSPhilippe Mathieu-Daudé {
3152a2b0a27dSPhilippe Mathieu-Daudé int64_t tempDL[2];
3153a2b0a27dSPhilippe Mathieu-Daudé
3154a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x1F;
3155a2b0a27dSPhilippe Mathieu-Daudé
3156a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_short_acc(tempDL, ac, shift, env);
3157a2b0a27dSPhilippe Mathieu-Daudé if ((tempDL[1] != 0 || (tempDL[0] & MIPSDSP_LHI) != 0) &&
3158a2b0a27dSPhilippe Mathieu-Daudé (tempDL[1] != 1 || (tempDL[0] & MIPSDSP_LHI) != MIPSDSP_LHI)) {
3159a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3160a2b0a27dSPhilippe Mathieu-Daudé }
3161a2b0a27dSPhilippe Mathieu-Daudé
3162a2b0a27dSPhilippe Mathieu-Daudé tempDL[0] += 1;
3163a2b0a27dSPhilippe Mathieu-Daudé if (tempDL[0] == 0) {
3164a2b0a27dSPhilippe Mathieu-Daudé tempDL[1] += 1;
3165a2b0a27dSPhilippe Mathieu-Daudé }
3166a2b0a27dSPhilippe Mathieu-Daudé
3167a2b0a27dSPhilippe Mathieu-Daudé if (((tempDL[1] & 0x01) != 0 || (tempDL[0] & MIPSDSP_LHI) != 0) &&
3168a2b0a27dSPhilippe Mathieu-Daudé ((tempDL[1] & 0x01) != 1 || (tempDL[0] & MIPSDSP_LHI) != MIPSDSP_LHI)) {
3169a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3170a2b0a27dSPhilippe Mathieu-Daudé }
3171a2b0a27dSPhilippe Mathieu-Daudé
3172a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)(tempDL[0] >> 1);
3173a2b0a27dSPhilippe Mathieu-Daudé }
3174a2b0a27dSPhilippe Mathieu-Daudé
helper_extr_rs_w(target_ulong ac,target_ulong shift,CPUMIPSState * env)3175a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_extr_rs_w(target_ulong ac, target_ulong shift,
3176a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3177a2b0a27dSPhilippe Mathieu-Daudé {
3178a2b0a27dSPhilippe Mathieu-Daudé int32_t tempI, temp64;
3179a2b0a27dSPhilippe Mathieu-Daudé int64_t tempDL[2];
3180a2b0a27dSPhilippe Mathieu-Daudé
3181a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x1F;
3182a2b0a27dSPhilippe Mathieu-Daudé
3183a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_short_acc(tempDL, ac, shift, env);
3184a2b0a27dSPhilippe Mathieu-Daudé if ((tempDL[1] != 0 || (tempDL[0] & MIPSDSP_LHI) != 0) &&
3185a2b0a27dSPhilippe Mathieu-Daudé (tempDL[1] != 1 || (tempDL[0] & MIPSDSP_LHI) != MIPSDSP_LHI)) {
3186a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3187a2b0a27dSPhilippe Mathieu-Daudé }
3188a2b0a27dSPhilippe Mathieu-Daudé tempDL[0] += 1;
3189a2b0a27dSPhilippe Mathieu-Daudé if (tempDL[0] == 0) {
3190a2b0a27dSPhilippe Mathieu-Daudé tempDL[1] += 1;
3191a2b0a27dSPhilippe Mathieu-Daudé }
3192a2b0a27dSPhilippe Mathieu-Daudé tempI = tempDL[0] >> 1;
3193a2b0a27dSPhilippe Mathieu-Daudé
3194a2b0a27dSPhilippe Mathieu-Daudé if (((tempDL[1] & 0x01) != 0 || (tempDL[0] & MIPSDSP_LHI) != 0) &&
3195a2b0a27dSPhilippe Mathieu-Daudé ((tempDL[1] & 0x01) != 1 || (tempDL[0] & MIPSDSP_LHI) != MIPSDSP_LHI)) {
3196a2b0a27dSPhilippe Mathieu-Daudé temp64 = tempDL[1] & 0x01;
3197a2b0a27dSPhilippe Mathieu-Daudé if (temp64 == 0) {
3198a2b0a27dSPhilippe Mathieu-Daudé tempI = 0x7FFFFFFF;
3199a2b0a27dSPhilippe Mathieu-Daudé } else {
3200a2b0a27dSPhilippe Mathieu-Daudé tempI = 0x80000000;
3201a2b0a27dSPhilippe Mathieu-Daudé }
3202a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3203a2b0a27dSPhilippe Mathieu-Daudé }
3204a2b0a27dSPhilippe Mathieu-Daudé
3205a2b0a27dSPhilippe Mathieu-Daudé return (target_long)tempI;
3206a2b0a27dSPhilippe Mathieu-Daudé }
3207a2b0a27dSPhilippe Mathieu-Daudé
3208a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_dextr_w(target_ulong ac,target_ulong shift,CPUMIPSState * env)3209a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_w(target_ulong ac, target_ulong shift,
3210a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3211a2b0a27dSPhilippe Mathieu-Daudé {
3212a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[3];
3213a2b0a27dSPhilippe Mathieu-Daudé
3214a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
3215a2b0a27dSPhilippe Mathieu-Daudé
3216a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_acc(temp, ac, shift, env);
3217a2b0a27dSPhilippe Mathieu-Daudé
3218a2b0a27dSPhilippe Mathieu-Daudé return (int64_t)(int32_t)(temp[0] >> 1);
3219a2b0a27dSPhilippe Mathieu-Daudé }
3220a2b0a27dSPhilippe Mathieu-Daudé
helper_dextr_r_w(target_ulong ac,target_ulong shift,CPUMIPSState * env)3221a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_r_w(target_ulong ac, target_ulong shift,
3222a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3223a2b0a27dSPhilippe Mathieu-Daudé {
3224a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[3];
3225a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp128;
3226a2b0a27dSPhilippe Mathieu-Daudé
3227a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
3228a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_acc(temp, ac, shift, env);
3229a2b0a27dSPhilippe Mathieu-Daudé
3230a2b0a27dSPhilippe Mathieu-Daudé temp[0] += 1;
3231a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] == 0) {
3232a2b0a27dSPhilippe Mathieu-Daudé temp[1] += 1;
3233a2b0a27dSPhilippe Mathieu-Daudé if (temp[1] == 0) {
3234a2b0a27dSPhilippe Mathieu-Daudé temp[2] += 1;
3235a2b0a27dSPhilippe Mathieu-Daudé }
3236a2b0a27dSPhilippe Mathieu-Daudé }
3237a2b0a27dSPhilippe Mathieu-Daudé
3238a2b0a27dSPhilippe Mathieu-Daudé temp128 = temp[2] & 0x01;
3239a2b0a27dSPhilippe Mathieu-Daudé
3240a2b0a27dSPhilippe Mathieu-Daudé if ((temp128 != 0 || temp[1] != 0) &&
3241a2b0a27dSPhilippe Mathieu-Daudé (temp128 != 1 || temp[1] != ~0ull)) {
3242a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3243a2b0a27dSPhilippe Mathieu-Daudé }
3244a2b0a27dSPhilippe Mathieu-Daudé
3245a2b0a27dSPhilippe Mathieu-Daudé return (int64_t)(int32_t)(temp[0] >> 1);
3246a2b0a27dSPhilippe Mathieu-Daudé }
3247a2b0a27dSPhilippe Mathieu-Daudé
helper_dextr_rs_w(target_ulong ac,target_ulong shift,CPUMIPSState * env)3248a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_rs_w(target_ulong ac, target_ulong shift,
3249a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3250a2b0a27dSPhilippe Mathieu-Daudé {
3251a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[3];
3252a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp128;
3253a2b0a27dSPhilippe Mathieu-Daudé
3254a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
3255a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_acc(temp, ac, shift, env);
3256a2b0a27dSPhilippe Mathieu-Daudé
3257a2b0a27dSPhilippe Mathieu-Daudé temp[0] += 1;
3258a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] == 0) {
3259a2b0a27dSPhilippe Mathieu-Daudé temp[1] += 1;
3260a2b0a27dSPhilippe Mathieu-Daudé if (temp[1] == 0) {
3261a2b0a27dSPhilippe Mathieu-Daudé temp[2] += 1;
3262a2b0a27dSPhilippe Mathieu-Daudé }
3263a2b0a27dSPhilippe Mathieu-Daudé }
3264a2b0a27dSPhilippe Mathieu-Daudé
3265a2b0a27dSPhilippe Mathieu-Daudé temp128 = temp[2] & 0x01;
3266a2b0a27dSPhilippe Mathieu-Daudé
3267a2b0a27dSPhilippe Mathieu-Daudé if ((temp128 != 0 || temp[1] != 0) &&
3268a2b0a27dSPhilippe Mathieu-Daudé (temp128 != 1 || temp[1] != ~0ull)) {
3269a2b0a27dSPhilippe Mathieu-Daudé if (temp128 == 0) {
3270a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0x0FFFFFFFF;
3271a2b0a27dSPhilippe Mathieu-Daudé } else {
3272a2b0a27dSPhilippe Mathieu-Daudé temp[0] = 0x0100000000ULL;
3273a2b0a27dSPhilippe Mathieu-Daudé }
3274a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3275a2b0a27dSPhilippe Mathieu-Daudé }
3276a2b0a27dSPhilippe Mathieu-Daudé
3277a2b0a27dSPhilippe Mathieu-Daudé return (int64_t)(int32_t)(temp[0] >> 1);
3278a2b0a27dSPhilippe Mathieu-Daudé }
3279a2b0a27dSPhilippe Mathieu-Daudé
helper_dextr_l(target_ulong ac,target_ulong shift,CPUMIPSState * env)3280a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_l(target_ulong ac, target_ulong shift,
3281a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3282a2b0a27dSPhilippe Mathieu-Daudé {
3283a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[3];
3284a2b0a27dSPhilippe Mathieu-Daudé
3285a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
3286a2b0a27dSPhilippe Mathieu-Daudé
3287a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_acc(temp, ac, shift, env);
3288a2b0a27dSPhilippe Mathieu-Daudé
3289*66997c42SMarkus Armbruster return (temp[1] << 63) | (temp[0] >> 1);
3290a2b0a27dSPhilippe Mathieu-Daudé }
3291a2b0a27dSPhilippe Mathieu-Daudé
helper_dextr_r_l(target_ulong ac,target_ulong shift,CPUMIPSState * env)3292a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_r_l(target_ulong ac, target_ulong shift,
3293a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3294a2b0a27dSPhilippe Mathieu-Daudé {
3295a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[3];
3296a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp128;
3297a2b0a27dSPhilippe Mathieu-Daudé
3298a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
3299a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_acc(temp, ac, shift, env);
3300a2b0a27dSPhilippe Mathieu-Daudé
3301a2b0a27dSPhilippe Mathieu-Daudé temp[0] += 1;
3302a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] == 0) {
3303a2b0a27dSPhilippe Mathieu-Daudé temp[1] += 1;
3304a2b0a27dSPhilippe Mathieu-Daudé if (temp[1] == 0) {
3305a2b0a27dSPhilippe Mathieu-Daudé temp[2] += 1;
3306a2b0a27dSPhilippe Mathieu-Daudé }
3307a2b0a27dSPhilippe Mathieu-Daudé }
3308a2b0a27dSPhilippe Mathieu-Daudé
3309a2b0a27dSPhilippe Mathieu-Daudé temp128 = temp[2] & 0x01;
3310a2b0a27dSPhilippe Mathieu-Daudé
3311a2b0a27dSPhilippe Mathieu-Daudé if ((temp128 != 0 || temp[1] != 0) &&
3312a2b0a27dSPhilippe Mathieu-Daudé (temp128 != 1 || temp[1] != ~0ull)) {
3313a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3314a2b0a27dSPhilippe Mathieu-Daudé }
3315a2b0a27dSPhilippe Mathieu-Daudé
3316*66997c42SMarkus Armbruster return (temp[1] << 63) | (temp[0] >> 1);
3317a2b0a27dSPhilippe Mathieu-Daudé }
3318a2b0a27dSPhilippe Mathieu-Daudé
helper_dextr_rs_l(target_ulong ac,target_ulong shift,CPUMIPSState * env)3319a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_rs_l(target_ulong ac, target_ulong shift,
3320a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3321a2b0a27dSPhilippe Mathieu-Daudé {
3322a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp[3];
3323a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp128;
3324a2b0a27dSPhilippe Mathieu-Daudé
3325a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x3F;
3326a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rndrashift_acc(temp, ac, shift, env);
3327a2b0a27dSPhilippe Mathieu-Daudé
3328a2b0a27dSPhilippe Mathieu-Daudé temp[0] += 1;
3329a2b0a27dSPhilippe Mathieu-Daudé if (temp[0] == 0) {
3330a2b0a27dSPhilippe Mathieu-Daudé temp[1] += 1;
3331a2b0a27dSPhilippe Mathieu-Daudé if (temp[1] == 0) {
3332a2b0a27dSPhilippe Mathieu-Daudé temp[2] += 1;
3333a2b0a27dSPhilippe Mathieu-Daudé }
3334a2b0a27dSPhilippe Mathieu-Daudé }
3335a2b0a27dSPhilippe Mathieu-Daudé
3336a2b0a27dSPhilippe Mathieu-Daudé temp128 = temp[2] & 0x01;
3337a2b0a27dSPhilippe Mathieu-Daudé
3338a2b0a27dSPhilippe Mathieu-Daudé if ((temp128 != 0 || temp[1] != 0) &&
3339a2b0a27dSPhilippe Mathieu-Daudé (temp128 != 1 || temp[1] != ~0ull)) {
3340a2b0a27dSPhilippe Mathieu-Daudé if (temp128 == 0) {
3341a2b0a27dSPhilippe Mathieu-Daudé temp[1] &= ~0x00ull - 1;
3342a2b0a27dSPhilippe Mathieu-Daudé temp[0] |= ~0x00ull - 1;
3343a2b0a27dSPhilippe Mathieu-Daudé } else {
3344a2b0a27dSPhilippe Mathieu-Daudé temp[1] |= 0x01;
3345a2b0a27dSPhilippe Mathieu-Daudé temp[0] &= 0x01;
3346a2b0a27dSPhilippe Mathieu-Daudé }
3347a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3348a2b0a27dSPhilippe Mathieu-Daudé }
3349a2b0a27dSPhilippe Mathieu-Daudé
3350*66997c42SMarkus Armbruster return (temp[1] << 63) | (temp[0] >> 1);
3351a2b0a27dSPhilippe Mathieu-Daudé }
3352a2b0a27dSPhilippe Mathieu-Daudé #endif
3353a2b0a27dSPhilippe Mathieu-Daudé
helper_extr_s_h(target_ulong ac,target_ulong shift,CPUMIPSState * env)3354a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_extr_s_h(target_ulong ac, target_ulong shift,
3355a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3356a2b0a27dSPhilippe Mathieu-Daudé {
3357a2b0a27dSPhilippe Mathieu-Daudé int64_t temp, acc;
3358a2b0a27dSPhilippe Mathieu-Daudé
3359a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x1F;
3360a2b0a27dSPhilippe Mathieu-Daudé
3361a2b0a27dSPhilippe Mathieu-Daudé acc = ((int64_t)env->active_tc.HI[ac] << 32) |
3362a2b0a27dSPhilippe Mathieu-Daudé ((int64_t)env->active_tc.LO[ac] & 0xFFFFFFFF);
3363a2b0a27dSPhilippe Mathieu-Daudé
3364a2b0a27dSPhilippe Mathieu-Daudé temp = acc >> shift;
3365a2b0a27dSPhilippe Mathieu-Daudé
3366a2b0a27dSPhilippe Mathieu-Daudé if (temp > (int64_t)0x7FFF) {
3367a2b0a27dSPhilippe Mathieu-Daudé temp = 0x00007FFF;
3368a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3369a2b0a27dSPhilippe Mathieu-Daudé } else if (temp < (int64_t)0xFFFFFFFFFFFF8000ULL) {
3370a2b0a27dSPhilippe Mathieu-Daudé temp = 0xFFFF8000;
3371a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3372a2b0a27dSPhilippe Mathieu-Daudé }
3373a2b0a27dSPhilippe Mathieu-Daudé
3374a2b0a27dSPhilippe Mathieu-Daudé return (target_long)(int32_t)(temp & 0xFFFFFFFF);
3375a2b0a27dSPhilippe Mathieu-Daudé }
3376a2b0a27dSPhilippe Mathieu-Daudé
3377a2b0a27dSPhilippe Mathieu-Daudé
3378a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_dextr_s_h(target_ulong ac,target_ulong shift,CPUMIPSState * env)3379a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextr_s_h(target_ulong ac, target_ulong shift,
3380a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3381a2b0a27dSPhilippe Mathieu-Daudé {
3382a2b0a27dSPhilippe Mathieu-Daudé int64_t temp[2];
3383a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp127;
3384a2b0a27dSPhilippe Mathieu-Daudé
3385a2b0a27dSPhilippe Mathieu-Daudé shift = shift & 0x1F;
3386a2b0a27dSPhilippe Mathieu-Daudé
3387a2b0a27dSPhilippe Mathieu-Daudé mipsdsp_rashift_acc((uint64_t *)temp, ac, shift, env);
3388a2b0a27dSPhilippe Mathieu-Daudé
3389a2b0a27dSPhilippe Mathieu-Daudé temp127 = (temp[1] >> 63) & 0x01;
3390a2b0a27dSPhilippe Mathieu-Daudé
3391a2b0a27dSPhilippe Mathieu-Daudé if ((temp127 == 0) && (temp[1] > 0 || temp[0] > 32767)) {
3392a2b0a27dSPhilippe Mathieu-Daudé temp[0] &= 0xFFFF0000;
3393a2b0a27dSPhilippe Mathieu-Daudé temp[0] |= 0x00007FFF;
3394a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3395a2b0a27dSPhilippe Mathieu-Daudé } else if ((temp127 == 1) &&
3396a2b0a27dSPhilippe Mathieu-Daudé (temp[1] < 0xFFFFFFFFFFFFFFFFll
3397a2b0a27dSPhilippe Mathieu-Daudé || temp[0] < 0xFFFFFFFFFFFF1000ll)) {
3398a2b0a27dSPhilippe Mathieu-Daudé temp[0] &= 0xFFFF0000;
3399a2b0a27dSPhilippe Mathieu-Daudé temp[0] |= 0x00008000;
3400a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_overflow_flag(1, 23, env);
3401a2b0a27dSPhilippe Mathieu-Daudé }
3402a2b0a27dSPhilippe Mathieu-Daudé
3403a2b0a27dSPhilippe Mathieu-Daudé return (int64_t)(int16_t)(temp[0] & MIPSDSP_LO);
3404a2b0a27dSPhilippe Mathieu-Daudé }
3405a2b0a27dSPhilippe Mathieu-Daudé
3406a2b0a27dSPhilippe Mathieu-Daudé #endif
3407a2b0a27dSPhilippe Mathieu-Daudé
helper_extp(target_ulong ac,target_ulong size,CPUMIPSState * env)3408a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_extp(target_ulong ac, target_ulong size, CPUMIPSState *env)
3409a2b0a27dSPhilippe Mathieu-Daudé {
3410a2b0a27dSPhilippe Mathieu-Daudé int32_t start_pos;
3411a2b0a27dSPhilippe Mathieu-Daudé int sub;
3412a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
3413a2b0a27dSPhilippe Mathieu-Daudé uint64_t acc;
3414a2b0a27dSPhilippe Mathieu-Daudé
3415a2b0a27dSPhilippe Mathieu-Daudé size = size & 0x1F;
3416a2b0a27dSPhilippe Mathieu-Daudé
3417a2b0a27dSPhilippe Mathieu-Daudé temp = 0;
3418a2b0a27dSPhilippe Mathieu-Daudé start_pos = get_DSPControl_pos(env);
3419a2b0a27dSPhilippe Mathieu-Daudé sub = start_pos - (size + 1);
3420a2b0a27dSPhilippe Mathieu-Daudé if (sub >= -1) {
3421a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) |
3422a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO);
3423a2b0a27dSPhilippe Mathieu-Daudé temp = (acc >> (start_pos - size)) & (~0U >> (31 - size));
3424a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(0, env);
3425a2b0a27dSPhilippe Mathieu-Daudé } else {
3426a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(1, env);
3427a2b0a27dSPhilippe Mathieu-Daudé }
3428a2b0a27dSPhilippe Mathieu-Daudé
3429a2b0a27dSPhilippe Mathieu-Daudé return (target_ulong)temp;
3430a2b0a27dSPhilippe Mathieu-Daudé }
3431a2b0a27dSPhilippe Mathieu-Daudé
helper_extpdp(target_ulong ac,target_ulong size,CPUMIPSState * env)3432a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_extpdp(target_ulong ac, target_ulong size,
3433a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3434a2b0a27dSPhilippe Mathieu-Daudé {
3435a2b0a27dSPhilippe Mathieu-Daudé int32_t start_pos;
3436a2b0a27dSPhilippe Mathieu-Daudé int sub;
3437a2b0a27dSPhilippe Mathieu-Daudé uint32_t temp;
3438a2b0a27dSPhilippe Mathieu-Daudé uint64_t acc;
3439a2b0a27dSPhilippe Mathieu-Daudé
3440a2b0a27dSPhilippe Mathieu-Daudé size = size & 0x1F;
3441a2b0a27dSPhilippe Mathieu-Daudé temp = 0;
3442a2b0a27dSPhilippe Mathieu-Daudé start_pos = get_DSPControl_pos(env);
3443a2b0a27dSPhilippe Mathieu-Daudé sub = start_pos - (size + 1);
3444a2b0a27dSPhilippe Mathieu-Daudé if (sub >= -1) {
3445a2b0a27dSPhilippe Mathieu-Daudé acc = ((uint64_t)env->active_tc.HI[ac] << 32) |
3446a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO);
3447a2b0a27dSPhilippe Mathieu-Daudé temp = extract64(acc, start_pos - size, size + 1);
3448a2b0a27dSPhilippe Mathieu-Daudé
3449a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_pos(sub, env);
3450a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(0, env);
3451a2b0a27dSPhilippe Mathieu-Daudé } else {
3452a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(1, env);
3453a2b0a27dSPhilippe Mathieu-Daudé }
3454a2b0a27dSPhilippe Mathieu-Daudé
3455a2b0a27dSPhilippe Mathieu-Daudé return (target_ulong)temp;
3456a2b0a27dSPhilippe Mathieu-Daudé }
3457a2b0a27dSPhilippe Mathieu-Daudé
3458a2b0a27dSPhilippe Mathieu-Daudé
3459a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_dextp(target_ulong ac,target_ulong size,CPUMIPSState * env)3460a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextp(target_ulong ac, target_ulong size, CPUMIPSState *env)
3461a2b0a27dSPhilippe Mathieu-Daudé {
3462a2b0a27dSPhilippe Mathieu-Daudé int start_pos;
3463a2b0a27dSPhilippe Mathieu-Daudé int len;
3464a2b0a27dSPhilippe Mathieu-Daudé int sub;
3465a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempB, tempA;
3466a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp;
3467a2b0a27dSPhilippe Mathieu-Daudé
3468a2b0a27dSPhilippe Mathieu-Daudé temp = 0;
3469a2b0a27dSPhilippe Mathieu-Daudé
3470a2b0a27dSPhilippe Mathieu-Daudé size = size & 0x3F;
3471a2b0a27dSPhilippe Mathieu-Daudé start_pos = get_DSPControl_pos(env);
3472a2b0a27dSPhilippe Mathieu-Daudé len = start_pos - size;
3473a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.HI[ac];
3474a2b0a27dSPhilippe Mathieu-Daudé tempA = env->active_tc.LO[ac];
3475a2b0a27dSPhilippe Mathieu-Daudé
3476a2b0a27dSPhilippe Mathieu-Daudé sub = start_pos - (size + 1);
3477a2b0a27dSPhilippe Mathieu-Daudé
3478a2b0a27dSPhilippe Mathieu-Daudé if (sub >= -1) {
3479a2b0a27dSPhilippe Mathieu-Daudé temp = (tempB << (64 - len)) | (tempA >> len);
3480a2b0a27dSPhilippe Mathieu-Daudé temp = temp & ((1ULL << (size + 1)) - 1);
3481a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(0, env);
3482a2b0a27dSPhilippe Mathieu-Daudé } else {
3483a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(1, env);
3484a2b0a27dSPhilippe Mathieu-Daudé }
3485a2b0a27dSPhilippe Mathieu-Daudé
3486a2b0a27dSPhilippe Mathieu-Daudé return temp;
3487a2b0a27dSPhilippe Mathieu-Daudé }
3488a2b0a27dSPhilippe Mathieu-Daudé
helper_dextpdp(target_ulong ac,target_ulong size,CPUMIPSState * env)3489a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_dextpdp(target_ulong ac, target_ulong size,
3490a2b0a27dSPhilippe Mathieu-Daudé CPUMIPSState *env)
3491a2b0a27dSPhilippe Mathieu-Daudé {
3492a2b0a27dSPhilippe Mathieu-Daudé int start_pos;
3493a2b0a27dSPhilippe Mathieu-Daudé int len;
3494a2b0a27dSPhilippe Mathieu-Daudé int sub;
3495a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempB, tempA;
3496a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp;
3497a2b0a27dSPhilippe Mathieu-Daudé
3498a2b0a27dSPhilippe Mathieu-Daudé temp = 0;
3499a2b0a27dSPhilippe Mathieu-Daudé size = size & 0x3F;
3500a2b0a27dSPhilippe Mathieu-Daudé start_pos = get_DSPControl_pos(env);
3501a2b0a27dSPhilippe Mathieu-Daudé len = start_pos - size;
3502a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.HI[ac];
3503a2b0a27dSPhilippe Mathieu-Daudé tempA = env->active_tc.LO[ac];
3504a2b0a27dSPhilippe Mathieu-Daudé
3505a2b0a27dSPhilippe Mathieu-Daudé sub = start_pos - (size + 1);
3506a2b0a27dSPhilippe Mathieu-Daudé
3507a2b0a27dSPhilippe Mathieu-Daudé if (sub >= -1) {
3508a2b0a27dSPhilippe Mathieu-Daudé temp = (tempB << (64 - len)) | (tempA >> len);
3509a2b0a27dSPhilippe Mathieu-Daudé temp = temp & ((1ULL << (size + 1)) - 1);
3510a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_pos(sub, env);
3511a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(0, env);
3512a2b0a27dSPhilippe Mathieu-Daudé } else {
3513a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_efi(1, env);
3514a2b0a27dSPhilippe Mathieu-Daudé }
3515a2b0a27dSPhilippe Mathieu-Daudé
3516a2b0a27dSPhilippe Mathieu-Daudé return temp;
3517a2b0a27dSPhilippe Mathieu-Daudé }
3518a2b0a27dSPhilippe Mathieu-Daudé
3519a2b0a27dSPhilippe Mathieu-Daudé #endif
3520a2b0a27dSPhilippe Mathieu-Daudé
helper_shilo(target_ulong ac,target_ulong rs,CPUMIPSState * env)3521a2b0a27dSPhilippe Mathieu-Daudé void helper_shilo(target_ulong ac, target_ulong rs, CPUMIPSState *env)
3522a2b0a27dSPhilippe Mathieu-Daudé {
3523a2b0a27dSPhilippe Mathieu-Daudé int8_t rs5_0;
3524a2b0a27dSPhilippe Mathieu-Daudé uint64_t temp, acc;
3525a2b0a27dSPhilippe Mathieu-Daudé
3526a2b0a27dSPhilippe Mathieu-Daudé rs5_0 = rs & 0x3F;
3527a2b0a27dSPhilippe Mathieu-Daudé rs5_0 = (int8_t)(rs5_0 << 2) >> 2;
3528a2b0a27dSPhilippe Mathieu-Daudé
3529a2b0a27dSPhilippe Mathieu-Daudé if (unlikely(rs5_0 == 0)) {
3530a2b0a27dSPhilippe Mathieu-Daudé return;
3531a2b0a27dSPhilippe Mathieu-Daudé }
3532a2b0a27dSPhilippe Mathieu-Daudé
3533a2b0a27dSPhilippe Mathieu-Daudé acc = (((uint64_t)env->active_tc.HI[ac] << 32) & MIPSDSP_LHI) |
3534a2b0a27dSPhilippe Mathieu-Daudé ((uint64_t)env->active_tc.LO[ac] & MIPSDSP_LLO);
3535a2b0a27dSPhilippe Mathieu-Daudé
3536a2b0a27dSPhilippe Mathieu-Daudé if (rs5_0 > 0) {
3537a2b0a27dSPhilippe Mathieu-Daudé temp = acc >> rs5_0;
3538a2b0a27dSPhilippe Mathieu-Daudé } else {
3539a2b0a27dSPhilippe Mathieu-Daudé temp = acc << -rs5_0;
3540a2b0a27dSPhilippe Mathieu-Daudé }
3541a2b0a27dSPhilippe Mathieu-Daudé
3542a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_ulong)(int32_t)((temp & MIPSDSP_LHI) >> 32);
3543a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_ulong)(int32_t)(temp & MIPSDSP_LLO);
3544a2b0a27dSPhilippe Mathieu-Daudé }
3545a2b0a27dSPhilippe Mathieu-Daudé
3546a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_dshilo(target_ulong shift,target_ulong ac,CPUMIPSState * env)3547a2b0a27dSPhilippe Mathieu-Daudé void helper_dshilo(target_ulong shift, target_ulong ac, CPUMIPSState *env)
3548a2b0a27dSPhilippe Mathieu-Daudé {
3549a2b0a27dSPhilippe Mathieu-Daudé int8_t shift_t;
3550a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempB, tempA;
3551a2b0a27dSPhilippe Mathieu-Daudé
3552a2b0a27dSPhilippe Mathieu-Daudé shift_t = (int8_t)(shift << 1) >> 1;
3553a2b0a27dSPhilippe Mathieu-Daudé
3554a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.HI[ac];
3555a2b0a27dSPhilippe Mathieu-Daudé tempA = env->active_tc.LO[ac];
3556a2b0a27dSPhilippe Mathieu-Daudé
3557a2b0a27dSPhilippe Mathieu-Daudé if (shift_t != 0) {
3558a2b0a27dSPhilippe Mathieu-Daudé if (shift_t >= 0) {
3559a2b0a27dSPhilippe Mathieu-Daudé tempA = (tempB << (64 - shift_t)) | (tempA >> shift_t);
3560a2b0a27dSPhilippe Mathieu-Daudé tempB = tempB >> shift_t;
3561a2b0a27dSPhilippe Mathieu-Daudé } else {
3562a2b0a27dSPhilippe Mathieu-Daudé shift_t = -shift_t;
3563a2b0a27dSPhilippe Mathieu-Daudé tempB = (tempB << shift_t) | (tempA >> (64 - shift_t));
3564a2b0a27dSPhilippe Mathieu-Daudé tempA = tempA << shift_t;
3565a2b0a27dSPhilippe Mathieu-Daudé }
3566a2b0a27dSPhilippe Mathieu-Daudé }
3567a2b0a27dSPhilippe Mathieu-Daudé
3568a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = tempB;
3569a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = tempA;
3570a2b0a27dSPhilippe Mathieu-Daudé }
3571a2b0a27dSPhilippe Mathieu-Daudé
3572a2b0a27dSPhilippe Mathieu-Daudé #endif
helper_mthlip(target_ulong ac,target_ulong rs,CPUMIPSState * env)3573a2b0a27dSPhilippe Mathieu-Daudé void helper_mthlip(target_ulong ac, target_ulong rs, CPUMIPSState *env)
3574a2b0a27dSPhilippe Mathieu-Daudé {
3575a2b0a27dSPhilippe Mathieu-Daudé int32_t tempA, tempB, pos;
3576a2b0a27dSPhilippe Mathieu-Daudé
3577a2b0a27dSPhilippe Mathieu-Daudé tempA = rs;
3578a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.LO[ac];
3579a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac] = (target_long)tempB;
3580a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac] = (target_long)tempA;
3581a2b0a27dSPhilippe Mathieu-Daudé pos = get_DSPControl_pos(env);
3582a2b0a27dSPhilippe Mathieu-Daudé
3583a2b0a27dSPhilippe Mathieu-Daudé if (pos > 32) {
3584a2b0a27dSPhilippe Mathieu-Daudé return;
3585a2b0a27dSPhilippe Mathieu-Daudé } else {
3586a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_pos(pos + 32, env);
3587a2b0a27dSPhilippe Mathieu-Daudé }
3588a2b0a27dSPhilippe Mathieu-Daudé }
3589a2b0a27dSPhilippe Mathieu-Daudé
3590a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
helper_dmthlip(target_ulong rs,target_ulong ac,CPUMIPSState * env)3591a2b0a27dSPhilippe Mathieu-Daudé void helper_dmthlip(target_ulong rs, target_ulong ac, CPUMIPSState *env)
3592a2b0a27dSPhilippe Mathieu-Daudé {
3593a2b0a27dSPhilippe Mathieu-Daudé uint8_t ac_t;
3594a2b0a27dSPhilippe Mathieu-Daudé uint8_t pos;
3595a2b0a27dSPhilippe Mathieu-Daudé uint64_t tempB, tempA;
3596a2b0a27dSPhilippe Mathieu-Daudé
3597a2b0a27dSPhilippe Mathieu-Daudé ac_t = ac & 0x3;
3598a2b0a27dSPhilippe Mathieu-Daudé
3599a2b0a27dSPhilippe Mathieu-Daudé tempA = rs;
3600a2b0a27dSPhilippe Mathieu-Daudé tempB = env->active_tc.LO[ac_t];
3601a2b0a27dSPhilippe Mathieu-Daudé
3602a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.HI[ac_t] = tempB;
3603a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.LO[ac_t] = tempA;
3604a2b0a27dSPhilippe Mathieu-Daudé
3605a2b0a27dSPhilippe Mathieu-Daudé pos = get_DSPControl_pos(env);
3606a2b0a27dSPhilippe Mathieu-Daudé
3607a2b0a27dSPhilippe Mathieu-Daudé if (pos <= 64) {
3608a2b0a27dSPhilippe Mathieu-Daudé pos = pos + 64;
3609a2b0a27dSPhilippe Mathieu-Daudé set_DSPControl_pos(pos, env);
3610a2b0a27dSPhilippe Mathieu-Daudé }
3611a2b0a27dSPhilippe Mathieu-Daudé }
3612a2b0a27dSPhilippe Mathieu-Daudé #endif
3613a2b0a27dSPhilippe Mathieu-Daudé
cpu_wrdsp(uint32_t rs,uint32_t mask_num,CPUMIPSState * env)3614a2b0a27dSPhilippe Mathieu-Daudé void cpu_wrdsp(uint32_t rs, uint32_t mask_num, CPUMIPSState *env)
3615a2b0a27dSPhilippe Mathieu-Daudé {
3616a2b0a27dSPhilippe Mathieu-Daudé uint8_t mask[6];
3617a2b0a27dSPhilippe Mathieu-Daudé uint8_t i;
3618a2b0a27dSPhilippe Mathieu-Daudé uint32_t newbits, overwrite;
3619a2b0a27dSPhilippe Mathieu-Daudé target_ulong dsp;
3620a2b0a27dSPhilippe Mathieu-Daudé
3621a2b0a27dSPhilippe Mathieu-Daudé newbits = 0x00;
3622a2b0a27dSPhilippe Mathieu-Daudé overwrite = 0xFFFFFFFF;
3623a2b0a27dSPhilippe Mathieu-Daudé dsp = env->active_tc.DSPControl;
3624a2b0a27dSPhilippe Mathieu-Daudé
3625a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 6; i++) {
3626a2b0a27dSPhilippe Mathieu-Daudé mask[i] = (mask_num >> i) & 0x01;
3627a2b0a27dSPhilippe Mathieu-Daudé }
3628a2b0a27dSPhilippe Mathieu-Daudé
3629a2b0a27dSPhilippe Mathieu-Daudé if (mask[0] == 1) {
3630a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
3631a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0xFFFFFF80;
3632a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0xFFFFFF80;
3633a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x0000007F & rs;
3634a2b0a27dSPhilippe Mathieu-Daudé #else
3635a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0xFFFFFFC0;
3636a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0xFFFFFFC0;
3637a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x0000003F & rs;
3638a2b0a27dSPhilippe Mathieu-Daudé #endif
3639a2b0a27dSPhilippe Mathieu-Daudé }
3640a2b0a27dSPhilippe Mathieu-Daudé
3641a2b0a27dSPhilippe Mathieu-Daudé if (mask[1] == 1) {
3642a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0xFFFFE07F;
3643a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0xFFFFE07F;
3644a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x00001F80 & rs;
3645a2b0a27dSPhilippe Mathieu-Daudé }
3646a2b0a27dSPhilippe Mathieu-Daudé
3647a2b0a27dSPhilippe Mathieu-Daudé if (mask[2] == 1) {
3648a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0xFFFFDFFF;
3649a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0xFFFFDFFF;
3650a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x00002000 & rs;
3651a2b0a27dSPhilippe Mathieu-Daudé }
3652a2b0a27dSPhilippe Mathieu-Daudé
3653a2b0a27dSPhilippe Mathieu-Daudé if (mask[3] == 1) {
3654a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0xFF00FFFF;
3655a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0xFF00FFFF;
3656a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x00FF0000 & rs;
3657a2b0a27dSPhilippe Mathieu-Daudé }
3658a2b0a27dSPhilippe Mathieu-Daudé
3659a2b0a27dSPhilippe Mathieu-Daudé if (mask[4] == 1) {
3660a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0x00FFFFFF;
3661a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0x00FFFFFF;
3662a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
3663a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0xFF000000 & rs;
3664a2b0a27dSPhilippe Mathieu-Daudé #else
3665a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x0F000000 & rs;
3666a2b0a27dSPhilippe Mathieu-Daudé #endif
3667a2b0a27dSPhilippe Mathieu-Daudé }
3668a2b0a27dSPhilippe Mathieu-Daudé
3669a2b0a27dSPhilippe Mathieu-Daudé if (mask[5] == 1) {
3670a2b0a27dSPhilippe Mathieu-Daudé overwrite &= 0xFFFFBFFF;
3671a2b0a27dSPhilippe Mathieu-Daudé newbits &= 0xFFFFBFFF;
3672a2b0a27dSPhilippe Mathieu-Daudé newbits |= 0x00004000 & rs;
3673a2b0a27dSPhilippe Mathieu-Daudé }
3674a2b0a27dSPhilippe Mathieu-Daudé
3675a2b0a27dSPhilippe Mathieu-Daudé dsp = dsp & overwrite;
3676a2b0a27dSPhilippe Mathieu-Daudé dsp = dsp | newbits;
3677a2b0a27dSPhilippe Mathieu-Daudé env->active_tc.DSPControl = dsp;
3678a2b0a27dSPhilippe Mathieu-Daudé }
3679a2b0a27dSPhilippe Mathieu-Daudé
helper_wrdsp(target_ulong rs,target_ulong mask_num,CPUMIPSState * env)3680a2b0a27dSPhilippe Mathieu-Daudé void helper_wrdsp(target_ulong rs, target_ulong mask_num, CPUMIPSState *env)
3681a2b0a27dSPhilippe Mathieu-Daudé {
3682a2b0a27dSPhilippe Mathieu-Daudé cpu_wrdsp(rs, mask_num, env);
3683a2b0a27dSPhilippe Mathieu-Daudé }
3684a2b0a27dSPhilippe Mathieu-Daudé
cpu_rddsp(uint32_t mask_num,CPUMIPSState * env)3685a2b0a27dSPhilippe Mathieu-Daudé uint32_t cpu_rddsp(uint32_t mask_num, CPUMIPSState *env)
3686a2b0a27dSPhilippe Mathieu-Daudé {
3687a2b0a27dSPhilippe Mathieu-Daudé uint8_t mask[6];
3688a2b0a27dSPhilippe Mathieu-Daudé uint32_t ruler, i;
3689a2b0a27dSPhilippe Mathieu-Daudé target_ulong temp;
3690a2b0a27dSPhilippe Mathieu-Daudé target_ulong dsp;
3691a2b0a27dSPhilippe Mathieu-Daudé
3692a2b0a27dSPhilippe Mathieu-Daudé ruler = 0x01;
3693a2b0a27dSPhilippe Mathieu-Daudé for (i = 0; i < 6; i++) {
3694a2b0a27dSPhilippe Mathieu-Daudé mask[i] = (mask_num & ruler) >> i ;
3695a2b0a27dSPhilippe Mathieu-Daudé ruler = ruler << 1;
3696a2b0a27dSPhilippe Mathieu-Daudé }
3697a2b0a27dSPhilippe Mathieu-Daudé
3698a2b0a27dSPhilippe Mathieu-Daudé temp = 0x00;
3699a2b0a27dSPhilippe Mathieu-Daudé dsp = env->active_tc.DSPControl;
3700a2b0a27dSPhilippe Mathieu-Daudé
3701a2b0a27dSPhilippe Mathieu-Daudé if (mask[0] == 1) {
3702a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
3703a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x7F;
3704a2b0a27dSPhilippe Mathieu-Daudé #else
3705a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x3F;
3706a2b0a27dSPhilippe Mathieu-Daudé #endif
3707a2b0a27dSPhilippe Mathieu-Daudé }
3708a2b0a27dSPhilippe Mathieu-Daudé
3709a2b0a27dSPhilippe Mathieu-Daudé if (mask[1] == 1) {
3710a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x1F80;
3711a2b0a27dSPhilippe Mathieu-Daudé }
3712a2b0a27dSPhilippe Mathieu-Daudé
3713a2b0a27dSPhilippe Mathieu-Daudé if (mask[2] == 1) {
3714a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x2000;
3715a2b0a27dSPhilippe Mathieu-Daudé }
3716a2b0a27dSPhilippe Mathieu-Daudé
3717a2b0a27dSPhilippe Mathieu-Daudé if (mask[3] == 1) {
3718a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x00FF0000;
3719a2b0a27dSPhilippe Mathieu-Daudé }
3720a2b0a27dSPhilippe Mathieu-Daudé
3721a2b0a27dSPhilippe Mathieu-Daudé if (mask[4] == 1) {
3722a2b0a27dSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
3723a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0xFF000000;
3724a2b0a27dSPhilippe Mathieu-Daudé #else
3725a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x0F000000;
3726a2b0a27dSPhilippe Mathieu-Daudé #endif
3727a2b0a27dSPhilippe Mathieu-Daudé }
3728a2b0a27dSPhilippe Mathieu-Daudé
3729a2b0a27dSPhilippe Mathieu-Daudé if (mask[5] == 1) {
3730a2b0a27dSPhilippe Mathieu-Daudé temp |= dsp & 0x4000;
3731a2b0a27dSPhilippe Mathieu-Daudé }
3732a2b0a27dSPhilippe Mathieu-Daudé
3733a2b0a27dSPhilippe Mathieu-Daudé return temp;
3734a2b0a27dSPhilippe Mathieu-Daudé }
3735a2b0a27dSPhilippe Mathieu-Daudé
helper_rddsp(target_ulong mask_num,CPUMIPSState * env)3736a2b0a27dSPhilippe Mathieu-Daudé target_ulong helper_rddsp(target_ulong mask_num, CPUMIPSState *env)
3737a2b0a27dSPhilippe Mathieu-Daudé {
3738a2b0a27dSPhilippe Mathieu-Daudé return cpu_rddsp(mask_num, env);
3739a2b0a27dSPhilippe Mathieu-Daudé }
3740a2b0a27dSPhilippe Mathieu-Daudé
3741a2b0a27dSPhilippe Mathieu-Daudé
3742a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_LHI
3743a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_LLO
3744a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_HI
3745a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_LO
3746a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_Q3
3747a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_Q2
3748a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_Q1
3749a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_Q0
3750a2b0a27dSPhilippe Mathieu-Daudé
3751a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_SPLIT32_8
3752a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_SPLIT32_16
3753a2b0a27dSPhilippe Mathieu-Daudé
3754a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_RETURN32_8
3755a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_RETURN32_16
3756a2b0a27dSPhilippe Mathieu-Daudé
3757a2b0a27dSPhilippe Mathieu-Daudé #ifdef TARGET_MIPS64
3758a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_SPLIT64_16
3759a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_SPLIT64_32
3760a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_RETURN64_16
3761a2b0a27dSPhilippe Mathieu-Daudé #undef MIPSDSP_RETURN64_32
3762a2b0a27dSPhilippe Mathieu-Daudé #endif
3763