Skip to content

Commit 2e29c91

Browse files
author
Mogball
committed
Revert "[Coro] [async] Disable inlining in async coroutine splitting (llvm#80904)"
This reverts commit b1ac052. This commit breaks coroutine splitting for non-swift calling convention functions. In this example: ```ll ; ModuleID = 'repro.ll' source_filename = "stdlib/test/runtime/test_llcl.mojo" target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-i128:128-f80:128-n8:16:32:64-S128" target triple = "x86_64-unknown-linux-gnu" @0 = internal constant { i32, i32 } { i32 trunc (i64 sub (i64 ptrtoint (ptr @craSH to i64), i64 ptrtoint (ptr getelementptr inbounds ({ i32, i32 }, ptr @0, i32 0, i32 1) to i64)) to i32), i32 64 } define dso_local void @af_suspend_fn(ptr %0, i64 %1, ptr %2) #0 { ret void } define dso_local void @craSH(ptr %0) #0 { %2 = call token @llvm.coro.id.async(i32 64, i32 8, i32 0, ptr @0) %3 = call ptr @llvm.coro.begin(token %2, ptr null) %4 = getelementptr inbounds { ptr, { ptr, ptr }, i64, { ptr, i1 }, i64, i64 }, ptr poison, i32 0, i32 0 %5 = call ptr @llvm.coro.async.resume() store ptr %5, ptr %4, align 8 %6 = call { ptr, ptr, ptr } (i32, ptr, ptr, ...) @llvm.coro.suspend.async.sl_p0p0p0s(i32 0, ptr %5, ptr @ctxt_proj_fn, ptr @af_suspend_fn, ptr poison, i64 -1, ptr poison) ret void } define dso_local ptr @ctxt_proj_fn(ptr %0) #0 { ret ptr %0 } ; Function Attrs: nomerge nounwind declare { ptr, ptr, ptr } @llvm.coro.suspend.async.sl_p0p0p0s(i32, ptr, ptr, ...) #1 ; Function Attrs: nounwind declare token @llvm.coro.id.async(i32, i32, i32, ptr) #2 ; Function Attrs: nounwind declare ptr @llvm.coro.begin(token, ptr writeonly) #2 ; Function Attrs: nomerge nounwind declare ptr @llvm.coro.async.resume() #1 attributes #0 = { "target-features"="+adx,+aes,+avx,+avx2,+bmi,+bmi2,+clflushopt,+clwb,+clzero,+crc32,+cx16,+cx8,+f16c,+fma,+fsgsbase,+fxsr,+invpcid,+lzcnt,+mmx,+movbe,+mwaitx,+pclmul,+pku,+popcnt,+prfchw,+rdpid,+rdpru,+rdrnd,+rdseed,+sahf,+sha,+sse,+sse2,+sse3,+sse4.1,+sse4.2,+sse4a,+ssse3,+vaes,+vpclmulqdq,+wbnoinvd,+x87,+xsave,+xsavec,+xsaveopt,+xsaves" } attributes #1 = { nomerge nounwind } attributes #2 = { nounwind } ``` This verifier crashes after the `coro-split` pass with ``` cannot guarantee tail call due to mismatched parameter counts musttail call void @af_suspend_fn(ptr poison, i64 -1, ptr poison) LLVM ERROR: Broken function PLEASE submit a bug report to https://github.com/llvm/llvm-project/issues/ and include the crash backtrace. Stack dump: 0. Program arguments: opt ../../../reduced.ll -O0 #0 0x00007f1d89645c0e __interceptor_backtrace.part.0 /build/gcc-11-XeT9lY/gcc-11-11.4.0/build/x86_64-linux-gnu/libsanitizer/asan/../../../../src/libsanitizer/sanitizer_common/sanitizer_common_interceptors.inc:4193:28 #1 0x0000556d94d254f7 llvm::sys::PrintStackTrace(llvm::raw_ostream&, int) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Support/Unix/Signals.inc:723:22 #2 0x0000556d94d19a2f llvm::sys::RunSignalHandlers() /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Support/Signals.cpp:105:20 #3 0x0000556d94d1aa42 SignalHandler(int) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Support/Unix/Signals.inc:371:36 llvm#4 0x00007f1d88e42520 (/lib/x86_64-linux-gnu/libc.so.6+0x42520) llvm#5 0x00007f1d88e969fc __pthread_kill_implementation ./nptl/pthread_kill.c:44:76 llvm#6 0x00007f1d88e969fc __pthread_kill_internal ./nptl/pthread_kill.c:78:10 llvm#7 0x00007f1d88e969fc pthread_kill ./nptl/pthread_kill.c:89:10 llvm#8 0x00007f1d88e42476 gsignal ./signal/../sysdeps/posix/raise.c:27:6 llvm#9 0x00007f1d88e287f3 abort ./stdlib/abort.c:81:7 llvm#10 0x0000556d8944be01 std::vector<llvm::json::Value, std::allocator<llvm::json::Value>>::size() const /usr/include/c++/11/bits/stl_vector.h:919:40 llvm#11 0x0000556d8944be01 bool std::operator==<llvm::json::Value, std::allocator<llvm::json::Value>>(std::vector<llvm::json::Value, std::allocator<llvm::json::Value>> const&, std::vector<llvm::json::Value, std::allocator<llvm::json::Value>> const&) /usr/include/c++/11/bits/stl_vector.h:1893:23 llvm#12 0x0000556d8944be01 llvm::json::operator==(llvm::json::Array const&, llvm::json::Array const&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/Support/JSON.h:572:69 llvm#13 0x0000556d8944be01 llvm::json::operator==(llvm::json::Value const&, llvm::json::Value const&) (.cold) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Support/JSON.cpp:204:28 llvm#14 0x0000556d949ed2bd llvm::report_fatal_error(char const*, bool) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Support/ErrorHandling.cpp:82:70 llvm#15 0x0000556d8e37e876 llvm::SmallVectorBase<unsigned int>::size() const /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallVector.h:91:32 llvm#16 0x0000556d8e37e876 llvm::SmallVectorTemplateCommon<llvm::DiagnosticInfoOptimizationBase::Argument, void>::end() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallVector.h:282:41 llvm#17 0x0000556d8e37e876 llvm::SmallVector<llvm::DiagnosticInfoOptimizationBase::Argument, 4u>::~SmallVector() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallVector.h:1215:24 llvm#18 0x0000556d8e37e876 llvm::DiagnosticInfoOptimizationBase::~DiagnosticInfoOptimizationBase() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/DiagnosticInfo.h:413:7 llvm#19 0x0000556d8e37e876 llvm::DiagnosticInfoIROptimization::~DiagnosticInfoIROptimization() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/DiagnosticInfo.h:622:7 llvm#20 0x0000556d8e37e876 llvm::OptimizationRemark::~OptimizationRemark() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/DiagnosticInfo.h:689:7 llvm#21 0x0000556d8e37e876 operator() /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Transforms/Coroutines/CoroSplit.cpp:2213:14 llvm#22 0x0000556d8e37e876 emit<llvm::CoroSplitPass::run(llvm::LazyCallGraph::SCC&, llvm::CGSCCAnalysisManager&, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&)::<lambda()> > /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/Analysis/OptimizationRemarkEmitter.h:83:12 llvm#23 0x0000556d8e37e876 llvm::CoroSplitPass::run(llvm::LazyCallGraph::SCC&, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>&, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Transforms/Coroutines/CoroSplit.cpp:2212:13 llvm#24 0x0000556d8c36ecb1 llvm::detail::PassModel<llvm::LazyCallGraph::SCC, llvm::CoroSplitPass, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&>::run(llvm::LazyCallGraph::SCC&, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>&, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/PassManagerInternal.h:91:3 llvm#25 0x0000556d91c1a84f llvm::PassManager<llvm::LazyCallGraph::SCC, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&>::run(llvm::LazyCallGraph::SCC&, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>&, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Analysis/CGSCCPassManager.cpp:90:12 llvm#26 0x0000556d8c3690d1 llvm::detail::PassModel<llvm::LazyCallGraph::SCC, llvm::PassManager<llvm::LazyCallGraph::SCC, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&>, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&>::run(llvm::LazyCallGraph::SCC&, llvm::AnalysisManager<llvm::LazyCallGraph::SCC, llvm::LazyCallGraph&>&, llvm::LazyCallGraph&, llvm::CGSCCUpdateResult&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/PassManagerInternal.h:91:3 llvm#27 0x0000556d91c2162d llvm::ModuleToPostOrderCGSCCPassAdaptor::run(llvm::Module&, llvm::AnalysisManager<llvm::Module>&) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Analysis/CGSCCPassManager.cpp:278:18 llvm#28 0x0000556d8c369035 llvm::detail::PassModel<llvm::Module, llvm::ModuleToPostOrderCGSCCPassAdaptor, llvm::AnalysisManager<llvm::Module>>::run(llvm::Module&, llvm::AnalysisManager<llvm::Module>&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/PassManagerInternal.h:91:3 llvm#29 0x0000556d9457abc5 llvm::PassManager<llvm::Module, llvm::AnalysisManager<llvm::Module>>::run(llvm::Module&, llvm::AnalysisManager<llvm::Module>&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/PassManager.h:247:20 llvm#30 0x0000556d8e30979e llvm::CoroConditionalWrapper::run(llvm::Module&, llvm::AnalysisManager<llvm::Module>&) /home/ubuntu/modular/third-party/llvm-project/llvm/lib/Transforms/Coroutines/CoroConditionalWrapper.cpp:19:74 llvm#31 0x0000556d8c365755 llvm::detail::PassModel<llvm::Module, llvm::CoroConditionalWrapper, llvm::AnalysisManager<llvm::Module>>::run(llvm::Module&, llvm::AnalysisManager<llvm::Module>&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/PassManagerInternal.h:91:3 llvm#32 0x0000556d9457abc5 llvm::PassManager<llvm::Module, llvm::AnalysisManager<llvm::Module>>::run(llvm::Module&, llvm::AnalysisManager<llvm::Module>&) /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/PassManager.h:247:20 llvm#33 0x0000556d89818556 llvm::SmallPtrSetImplBase::isSmall() const /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallPtrSet.h:196:33 llvm#34 0x0000556d89818556 llvm::SmallPtrSetImplBase::~SmallPtrSetImplBase() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallPtrSet.h:84:17 llvm#35 0x0000556d89818556 llvm::SmallPtrSetImpl<llvm::AnalysisKey*>::~SmallPtrSetImpl() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallPtrSet.h:321:7 llvm#36 0x0000556d89818556 llvm::SmallPtrSet<llvm::AnalysisKey*, 2u>::~SmallPtrSet() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/ADT/SmallPtrSet.h:427:7 llvm#37 0x0000556d89818556 llvm::PreservedAnalyses::~PreservedAnalyses() /home/ubuntu/modular/third-party/llvm-project/llvm/include/llvm/IR/Analysis.h:109:7 llvm#38 0x0000556d89818556 llvm::runPassPipeline(llvm::StringRef, llvm::Module&, llvm::TargetMachine*, llvm::TargetLibraryInfoImpl*, llvm::ToolOutputFile*, llvm::ToolOutputFile*, llvm::ToolOutputFile*, llvm::StringRef, llvm::ArrayRef<llvm::PassPlugin>, llvm::ArrayRef<std::function<void (llvm::PassBuilder&)>>, llvm::opt_tool::OutputKind, llvm::opt_tool::VerifierKind, bool, bool, bool, bool, bool, bool, bool) /home/ubuntu/modular/third-party/llvm-project/llvm/tools/opt/NewPMDriver.cpp:532:10 llvm#39 0x0000556d897e3939 optMain /home/ubuntu/modular/third-party/llvm-project/llvm/tools/opt/optdriver.cpp:737:27 llvm#40 0x0000556d89455461 main /home/ubuntu/modular/third-party/llvm-project/llvm/tools/opt/opt.cpp:25:33 llvm#41 0x00007f1d88e29d90 __libc_start_call_main ./csu/../sysdeps/nptl/libc_start_call_main.h:58:16 llvm#42 0x00007f1d88e29e40 call_init ./csu/../csu/libc-start.c:128:20 llvm#43 0x00007f1d88e29e40 __libc_start_main ./csu/../csu/libc-start.c:379:5 llvm#44 0x0000556d897b6335 _start (/home/ubuntu/modular/.derived/third-party/llvm-project/build-relwithdebinfo-asan/bin/opt+0x150c335) Aborted (core dumped)
1 parent 2cd59bd commit 2e29c91

8 files changed

+117
-262
lines changed

llvm/lib/Transforms/Coroutines/CoroSplit.cpp

+47-34
Original file line numberDiff line numberDiff line change
@@ -117,8 +117,8 @@ class CoroCloner {
117117
/// Create a cloner for a switch lowering.
118118
CoroCloner(Function &OrigF, const Twine &Suffix, coro::Shape &Shape,
119119
Kind FKind)
120-
: OrigF(OrigF), NewF(nullptr), Suffix(Suffix), Shape(Shape),
121-
FKind(FKind), Builder(OrigF.getContext()) {
120+
: OrigF(OrigF), NewF(nullptr), Suffix(Suffix), Shape(Shape), FKind(FKind),
121+
Builder(OrigF.getContext()) {
122122
assert(Shape.ABI == coro::ABI::Switch);
123123
}
124124

@@ -170,8 +170,7 @@ class CoroCloner {
170170
static void maybeFreeRetconStorage(IRBuilder<> &Builder,
171171
const coro::Shape &Shape, Value *FramePtr,
172172
CallGraph *CG) {
173-
assert(Shape.ABI == coro::ABI::Retcon ||
174-
Shape.ABI == coro::ABI::RetconOnce);
173+
assert(Shape.ABI == coro::ABI::Retcon || Shape.ABI == coro::ABI::RetconOnce);
175174
if (Shape.RetconLowering.IsFrameInlineInStorage)
176175
return;
177176

@@ -208,12 +207,17 @@ static bool replaceCoroEndAsync(AnyCoroEndInst *End) {
208207
// Insert the return instruction.
209208
Builder.SetInsertPoint(End);
210209
Builder.CreateRetVoid();
210+
InlineFunctionInfo FnInfo;
211211

212212
// Remove the rest of the block, by splitting it into an unreachable block.
213213
auto *BB = End->getParent();
214214
BB->splitBasicBlock(End);
215215
BB->getTerminator()->eraseFromParent();
216216

217+
auto InlineRes = InlineFunction(*MustTailCall, FnInfo);
218+
assert(InlineRes.isSuccess() && "Expected inlining to succeed");
219+
(void)InlineRes;
220+
217221
// We have cleaned up the coro.end block above.
218222
return false;
219223
}
@@ -264,7 +268,7 @@ static void replaceFallthroughCoroEnd(AnyCoroEndInst *End,
264268

265269
if (auto *RetStructTy = dyn_cast<StructType>(RetTy)) {
266270
assert(RetStructTy->getNumElements() == NumReturns &&
267-
"numbers of returns should match resume function singature");
271+
"numbers of returns should match resume function singature");
268272
Value *ReturnValue = UndefValue::get(RetStructTy);
269273
unsigned Idx = 0;
270274
for (Value *RetValEl : CoroResults->return_values())
@@ -277,7 +281,8 @@ static void replaceFallthroughCoroEnd(AnyCoroEndInst *End,
277281
assert(NumReturns == 1);
278282
Builder.CreateRet(*CoroResults->retval_begin());
279283
}
280-
CoroResults->replaceAllUsesWith(ConstantTokenNone::get(CoroResults->getContext()));
284+
CoroResults->replaceAllUsesWith(
285+
ConstantTokenNone::get(CoroResults->getContext()));
281286
CoroResults->eraseFromParent();
282287
break;
283288
}
@@ -291,7 +296,7 @@ static void replaceFallthroughCoroEnd(AnyCoroEndInst *End,
291296
auto RetTy = Shape.getResumeFunctionType()->getReturnType();
292297
auto RetStructTy = dyn_cast<StructType>(RetTy);
293298
PointerType *ContinuationTy =
294-
cast<PointerType>(RetStructTy ? RetStructTy->getElementType(0) : RetTy);
299+
cast<PointerType>(RetStructTy ? RetStructTy->getElementType(0) : RetTy);
295300

296301
Value *ReturnValue = ConstantPointerNull::get(ContinuationTy);
297302
if (RetStructTy) {
@@ -480,11 +485,12 @@ void CoroCloner::replaceRetconOrAsyncSuspendUses() {
480485
Shape.ABI == coro::ABI::Async);
481486

482487
auto NewS = VMap[ActiveSuspend];
483-
if (NewS->use_empty()) return;
488+
if (NewS->use_empty())
489+
return;
484490

485491
// Copy out all the continuation arguments after the buffer pointer into
486492
// an easily-indexed data structure for convenience.
487-
SmallVector<Value*, 8> Args;
493+
SmallVector<Value *, 8> Args;
488494
// The async ABI includes all arguments -- including the first argument.
489495
bool IsAsyncABI = Shape.ABI == coro::ABI::Async;
490496
for (auto I = IsAsyncABI ? NewF->arg_begin() : std::next(NewF->arg_begin()),
@@ -511,7 +517,8 @@ void CoroCloner::replaceRetconOrAsyncSuspendUses() {
511517
}
512518

513519
// If we have no remaining uses, we're done.
514-
if (NewS->use_empty()) return;
520+
if (NewS->use_empty())
521+
return;
515522

516523
// Otherwise, we need to create an aggregate.
517524
Value *Agg = PoisonValue::get(NewS->getType());
@@ -549,7 +556,8 @@ void CoroCloner::replaceCoroSuspends() {
549556

550557
for (AnyCoroSuspendInst *CS : Shape.CoroSuspends) {
551558
// The active suspend was handled earlier.
552-
if (CS == ActiveSuspend) continue;
559+
if (CS == ActiveSuspend)
560+
continue;
553561

554562
auto *MappedCS = cast<AnyCoroSuspendInst>(VMap[CS]);
555563
MappedCS->replaceAllUsesWith(SuspendResult);
@@ -707,7 +715,7 @@ void CoroCloner::replaceEntryBlock() {
707715
// In switch-lowering, we built a resume-entry block in the original
708716
// function. Make the entry block branch to this.
709717
auto *SwitchBB =
710-
cast<BasicBlock>(VMap[Shape.SwitchLowering.ResumeEntryBlock]);
718+
cast<BasicBlock>(VMap[Shape.SwitchLowering.ResumeEntryBlock]);
711719
Builder.CreateBr(SwitchBB);
712720
break;
713721
}
@@ -1055,7 +1063,7 @@ void CoroCloner::create() {
10551063
// to suppress deallocation code.
10561064
if (Shape.ABI == coro::ABI::Switch)
10571065
coro::replaceCoroFree(cast<CoroIdInst>(VMap[Shape.CoroBegin->getId()]),
1058-
/*Elide=*/ FKind == CoroCloner::Kind::SwitchCleanup);
1066+
/*Elide=*/FKind == CoroCloner::Kind::SwitchCleanup);
10591067
}
10601068

10611069
static void updateAsyncFuncPointerContextSize(coro::Shape &Shape) {
@@ -1842,8 +1850,13 @@ static void splitAsyncCoroutine(Function &F, coro::Shape &Shape,
18421850
SmallVector<Value *, 8> Args(Suspend->args());
18431851
auto FnArgs = ArrayRef<Value *>(Args).drop_front(
18441852
CoroSuspendAsyncInst::MustTailCallFuncArg + 1);
1845-
coro::createMustTailCall(Suspend->getDebugLoc(), Fn, TTI, FnArgs, Builder);
1853+
auto *TailCall = coro::createMustTailCall(Suspend->getDebugLoc(), Fn, TTI,
1854+
FnArgs, Builder);
18461855
Builder.CreateRetVoid();
1856+
InlineFunctionInfo FnInfo;
1857+
auto InlineRes = InlineFunction(*TailCall, FnInfo);
1858+
assert(InlineRes.isSuccess() && "Expected inlining to succeed");
1859+
(void)InlineRes;
18471860

18481861
// Replace the lvm.coro.async.resume intrisic call.
18491862
replaceAsyncResumeFunction(Suspend, Continuation);
@@ -1860,8 +1873,7 @@ static void splitAsyncCoroutine(Function &F, coro::Shape &Shape,
18601873

18611874
static void splitRetconCoroutine(Function &F, coro::Shape &Shape,
18621875
SmallVectorImpl<Function *> &Clones) {
1863-
assert(Shape.ABI == coro::ABI::Retcon ||
1864-
Shape.ABI == coro::ABI::RetconOnce);
1876+
assert(Shape.ABI == coro::ABI::Retcon || Shape.ABI == coro::ABI::RetconOnce);
18651877
assert(Clones.empty());
18661878

18671879
// Reset various things that the optimizer might have decided it
@@ -1887,7 +1899,7 @@ static void splitRetconCoroutine(Function &F, coro::Shape &Shape,
18871899
// FIXME: pass the required alignment
18881900
RawFramePtr = Shape.emitAlloc(Builder, Builder.getInt64(Size), nullptr);
18891901
RawFramePtr =
1890-
Builder.CreateBitCast(RawFramePtr, Shape.CoroBegin->getType());
1902+
Builder.CreateBitCast(RawFramePtr, Shape.CoroBegin->getType());
18911903

18921904
// Stash the allocated frame pointer in the continuation storage.
18931905
Builder.CreateStore(RawFramePtr, Id->getStorage());
@@ -1927,8 +1939,8 @@ static void splitRetconCoroutine(Function &F, coro::Shape &Shape,
19271939
// Create the unified return block.
19281940
if (!ReturnBB) {
19291941
// Place it before the first suspend.
1930-
ReturnBB = BasicBlock::Create(F.getContext(), "coro.return", &F,
1931-
NewSuspendBB);
1942+
ReturnBB =
1943+
BasicBlock::Create(F.getContext(), "coro.return", &F, NewSuspendBB);
19321944
Shape.RetconLowering.ReturnBlock = ReturnBB;
19331945

19341946
IRBuilder<> Builder(ReturnBB);
@@ -1942,8 +1954,8 @@ static void splitRetconCoroutine(Function &F, coro::Shape &Shape,
19421954

19431955
// Next, all the directly-yielded values.
19441956
for (auto *ResultTy : Shape.getRetconResultTypes())
1945-
ReturnPHIs.push_back(Builder.CreatePHI(ResultTy,
1946-
Shape.CoroSuspends.size()));
1957+
ReturnPHIs.push_back(
1958+
Builder.CreatePHI(ResultTy, Shape.CoroSuspends.size()));
19471959

19481960
// Build the return value.
19491961
auto RetTy = F.getReturnType();
@@ -1952,9 +1964,9 @@ static void splitRetconCoroutine(Function &F, coro::Shape &Shape,
19521964
// We can't rely on the types matching up because that type would
19531965
// have to be infinite.
19541966
auto CastedContinuationTy =
1955-
(ReturnPHIs.size() == 1 ? RetTy : RetTy->getStructElementType(0));
1967+
(ReturnPHIs.size() == 1 ? RetTy : RetTy->getStructElementType(0));
19561968
auto *CastedContinuation =
1957-
Builder.CreateBitCast(ReturnPHIs[0], CastedContinuationTy);
1969+
Builder.CreateBitCast(ReturnPHIs[0], CastedContinuationTy);
19581970

19591971
Value *RetV;
19601972
if (ReturnPHIs.size() == 1) {
@@ -1988,17 +2000,18 @@ static void splitRetconCoroutine(Function &F, coro::Shape &Shape,
19882000
}
19892001

19902002
namespace {
1991-
class PrettyStackTraceFunction : public PrettyStackTraceEntry {
1992-
Function &F;
1993-
public:
1994-
PrettyStackTraceFunction(Function &F) : F(F) {}
1995-
void print(raw_ostream &OS) const override {
1996-
OS << "While splitting coroutine ";
1997-
F.printAsOperand(OS, /*print type*/ false, F.getParent());
1998-
OS << "\n";
1999-
}
2000-
};
2001-
}
2003+
class PrettyStackTraceFunction : public PrettyStackTraceEntry {
2004+
Function &F;
2005+
2006+
public:
2007+
PrettyStackTraceFunction(Function &F) : F(F) {}
2008+
void print(raw_ostream &OS) const override {
2009+
OS << "While splitting coroutine ";
2010+
F.printAsOperand(OS, /*print type*/ false, F.getParent());
2011+
OS << "\n";
2012+
}
2013+
};
2014+
} // namespace
20022015

20032016
static coro::Shape
20042017
splitCoroutine(Function &F, SmallVectorImpl<Function *> &Clones,

llvm/test/Transforms/Coroutines/coro-async-addr-lifetime-infinite-loop-bug.ll

+4-4
Original file line numberDiff line numberDiff line change
@@ -22,8 +22,8 @@ declare void @my_other_async_function(ptr %async.ctxt)
2222
i32 128 ; Initial async context size without space for frame
2323
}>
2424

25-
define swifttailcc void @my_other_async_function_fp.apply(ptr %fnPtr, ptr %async.ctxt) alwaysinline {
26-
tail call swifttailcc void %fnPtr(ptr %async.ctxt)
25+
define swiftcc void @my_other_async_function_fp.apply(ptr %fnPtr, ptr %async.ctxt) {
26+
tail call swiftcc void %fnPtr(ptr %async.ctxt)
2727
ret void
2828
}
2929

@@ -37,12 +37,12 @@ entry:
3737

3838
; The address of alloca escapes but the analysis based on lifetimes fails to see
3939
; that it can't localize this alloca.
40-
; CHECK: define swifttailcc void @my_async_function(ptr swiftasync %async.ctxt) {
40+
; CHECK: define swiftcc void @my_async_function(ptr swiftasync %async.ctxt) {
4141
; CHECK: entry:
4242
; CHECK-NOT: ret
4343
; CHECK-NOT: [[ESCAPED_ADDR:%.*]] = alloca i64, align 8
4444
; CHECK: ret
45-
define swifttailcc void @my_async_function(ptr swiftasync %async.ctxt) {
45+
define swiftcc void @my_async_function(ptr swiftasync %async.ctxt) {
4646
entry:
4747
%escaped_addr = alloca i64
4848

llvm/test/Transforms/Coroutines/coro-async-addr-lifetime-start-bug.ll

+3-3
Original file line numberDiff line numberDiff line change
@@ -22,8 +22,8 @@ declare void @my_other_async_function(ptr %async.ctxt)
2222
i32 128 ; Initial async context size without space for frame
2323
}>
2424

25-
define swifttailcc void @my_other_async_function_fp.apply(ptr %fnPtr, ptr %async.ctxt) alwaysinline {
26-
tail call swifttailcc void %fnPtr(ptr %async.ctxt)
25+
define swiftcc void @my_other_async_function_fp.apply(ptr %fnPtr, ptr %async.ctxt) {
26+
tail call swiftcc void %fnPtr(ptr %async.ctxt)
2727
ret void
2828
}
2929

@@ -36,7 +36,7 @@ entry:
3636
ret ptr %resume_ctxt
3737
}
3838

39-
define swifttailcc void @my_async_function(ptr swiftasync %async.ctxt) {
39+
define swiftcc void @my_async_function(ptr swiftasync %async.ctxt) {
4040
entry:
4141
%escaped_addr = alloca i64
4242

llvm/test/Transforms/Coroutines/coro-async-dyn-align.ll

+5-5
Original file line numberDiff line numberDiff line change
@@ -33,20 +33,20 @@ declare swiftcc void @asyncReturn(ptr)
3333
declare swiftcc void @asyncSuspend(ptr)
3434
declare {ptr} @llvm.coro.suspend.async(i32, ptr, ptr, ...)
3535

36-
define swifttailcc void @my_async_function.my_other_async_function_fp.apply(ptr %fnPtr, ptr %async.ctxt) alwaysinline {
37-
musttail call swifttailcc void %fnPtr(ptr %async.ctxt)
36+
define swiftcc void @my_async_function.my_other_async_function_fp.apply(ptr %fnPtr, ptr %async.ctxt) {
37+
tail call swiftcc void %fnPtr(ptr %async.ctxt)
3838
ret void
3939
}
4040

41-
define ptr @__swift_async_resume_project_context(ptr %ctxt) alwaysinline {
41+
define ptr @__swift_async_resume_project_context(ptr %ctxt) {
4242
entry:
4343
%resume_ctxt = load ptr, ptr %ctxt, align 8
4444
ret ptr %resume_ctxt
4545
}
4646

4747

4848
; CHECK: %my_async_function.Frame = type { i64, [48 x i8], i64, i64, [16 x i8], ptr, i64, ptr }
49-
; CHECK: define swifttailcc void @my_async_function
49+
; CHECK: define swiftcc void @my_async_function
5050
; CHECK: [[T0:%.*]] = getelementptr inbounds %my_async_function.Frame, ptr %async.ctx.frameptr, i32 0, i32 3
5151
; CHECK: [[T1:%.*]] = ptrtoint ptr [[T0]] to i64
5252
; CHECK: [[T2:%.*]] = add i64 [[T1]], 31
@@ -60,7 +60,7 @@ entry:
6060
; CHECK: store i64 2, ptr [[T4]]
6161
; CHECK: store i64 3, ptr [[T9]]
6262

63-
define swifttailcc void @my_async_function(ptr swiftasync %async.ctxt) presplitcoroutine {
63+
define swiftcc void @my_async_function(ptr swiftasync %async.ctxt) presplitcoroutine {
6464
entry:
6565
%tmp = alloca i64, align 8
6666
%tmp2 = alloca i64, align 16

0 commit comments

Comments
 (0)