Test source: git
Source: <stdin> ---------------------------------------- @a = global 4000 bytes, align 16 @b = global 4000 bytes, align 16 define void @_Z3foov() { %entry: br label %for.cond %for.cond: %i.0 = phi i32 [ 0, %entry ], [ %inc, %for.body ] %cmp = icmp ult i32 %i.0, 1000 br i1 %cmp, label %for.body, label %for.cond.cleanup %for.body: %0 = zext i32 %i.0 to i64 %arrayidx = gep inbounds * @a, 4000 x i64 0, 4 x i64 %0 %arrayidx2 = gep inbounds * @b, 4000 x i64 0, 4 x i64 %0 %1 = load float, * %arrayidx, align 4 %2 = load float, * %arrayidx2, align 4 %cmp.i = fcmp fast olt float %1, %2 %__b.__a.i = select i1 %cmp.i, * %arrayidx2, * %arrayidx %3 = bitcast * %__b.__a.i to * %4 = load i32, * %3, align 4 %5 = bitcast * %arrayidx to * store i32 %4, * %5, align 4 %inc = add nsw nuw i32 %i.0, 1 br label %for.cond %for.cond.cleanup: ret void } => @a = global 4000 bytes, align 16 @b = global 4000 bytes, align 16 define void @_Z3foov() { %entry: br label %for.cond %for.cond: %i.0 = phi i32 [ 0, %entry ], [ %inc, %for.body ] %cmp = icmp ult i32 %i.0, 1000 br i1 %cmp, label %for.body, label %for.cond.cleanup %for.body: %0 = zext i32 %i.0 to i64 %arrayidx = gep inbounds * @a, 4000 x i64 0, 4 x i64 %0 %arrayidx2 = gep inbounds * @b, 4000 x i64 0, 4 x i64 %0 %1 = load float, * %arrayidx, align 4 %2 = load float, * %arrayidx2, align 4 %cmp.i = fcmp fast olt float %1, %2 %3 = select i1 %cmp.i, float %2, float %1 store float %3, * %arrayidx, align 4 %inc = add nsw nuw i32 %i.0, 1 br label %for.cond %for.cond.cleanup: ret void } Transformation doesn't verify! ERROR: The source program doesn't reach a return instruction. Consider increasing the unroll factor if it has loops ---------------------------------------- define i32 @store_bitcasted_load(i1 %cond, * dereferenceable(4) %addr1, * dereferenceable(4) %addr2) { %0: %sel = select i1 %cond, * dereferenceable(4) %addr1, * dereferenceable(4) %addr2 %bc1 = bitcast * %sel to * %ld = load i32, * %bc1, align 4 ret i32 %ld } => define i32 @store_bitcasted_load(i1 %cond, * dereferenceable(4) %addr1, * dereferenceable(4) %addr2) { %0: %sel = select i1 %cond, * dereferenceable(4) %addr1, * dereferenceable(4) %addr2 %bc1 = bitcast * %sel to * %ld = load i32, * %bc1, align 4 ret i32 %ld } Transformation seems to be correct! (syntactically equal) ---------------------------------------- define void @bitcasted_store(i1 %cond, * %loadaddr1, * %loadaddr2, * %storeaddr) { %0: %sel = select i1 %cond, * %loadaddr1, * %loadaddr2 %int_load_addr = bitcast * %sel to * %ld = load i32, * %int_load_addr, align 4 %int_store_addr = bitcast * %storeaddr to * store i32 %ld, * %int_store_addr, align 4 ret void } => define void @bitcasted_store(i1 %cond, * %loadaddr1, * %loadaddr2, * %storeaddr) { %0: %sel = select i1 %cond, * %loadaddr1, * %loadaddr2 %int_load_addr = bitcast * %sel to * %ld = load i32, * %int_load_addr, align 4 %int_store_addr = bitcast * %storeaddr to * store i32 %ld, * %int_store_addr, align 4 ret void } Transformation seems to be correct! (syntactically equal) ---------------------------------------- define void @bitcasted_minmax_with_select_of_pointers(* %loadaddr1, * %loadaddr2, * %storeaddr) { %0: %ld1 = load float, * %loadaddr1, align 4 %ld2 = load float, * %loadaddr2, align 4 %cond = fcmp ogt float %ld1, %ld2 %sel = select i1 %cond, * %loadaddr1, * %loadaddr2 %int_load_addr = bitcast * %sel to * %ld = load i32, * %int_load_addr, align 4 %int_store_addr = bitcast * %storeaddr to * store i32 %ld, * %int_store_addr, align 4 ret void } => define void @bitcasted_minmax_with_select_of_pointers(* %loadaddr1, * %loadaddr2, * %storeaddr) { %0: %ld1 = load float, * %loadaddr1, align 4 %ld2 = load float, * %loadaddr2, align 4 %cond = fcmp ogt float %ld1, %ld2 %ld3 = select i1 %cond, float %ld1, float %ld2 store float %ld3, * %storeaddr, align 4 ret void } Transformation doesn't verify! ERROR: Mismatch in memory Example: * %loadaddr1 = pointer(non-local, block_id=1, offset=0) * %loadaddr2 = pointer(non-local, block_id=2, offset=0) * %storeaddr = pointer(non-local, block_id=1, offset=0) Source: float %ld1 = #x00000000 (+0.0) float %ld2 = NaN i1 %cond = #x0 (0) * %sel = pointer(non-local, block_id=2, offset=0) * %int_load_addr = pointer(non-local, block_id=2, offset=0) i32 %ld = #x7f800400 (2139096064) * %int_store_addr = pointer(non-local, block_id=1, offset=0) SOURCE MEMORY STATE =================== NON-LOCAL BLOCKS: Block 0 > size: 0 align: 4 alloc type: 0 Block 1 > size: 9 align: 32 alloc type: 0 Block 2 > size: 8 align: 512 alloc type: 0 Block 3 > align: 2 alloc type: 0 Target: float %ld1 = #x00000000 (+0.0) float %ld2 = NaN i1 %cond = #x0 (0) float %ld3 = NaN Mismatch in pointer(non-local, block_id=1, offset=0) Source value: #bp Target value: #bp ------------------- SMT STATS ------------------- Num queries: 3 Num invalid: 0 Num skips: 0 Num trivial: 18 (85.7%) Num timeout: 0 (0.0%) Num errors: 0 (0.0%) Num SAT: 2 (66.7%) Num UNSAT: 1 (33.3%) Alive2: Transform doesn't verify; aborting!
+ : 'RUN: at line 2' + /home/nlopes/alive2/build/opt-alive.sh -instcombine -S -data-layout=e-m:e-i64:64-f80:128-n8:16:32:64-S128 + /home/nlopes/llvm/build/bin/FileCheck /home/nlopes/llvm/llvm/test/Transforms/InstCombine/load-bitcast-select.ll FileCheck error: '<stdin>' is empty. FileCheck command line: /home/nlopes/llvm/build/bin/FileCheck /home/nlopes/llvm/llvm/test/Transforms/InstCombine/load-bitcast-select.ll