From: Peter Collingbourne Date: Thu, 9 Mar 2017 01:11:15 +0000 (+0000) Subject: WholeProgramDevirt: Implement importing for uniform ret val opt. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=1166572ca3792f8c0d91a40d85396321e534429a;p=llvm WholeProgramDevirt: Implement importing for uniform ret val opt. Differential Revision: https://reviews.llvm.org/D29854 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@297350 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Transforms/IPO/WholeProgramDevirt.cpp b/lib/Transforms/IPO/WholeProgramDevirt.cpp index f321a8c8eb7..51d30f3c244 100644 --- a/lib/Transforms/IPO/WholeProgramDevirt.cpp +++ b/lib/Transforms/IPO/WholeProgramDevirt.cpp @@ -1126,6 +1126,24 @@ void DevirtModule::importResolution(VTableSlot Slot, VTableSlotInfo &SlotInfo) { applySingleImplDevirt(SlotInfo, SingleImpl, IsExported); assert(!IsExported); } + + for (auto &CSByConstantArg : SlotInfo.ConstCSInfo) { + auto I = Res.ResByArg.find(CSByConstantArg.first); + if (I == Res.ResByArg.end()) + continue; + auto &ResByArg = I->second; + // FIXME: We should figure out what to do about the "function name" argument + // to the apply* functions, as the function names are unavailable during the + // importing phase. For now we just pass the empty string. This does not + // impact correctness because the function names are just used for remarks. + switch (ResByArg.TheKind) { + case WholeProgramDevirtResolution::ByArg::UniformRetVal: + applyUniformRetValOpt(CSByConstantArg.second, "", ResByArg.Info); + break; + default: + break; + } + } } void DevirtModule::removeRedundantTypeTests() { diff --git a/test/Transforms/WholeProgramDevirt/Inputs/import-uniform-ret-val.yaml b/test/Transforms/WholeProgramDevirt/Inputs/import-uniform-ret-val.yaml new file mode 100644 index 00000000000..f1daae63b67 --- /dev/null +++ b/test/Transforms/WholeProgramDevirt/Inputs/import-uniform-ret-val.yaml @@ -0,0 +1,19 @@ +--- +TypeIdMap: + typeid1: + WPDRes: + 0: + Kind: Indir + ResByArg: + 1: + Kind: UniformRetVal + Info: 42 + typeid2: + WPDRes: + 8: + Kind: Indir + ResByArg: + 1: + Kind: UniformRetVal + Info: 42 +... diff --git a/test/Transforms/WholeProgramDevirt/import.ll b/test/Transforms/WholeProgramDevirt/import.ll index f0727b4c023..b54eeb7314a 100644 --- a/test/Transforms/WholeProgramDevirt/import.ll +++ b/test/Transforms/WholeProgramDevirt/import.ll @@ -1,8 +1,12 @@ ; RUN: opt -S -wholeprogramdevirt -wholeprogramdevirt-summary-action=import -wholeprogramdevirt-read-summary=%S/Inputs/import-single-impl.yaml < %s | FileCheck --check-prefixes=CHECK,SINGLE-IMPL %s +; RUN: opt -S -wholeprogramdevirt -wholeprogramdevirt-summary-action=import -wholeprogramdevirt-read-summary=%S/Inputs/import-uniform-ret-val.yaml < %s | FileCheck --check-prefixes=CHECK,UNIFORM-RET-VAL %s target datalayout = "e-p:64:64" target triple = "x86_64-unknown-linux-gnu" +; Test cases where the argument values are known and we can apply virtual +; constant propagation. + ; CHECK: define i32 @call1 define i32 @call1(i8* %obj) { %vtableptr = bitcast i8* %obj to [3 x i8*]** @@ -15,9 +19,13 @@ define i32 @call1(i8* %obj) { %fptr_casted = bitcast i8* %fptr to i32 (i8*, i32)* ; SINGLE-IMPL: call i32 bitcast (void ()* @singleimpl1 to i32 (i8*, i32)*) %result = call i32 %fptr_casted(i8* %obj, i32 1) + ; UNIFORM-RET-VAL: ret i32 42 ret i32 %result } +; Test cases where the argument values are unknown, so we cannot apply virtual +; constant propagation. + ; CHECK: define i1 @call2 define i1 @call2(i8* %obj) { %vtableptr = bitcast i8* %obj to [1 x i8*]** @@ -32,6 +40,7 @@ define i1 @call2(i8* %obj) { cont: %fptr_casted = bitcast i8* %fptr to i1 (i8*, i32)* ; SINGLE-IMPL: call i1 bitcast (void ()* @singleimpl2 to i1 (i8*, i32)*) + ; UNIFORM-RET-VAL: call i1 % %result = call i1 %fptr_casted(i8* %obj, i32 undef) ret i1 %result