[FileCheck] Support comment directives
Sometimes you want to disable a FileCheck directive without removing
it entirely, or you want to write comments that mention a directive by
name. The `COM:` directive makes it easy to do this. For example,
you might have:
```
; X32: pinsrd_1:
; X32: pinsrd $1, 4(%esp), %xmm0
; COM: FIXME: X64 isn't working correctly yet for this part of codegen, but
; COM: X64 will have something similar to X32:
; COM:
; COM: X64: pinsrd_1:
; COM: X64: pinsrd $1, %edi, %xmm0
```
Without this patch, you need to use some combination of rewording and
directive syntax mangling to prevent FileCheck from recognizing the
commented occurrences of `X32:` and `X64:` above as directives.
Moreover, FileCheck diagnostics have been proposed that might complain
about the occurrences of `X64` that don't have the trailing `:`
because they look like directive typos:
<http://lists.llvm.org/pipermail/llvm-dev/2020-April/140610.html>
I think dodging all these problems can prove tedious for test authors,
and directive syntax mangling already makes the purpose of existing
test code unclear. `COM:` can avoid all these problems.
This patch also updates the small set of existing tests that define
`COM` as a check prefix:
- clang/test/CodeGen/default-address-space.c
- clang/test/CodeGenOpenCL/addr-space-struct-arg.cl
- clang/test/Driver/hip-device-libs.hip
- llvm/test/Assembler/drop-debug-info-nonzero-alloca.ll
I think lit should support `COM:` as well. Perhaps `clang -verify`
should too.
Reviewed By: jhenderson, thopre
Differential Revision: https://reviews.llvm.org/D79276
2020-05-05 06:05:55 +08:00
|
|
|
// RUN: %clang_cc1 %s -emit-llvm -o - -O0 -ffake-address-space-map -triple i686-pc-darwin | FileCheck -enable-var-scope -check-prefixes=ALL,X86 %s
|
|
|
|
// RUN: %clang_cc1 %s -emit-llvm -o - -O0 -triple amdgcn | FileCheck -enable-var-scope -check-prefixes=ALL,AMDGCN %s
|
|
|
|
// RUN: %clang_cc1 %s -emit-llvm -o - -cl-std=CL2.0 -O0 -triple amdgcn | FileCheck -enable-var-scope -check-prefixes=ALL,AMDGCN,AMDGCN20 %s
|
2018-12-10 20:03:00 +08:00
|
|
|
// RUN: %clang_cc1 %s -emit-llvm -o - -cl-std=CL1.2 -O0 -triple spir-unknown-unknown-unknown | FileCheck -enable-var-scope -check-prefixes=SPIR %s
|
|
|
|
|
|
|
|
typedef int int2 __attribute__((ext_vector_type(2)));
|
2014-10-27 20:37:26 +08:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
int cells[9];
|
|
|
|
} Mat3X3;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
int cells[16];
|
|
|
|
} Mat4X4;
|
|
|
|
|
2017-08-10 05:44:58 +08:00
|
|
|
typedef struct {
|
|
|
|
int cells[1024];
|
|
|
|
} Mat32X32;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
int cells[4096];
|
|
|
|
} Mat64X64;
|
|
|
|
|
2017-06-30 02:47:45 +08:00
|
|
|
struct StructOneMember {
|
|
|
|
int2 x;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct StructTwoMember {
|
|
|
|
int2 x;
|
|
|
|
int2 y;
|
|
|
|
};
|
|
|
|
|
2017-08-10 05:44:58 +08:00
|
|
|
struct LargeStructOneMember {
|
|
|
|
int2 x[100];
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LargeStructTwoMember {
|
|
|
|
int2 x[40];
|
|
|
|
int2 y[20];
|
|
|
|
};
|
|
|
|
|
2018-03-15 23:25:19 +08:00
|
|
|
#if __OPENCL_C_VERSION__ >= 200
|
|
|
|
struct LargeStructOneMember g_s;
|
|
|
|
#endif
|
2017-08-10 05:44:58 +08:00
|
|
|
|
2020-03-25 00:36:19 +08:00
|
|
|
// X86-LABEL: define void @foo(%struct.Mat4X4* noalias sret align 4 %agg.result, %struct.Mat3X3* byval(%struct.Mat3X3) align 4 %in)
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define %struct.Mat4X4 @foo([9 x i32] %in.coerce)
|
2014-10-27 20:37:26 +08:00
|
|
|
Mat4X4 __attribute__((noinline)) foo(Mat3X3 in) {
|
|
|
|
Mat4X4 out;
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
[FileCheck] Support comment directives
Sometimes you want to disable a FileCheck directive without removing
it entirely, or you want to write comments that mention a directive by
name. The `COM:` directive makes it easy to do this. For example,
you might have:
```
; X32: pinsrd_1:
; X32: pinsrd $1, 4(%esp), %xmm0
; COM: FIXME: X64 isn't working correctly yet for this part of codegen, but
; COM: X64 will have something similar to X32:
; COM:
; COM: X64: pinsrd_1:
; COM: X64: pinsrd $1, %edi, %xmm0
```
Without this patch, you need to use some combination of rewording and
directive syntax mangling to prevent FileCheck from recognizing the
commented occurrences of `X32:` and `X64:` above as directives.
Moreover, FileCheck diagnostics have been proposed that might complain
about the occurrences of `X64` that don't have the trailing `:`
because they look like directive typos:
<http://lists.llvm.org/pipermail/llvm-dev/2020-April/140610.html>
I think dodging all these problems can prove tedious for test authors,
and directive syntax mangling already makes the purpose of existing
test code unclear. `COM:` can avoid all these problems.
This patch also updates the small set of existing tests that define
`COM` as a check prefix:
- clang/test/CodeGen/default-address-space.c
- clang/test/CodeGenOpenCL/addr-space-struct-arg.cl
- clang/test/Driver/hip-device-libs.hip
- llvm/test/Assembler/drop-debug-info-nonzero-alloca.ll
I think lit should support `COM:` as well. Perhaps `clang -verify`
should too.
Reviewed By: jhenderson, thopre
Differential Revision: https://reviews.llvm.org/D79276
2020-05-05 06:05:55 +08:00
|
|
|
// ALL-LABEL: define {{.*}} void @ker
|
2017-06-30 02:47:45 +08:00
|
|
|
// Expect two mem copies: one for the argument "in", and one for
|
|
|
|
// the return value.
|
|
|
|
// X86: call void @llvm.memcpy.p0i8.p1i8.i32(i8*
|
|
|
|
// X86: call void @llvm.memcpy.p1i8.p0i8.i32(i8 addrspace(1)*
|
2017-08-10 05:44:58 +08:00
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN: load [9 x i32], [9 x i32] addrspace(1)*
|
|
|
|
// AMDGCN: call %struct.Mat4X4 @foo([9 x i32]
|
|
|
|
// AMDGCN: call void @llvm.memcpy.p1i8.p5i8.i64(i8 addrspace(1)*
|
2014-10-27 20:37:26 +08:00
|
|
|
kernel void ker(global Mat3X3 *in, global Mat4X4 *out) {
|
|
|
|
out[0] = foo(in[1]);
|
|
|
|
}
|
|
|
|
|
2020-03-25 00:36:19 +08:00
|
|
|
// X86-LABEL: define void @foo_large(%struct.Mat64X64* noalias sret align 4 %agg.result, %struct.Mat32X32* byval(%struct.Mat32X32) align 4 %in)
|
|
|
|
// AMDGCN-LABEL: define void @foo_large(%struct.Mat64X64 addrspace(5)* noalias sret align 4 %agg.result, %struct.Mat32X32 addrspace(5)* byval(%struct.Mat32X32) align 4 %in)
|
2017-08-10 05:44:58 +08:00
|
|
|
Mat64X64 __attribute__((noinline)) foo_large(Mat32X32 in) {
|
|
|
|
Mat64X64 out;
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
[FileCheck] Support comment directives
Sometimes you want to disable a FileCheck directive without removing
it entirely, or you want to write comments that mention a directive by
name. The `COM:` directive makes it easy to do this. For example,
you might have:
```
; X32: pinsrd_1:
; X32: pinsrd $1, 4(%esp), %xmm0
; COM: FIXME: X64 isn't working correctly yet for this part of codegen, but
; COM: X64 will have something similar to X32:
; COM:
; COM: X64: pinsrd_1:
; COM: X64: pinsrd $1, %edi, %xmm0
```
Without this patch, you need to use some combination of rewording and
directive syntax mangling to prevent FileCheck from recognizing the
commented occurrences of `X32:` and `X64:` above as directives.
Moreover, FileCheck diagnostics have been proposed that might complain
about the occurrences of `X64` that don't have the trailing `:`
because they look like directive typos:
<http://lists.llvm.org/pipermail/llvm-dev/2020-April/140610.html>
I think dodging all these problems can prove tedious for test authors,
and directive syntax mangling already makes the purpose of existing
test code unclear. `COM:` can avoid all these problems.
This patch also updates the small set of existing tests that define
`COM` as a check prefix:
- clang/test/CodeGen/default-address-space.c
- clang/test/CodeGenOpenCL/addr-space-struct-arg.cl
- clang/test/Driver/hip-device-libs.hip
- llvm/test/Assembler/drop-debug-info-nonzero-alloca.ll
I think lit should support `COM:` as well. Perhaps `clang -verify`
should too.
Reviewed By: jhenderson, thopre
Differential Revision: https://reviews.llvm.org/D79276
2020-05-05 06:05:55 +08:00
|
|
|
// ALL-LABEL: define {{.*}} void @ker_large
|
2017-08-10 05:44:58 +08:00
|
|
|
// Expect two mem copies: one for the argument "in", and one for
|
|
|
|
// the return value.
|
|
|
|
// X86: call void @llvm.memcpy.p0i8.p1i8.i32(i8*
|
|
|
|
// X86: call void @llvm.memcpy.p1i8.p0i8.i32(i8 addrspace(1)*
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN: call void @llvm.memcpy.p5i8.p1i8.i64(i8 addrspace(5)*
|
|
|
|
// AMDGCN: call void @llvm.memcpy.p1i8.p5i8.i64(i8 addrspace(1)*
|
2017-08-10 05:44:58 +08:00
|
|
|
kernel void ker_large(global Mat32X32 *in, global Mat64X64 *out) {
|
|
|
|
out[0] = foo_large(in[1]);
|
|
|
|
}
|
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define void @FuncOneMember(<2 x i32> %u.coerce)
|
2017-06-30 02:47:45 +08:00
|
|
|
void FuncOneMember(struct StructOneMember u) {
|
|
|
|
u.x = (int2)(0, 0);
|
|
|
|
}
|
|
|
|
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN-LABEL: define void @FuncOneLargeMember(%struct.LargeStructOneMember addrspace(5)* byval(%struct.LargeStructOneMember) align 8 %u)
|
2018-03-15 23:25:19 +08:00
|
|
|
// AMDGCN-NOT: addrspacecast
|
|
|
|
// AMDGCN: store <2 x i32> %{{.*}}, <2 x i32> addrspace(5)*
|
2017-08-10 05:44:58 +08:00
|
|
|
void FuncOneLargeMember(struct LargeStructOneMember u) {
|
|
|
|
u.x[0] = (int2)(0, 0);
|
|
|
|
}
|
|
|
|
|
2018-03-15 23:25:19 +08:00
|
|
|
// AMDGCN20-LABEL: define void @test_indirect_arg_globl()
|
|
|
|
// AMDGCN20: %[[byval_temp:.*]] = alloca %struct.LargeStructOneMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN20: %[[r0:.*]] = bitcast %struct.LargeStructOneMember addrspace(5)* %[[byval_temp]] to i8 addrspace(5)*
|
|
|
|
// AMDGCN20: call void @llvm.memcpy.p5i8.p1i8.i64(i8 addrspace(5)* align 8 %[[r0]], i8 addrspace(1)* align 8 bitcast (%struct.LargeStructOneMember addrspace(1)* @g_s to i8 addrspace(1)*), i64 800, i1 false)
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN20: call void @FuncOneLargeMember(%struct.LargeStructOneMember addrspace(5)* byval(%struct.LargeStructOneMember) align 8 %[[byval_temp]])
|
2018-03-15 23:25:19 +08:00
|
|
|
#if __OPENCL_C_VERSION__ >= 200
|
|
|
|
void test_indirect_arg_globl(void) {
|
|
|
|
FuncOneLargeMember(g_s);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// AMDGCN-LABEL: define amdgpu_kernel void @test_indirect_arg_local()
|
|
|
|
// AMDGCN: %[[byval_temp:.*]] = alloca %struct.LargeStructOneMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN: %[[r0:.*]] = bitcast %struct.LargeStructOneMember addrspace(5)* %[[byval_temp]] to i8 addrspace(5)*
|
|
|
|
// AMDGCN: call void @llvm.memcpy.p5i8.p3i8.i64(i8 addrspace(5)* align 8 %[[r0]], i8 addrspace(3)* align 8 bitcast (%struct.LargeStructOneMember addrspace(3)* @test_indirect_arg_local.l_s to i8 addrspace(3)*), i64 800, i1 false)
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN: call void @FuncOneLargeMember(%struct.LargeStructOneMember addrspace(5)* byval(%struct.LargeStructOneMember) align 8 %[[byval_temp]])
|
2018-03-15 23:25:19 +08:00
|
|
|
kernel void test_indirect_arg_local(void) {
|
|
|
|
local struct LargeStructOneMember l_s;
|
|
|
|
FuncOneLargeMember(l_s);
|
|
|
|
}
|
|
|
|
|
|
|
|
// AMDGCN-LABEL: define void @test_indirect_arg_private()
|
|
|
|
// AMDGCN: %[[p_s:.*]] = alloca %struct.LargeStructOneMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN-NOT: @llvm.memcpy
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN-NEXT: call void @FuncOneLargeMember(%struct.LargeStructOneMember addrspace(5)* byval(%struct.LargeStructOneMember) align 8 %[[p_s]])
|
2018-03-15 23:25:19 +08:00
|
|
|
void test_indirect_arg_private(void) {
|
|
|
|
struct LargeStructOneMember p_s;
|
|
|
|
FuncOneLargeMember(p_s);
|
|
|
|
}
|
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define amdgpu_kernel void @KernelOneMember
|
|
|
|
// AMDGCN-SAME: (<2 x i32> %[[u_coerce:.*]])
|
|
|
|
// AMDGCN: %[[u:.*]] = alloca %struct.StructOneMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN: %[[coerce_dive:.*]] = getelementptr inbounds %struct.StructOneMember, %struct.StructOneMember addrspace(5)* %[[u]], i32 0, i32 0
|
|
|
|
// AMDGCN: store <2 x i32> %[[u_coerce]], <2 x i32> addrspace(5)* %[[coerce_dive]]
|
|
|
|
// AMDGCN: call void @FuncOneMember(<2 x i32>
|
2017-06-30 02:47:45 +08:00
|
|
|
kernel void KernelOneMember(struct StructOneMember u) {
|
|
|
|
FuncOneMember(u);
|
|
|
|
}
|
|
|
|
|
2018-12-10 20:03:00 +08:00
|
|
|
// SPIR: call void @llvm.memcpy.p0i8.p1i8.i32
|
|
|
|
// SPIR-NOT: addrspacecast
|
|
|
|
kernel void KernelOneMemberSpir(global struct StructOneMember* u) {
|
|
|
|
FuncOneMember(*u);
|
|
|
|
}
|
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define amdgpu_kernel void @KernelLargeOneMember(
|
|
|
|
// AMDGCN: %[[U:.*]] = alloca %struct.LargeStructOneMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN: store %struct.LargeStructOneMember %u.coerce, %struct.LargeStructOneMember addrspace(5)* %[[U]], align 8
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN: call void @FuncOneLargeMember(%struct.LargeStructOneMember addrspace(5)* byval(%struct.LargeStructOneMember) align 8 %[[U]])
|
2017-08-10 05:44:58 +08:00
|
|
|
kernel void KernelLargeOneMember(struct LargeStructOneMember u) {
|
|
|
|
FuncOneLargeMember(u);
|
|
|
|
}
|
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define void @FuncTwoMember(<2 x i32> %u.coerce0, <2 x i32> %u.coerce1)
|
2017-06-30 02:47:45 +08:00
|
|
|
void FuncTwoMember(struct StructTwoMember u) {
|
2017-08-10 05:44:58 +08:00
|
|
|
u.y = (int2)(0, 0);
|
|
|
|
}
|
|
|
|
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN-LABEL: define void @FuncLargeTwoMember(%struct.LargeStructTwoMember addrspace(5)* byval(%struct.LargeStructTwoMember) align 8 %u)
|
2017-08-10 05:44:58 +08:00
|
|
|
void FuncLargeTwoMember(struct LargeStructTwoMember u) {
|
|
|
|
u.y[0] = (int2)(0, 0);
|
2017-06-30 02:47:45 +08:00
|
|
|
}
|
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define amdgpu_kernel void @KernelTwoMember
|
|
|
|
// AMDGCN-SAME: (%struct.StructTwoMember %[[u_coerce:.*]])
|
|
|
|
// AMDGCN: %[[u:.*]] = alloca %struct.StructTwoMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN: %[[LD0:.*]] = load <2 x i32>, <2 x i32> addrspace(5)*
|
|
|
|
// AMDGCN: %[[LD1:.*]] = load <2 x i32>, <2 x i32> addrspace(5)*
|
|
|
|
// AMDGCN: call void @FuncTwoMember(<2 x i32> %[[LD0]], <2 x i32> %[[LD1]])
|
2017-06-30 02:47:45 +08:00
|
|
|
kernel void KernelTwoMember(struct StructTwoMember u) {
|
|
|
|
FuncTwoMember(u);
|
|
|
|
}
|
2017-08-10 05:44:58 +08:00
|
|
|
|
2018-02-16 03:12:41 +08:00
|
|
|
// AMDGCN-LABEL: define amdgpu_kernel void @KernelLargeTwoMember
|
|
|
|
// AMDGCN-SAME: (%struct.LargeStructTwoMember %[[u_coerce:.*]])
|
|
|
|
// AMDGCN: %[[u:.*]] = alloca %struct.LargeStructTwoMember, align 8, addrspace(5)
|
|
|
|
// AMDGCN: store %struct.LargeStructTwoMember %[[u_coerce]], %struct.LargeStructTwoMember addrspace(5)* %[[u]]
|
2019-06-06 05:12:14 +08:00
|
|
|
// AMDGCN: call void @FuncLargeTwoMember(%struct.LargeStructTwoMember addrspace(5)* byval(%struct.LargeStructTwoMember) align 8 %[[u]])
|
2017-08-10 05:44:58 +08:00
|
|
|
kernel void KernelLargeTwoMember(struct LargeStructTwoMember u) {
|
|
|
|
FuncLargeTwoMember(u);
|
|
|
|
}
|