Home | History | Annotate | Download | only in CodeGen
      1 // RUN: %clang_cc1 -O1 -triple arm64-apple-ios7 -target-feature +neon -ffreestanding -S -o - -emit-llvm %s | FileCheck %s
      2 // RUN: %clang_cc1 -O1 -triple arm64-apple-ios7 -target-feature +neon -ffreestanding -S -o - %s | FileCheck -check-prefix=CHECK-CODEGEN %s
      3 // REQUIRES: aarch64-registered-target
      4 // Test ARM64 SIMD max/min intrinsics
      5 
      6 #include <arm_neon.h>
      7 
      8 // Test a represntative sample of 8 and 16, signed and unsigned, 64 and 128 bit reduction
      9 int8_t test_vmaxv_s8(int8x8_t a1) {
     10   // CHECK-LABEL: define i8 @test_vmaxv_s8(
     11   return vmaxv_s8(a1);
     12   // CHECK: call i32 @llvm.aarch64.neon.smaxv.i32.v8i8(
     13 }
     14 
     15 uint16_t test_vminvq_u16(uint16x8_t a1) {
     16   // CHECK-LABEL: define i16 @test_vminvq_u16(
     17   return vminvq_u16(a1);
     18   // CHECK: call i32 @llvm.aarch64.neon.uminv.i32.v8i16(
     19 }
     20 
     21 // Test a represntative sample of 8 and 16, signed and unsigned, 64 and 128 bit pairwise
     22 uint8x8_t test_vmin_u8(uint8x8_t a1, uint8x8_t a2) {
     23   // CHECK-LABEL: define <8 x i8> @test_vmin_u8(
     24   return vmin_u8(a1, a2);
     25   // CHECK: call <8 x i8> @llvm.aarch64.neon.umin.v8i8(
     26 }
     27 
     28 uint8x16_t test_vminq_u8(uint8x16_t a1, uint8x16_t a2) {
     29   // CHECK-LABEL: define <16 x i8> @test_vminq_u8(
     30   return vminq_u8(a1, a2);
     31   // CHECK: call <16 x i8> @llvm.aarch64.neon.umin.v16i8(
     32 }
     33 
     34 int16x8_t test_vmaxq_s16(int16x8_t a1, int16x8_t a2) {
     35   // CHECK-LABEL: define <8 x i16> @test_vmaxq_s16(
     36   return vmaxq_s16(a1, a2);
     37   // CHECK: call <8 x i16> @llvm.aarch64.neon.smax.v8i16(
     38 }
     39 
     40 // Test the more complicated cases of [suf]32 and f64
     41 float64x2_t test_vmaxq_f64(float64x2_t a1, float64x2_t a2) {
     42   // CHECK-LABEL: define <2 x double> @test_vmaxq_f64(
     43   return vmaxq_f64(a1, a2);
     44   // CHECK: call <2 x double> @llvm.aarch64.neon.fmax.v2f64(
     45 }
     46 
     47 float32x4_t test_vmaxq_f32(float32x4_t a1, float32x4_t a2) {
     48   // CHECK-LABEL: define <4 x float> @test_vmaxq_f32(
     49   return vmaxq_f32(a1, a2);
     50   // CHECK: call <4 x float> @llvm.aarch64.neon.fmax.v4f32(
     51 }
     52 
     53 float64x2_t test_vminq_f64(float64x2_t a1, float64x2_t a2) {
     54   // CHECK-LABEL: define <2 x double> @test_vminq_f64(
     55   return vminq_f64(a1, a2);
     56   // CHECK: call <2 x double> @llvm.aarch64.neon.fmin.v2f64(
     57 }
     58 
     59 float32x2_t test_vmax_f32(float32x2_t a1, float32x2_t a2) {
     60   // CHECK-LABEL: define <2 x float> @test_vmax_f32(
     61   return vmax_f32(a1, a2);
     62   // CHECK: call <2 x float> @llvm.aarch64.neon.fmax.v2f32(
     63 }
     64 
     65 int32x2_t test_vmax_s32(int32x2_t a1, int32x2_t a2) {
     66   // CHECK-LABEL: define <2 x i32> @test_vmax_s32(
     67   return vmax_s32(a1, a2);
     68   // CHECK: call <2 x i32> @llvm.aarch64.neon.smax.v2i32(
     69 }
     70 
     71 uint32x2_t test_vmin_u32(uint32x2_t a1, uint32x2_t a2) {
     72   // CHECK-LABEL: define <2 x i32> @test_vmin_u32(
     73   return vmin_u32(a1, a2);
     74   // CHECK: call <2 x i32> @llvm.aarch64.neon.umin.v2i32(
     75 }
     76 
     77 float32_t test_vmaxnmv_f32(float32x2_t a1) {
     78   // CHECK-LABEL: define float @test_vmaxnmv_f32(
     79   return vmaxnmv_f32(a1);
     80   // CHECK: llvm.aarch64.neon.fmaxnmv.f32.v2f32
     81   // CHECK-NEXT: ret
     82 }
     83 
     84 // this doesn't translate into a valid instruction, regardless of what the
     85 // ARM doc says.
     86 #if 0
     87 float64_t test_vmaxnmvq_f64(float64x2_t a1) {
     88   // CHECK@ test_vmaxnmvq_f64
     89   return vmaxnmvq_f64(a1);
     90   // CHECK@ llvm.aarch64.neon.saddlv.i64.v2i32
     91   // CHECK-NEXT@ ret
     92 }
     93 #endif
     94 
     95 float32_t test_vmaxnmvq_f32(float32x4_t a1) {
     96   // CHECK-LABEL: define float @test_vmaxnmvq_f32(
     97   return vmaxnmvq_f32(a1);
     98   // CHECK: call float @llvm.aarch64.neon.fmaxnmv.f32.v4f32(
     99   // CHECK-NEXT: ret
    100 }
    101 
    102 float32_t test_vmaxv_f32(float32x2_t a1) {
    103   // CHECK-LABEL: define float @test_vmaxv_f32(
    104   return vmaxv_f32(a1);
    105   // CHECK: call float @llvm.aarch64.neon.fmaxv.f32.v2f32(
    106   // FIXME check that the 2nd and 3rd arguments are the same V register below
    107   // CHECK-CODEGEN: fmaxp.2s
    108   // CHECK-NEXT: ret
    109 }
    110 
    111 int32_t test_vmaxv_s32(int32x2_t a1) {
    112   // CHECK-LABEL: define i32 @test_vmaxv_s32(
    113   return vmaxv_s32(a1);
    114   // CHECK: call i32 @llvm.aarch64.neon.smaxv.i32.v2i32(
    115   // FIXME check that the 2nd and 3rd arguments are the same V register below
    116   // CHECK-CODEGEN: smaxp.2s
    117   // CHECK-NEXT: ret
    118 }
    119 
    120 uint32_t test_vmaxv_u32(uint32x2_t a1) {
    121   // CHECK-LABEL: define i32 @test_vmaxv_u32(
    122   return vmaxv_u32(a1);
    123   // CHECK: call i32 @llvm.aarch64.neon.umaxv.i32.v2i32(
    124   // FIXME check that the 2nd and 3rd arguments are the same V register below
    125   // CHECK-CODEGEN: umaxp.2s
    126   // CHECK-NEXT: ret
    127 }
    128 
    129 // FIXME punt on this for now; don't forget to fix CHECKs
    130 #if 0
    131 float64_t test_vmaxvq_f64(float64x2_t a1) {
    132   // CHECK@ test_vmaxvq_f64
    133   return vmaxvq_f64(a1);
    134   // CHECK@ llvm.aarch64.neon.fmaxv.i64.v2f64
    135   // CHECK-NEXT@ ret
    136 }
    137 #endif
    138 
    139 float32_t test_vmaxvq_f32(float32x4_t a1) {
    140   // CHECK-LABEL: define float @test_vmaxvq_f32(
    141   return vmaxvq_f32(a1);
    142   // CHECK: call float @llvm.aarch64.neon.fmaxv.f32.v4f32(
    143   // CHECK-NEXT: ret
    144 }
    145 
    146 float32_t test_vminnmv_f32(float32x2_t a1) {
    147   // CHECK-LABEL: define float @test_vminnmv_f32(
    148   return vminnmv_f32(a1);
    149   // CHECK: call float @llvm.aarch64.neon.fminnmv.f32.v2f32(
    150   // CHECK-NEXT: ret
    151 }
    152 
    153 float32_t test_vminvq_f32(float32x4_t a1) {
    154   // CHECK-LABEL: define float @test_vminvq_f32(
    155   return vminvq_f32(a1);
    156   // CHECK: call float @llvm.aarch64.neon.fminv.f32.v4f32(
    157   // CHECK-NEXT: ret
    158 }
    159 
    160 // this doesn't translate into a valid instruction, regardless of what the ARM
    161 // doc says.
    162 #if 0
    163 float64_t test_vminnmvq_f64(float64x2_t a1) {
    164   // CHECK@ test_vminnmvq_f64
    165   return vminnmvq_f64(a1);
    166   // CHECK@ llvm.aarch64.neon.saddlv.i64.v2i32
    167   // CHECK-NEXT@ ret
    168 }
    169 #endif
    170 
    171 float32_t test_vminnmvq_f32(float32x4_t a1) {
    172   // CHECK-LABEL: define float @test_vminnmvq_f32(
    173   return vminnmvq_f32(a1);
    174   // CHECK: call float @llvm.aarch64.neon.fminnmv.f32.v4f32(
    175   // CHECK-NEXT: ret
    176 }
    177 
    178 float32_t test_vminv_f32(float32x2_t a1) {
    179   // CHECK-LABEL: define float @test_vminv_f32(
    180   return vminv_f32(a1);
    181   // CHECK: call float @llvm.aarch64.neon.fminv.f32.v2f32(
    182   // CHECK-NEXT: ret
    183 }
    184 
    185 int32_t test_vminv_s32(int32x2_t a1) {
    186   // CHECK-LABEL: define i32 @test_vminv_s32(
    187   return vminv_s32(a1);
    188   // CHECK: call i32 @llvm.aarch64.neon.sminv.i32.v2i32(
    189   // CHECK-CODEGEN: sminp.2s
    190   // CHECK-NEXT: ret
    191 }
    192 
    193 uint32_t test_vminv_u32(uint32x2_t a1) {
    194   // CHECK-LABEL: define i32 @test_vminv_u32(
    195   return vminv_u32(a1);
    196   // CHECK: call i32 @llvm.aarch64.neon.uminv.i32.v2i32(
    197 }
    198 
    199 // FIXME punt on this for now; don't forget to fix CHECKs
    200 #if 0
    201 float64_t test_vminvq_f64(float64x2_t a1) {
    202   // CHECK@ test_vminvq_f64
    203   return vminvq_f64(a1);
    204   // CHECK@ llvm.aarch64.neon.saddlv.i64.v2i32
    205   // CHECK-NEXT@ ret
    206 }
    207 #endif
    208