]> git.ipfire.org Git - thirdparty/gcc.git/blame - gcc/config/arm/vfp.md
re PR fortran/51306 (MOVE_ALLOC: Make more middle end friendlier)
[thirdparty/gcc.git] / gcc / config / arm / vfp.md
CommitLineData
3f4d9b98 1;; ARM VFP instruction patterns
d652f226
JJ
2;; Copyright (C) 2003, 2005, 2006, 2007, 2008, 2010
3;; Free Software Foundation, Inc.
3f4d9b98 4;; Written by CodeSourcery.
9b66ebb1
PB
5;;
6;; This file is part of GCC.
7;;
8;; GCC is free software; you can redistribute it and/or modify it
9;; under the terms of the GNU General Public License as published by
2f83c7d6 10;; the Free Software Foundation; either version 3, or (at your option)
9b66ebb1
PB
11;; any later version.
12;;
13;; GCC is distributed in the hope that it will be useful, but
14;; WITHOUT ANY WARRANTY; without even the implied warranty of
15;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16;; General Public License for more details.
17;;
18;; You should have received a copy of the GNU General Public License
2f83c7d6
NC
19;; along with GCC; see the file COPYING3. If not see
20;; <http://www.gnu.org/licenses/>. */
9b66ebb1
PB
21
22;; Additional register numbers
23(define_constants
f1adb0a9 24 [(VFPCC_REGNUM 127)]
9b66ebb1
PB
25)
26
9b66ebb1 27;; The VFP "type" attributes differ from those used in the FPA model.
51c69ddb
PB
28;; fcpys Single precision cpy.
29;; ffariths Single precision abs, neg.
30;; ffarithd Double precision abs, neg, cpy.
31;; fadds Single precision add/sub.
32;; faddd Double precision add/sub.
33;; fconsts Single precision load immediate.
34;; fconstd Double precision load immediate.
35;; fcmps Single precision comparison.
36;; fcmpd Double precision comparison.
a8e17e9e
PB
37;; fmuls Single precision multiply.
38;; fmuld Double precision multiply.
39;; fmacs Single precision multiply-accumulate.
40;; fmacd Double precision multiply-accumulate.
9b66ebb1
PB
41;; fdivs Single precision sqrt or division.
42;; fdivd Double precision sqrt or division.
75fe7b2f
RE
43;; f_flag fmstat operation
44;; f_load[sd] Floating point load from memory.
45;; f_store[sd] Floating point store to memory.
9b66ebb1
PB
46;; f_2_r Transfer vfp to arm reg.
47;; r_2_f Transfer arm to vfp reg.
75fe7b2f 48;; f_cvt Convert floating<->integral
9b66ebb1 49
9b66ebb1
PB
50;; SImode moves
51;; ??? For now do not allow loading constants into vfp regs. This causes
59b9a953 52;; problems because small constants get converted into adds.
9b66ebb1 53(define_insn "*arm_movsi_vfp"
f5c630c3 54 [(set (match_operand:SI 0 "nonimmediate_operand" "=rk,r,r,r,rk,m ,*t,r,*t,*t, *Uv")
d58bc084 55 (match_operand:SI 1 "general_operand" "rk, I,K,j,mi,rk,r,*t,*t,*Uvi,*t"))]
9b66ebb1
PB
56 "TARGET_ARM && TARGET_VFP && TARGET_HARD_FLOAT
57 && ( s_register_operand (operands[0], SImode)
58 || s_register_operand (operands[1], SImode))"
5b3e6663
PB
59 "*
60 switch (which_alternative)
61 {
f5c630c3 62 case 0: case 1:
5b3e6663 63 return \"mov%?\\t%0, %1\";
5b3e6663 64 case 2:
f5c630c3 65 return \"mvn%?\\t%0, #%B1\";
5b3e6663 66 case 3:
f5c630c3 67 return \"movw%?\\t%0, %1\";
5b3e6663 68 case 4:
f5c630c3 69 return \"ldr%?\\t%0, %1\";
5b3e6663 70 case 5:
f5c630c3 71 return \"str%?\\t%1, %0\";
5b3e6663 72 case 6:
f5c630c3 73 return \"fmsr%?\\t%0, %1\\t%@ int\";
5b3e6663 74 case 7:
f5c630c3
PB
75 return \"fmrs%?\\t%0, %1\\t%@ int\";
76 case 8:
5b3e6663 77 return \"fcpys%?\\t%0, %1\\t%@ int\";
f5c630c3 78 case 9: case 10:
5b3e6663
PB
79 return output_move_vfp (operands);
80 default:
81 gcc_unreachable ();
82 }
83 "
84 [(set_attr "predicable" "yes")
51c69ddb 85 (set_attr "type" "*,*,*,*,load1,store1,r_2_f,f_2_r,fcpys,f_loads,f_stores")
f8045c41 86 (set_attr "insn" "mov,mov,mvn,mov,*,*,*,*,*,*,*")
f5c630c3
PB
87 (set_attr "pool_range" "*,*,*,*,4096,*,*,*,*,1020,*")
88 (set_attr "neg_pool_range" "*,*,*,*,4084,*,*,*,*,1008,*")]
5b3e6663
PB
89)
90
28907f9a
MS
91;; See thumb2.md:thumb2_movsi_insn for an explanation of the split
92;; high/low register alternatives for loads and stores here.
5b3e6663 93(define_insn "*thumb2_movsi_vfp"
28907f9a
MS
94 [(set (match_operand:SI 0 "nonimmediate_operand" "=rk,r,r,r, l,*hk,m, *m,*t, r,*t,*t, *Uv")
95 (match_operand:SI 1 "general_operand" "rk, I,K,j,mi,*mi,l,*hk, r,*t,*t,*Uvi,*t"))]
5b3e6663
PB
96 "TARGET_THUMB2 && TARGET_VFP && TARGET_HARD_FLOAT
97 && ( s_register_operand (operands[0], SImode)
98 || s_register_operand (operands[1], SImode))"
99 "*
100 switch (which_alternative)
101 {
f5c630c3 102 case 0: case 1:
5b3e6663 103 return \"mov%?\\t%0, %1\";
5b3e6663 104 case 2:
f5c630c3 105 return \"mvn%?\\t%0, #%B1\";
5b3e6663 106 case 3:
f5c630c3 107 return \"movw%?\\t%0, %1\";
5b3e6663 108 case 4:
5b3e6663 109 case 5:
28907f9a 110 return \"ldr%?\\t%0, %1\";
5b3e6663 111 case 6:
5b3e6663 112 case 7:
28907f9a 113 return \"str%?\\t%1, %0\";
f5c630c3 114 case 8:
28907f9a
MS
115 return \"fmsr%?\\t%0, %1\\t%@ int\";
116 case 9:
117 return \"fmrs%?\\t%0, %1\\t%@ int\";
118 case 10:
5b3e6663 119 return \"fcpys%?\\t%0, %1\\t%@ int\";
28907f9a 120 case 11: case 12:
5b3e6663
PB
121 return output_move_vfp (operands);
122 default:
123 gcc_unreachable ();
124 }
125 "
9b66ebb1 126 [(set_attr "predicable" "yes")
837b01f6 127 (set_attr "type" "*,*,*,*,load1,load1,store1,store1,r_2_f,f_2_r,fcpys,f_loads,f_stores")
f8045c41 128 (set_attr "insn" "mov,mov,mvn,mov,*,*,*,*,*,*,*,*,*")
28907f9a
MS
129 (set_attr "pool_range" "*,*,*,*,1020,4096,*,*,*,*,*,1020,*")
130 (set_attr "neg_pool_range" "*,*,*,*, 0, 0,*,*,*,*,*,1008,*")]
9b66ebb1
PB
131)
132
133
134;; DImode moves
135
0127c76f
RR
136(define_insn "*movdi_vfp"
137 [(set (match_operand:DI 0 "nonimmediate_di_operand" "=r,r,r,r,r,r,m,w,r,w,w, Uv")
138 (match_operand:DI 1 "di_operand" "r,rDa,Db,Dc,mi,mi,r,r,w,w,Uvi,w"))]
139 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP && arm_tune != cortexa8
7baa7c13
BS
140 && ( register_operand (operands[0], DImode)
141 || register_operand (operands[1], DImode))"
142 "*
143 switch (which_alternative)
144 {
145 case 0:
7baa7c13
BS
146 case 1:
147 case 2:
7baa7c13 148 case 3:
0127c76f 149 return \"#\";
7baa7c13 150 case 4:
7baa7c13 151 case 5:
0127c76f 152 case 6:
3598da80 153 return output_move_double (operands, true, NULL);
0127c76f
RR
154 case 7:
155 return \"fmdrr%?\\t%P0, %Q1, %R1\\t%@ int\";
156 case 8:
157 return \"fmrrd%?\\t%Q0, %R0, %P1\\t%@ int\";
158 case 9:
7baa7c13
BS
159 if (TARGET_VFP_SINGLE)
160 return \"fcpys%?\\t%0, %1\\t%@ int\;fcpys%?\\t%p0, %p1\\t%@ int\";
161 else
162 return \"fcpyd%?\\t%P0, %P1\\t%@ int\";
0127c76f 163 case 10: case 11:
7baa7c13
BS
164 return output_move_vfp (operands);
165 default:
166 gcc_unreachable ();
167 }
168 "
0127c76f
RR
169 [(set_attr "type" "*,*,*,*,load2,load2,store2,r_2_f,f_2_r,ffarithd,f_loadd,f_stored")
170 (set_attr "neon_type" "*,*,*,*,*,*,*,neon_mcr_2_mcrr,neon_mrrc,neon_vmov,*,*")
171 (set (attr "length") (cond [(eq_attr "alternative" "1,4,5,6") (const_int 8)
172 (eq_attr "alternative" "2") (const_int 12)
173 (eq_attr "alternative" "3") (const_int 16)
174 (eq_attr "alternative" "9")
175 (if_then_else
b75b1be2 176 (match_test "TARGET_VFP_SINGLE")
0127c76f
RR
177 (const_int 8)
178 (const_int 4))]
179 (const_int 4)))
180 (set_attr "pool_range" "*,*,*,*,1020,4096,*,*,*,*,1020,*")
181 (set_attr "neg_pool_range" "*,*,*,*,1008,0,*,*,*,*,1008,*")
182 (set_attr "arch" "t2,any,any,any,a,t2,any,any,any,any,any,any")]
7baa7c13
BS
183)
184
0127c76f
RR
185(define_insn "*movdi_vfp_cortexa8"
186 [(set (match_operand:DI 0 "nonimmediate_di_operand" "=r,r,r,r,r,r,m,w,!r,w,w, Uv")
187 (match_operand:DI 1 "di_operand" "r,rDa,Db,Dc,mi,mi,r,r,w,w,Uvi,w"))]
188 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP && arm_tune == cortexa8
189 && ( register_operand (operands[0], DImode)
190 || register_operand (operands[1], DImode))"
9b66ebb1
PB
191 "*
192 switch (which_alternative)
193 {
9b901d50 194 case 0:
9b901d50
RE
195 case 1:
196 case 2:
9b66ebb1 197 case 3:
0127c76f 198 return \"#\";
9b66ebb1 199 case 4:
9b66ebb1 200 case 5:
0127c76f 201 case 6:
3598da80 202 return output_move_double (operands, true, NULL);
0127c76f 203 case 7:
5b3e6663 204 return \"fmdrr%?\\t%P0, %Q1, %R1\\t%@ int\";
0127c76f 205 case 8:
5b3e6663 206 return \"fmrrd%?\\t%Q0, %R0, %P1\\t%@ int\";
0127c76f
RR
207 case 9:
208 return \"fcpyd%?\\t%P0, %P1\\t%@ int\";
209 case 10: case 11:
5b3e6663
PB
210 return output_move_vfp (operands);
211 default:
0127c76f 212 gcc_unreachable ();
5b3e6663
PB
213 }
214 "
0127c76f 215 [(set_attr "type" "*,*,*,*,load2,load2,store2,r_2_f,f_2_r,ffarithd,f_loadd,f_stored")
3598da80
RR
216 (set (attr "length") (cond [(eq_attr "alternative" "1") (const_int 8)
217 (eq_attr "alternative" "2") (const_int 12)
218 (eq_attr "alternative" "3") (const_int 16)
219 (eq_attr "alternative" "4,5,6")
220 (symbol_ref
221 "arm_count_output_move_double_insns (operands) \
222 * 4")]
223 (const_int 4)))
0127c76f
RR
224 (set_attr "predicable" "yes")
225 (set_attr "pool_range" "*,*,*,*,1020,4096,*,*,*,*,1020,*")
226 (set_attr "neg_pool_range" "*,*,*,*,1008,0,*,*,*,*,1008,*")
3598da80
RR
227 (set (attr "ce_count")
228 (symbol_ref "get_attr_length (insn) / 4"))
0127c76f
RR
229 (set_attr "arch" "t2,any,any,any,a,t2,any,any,any,any,any,any")]
230 )
5b3e6663 231
0fd8c3ad 232;; HFmode moves
e0dc3601 233(define_insn "*movhf_vfp_neon"
0fd8c3ad
SL
234 [(set (match_operand:HF 0 "nonimmediate_operand" "= t,Um,r,m,t,r,t,r,r")
235 (match_operand:HF 1 "general_operand" " Um, t,m,r,t,r,r,t,F"))]
236 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_NEON_FP16
237 && ( s_register_operand (operands[0], HFmode)
238 || s_register_operand (operands[1], HFmode))"
239 "*
240 switch (which_alternative)
241 {
242 case 0: /* S register from memory */
243 return \"vld1.16\\t{%z0}, %A1\";
244 case 1: /* memory from S register */
245 return \"vst1.16\\t{%z1}, %A0\";
246 case 2: /* ARM register from memory */
247 return \"ldrh\\t%0, %1\\t%@ __fp16\";
248 case 3: /* memory from ARM register */
249 return \"strh\\t%1, %0\\t%@ __fp16\";
250 case 4: /* S register from S register */
251 return \"fcpys\\t%0, %1\";
252 case 5: /* ARM register from ARM register */
253 return \"mov\\t%0, %1\\t%@ __fp16\";
254 case 6: /* S register from ARM register */
255 return \"fmsr\\t%0, %1\";
256 case 7: /* ARM register from S register */
257 return \"fmrs\\t%0, %1\";
258 case 8: /* ARM register from constant */
259 {
260 REAL_VALUE_TYPE r;
261 long bits;
262 rtx ops[4];
263
264 REAL_VALUE_FROM_CONST_DOUBLE (r, operands[1]);
265 bits = real_to_target (NULL, &r, HFmode);
266 ops[0] = operands[0];
267 ops[1] = GEN_INT (bits);
268 ops[2] = GEN_INT (bits & 0xff00);
269 ops[3] = GEN_INT (bits & 0x00ff);
270
271 if (arm_arch_thumb2)
272 output_asm_insn (\"movw\\t%0, %1\", ops);
273 else
274 output_asm_insn (\"mov\\t%0, %2\;orr\\t%0, %0, %3\", ops);
275 return \"\";
276 }
277 default:
278 gcc_unreachable ();
279 }
280 "
281 [(set_attr "conds" "unconditional")
282 (set_attr "type" "*,*,load1,store1,fcpys,*,r_2_f,f_2_r,*")
283 (set_attr "neon_type" "neon_vld1_1_2_regs,neon_vst1_1_2_regs_vst2_2_regs,*,*,*,*,*,*,*")
284 (set_attr "length" "4,4,4,4,4,4,4,4,8")]
285)
286
e0dc3601
PB
287;; FP16 without element load/store instructions.
288(define_insn "*movhf_vfp"
289 [(set (match_operand:HF 0 "nonimmediate_operand" "=r,m,t,r,t,r,r")
290 (match_operand:HF 1 "general_operand" " m,r,t,r,r,t,F"))]
291 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_FP16 && !TARGET_NEON_FP16
292 && ( s_register_operand (operands[0], HFmode)
293 || s_register_operand (operands[1], HFmode))"
294 "*
295 switch (which_alternative)
296 {
297 case 0: /* ARM register from memory */
298 return \"ldrh\\t%0, %1\\t%@ __fp16\";
299 case 1: /* memory from ARM register */
300 return \"strh\\t%1, %0\\t%@ __fp16\";
301 case 2: /* S register from S register */
302 return \"fcpys\\t%0, %1\";
303 case 3: /* ARM register from ARM register */
304 return \"mov\\t%0, %1\\t%@ __fp16\";
305 case 4: /* S register from ARM register */
306 return \"fmsr\\t%0, %1\";
307 case 5: /* ARM register from S register */
308 return \"fmrs\\t%0, %1\";
309 case 6: /* ARM register from constant */
310 {
311 REAL_VALUE_TYPE r;
312 long bits;
313 rtx ops[4];
314
315 REAL_VALUE_FROM_CONST_DOUBLE (r, operands[1]);
316 bits = real_to_target (NULL, &r, HFmode);
317 ops[0] = operands[0];
318 ops[1] = GEN_INT (bits);
319 ops[2] = GEN_INT (bits & 0xff00);
320 ops[3] = GEN_INT (bits & 0x00ff);
321
322 if (arm_arch_thumb2)
323 output_asm_insn (\"movw\\t%0, %1\", ops);
324 else
325 output_asm_insn (\"mov\\t%0, %2\;orr\\t%0, %0, %3\", ops);
326 return \"\";
327 }
328 default:
329 gcc_unreachable ();
330 }
331 "
332 [(set_attr "conds" "unconditional")
333 (set_attr "type" "load1,store1,fcpys,*,r_2_f,f_2_r,*")
334 (set_attr "length" "4,4,4,4,4,4,8")]
335)
336
9b66ebb1
PB
337
338;; SFmode moves
221b2a64
PB
339;; Disparage the w<->r cases because reloading an invalid address is
340;; preferable to loading the value via integer registers.
9b66ebb1
PB
341
342(define_insn "*movsf_vfp"
f1adb0a9
JB
343 [(set (match_operand:SF 0 "nonimmediate_operand" "=t,?r,t ,t ,Uv,r ,m,t,r")
344 (match_operand:SF 1 "general_operand" " ?r,t,Dv,UvE,t, mE,r,t,r"))]
9b66ebb1
PB
345 "TARGET_ARM && TARGET_HARD_FLOAT && TARGET_VFP
346 && ( s_register_operand (operands[0], SFmode)
347 || s_register_operand (operands[1], SFmode))"
5b3e6663
PB
348 "*
349 switch (which_alternative)
350 {
351 case 0:
352 return \"fmsr%?\\t%0, %1\";
353 case 1:
354 return \"fmrs%?\\t%0, %1\";
f1adb0a9
JB
355 case 2:
356 return \"fconsts%?\\t%0, #%G1\";
357 case 3: case 4:
5b3e6663 358 return output_move_vfp (operands);
5b3e6663 359 case 5:
f1adb0a9 360 return \"ldr%?\\t%0, %1\\t%@ float\";
5b3e6663 361 case 6:
f1adb0a9 362 return \"str%?\\t%1, %0\\t%@ float\";
5b3e6663 363 case 7:
f1adb0a9
JB
364 return \"fcpys%?\\t%0, %1\";
365 case 8:
5b3e6663
PB
366 return \"mov%?\\t%0, %1\\t%@ float\";
367 default:
368 gcc_unreachable ();
369 }
370 "
9b66ebb1 371 [(set_attr "predicable" "yes")
f1adb0a9 372 (set_attr "type"
51c69ddb 373 "r_2_f,f_2_r,fconsts,f_loads,f_stores,load1,store1,fcpys,*")
f8045c41 374 (set_attr "insn" "*,*,*,*,*,*,*,*,mov")
f1adb0a9
JB
375 (set_attr "pool_range" "*,*,*,1020,*,4096,*,*,*")
376 (set_attr "neg_pool_range" "*,*,*,1008,*,4080,*,*,*")]
9b66ebb1
PB
377)
378
5b3e6663 379(define_insn "*thumb2_movsf_vfp"
f1adb0a9
JB
380 [(set (match_operand:SF 0 "nonimmediate_operand" "=t,?r,t, t ,Uv,r ,m,t,r")
381 (match_operand:SF 1 "general_operand" " ?r,t,Dv,UvE,t, mE,r,t,r"))]
5b3e6663
PB
382 "TARGET_THUMB2 && TARGET_HARD_FLOAT && TARGET_VFP
383 && ( s_register_operand (operands[0], SFmode)
384 || s_register_operand (operands[1], SFmode))"
385 "*
386 switch (which_alternative)
387 {
388 case 0:
389 return \"fmsr%?\\t%0, %1\";
390 case 1:
391 return \"fmrs%?\\t%0, %1\";
f1adb0a9
JB
392 case 2:
393 return \"fconsts%?\\t%0, #%G1\";
394 case 3: case 4:
5b3e6663 395 return output_move_vfp (operands);
5b3e6663 396 case 5:
f1adb0a9 397 return \"ldr%?\\t%0, %1\\t%@ float\";
5b3e6663 398 case 6:
f1adb0a9 399 return \"str%?\\t%1, %0\\t%@ float\";
5b3e6663 400 case 7:
f1adb0a9
JB
401 return \"fcpys%?\\t%0, %1\";
402 case 8:
5b3e6663
PB
403 return \"mov%?\\t%0, %1\\t%@ float\";
404 default:
405 gcc_unreachable ();
406 }
407 "
408 [(set_attr "predicable" "yes")
f1adb0a9 409 (set_attr "type"
837b01f6 410 "r_2_f,f_2_r,fconsts,f_loads,f_stores,load1,store1,fcpys,*")
f8045c41 411 (set_attr "insn" "*,*,*,*,*,*,*,*,mov")
f1adb0a9
JB
412 (set_attr "pool_range" "*,*,*,1020,*,4092,*,*,*")
413 (set_attr "neg_pool_range" "*,*,*,1008,*,0,*,*,*")]
5b3e6663
PB
414)
415
9b66ebb1
PB
416
417;; DFmode moves
418
419(define_insn "*movdf_vfp"
95f89bb3
RR
420 [(set (match_operand:DF 0 "nonimmediate_soft_df_operand" "=w,?r,w ,w ,Uv,r, m,w,r")
421 (match_operand:DF 1 "soft_df_operand" " ?r,w,Dy,UvF,w ,mF,r,w,r"))]
d5b6e637
PB
422 "TARGET_ARM && TARGET_HARD_FLOAT && TARGET_VFP
423 && ( register_operand (operands[0], DFmode)
424 || register_operand (operands[1], DFmode))"
9b66ebb1
PB
425 "*
426 {
427 switch (which_alternative)
428 {
429 case 0:
430 return \"fmdrr%?\\t%P0, %Q1, %R1\";
431 case 1:
432 return \"fmrrd%?\\t%Q0, %R0, %P1\";
f1adb0a9 433 case 2:
e0dc3601 434 gcc_assert (TARGET_VFP_DOUBLE);
f1adb0a9
JB
435 return \"fconstd%?\\t%P0, #%G1\";
436 case 3: case 4:
5b3e6663 437 return output_move_vfp (operands);
95f89bb3 438 case 5: case 6:
3598da80 439 return output_move_double (operands, true, NULL);
9b901d50 440 case 7:
e0dc3601
PB
441 if (TARGET_VFP_SINGLE)
442 return \"fcpys%?\\t%0, %1\;fcpys%?\\t%p0, %p1\";
443 else
444 return \"fcpyd%?\\t%P0, %P1\";
f1adb0a9 445 case 8:
9b901d50 446 return \"#\";
9b66ebb1 447 default:
e6d29d15 448 gcc_unreachable ();
9b66ebb1
PB
449 }
450 }
451 "
f1adb0a9 452 [(set_attr "type"
51c69ddb 453 "r_2_f,f_2_r,fconstd,f_loadd,f_stored,load2,store2,ffarithd,*")
95f89bb3 454 (set (attr "length") (cond [(eq_attr "alternative" "5,6,8") (const_int 8)
e0dc3601
PB
455 (eq_attr "alternative" "7")
456 (if_then_else
b75b1be2 457 (match_test "TARGET_VFP_SINGLE")
e0dc3601
PB
458 (const_int 8)
459 (const_int 4))]
460 (const_int 4)))
54f52b81 461 (set_attr "predicable" "yes")
f1adb0a9
JB
462 (set_attr "pool_range" "*,*,*,1020,*,1020,*,*,*")
463 (set_attr "neg_pool_range" "*,*,*,1008,*,1008,*,*,*")]
9b66ebb1
PB
464)
465
5b3e6663 466(define_insn "*thumb2_movdf_vfp"
95f89bb3
RR
467 [(set (match_operand:DF 0 "nonimmediate_soft_df_operand" "=w,?r,w ,w ,Uv,r ,m,w,r")
468 (match_operand:DF 1 "soft_df_operand" " ?r,w,Dy,UvF,w, mF,r, w,r"))]
5b3e6663
PB
469 "TARGET_THUMB2 && TARGET_HARD_FLOAT && TARGET_VFP"
470 "*
471 {
472 switch (which_alternative)
473 {
474 case 0:
475 return \"fmdrr%?\\t%P0, %Q1, %R1\";
476 case 1:
477 return \"fmrrd%?\\t%Q0, %R0, %P1\";
f1adb0a9 478 case 2:
e0dc3601 479 gcc_assert (TARGET_VFP_DOUBLE);
f1adb0a9 480 return \"fconstd%?\\t%P0, #%G1\";
95f89bb3 481 case 3: case 4:
5b3e6663 482 return output_move_vfp (operands);
95f89bb3 483 case 5: case 6: case 8:
3598da80 484 return output_move_double (operands, true, NULL);
f1adb0a9 485 case 7:
e0dc3601
PB
486 if (TARGET_VFP_SINGLE)
487 return \"fcpys%?\\t%0, %1\;fcpys%?\\t%p0, %p1\";
488 else
489 return \"fcpyd%?\\t%P0, %P1\";
5b3e6663
PB
490 default:
491 abort ();
492 }
493 }
494 "
f1adb0a9 495 [(set_attr "type"
95f89bb3
RR
496 "r_2_f,f_2_r,fconstd,f_loadd,f_stored,load2,store2,ffarithd,*")
497 (set (attr "length") (cond [(eq_attr "alternative" "5,6,8") (const_int 8)
e0dc3601
PB
498 (eq_attr "alternative" "7")
499 (if_then_else
b75b1be2 500 (match_test "TARGET_VFP_SINGLE")
e0dc3601
PB
501 (const_int 8)
502 (const_int 4))]
503 (const_int 4)))
95f89bb3
RR
504 (set_attr "pool_range" "*,*,*,1020,*,4096,*,*,*")
505 (set_attr "neg_pool_range" "*,*,*,1008,*,0,*,*,*")]
5b3e6663
PB
506)
507
9b66ebb1
PB
508
509;; Conditional move patterns
510
511(define_insn "*movsfcc_vfp"
f1adb0a9 512 [(set (match_operand:SF 0 "s_register_operand" "=t,t,t,t,t,t,?r,?r,?r")
9b66ebb1
PB
513 (if_then_else:SF
514 (match_operator 3 "arm_comparison_operator"
515 [(match_operand 4 "cc_register" "") (const_int 0)])
f1adb0a9
JB
516 (match_operand:SF 1 "s_register_operand" "0,t,t,0,?r,?r,0,t,t")
517 (match_operand:SF 2 "s_register_operand" "t,0,t,?r,0,?r,t,0,t")))]
9b66ebb1
PB
518 "TARGET_ARM && TARGET_HARD_FLOAT && TARGET_VFP"
519 "@
520 fcpys%D3\\t%0, %2
521 fcpys%d3\\t%0, %1
522 fcpys%D3\\t%0, %2\;fcpys%d3\\t%0, %1
523 fmsr%D3\\t%0, %2
524 fmsr%d3\\t%0, %1
525 fmsr%D3\\t%0, %2\;fmsr%d3\\t%0, %1
526 fmrs%D3\\t%0, %2
527 fmrs%d3\\t%0, %1
528 fmrs%D3\\t%0, %2\;fmrs%d3\\t%0, %1"
529 [(set_attr "conds" "use")
530 (set_attr "length" "4,4,8,4,4,8,4,4,8")
51c69ddb 531 (set_attr "type" "fcpys,fcpys,fcpys,r_2_f,r_2_f,r_2_f,f_2_r,f_2_r,f_2_r")]
9b66ebb1
PB
532)
533
5b3e6663 534(define_insn "*thumb2_movsfcc_vfp"
f1adb0a9 535 [(set (match_operand:SF 0 "s_register_operand" "=t,t,t,t,t,t,?r,?r,?r")
5b3e6663
PB
536 (if_then_else:SF
537 (match_operator 3 "arm_comparison_operator"
538 [(match_operand 4 "cc_register" "") (const_int 0)])
f1adb0a9
JB
539 (match_operand:SF 1 "s_register_operand" "0,t,t,0,?r,?r,0,t,t")
540 (match_operand:SF 2 "s_register_operand" "t,0,t,?r,0,?r,t,0,t")))]
5b3e6663
PB
541 "TARGET_THUMB2 && TARGET_HARD_FLOAT && TARGET_VFP"
542 "@
543 it\\t%D3\;fcpys%D3\\t%0, %2
544 it\\t%d3\;fcpys%d3\\t%0, %1
545 ite\\t%D3\;fcpys%D3\\t%0, %2\;fcpys%d3\\t%0, %1
546 it\\t%D3\;fmsr%D3\\t%0, %2
547 it\\t%d3\;fmsr%d3\\t%0, %1
548 ite\\t%D3\;fmsr%D3\\t%0, %2\;fmsr%d3\\t%0, %1
549 it\\t%D3\;fmrs%D3\\t%0, %2
550 it\\t%d3\;fmrs%d3\\t%0, %1
551 ite\\t%D3\;fmrs%D3\\t%0, %2\;fmrs%d3\\t%0, %1"
552 [(set_attr "conds" "use")
553 (set_attr "length" "6,6,10,6,6,10,6,6,10")
51c69ddb 554 (set_attr "type" "fcpys,fcpys,fcpys,r_2_f,r_2_f,r_2_f,f_2_r,f_2_r,f_2_r")]
5b3e6663
PB
555)
556
9b66ebb1
PB
557(define_insn "*movdfcc_vfp"
558 [(set (match_operand:DF 0 "s_register_operand" "=w,w,w,w,w,w,?r,?r,?r")
559 (if_then_else:DF
560 (match_operator 3 "arm_comparison_operator"
561 [(match_operand 4 "cc_register" "") (const_int 0)])
562 (match_operand:DF 1 "s_register_operand" "0,w,w,0,?r,?r,0,w,w")
563 (match_operand:DF 2 "s_register_operand" "w,0,w,?r,0,?r,w,0,w")))]
e0dc3601 564 "TARGET_ARM && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
565 "@
566 fcpyd%D3\\t%P0, %P2
567 fcpyd%d3\\t%P0, %P1
568 fcpyd%D3\\t%P0, %P2\;fcpyd%d3\\t%P0, %P1
569 fmdrr%D3\\t%P0, %Q2, %R2
570 fmdrr%d3\\t%P0, %Q1, %R1
571 fmdrr%D3\\t%P0, %Q2, %R2\;fmdrr%d3\\t%P0, %Q1, %R1
572 fmrrd%D3\\t%Q0, %R0, %P2
573 fmrrd%d3\\t%Q0, %R0, %P1
574 fmrrd%D3\\t%Q0, %R0, %P2\;fmrrd%d3\\t%Q0, %R0, %P1"
575 [(set_attr "conds" "use")
576 (set_attr "length" "4,4,8,4,4,8,4,4,8")
51c69ddb 577 (set_attr "type" "ffarithd,ffarithd,ffarithd,r_2_f,r_2_f,r_2_f,f_2_r,f_2_r,f_2_r")]
9b66ebb1
PB
578)
579
5b3e6663
PB
580(define_insn "*thumb2_movdfcc_vfp"
581 [(set (match_operand:DF 0 "s_register_operand" "=w,w,w,w,w,w,?r,?r,?r")
582 (if_then_else:DF
583 (match_operator 3 "arm_comparison_operator"
584 [(match_operand 4 "cc_register" "") (const_int 0)])
585 (match_operand:DF 1 "s_register_operand" "0,w,w,0,?r,?r,0,w,w")
586 (match_operand:DF 2 "s_register_operand" "w,0,w,?r,0,?r,w,0,w")))]
e0dc3601 587 "TARGET_THUMB2 && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
5b3e6663
PB
588 "@
589 it\\t%D3\;fcpyd%D3\\t%P0, %P2
590 it\\t%d3\;fcpyd%d3\\t%P0, %P1
591 ite\\t%D3\;fcpyd%D3\\t%P0, %P2\;fcpyd%d3\\t%P0, %P1
592 it\t%D3\;fmdrr%D3\\t%P0, %Q2, %R2
593 it\t%d3\;fmdrr%d3\\t%P0, %Q1, %R1
594 ite\\t%D3\;fmdrr%D3\\t%P0, %Q2, %R2\;fmdrr%d3\\t%P0, %Q1, %R1
595 it\t%D3\;fmrrd%D3\\t%Q0, %R0, %P2
596 it\t%d3\;fmrrd%d3\\t%Q0, %R0, %P1
597 ite\\t%D3\;fmrrd%D3\\t%Q0, %R0, %P2\;fmrrd%d3\\t%Q0, %R0, %P1"
598 [(set_attr "conds" "use")
599 (set_attr "length" "6,6,10,6,6,10,6,6,10")
51c69ddb 600 (set_attr "type" "ffarithd,ffarithd,ffarithd,r_2_f,r_2_f,r_2_f,f_2_r,f_2_r,f_2_r")]
5b3e6663
PB
601)
602
9b66ebb1
PB
603
604;; Sign manipulation functions
605
606(define_insn "*abssf2_vfp"
f1adb0a9
JB
607 [(set (match_operand:SF 0 "s_register_operand" "=t")
608 (abs:SF (match_operand:SF 1 "s_register_operand" "t")))]
5b3e6663 609 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
610 "fabss%?\\t%0, %1"
611 [(set_attr "predicable" "yes")
51c69ddb 612 (set_attr "type" "ffariths")]
9b66ebb1
PB
613)
614
615(define_insn "*absdf2_vfp"
616 [(set (match_operand:DF 0 "s_register_operand" "=w")
617 (abs:DF (match_operand:DF 1 "s_register_operand" "w")))]
e0dc3601 618 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
619 "fabsd%?\\t%P0, %P1"
620 [(set_attr "predicable" "yes")
51c69ddb 621 (set_attr "type" "ffarithd")]
9b66ebb1
PB
622)
623
624(define_insn "*negsf2_vfp"
f1adb0a9
JB
625 [(set (match_operand:SF 0 "s_register_operand" "=t,?r")
626 (neg:SF (match_operand:SF 1 "s_register_operand" "t,r")))]
5b3e6663 627 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
81632f11
RE
628 "@
629 fnegs%?\\t%0, %1
630 eor%?\\t%0, %1, #-2147483648"
9b66ebb1 631 [(set_attr "predicable" "yes")
51c69ddb 632 (set_attr "type" "ffariths")]
9b66ebb1
PB
633)
634
81632f11
RE
635(define_insn_and_split "*negdf2_vfp"
636 [(set (match_operand:DF 0 "s_register_operand" "=w,?r,?r")
637 (neg:DF (match_operand:DF 1 "s_register_operand" "w,0,r")))]
e0dc3601 638 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
81632f11
RE
639 "@
640 fnegd%?\\t%P0, %P1
641 #
642 #"
e0dc3601 643 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE && reload_completed
81632f11
RE
644 && arm_general_register_operand (operands[0], DFmode)"
645 [(set (match_dup 0) (match_dup 1))]
646 "
647 if (REGNO (operands[0]) == REGNO (operands[1]))
648 {
649 operands[0] = gen_highpart (SImode, operands[0]);
650 operands[1] = gen_rtx_XOR (SImode, operands[0], GEN_INT (0x80000000));
651 }
652 else
653 {
654 rtx in_hi, in_lo, out_hi, out_lo;
655
656 in_hi = gen_rtx_XOR (SImode, gen_highpart (SImode, operands[1]),
657 GEN_INT (0x80000000));
658 in_lo = gen_lowpart (SImode, operands[1]);
659 out_hi = gen_highpart (SImode, operands[0]);
660 out_lo = gen_lowpart (SImode, operands[0]);
661
662 if (REGNO (in_lo) == REGNO (out_hi))
663 {
664 emit_insn (gen_rtx_SET (SImode, out_lo, in_lo));
665 operands[0] = out_hi;
666 operands[1] = in_hi;
667 }
668 else
669 {
670 emit_insn (gen_rtx_SET (SImode, out_hi, in_hi));
671 operands[0] = out_lo;
672 operands[1] = in_lo;
673 }
674 }
675 "
9b66ebb1 676 [(set_attr "predicable" "yes")
81632f11 677 (set_attr "length" "4,4,8")
51c69ddb 678 (set_attr "type" "ffarithd")]
9b66ebb1
PB
679)
680
681
682;; Arithmetic insns
683
684(define_insn "*addsf3_vfp"
f1adb0a9
JB
685 [(set (match_operand:SF 0 "s_register_operand" "=t")
686 (plus:SF (match_operand:SF 1 "s_register_operand" "t")
687 (match_operand:SF 2 "s_register_operand" "t")))]
5b3e6663 688 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
689 "fadds%?\\t%0, %1, %2"
690 [(set_attr "predicable" "yes")
51c69ddb 691 (set_attr "type" "fadds")]
9b66ebb1
PB
692)
693
694(define_insn "*adddf3_vfp"
695 [(set (match_operand:DF 0 "s_register_operand" "=w")
696 (plus:DF (match_operand:DF 1 "s_register_operand" "w")
697 (match_operand:DF 2 "s_register_operand" "w")))]
e0dc3601 698 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
699 "faddd%?\\t%P0, %P1, %P2"
700 [(set_attr "predicable" "yes")
51c69ddb 701 (set_attr "type" "faddd")]
9b66ebb1
PB
702)
703
704
705(define_insn "*subsf3_vfp"
f1adb0a9
JB
706 [(set (match_operand:SF 0 "s_register_operand" "=t")
707 (minus:SF (match_operand:SF 1 "s_register_operand" "t")
708 (match_operand:SF 2 "s_register_operand" "t")))]
5b3e6663 709 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
710 "fsubs%?\\t%0, %1, %2"
711 [(set_attr "predicable" "yes")
51c69ddb 712 (set_attr "type" "fadds")]
9b66ebb1
PB
713)
714
715(define_insn "*subdf3_vfp"
716 [(set (match_operand:DF 0 "s_register_operand" "=w")
717 (minus:DF (match_operand:DF 1 "s_register_operand" "w")
718 (match_operand:DF 2 "s_register_operand" "w")))]
e0dc3601 719 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
720 "fsubd%?\\t%P0, %P1, %P2"
721 [(set_attr "predicable" "yes")
51c69ddb 722 (set_attr "type" "faddd")]
9b66ebb1
PB
723)
724
725
726;; Division insns
727
728(define_insn "*divsf3_vfp"
0498a2be 729 [(set (match_operand:SF 0 "s_register_operand" "=t")
f1adb0a9
JB
730 (div:SF (match_operand:SF 1 "s_register_operand" "t")
731 (match_operand:SF 2 "s_register_operand" "t")))]
5b3e6663 732 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
733 "fdivs%?\\t%0, %1, %2"
734 [(set_attr "predicable" "yes")
735 (set_attr "type" "fdivs")]
736)
737
738(define_insn "*divdf3_vfp"
0498a2be 739 [(set (match_operand:DF 0 "s_register_operand" "=w")
9b66ebb1
PB
740 (div:DF (match_operand:DF 1 "s_register_operand" "w")
741 (match_operand:DF 2 "s_register_operand" "w")))]
e0dc3601 742 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
743 "fdivd%?\\t%P0, %P1, %P2"
744 [(set_attr "predicable" "yes")
745 (set_attr "type" "fdivd")]
746)
747
748
749;; Multiplication insns
750
751(define_insn "*mulsf3_vfp"
0498a2be 752 [(set (match_operand:SF 0 "s_register_operand" "=t")
f1adb0a9
JB
753 (mult:SF (match_operand:SF 1 "s_register_operand" "t")
754 (match_operand:SF 2 "s_register_operand" "t")))]
5b3e6663 755 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
756 "fmuls%?\\t%0, %1, %2"
757 [(set_attr "predicable" "yes")
a8e17e9e 758 (set_attr "type" "fmuls")]
9b66ebb1
PB
759)
760
761(define_insn "*muldf3_vfp"
0498a2be 762 [(set (match_operand:DF 0 "s_register_operand" "=w")
9b66ebb1
PB
763 (mult:DF (match_operand:DF 1 "s_register_operand" "w")
764 (match_operand:DF 2 "s_register_operand" "w")))]
e0dc3601 765 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
766 "fmuld%?\\t%P0, %P1, %P2"
767 [(set_attr "predicable" "yes")
a8e17e9e 768 (set_attr "type" "fmuld")]
9b66ebb1
PB
769)
770
9b66ebb1 771(define_insn "*mulsf3negsf_vfp"
0498a2be 772 [(set (match_operand:SF 0 "s_register_operand" "=t")
f1adb0a9
JB
773 (mult:SF (neg:SF (match_operand:SF 1 "s_register_operand" "t"))
774 (match_operand:SF 2 "s_register_operand" "t")))]
5b3e6663 775 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
776 "fnmuls%?\\t%0, %1, %2"
777 [(set_attr "predicable" "yes")
a8e17e9e 778 (set_attr "type" "fmuls")]
9b66ebb1
PB
779)
780
781(define_insn "*muldf3negdf_vfp"
0498a2be 782 [(set (match_operand:DF 0 "s_register_operand" "=w")
9b66ebb1
PB
783 (mult:DF (neg:DF (match_operand:DF 1 "s_register_operand" "w"))
784 (match_operand:DF 2 "s_register_operand" "w")))]
e0dc3601 785 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
786 "fnmuld%?\\t%P0, %P1, %P2"
787 [(set_attr "predicable" "yes")
a8e17e9e 788 (set_attr "type" "fmuld")]
9b66ebb1
PB
789)
790
791
792;; Multiply-accumulate insns
793
794;; 0 = 1 * 2 + 0
795(define_insn "*mulsf3addsf_vfp"
f1adb0a9
JB
796 [(set (match_operand:SF 0 "s_register_operand" "=t")
797 (plus:SF (mult:SF (match_operand:SF 2 "s_register_operand" "t")
798 (match_operand:SF 3 "s_register_operand" "t"))
9b66ebb1 799 (match_operand:SF 1 "s_register_operand" "0")))]
5b3e6663 800 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
801 "fmacs%?\\t%0, %2, %3"
802 [(set_attr "predicable" "yes")
a8e17e9e 803 (set_attr "type" "fmacs")]
9b66ebb1
PB
804)
805
806(define_insn "*muldf3adddf_vfp"
807 [(set (match_operand:DF 0 "s_register_operand" "=w")
808 (plus:DF (mult:DF (match_operand:DF 2 "s_register_operand" "w")
809 (match_operand:DF 3 "s_register_operand" "w"))
810 (match_operand:DF 1 "s_register_operand" "0")))]
e0dc3601 811 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
812 "fmacd%?\\t%P0, %P2, %P3"
813 [(set_attr "predicable" "yes")
a8e17e9e 814 (set_attr "type" "fmacd")]
9b66ebb1
PB
815)
816
817;; 0 = 1 * 2 - 0
818(define_insn "*mulsf3subsf_vfp"
f1adb0a9
JB
819 [(set (match_operand:SF 0 "s_register_operand" "=t")
820 (minus:SF (mult:SF (match_operand:SF 2 "s_register_operand" "t")
821 (match_operand:SF 3 "s_register_operand" "t"))
9b66ebb1 822 (match_operand:SF 1 "s_register_operand" "0")))]
5b3e6663 823 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
824 "fmscs%?\\t%0, %2, %3"
825 [(set_attr "predicable" "yes")
a8e17e9e 826 (set_attr "type" "fmacs")]
9b66ebb1
PB
827)
828
829(define_insn "*muldf3subdf_vfp"
830 [(set (match_operand:DF 0 "s_register_operand" "=w")
831 (minus:DF (mult:DF (match_operand:DF 2 "s_register_operand" "w")
832 (match_operand:DF 3 "s_register_operand" "w"))
833 (match_operand:DF 1 "s_register_operand" "0")))]
e0dc3601 834 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
835 "fmscd%?\\t%P0, %P2, %P3"
836 [(set_attr "predicable" "yes")
a8e17e9e 837 (set_attr "type" "fmacd")]
9b66ebb1
PB
838)
839
840;; 0 = -(1 * 2) + 0
841(define_insn "*mulsf3negsfaddsf_vfp"
f1adb0a9 842 [(set (match_operand:SF 0 "s_register_operand" "=t")
9b66ebb1 843 (minus:SF (match_operand:SF 1 "s_register_operand" "0")
f1adb0a9
JB
844 (mult:SF (match_operand:SF 2 "s_register_operand" "t")
845 (match_operand:SF 3 "s_register_operand" "t"))))]
5b3e6663 846 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
847 "fnmacs%?\\t%0, %2, %3"
848 [(set_attr "predicable" "yes")
a8e17e9e 849 (set_attr "type" "fmacs")]
9b66ebb1
PB
850)
851
852(define_insn "*fmuldf3negdfadddf_vfp"
853 [(set (match_operand:DF 0 "s_register_operand" "=w")
854 (minus:DF (match_operand:DF 1 "s_register_operand" "0")
855 (mult:DF (match_operand:DF 2 "s_register_operand" "w")
856 (match_operand:DF 3 "s_register_operand" "w"))))]
e0dc3601 857 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
858 "fnmacd%?\\t%P0, %P2, %P3"
859 [(set_attr "predicable" "yes")
a8e17e9e 860 (set_attr "type" "fmacd")]
9b66ebb1
PB
861)
862
863
864;; 0 = -(1 * 2) - 0
865(define_insn "*mulsf3negsfsubsf_vfp"
f1adb0a9 866 [(set (match_operand:SF 0 "s_register_operand" "=t")
9b66ebb1 867 (minus:SF (mult:SF
f1adb0a9
JB
868 (neg:SF (match_operand:SF 2 "s_register_operand" "t"))
869 (match_operand:SF 3 "s_register_operand" "t"))
9b66ebb1 870 (match_operand:SF 1 "s_register_operand" "0")))]
5b3e6663 871 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
872 "fnmscs%?\\t%0, %2, %3"
873 [(set_attr "predicable" "yes")
a8e17e9e 874 (set_attr "type" "fmacs")]
9b66ebb1
PB
875)
876
877(define_insn "*muldf3negdfsubdf_vfp"
878 [(set (match_operand:DF 0 "s_register_operand" "=w")
879 (minus:DF (mult:DF
880 (neg:DF (match_operand:DF 2 "s_register_operand" "w"))
881 (match_operand:DF 3 "s_register_operand" "w"))
882 (match_operand:DF 1 "s_register_operand" "0")))]
e0dc3601 883 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
884 "fnmscd%?\\t%P0, %P2, %P3"
885 [(set_attr "predicable" "yes")
a8e17e9e 886 (set_attr "type" "fmacd")]
9b66ebb1
PB
887)
888
889
890;; Conversion routines
891
892(define_insn "*extendsfdf2_vfp"
893 [(set (match_operand:DF 0 "s_register_operand" "=w")
f1adb0a9 894 (float_extend:DF (match_operand:SF 1 "s_register_operand" "t")))]
e0dc3601 895 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
896 "fcvtds%?\\t%P0, %1"
897 [(set_attr "predicable" "yes")
75fe7b2f 898 (set_attr "type" "f_cvt")]
9b66ebb1
PB
899)
900
901(define_insn "*truncdfsf2_vfp"
f1adb0a9 902 [(set (match_operand:SF 0 "s_register_operand" "=t")
9b66ebb1 903 (float_truncate:SF (match_operand:DF 1 "s_register_operand" "w")))]
e0dc3601 904 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
905 "fcvtsd%?\\t%0, %P1"
906 [(set_attr "predicable" "yes")
75fe7b2f 907 (set_attr "type" "f_cvt")]
9b66ebb1
PB
908)
909
0fd8c3ad
SL
910(define_insn "extendhfsf2"
911 [(set (match_operand:SF 0 "s_register_operand" "=t")
912 (float_extend:SF (match_operand:HF 1 "s_register_operand" "t")))]
e0dc3601 913 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_FP16"
0fd8c3ad
SL
914 "vcvtb%?.f32.f16\\t%0, %1"
915 [(set_attr "predicable" "yes")
916 (set_attr "type" "f_cvt")]
917)
918
919(define_insn "truncsfhf2"
920 [(set (match_operand:HF 0 "s_register_operand" "=t")
921 (float_truncate:HF (match_operand:SF 1 "s_register_operand" "t")))]
e0dc3601 922 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_FP16"
0fd8c3ad
SL
923 "vcvtb%?.f16.f32\\t%0, %1"
924 [(set_attr "predicable" "yes")
925 (set_attr "type" "f_cvt")]
926)
927
9b66ebb1 928(define_insn "*truncsisf2_vfp"
f1adb0a9
JB
929 [(set (match_operand:SI 0 "s_register_operand" "=t")
930 (fix:SI (fix:SF (match_operand:SF 1 "s_register_operand" "t"))))]
5b3e6663 931 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
932 "ftosizs%?\\t%0, %1"
933 [(set_attr "predicable" "yes")
75fe7b2f 934 (set_attr "type" "f_cvt")]
9b66ebb1
PB
935)
936
937(define_insn "*truncsidf2_vfp"
f1adb0a9 938 [(set (match_operand:SI 0 "s_register_operand" "=t")
9b66ebb1 939 (fix:SI (fix:DF (match_operand:DF 1 "s_register_operand" "w"))))]
e0dc3601 940 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
941 "ftosizd%?\\t%0, %P1"
942 [(set_attr "predicable" "yes")
75fe7b2f 943 (set_attr "type" "f_cvt")]
9b66ebb1
PB
944)
945
6f6c1f6d
PB
946
947(define_insn "fixuns_truncsfsi2"
f1adb0a9
JB
948 [(set (match_operand:SI 0 "s_register_operand" "=t")
949 (unsigned_fix:SI (fix:SF (match_operand:SF 1 "s_register_operand" "t"))))]
5b3e6663 950 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
6f6c1f6d
PB
951 "ftouizs%?\\t%0, %1"
952 [(set_attr "predicable" "yes")
75fe7b2f 953 (set_attr "type" "f_cvt")]
6f6c1f6d
PB
954)
955
956(define_insn "fixuns_truncdfsi2"
f1adb0a9
JB
957 [(set (match_operand:SI 0 "s_register_operand" "=t")
958 (unsigned_fix:SI (fix:DF (match_operand:DF 1 "s_register_operand" "t"))))]
e0dc3601 959 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
6f6c1f6d
PB
960 "ftouizd%?\\t%0, %P1"
961 [(set_attr "predicable" "yes")
75fe7b2f 962 (set_attr "type" "f_cvt")]
6f6c1f6d
PB
963)
964
965
9b66ebb1 966(define_insn "*floatsisf2_vfp"
f1adb0a9
JB
967 [(set (match_operand:SF 0 "s_register_operand" "=t")
968 (float:SF (match_operand:SI 1 "s_register_operand" "t")))]
5b3e6663 969 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
970 "fsitos%?\\t%0, %1"
971 [(set_attr "predicable" "yes")
75fe7b2f 972 (set_attr "type" "f_cvt")]
9b66ebb1
PB
973)
974
975(define_insn "*floatsidf2_vfp"
976 [(set (match_operand:DF 0 "s_register_operand" "=w")
f1adb0a9 977 (float:DF (match_operand:SI 1 "s_register_operand" "t")))]
e0dc3601 978 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
979 "fsitod%?\\t%P0, %1"
980 [(set_attr "predicable" "yes")
75fe7b2f 981 (set_attr "type" "f_cvt")]
9b66ebb1
PB
982)
983
984
6f6c1f6d 985(define_insn "floatunssisf2"
f1adb0a9
JB
986 [(set (match_operand:SF 0 "s_register_operand" "=t")
987 (unsigned_float:SF (match_operand:SI 1 "s_register_operand" "t")))]
5b3e6663 988 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
6f6c1f6d
PB
989 "fuitos%?\\t%0, %1"
990 [(set_attr "predicable" "yes")
75fe7b2f 991 (set_attr "type" "f_cvt")]
6f6c1f6d
PB
992)
993
994(define_insn "floatunssidf2"
995 [(set (match_operand:DF 0 "s_register_operand" "=w")
f1adb0a9 996 (unsigned_float:DF (match_operand:SI 1 "s_register_operand" "t")))]
e0dc3601 997 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
6f6c1f6d
PB
998 "fuitod%?\\t%P0, %1"
999 [(set_attr "predicable" "yes")
75fe7b2f 1000 (set_attr "type" "f_cvt")]
6f6c1f6d
PB
1001)
1002
1003
9b66ebb1
PB
1004;; Sqrt insns.
1005
1006(define_insn "*sqrtsf2_vfp"
f1adb0a9
JB
1007 [(set (match_operand:SF 0 "s_register_operand" "=t")
1008 (sqrt:SF (match_operand:SF 1 "s_register_operand" "t")))]
5b3e6663 1009 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
1010 "fsqrts%?\\t%0, %1"
1011 [(set_attr "predicable" "yes")
1012 (set_attr "type" "fdivs")]
1013)
1014
1015(define_insn "*sqrtdf2_vfp"
1016 [(set (match_operand:DF 0 "s_register_operand" "=w")
1017 (sqrt:DF (match_operand:DF 1 "s_register_operand" "w")))]
e0dc3601 1018 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
1019 "fsqrtd%?\\t%P0, %P1"
1020 [(set_attr "predicable" "yes")
1021 (set_attr "type" "fdivd")]
1022)
1023
1024
1025;; Patterns to split/copy vfp condition flags.
1026
1027(define_insn "*movcc_vfp"
1028 [(set (reg CC_REGNUM)
1029 (reg VFPCC_REGNUM))]
5b3e6663 1030 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
1031 "fmstat%?"
1032 [(set_attr "conds" "set")
75fe7b2f 1033 (set_attr "type" "f_flag")]
9b66ebb1
PB
1034)
1035
1036(define_insn_and_split "*cmpsf_split_vfp"
1037 [(set (reg:CCFP CC_REGNUM)
f1adb0a9
JB
1038 (compare:CCFP (match_operand:SF 0 "s_register_operand" "t")
1039 (match_operand:SF 1 "vfp_compare_operand" "tG")))]
5b3e6663 1040 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1 1041 "#"
5b3e6663 1042 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
1043 [(set (reg:CCFP VFPCC_REGNUM)
1044 (compare:CCFP (match_dup 0)
1045 (match_dup 1)))
1046 (set (reg:CCFP CC_REGNUM)
1047 (reg:CCFP VFPCC_REGNUM))]
1048 ""
1049)
1050
1051(define_insn_and_split "*cmpsf_trap_split_vfp"
1052 [(set (reg:CCFPE CC_REGNUM)
f1adb0a9
JB
1053 (compare:CCFPE (match_operand:SF 0 "s_register_operand" "t")
1054 (match_operand:SF 1 "vfp_compare_operand" "tG")))]
5b3e6663 1055 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1 1056 "#"
5b3e6663 1057 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
1058 [(set (reg:CCFPE VFPCC_REGNUM)
1059 (compare:CCFPE (match_dup 0)
1060 (match_dup 1)))
1061 (set (reg:CCFPE CC_REGNUM)
1062 (reg:CCFPE VFPCC_REGNUM))]
1063 ""
1064)
1065
1066(define_insn_and_split "*cmpdf_split_vfp"
1067 [(set (reg:CCFP CC_REGNUM)
1068 (compare:CCFP (match_operand:DF 0 "s_register_operand" "w")
1069 (match_operand:DF 1 "vfp_compare_operand" "wG")))]
e0dc3601 1070 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1 1071 "#"
e0dc3601 1072 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
1073 [(set (reg:CCFP VFPCC_REGNUM)
1074 (compare:CCFP (match_dup 0)
1075 (match_dup 1)))
1076 (set (reg:CCFP CC_REGNUM)
510bc854 1077 (reg:CCFP VFPCC_REGNUM))]
9b66ebb1
PB
1078 ""
1079)
1080
1081(define_insn_and_split "*cmpdf_trap_split_vfp"
1082 [(set (reg:CCFPE CC_REGNUM)
1083 (compare:CCFPE (match_operand:DF 0 "s_register_operand" "w")
1084 (match_operand:DF 1 "vfp_compare_operand" "wG")))]
e0dc3601 1085 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1 1086 "#"
e0dc3601 1087 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
1088 [(set (reg:CCFPE VFPCC_REGNUM)
1089 (compare:CCFPE (match_dup 0)
1090 (match_dup 1)))
1091 (set (reg:CCFPE CC_REGNUM)
1092 (reg:CCFPE VFPCC_REGNUM))]
1093 ""
1094)
1095
1096
1097;; Comparison patterns
1098
1099(define_insn "*cmpsf_vfp"
1100 [(set (reg:CCFP VFPCC_REGNUM)
f1adb0a9
JB
1101 (compare:CCFP (match_operand:SF 0 "s_register_operand" "t,t")
1102 (match_operand:SF 1 "vfp_compare_operand" "t,G")))]
5b3e6663 1103 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
1104 "@
1105 fcmps%?\\t%0, %1
1106 fcmpzs%?\\t%0"
1107 [(set_attr "predicable" "yes")
51c69ddb 1108 (set_attr "type" "fcmps")]
9b66ebb1
PB
1109)
1110
1111(define_insn "*cmpsf_trap_vfp"
1112 [(set (reg:CCFPE VFPCC_REGNUM)
f1adb0a9
JB
1113 (compare:CCFPE (match_operand:SF 0 "s_register_operand" "t,t")
1114 (match_operand:SF 1 "vfp_compare_operand" "t,G")))]
5b3e6663 1115 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
9b66ebb1
PB
1116 "@
1117 fcmpes%?\\t%0, %1
1118 fcmpezs%?\\t%0"
1119 [(set_attr "predicable" "yes")
7dd8ecf0 1120 (set_attr "type" "fcmps")]
9b66ebb1
PB
1121)
1122
1123(define_insn "*cmpdf_vfp"
1124 [(set (reg:CCFP VFPCC_REGNUM)
1125 (compare:CCFP (match_operand:DF 0 "s_register_operand" "w,w")
1126 (match_operand:DF 1 "vfp_compare_operand" "w,G")))]
e0dc3601 1127 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
1128 "@
1129 fcmpd%?\\t%P0, %P1
1130 fcmpzd%?\\t%P0"
1131 [(set_attr "predicable" "yes")
51c69ddb 1132 (set_attr "type" "fcmpd")]
9b66ebb1
PB
1133)
1134
1135(define_insn "*cmpdf_trap_vfp"
1136 [(set (reg:CCFPE VFPCC_REGNUM)
1137 (compare:CCFPE (match_operand:DF 0 "s_register_operand" "w,w")
1138 (match_operand:DF 1 "vfp_compare_operand" "w,G")))]
e0dc3601 1139 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP_DOUBLE"
9b66ebb1
PB
1140 "@
1141 fcmped%?\\t%P0, %P1
1142 fcmpezd%?\\t%P0"
1143 [(set_attr "predicable" "yes")
51c69ddb 1144 (set_attr "type" "fcmpd")]
9b66ebb1
PB
1145)
1146
1147
1148;; Store multiple insn used in function prologue.
1149
1150(define_insn "*push_multi_vfp"
1151 [(match_parallel 2 "multi_register_push"
1152 [(set (match_operand:BLK 0 "memory_operand" "=m")
4e6f5666 1153 (unspec:BLK [(match_operand:DF 1 "vfp_register_operand" "")]
9b66ebb1 1154 UNSPEC_PUSH_MULT))])]
5b3e6663 1155 "TARGET_32BIT && TARGET_HARD_FLOAT && TARGET_VFP"
8edfc4cc 1156 "* return vfp_output_fstmd (operands);"
75fe7b2f 1157 [(set_attr "type" "f_stored")]
9b66ebb1
PB
1158)
1159
1160
1161;; Unimplemented insns:
1162;; fldm*
1163;; fstm*
1164;; fmdhr et al (VFPv1)
59b9a953 1165;; Support for xD (single precision only) variants.
9b66ebb1 1166;; fmrrs, fmsrr