blob: 7e0b6d6f00c4902e05fc28b25f8e03903452bf76 [file]
// RUN: mlir-translate -mlir-to-llvmir %s | FileCheck %s
// Regression test: taskloop loop bounds defined outside omp.taskloop must not
// leave the outlined loop body using the original function's casted step
// value. The outlined loop preheader has to reload the task's step from the
// task shared values structure.
omp.private {type = private} @_QFtestEi_private_i32 : i32
llvm.func @_QPtest() {
%c1_i64 = llvm.mlir.constant(1 : i64) : i64
%i = llvm.alloca %c1_i64 x i32 {bindc_name = "i"} : (i64) -> !llvm.ptr
%lb.addr = llvm.alloca %c1_i64 x i32 {bindc_name = "lb"} : (i64) -> !llvm.ptr
%ub.addr = llvm.alloca %c1_i64 x i32 {bindc_name = "ub"} : (i64) -> !llvm.ptr
%step.addr = llvm.alloca %c1_i64 x i32 {bindc_name = "step"} : (i64) -> !llvm.ptr
%c1_i32 = llvm.mlir.constant(1 : i32) : i32
%c10_i32 = llvm.mlir.constant(10 : i32) : i32
%c2_i32 = llvm.mlir.constant(2 : i32) : i32
llvm.store %c1_i32, %lb.addr : i32, !llvm.ptr
llvm.store %c10_i32, %ub.addr : i32, !llvm.ptr
llvm.store %c2_i32, %step.addr : i32, !llvm.ptr
%lb = llvm.load %lb.addr : !llvm.ptr -> i32
%ub = llvm.load %ub.addr : !llvm.ptr -> i32
%step = llvm.load %step.addr : !llvm.ptr -> i32
omp.taskloop.context private(@_QFtestEi_private_i32 %i -> %arg0 : !llvm.ptr) {
omp.taskloop.wrapper {
omp.loop_nest (%iv) : i32 = (%lb) to (%ub) inclusive step (%step) {
llvm.store %iv, %arg0 : i32, !llvm.ptr
omp.yield
}
}
omp.terminator
}
llvm.return
}
// CHECK-LABEL: define void @_QPtest() {
// CHECK: call void @__kmpc_taskloop(
// CHECK-LABEL: define internal void @_QPtest..omp_par(
// CHECK: %[[GEPSTEP:.*]] = getelementptr {{.*}}, ptr %{{.*}}, i32 0, i32 2
// CHECK: %[[LOADSTEP:.*]] = load i64, ptr %[[GEPSTEP]]
// CHECK: %[[SUB:.*]] = sub i64 %{{.*}}, %{{.*}}
// CHECK: %[[DIV:.*]] = sdiv i64 %[[SUB]], %[[LOADSTEP]]