[Beignet] [PATCH] GBE: fix a legalize pass bug when bitcast wide integer to incompaitble vector.
Zhigang Gong
zhigang.gong at intel.com
Thu Oct 16 21:25:29 PDT 2014
Our wide integer legalize pass will assume the source type of a wide
integer must be the same as the final use type. But this is not always
true. The following case is a real example:
%conv.i.i.14 = sext i8 %usrc0.i.sroa.0.14.extract.trunc to i32
%call.i.i2.14 = tail call i32 @__gen_ocl_abs(i32 %conv.i.i.14) #5
%conv1.i.i.14.mask = and i32 %call.i.i2.14, 255
%uret.i.sroa.0.14.insert.ext = zext i32 %conv1.i.i.14.mask to i128
%uret.i.sroa.0.14.insert.shift = shl nuw nsw i128 %uret.i.sroa.0.14.insert.ext, 112
......
%uret.i.sroa.0.15.insert.mask = or i128 %uret.i.sroa.0.14.insert.mask.masked, %uret.i.sroa.0.14.insert.shift
%uret.i.sroa.0.15.insert.insert = or i128 %uret.i.sroa.0.15.insert.mask, %uret.i.sroa.0.15.insert.shift
%2 = bitcast i128 %uret.i.sroa.0.15.insert.insert to <16 x i8>
The wide integer i128 %uret.i.sroa.0.16.insert.insert is from an
i32 integer %conv1.i.i.14.mask. But the use of it is i8 vector
which breaks our assumption.
According to ruiling's good suggestion, we always bitcast the wide integer to
a compatible vector, take the above example:
%3 = bitcast i128 %uret.i.sroa.0.15.insert.insert to <4 x i32>
Then insert a bit cast instruction to convert it to the original destination
%2 = bitcast <4 x i32> %3 to <16 x i8>
Signed-off-by: Zhigang Gong <zhigang.gong at intel.com>
---
backend/src/llvm/llvm_legalize.cpp | 16 ++++++++++++----
1 file changed, 12 insertions(+), 4 deletions(-)
diff --git a/backend/src/llvm/llvm_legalize.cpp b/backend/src/llvm/llvm_legalize.cpp
index 7a78a97..3a2cabb 100644
--- a/backend/src/llvm/llvm_legalize.cpp
+++ b/backend/src/llvm/llvm_legalize.cpp
@@ -392,6 +392,7 @@ namespace gbe {
}
valueMap.insert(std::make_pair(p, v2));
}
+
void Legalize::legalizeBitCast(IRBuilder<> &Builder, Instruction *p) {
SmallVector<Value*, 16> split;
Type *dstTy = p->getType();
@@ -412,16 +413,23 @@ namespace gbe {
//bitcast from large integer to vector, so we do insertElement to build the vector
ValueMapIter iter = valueMap.find(p->getOperand(0));
SmallVectorImpl<Value*> &opVec = iter->second;
- Type *elemTy = cast<VectorType>(dstTy)->getElementType();
- GBE_ASSERT(elemTy == opVec[0]->getType());
+ Type *splitTy = opVec[0]->getType();
+ GBE_ASSERT(dstTy->getPrimitiveSizeInBits() % splitTy->getPrimitiveSizeInBits() == 0);
+ GBE_ASSERT(dstTy->getPrimitiveSizeInBits() / splitTy->getPrimitiveSizeInBits() == opVec.size());
Value *vec = NULL;
Type *idxTy = IntegerType::get(p->getContext(), 32);
for (unsigned i = 0; i < opVec.size(); ++i) {
- Value *tmp = vec ? vec : UndefValue::get(dstTy);
+ Value *tmp = vec ? vec : UndefValue::get(VectorType::get(splitTy, opVec.size()));
Value *idx = ConstantInt::get(idxTy, i);
vec = Builder.CreateInsertElement(tmp, opVec[i], idx);
}
- p->replaceAllUsesWith(vec);
+ Type *elemTy = cast<VectorType>(dstTy)->getElementType();
+ if (elemTy == opVec[0]->getType())
+ p->replaceAllUsesWith(vec);
+ else {
+ Value *newVec = Builder.CreateBitCast(vec, dstTy);
+ p->replaceAllUsesWith(newVec);
+ }
} else {
p->dump(); GBE_ASSERT(0 && "Unsupported bitcast");
}
--
1.8.3.2
More information about the Beignet
mailing list