From 7247c0a42119931ca6f1fb393f89fe5e26aef5a2 Mon Sep 17 00:00:00 2001 From: Dmitry Venikov Date: Mon, 15 Jul 2019 14:47:45 +0000 Subject: [PATCH] [PatternMatch] Implement matching code for LibFunc Summary: Provides m_LibFunc pattern that can be used to match LibFuncs. Reviewers: spatel, hfinkel, efriedma, lebedev.ri Reviewed By: lebedev.ri Subscribers: lebedev.ri, llvm-commits Tags: #llvm Differential Revision: https://reviews.llvm.org/D42047 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@366069 91177308-0d34-0410-b5e6-96231b3b80d8 --- include/llvm/IR/PatternMatch.h | 52 ++++++++++++++++++++++++++++++++++ unittests/IR/PatternMatch.cpp | 38 ++++++++++++++++++++++++- 2 files changed, 89 insertions(+), 1 deletion(-) diff --git a/include/llvm/IR/PatternMatch.h b/include/llvm/IR/PatternMatch.h index 0f03d7cc56b..b45a6b0bc8b 100644 --- a/include/llvm/IR/PatternMatch.h +++ b/include/llvm/IR/PatternMatch.h @@ -30,6 +30,7 @@ #include "llvm/ADT/APFloat.h" #include "llvm/ADT/APInt.h" +#include "llvm/Analysis/TargetLibraryInfo.h" #include "llvm/IR/Constant.h" #include "llvm/IR/Constants.h" #include "llvm/IR/InstrTypes.h" @@ -1812,6 +1813,57 @@ template inline Signum_match m_Signum(const Val_t &V) { return Signum_match(V); } +/// \brief LibFunc matchers. +struct LibFunc_match { + LibFunc F; + TargetLibraryInfo TLI; + + LibFunc_match(LibFunc Func, TargetLibraryInfo TargetLI) + : F(Func), TLI(TargetLI) {} + + template bool match(OpTy *V) { + LibFunc LF; + if (const auto *CI = dyn_cast(V)) + if (!CI->isNoBuiltin() && CI->getCalledFunction() && + TLI.getLibFunc(*CI->getCalledFunction(), LF) && + LF == F && TLI.has(LF)) + return true; + return false; + } +}; + +/// LibFunc matches are combinations of Name matchers, and argument +/// matchers. +template +struct m_LibFunc_Ty; +template struct m_LibFunc_Ty { + using Ty = match_combine_and>; +}; +template struct m_LibFunc_Ty { + using Ty = + match_combine_and::Ty, + Argument_match>; +}; + +/// \brief Match LibFunc calls like this: +/// m_LibFunc(m_Value(X)) +template +inline LibFunc_match m_LibFunc(TargetLibraryInfo TLI) { + return LibFunc_match(F, TLI); +} + +template +inline typename m_LibFunc_Ty::Ty +m_LibFunc(const TargetLibraryInfo TLI, const T0 &Op0) { + return m_CombineAnd(m_LibFunc(TLI), m_Argument<0>(Op0)); +} + +template +inline typename m_LibFunc_Ty::Ty +m_LibFunc(const TargetLibraryInfo TLI, const T0 &Op0, const T1 &Op1) { + return m_CombineAnd(m_LibFunc(TLI, Op0), m_Argument<1>(Op1)); +} + } // end namespace PatternMatch } // end namespace llvm diff --git a/unittests/IR/PatternMatch.cpp b/unittests/IR/PatternMatch.cpp index 600494fba26..3f8e6918019 100644 --- a/unittests/IR/PatternMatch.cpp +++ b/unittests/IR/PatternMatch.cpp @@ -35,13 +35,15 @@ struct PatternMatchTest : ::testing::Test { Function *F; BasicBlock *BB; IRBuilder IRB; + TargetLibraryInfoImpl TLII; + TargetLibraryInfo TLI; PatternMatchTest() : M(new Module("PatternMatchTestModule", Ctx)), F(Function::Create( FunctionType::get(Type::getVoidTy(Ctx), /* IsVarArg */ false), Function::ExternalLinkage, "f", M.get())), - BB(BasicBlock::Create(Ctx, "entry", F)), IRB(BB) {} + BB(BasicBlock::Create(Ctx, "entry", F)), IRB(BB), TLI(TLII) {} }; TEST_F(PatternMatchTest, OneUse) { @@ -1008,6 +1010,40 @@ TEST_F(PatternMatchTest, FloatingPointFNeg) { EXPECT_FALSE(match(V3, m_FNeg(m_Value(Match)))); } +TEST_F(PatternMatchTest, LibFunc) { + Type *FltTy = IRB.getFloatTy(); + Value *One = ConstantFP::get(FltTy, 1.0); + Value *Two = ConstantFP::get(FltTy, 2.0); + Value *MatchOne, *MatchTwo; + + StringRef TanName = TLI.getName(LibFunc_tan); + FunctionCallee TanCallee = M->getOrInsertFunction(TanName, FltTy, FltTy); + CallInst *Tan = IRB.CreateCall(TanCallee, One, TanName); + + StringRef PowName = TLI.getName(LibFunc_pow); + FunctionCallee PowCallee = M->getOrInsertFunction(PowName, FltTy, FltTy, FltTy); + CallInst *Pow = IRB.CreateCall(PowCallee, {One, Two}, PowName); + + EXPECT_TRUE(match(Tan, m_LibFunc(TLI))); + EXPECT_FALSE(match(Tan, m_LibFunc(TLI))); + EXPECT_FALSE(match(Pow, m_LibFunc(TLI))); + + EXPECT_TRUE(match(Tan, m_LibFunc(TLI, m_Value(MatchOne)))); + EXPECT_EQ(One, MatchOne); + EXPECT_FALSE(match(Tan, m_LibFunc(TLI, m_Value()))); + + EXPECT_TRUE(match(Pow, m_LibFunc(TLI, m_Value(MatchOne), + m_Value(MatchTwo)))); + EXPECT_EQ(One, MatchOne); + EXPECT_EQ(Two, MatchTwo); + EXPECT_FALSE(match(Pow, m_LibFunc(TLI, m_Value(), m_Value()))); + + TLII.disableAllFunctions(); + EXPECT_FALSE(match(Tan, m_LibFunc(TLI))); + EXPECT_FALSE(match(Tan, m_LibFunc(TLI, m_Value()))); + EXPECT_FALSE(match(Pow, m_LibFunc(TLI, m_Value(), m_Value()))); +} + template struct MutableConstTest : PatternMatchTest { }; typedef ::testing::Types, -- 2.40.0