The original commit failed to handle "shift assign" (<<=), which
broke the test mentioned in r228406. This is now fixed and the
test added to the lit tests under SemaOpenCL.
*** Original commit message from r228382 ***
OpenCL: handle shift operator with vector operands
Introduce a number of checks:
1. If LHS is a scalar, then RHS cannot be a vector.
2. Operands must be of integer type.
3. If both are vectors, then the number of elements must match.
Relax the requirement for "usual arithmetic conversions":
When LHS is a vector, a scalar RHS can simply be expanded into a
vector; OpenCL does not require that its rank be lower than the LHS.
For example, the following code is not an error even if the implicit
type of the constant literal is "int".
char2 foo(char2 v) { return v << 1; }
Consolidate existing tests under CodeGenOpenCL, and add more tests
under SemaOpenCL.
llvm-svn: 230464
74 lines
2.3 KiB
Common Lisp
74 lines
2.3 KiB
Common Lisp
// RUN: %clang_cc1 -x cl -O1 -emit-llvm %s -o - -triple x86_64-linux-gnu | FileCheck %s -check-prefix=OPT
|
|
// RUN: %clang_cc1 -x cl -O0 -emit-llvm %s -o - -triple x86_64-linux-gnu | FileCheck %s -check-prefix=NOOPT
|
|
|
|
// OpenCL essentially reduces all shift amounts to the last word-size
|
|
// bits before evaluating. Test this both for variables and constants
|
|
// evaluated in the front-end.
|
|
|
|
// OPT: @gtest1 = constant i64 2147483648
|
|
__constant const unsigned long gtest1 = 1UL << 31;
|
|
|
|
// NOOPT: @negativeShift32
|
|
int negativeShift32(int a,int b) {
|
|
// NOOPT: %array0 = alloca [256 x i8]
|
|
char array0[((int)1)<<40];
|
|
// NOOPT: %array1 = alloca [256 x i8]
|
|
char array1[((int)1)<<(-24)];
|
|
|
|
// NOOPT: ret i32 65536
|
|
return ((int)1)<<(-16);
|
|
}
|
|
|
|
//OPT: @positiveShift32
|
|
int positiveShift32(int a,int b) {
|
|
//OPT: [[M32:%.+]] = and i32 %b, 31
|
|
//OPT-NEXT: [[C32:%.+]] = shl i32 %a, [[M32]]
|
|
int c = a<<b;
|
|
int d = ((int)1)<<33;
|
|
//OPT-NEXT: [[E32:%.+]] = add nsw i32 [[C32]], 2
|
|
int e = c + d;
|
|
//OPT-NEXT: ret i32 [[E32]]
|
|
return e;
|
|
}
|
|
|
|
//OPT: @positiveShift64
|
|
long positiveShift64(long a,long b) {
|
|
//OPT: [[M64:%.+]] = and i64 %b, 63
|
|
//OPT-NEXT: [[C64:%.+]] = ashr i64 %a, [[M64]]
|
|
long c = a>>b;
|
|
long d = ((long)8)>>65;
|
|
//OPT-NEXT: [[E64:%.+]] = add nsw i64 [[C64]], 4
|
|
long e = c + d;
|
|
//OPT-NEXT: ret i64 [[E64]]
|
|
return e;
|
|
}
|
|
|
|
typedef __attribute__((ext_vector_type(4))) int int4;
|
|
|
|
//OPT: @vectorVectorTest
|
|
int4 vectorVectorTest(int4 a,int4 b) {
|
|
//OPT: [[VM:%.+]] = and <4 x i32> %b, <i32 31, i32 31, i32 31, i32 31>
|
|
//OPT-NEXT: [[VC:%.+]] = shl <4 x i32> %a, [[VM]]
|
|
int4 c = a << b;
|
|
//OPT-NEXT: [[VF:%.+]] = add <4 x i32> [[VC]], <i32 2, i32 4, i32 16, i32 8>
|
|
int4 d = {1, 1, 1, 1};
|
|
int4 e = {33, 34, -28, -29};
|
|
int4 f = c + (d << e);
|
|
//OPT-NEXT: ret <4 x i32> [[VF]]
|
|
return f;
|
|
}
|
|
|
|
//OPT: @vectorScalarTest
|
|
int4 vectorScalarTest(int4 a,int b) {
|
|
//OPT: [[SP0:%.+]] = insertelement <4 x i32> undef, i32 %b, i32 0
|
|
//OPT: [[SP1:%.+]] = shufflevector <4 x i32> [[SP0]], <4 x i32> undef, <4 x i32> zeroinitializer
|
|
//OPT: [[VSM:%.+]] = and <4 x i32> [[SP1]], <i32 31, i32 31, i32 31, i32 31>
|
|
//OPT-NEXT: [[VSC:%.+]] = shl <4 x i32> %a, [[VSM]]
|
|
int4 c = a << b;
|
|
//OPT-NEXT: [[VSF:%.+]] = add <4 x i32> [[VSC]], <i32 4, i32 4, i32 4, i32 4>
|
|
int4 d = {1, 1, 1, 1};
|
|
int4 f = c + (d << 34);
|
|
//OPT-NEXT: ret <4 x i32> [[VSF]]
|
|
return f;
|
|
}
|