Index: mlir/include/mlir/Dialect/LLVMIR/LLVMAttrDefs.td =================================================================== --- mlir/include/mlir/Dialect/LLVMIR/LLVMAttrDefs.td +++ mlir/include/mlir/Dialect/LLVMIR/LLVMAttrDefs.td @@ -274,11 +274,11 @@ let parameters = (ins "DIScopeAttr":$scope, "StringAttr":$name, - "DIFileAttr":$file, - "unsigned":$line, - "unsigned":$arg, - "unsigned":$alignInBits, - "DITypeAttr":$type + OptionalParameter<"DIFileAttr">:$file, + OptionalParameter<"unsigned">:$line, + OptionalParameter<"unsigned">:$arg, + OptionalParameter<"unsigned">:$alignInBits, + OptionalParameter<"DITypeAttr">:$type ); let builders = [ AttrBuilderWithInferredContext<(ins Index: mlir/include/mlir/Dialect/LLVMIR/LLVMIntrinsicOps.td =================================================================== --- mlir/include/mlir/Dialect/LLVMIR/LLVMIntrinsicOps.td +++ mlir/include/mlir/Dialect/LLVMIR/LLVMIntrinsicOps.td @@ -262,7 +262,7 @@ // Debug function intrinsics. // -class LLVM_DbgIntrOp : LLVM_Op { +class LLVM_DbgIntrOp : LLVM_IntrOp { let llvmBuilder = [{ llvm::Module *module = builder.GetInsertBlock()->getModule(); llvm::LLVMContext &ctx = module->getContext(); @@ -276,27 +276,34 @@ llvm::MetadataAsValue::get(ctx, llvm::DIExpression::get(ctx, llvm::None)), }); }]; + let mlirBuilder = [{ + // Drop all debug intrinsics with a non-empty debug expression. + // TODO: Stop dropping debug intrinsics once debug expressions are + // convertible to MLIR. + auto *dbgIntr = cast(inst); + if (dbgIntr->getExpression()->getNumElements() == 0) + $_builder.create<$_qualCppClassName>($_location, + $}] # argName # [{, $_var_attr($varInfo)); + }]; + let assemblyFormat = [{ + qualified($varInfo) `=` $}] # argName # + [{ `:` qualified(type($}] # argName # [{)) attr-dict + }]; } -def LLVM_DbgAddrOp : LLVM_DbgIntrOp<"dbg.addr"> { +def LLVM_DbgAddrOp : LLVM_DbgIntrOp<"dbg.addr", "addr"> { let summary = "Describe the current address of a local debug info variable."; let arguments = (ins LLVM_AnyPointer:$addr, LLVM_DILocalVariableAttr:$varInfo); - - let assemblyFormat = "qualified($varInfo) `=` $addr `:` type($addr) attr-dict"; } -def LLVM_DbgDeclareOp : LLVM_DbgIntrOp<"dbg.declare"> { +def LLVM_DbgDeclareOp : LLVM_DbgIntrOp<"dbg.declare", "addr"> { let summary = "Declare the address of a local debug info variable."; let arguments = (ins LLVM_AnyPointer:$addr, LLVM_DILocalVariableAttr:$varInfo); - - let assemblyFormat = "qualified($varInfo) `=` $addr `:` type($addr) attr-dict"; } -def LLVM_DbgValueOp : LLVM_DbgIntrOp<"dbg.value"> { +def LLVM_DbgValueOp : LLVM_DbgIntrOp<"dbg.value", "value"> { let summary = "Describe the current value of a local debug info variable."; let arguments = (ins LLVM_Type:$value, LLVM_DILocalVariableAttr:$varInfo); - - let assemblyFormat = "qualified($varInfo) `=` $value `:` type($value) attr-dict"; } // Index: mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td =================================================================== --- mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td +++ mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td @@ -234,6 +234,7 @@ // name matches the result name, by a reference to store the // result of the newly created MLIR operation to; // - $_int_attr - substituted by a call to an integer attribute matcher; + // - $_var_attr - substituted by a call to a variable attribute matcher; // - $_resultType - substituted with the MLIR result type; // - $_location - substituted with the MLIR location; // - $_builder - substituted with the MLIR builder; Index: mlir/lib/Target/LLVMIR/ConvertFromLLVMIR.cpp =================================================================== --- mlir/lib/Target/LLVMIR/ConvertFromLLVMIR.cpp +++ mlir/lib/Target/LLVMIR/ConvertFromLLVMIR.cpp @@ -372,7 +372,10 @@ SmallVector convertValues(ArrayRef values); /// Converts `value` to an integer attribute. Asserts if the conversion fails. - IntegerAttr matchIntegerAttr(Value value); + IntegerAttr matchIntegerAttr(llvm::Value *value); + + /// Converts `value` to a local variable attribute. + DILocalVariableAttr matchLocalVariableAttr(llvm::Value *value); /// Translates the debug location. Location translateLoc(llvm::DILocation *loc) { @@ -852,6 +855,12 @@ } Value Importer::convertValue(llvm::Value *value) { + // A value may be wrapped as metadata, for example, when passed to a debug + // intrinsic. Unwrap these values before the conversion. + if (auto *nodeAsVal = dyn_cast(value)) + if (auto *node = dyn_cast(nodeAsVal->getMetadata())) + value = node->getValue(); + // Return the mapped value if it has been converted before. if (valueMapping.count(value)) return lookupValue(value); @@ -872,14 +881,20 @@ return remapped; } -IntegerAttr Importer::matchIntegerAttr(Value value) { +IntegerAttr Importer::matchIntegerAttr(llvm::Value *value) { IntegerAttr integerAttr; - bool success = matchPattern(value, m_Constant(&integerAttr)); + bool success = matchPattern(convertValue(value), m_Constant(&integerAttr)); assert(success && "expected a constant value"); (void)success; return integerAttr; } +DILocalVariableAttr Importer::matchLocalVariableAttr(llvm::Value *value) { + auto *nodeAsVal = cast(value); + auto *node = cast(nodeAsVal->getMetadata()); + return debugImporter.translate(node); +} + LogicalResult Importer::convertBranchArgs(llvm::Instruction *branch, llvm::BasicBlock *target, SmallVectorImpl &blockArguments) { Index: mlir/test/Dialect/LLVMIR/debuginfo.mlir =================================================================== --- mlir/test/Dialect/LLVMIR/debuginfo.mlir +++ mlir/test/Dialect/LLVMIR/debuginfo.mlir @@ -81,16 +81,16 @@ file = #file, line = 4, scopeLine = 4, subprogramFlags = "Definition", type = #spType1 > -// CHECK-DAG: #[[VAR0:.*]] = #llvm.di_local_variable +// CHECK-DAG: #[[VAR0:.*]] = #llvm.di_local_variable #var0 = #llvm.di_local_variable< - scope = #sp0, name = "arg", file = #file, - line = 6, arg = 1, alignInBits = 0, type = #int0 + scope = #sp0, name = "alloc", file = #file, + line = 6, arg = 1, alignInBits = 32, type = #int0 > -// CHECK-DAG: #[[VAR1:.*]] = #llvm.di_local_variable +// CHECK-DAG: #[[VAR1:.*]] = #llvm.di_local_variable #var1 = #llvm.di_local_variable< - scope = #sp1, name = "arg", file = #file, - line = 7, arg = 2, alignInBits = 0, type = #int1 + // Omit the optional parameters. + scope = #sp1, name = "arg" > // CHECK: llvm.func @addr(%[[ARG:.*]]: i64) @@ -99,16 +99,16 @@ %allocCount = llvm.mlir.constant(1 : i32) : i32 %alloc = llvm.alloca %allocCount x i64 : (i32) -> !llvm.ptr - // CHECK: llvm.dbg.addr #[[VAR0]] = %[[ALLOC]] - // CHECK: llvm.dbg.declare #[[VAR0]] = %[[ALLOC]] - llvm.dbg.addr #var0 = %alloc : !llvm.ptr - llvm.dbg.declare #var0 = %alloc : !llvm.ptr + // CHECK: llvm.intr.dbg.addr #[[VAR0]] = %[[ALLOC]] + // CHECK: llvm.intr.dbg.declare #[[VAR0]] = %[[ALLOC]] + llvm.intr.dbg.addr #var0 = %alloc : !llvm.ptr + llvm.intr.dbg.declare #var0 = %alloc : !llvm.ptr llvm.return } // CHECK: llvm.func @value(%[[ARG:.*]]: i32) llvm.func @value(%arg: i32) -> i32 { - // CHECK: llvm.dbg.value #[[VAR1]] = %[[ARG]] - llvm.dbg.value #var1 = %arg : i32 + // CHECK: llvm.intr.dbg.value #[[VAR1]] = %[[ARG]] + llvm.intr.dbg.value #var1 = %arg : i32 llvm.return %arg : i32 } Index: mlir/test/Target/LLVMIR/Import/debug-info.ll =================================================================== --- mlir/test/Target/LLVMIR/Import/debug-info.ll +++ mlir/test/Target/LLVMIR/Import/debug-info.ll @@ -226,3 +226,43 @@ ; Verify the module location is set to the source filename. ; CHECK: loc("debug-info.ll":0:0) source_filename = "debug-info.ll" + +; // ----- + +; CHECK: #[[$SP:.+]] = #llvm.di_subprogram< +; CHECK: #[[$VAR0:.+]] = #llvm.di_local_variable +; CHECK: #[[$VAR1:.+]] = #llvm.di_local_variable + +; CHECK-LABEL: @intrinsic +; CHECK-SAME: %[[ARG0:[a-zA-Z0-9]+]] +; CHECK-SAME: %[[ARG1:[a-zA-Z0-9]+]] +define void @intrinsic(i64 %0, ptr %1) { + ; CHECK: llvm.intr.dbg.value #[[$VAR0]] = %[[ARG0]] : i64 loc(#[[LOC0:.+]]) + call void @llvm.dbg.value(metadata i64 %0, metadata !5, metadata !DIExpression()), !dbg !7 + ; CHECK: llvm.intr.dbg.addr #[[$VAR1]] = %[[ARG1]] : !llvm.ptr loc(#[[LOC1:.+]]) + call void @llvm.dbg.addr(metadata ptr %1, metadata !6, metadata !DIExpression()), !dbg !8 + ; CHECK: llvm.intr.dbg.declare #[[$VAR1]] = %[[ARG1]] : !llvm.ptr loc(#[[LOC2:.+]]) + call void @llvm.dbg.declare(metadata ptr %1, metadata !6, metadata !DIExpression()), !dbg !9 + ret void +} + +; CHECK: #[[LOC0]] = loc(fused<#[[$SP]]>["debug-info.ll":1:2]) +; CHECK: #[[LOC1]] = loc(fused<#[[$SP]]>["debug-info.ll":2:2]) +; CHECK: #[[LOC2]] = loc(fused<#[[$SP]]>["debug-info.ll":3:2]) + +declare void @llvm.dbg.value(metadata, metadata, metadata) +declare void @llvm.dbg.addr(metadata, metadata, metadata) +declare void @llvm.dbg.declare(metadata, metadata, metadata) + +!llvm.dbg.cu = !{!1} +!llvm.module.flags = !{!0} +!0 = !{i32 2, !"Debug Info Version", i32 3} +!1 = distinct !DICompileUnit(language: DW_LANG_C, file: !2) +!2 = !DIFile(filename: "debug-info.ll", directory: "/") +!3 = distinct !DISubprogram(name: "intrinsic", scope: !2, file: !2, line: 42, scopeLine: 42, spFlags: DISPFlagDefinition, unit: !1) +!4 = !DIBasicType(name: "int") +!5 = !DILocalVariable(scope: !3, name: "arg", file: !2, line: 1, arg: 1, align: 32, type: !4); +!6 = !DILocalVariable(scope: !3, name: "arg") +!7 = !DILocation(line: 1, column: 2, scope: !3) +!8 = !DILocation(line: 2, column: 2, scope: !3) +!9 = !DILocation(line: 3, column: 2, scope: !3) Index: mlir/test/Target/LLVMIR/llvmir-debug.mlir =================================================================== --- mlir/test/Target/LLVMIR/llvmir-debug.mlir +++ mlir/test/Target/LLVMIR/llvmir-debug.mlir @@ -64,6 +64,7 @@ > #fileScope = #llvm.di_lexical_block_file #variable = #llvm.di_local_variable +#variableAddr = #llvm.di_local_variable // CHECK-LABEL: define void @func_with_debug( // CHECK-SAME: i64 %[[ARG:.*]]) !dbg ![[FUNC_LOC:[0-9]+]] @@ -73,11 +74,11 @@ %alloc = llvm.alloca %allocCount x i64 : (i32) -> !llvm.ptr // CHECK: call void @llvm.dbg.value(metadata i64 %[[ARG]], metadata ![[VAR_LOC:[0-9]+]], metadata !DIExpression()) - // CHECK: call void @llvm.dbg.addr(metadata ptr %[[ALLOC]], metadata ![[VAR_LOC]], metadata !DIExpression()) - // CHECK: call void @llvm.dbg.declare(metadata ptr %[[ALLOC]], metadata ![[VAR_LOC]], metadata !DIExpression()) - llvm.dbg.value #variable = %arg : i64 - llvm.dbg.addr #variable = %alloc : !llvm.ptr - llvm.dbg.declare #variable = %alloc : !llvm.ptr + // CHECK: call void @llvm.dbg.addr(metadata ptr %[[ALLOC]], metadata ![[ADDR_LOC:[0-9]+]], metadata !DIExpression()) + // CHECK: call void @llvm.dbg.declare(metadata ptr %[[ALLOC]], metadata ![[ADDR_LOC]], metadata !DIExpression()) + llvm.intr.dbg.value #variable = %arg : i64 + llvm.intr.dbg.addr #variableAddr = %alloc : !llvm.ptr + llvm.intr.dbg.declare #variableAddr = %alloc : !llvm.ptr // CHECK: call void @func_no_debug(), !dbg ![[CALLSITE_LOC:[0-9]+]] llvm.call @func_no_debug() : () -> () loc(callsite("mysource.cc":3:4 at "mysource.cc":5:6)) @@ -116,6 +117,7 @@ // CHECK: ![[VAR_LOC]] = !DILocalVariable(name: "arg", arg: 1, scope: ![[VAR_SCOPE:.*]], file: ![[CU_FILE_LOC]], line: 6, type: ![[ARG_TYPE]]) // CHECK: ![[VAR_SCOPE]] = distinct !DILexicalBlockFile(scope: ![[FUNC_LOC]], file: ![[CU_FILE_LOC]], discriminator: 0) +// CHECK: ![[ADDR_LOC]] = !DILocalVariable(name: "alloc", scope: ![[VAR_SCOPE:.*]]) // CHECK-DAG: ![[CALLSITE_LOC]] = !DILocation(line: 3, column: 4, // CHECK-DAG: ![[FILE_LOC]] = !DILocation(line: 1, column: 2, Index: mlir/tools/mlir-tblgen/LLVMIRConversionGen.cpp =================================================================== --- mlir/tools/mlir-tblgen/LLVMIRConversionGen.cpp +++ mlir/tools/mlir-tblgen/LLVMIRConversionGen.cpp @@ -237,18 +237,24 @@ return emitError( record, "expected non-negative operand index for argument " + name); } - bool isVariadicOperand = isVariadicOperandName(op, name); - auto result = - isVariadicOperand - ? formatv("convertValues(llvmOperands.drop_front({0}))", idx) - : formatv("convertValue(llvmOperands[{0}])", idx); - bs << result; + if (isAttributeName(op, name)) { + bs << formatv("llvmOperands[{0}]", idx); + } else { + bool isVariadicOperand = isVariadicOperandName(op, name); + auto result = + isVariadicOperand + ? formatv("convertValues(llvmOperands.drop_front({0}))", idx) + : formatv("convertValue(llvmOperands[{0}])", idx); + bs << result; + } } else if (isResultName(op, name)) { if (op.getNumResults() != 1) return emitError(record, "expected op to have one result"); bs << formatv("mapValue(inst)"); } else if (name == "_int_attr") { bs << "matchIntegerAttr"; + } else if (name == "_var_attr") { + bs << "matchLocalVariableAttr"; } else if (name == "_resultType") { bs << "convertType(inst->getType())"; } else if (name == "_location") { Index: openmp/libomptarget/test/api/omp_metadirective01.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective01.c @@ -0,0 +1,29 @@ +// RUN: %libomptarget-compile-run-and-check-generic + +#include +#include +#include + +void do_work(int *ptr, const int size) { + for (int i = 0; i < size; i ++) + ptr[i] = 1; +} + +int main() { + const int n = 1000; + const int buf_size = sizeof(int) * n; + const int dev = omp_get_default_device(); + int *ptr = (int *) malloc(buf_size); // possibly compiled on + + // Unified Shared Memory system + const int accessible = ptr != NULL ? 1 : 0; // omp_target_is_accessible(ptr, buf_size, dev); +#pragma omp metadirective when(user={condition(accessible)}: target firstprivate(ptr)) \ + default(target map(ptr[:n])) +{ + do_work(ptr, n); +} + +free(ptr); + +return 0; +} Index: openmp/libomptarget/test/api/omp_metadirective02.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective02.c @@ -0,0 +1,21 @@ +// RUN: %libomptarget-compile-run-and-check-generic + +#define N 100 +#include +#include +#include + +int main() { + double d[N]; + + float my_pi = 3.1415; + + for (int idev = 0; idev < omp_get_num_devices(); idev ++) { +#pragma omp target device(idev) +#pragma omp metadirective when(implementation={vendor(nvidia)}, device={arch("kepler")}: teams num_teams(512) thread_limit(32) map(tofrom: d[0:N])) when(implementation={vendor(amd)}, device={arch("fiji")}: teams num_teams(512) thread_limit(64) map(tofrom: d[0:N])) default(teams) + #pragma omp distribute parallel for + for (int i = 0; i < N; i ++) + d[i] = exp((M_PI - my_pi) * i); + } + return 0; +} Index: openmp/libomptarget/test/api/omp_metadirective03.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective03.c @@ -0,0 +1,59 @@ +// RUN: %libomptarget-compile-run-and-check-generic + +#include +#include +#include + +#define N 1048576 +enum Operation {MAX, MIN, NONE}; + +void set_monotonic_array(int a[], int size) { + for (int i = 0; i < size; i++) + a[i] = i; +} + +int test_kernel(Operation op) { + int a[N]; + set_monotonic_array(a, N); + int res = a[N/2]; + +#pragma omp metadirective \ + when (user = {condition(op == MAX)} : target teams distribute parallel for reduction(max: res)) \ + when (user = {condition(op == MIN)} : target teams distribute parallel for reduction(min: res)) \ + otherwise (parallel for private(res)) + for (int i = 0; i < N; i++) { + if (op == MAX) + res = (a[i] > res) ? a[i] : res; // Partial max (per-thread) + else if (op == MIN) + res = (a[i] < res) ? a[i] : res; // Partial min (per-thread) + else + res = -42; // No-op (assignment to private variable) + } + + int expc; + if (op == MAX) + expc = a[N-1]; + else if (op == MIN) + expc = a[0]; + else + expc = a[N/2]; + + if (res != expc) { + fprintf(stderr, "Error: Result = %d while expected = %d for operation = %d\n", res, expc, op); + return -1; + } else { + fprintf(stderr, "Pass: Result = expected = %d for operation = %d\n", res, op); + } + return 0; +} + +int main() { + if (test_kernel(MAX) == -1) + return -1; + if (test_kernel(MIN) == -1) + return -1; + if (test_kernel(NONE) == -1) + return -1; + + return 0; +} Index: openmp/libomptarget/test/api/omp_metadirective04.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective04.c @@ -0,0 +1,58 @@ +// RUN: %libomptarget-compile-run-and-check-generic + +#include +#include +#include + +#define N 1048576 + +void set_monotonic_array(int a[], int size) { + for (int i = 0; i < size; i++) + a[i] = i; +} + +int test_kernel() { + int a[N]; + set_monotonic_array(a, N); + int res = a[N/2]; + + omp_interop_t obj = omp_interop_none; + int dev = omp_get_default_device(); + +#pragma omp interop init(targetsync: obj) device(dev) + int id = (int) omp_get_interop_int(obj, omp_ipr_fr_id, NULL); + +#pragma omp metadirective \ + when (user = {condition(id == omp_ifr_cuda)} : target teams distribute parallel for reduction(max: res)) \ + when (user = {condition(id == omp_ifr_hip)} : target teams distribute parallel for reduction(min: res)) \ + otherwise (parallel for private(res)) + for (int i = 0; i < N; i++) { + if (id == omp_ifr_cuda) + res = (a[i] > res) ? a[i] : res; // Partial max (per-thread) + else if (id == omp_ifr_hip) + res = (a[i] < res) ? a[i] : res; // Partial min (per-thread) + else + res = -42; // No-op (assignment to private variable) + } + + int expc; + if (id == omp_ifr_cuda) + expc = a[N-1]; + else if (id == omp_ifr_hip) + expc = a[0]; + else + expc = a[N/2]; + + if (res != expc) { + fprintf(stderr, "Error: Result = %d while expected = %d for omp_get_interop_int() = %d\n", res, expc, id); + return -1; + } else { + fprintf(stderr, "Pass: Result = expected = %d for omp_get_interop_int() = %d\n", res, id); + } + return 0; +} + +int main() { + int status = test_kernel(); + return status; +} Index: openmp/libomptarget/test/api/omp_metadirective05.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective05.c @@ -0,0 +1,94 @@ +// RUN: %libomptarget-compile-run-and-check-generic + + +#include +#include +#include + +#define N 1048576 +enum Operation {MAX, MIN, NONE}; + +void set_monotonic_array(int a[], int size) { + for (int i = 0; i < size; i++) + a[i] = i; +} + +int test_kernel(Operation op) { + int a[N]; + set_monotonic_array(a, N); + int res1 = a[N/2]; + int res2 = a[N/2]; + + omp_interop_t obj = omp_interop_none; + int dev = omp_get_default_device(); + +#pragma omp interop init(targetsync: obj) device(dev) + int id = (int) omp_get_interop_int(obj, omp_ipr_fr_id, NULL); + +#pragma omp metadirective \ + when (user = {condition(id == omp_ifr_cuda && op == MAX)} : target teams distribute parallel for reduction(max: res1) reduction(max: res2)) \ + when (user = {condition(id == omp_ifr_cuda && op == MIN)} : target teams distribute parallel for reduction(max: res1) reduction(min: res2)) \ + when (user = {condition(id == omp_ifr_cuda && op == NONE)} : target teams distribute parallel for reduction(max: res1) private(res2)) \ + when (user = {condition(id == omp_ifr_hip && op == MAX)} : target teams distribute parallel for reduction(min: res1) reduction(max: res2)) \ + when (user = {condition(id == omp_ifr_hip && op == MIN)} : target teams distribute parallel for reduction(min: res1) reduction(min: res2)) \ + when (user = {condition(id == omp_ifr_hip && op == NONE)} : target teams distribute parallel for reduction(min: res1) private(res2)) \ + default (parallel for private(res1, res2)) + + for (int i = 0; i < N; i++) { + if (id == omp_ifr_cuda) + res1 = (a[i] > res1) ? a[i] : res1; // Partial max (per-thread) + else if (id == omp_ifr_hip) + res1 = (a[i] < res1) ? a[i] : res1; // Partial min (per-thread) + else + res1 = -42; // No-op (assignment to private variable) + + if (op == MAX) + res2 = (a[i] > res2) ? a[i] : res2; // Partial max (per-thread) + else if (op == MIN) + res2 = (a[i] < res2) ? a[i] : res2; // Partial min (per-thread) + else + res2 = -42; // No-op (assignment to private variable) + } + + int expc1; + if (id == omp_ifr_cuda) + expc1 = a[N-1]; + else if (id == omp_ifr_hip) + expc1 = a[0]; + else + expc1 = a[N/2]; + + if (res1 != expc1) { + fprintf(stderr, "Error: Result = %d while expected = %d for omp_get_interop_int() = %d\n", res1, expc1, id); + return -1; + } else { + fprintf(stderr, "Pass: Result = expected = %d for omp_get_interop_int() = %d\n", res1, id); + } + + int expc2; + if (op == MAX) + expc2 = a[N-1]; + else if (op == MIN) + expc2 = a[0]; + else + expc2 = a[N/2]; + + if (res2 != expc2) { + fprintf(stderr, "Error: Result = %d while expected = %d for operation = %d\n", res2, expc2, op); + return -1; + } else { + fprintf(stderr, "Pass: Result = expected = %d for operation = %d\n", res2, op); + } + return 0; +} + +int main() { + if (test_kernel(MAX) == -1) + return -1; + if (test_kernel(MIN) == -1) + return -1; + if (test_kernel(NONE) == -1) + return -1; + + return 0; +} Index: openmp/libomptarget/test/api/omp_metadirective06.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective06.c @@ -0,0 +1,85 @@ +// RUN: %libomptarget-compile-run-and-check-generic + + +#include +#include +#include + +#define N 1048576 +enum Operation {MAX, MIN, NONE}; + +void set_monotonic_array(int a[], int size) { + for (int i = 0; i < size; i++) + a[i] = i; +} + +int test_kernel(Operation op) { + int a[N]; + set_monotonic_array(a, N); + int res = a[N/2]; + + omp_interop_t obj; + int dev, id; + if (op == NONE) { + obj = omp_interop_none; + dev = omp_get_default_device(); + +#pragma omp interop init(targetsync: obj) device(dev) + } + + if (op == NONE) { + id = (int) omp_get_interop_int(obj, omp_ipr_fr_id, NULL); + } + +#pragma omp metadirective \ + when (user = {condition(op == MAX || op == NONE && id == omp_ifr_cuda)} : target teams distribute parallel for reduction(max: res)) \ + when (user = {condition(op == MIN || op == NONE && id == omp_ifr_hip)} : target teams distribute parallel for reduction(min: res)) \ + otherwise (parallel for private(res)) + for (int i = 0; i < N; i++) { + if (op == MAX || op == NONE && id == omp_ifr_cuda) + res = (a[i] > res) ? a[i] : res; // Partial max (per-thread) + else if (op == MIN || op == NONE && id == omp_ifr_hip) + res = (a[i] < res) ? a[i] : res; // Partial min (per-thread) + else + res = -42; // No-op (assignment to private variable) + } + + if (op == NONE) { +#pragma omp interop destroy(obj)) + } + + int expc; + if (op == MAX || op == NONE && id == omp_ifr_cuda) + expc = a[N-1]; + else if (op == MIN || op == NONE && id == omp_ifr_hip) + expc = a[0]; + else + expc = a[N/2]; + + printf("expc = %d res = %d\n ", expc, res); + + if (res != expc) { + if (op == NONE) + fprintf(stderr, "Error: Result = %d while expected = %d for operation = %d and omp_get_interop_int() = %d\n", res, expc, op, id); + else + fprintf(stderr, "Error: Result = %d while expected = %d for operation = %d\n", res, expc, op); + return -1; + } else { + if (op == NONE) + fprintf(stderr, "Pass: Result = expected = %d for operation = %d and omp_get_interop_int() = %d\n", res, op, id); + else + fprintf(stderr, "Pass: Result = expected = %d for operation = %d\n", res, op); + } + return 0; +} + +int main() { + if (test_kernel(MAX) == -1) + return -1; + if (test_kernel(MIN) == -1) + return -1; + if (test_kernel(NONE) == -1) + return -1; + + return 0; +} Index: openmp/libomptarget/test/api/omp_metadirective07.c =================================================================== --- /dev/null +++ openmp/libomptarget/test/api/omp_metadirective07.c @@ -0,0 +1,60 @@ +// RUN: %libomptarget-compile-run-and-check-generic + + +#include +#include +#include + +#define N 1048576 +enum Operation {MAX, MIN, NONE}; + +void set_monotonic_array(int a[], int size) { + for (int i = 0; i < size; i++) + a[i] = i; +} + +int test_kernel(Operation op) { + int a[N]; + set_monotonic_array(a, N); + int res = a[N/2]; + +#pragma omp metadirective \ + when (user = {condition(op == NONE)} : parallel for private(res)) \ + when (user = {condition(op == MIN)} : target teams distribute parallel for reduction(min: res)) \ + otherwise (target teams distribute parallel for reduction(max: res)) + for (int i = 0; i < N; i++) { + if (op == MAX) + res = (a[i] > res) ? a[i] : res; // Partial max (per-thread) + else if (op == MIN) + res = (a[i] < res) ? a[i] : res; // Partial min (per-thread) + else + res = -42; // No-op (assignment to private variable) + } + + int expc; + if (op == MAX) + expc = a[N-1]; + else if (op == MIN) + expc = a[0]; + else + expc = a[N/2]; + + if (res != expc) { + fprintf(stderr, "Error: Result = %d while expected = %d for operation = %d\n", res, expc, op); + return -1; + } else { + fprintf(stderr, "Pass: Result = expected = %d for operation = %d\n", res, op); + } + return 0; +} + +int main() { + if (test_kernel(MAX) == -1) + return -1; + if (test_kernel(MIN) == -1) + return -1; + if (test_kernel(NONE) == -1) + return -1; + + return 0; +}