mirror of
https://github.com/RPCS3/llvm.git
synced 2024-12-03 17:32:59 +00:00
[InstCombine] fix crash due to ignored addrspacecast
Summary: Part of the InstCombine code for simplifying GEPs looks through addrspacecasts. However, this was done by updating a variable also used by the next transformation, for marking GEPs as inbounds. This led to replacing a GEP with a similar instruction in a different addrspace, which caused an assertion failure in RAUW. This caused julia issue https://github.com/JuliaLang/julia/issues/27055 Patch by Jeff Bezanson <jeff@juliacomputing.com> Reviewed By: arsenm Differential Revision: https://reviews.llvm.org/D46722 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@332302 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
26b9bb1166
commit
abb2646f5a
@ -1937,16 +1937,17 @@ Instruction *InstCombiner::visitGetElementPtrInst(GetElementPtrInst &GEP) {
|
||||
// addrspacecast between types is canonicalized as a bitcast, then an
|
||||
// addrspacecast. To take advantage of the below bitcast + struct GEP, look
|
||||
// through the addrspacecast.
|
||||
Value *ASCStrippedPtrOp = PtrOp;
|
||||
if (auto *ASC = dyn_cast<AddrSpaceCastInst>(PtrOp)) {
|
||||
// X = bitcast A addrspace(1)* to B addrspace(1)*
|
||||
// Y = addrspacecast A addrspace(1)* to B addrspace(2)*
|
||||
// Z = gep Y, <...constant indices...>
|
||||
// Into an addrspacecasted GEP of the struct.
|
||||
if (auto *BC = dyn_cast<BitCastInst>(ASC->getOperand(0)))
|
||||
PtrOp = BC;
|
||||
ASCStrippedPtrOp = BC;
|
||||
}
|
||||
|
||||
if (auto *BCI = dyn_cast<BitCastInst>(PtrOp)) {
|
||||
if (auto *BCI = dyn_cast<BitCastInst>(ASCStrippedPtrOp)) {
|
||||
Value *SrcOp = BCI->getOperand(0);
|
||||
PointerType *SrcType = cast<PointerType>(BCI->getSrcTy());
|
||||
Type *SrcEltType = SrcType->getElementType();
|
||||
|
@ -32,3 +32,22 @@ entry:
|
||||
ret void
|
||||
}
|
||||
|
||||
declare void @escape_alloca(i16*)
|
||||
|
||||
; check that addrspacecast is not ignored (leading to an assertion failure)
|
||||
; when trying to mark a GEP as inbounds
|
||||
define { i8, i8 } @inbounds_after_addrspacecast() {
|
||||
top:
|
||||
; CHECK-LABEL: @inbounds_after_addrspacecast
|
||||
%0 = alloca i16, align 2
|
||||
call void @escape_alloca(i16* %0)
|
||||
%tmpcast = bitcast i16* %0 to [2 x i8]*
|
||||
; CHECK: addrspacecast [2 x i8]* %tmpcast to [2 x i8] addrspace(11)*
|
||||
%1 = addrspacecast [2 x i8]* %tmpcast to [2 x i8] addrspace(11)*
|
||||
; CHECK: getelementptr [2 x i8], [2 x i8] addrspace(11)* %1, i64 0, i64 1
|
||||
%2 = getelementptr [2 x i8], [2 x i8] addrspace(11)* %1, i64 0, i64 1
|
||||
; CHECK: addrspace(11)
|
||||
%3 = load i8, i8 addrspace(11)* %2, align 1
|
||||
%.fca.1.insert = insertvalue { i8, i8 } zeroinitializer, i8 %3, 1
|
||||
ret { i8, i8 } %.fca.1.insert
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user