1
0
mirror of https://github.com/RPCS3/llvm-mirror.git synced 2024-11-22 10:42:39 +01:00

Implement -fsemantic-interposition

First attempt at implementing -fsemantic-interposition.

Rely on GlobalValue::isInterposable that already captures most of the expected
behavior.

Rely on a ModuleFlag to state whether we should respect SemanticInterposition or
not. The default remains no.

So this should be a no-op if -fsemantic-interposition isn't used, and if it is,
isInterposable being already used in most optimisation, they should honor it
properly.

Note that it only impacts architecture compiled with -fPIC and no pie.

Differential Revision: https://reviews.llvm.org/D72829
This commit is contained in:
serge-sans-paille 2020-01-16 11:56:41 +01:00
parent e961032e7d
commit 1edc654103
7 changed files with 76 additions and 4 deletions

View File

@ -423,10 +423,10 @@ public:
}
/// Return true if this global's definition can be substituted with an
/// *arbitrary* definition at link time. We cannot do any IPO or inlinining
/// across interposable call edges, since the callee can be replaced with
/// something arbitrary at link time.
bool isInterposable() const { return isInterposableLinkage(getLinkage()); }
/// *arbitrary* definition at link time or load time. We cannot do any IPO or
/// inlining across interposable call edges, since the callee can be
/// replaced with something arbitrary.
bool isInterposable() const;
bool hasExternalLinkage() const { return isExternalLinkage(getLinkage()); }
bool hasAvailableExternallyLinkage() const {

View File

@ -848,6 +848,12 @@ public:
Metadata *getProfileSummary(bool IsCS);
/// @}
/// Returns whether semantic interposition is to be respected.
bool getSemanticInterposition() const;
/// Set whether semantic interposition is to be respected.
void setSemanticInterposition(bool);
/// Returns true if PLT should be avoided for RTLib calls.
bool getRtLibUseGOT() const;

View File

@ -94,6 +94,13 @@ void GlobalValue::eraseFromParent() {
llvm_unreachable("not a global");
}
bool GlobalValue::isInterposable() const {
if (isInterposableLinkage(getLinkage()))
return true;
return getParent() && getParent()->getSemanticInterposition() &&
!isDSOLocal();
}
unsigned GlobalValue::getAlignment() const {
if (auto *GA = dyn_cast<GlobalAlias>(this)) {
// In general we cannot compute this at the IR level, but we try.

View File

@ -555,6 +555,20 @@ Metadata *Module::getProfileSummary(bool IsCS) {
: getModuleFlag("ProfileSummary"));
}
bool Module::getSemanticInterposition() const {
Metadata *MF = getModuleFlag("SemanticInterposition");
auto *Val = cast_or_null<ConstantAsMetadata>(MF);
if (!Val)
return false;
return cast<ConstantInt>(Val->getValue())->getZExtValue();
}
void Module::setSemanticInterposition(bool SI) {
addModuleFlag(ModFlagBehavior::Error, "SemanticInterposition", SI);
}
void Module::setOwnedMemoryBuffer(std::unique_ptr<MemoryBuffer> MB) {
OwnedMemoryBuffer = std::move(MB);
}

View File

@ -1476,6 +1476,13 @@ Verifier::visitModuleFlag(const MDNode *Op,
"'Linker Options' named metadata no longer supported");
}
if (ID->getString() == "SemanticInterposition") {
ConstantInt *Value =
mdconst::dyn_extract_or_null<ConstantInt>(Op->getOperand(2));
Assert(Value,
"SemanticInterposition metadata requires constant integer argument");
}
if (ID->getString() == "CG Profile") {
for (const MDOperand &MDO : cast<MDNode>(Op->getOperand(2))->operands())
visitModuleFlagCGProfileEntry(MDO);

View File

@ -0,0 +1,26 @@
; Check that @callee1 gets inlined while @callee2 is not, because of
; SemanticInterposition.
; RUN: opt < %s -inline -S | FileCheck %s
define internal i32 @callee1(i32 %A) {
ret i32 %A
}
define i32 @callee2(i32 %A) {
ret i32 %A
}
; CHECK-LABEL: @caller
define i32 @caller(i32 %A) {
; CHECK-NOT: call i32 @callee1(i32 %A)
%A1 = call i32 @callee1(i32 %A)
; CHECK: %A2 = call i32 @callee2(i32 %A)
%A2 = call i32 @callee2(i32 %A)
; CHECK: add i32 %A, %A2
%R = add i32 %A1, %A2
ret i32 %R
}
!llvm.module.flags = !{!0}
!0 = !{i32 1, !"SemanticInterposition", i32 1}

View File

@ -0,0 +1,12 @@
; RUN: not llvm-as < %s -o /dev/null 2>&1 | FileCheck %s
target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128"
target triple = "x86_64-unknown-linux-gnu"
@foo = dso_local global i32 1, align 4
!llvm.module.flags = !{!0}
!0 = !{i32 1, !"SemanticInterposition", float 1.}
; CHECK: SemanticInterposition metadata requires constant integer argument