mirror of https://github.com/microsoft/clang.git
[OpenCL] An error shall occur if any scalar operand has greater rank than the type of the vector element
Summary: This is the fix for patch https://reviews.llvm.org/D33353 @uweigand, could you please verify that everything will be good on SystemZ? I added triple spir-unknown-unknown. Thank you in advance! Reviewers: uweigand Reviewed By: uweigand Subscribers: yaxunl, cfe-commits, bader, Anastasia, uweigand Differential Revision: https://reviews.llvm.org/D33648 git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@304191 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
bb482b389e
commit
f709b9aef7
|
@ -8312,6 +8312,9 @@ def err_opencl_bitfields : Error<
|
||||||
"bit-fields are not supported in OpenCL">;
|
"bit-fields are not supported in OpenCL">;
|
||||||
def err_opencl_vla : Error<
|
def err_opencl_vla : Error<
|
||||||
"variable length arrays are not supported in OpenCL">;
|
"variable length arrays are not supported in OpenCL">;
|
||||||
|
def err_opencl_scalar_type_rank_greater_than_vector_type : Error<
|
||||||
|
"scalar operand type has greater rank than the type of the vector "
|
||||||
|
"element. (%0 and %1)">;
|
||||||
def err_bad_kernel_param_type : Error<
|
def err_bad_kernel_param_type : Error<
|
||||||
"%0 cannot be used as the type of a kernel parameter">;
|
"%0 cannot be used as the type of a kernel parameter">;
|
||||||
def err_record_with_pointers_kernel_param : Error<
|
def err_record_with_pointers_kernel_param : Error<
|
||||||
|
|
|
@ -8074,28 +8074,38 @@ QualType Sema::InvalidLogicalVectorOperands(SourceLocation Loc, ExprResult &LHS,
|
||||||
/// rank; for C, Obj-C, and C++ we allow any real scalar conversion except
|
/// rank; for C, Obj-C, and C++ we allow any real scalar conversion except
|
||||||
/// for float->int.
|
/// for float->int.
|
||||||
///
|
///
|
||||||
|
/// OpenCL V2.0 6.2.6.p2:
|
||||||
|
/// An error shall occur if any scalar operand type has greater rank
|
||||||
|
/// than the type of the vector element.
|
||||||
|
///
|
||||||
/// \param scalar - if non-null, actually perform the conversions
|
/// \param scalar - if non-null, actually perform the conversions
|
||||||
/// \return true if the operation fails (but without diagnosing the failure)
|
/// \return true if the operation fails (but without diagnosing the failure)
|
||||||
static bool tryVectorConvertAndSplat(Sema &S, ExprResult *scalar,
|
static bool tryVectorConvertAndSplat(Sema &S, ExprResult *scalar,
|
||||||
QualType scalarTy,
|
QualType scalarTy,
|
||||||
QualType vectorEltTy,
|
QualType vectorEltTy,
|
||||||
QualType vectorTy) {
|
QualType vectorTy,
|
||||||
|
unsigned &DiagID) {
|
||||||
// The conversion to apply to the scalar before splatting it,
|
// The conversion to apply to the scalar before splatting it,
|
||||||
// if necessary.
|
// if necessary.
|
||||||
CastKind scalarCast = CK_Invalid;
|
CastKind scalarCast = CK_Invalid;
|
||||||
|
|
||||||
if (vectorEltTy->isIntegralType(S.Context)) {
|
if (vectorEltTy->isIntegralType(S.Context)) {
|
||||||
if (!scalarTy->isIntegralType(S.Context))
|
if (S.getLangOpts().OpenCL && (scalarTy->isRealFloatingType() ||
|
||||||
|
(scalarTy->isIntegerType() &&
|
||||||
|
S.Context.getIntegerTypeOrder(vectorEltTy, scalarTy) < 0))) {
|
||||||
|
DiagID = diag::err_opencl_scalar_type_rank_greater_than_vector_type;
|
||||||
return true;
|
return true;
|
||||||
if (S.getLangOpts().OpenCL &&
|
}
|
||||||
S.Context.getIntegerTypeOrder(vectorEltTy, scalarTy) < 0)
|
if (!scalarTy->isIntegralType(S.Context))
|
||||||
return true;
|
return true;
|
||||||
scalarCast = CK_IntegralCast;
|
scalarCast = CK_IntegralCast;
|
||||||
} else if (vectorEltTy->isRealFloatingType()) {
|
} else if (vectorEltTy->isRealFloatingType()) {
|
||||||
if (scalarTy->isRealFloatingType()) {
|
if (scalarTy->isRealFloatingType()) {
|
||||||
if (S.getLangOpts().OpenCL &&
|
if (S.getLangOpts().OpenCL &&
|
||||||
S.Context.getFloatingTypeOrder(vectorEltTy, scalarTy) < 0)
|
S.Context.getFloatingTypeOrder(vectorEltTy, scalarTy) < 0) {
|
||||||
|
DiagID = diag::err_opencl_scalar_type_rank_greater_than_vector_type;
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
scalarCast = CK_FloatingCast;
|
scalarCast = CK_FloatingCast;
|
||||||
}
|
}
|
||||||
else if (scalarTy->isIntegralType(S.Context))
|
else if (scalarTy->isIntegralType(S.Context))
|
||||||
|
@ -8341,10 +8351,12 @@ QualType Sema::CheckVectorOperands(ExprResult &LHS, ExprResult &RHS,
|
||||||
|
|
||||||
// If there's a vector type and a scalar, try to convert the scalar to
|
// If there's a vector type and a scalar, try to convert the scalar to
|
||||||
// the vector element type and splat.
|
// the vector element type and splat.
|
||||||
|
unsigned DiagID = diag::err_typecheck_vector_not_convertable;
|
||||||
if (!RHSVecType) {
|
if (!RHSVecType) {
|
||||||
if (isa<ExtVectorType>(LHSVecType)) {
|
if (isa<ExtVectorType>(LHSVecType)) {
|
||||||
if (!tryVectorConvertAndSplat(*this, &RHS, RHSType,
|
if (!tryVectorConvertAndSplat(*this, &RHS, RHSType,
|
||||||
LHSVecType->getElementType(), LHSType))
|
LHSVecType->getElementType(), LHSType,
|
||||||
|
DiagID))
|
||||||
return LHSType;
|
return LHSType;
|
||||||
} else {
|
} else {
|
||||||
if (!tryGCCVectorConvertAndSplat(*this, &RHS, &LHS))
|
if (!tryGCCVectorConvertAndSplat(*this, &RHS, &LHS))
|
||||||
|
@ -8355,7 +8367,7 @@ QualType Sema::CheckVectorOperands(ExprResult &LHS, ExprResult &RHS,
|
||||||
if (isa<ExtVectorType>(RHSVecType)) {
|
if (isa<ExtVectorType>(RHSVecType)) {
|
||||||
if (!tryVectorConvertAndSplat(*this, (IsCompAssign ? nullptr : &LHS),
|
if (!tryVectorConvertAndSplat(*this, (IsCompAssign ? nullptr : &LHS),
|
||||||
LHSType, RHSVecType->getElementType(),
|
LHSType, RHSVecType->getElementType(),
|
||||||
RHSType))
|
RHSType, DiagID))
|
||||||
return RHSType;
|
return RHSType;
|
||||||
} else {
|
} else {
|
||||||
if (LHS.get()->getValueKind() == VK_LValue ||
|
if (LHS.get()->getValueKind() == VK_LValue ||
|
||||||
|
@ -8431,7 +8443,7 @@ QualType Sema::CheckVectorOperands(ExprResult &LHS, ExprResult &RHS,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, use the generic diagnostic.
|
// Otherwise, use the generic diagnostic.
|
||||||
Diag(Loc, diag::err_typecheck_vector_not_convertable)
|
Diag(Loc, DiagID)
|
||||||
<< LHSType << RHSType
|
<< LHSType << RHSType
|
||||||
<< LHS.get()->getSourceRange() << RHS.get()->getSourceRange();
|
<< LHS.get()->getSourceRange() << RHS.get()->getSourceRange();
|
||||||
return QualType();
|
return QualType();
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
// RUN: %clang_cc1 %s -triple spir-unknown-unknown -verify -pedantic -fsyntax-only -cl-std=CL1.2
|
||||||
|
|
||||||
|
typedef float float2 __attribute__((ext_vector_type(2)));
|
||||||
|
typedef long long2 __attribute__((ext_vector_type(2)));
|
||||||
|
typedef int int2 __attribute__((ext_vector_type(2)));
|
||||||
|
|
||||||
|
kernel void foo1(float2 in, global float2 *out) { *out = in + 0.5;} // expected-error {{scalar operand type has greater rank than the type of the vector element. ('float2' (vector of 2 'float' values) and 'double')}}
|
||||||
|
|
||||||
|
kernel void foo2(float2 in, global float2 *out) { *out = 0.5 + in;} // expected-error {{scalar operand type has greater rank than the type of the vector element. ('double' and 'float2' (vector of 2 'float' values))}}
|
||||||
|
|
||||||
|
kernel void foo3(float2 in, global float2 *out) { *out = 0.5f + in;}
|
||||||
|
|
||||||
|
kernel void foo4(long2 in, global long2 *out) { *out = 5 + in;}
|
||||||
|
|
||||||
|
kernel void foo5(float2 in, global float2 *out) {
|
||||||
|
float* f;
|
||||||
|
*out = f + in; // expected-error{{cannot convert between vector and non-scalar values ('float *' and 'float2' (vector of 2 'float' values))}}
|
||||||
|
}
|
||||||
|
|
||||||
|
kernel void foo6(int2 in, global int2 *out) {
|
||||||
|
int* f;
|
||||||
|
*out = f + in; // expected-error{{cannot convert between vector and non-scalar values ('int *' and 'int2' (vector of 2 'int' values))}}
|
||||||
|
}
|
|
@ -89,7 +89,7 @@ float2 ntest04(int2 C, int2 X, float2 Y)
|
||||||
|
|
||||||
float2 ntest05(int2 C, int2 X, float Y)
|
float2 ntest05(int2 C, int2 X, float Y)
|
||||||
{
|
{
|
||||||
return C ? X : Y; // expected-error {{cannot convert between vector values of different size ('int2' (vector of 2 'int' values) and 'float')}}
|
return C ? X : Y; // expected-error {{scalar operand type has greater rank than the type of the vector element. ('int2' (vector of 2 'int' values) and 'float'}}
|
||||||
}
|
}
|
||||||
|
|
||||||
char2 ntest06(int2 C, char2 X, char2 Y)
|
char2 ntest06(int2 C, char2 X, char2 Y)
|
||||||
|
|
Loading…
Reference in New Issue