Files
clang-p2996/clang/test/CodeGenOpenCL/shifts.cl
Sameer Sahasrabuddhe a75db66eee Restores r228382, which was reverted in r228406.
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
2015-02-25 05:48:23 +00:00

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;
}