@@ -62,18 +62,27 @@ static BasicBlock *createResumeEntryBlock(Function &F, coro::Shape &Shape) {
62
62
Builder.CreateSwitch (Index, UnreachBB, Shape.CoroSuspends .size ());
63
63
Shape.ResumeSwitch = Switch;
64
64
65
- uint32_t SuspendIndex = 0 ;
66
- for (auto S : Shape.CoroSuspends ) {
65
+ size_t SuspendIndex = 0 ;
66
+ for (CoroSuspendInst * S : Shape.CoroSuspends ) {
67
67
ConstantInt *IndexVal = Shape.getIndex (SuspendIndex);
68
68
69
69
// Replace CoroSave with a store to Index:
70
70
// %index.addr = getelementptr %f.frame... (index field number)
71
71
// store i32 0, i32* %index.addr1
72
72
auto *Save = S->getCoroSave ();
73
73
Builder.SetInsertPoint (Save);
74
- auto *GepIndex = Builder.CreateConstInBoundsGEP2_32 (
75
- FrameTy, FramePtr, 0 , coro::Shape::IndexField, " index.addr" );
76
- Builder.CreateStore (IndexVal, GepIndex);
74
+ if (S->isFinal ()) {
75
+ // Final suspend point is represented by storing zero in ResumeFnAddr.
76
+ auto *GepIndex = Builder.CreateConstInBoundsGEP2_32 (FrameTy, FramePtr, 0 ,
77
+ 0 , " ResumeFn.addr" );
78
+ auto *NullPtr = ConstantPointerNull::get (cast<PointerType>(
79
+ cast<PointerType>(GepIndex->getType ())->getElementType ()));
80
+ Builder.CreateStore (NullPtr, GepIndex);
81
+ } else {
82
+ auto *GepIndex = Builder.CreateConstInBoundsGEP2_32 (
83
+ FrameTy, FramePtr, 0 , coro::Shape::IndexField, " index.addr" );
84
+ Builder.CreateStore (IndexVal, GepIndex);
85
+ }
77
86
Save->replaceAllUsesWith (ConstantTokenNone::get (C));
78
87
Save->eraseFromParent ();
79
88
@@ -135,6 +144,37 @@ static void replaceFallthroughCoroEnd(IntrinsicInst *End,
135
144
BB->getTerminator ()->eraseFromParent ();
136
145
}
137
146
147
+ // Rewrite final suspend point handling. We do not use suspend index to
148
+ // represent the final suspend point. Instead we zero-out ResumeFnAddr in the
149
+ // coroutine frame, since it is undefined behavior to resume a coroutine
150
+ // suspended at the final suspend point. Thus, in the resume function, we can
151
+ // simply remove the last case (when coro::Shape is built, the final suspend
152
+ // point (if present) is always the last element of CoroSuspends array).
153
+ // In the destroy function, we add a code sequence to check if ResumeFnAddress
154
+ // is Null, and if so, jump to the appropriate label to handle cleanup from the
155
+ // final suspend point.
156
+ static void handleFinalSuspend (IRBuilder<> &Builder, Value *FramePtr,
157
+ coro::Shape &Shape, SwitchInst *Switch,
158
+ bool IsDestroy) {
159
+ assert (Shape.HasFinalSuspend );
160
+ auto FinalCase = --Switch->case_end ();
161
+ BasicBlock *ResumeBB = FinalCase.getCaseSuccessor ();
162
+ Switch->removeCase (FinalCase);
163
+ if (IsDestroy) {
164
+ BasicBlock *OldSwitchBB = Switch->getParent ();
165
+ auto *NewSwitchBB = OldSwitchBB->splitBasicBlock (Switch, " Switch" );
166
+ Builder.SetInsertPoint (OldSwitchBB->getTerminator ());
167
+ auto *GepIndex = Builder.CreateConstInBoundsGEP2_32 (Shape.FrameTy , FramePtr,
168
+ 0 , 0 , " ResumeFn.addr" );
169
+ auto *Load = Builder.CreateLoad (GepIndex);
170
+ auto *NullPtr =
171
+ ConstantPointerNull::get (cast<PointerType>(Load->getType ()));
172
+ auto *Cond = Builder.CreateICmpEQ (Load, NullPtr);
173
+ Builder.CreateCondBr (Cond, ResumeBB, NewSwitchBB);
174
+ OldSwitchBB->getTerminator ()->eraseFromParent ();
175
+ }
176
+ }
177
+
138
178
// Create a resume clone by cloning the body of the original function, setting
139
179
// new entry block and replacing coro.suspend an appropriate value to force
140
180
// resume or cleanup pass for every suspend point.
@@ -205,6 +245,15 @@ static Function *createClone(Function &F, Twine Suffix, coro::Shape &Shape,
205
245
Value *OldVFrame = cast<Value>(VMap[Shape.CoroBegin ]);
206
246
OldVFrame->replaceAllUsesWith (NewVFrame);
207
247
248
+ // Rewrite final suspend handling as it is not done via switch (allows to
249
+ // remove final case from the switch, since it is undefined behavior to resume
250
+ // the coroutine suspended at the final suspend point.
251
+ if (Shape.HasFinalSuspend ) {
252
+ auto *Switch = cast<SwitchInst>(VMap[Shape.ResumeSwitch ]);
253
+ bool IsDestroy = FnIndex != 0 ;
254
+ handleFinalSuspend (Builder, NewFramePtr, Shape, Switch, IsDestroy);
255
+ }
256
+
208
257
// Replace coro suspend with the appropriate resume index.
209
258
// Replacing coro.suspend with (0) will result in control flow proceeding to
210
259
// a resume label associated with a suspend point, replacing it with (1) will
0 commit comments