Skip to content

Commit

Permalink
[AbstractCallSite] Look though constant cast expression when checking…
Browse files Browse the repository at this point in the history
… for callee use

Summary: That makes AbstractCallSite::isCallee(const Use *) behavior consistent with AbstractCallSite constructor.

Reviewers: jdoerfert

Reviewed By: jdoerfert

Subscribers: mgorny, llvm-commits

Tags: #llvm

Differential Revision: https://reviews.llvm.org/D79188
  • Loading branch information
sndmitriev committed Apr 30, 2020
1 parent 0d61dcf commit cfea3dc
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 0 deletions.
6 changes: 6 additions & 0 deletions llvm/include/llvm/IR/AbstractCallSite.h
Expand Up @@ -141,6 +141,12 @@ class AbstractCallSite {
assert(!CI.ParameterEncoding.empty() &&
"Callback without parameter encoding!");

// If the use is actually in a constant cast expression which itself
// has only one use, we look through the constant cast expression.
if (auto *CE = dyn_cast<ConstantExpr>(U->getUser()))
if (CE->getNumUses() == 1 && CE->isCast())
U = &*CE->use_begin();

return (int)CB->getArgOperandNo(U) == CI.ParameterEncoding[0];
}

Expand Down
55 changes: 55 additions & 0 deletions llvm/unittests/IR/AbstractCallSiteTest.cpp
@@ -0,0 +1,55 @@
//===----- 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/AsmParser/Parser.h"
#include "llvm/IR/AbstractCallSite.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<Module> parseIR(LLVMContext &C, const char *IR) {
SMDiagnostic Err;
std::unique_ptr<Module> 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<Module> 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.getCalledFunction(), Callback);
}
1 change: 1 addition & 0 deletions llvm/unittests/IR/CMakeLists.txt
Expand Up @@ -8,6 +8,7 @@ set(LLVM_LINK_COMPONENTS
)

add_llvm_unittest(IRTests
AbstractCallSiteTest.cpp
AsmWriterTest.cpp
AttributesTest.cpp
BasicBlockTest.cpp
Expand Down

0 comments on commit cfea3dc

Please sign in to comment.