xref: /openbmc/qemu/target/mips/tcg/dsp_helper.c (revision 48804eebd4a327e4b11f902ba80a00876ee53a43)
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