Index: clang/lib/CodeGen/CGBuiltin.cpp =================================================================== --- clang/lib/CodeGen/CGBuiltin.cpp +++ clang/lib/CodeGen/CGBuiltin.cpp @@ -7917,6 +7917,19 @@ getContext().GetBuiltinType(BuiltinID, Error, &ICEArguments); assert(Error == ASTContext::GE_None && "Should not codegen an error"); + llvm::Type *Ty = ConvertType(E->getType()); + if (BuiltinID >= SVE::BI__builtin_sve_reinterpret_s8_s8 && + BuiltinID <= SVE::BI__builtin_sve_reinterpret_f64_f64) { + Value *Val = EmitScalarExpr(E->getArg(0)); + // FIXME: For big endian this needs an additional REV, or needs a separate + // intrinsic that is code-generated as a no-op, because the LLVM bitcast + // instruction is defined as 'bitwise' equivalent from memory point of + // view (when storing/reloading), whereas the svreinterpret builtin + // implements bitwise equivalent cast from register point of view. + // LLVM CodeGen for a bitcast must add an explicit REV for big-endian. + return Builder.CreateBitCast(Val, Ty); + } + llvm::SmallVector Ops; for (unsigned i = 0, e = E->getNumArgs(); i != e; i++) { if ((ICEArguments & (1 << i)) == 0) @@ -7939,7 +7952,6 @@ auto *Builtin = findARMVectorIntrinsicInMap(AArch64SVEIntrinsicMap, BuiltinID, AArch64SVEIntrinsicsProvenSorted); SVETypeFlags TypeFlags(Builtin->TypeModifier); - llvm::Type *Ty = ConvertType(E->getType()); if (TypeFlags.isLoad()) return EmitSVEMaskedLoad(E, Ty, Ops, Builtin->LLVMIntrinsic, TypeFlags.isZExtReturn()); Index: clang/test/CodeGen/aarch64-sve-intrinsics/acle_sve_reinterpret.c =================================================================== --- /dev/null +++ clang/test/CodeGen/aarch64-sve-intrinsics/acle_sve_reinterpret.c @@ -0,0 +1,960 @@ +// RUN: %clang_cc1 -D__ARM_FEATURE_SVE -triple aarch64-none-linux-gnu -target-feature +sve -fallow-half-arguments-and-returns -S -O1 -Werror -Wall -emit-llvm -o - %s | FileCheck %s +// RUN: %clang_cc1 -D__ARM_FEATURE_SVE -DSVE_OVERLOADED_FORMS -triple aarch64-none-linux-gnu -target-feature +sve -fallow-half-arguments-and-returns -S -O1 -Werror -Wall -emit-llvm -o - %s | FileCheck %s + +#include + +#ifdef SVE_OVERLOADED_FORMS +// A simple used,unused... macro, long enough to represent any SVE builtin. +#define SVE_ACLE_FUNC(A1,A2_UNUSED,A3,A4_UNUSED) A1##A3 +#else +#define SVE_ACLE_FUNC(A1,A2,A3,A4) A1##A2##A3##A4 +#endif + +svint8_t test_svreinterpret_s8_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_s8 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s8,_s8,,)(op); +} + +svint8_t test_svreinterpret_s8_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_s16,,)(op); +} + +svint8_t test_svreinterpret_s8_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_s32,,)(op); +} + +svint8_t test_svreinterpret_s8_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_s64,,)(op); +} + +svint8_t test_svreinterpret_s8_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_u8 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s8,_u8,,)(op); +} + +svint8_t test_svreinterpret_s8_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_u16,,)(op); +} + +svint8_t test_svreinterpret_s8_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_u32,,)(op); +} + +svint8_t test_svreinterpret_s8_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_u64,,)(op); +} + +svint8_t test_svreinterpret_s8_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_f16,,)(op); +} + +svint8_t test_svreinterpret_s8_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_f32,,)(op); +} + +svint8_t test_svreinterpret_s8_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s8_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s8,_f64,,)(op); +} + +svint16_t test_svreinterpret_s16_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_s8,,)(op); +} + +svint16_t test_svreinterpret_s16_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_s16 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s16,_s16,,)(op); +} + +svint16_t test_svreinterpret_s16_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_s32,,)(op); +} + +svint16_t test_svreinterpret_s16_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_s64,,)(op); +} + +svint16_t test_svreinterpret_s16_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_u8,,)(op); +} + +svint16_t test_svreinterpret_s16_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_u16 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s16,_u16,,)(op); +} + +svint16_t test_svreinterpret_s16_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_u32,,)(op); +} + +svint16_t test_svreinterpret_s16_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_u64,,)(op); +} + +svint16_t test_svreinterpret_s16_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_f16,,)(op); +} + +svint16_t test_svreinterpret_s16_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_f32,,)(op); +} + +svint16_t test_svreinterpret_s16_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s16_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s16,_f64,,)(op); +} + +svint32_t test_svreinterpret_s32_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_s8,,)(op); +} + +svint32_t test_svreinterpret_s32_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_s16,,)(op); +} + +svint32_t test_svreinterpret_s32_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_s32 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s32,_s32,,)(op); +} + +svint32_t test_svreinterpret_s32_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_s64,,)(op); +} + +svint32_t test_svreinterpret_s32_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_u8,,)(op); +} + +svint32_t test_svreinterpret_s32_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_u16,,)(op); +} + +svint32_t test_svreinterpret_s32_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_u32 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s32,_u32,,)(op); +} + +svint32_t test_svreinterpret_s32_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_u64,,)(op); +} + +svint32_t test_svreinterpret_s32_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_f16,,)(op); +} + +svint32_t test_svreinterpret_s32_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_f32,,)(op); +} + +svint32_t test_svreinterpret_s32_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s32_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s32,_f64,,)(op); +} + +svint64_t test_svreinterpret_s64_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_s8,,)(op); +} + +svint64_t test_svreinterpret_s64_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_s16,,)(op); +} + +svint64_t test_svreinterpret_s64_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_s32,,)(op); +} + +svint64_t test_svreinterpret_s64_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_s64 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s64,_s64,,)(op); +} + +svint64_t test_svreinterpret_s64_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_u8,,)(op); +} + +svint64_t test_svreinterpret_s64_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_u16,,)(op); +} + +svint64_t test_svreinterpret_s64_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_u32,,)(op); +} + +svint64_t test_svreinterpret_s64_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_u64 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_s64,_u64,,)(op); +} + +svint64_t test_svreinterpret_s64_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_f16,,)(op); +} + +svint64_t test_svreinterpret_s64_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_f32,,)(op); +} + +svint64_t test_svreinterpret_s64_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_s64_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_s64,_f64,,)(op); +} + +svuint8_t test_svreinterpret_u8_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_s8 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u8,_s8,,)(op); +} + +svuint8_t test_svreinterpret_u8_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_s16,,)(op); +} + +svuint8_t test_svreinterpret_u8_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_s32,,)(op); +} + +svuint8_t test_svreinterpret_u8_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_s64,,)(op); +} + +svuint8_t test_svreinterpret_u8_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_u8 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u8,_u8,,)(op); +} + +svuint8_t test_svreinterpret_u8_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_u16,,)(op); +} + +svuint8_t test_svreinterpret_u8_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_u32,,)(op); +} + +svuint8_t test_svreinterpret_u8_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_u64,,)(op); +} + +svuint8_t test_svreinterpret_u8_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_f16,,)(op); +} + +svuint8_t test_svreinterpret_u8_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_f32,,)(op); +} + +svuint8_t test_svreinterpret_u8_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u8_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u8,_f64,,)(op); +} + +svuint16_t test_svreinterpret_u16_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_s8,,)(op); +} + +svuint16_t test_svreinterpret_u16_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_s16 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u16,_s16,,)(op); +} + +svuint16_t test_svreinterpret_u16_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_s32,,)(op); +} + +svuint16_t test_svreinterpret_u16_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_s64,,)(op); +} + +svuint16_t test_svreinterpret_u16_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_u8,,)(op); +} + +svuint16_t test_svreinterpret_u16_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_u16 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u16,_u16,,)(op); +} + +svuint16_t test_svreinterpret_u16_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_u32,,)(op); +} + +svuint16_t test_svreinterpret_u16_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_u64,,)(op); +} + +svuint16_t test_svreinterpret_u16_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_f16,,)(op); +} + +svuint16_t test_svreinterpret_u16_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_f32,,)(op); +} + +svuint16_t test_svreinterpret_u16_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u16_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u16,_f64,,)(op); +} + +svuint32_t test_svreinterpret_u32_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_s8,,)(op); +} + +svuint32_t test_svreinterpret_u32_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_s16,,)(op); +} + +svuint32_t test_svreinterpret_u32_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_s32 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u32,_s32,,)(op); +} + +svuint32_t test_svreinterpret_u32_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_s64,,)(op); +} + +svuint32_t test_svreinterpret_u32_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_u8,,)(op); +} + +svuint32_t test_svreinterpret_u32_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_u16,,)(op); +} + +svuint32_t test_svreinterpret_u32_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_u32 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u32,_u32,,)(op); +} + +svuint32_t test_svreinterpret_u32_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_u64,,)(op); +} + +svuint32_t test_svreinterpret_u32_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_f16,,)(op); +} + +svuint32_t test_svreinterpret_u32_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_f32,,)(op); +} + +svuint32_t test_svreinterpret_u32_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u32_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u32,_f64,,)(op); +} + +svuint64_t test_svreinterpret_u64_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_s8,,)(op); +} + +svuint64_t test_svreinterpret_u64_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_s16,,)(op); +} + +svuint64_t test_svreinterpret_u64_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_s32,,)(op); +} + +svuint64_t test_svreinterpret_u64_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_s64 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u64,_s64,,)(op); +} + +svuint64_t test_svreinterpret_u64_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_u8,,)(op); +} + +svuint64_t test_svreinterpret_u64_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_u16,,)(op); +} + +svuint64_t test_svreinterpret_u64_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_u32,,)(op); +} + +svuint64_t test_svreinterpret_u64_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_u64 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_u64,_u64,,)(op); +} + +svuint64_t test_svreinterpret_u64_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_f16,,)(op); +} + +svuint64_t test_svreinterpret_u64_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_f32,,)(op); +} + +svuint64_t test_svreinterpret_u64_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_u64_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_u64,_f64,,)(op); +} + +svfloat16_t test_svreinterpret_f16_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_s8,,)(op); +} + +svfloat16_t test_svreinterpret_f16_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_s16,,)(op); +} + +svfloat16_t test_svreinterpret_f16_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_s32,,)(op); +} + +svfloat16_t test_svreinterpret_f16_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_s64,,)(op); +} + +svfloat16_t test_svreinterpret_f16_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_u8,,)(op); +} + +svfloat16_t test_svreinterpret_f16_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_u16,,)(op); +} + +svfloat16_t test_svreinterpret_f16_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_u32,,)(op); +} + +svfloat16_t test_svreinterpret_f16_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_u64,,)(op); +} + +svfloat16_t test_svreinterpret_f16_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_f16 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_f16,_f16,,)(op); +} + +svfloat16_t test_svreinterpret_f16_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_f32,,)(op); +} + +svfloat16_t test_svreinterpret_f16_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f16_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f16,_f64,,)(op); +} + +svfloat32_t test_svreinterpret_f32_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_s8,,)(op); +} + +svfloat32_t test_svreinterpret_f32_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_s16,,)(op); +} + +svfloat32_t test_svreinterpret_f32_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_s32,,)(op); +} + +svfloat32_t test_svreinterpret_f32_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_s64,,)(op); +} + +svfloat32_t test_svreinterpret_f32_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_u8,,)(op); +} + +svfloat32_t test_svreinterpret_f32_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_u16,,)(op); +} + +svfloat32_t test_svreinterpret_f32_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_u32,,)(op); +} + +svfloat32_t test_svreinterpret_f32_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_u64,,)(op); +} + +svfloat32_t test_svreinterpret_f32_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_f16,,)(op); +} + +svfloat32_t test_svreinterpret_f32_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_f32 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_f32,_f32,,)(op); +} + +svfloat32_t test_svreinterpret_f32_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f32_f64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f32,_f64,,)(op); +} + +svfloat64_t test_svreinterpret_f64_s8(svint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_s8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_s8,,)(op); +} + +svfloat64_t test_svreinterpret_f64_s16(svint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_s16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_s16,,)(op); +} + +svfloat64_t test_svreinterpret_f64_s32(svint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_s32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_s32,,)(op); +} + +svfloat64_t test_svreinterpret_f64_s64(svint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_s64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_s64,,)(op); +} + +svfloat64_t test_svreinterpret_f64_u8(svuint8_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_u8 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_u8,,)(op); +} + +svfloat64_t test_svreinterpret_f64_u16(svuint16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_u16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_u16,,)(op); +} + +svfloat64_t test_svreinterpret_f64_u32(svuint32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_u32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_u32,,)(op); +} + +svfloat64_t test_svreinterpret_f64_u64(svuint64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_u64 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_u64,,)(op); +} + +svfloat64_t test_svreinterpret_f64_f16(svfloat16_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_f16 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_f16,,)(op); +} + +svfloat64_t test_svreinterpret_f64_f32(svfloat32_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_f32 + // CHECK: %[[CAST:.*]] = bitcast %op to + // CHECK: ret %[[CAST]] + return SVE_ACLE_FUNC(svreinterpret_f64,_f32,,)(op); +} + +svfloat64_t test_svreinterpret_f64_f64(svfloat64_t op) +{ + // CHECK-LABEL: test_svreinterpret_f64_f64 + // CHECK: ret %op + return SVE_ACLE_FUNC(svreinterpret_f64,_f64,,)(op); +} Index: clang/test/CodeGen/aarch64-sve-intrinsics/negative/big_endian.c =================================================================== --- /dev/null +++ clang/test/CodeGen/aarch64-sve-intrinsics/negative/big_endian.c @@ -0,0 +1,4 @@ +// RUN: %clang_cc1 -D__ARM_FEATURE_SVE -triple aarch64_be-none-linux-gnu -target-feature +sve -fallow-half-arguments-and-returns -fsyntax-only -verify %s + +// expected-error@* {{Big endian is currently not supported for arm_sve.h}} +#include Index: clang/utils/TableGen/SveEmitter.cpp =================================================================== --- clang/utils/TableGen/SveEmitter.cpp +++ clang/utils/TableGen/SveEmitter.cpp @@ -237,6 +237,23 @@ class SVEEmitter { private: + // The reinterpret builtins are generated separately because they + // need the cross product of all types (121 functions in total), + // which is inconvenient to specify in the arm_sve.td file or + // generate in CGBuiltin.cpp. + struct ReinterpretTypeInfo { + const char *Suffix; + const char *Type; + const char *BuiltinType; + }; + SmallVector Reinterprets = { + {"s8", "svint8_t", "q16Sc"}, {"s16", "svint16_t", "q8Ss"}, + {"s32", "svint32_t", "q4Si"}, {"s64", "svint64_t", "q2SWi"}, + {"u8", "svuint8_t", "q16Uc"}, {"u16", "svuint16_t", "q8Us"}, + {"u32", "svuint32_t", "q4Ui"}, {"u64", "svuint64_t", "q2UWi"}, + {"f16", "svfloat16_t", "q8h"}, {"f32", "svfloat32_t", "q4f"}, + {"f64", "svfloat64_t", "q2d"}}; + RecordKeeper &Records; llvm::StringMap EltTypes; llvm::StringMap MemEltTypes; @@ -1008,6 +1025,10 @@ OS << "#error \"SVE support not enabled\"\n"; OS << "#else\n\n"; + OS << "#if !defined(__LITTLE_ENDIAN__)\n"; + OS << "#error \"Big endian is currently not supported for arm_sve.h\"\n"; + OS << "#endif\n"; + OS << "#include \n\n"; OS << "#ifdef __cplusplus\n"; OS << "extern \"C\" {\n"; @@ -1074,6 +1095,22 @@ OS << "#define __aio static inline __attribute__((__always_inline__, " "__nodebug__, __overloadable__))\n\n"; + // Add reinterpret functions. + for (auto ShortForm : { false, true } ) + for (const ReinterpretTypeInfo &From : Reinterprets) + for (const ReinterpretTypeInfo &To : Reinterprets) { + if (ShortForm) { + OS << "__aio " << From.Type << " svreinterpret_" << From.Suffix; + OS << "(" << To.Type << " op) {\n"; + OS << " return __builtin_sve_reinterpret_" << From.Suffix << "_" + << To.Suffix << "(op);\n"; + OS << "}\n\n"; + } else + OS << "#define svreinterpret_" << From.Suffix << "_" << To.Suffix + << "(...) __builtin_sve_reinterpret_" << From.Suffix << "_" + << To.Suffix << "(__VA_ARGS__)\n"; + } + SmallVector, 128> Defs; std::vector RV = Records.getAllDerivedDefinitions("Inst"); for (auto *R : RV) @@ -1148,8 +1185,16 @@ OS << "BUILTIN(__builtin_sve_" << Def->getMangledName() << ", \"" << Def->getBuiltinTypeStr() << "\", \"n\")\n"; } + + // Add reinterpret builtins + for (const ReinterpretTypeInfo &From : Reinterprets) + for (const ReinterpretTypeInfo &To : Reinterprets) + OS << "BUILTIN(__builtin_sve_reinterpret_" << From.Suffix << "_" + << To.Suffix << +", \"" << From.BuiltinType << To.BuiltinType + << "\", \"n\")\n"; + OS << "#endif\n\n"; -} + } void SVEEmitter::createCodeGenMap(raw_ostream &OS) { std::vector RV = Records.getAllDerivedDefinitions("Inst");