Option to ignore assume like intrinsic uses in hasAddressTaken()

Differential Revision: https://reviews.llvm.org/D96081
This commit is contained in:
Stanislav Mekhanoshin
2021-02-04 14:38:40 -08:00
parent 7f6e331645
commit 29e2d9461a
4 changed files with 47 additions and 6 deletions

View File

@@ -872,11 +872,12 @@ public:
/// hasAddressTaken - returns true if there are any uses of this function
/// other than direct calls or invokes to it, or blockaddress expressions.
/// Optionally passes back an offending user for diagnostic purposes and
/// ignores callback uses.
/// Optionally passes back an offending user for diagnostic purposes,
/// ignores callback uses, and assume like pointer annotation calls.
///
bool hasAddressTaken(const User ** = nullptr,
bool IgnoreCallbackUses = false) const;
bool IgnoreCallbackUses = false,
bool IgnoreAssumeLikeCalls = false) const;
/// isDefTriviallyDead - Return true if it is trivially safe to remove
/// this function definition from the module (because it isn't externally

View File

@@ -80,7 +80,8 @@ void CallGraph::addToCallGraph(Function *F) {
// If this function has external linkage or has its address taken and
// it is not a callback, then anything could call it.
if (!F->hasLocalLinkage() ||
F->hasAddressTaken(nullptr, /*IgnoreCallbackUses=*/true))
F->hasAddressTaken(nullptr, /*IgnoreCallbackUses=*/true,
/* IgnoreAssumeLikeCalls */ true))
ExternalCallingNode->addCalledFunction(nullptr, Node);
populateCallGraphNode(Node);

View File

@@ -31,6 +31,7 @@
#include "llvm/IR/InstIterator.h"
#include "llvm/IR/Instruction.h"
#include "llvm/IR/Instructions.h"
#include "llvm/IR/IntrinsicInst.h"
#include "llvm/IR/Intrinsics.h"
#include "llvm/IR/IntrinsicsAArch64.h"
#include "llvm/IR/IntrinsicsAMDGPU.h"
@@ -1578,9 +1579,10 @@ Optional<Function *> Intrinsic::remangleIntrinsicFunction(Function *F) {
/// hasAddressTaken - returns true if there are any uses of this function
/// other than direct calls or invokes to it. Optionally ignores callback
/// uses.
/// uses and assume like pointer annotation calls.
bool Function::hasAddressTaken(const User **PutOffender,
bool IgnoreCallbackUses) const {
bool IgnoreCallbackUses,
bool IgnoreAssumeLikeCalls) const {
for (const Use &U : uses()) {
const User *FU = U.getUser();
if (isa<BlockAddress>(FU))
@@ -1594,6 +1596,17 @@ bool Function::hasAddressTaken(const User **PutOffender,
const auto *Call = dyn_cast<CallBase>(FU);
if (!Call) {
if (IgnoreAssumeLikeCalls) {
if (const auto *FI = dyn_cast<Instruction>(FU)) {
if (FI->isCast() && !FI->user_empty() &&
llvm::all_of(FU->users(), [](const User *U) {
if (const auto *I = dyn_cast<IntrinsicInst>(U))
return I->isAssumeLikeIntrinsic();
return false;
}))
continue;
}
}
if (PutOffender)
*PutOffender = FU;
return true;

View File

@@ -0,0 +1,26 @@
; RUN: opt < %s -print-callgraph -disable-output 2>&1 | FileCheck %s
; CHECK: Call graph node <<null function>><<{{.*}}>> #uses=0
; CHECK-NEXT: CS<None> calls function 'cast_only'
; CHECK-NEXT: CS<None> calls function 'llvm.lifetime.start.p0i8'
; CHECK-EMPTY:
; CHECK-NEXT: Call graph node for function: 'cast_only'<<{{.*}}>> #uses=1
; CHECK-EMPTY:
; CHECK-NEXT: Call graph node for function: 'llvm.lifetime.start.p0i8'<<{{.*}}>> #uses=1
; CHECK-EMPTY:
; CHECK-NEXT: Call graph node for function: 'used_by_lifetime'<<{{.*}}>> #uses=0
; CHECK-EMPTY:
define internal void @used_by_lifetime() {
entry:
%c = bitcast void()* @used_by_lifetime to i8*
call void @llvm.lifetime.start.p0i8(i64 4, i8* %c)
ret void
}
define internal void @cast_only() {
entry:
%c = bitcast void()* @cast_only to i8*
ret void
}
declare void @llvm.lifetime.start.p0i8(i64, i8* nocapture)