; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt -S -passes=instsimplify < %s | FileCheck %s
define <4 x i32> @test1(<4 x i32> %A) {
; CHECK-LABEL: @test1(
; CHECK-NEXT: ret <4 x i32> poison
;
%I = insertelement <4 x i32> %A, i32 5, i64 4294967296
ret <4 x i32> %I
}
define <4 x i32> @test2(<4 x i32> %A) {
; CHECK-LABEL: @test2(
; CHECK-NEXT: ret <4 x i32> poison
;
%I = insertelement <4 x i32> %A, i32 5, i64 4
ret <4 x i32> %I
}
define <4 x i32> @test3(<4 x i32> %A) {
; CHECK-LABEL: @test3(
; CHECK-NEXT: [[I:%.*]] = insertelement <4 x i32> [[A:%.*]], i32 5, i64 1
; CHECK-NEXT: ret <4 x i32> [[I]]
;
%I = insertelement <4 x i32> %A, i32 5, i64 1
ret <4 x i32> %I
}
define <4 x i32> @test4(<4 x i32> %A) {
; CHECK-LABEL: @test4(
; CHECK-NEXT: ret <4 x i32> poison
;
%I = insertelement <4 x i32> %A, i32 5, i128 100
ret <4 x i32> %I
}
define <4 x i32> @test5(<4 x i32> %A) {
; CHECK-LABEL: @test5(
; CHECK-NEXT: ret <4 x i32> poison
;
%I = insertelement <4 x i32> %A, i32 5, i64 undef
ret <4 x i32> %I
}
define <4 x i32> @test5_poison(<4 x i32> %A) {
; CHECK-LABEL: @test5_poison(
; CHECK-NEXT: ret <4 x i32> poison
;
%I = insertelement <4 x i32> %A, i32 5, i64 poison
ret <4 x i32> %I
}
define <4 x i32> @elem_poison(<4 x i32> %A) {
; CHECK-LABEL: @elem_poison(
; CHECK-NEXT: ret <4 x i32> [[A:%.*]]
;
%B = insertelement <4 x i32> %A, i32 poison, i32 1
ret <4 x i32> %B
}
; The undef may be replacing a poison value, so it is not safe to just return 'A'.
define <4 x i32> @PR1286(<4 x i32> %A) {
; CHECK-LABEL: @PR1286(
; CHECK-NEXT: [[B:%.*]] = insertelement <4 x i32> [[A:%.*]], i32 undef, i32 1
; CHECK-NEXT: ret <4 x i32> [[B]]
;
%B = insertelement <4 x i32> %A, i32 undef, i32 1
ret <4 x i32> %B
}
; Constant is not poison, so this can simplify.
define <2 x i32> @undef_into_constant_vector_with_variable_index(<2 x i32> %A, i32 %Index) {
; CHECK-LABEL: @undef_into_constant_vector_with_variable_index(
; CHECK-NEXT: ret <2 x i32> <i32 42, i32 -42>
;
%B = insertelement <2 x i32> <i32 42, i32 -42>, i32 undef, i32 %Index
ret <2 x i32> %B
}
define <8 x i8> @extract_insert_same_vec_and_index(<8 x i8> %in) {
; CHECK-LABEL: @extract_insert_same_vec_and_index(
; CHECK-NEXT: ret <8 x i8> [[IN:%.*]]
;
%val = extractelement <8 x i8> %in, i32 5
%vec = insertelement <8 x i8> %in, i8 %val, i32 5
ret <8 x i8> %vec
}
define <8 x i8> @extract_insert_same_vec_and_index2(<8 x i8> %in, i32 %index) {
; CHECK-LABEL: @extract_insert_same_vec_and_index2(
; CHECK-NEXT: ret <8 x i8> [[IN:%.*]]
;
%val = extractelement <8 x i8> %in, i32 %index
%vec = insertelement <8 x i8> %in, i8 %val, i32 %index
ret <8 x i8> %vec
}
; The insert is in an unreachable block, so it is allowed to point to itself.
; This would crash via stack overflow.
define void @PR43218() {
; CHECK-LABEL: @PR43218(
; CHECK-NEXT: end:
; CHECK-NEXT: ret void
; CHECK: unreachable_infloop:
; CHECK-NEXT: [[EXTRACT:%.*]] = extractelement <2 x i64> [[BOGUS:%.*]], i32 0
; CHECK-NEXT: [[T0:%.*]] = inttoptr i64 [[EXTRACT]] to ptr
; CHECK-NEXT: [[BOGUS]] = insertelement <2 x i64> [[BOGUS]], i64 undef, i32 1
; CHECK-NEXT: br label [[UNREACHABLE_INFLOOP:%.*]]
;
end:
ret void
unreachable_infloop:
%extract = extractelement <2 x i64> %bogus, i32 0
%t0 = inttoptr i64 %extract to ptr
%bogus = insertelement <2 x i64> %bogus, i64 undef, i32 1
br label %unreachable_infloop
}
define <4 x i32> @insert_into_splat(i32 %index) {
; CHECK-LABEL: @insert_into_splat(
; CHECK-NEXT: ret <4 x i32> <i32 3, i32 3, i32 3, i32 3>
;
%I = insertelement <4 x i32> <i32 3, i32 3, i32 3, i32 3>, i32 3, i32 %index
ret <4 x i32> %I
}