compnerd created this revision. compnerd added a reviewer: gottesmm. Herald added a subscriber: cfe-commits. compnerd added a reviewer: ahatanak. compnerd added subscribers: rnk, majnemer.
In the case that the CallInst that is being moved has an associated operand bundle which is a funclet, the move will construct an invalid instruction. The new site will have a different token and needs to be reassociated with the new instruction. Unfortunately, there is no way to alter the bundle after the construction of the instruction. Replace the call instruction cloning with a custom helper to clone the instruction and reassociate the funclet token. Repository: rC Clang https://reviews.llvm.org/D44327 Files: lib/Transforms/ObjCARC/ObjCARCOpts.cpp test/Transforms/ObjCARC/funclet.ll
Index: test/Transforms/ObjCARC/funclet.ll =================================================================== --- /dev/null +++ test/Transforms/ObjCARC/funclet.ll @@ -0,0 +1,54 @@ +; RUN: %opt -mtriple x86_64-unknown-windows-msvc -objc-arc -S -o - %s | FileCheck %s + +declare zeroext i1 @"\01?g@@YA_NXZ"() local_unnamed_addr +declare i8* @"\01?h@@YAPEAUobjc_object@@XZ"() local_unnamed_addr + +declare dllimport void @objc_release(i8*) local_unnamed_addr +declare dllimport i8* @objc_retainAutoreleasedReturnValue(i8* returned) local_unnamed_addr + +declare i32 @__CxxFrameHandler3(...) + +define void @"\01?f@@YAXXZ"() local_unnamed_addr personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) { +entry: + %call = invoke zeroext i1 @"\01?g@@YA_NXZ"() + to label %invoke.cont unwind label %ehcleanup6 + +invoke.cont: ; preds = %entry + br i1 %call, label %if.then, label %if.end + +if.then: ; preds = %invoke.cont + %call2 = invoke i8* @"\01?h@@YAPEAUobjc_object@@XZ"() + to label %invoke.cont1 unwind label %ehcleanup6 + +invoke.cont1: ; preds = %if.then + %0 = tail call i8* @objc_retainAutoreleasedReturnValue(i8* %call2) + tail call void @objc_release(i8* null), !clang.imprecise_release !1 + br label %if.end + +if.end: ; preds = %invoke.cont1, %invoke.cont + %a.0 = phi i8* [ %call2, %invoke.cont1 ], [ null, %invoke.cont ] + %call4 = invoke zeroext i1 @"\01?g@@YA_NXZ"() + to label %invoke.cont3 unwind label %ehcleanup + +invoke.cont3: ; preds = %if.end + tail call void @objc_release(i8* null), !clang.imprecise_release !1 + tail call void @objc_release(i8* %a.0), !clang.imprecise_release !1 + ret void + +ehcleanup: ; preds = %if.end + %1 = cleanuppad within none [] + call void @objc_release(i8* null) [ "funclet"(token %1) ], !clang.imprecise_release !1 + cleanupret from %1 unwind label %ehcleanup6 + +ehcleanup6: ; preds = %ehcleanup, %if.then, %entry + %a.1 = phi i8* [ %a.0, %ehcleanup ], [ null, %if.then ], [ null, %entry ] + %2 = cleanuppad within none [] + call void @objc_release(i8* %a.1) [ "funclet"(token %2) ], !clang.imprecise_release !1 + cleanupret from %2 unwind to caller +} + +; CHECK: call void @objc_release(i8* {{.*}}) {{.*}}[ "funclet"(token %1) ] +; CHECK-NOT: call void @objc_release(i8* {{.*}}) {{.*}}[ "funclet"(token %2) ] + +!1 = !{} + Index: lib/Transforms/ObjCARC/ObjCARCOpts.cpp =================================================================== --- lib/Transforms/ObjCARC/ObjCARCOpts.cpp +++ lib/Transforms/ObjCARC/ObjCARCOpts.cpp @@ -684,6 +684,27 @@ DEBUG(dbgs() << "New: " << *AutoreleaseRV << "\n"); } +namespace { +Instruction *CloneCallInstForBB(Instruction &I, BasicBlock &BB) { + auto *CI = dyn_cast<CallInst>(&I); + assert(CI && "CloneCallInst must receive a CallInst"); + + SmallVector<OperandBundleDef, 1> OpBundles; + for (unsigned I = 0, E = CI->getNumOperandBundles(); I != E; ++I) { + auto Bundle = CI->getOperandBundleAt(I); + // funclets will be reassociated in the future + if (Bundle.getTagID() == LLVMContext::OB_funclet) + continue; + OpBundles.emplace_back(Bundle); + } + + if (auto *CleanupPad = dyn_cast<CleanupPadInst>(BB.getFirstNonPHI())) + OpBundles.emplace_back("funclet", CleanupPad); + + return CallInst::Create(CI, OpBundles); +} +} + /// Visit each call, one at a time, and make simplifications without doing any /// additional analysis. void ObjCARCOpt::OptimizeIndividualCalls(Function &F) { @@ -927,9 +948,10 @@ Value *Incoming = GetRCIdentityRoot(PN->getIncomingValue(i)); if (!IsNullOrUndef(Incoming)) { - CallInst *Clone = cast<CallInst>(CInst->clone()); Value *Op = PN->getIncomingValue(i); Instruction *InsertPos = &PN->getIncomingBlock(i)->back(); + CallInst *Clone = cast<CallInst>( + CloneCallInstForBB(*CInst, *InsertPos->getParent())); if (Op->getType() != ParamTy) Op = new BitCastInst(Op, ParamTy, "", InsertPos); Clone->setArgOperand(0, Op);
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits