diff options
author | Shraiysh <Shraiysh.Vaishay@amd.com> | 2023-10-11 11:36:03 -0400 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-10-11 10:36:03 -0500 |
commit | e41eaf4896d96b5e9c81959130cc61d453f471fa (patch) | |
tree | 98d106b05ee498be7efca618ac0e712107b98cdd /llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp | |
parent | 315ab3c44b56b94725d629137c6c402e676ecb04 (diff) | |
download | llvm-e41eaf4896d96b5e9c81959130cc61d453f471fa.zip llvm-e41eaf4896d96b5e9c81959130cc61d453f471fa.tar.gz llvm-e41eaf4896d96b5e9c81959130cc61d453f471fa.tar.bz2 |
[OpenMPIRBuilder] Add ThreadLimit and NumTeams clauses to teams construct (#68364)
This patch adds support for `thread_limit` and bounds on `num_teams`
clause for the teams construct in OpenMP.
Added testcases for the same.
Diffstat (limited to 'llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp')
-rw-r--r-- | llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp | 210 |
1 files changed, 210 insertions, 0 deletions
diff --git a/llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp b/llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp index c60aca2..d770fac 100644 --- a/llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp +++ b/llvm/unittests/Frontend/OpenMPIRBuilderTest.cpp @@ -4074,6 +4074,216 @@ TEST_F(OpenMPIRBuilderTest, CreateTeams) { [](Instruction &inst) { return isa<ICmpInst>(&inst); })); } +TEST_F(OpenMPIRBuilderTest, CreateTeamsWithThreadLimit) { + using InsertPointTy = OpenMPIRBuilder::InsertPointTy; + OpenMPIRBuilder OMPBuilder(*M); + OMPBuilder.initialize(); + F->setName("func"); + IRBuilder<> &Builder = OMPBuilder.Builder; + Builder.SetInsertPoint(BB); + + Function *FakeFunction = + Function::Create(FunctionType::get(Builder.getVoidTy(), false), + GlobalValue::ExternalLinkage, "fakeFunction", M.get()); + + auto BodyGenCB = [&](InsertPointTy AllocaIP, InsertPointTy CodeGenIP) { + Builder.restoreIP(CodeGenIP); + Builder.CreateCall(FakeFunction, {}); + }; + + // `F` has an argument - an integer, so we use that as the thread limit. + Builder.restoreIP(OMPBuilder.createTeams(/*=*/Builder, BodyGenCB, + /*NumTeamsLower=*/nullptr, + /*NumTeamsUpper=*/nullptr, + /*ThreadLimit=*/F->arg_begin())); + + Builder.CreateRetVoid(); + OMPBuilder.finalize(); + + ASSERT_FALSE(verifyModule(*M)); + + CallInst *PushNumTeamsCallInst = + findSingleCall(F, OMPRTL___kmpc_push_num_teams_51, OMPBuilder); + ASSERT_NE(PushNumTeamsCallInst, nullptr); + + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(2), Builder.getInt32(0)); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(3), Builder.getInt32(0)); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(4), &*F->arg_begin()); + + // Verifying that the next instruction to execute is kmpc_fork_teams + BranchInst *BrInst = + dyn_cast<BranchInst>(PushNumTeamsCallInst->getNextNonDebugInstruction()); + ASSERT_NE(BrInst, nullptr); + ASSERT_EQ(BrInst->getNumSuccessors(), 1U); + Instruction *NextInstruction = + BrInst->getSuccessor(0)->getFirstNonPHIOrDbgOrLifetime(); + CallInst *ForkTeamsCI = dyn_cast_if_present<CallInst>(NextInstruction); + ASSERT_NE(ForkTeamsCI, nullptr); + EXPECT_EQ(ForkTeamsCI->getCalledFunction(), + OMPBuilder.getOrCreateRuntimeFunctionPtr(OMPRTL___kmpc_fork_teams)); +} + +TEST_F(OpenMPIRBuilderTest, CreateTeamsWithNumTeamsUpper) { + using InsertPointTy = OpenMPIRBuilder::InsertPointTy; + OpenMPIRBuilder OMPBuilder(*M); + OMPBuilder.initialize(); + F->setName("func"); + IRBuilder<> &Builder = OMPBuilder.Builder; + Builder.SetInsertPoint(BB); + + Function *FakeFunction = + Function::Create(FunctionType::get(Builder.getVoidTy(), false), + GlobalValue::ExternalLinkage, "fakeFunction", M.get()); + + auto BodyGenCB = [&](InsertPointTy AllocaIP, InsertPointTy CodeGenIP) { + Builder.restoreIP(CodeGenIP); + Builder.CreateCall(FakeFunction, {}); + }; + + // `F` already has an integer argument, so we use that as upper bound to + // `num_teams` + Builder.restoreIP(OMPBuilder.createTeams(Builder, BodyGenCB, + /*NumTeamsLower=*/nullptr, + /*NumTeamsUpper=*/F->arg_begin())); + + Builder.CreateRetVoid(); + OMPBuilder.finalize(); + + ASSERT_FALSE(verifyModule(*M)); + + CallInst *PushNumTeamsCallInst = + findSingleCall(F, OMPRTL___kmpc_push_num_teams_51, OMPBuilder); + ASSERT_NE(PushNumTeamsCallInst, nullptr); + + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(2), &*F->arg_begin()); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(3), &*F->arg_begin()); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(4), Builder.getInt32(0)); + + // Verifying that the next instruction to execute is kmpc_fork_teams + BranchInst *BrInst = + dyn_cast<BranchInst>(PushNumTeamsCallInst->getNextNonDebugInstruction()); + ASSERT_NE(BrInst, nullptr); + ASSERT_EQ(BrInst->getNumSuccessors(), 1U); + Instruction *NextInstruction = + BrInst->getSuccessor(0)->getFirstNonPHIOrDbgOrLifetime(); + CallInst *ForkTeamsCI = dyn_cast_if_present<CallInst>(NextInstruction); + ASSERT_NE(ForkTeamsCI, nullptr); + EXPECT_EQ(ForkTeamsCI->getCalledFunction(), + OMPBuilder.getOrCreateRuntimeFunctionPtr(OMPRTL___kmpc_fork_teams)); +} + +TEST_F(OpenMPIRBuilderTest, CreateTeamsWithNumTeamsBoth) { + using InsertPointTy = OpenMPIRBuilder::InsertPointTy; + OpenMPIRBuilder OMPBuilder(*M); + OMPBuilder.initialize(); + F->setName("func"); + IRBuilder<> &Builder = OMPBuilder.Builder; + Builder.SetInsertPoint(BB); + + Function *FakeFunction = + Function::Create(FunctionType::get(Builder.getVoidTy(), false), + GlobalValue::ExternalLinkage, "fakeFunction", M.get()); + + Value *NumTeamsLower = + Builder.CreateAdd(F->arg_begin(), Builder.getInt32(5), "numTeamsLower"); + Value *NumTeamsUpper = + Builder.CreateAdd(F->arg_begin(), Builder.getInt32(10), "numTeamsUpper"); + + auto BodyGenCB = [&](InsertPointTy AllocaIP, InsertPointTy CodeGenIP) { + Builder.restoreIP(CodeGenIP); + Builder.CreateCall(FakeFunction, {}); + }; + + // `F` already has an integer argument, so we use that as upper bound to + // `num_teams` + Builder.restoreIP( + OMPBuilder.createTeams(Builder, BodyGenCB, NumTeamsLower, NumTeamsUpper)); + + Builder.CreateRetVoid(); + OMPBuilder.finalize(); + + ASSERT_FALSE(verifyModule(*M)); + + CallInst *PushNumTeamsCallInst = + findSingleCall(F, OMPRTL___kmpc_push_num_teams_51, OMPBuilder); + ASSERT_NE(PushNumTeamsCallInst, nullptr); + + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(2), NumTeamsLower); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(3), NumTeamsUpper); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(4), Builder.getInt32(0)); + + // Verifying that the next instruction to execute is kmpc_fork_teams + BranchInst *BrInst = + dyn_cast<BranchInst>(PushNumTeamsCallInst->getNextNonDebugInstruction()); + ASSERT_NE(BrInst, nullptr); + ASSERT_EQ(BrInst->getNumSuccessors(), 1U); + Instruction *NextInstruction = + BrInst->getSuccessor(0)->getFirstNonPHIOrDbgOrLifetime(); + CallInst *ForkTeamsCI = dyn_cast_if_present<CallInst>(NextInstruction); + ASSERT_NE(ForkTeamsCI, nullptr); + EXPECT_EQ(ForkTeamsCI->getCalledFunction(), + OMPBuilder.getOrCreateRuntimeFunctionPtr(OMPRTL___kmpc_fork_teams)); +} + +TEST_F(OpenMPIRBuilderTest, CreateTeamsWithNumTeamsAndThreadLimit) { + using InsertPointTy = OpenMPIRBuilder::InsertPointTy; + OpenMPIRBuilder OMPBuilder(*M); + OMPBuilder.initialize(); + F->setName("func"); + IRBuilder<> &Builder = OMPBuilder.Builder; + Builder.SetInsertPoint(BB); + + BasicBlock *CodegenBB = splitBB(Builder, true); + Builder.SetInsertPoint(CodegenBB); + + // Generate values for `num_teams` and `thread_limit` using the first argument + // of the testing function. + Value *NumTeamsLower = + Builder.CreateAdd(F->arg_begin(), Builder.getInt32(5), "numTeamsLower"); + Value *NumTeamsUpper = + Builder.CreateAdd(F->arg_begin(), Builder.getInt32(10), "numTeamsUpper"); + Value *ThreadLimit = + Builder.CreateAdd(F->arg_begin(), Builder.getInt32(20), "threadLimit"); + + Function *FakeFunction = + Function::Create(FunctionType::get(Builder.getVoidTy(), false), + GlobalValue::ExternalLinkage, "fakeFunction", M.get()); + + auto BodyGenCB = [&](InsertPointTy AllocaIP, InsertPointTy CodeGenIP) { + Builder.restoreIP(CodeGenIP); + Builder.CreateCall(FakeFunction, {}); + }; + + OpenMPIRBuilder::LocationDescription Loc({Builder.saveIP(), DL}); + Builder.restoreIP(OMPBuilder.createTeams(Builder, BodyGenCB, NumTeamsLower, + NumTeamsUpper, ThreadLimit)); + + Builder.CreateRetVoid(); + OMPBuilder.finalize(); + + ASSERT_FALSE(verifyModule(*M)); + + CallInst *PushNumTeamsCallInst = + findSingleCall(F, OMPRTL___kmpc_push_num_teams_51, OMPBuilder); + ASSERT_NE(PushNumTeamsCallInst, nullptr); + + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(2), NumTeamsLower); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(3), NumTeamsUpper); + EXPECT_EQ(PushNumTeamsCallInst->getArgOperand(4), ThreadLimit); + + // Verifying that the next instruction to execute is kmpc_fork_teams + BranchInst *BrInst = + dyn_cast<BranchInst>(PushNumTeamsCallInst->getNextNonDebugInstruction()); + ASSERT_NE(BrInst, nullptr); + ASSERT_EQ(BrInst->getNumSuccessors(), 1U); + Instruction *NextInstruction = + BrInst->getSuccessor(0)->getFirstNonPHIOrDbgOrLifetime(); + CallInst *ForkTeamsCI = dyn_cast_if_present<CallInst>(NextInstruction); + ASSERT_NE(ForkTeamsCI, nullptr); + EXPECT_EQ(ForkTeamsCI->getCalledFunction(), + OMPBuilder.getOrCreateRuntimeFunctionPtr(OMPRTL___kmpc_fork_teams)); +} + /// Returns the single instruction of InstTy type in BB that uses the value V. /// If there is more than one such instruction, returns null. template <typename InstTy> |