//===----- AbstractCallSiteTest.cpp - AbstractCallSite Unittests ----------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "llvm/IR/AbstractCallSite.h" #include "llvm/AsmParser/Parser.h" #include "llvm/IR/Argument.h" #include "llvm/IR/Function.h" #include "llvm/IR/Module.h" #include "llvm/Support/SourceMgr.h" #include "gtest/gtest.h" using namespace llvm; static std::unique_ptr parseIR(LLVMContext &C, const char *IR) { SMDiagnostic Err; std::unique_ptr Mod = parseAssemblyString(IR, Err, C); if (!Mod) Err.print("AbstractCallSiteTests", errs()); return Mod; } TEST(AbstractCallSite, CallbackCall) { LLVMContext C; const char *IR = "define void @callback(i8* %X, i32* %A) {\n" " ret void\n" "}\n" "define void @foo(i32* %A) {\n" " call void (i32, void (i8*, ...)*, ...) @broker(i32 1, void (i8*, ...)* bitcast (void (i8*, i32*)* @callback to void (i8*, ...)*), i32* %A)\n" " ret void\n" "}\n" "declare !callback !0 void @broker(i32, void (i8*, ...)*, ...)\n" "!0 = !{!1}\n" "!1 = !{i64 1, i64 -1, i1 true}"; std::unique_ptr M = parseIR(C, IR); ASSERT_TRUE(M); Function *Callback = M->getFunction("callback"); ASSERT_NE(Callback, nullptr); const Use *CallbackUse = Callback->getSingleUndroppableUse(); ASSERT_NE(CallbackUse, nullptr); AbstractCallSite ACS(CallbackUse); EXPECT_TRUE(ACS); EXPECT_TRUE(ACS.isCallbackCall()); EXPECT_TRUE(ACS.isCallee(CallbackUse)); EXPECT_EQ(ACS.getCalleeUseForCallback(), *CallbackUse); EXPECT_EQ(ACS.getCalledFunction(), Callback); // The callback metadata {CallbackNo, Arg0No, ..., isVarArg} = {1, -1, true} EXPECT_EQ(ACS.getCallArgOperandNoForCallee(), 1); // Though the callback metadata only specifies ONE unfixed argument No, the // callback callee is vararg, hence the third arg is also considered as // another arg for the callback. EXPECT_EQ(ACS.getNumArgOperands(), 2u); Argument *Param0 = Callback->getArg(0), *Param1 = Callback->getArg(1); ASSERT_TRUE(Param0 && Param1); EXPECT_EQ(ACS.getCallArgOperandNo(*Param0), -1); EXPECT_EQ(ACS.getCallArgOperandNo(*Param1), 2); } TEST(AbstractCallSite, DirectCall) { LLVMContext C; const char *IR = "declare void @bar(i32 %x, i32 %y)\n" "define void @foo() {\n" " call void @bar(i32 1, i32 2)\n" " ret void\n" "}\n"; std::unique_ptr M = parseIR(C, IR); ASSERT_TRUE(M); Function *Callee = M->getFunction("bar"); ASSERT_NE(Callee, nullptr); const Use *DirectCallUse = Callee->getSingleUndroppableUse(); ASSERT_NE(DirectCallUse, nullptr); AbstractCallSite ACS(DirectCallUse); EXPECT_TRUE(ACS); EXPECT_TRUE(ACS.isDirectCall()); EXPECT_TRUE(ACS.isCallee(DirectCallUse)); EXPECT_EQ(ACS.getCalledFunction(), Callee); EXPECT_EQ(ACS.getNumArgOperands(), 2u); Argument *ArgX = Callee->getArg(0); ASSERT_NE(ArgX, nullptr); Value *CAO1 = ACS.getCallArgOperand(*ArgX); Value *CAO2 = ACS.getCallArgOperand(0); ASSERT_NE(CAO2, nullptr); // The two call arg operands should be the same object, since they are both // the first argument of the call. EXPECT_EQ(CAO2, CAO1); ConstantInt *FirstArgInt = dyn_cast(CAO2); ASSERT_NE(FirstArgInt, nullptr); EXPECT_EQ(FirstArgInt->getZExtValue(), 1ull); EXPECT_EQ(ACS.getCallArgOperandNo(*ArgX), 0); EXPECT_EQ(ACS.getCallArgOperandNo(0), 0); EXPECT_EQ(ACS.getCallArgOperandNo(1), 1); } TEST(AbstractCallSite, IndirectCall) { LLVMContext C; const char *IR = "define void @foo(ptr %0) {\n" " call void %0(i32 1, i32 2)\n" " ret void\n" "}\n"; std::unique_ptr M = parseIR(C, IR); ASSERT_TRUE(M); Function *Fun = M->getFunction("foo"); ASSERT_NE(Fun, nullptr); Argument *ArgAsCallee = Fun->getArg(0); ASSERT_NE(ArgAsCallee, nullptr); const Use *IndCallUse = ArgAsCallee->getSingleUndroppableUse(); ASSERT_NE(IndCallUse, nullptr); AbstractCallSite ACS(IndCallUse); EXPECT_TRUE(ACS); EXPECT_TRUE(ACS.isIndirectCall()); EXPECT_TRUE(ACS.isCallee(IndCallUse)); EXPECT_EQ(ACS.getCalledFunction(), nullptr); EXPECT_EQ(ACS.getCalledOperand(), ArgAsCallee); EXPECT_EQ(ACS.getNumArgOperands(), 2u); Value *CalledOperand = ACS.getCallArgOperand(0); ASSERT_NE(CalledOperand, nullptr); ConstantInt *FirstArgInt = dyn_cast(CalledOperand); ASSERT_NE(FirstArgInt, nullptr); EXPECT_EQ(FirstArgInt->getZExtValue(), 1ull); EXPECT_EQ(ACS.getCallArgOperandNo(0), 0); EXPECT_EQ(ACS.getCallArgOperandNo(1), 1); }