blob: b135f76f709f10645bf7c33cdd91d270b4bf46c8 [file] [log] [blame]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt < %s -passes=instcombine -S | FileCheck %s
declare noalias ptr @calloc(i32, i32) nounwind allockind("alloc,zeroed") allocsize(0,1) "alloc-family"="malloc"
declare void @free(ptr) allockind("free") "alloc-family"="malloc"
; Test load from uninitialized alloca - should be removed and replaced with undef
define i32 @test_load_uninitialized_alloca() {
; CHECK-LABEL: @test_load_uninitialized_alloca(
; CHECK-NEXT: ret i32 undef
;
%a = alloca i32
%v = load i32, ptr %a
ret i32 %v
}
; Test load from zero-initialized malloc - should be removed and replaced with zero
define i32 @test_load_zero_initialized_malloc() {
; CHECK-LABEL: @test_load_zero_initialized_malloc(
; CHECK-NEXT: ret i32 0
;
%a = call ptr @calloc(i32 1, i32 4)
%v = load i32, ptr %a
call void @free(ptr %a)
ret i32 %v
}
; Test memcpy from uninitialized source - should be removed
define void @test_memcpy_from_uninitialized_alloca(ptr %dest) {
; CHECK-LABEL: @test_memcpy_from_uninitialized_alloca(
; CHECK-NEXT: ret void
;
%src = alloca i32, align 1
call void @llvm.memcpy.p0.p0.i32(ptr %src, ptr %src, i32 4, i1 false)
call void @llvm.memcpy.p0.p0.i32(ptr %dest, ptr %src, i32 4, i1 false)
ret void
}
; Test memcpy from zeroed source - should transform to memset with zero
define void @test_memcpy_from_uninitialized_calloc(ptr %dest) {
; CHECK-LABEL: @test_memcpy_from_uninitialized_calloc(
; CHECK-NEXT: call void @llvm.memset.p0.i32(ptr noundef nonnull align 1 dereferenceable(16) [[DEST:%.*]], i8 0, i32 16, i1 false)
; CHECK-NEXT: ret void
;
%src = call ptr @calloc(i32 1, i32 16)
call void @llvm.memcpy.p0.p0.i32(ptr %dest, ptr %src, i32 16, i1 false)
call void @free(ptr %src)
ret void
}
; Test mixed read/write pattern - should not be removable due to write before read
define i32 @test_write_then_read_alloca() {
; CHECK-LABEL: @test_write_then_read_alloca(
; CHECK-NEXT: [[A:%.*]] = alloca i32, align 4
; CHECK-NEXT: store i8 42, ptr [[A]], align 1
; CHECK-NEXT: [[V:%.*]] = load i32, ptr [[A]], align 4
; CHECK-NEXT: ret i32 [[V]]
;
%a = alloca i32
store i8 42, ptr %a
%v = load i32, ptr %a
ret i32 %v
}
; Test read then write pattern - should not be removable due to conflicting access
define void @test_read_then_write_alloca() {
; CHECK-LABEL: @test_read_then_write_alloca(
; CHECK-NEXT: [[A:%.*]] = alloca i32, align 4
; CHECK-NEXT: [[V:%.*]] = load i32, ptr [[A]], align 4
; CHECK-NEXT: [[V8:%.*]] = trunc i32 [[V]] to i8
; CHECK-NEXT: store i8 [[V8]], ptr [[A]], align 1
; CHECK-NEXT: ret void
;
%a = alloca i32
%v = load i32, ptr %a
%v8 = trunc i32 %v to i8
store i8 %v8, ptr %a
ret void
}
; Test load through GEP from uninitialized alloca
define i8 @test_load_gep_uninitialized_alloca() {
; CHECK-LABEL: @test_load_gep_uninitialized_alloca(
; CHECK-NEXT: ret i8 undef
;
%a = alloca [4 x i8]
%gep = getelementptr [4 x i8], ptr %a, i32 0, i32 2
%v = load i8, ptr %gep
ret i8 %v
}
; Test load through bitcast from uninitialized alloca
define i16 @test_load_bitcast_uninitialized_alloca() {
; CHECK-LABEL: @test_load_bitcast_uninitialized_alloca(
; CHECK-NEXT: ret i16 undef
;
%a = alloca i32
%bc = bitcast ptr %a to ptr
%v = load i16, ptr %bc
ret i16 %v
}
; Test memmove from zero-initialized malloc
define void @test_memmove_from_zero_initialized_malloc(ptr %dest) {
; CHECK-LABEL: @test_memmove_from_zero_initialized_malloc(
; CHECK-NEXT: call void @llvm.memset.p0.i32(ptr noundef nonnull align 1 dereferenceable(32) [[DEST:%.*]], i8 0, i32 32, i1 false)
; CHECK-NEXT: ret void
;
%src = call ptr @calloc(i32 8, i32 4)
call void @llvm.memmove.p0.p0.i32(ptr %dest, ptr %src, i32 32, i1 false)
call void @free(ptr %src)
ret void
}
; Test multiple loads from same uninitialized alloca
define { i32, i32 } @test_multiple_loads_uninitialized_alloca() {
; CHECK-LABEL: @test_multiple_loads_uninitialized_alloca(
; CHECK-NEXT: ret { i32, i32 } undef
;
%a = alloca [2 x i32]
%gep1 = getelementptr [2 x i32], ptr %a, i32 0, i32 0
%gep2 = getelementptr [2 x i32], ptr %a, i32 0, i32 1
%v1 = load i32, ptr %gep1
%v2 = load i32, ptr %gep2
%ret = insertvalue { i32, i32 } { i32 undef, i32 poison }, i32 %v1, 0
%ret2 = insertvalue { i32, i32 } %ret, i32 %v2, 1
ret { i32, i32 } %ret2
}
; Test that volatile operations prevent removal
define i32 @test_volatile_load_prevents_removal() {
; CHECK-LABEL: @test_volatile_load_prevents_removal(
; CHECK-NEXT: [[A:%.*]] = alloca i32, align 4
; CHECK-NEXT: [[V:%.*]] = load volatile i32, ptr [[A]], align 4
; CHECK-NEXT: ret i32 [[V]]
;
%a = alloca i32
%v = load volatile i32, ptr %a
ret i32 %v
}