[ARM] Add test to check pcs of ARM ABI runtime floating point helpers
authorPeter Smith <peter.smith@linaro.org>
Fri, 28 Jul 2017 09:21:00 +0000 (09:21 +0000)
committerPeter Smith <peter.smith@linaro.org>
Fri, 28 Jul 2017 09:21:00 +0000 (09:21 +0000)
The ARM Runtime ABI document (IHI0043) defines the AEABI floating point
helper functions in section 4.1.2 The floating-point helper functions.
The functions listed in this section must always use the base AAPCS calling
convention.

This test generates calls to all the helper functions that llvm supports
and checks that the base AAPCS calling convention has been used. We test
the equivalent of -mfloat-abi=soft, -mfloat-abi=softfp, -mfloat-abi=hardfp
with an FPU that supports single and double precision, and one that only
supports double precision.

Differential Revision: https://reviews.llvm.org/D35904

llvm-svn: 309371

llvm/test/CodeGen/ARM/float-helpers.s [new file with mode: 0644]

diff --git a/llvm/test/CodeGen/ARM/float-helpers.s b/llvm/test/CodeGen/ARM/float-helpers.s
new file mode 100644 (file)
index 0000000..c861f1f
--- /dev/null
@@ -0,0 +1,1195 @@
+; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
+; RUN: llc -asm-verbose=false -mattr=-vfp2 -mtriple=arm-eabi < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-SOFT
+; RUN: llc -asm-verbose=false -mattr=-vfp2 -mtriple=arm-eabi -meabi=gnu < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-SOFT
+; RUN: llc -asm-verbose=false -mattr=+vfp3 -mtriple=arm-eabi < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-SOFTFP
+; RUN: llc -asm-verbose=false -mattr=+vfp3 -meabi=gnu -mtriple=arm-eabi < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-SOFTFP
+; RUN: llc -asm-verbose=false -mattr=+vfp3 -float-abi=hard -mtriple=arm-eabi < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-HARDFP-SP -check-prefix=CHECK-HARDFP-DP
+; RUN: llc -asm-verbose=false -mattr=+vfp3 -float-abi=hard -meabi=gnu -mtriple=arm-eabi < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-HARDFP-SP -check-prefix=CHECK-HARDFP-DP
+; RUN: llc -asm-verbose=false -mattr=+vfp3,+fp-only-sp -float-abi=hard -mtriple=arm-eabi < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-HARDFP-SP -check-prefix=CHECK-HARDFP-SPONLY
+; RUN: llc -asm-verbose=false -mattr=+vfp3,+fp-only-sp -float-abi=hard -mtriple=arm-eabi -meabi=gnu < %s | FileCheck %s -check-prefix=CHECK-ALL -check-prefix=CHECK-HARDFP-SP -check-prefix=CHECK-HARDFP-SPONLY
+
+; The Runtime ABI for the ARM Architecture IHI0043 section 4.1.2 The
+; floating-point helper functions to always use the base AAPCS (soft-float)
+; calling convention.
+
+; In this test we cover the following configurations:
+; CHECK-SOFT -mfloat-abi=soft
+;     * expect no use of floating point instructions
+;     * expect to use __aeabi_ helper function in each function
+; CHECK-SOFTFP -mfloat-abi=softfp
+;     * all functions use base AAPCS
+;     * floating point instructions permitted, so __aeabi_ helpers only
+;       expected when there is no available instruction.
+; CHECK-HARD-FP-SP -mfloat-abi=hardfp (single precision instructions)
+;     * all non Runtime ABI helper functions use AAPCS VFP
+;     * floating point instructions permitted, so __aeabi_ helpers only
+;       expected when there is no available instruction.
+; CHECK-HARD-FP-DP -mfloat-abi=hardfp (double precision instructions)
+; CHECK-HARD_FP_SPONLY -mfloat-abi=hardfp (double precision but single
+;                      precision only FPU)
+;     * as CHECK-HARD-FP-SP, but we split up the double precision helper
+;       functions so we can test a single precision only FPU, which has to use
+;       helper function for all double precision operations.
+
+; In all cases we must use base AAPCS when calling a helper function from
+; section 4.1.2.
+
+target datalayout = "e-m:e-p:32:32-i64:64-v128:64:128-a:0:32-n32-S64"
+target triple = "arm-eabi"
+
+define float @fadd(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fadd:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fadd
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fadd:
+; CHECK-SOFTFP:         vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vmov s2, r0
+; CHECK-SOFTFP-NEXT:    vadd.f32 s0, s2, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fadd:
+; CHECK-HARDFP-SP:         vadd.f32 s0, s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %add = fadd float %a, %b
+  ret float %add
+}
+
+define float @fdiv(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fdiv:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fdiv
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fdiv:
+; CHECK-SOFTFP:         vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vmov s2, r0
+; CHECK-SOFTFP-NEXT:    vdiv.f32 s0, s2, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fdiv:
+; CHECK-HARDFP-SP:         vdiv.f32 s0, s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %div = fdiv float %a, %b
+  ret float %div
+}
+
+define float @fmul(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fmul:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fmul
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fmul:
+; CHECK-SOFTFP:         vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vmov s2, r0
+; CHECK-SOFTFP-NEXT:    vmul.f32 s0, s2, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fmul:
+; CHECK-HARDFP-SP:         vmul.f32 s0, s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %mul = fmul float %a, %b
+  ret float %mul
+}
+
+define float @fsub(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fsub:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fsub
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fsub:
+; CHECK-SOFTFP:         vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vmov s2, r0
+; CHECK-SOFTFP-NEXT:    vsub.f32 s0, s2, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fsub:
+; CHECK-HARDFP-SP:         vsub.f32 s0, s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %sub = fsub float %a, %b
+  ret float %sub
+}
+
+define i32 @fcmpeq(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fcmpeq:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fcmpeq
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fcmpeq:
+; CHECK-SOFTFP:         vmov s2, r0
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vcmp.f32 s2, s0
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    moveq r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fcmpeq:
+; CHECK-HARDFP-SP:         vcmp.f32 s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov r0, #0
+; CHECK-HARDFP-SP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-SP-NEXT:    moveq r0, #1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp oeq float %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @fcmplt(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fcmplt:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fcmplt
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fcmplt:
+; CHECK-SOFTFP:         vmov s2, r0
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vcmpe.f32 s2, s0
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movmi r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fcmplt:
+; CHECK-HARDFP-SP:         vcmpe.f32 s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov r0, #0
+; CHECK-HARDFP-SP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-SP-NEXT:    movmi r0, #1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp olt float %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @fcmple(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fcmple:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fcmple
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fcmple:
+; CHECK-SOFTFP:         vmov s2, r0
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vcmpe.f32 s2, s0
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movls r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fcmple:
+; CHECK-HARDFP-SP:         vcmpe.f32 s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov r0, #0
+; CHECK-HARDFP-SP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-SP-NEXT:    movls r0, #1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp ole float %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @fcmpge(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fcmpge:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fcmpge
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fcmpge:
+; CHECK-SOFTFP:         vmov s2, r0
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vcmpe.f32 s2, s0
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movge r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fcmpge:
+; CHECK-HARDFP-SP:         vcmpe.f32 s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov r0, #0
+; CHECK-HARDFP-SP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-SP-NEXT:    movge r0, #1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp oge float %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @fcmpgt(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fcmpgt:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fcmpgt
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fcmpgt:
+; CHECK-SOFTFP:         vmov s2, r0
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vcmpe.f32 s2, s0
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movgt r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fcmpgt:
+; CHECK-HARDFP-SP:         vcmpe.f32 s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov r0, #0
+; CHECK-HARDFP-SP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-SP-NEXT:    movgt r0, #1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp ogt float %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @fcmpun(float %a, float %b) #0 {
+; CHECK-SOFT-LABEL: fcmpun:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_fcmpun
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: fcmpun:
+; CHECK-SOFTFP:         vmov s2, r0
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vmov s0, r1
+; CHECK-SOFTFP-NEXT:    vcmpe.f32 s2, s0
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movvs r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: fcmpun:
+; CHECK-HARDFP-SP:         vcmpe.f32 s0, s1
+; CHECK-HARDFP-SP-NEXT:    mov r0, #0
+; CHECK-HARDFP-SP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-SP-NEXT:    movvs r0, #1
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp uno float %a, %b
+  %0 = zext i1 %cmp to i32
+  ret i32 %0
+}
+
+define double @dadd(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dadd:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dadd
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dadd:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    vadd.f64 d16, d17, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dadd:
+; CHECK-HARDFP-DP:         vadd.f64 d0, d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dadd:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dadd
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %add = fadd double %a, %b
+  ret double %add
+}
+
+define double @ddiv(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: ddiv:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_ddiv
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: ddiv:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    vdiv.f64 d16, d17, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: ddiv:
+; CHECK-HARDFP-DP:         vdiv.f64 d0, d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: ddiv:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_ddiv
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %div = fdiv double %a, %b
+  ret double %div
+}
+
+define double @dmul(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dmul:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dmul
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dmul:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    vmul.f64 d16, d17, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dmul:
+; CHECK-HARDFP-DP:         vmul.f64 d0, d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dmul:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dmul
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %mul = fmul double %a, %b
+  ret double %mul
+}
+
+define double @dsub(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dsub:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dsub
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dsub:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    vsub.f64 d16, d17, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dsub:
+; CHECK-HARDFP-DP:         vsub.f64 d0, d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dsub:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dsub
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %sub = fsub double %a, %b
+  ret double %sub
+}
+
+define i32 @dcmpeq(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dcmpeq:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dcmpeq
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dcmpeq:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vcmp.f64 d17, d16
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    moveq r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dcmpeq:
+; CHECK-HARDFP-DP:         vcmp.f64 d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov r0, #0
+; CHECK-HARDFP-DP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-DP-NEXT:    moveq r0, #1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dcmpeq:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dcmpeq
+; CHECK-HARDFP-SPONLY-NEXT:    cmp r0, #0
+; CHECK-HARDFP-SPONLY-NEXT:    movne r0, #1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp oeq double %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @dcmplt(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dcmplt:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dcmplt
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dcmplt:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vcmpe.f64 d17, d16
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movmi r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dcmplt:
+; CHECK-HARDFP-DP:         vcmpe.f64 d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov r0, #0
+; CHECK-HARDFP-DP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-DP-NEXT:    movmi r0, #1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dcmplt:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dcmplt
+; CHECK-HARDFP-SPONLY-NEXT:    cmp r0, #0
+; CHECK-HARDFP-SPONLY-NEXT:    movne r0, #1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp olt double %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @dcmple(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dcmple:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dcmple
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dcmple:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vcmpe.f64 d17, d16
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movls r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dcmple:
+; CHECK-HARDFP-DP:         vcmpe.f64 d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov r0, #0
+; CHECK-HARDFP-DP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-DP-NEXT:    movls r0, #1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dcmple:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dcmple
+; CHECK-HARDFP-SPONLY-NEXT:    cmp r0, #0
+; CHECK-HARDFP-SPONLY-NEXT:    movne r0, #1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp ole double %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @dcmpge(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dcmpge:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dcmpge
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dcmpge:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vcmpe.f64 d17, d16
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movge r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dcmpge:
+; CHECK-HARDFP-DP:         vcmpe.f64 d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov r0, #0
+; CHECK-HARDFP-DP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-DP-NEXT:    movge r0, #1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dcmpge:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dcmpge
+; CHECK-HARDFP-SPONLY-NEXT:    cmp r0, #0
+; CHECK-HARDFP-SPONLY-NEXT:    movne r0, #1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp oge double %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @dcmpgt(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dcmpgt:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dcmpgt
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dcmpgt:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vcmpe.f64 d17, d16
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movgt r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dcmpgt:
+; CHECK-HARDFP-DP:         vcmpe.f64 d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov r0, #0
+; CHECK-HARDFP-DP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-DP-NEXT:    movgt r0, #1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dcmpgt:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dcmpgt
+; CHECK-HARDFP-SPONLY-NEXT:    cmp r0, #0
+; CHECK-HARDFP-SPONLY-NEXT:    movne r0, #1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp ogt double %a, %b
+  %conv = zext i1 %cmp to i32
+  ret i32 %conv
+}
+
+define i32 @dcmpun(double %a, double %b) #0 {
+; CHECK-SOFT-LABEL: dcmpun:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_dcmpun
+; CHECK-SOFT-NEXT:    cmp r0, #0
+; CHECK-SOFT-NEXT:    movne r0, #1
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: dcmpun:
+; CHECK-SOFTFP:         vmov d16, r2, r3
+; CHECK-SOFTFP-NEXT:    vmov d17, r0, r1
+; CHECK-SOFTFP-NEXT:    mov r0, #0
+; CHECK-SOFTFP-NEXT:    vcmpe.f64 d17, d16
+; CHECK-SOFTFP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-SOFTFP-NEXT:    movvs r0, #1
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: dcmpun:
+; CHECK-HARDFP-DP:         vcmpe.f64 d0, d1
+; CHECK-HARDFP-DP-NEXT:    mov r0, #0
+; CHECK-HARDFP-DP-NEXT:    vmrs APSR_nzcv, fpscr
+; CHECK-HARDFP-DP-NEXT:    movvs r0, #1
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: dcmpun:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r2, r3, d1
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_dcmpun
+; CHECK-HARDFP-SPONLY-NEXT:    cmp r0, #0
+; CHECK-HARDFP-SPONLY-NEXT:    movne r0, #1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %cmp = fcmp uno double %a, %b
+  %0 = zext i1 %cmp to i32
+  ret i32 %0
+}
+
+define i32 @d2iz(double %a) #0 {
+; CHECK-SOFT-LABEL: d2iz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_d2iz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: d2iz:
+; CHECK-SOFTFP:         vmov d16, r0, r1
+; CHECK-SOFTFP-NEXT:    vcvt.s32.f64 s0, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: d2iz:
+; CHECK-HARDFP-DP:         vcvt.s32.f64 s0, d0
+; CHECK-HARDFP-DP-NEXT:    vmov r0, s0
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: d2iz:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_d2iz
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %conv = fptosi double %a to i32
+  ret i32 %conv
+}
+
+define i32 @d2uiz(double %a) #0 {
+; CHECK-SOFT-LABEL: d2uiz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_d2uiz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: d2uiz:
+; CHECK-SOFTFP:         vmov d16, r0, r1
+; CHECK-SOFTFP-NEXT:    vcvt.u32.f64 s0, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: d2uiz:
+; CHECK-HARDFP-DP:         vcvt.u32.f64 s0, d0
+; CHECK-HARDFP-DP-NEXT:    vmov r0, s0
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: d2uiz:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_d2uiz
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %conv = fptoui double %a to i32
+  ret i32 %conv
+}
+
+define i64 @d2lz(double %a) #0 {
+; CHECK-SOFT-LABEL: d2lz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_d2lz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: d2lz:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_d2lz
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: d2lz:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_d2lz
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = fptosi double %a to i64
+  ret i64 %conv
+}
+
+define i64 @d2ulz(double %a) #0 {
+; CHECK-SOFT-LABEL: d2ulz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_d2ulz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: d2ulz:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_d2ulz
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: d2ulz:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_d2ulz
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = fptoui double %a to i64
+  ret i64 %conv
+}
+
+define i32 @f2iz(float %a) #0 {
+; CHECK-SOFT-LABEL: f2iz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_f2iz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: f2iz:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.s32.f32 s0, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: f2iz:
+; CHECK-HARDFP-SP:         vcvt.s32.f32 s0, s0
+; CHECK-HARDFP-SP-NEXT:    vmov r0, s0
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = fptosi float %a to i32
+  ret i32 %conv
+}
+
+define i32 @f2uiz(float %a) #0 {
+; CHECK-SOFT-LABEL: f2uiz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_f2uiz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: f2uiz:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.u32.f32 s0, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: f2uiz:
+; CHECK-HARDFP-SP:         vcvt.u32.f32 s0, s0
+; CHECK-HARDFP-SP-NEXT:    vmov r0, s0
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = fptoui float %a to i32
+  ret i32 %conv
+}
+
+define i64 @f2lz(float %a) #0 {
+; CHECK-SOFT-LABEL: f2lz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_f2lz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: f2lz:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_f2lz
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: f2lz:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    vmov r0, s0
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_f2lz
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = fptosi float %a to i64
+  ret i64 %conv
+}
+
+define i64 @f2ulz(float %a) #0 {
+; CHECK-SOFT-LABEL: f2ulz:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_f2ulz
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: f2ulz:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_f2ulz
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: f2ulz:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    vmov r0, s0
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_f2ulz
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = fptoui float %a to i64
+  ret i64 %conv
+}
+
+define float @d2f(double %a) #0 {
+; CHECK-SOFT-LABEL: d2f:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_d2f
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: d2f:
+; CHECK-SOFTFP:         vmov d16, r0, r1
+; CHECK-SOFTFP-NEXT:    vcvt.f32.f64 s0, d16
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: d2f:
+; CHECK-HARDFP-DP:         vcvt.f32.f64 s0, d0
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: d2f:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, r1, d0
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_d2f
+; CHECK-HARDFP-SPONLY-NEXT:    vmov s0, r0
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %conv = fptrunc double %a to float
+  ret float %conv
+}
+
+define double @f2d(float %a) #0 {
+; CHECK-SOFT-LABEL: f2d:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_f2d
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: f2d:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.f64.f32 d16, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: f2d:
+; CHECK-HARDFP-DP:         vcvt.f64.f32 d0, s0
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: f2d:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    vmov r0, s0
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_f2d
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %conv = fpext float %a to double
+  ret double %conv
+}
+
+define double @i2d(i32 %a) #0 {
+; CHECK-SOFT-LABEL: i2d:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_i2d
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: i2d:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.f64.s32 d16, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: i2d:
+; CHECK-HARDFP-DP:         vmov s0, r0
+; CHECK-HARDFP-DP-NEXT:    vcvt.f64.s32 d0, s0
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: i2d:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_i2d
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %conv = sitofp i32 %a to double
+  ret double %conv
+}
+
+define double @ui2d(i32 %a) #0 {
+; CHECK-SOFT-LABEL: ui2d:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_ui2d
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: ui2d:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.f64.u32 d16, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, r1, d16
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-DP-LABEL: ui2d:
+; CHECK-HARDFP-DP:         vmov s0, r0
+; CHECK-HARDFP-DP-NEXT:    vcvt.f64.u32 d0, s0
+; CHECK-HARDFP-DP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SPONLY-LABEL: ui2d:
+; CHECK-HARDFP-SPONLY:         .save {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    bl __aeabi_ui2d
+; CHECK-HARDFP-SPONLY-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SPONLY-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SPONLY-NEXT:    mov pc, lr
+entry:
+  %conv = uitofp i32 %a to double
+  ret double %conv
+}
+
+define double @l2d(i64 %a) #0 {
+; CHECK-SOFT-LABEL: l2d:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_l2d
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: l2d:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_l2d
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: l2d:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_l2d
+; CHECK-HARDFP-SP-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = sitofp i64 %a to double
+  ret double %conv
+}
+
+define double @ul2d(i64 %a) #0 {
+; CHECK-SOFT-LABEL: ul2d:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_ul2d
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: ul2d:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_ul2d
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: ul2d:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_ul2d
+; CHECK-HARDFP-SP-NEXT:    vmov d0, r0, r1
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = uitofp i64 %a to double
+  ret double %conv
+}
+
+define float @i2f(i32 %a) #0 {
+; CHECK-SOFT-LABEL: i2f:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_i2f
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: i2f:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.f32.s32 s0, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: i2f:
+; CHECK-HARDFP-SP:         vmov s0, r0
+; CHECK-HARDFP-SP-NEXT:    vcvt.f32.s32 s0, s0
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = sitofp i32 %a to float
+  ret float %conv
+}
+
+define float @ui2f(i32 %a) #0 {
+; CHECK-SOFT-LABEL: ui2f:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_ui2f
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: ui2f:
+; CHECK-SOFTFP:         vmov s0, r0
+; CHECK-SOFTFP-NEXT:    vcvt.f32.u32 s0, s0
+; CHECK-SOFTFP-NEXT:    vmov r0, s0
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: ui2f:
+; CHECK-HARDFP-SP:         vmov s0, r0
+; CHECK-HARDFP-SP-NEXT:    vcvt.f32.u32 s0, s0
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = uitofp i32 %a to float
+  ret float %conv
+}
+
+define float @l2f(i64 %a) #0 {
+; CHECK-SOFT-LABEL: l2f:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_l2f
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: l2f:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_l2f
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: l2f:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_l2f
+; CHECK-HARDFP-SP-NEXT:    vmov s0, r0
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = sitofp i64 %a to float
+  ret float %conv
+}
+
+define float @ul2f(i64 %a) #0 {
+; CHECK-SOFT-LABEL: ul2f:
+; CHECK-SOFT:         .save {r11, lr}
+; CHECK-SOFT-NEXT:    push {r11, lr}
+; CHECK-SOFT-NEXT:    bl __aeabi_ul2f
+; CHECK-SOFT-NEXT:    pop {r11, lr}
+; CHECK-SOFT-NEXT:    mov pc, lr
+;
+; CHECK-SOFTFP-LABEL: ul2f:
+; CHECK-SOFTFP:         .save {r11, lr}
+; CHECK-SOFTFP-NEXT:    push {r11, lr}
+; CHECK-SOFTFP-NEXT:    bl __aeabi_ul2f
+; CHECK-SOFTFP-NEXT:    pop {r11, lr}
+; CHECK-SOFTFP-NEXT:    mov pc, lr
+;
+; CHECK-HARDFP-SP-LABEL: ul2f:
+; CHECK-HARDFP-SP:         .save {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    push {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    bl __aeabi_ul2f
+; CHECK-HARDFP-SP-NEXT:    vmov s0, r0
+; CHECK-HARDFP-SP-NEXT:    pop {r11, lr}
+; CHECK-HARDFP-SP-NEXT:    mov pc, lr
+entry:
+  %conv = uitofp i64 %a to float
+  ret float %conv
+}
+attributes #0 = { nounwind }