; The only pass criterion is that spirv-val considers output valid. ; RUN: %if spirv-tools %{ llc -O0 -mtriple=spirv64-unknown-unknown %s -o - -filetype=obj | spirv-val %} %subgr = type { i64, i64 } %t_range = type { %t_arr } %t_arr = type { [1 x i64] } %t_arr2 = type { [4 x i32] } define internal spir_func noundef i32 @geti32() { entry: ret i32 100 } define internal spir_func noundef i64 @geti64() { entry: ret i64 200 } define internal spir_func void @enable_if(ptr addrspace(4) noundef align 8 dereferenceable_or_null(8) %this, i64 noundef %dim0) { entry: %this.addr = alloca ptr addrspace(4), align 8 %dim0.addr = alloca i64, align 8 store ptr addrspace(4) %this, ptr %this.addr, align 8 store i64 %dim0, ptr %dim0.addr, align 8 %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %0 = load i64, ptr %dim0.addr, align 8 call spir_func void @enable_if_2(ptr addrspace(4) noundef align 8 dereferenceable_or_null(8) %this1, i64 noundef %0) ret void } define internal spir_func void @test(ptr addrspace(4) noundef align 8 dereferenceable_or_null(16) %this, ptr addrspace(4) noundef align 4 dereferenceable(16) %bits, ptr noundef byval(%t_range) align 8 %pos) { entry: %this.addr = alloca ptr addrspace(4), align 8 %bits.addr = alloca ptr addrspace(4), align 8 %cur_pos = alloca i64, align 8 %__range4 = alloca ptr addrspace(4), align 8 %__begin0 = alloca ptr addrspace(4), align 8 %__end0 = alloca ptr addrspace(4), align 8 %cleanup.dest.slot = alloca i32, align 4 %elem = alloca ptr addrspace(4), align 8 %agg.tmp = alloca %t_range, align 8 %agg.tmp.ascast = addrspacecast ptr %agg.tmp to ptr addrspace(4) store ptr addrspace(4) %this, ptr %this.addr, align 8 store ptr addrspace(4) %bits, ptr %bits.addr, align 8 %pos.ascast = addrspacecast ptr %pos to ptr addrspace(4) %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %call = call spir_func noundef i64 @getp(ptr addrspace(4) noundef align 8 dereferenceable_or_null(8) %pos.ascast, i32 noundef 0) store i64 %call, ptr %cur_pos, align 8 %0 = load ptr addrspace(4), ptr %bits.addr, align 8 store ptr addrspace(4) %0, ptr %__range4, align 8 %1 = load ptr addrspace(4), ptr %__range4, align 8 %call2 = call spir_func noundef ptr addrspace(4) @beginp(ptr addrspace(4) noundef align 4 dereferenceable_or_null(16) %1) store ptr addrspace(4) %call2, ptr %__begin0, align 8 %2 = load ptr addrspace(4), ptr %__range4, align 8 %call3 = call spir_func noundef ptr addrspace(4) @endp(ptr addrspace(4) noundef align 4 dereferenceable_or_null(16) %2) store ptr addrspace(4) %call3, ptr %__end0, align 8 br label %for.cond for.cond: ; preds = %for.inc, %entry %3 = load ptr addrspace(4), ptr %__begin0, align 8 %4 = load ptr addrspace(4), ptr %__end0, align 8 %cmp = icmp ne ptr addrspace(4) %3, %4 br i1 %cmp, label %for.body, label %for.cond.cleanup for.cond.cleanup: ; preds = %for.cond br label %for.end for.body: ; preds = %for.cond %5 = load ptr addrspace(4), ptr %__begin0, align 8 store ptr addrspace(4) %5, ptr %elem, align 8 %6 = load i64, ptr %cur_pos, align 8 %call4 = call spir_func noundef i32 @maskp(ptr addrspace(4) noundef align 8 dereferenceable_or_null(16) %this1) %conv = zext i32 %call4 to i64 %cmp5 = icmp ult i64 %6, %conv br i1 %cmp5, label %if.then, label %if.else if.then: ; preds = %for.body %7 = load ptr addrspace(4), ptr %elem, align 8 %8 = load i64, ptr %cur_pos, align 8 call spir_func void @enable_if(ptr addrspace(4) noundef align 8 dereferenceable_or_null(8) %agg.tmp.ascast, i64 noundef %8) call spir_func void @extract_bits(ptr addrspace(4) noundef align 8 dereferenceable_or_null(16) %this1, ptr addrspace(4) noundef align 4 dereferenceable(4) %7, ptr noundef byval(%t_range) align 8 %agg.tmp) %9 = load i64, ptr %cur_pos, align 8 %add = add i64 %9, 32 store i64 %add, ptr %cur_pos, align 8 br label %if.end if.else: ; preds = %for.body %10 = load ptr addrspace(4), ptr %elem, align 8 store i32 0, ptr addrspace(4) %10, align 4 br label %if.end if.end: ; preds = %if.else, %if.then br label %for.inc for.inc: ; preds = %if.end %11 = load ptr addrspace(4), ptr %__begin0, align 8 %incdec.ptr = getelementptr inbounds nuw i32, ptr addrspace(4) %11, i32 1 store ptr addrspace(4) %incdec.ptr, ptr %__begin0, align 8 br label %for.cond for.end: ; preds = %for.cond.cleanup ret void } define internal spir_func noundef i64 @getp(ptr addrspace(4) noundef align 8 dereferenceable_or_null(8) %this, i32 noundef %dimension) { entry: %this.addr.i = alloca ptr addrspace(4), align 8 %dimension.addr.i = alloca i32, align 4 %retval = alloca i64, align 8 %this.addr = alloca ptr addrspace(4), align 8 %dimension.addr = alloca i32, align 4 %retval.ascast = addrspacecast ptr %retval to ptr addrspace(4) store ptr addrspace(4) %this, ptr %this.addr, align 8 store i32 %dimension, ptr %dimension.addr, align 4 %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %0 = load i32, ptr %dimension.addr, align 4 store ptr addrspace(4) %this1, ptr %this.addr.i, align 8 store i32 %0, ptr %dimension.addr.i, align 4 %this1.i = load ptr addrspace(4), ptr %this.addr.i, align 8 %common_array1 = bitcast ptr addrspace(4) %this1 to ptr addrspace(4) %1 = load i32, ptr %dimension.addr, align 4 %idxprom = sext i32 %1 to i64 %arrayidx = getelementptr inbounds [1 x i64], ptr addrspace(4) %common_array1, i64 0, i64 %idxprom %2 = load i64, ptr addrspace(4) %arrayidx, align 8 ret i64 %2 } define internal spir_func noundef ptr addrspace(4) @beginp(ptr addrspace(4) noundef align 4 dereferenceable_or_null(16) %this) { entry: %retval = alloca ptr addrspace(4), align 8 %this.addr = alloca ptr addrspace(4), align 8 %retval.ascast = addrspacecast ptr %retval to ptr addrspace(4) store ptr addrspace(4) %this, ptr %this.addr, align 8 %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %MData1 = bitcast ptr addrspace(4) %this1 to ptr addrspace(4) %arraydecay2 = bitcast ptr addrspace(4) %MData1 to ptr addrspace(4) ret ptr addrspace(4) %arraydecay2 } define internal spir_func noundef ptr addrspace(4) @endp(ptr addrspace(4) noundef align 4 dereferenceable_or_null(16) %this) { entry: %retval = alloca ptr addrspace(4), align 8 %this.addr = alloca ptr addrspace(4), align 8 %retval.ascast = addrspacecast ptr %retval to ptr addrspace(4) store ptr addrspace(4) %this, ptr %this.addr, align 8 %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %MData1 = bitcast ptr addrspace(4) %this1 to ptr addrspace(4) %arraydecay2 = bitcast ptr addrspace(4) %MData1 to ptr addrspace(4) %add.ptr = getelementptr inbounds nuw i32, ptr addrspace(4) %arraydecay2, i64 4 ret ptr addrspace(4) %add.ptr } define internal spir_func noundef i32 @maskp(ptr addrspace(4) noundef align 8 dereferenceable_or_null(16) %this) { entry: %retval = alloca i32, align 4 %this.addr = alloca ptr addrspace(4), align 8 %retval.ascast = addrspacecast ptr %retval to ptr addrspace(4) store ptr addrspace(4) %this, ptr %this.addr, align 8 %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %bits_num = getelementptr inbounds nuw %subgr, ptr addrspace(4) %this1, i32 0, i32 1 %0 = load i64, ptr addrspace(4) %bits_num, align 8 %conv = trunc i64 %0 to i32 ret i32 %conv } define internal spir_func void @enable_if_2(ptr addrspace(4) noundef align 8 dereferenceable_or_null(8) %this, i64 noundef %dim0) { entry: %this.addr = alloca ptr addrspace(4), align 8 %dim0.addr = alloca i64, align 8 store ptr addrspace(4) %this, ptr %this.addr, align 8 store i64 %dim0, ptr %dim0.addr, align 8 %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %common_array1 = bitcast ptr addrspace(4) %this1 to ptr addrspace(4) %0 = load i64, ptr %dim0.addr, align 8 store i64 %0, ptr addrspace(4) %common_array1, align 8 ret void } define internal spir_func void @extract_bits(ptr addrspace(4) noundef align 8 dereferenceable_or_null(16) %this, ptr addrspace(4) noundef align 4 dereferenceable(4) %bits, ptr noundef byval(%t_range) align 8 %pos) { entry: %this.addr = alloca ptr addrspace(4), align 8 %bits.addr = alloca ptr addrspace(4), align 8 %Res = alloca i64, align 8 store ptr addrspace(4) %this, ptr %this.addr, align 8 store ptr addrspace(4) %bits, ptr %bits.addr, align 8 %pos.ascast = addrspacecast ptr %pos to ptr addrspace(4) %this1 = load ptr addrspace(4), ptr %this.addr, align 8 %Bits1 = bitcast ptr addrspace(4) %this1 to ptr addrspace(4) %0 = load i64, ptr addrspace(4) %Bits1, align 8 store i64 %0, ptr %Res, align 8 %bits_num = getelementptr inbounds nuw %subgr, ptr addrspace(4) %this1, i32 0, i32 1 %1 = load i64, ptr addrspace(4) %bits_num, align 8 %call = call spir_func noundef i64 @geti64() %2 = load i64, ptr %Res, align 8 %and = and i64 %2, %call store i64 %and, ptr %Res, align 8 %call2 = call spir_func noundef i64 @geti64() %call3 = call spir_func noundef i32 @geti32() %conv = zext i32 %call3 to i64 %cmp = icmp ult i64 %call2, %conv br i1 %cmp, label %if.then, label %if.else if.else: ; preds = %entry %3 = load ptr addrspace(4), ptr %bits.addr, align 8 store i32 0, ptr addrspace(4) %3, align 4 br label %if.end11 if.then: ; preds = %entry %call4 = call spir_func noundef i64 @geti64() %cmp5 = icmp ugt i64 %call4, 0 br i1 %cmp5, label %if.then6, label %if.end if.then6: ; preds = %if.then %call7 = call spir_func noundef i64 @geti64() %4 = load i64, ptr %Res, align 8 %shr = lshr i64 %4, %call7 store i64 %shr, ptr %Res, align 8 br label %if.end if.end: ; preds = %if.then6, %if.then %call8 = call spir_func noundef i64 @geti64() %5 = load i64, ptr %Res, align 8 %and9 = and i64 %5, %call8 store i64 %and9, ptr %Res, align 8 %6 = load i64, ptr %Res, align 8 %conv10 = trunc i64 %6 to i32 %7 = load ptr addrspace(4), ptr %bits.addr, align 8 store i32 %conv10, ptr addrspace(4) %7, align 4 br label %if.end11 if.end11: ; preds = %if.else, %if.end ret void }