LLVM 23.0.0git
CoroShape.h
Go to the documentation of this file.
1//===- CoroShape.h - Coroutine info for lowering --------------*- C++ -*---===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8// This file declares the shape info struct that is required by many coroutine
9// utility methods.
10//===----------------------------------------------------------------------===//
11
12#ifndef LLVM_TRANSFORMS_COROUTINES_COROSHAPE_H
13#define LLVM_TRANSFORMS_COROUTINES_COROSHAPE_H
14
15#include "llvm/IR/IRBuilder.h"
16#include "llvm/IR/PassManager.h"
19
20namespace llvm {
21
22class CallGraph;
23
24namespace coro {
25
26enum class ABI {
27 /// The "resume-switch" lowering, where there are separate resume and
28 /// destroy functions that are shared between all suspend points. The
29 /// coroutine frame implicitly stores the resume and destroy functions,
30 /// the current index, and any promise value.
32
33 /// The "returned-continuation" lowering, where each suspend point creates a
34 /// single continuation function that is used for both resuming and
35 /// destroying. Does not support promises.
37
38 /// The "unique returned-continuation" lowering, where each suspend point
39 /// creates a single continuation function that is used for both resuming
40 /// and destroying. Does not support promises. The function is known to
41 /// suspend at most once during its execution, and the return value of
42 /// the continuation is void.
44
45 /// The "async continuation" lowering, where each suspend point creates a
46 /// single continuation function. The continuation function is available as an
47 /// intrinsic.
49};
50
51// Holds structural Coroutine Intrinsics for a particular function and other
52// values used during CoroSplit pass.
53struct Shape {
62
63 // Values invalidated by replaceSwiftErrorOps()
65
66 void clear() {
67 CoroBegin = nullptr;
68 CoroEnds.clear();
69 CoroIsInRampInsts.clear();
70 CoroSizes.clear();
71 CoroAligns.clear();
72 CoroSuspends.clear();
73 CoroAwaitSuspends.clear();
74 SymmetricTransfers.clear();
75
76 SwiftErrorOps.clear();
77
78 FramePtr = nullptr;
79 AllocaSpillBlock = nullptr;
80 }
81
82 // Scan the function and collect the above intrinsics for later processing
85 SmallVectorImpl<CoroSaveInst *> &UnusedCoroSaves);
86 // If for some reason, we were not able to find coro.begin, bailout.
87 LLVM_ABI void
90 // Perform ABI related initial transformation
92 // Remove orphaned and unnecessary intrinsics
93 LLVM_ABI void
95 SmallVectorImpl<CoroSaveInst *> &UnusedCoroSaves);
96
98
101 Value *FramePtr = nullptr;
103
116
124
137
138 union {
142 };
143
146 return cast<CoroIdInst>(CoroBegin->getId());
147 }
148
153
158
161 assert(SwitchLowering.IndexType && "index type not assigned");
162 return SwitchLowering.IndexType;
163 }
165 return ConstantInt::get(getIndexType(), Value);
166 }
167
170 assert(CoroBegin && "CoroBegin not assigned");
171 return PointerType::getUnqual(CoroBegin->getContext());
172 }
173
175 switch (ABI) {
177 return FunctionType::get(Type::getVoidTy(CoroBegin->getContext()),
178 PointerType::getUnqual(CoroBegin->getContext()),
179 /*IsVarArg=*/false);
182 return RetconLowering.ResumePrototype->getFunctionType();
183 case coro::ABI::Async:
184 // Not used. The function type depends on the active suspend.
185 return nullptr;
186 }
187
188 llvm_unreachable("Unknown coro::ABI enum");
189 }
190
193 auto FTy = CoroBegin->getFunction()->getFunctionType();
194
195 // The safety of all this is checked by checkWFRetconPrototype.
196 if (auto STy = dyn_cast<StructType>(FTy->getReturnType())) {
197 return STy->elements().slice(1);
198 } else {
199 return ArrayRef<Type *>();
200 }
201 }
202
205
206 // The safety of all this is checked by checkWFRetconPrototype.
207 auto FTy = RetconLowering.ResumePrototype->getFunctionType();
208 return FTy->params().slice(1);
209 }
210
212 switch (ABI) {
214 // Use the platform C calling convention so that resume/destroy
215 // function pointers stored in the coroutine frame are
216 // interoperable with other compilers.
217 return CallingConv::C;
218
221 return RetconLowering.ResumePrototype->getCallingConv();
222 case coro::ABI::Async:
223 return AsyncLowering.AsyncCC;
224 }
225 llvm_unreachable("Unknown coro::ABI enum");
226 }
227
229 if (ABI == coro::ABI::Switch)
230 return SwitchLowering.PromiseAlloca;
231 return nullptr;
232 }
233
235 if (auto *I = dyn_cast<Instruction>(FramePtr)) {
236 BasicBlock::iterator It = std::next(I->getIterator());
237 It.setHeadBit(true); // Copy pre-RemoveDIs behaviour.
238 return It;
239 }
240 return cast<Argument>(FramePtr)->getParent()->getEntryBlock().begin();
241 }
242
243 /// Allocate memory according to the rules of the active lowering.
244 ///
245 /// \param CG - if non-null, will be updated for the new call
247 CallGraph *CG) const;
248
249 /// Deallocate memory according to the rules of the active lowering.
250 ///
251 /// \param CG - if non-null, will be updated for the new call
252 LLVM_ABI void emitDealloc(IRBuilder<> &Builder, Value *Ptr,
253 CallGraph *CG) const;
254
255 Shape() = default;
256 explicit Shape(Function &F) {
258 SmallVector<CoroSaveInst *, 2> UnusedCoroSaves;
259
260 analyze(F, CoroFrames, UnusedCoroSaves);
261 if (!CoroBegin) {
262 invalidateCoroutine(F, CoroFrames);
263 return;
264 }
265 cleanCoroutine(CoroFrames, UnusedCoroSaves);
266 }
267};
268
269} // end namespace coro
270
271} // end namespace llvm
272
273#endif // LLVM_TRANSFORMS_COROUTINES_COROSHAPE_H
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
#define LLVM_ABI
Definition Compiler.h:213
This header defines various interfaces for pass management in LLVM.
#define F(x, y, z)
Definition MD5.cpp:54
#define I(x, y, z)
Definition MD5.cpp:57
an instruction to allocate memory on the stack
This represents either the llvm.coro.id.retcon or llvm.coro.id.retcon.once instruction.
Definition CoroInstr.h:237
Represent a constant reference to an array (0 or more elements consecutively in memory),...
Definition ArrayRef.h:40
LLVM Basic Block Representation.
Definition BasicBlock.h:62
InstListType::iterator iterator
Instruction iterators...
Definition BasicBlock.h:170
The basic data container for the call graph of a Module of IR.
Definition CallGraph.h:72
This is the shared class of boolean and integer constants.
Definition Constants.h:87
This class represents the llvm.coro.begin or llvm.coro.begin.custom.abi instructions.
Definition CoroInstr.h:475
This represents the llvm.coro.id.async instruction.
Definition CoroInstr.h:306
This represents the llvm.coro.id instruction.
Definition CoroInstr.h:148
Class to represent function types.
static LLVM_ABI FunctionType * get(Type *Result, ArrayRef< Type * > Params, bool isVarArg)
This static method is the primary way of constructing a FunctionType.
This provides a uniform API for creating instructions and inserting them into a basic block: either a...
Definition IRBuilder.h:2858
Class to represent integer types.
Class to represent pointers.
static PointerType * getUnqual(Type *ElementType)
This constructs a pointer to an object of the specified type in the default address space (address sp...
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Multiway switch.
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
Definition Type.cpp:282
LLVM Value Representation.
Definition Value.h:75
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
@ C
The default llvm calling convention, compatible with C.
Definition CallingConv.h:34
@ Async
The "async continuation" lowering, where each suspend point creates a single continuation function.
Definition CoroShape.h:48
@ RetconOnce
The "unique returned-continuation" lowering, where each suspend point creates a single continuation f...
Definition CoroShape.h:43
@ Retcon
The "returned-continuation" lowering, where each suspend point creates a single continuation function...
Definition CoroShape.h:36
@ Switch
The "resume-switch" lowering, where there are separate resume and destroy functions that are shared b...
Definition CoroShape.h:31
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
ArrayRef(const T &OneElt) -> ArrayRef< T >
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:559
This struct is a compact representation of a valid (non-zero power of two) alignment.
Definition Alignment.h:39
SmallVector< CallInst *, 2 > SymmetricTransfers
Definition CoroShape.h:61
SmallVector< CoroAwaitSuspendInst *, 4 > CoroAwaitSuspends
Definition CoroShape.h:60
AsyncLoweringStorage AsyncLowering
Definition CoroShape.h:141
FunctionType * getResumeFunctionType() const
Definition CoroShape.h:174
IntegerType * getIndexType() const
Definition CoroShape.h:159
AnyCoroIdRetconInst * getRetconCoroId() const
Definition CoroShape.h:149
PointerType * getSwitchResumePointerType() const
Definition CoroShape.h:168
CoroIdInst * getSwitchCoroId() const
Definition CoroShape.h:144
LLVM_ABI void initABI()
SmallVector< CoroSizeInst *, 2 > CoroSizes
Definition CoroShape.h:57
CallingConv::ID getResumeFunctionCC() const
Definition CoroShape.h:211
coro::ABI ABI
Definition CoroShape.h:97
Shape(Function &F)
Definition CoroShape.h:256
ArrayRef< Type * > getRetconResumeTypes() const
Definition CoroShape.h:203
SmallVector< AnyCoroSuspendInst *, 4 > CoroSuspends
Definition CoroShape.h:59
uint64_t FrameSize
Definition CoroShape.h:100
LLVM_ABI Value * emitAlloc(IRBuilder<> &Builder, Value *Size, CallGraph *CG) const
Allocate memory according to the rules of the active lowering.
LLVM_ABI void cleanCoroutine(SmallVectorImpl< CoroFrameInst * > &CoroFrames, SmallVectorImpl< CoroSaveInst * > &UnusedCoroSaves)
ConstantInt * getIndex(uint64_t Value) const
Definition CoroShape.h:164
AllocaInst * getPromiseAlloca() const
Definition CoroShape.h:228
SwitchLoweringStorage SwitchLowering
Definition CoroShape.h:139
CoroBeginInst * CoroBegin
Definition CoroShape.h:54
BasicBlock::iterator getInsertPtAfterFramePtr() const
Definition CoroShape.h:234
ArrayRef< Type * > getRetconResultTypes() const
Definition CoroShape.h:191
SmallVector< CoroIsInRampInst *, 2 > CoroIsInRampInsts
Definition CoroShape.h:56
LLVM_ABI void emitDealloc(IRBuilder<> &Builder, Value *Ptr, CallGraph *CG) const
Deallocate memory according to the rules of the active lowering.
RetconLoweringStorage RetconLowering
Definition CoroShape.h:140
SmallVector< CoroAlignInst *, 2 > CoroAligns
Definition CoroShape.h:58
CoroIdAsyncInst * getAsyncCoroId() const
Definition CoroShape.h:154
SmallVector< AnyCoroEndInst *, 4 > CoroEnds
Definition CoroShape.h:55
SmallVector< CallInst *, 2 > SwiftErrorOps
Definition CoroShape.h:64
LLVM_ABI void invalidateCoroutine(Function &F, SmallVectorImpl< CoroFrameInst * > &CoroFrames)
BasicBlock * AllocaSpillBlock
Definition CoroShape.h:102
LLVM_ABI void analyze(Function &F, SmallVectorImpl< CoroFrameInst * > &CoroFrames, SmallVectorImpl< CoroSaveInst * > &UnusedCoroSaves)