; 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 }