LLVM 23.0.0git
WebAssemblyFrameLowering.cpp
Go to the documentation of this file.
1//===-- WebAssemblyFrameLowering.cpp - WebAssembly Frame Lowering ----------==//
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///
9/// \file
10/// This file contains the WebAssembly implementation of
11/// TargetFrameLowering class.
12///
13/// On WebAssembly, there aren't a lot of things to do here. There are no
14/// callee-saved registers to save, and no spill slots.
15///
16/// The stack grows downward.
17///
18//===----------------------------------------------------------------------===//
19
23#include "WebAssembly.h"
34#include "llvm/MC/MCAsmInfo.h"
35using namespace llvm;
36
37#define DEBUG_TYPE "wasm-frame-info"
38
39// TODO: wasm64
40// TODO: Emit TargetOpcode::CFI_INSTRUCTION instructions
41
42// In an ideal world, when objects are added to the MachineFrameInfo by
43// FunctionLoweringInfo::set, we could somehow hook into target-specific code to
44// ensure they are assigned the right stack ID. However there isn't a hook that
45// runs between then and DAG building time, though, so instead we hoist stack
46// objects lazily when they are first used, and comprehensively after the DAG is
47// built via the PreprocessISelDAG hook, called by the
48// SelectionDAGISel::runOnMachineFunction. We have to do it in two places
49// because we want to do it while building the selection DAG for uses of alloca,
50// but not all alloca instructions are used so we have to follow up afterwards.
51std::optional<unsigned>
53 int FrameIndex) {
55
56 // If already hoisted to a local, done.
57 if (MFI.getStackID(FrameIndex) == TargetStackID::WasmLocal)
58 return static_cast<unsigned>(MFI.getObjectOffset(FrameIndex));
59
60 // If not allocated in the object address space, this object will be in
61 // linear memory.
62 const AllocaInst *AI = MFI.getObjectAllocation(FrameIndex);
64 return std::nullopt;
65
66 // Otherwise, allocate this object in the named value stack, outside of linear
67 // memory.
68 SmallVector<EVT, 4> ValueVTs;
69 const WebAssemblyTargetLowering &TLI =
70 *MF.getSubtarget<WebAssemblySubtarget>().getTargetLowering();
72 ComputeValueVTs(TLI, MF.getDataLayout(), AI->getAllocatedType(), ValueVTs);
73 MFI.setStackID(FrameIndex, TargetStackID::WasmLocal);
74 // Abuse SP offset to record the index of the first local in the object.
75 unsigned Local = FuncInfo->getParams().size() + FuncInfo->getLocals().size();
76 MFI.setObjectOffset(FrameIndex, Local);
77 // Allocate WebAssembly locals for each non-aggregate component of the
78 // allocation.
79 for (EVT ValueVT : ValueVTs)
80 FuncInfo->addLocal(ValueVT.getSimpleVT());
81 // Abuse object size to record number of WebAssembly locals allocated to
82 // this object.
83 MFI.setObjectSize(FrameIndex, ValueVTs.size());
84 return Local;
85}
86
87/// We need a base pointer in the case of having items on the stack that
88/// require stricter alignment than the stack pointer itself. Because we need
89/// to shift the stack pointer by some unknown amount to force the alignment,
90/// we need to record the value of the stack pointer on entry to the function.
91bool WebAssemblyFrameLowering::hasBP(const MachineFunction &MF) const {
92 const auto *RegInfo =
93 MF.getSubtarget<WebAssemblySubtarget>().getRegisterInfo();
94 return RegInfo->hasStackRealignment(MF);
95}
96
97/// Return true if the specified function should have a dedicated frame pointer
98/// register.
100 const MachineFrameInfo &MFI = MF.getFrameInfo();
101
102 // When we have var-sized objects, we move the stack pointer by an unknown
103 // amount, and need to emit a frame pointer to restore the stack to where we
104 // were on function entry.
105 // If we already need a base pointer, we use that to fix up the stack pointer.
106 // If there are no fixed-size objects, we would have no use of a frame
107 // pointer, and thus should not emit one.
108 bool HasFixedSizedObjects = MFI.getStackSize() > 0;
109 bool NeedsFixedReference = !hasBP(MF) || HasFixedSizedObjects;
110
111 return MFI.isFrameAddressTaken() ||
112 (MFI.hasVarSizedObjects() && NeedsFixedReference) ||
113 MFI.hasStackMap() || MFI.hasPatchPoint();
114}
115
116/// Under normal circumstances, when a frame pointer is not required, we reserve
117/// argument space for call sites in the function immediately on entry to the
118/// current function. This eliminates the need for add/sub sp brackets around
119/// call sites. Returns true if the call frame is included as part of the stack
120/// frame.
125
126// Returns true if this function needs a local user-space stack pointer for its
127// local frame (not for exception handling).
128bool WebAssemblyFrameLowering::needsSPForLocalFrame(
129 const MachineFunction &MF) const {
130 auto &MFI = MF.getFrameInfo();
131 auto &MRI = MF.getRegInfo();
132 // llvm.stacksave can explicitly read SP register and it can appear without
133 // dynamic alloca.
134 bool HasExplicitSPUse =
135 any_of(MRI.use_operands(getSPReg(MF)),
136 [](MachineOperand &MO) { return !MO.isImplicit(); });
137
138 // With libcall thread context, we need SP in the prolog when debug
139 // info is present so we can allocate a local for DWARF to reference.
140 bool NeedsSPForDebug =
141 MF.getFunction().getSubprogram() &&
142 MF.getSubtarget<WebAssemblySubtarget>().hasLibcallThreadContext();
143
144 return MFI.getStackSize() || MFI.adjustsStack() || hasFP(MF) ||
145 HasExplicitSPUse || NeedsSPForDebug;
146}
147
148// In function with EH pads, we need to make a copy of the value of
149// the stack pointer in the SP32/64 register, in order to use it when
150// restoring the stack pointer after an exception is caught.
152 const MachineFunction &MF) const {
153 auto EHType = MF.getTarget().getMCAsmInfo().getExceptionHandlingType();
154 return EHType == ExceptionHandling::Wasm &&
156}
157
158/// Returns true if this function needs a local user-space stack pointer.
159/// Unlike a machine stack pointer, the wasm user stack pointer is a global
160/// variable or managed by library calls, so it is loaded
161/// into a register in the prolog.
162bool WebAssemblyFrameLowering::needsSP(const MachineFunction &MF) const {
163 return needsSPForLocalFrame(MF) || needsPrologForEH(MF);
164}
165
166/// Returns true if the local user-space stack pointer needs to be written back
167/// to the stack pointer global/thread context by this function (this is not
168/// meaningful if needsSP is false). If false, the stack red zone can be used
169/// and only a local SP is needed.
170bool WebAssemblyFrameLowering::needsSPWriteback(
171 const MachineFunction &MF) const {
172 auto &MFI = MF.getFrameInfo();
173 assert(needsSP(MF));
174 // When we don't need a local stack pointer for its local frame but only to
175 // support EH, we don't need to write SP back in the epilog, because we don't
176 // bump down the stack pointer in the prolog. We need to write SP back in the
177 // epilog only if
178 // 1. We need SP not only for EH support but also because we actually use
179 // stack or we have a frame address taken.
180 // 2. We cannot use the red zone.
181 bool CanUseRedZone = MFI.getStackSize() <= RedZoneSize && !MFI.hasCalls() &&
182 !MF.getFunction().hasFnAttribute(Attribute::NoRedZone);
183 return needsSPForLocalFrame(MF) && !CanUseRedZone;
184}
185
187 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
188 ? WebAssembly::SP64
189 : WebAssembly::SP32;
190}
191
193 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
194 ? WebAssembly::FP64
195 : WebAssembly::FP32;
196}
197
198unsigned
200 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
201 ? WebAssembly::CONST_I64
202 : WebAssembly::CONST_I32;
203}
204
206 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
207 ? WebAssembly::ADD_I64
208 : WebAssembly::ADD_I32;
209}
210
212 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
213 ? WebAssembly::SUB_I64
214 : WebAssembly::SUB_I32;
215}
216
218 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
219 ? WebAssembly::AND_I64
220 : WebAssembly::AND_I32;
221}
222
223unsigned
225 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
226 ? WebAssembly::GLOBAL_GET_I64
227 : WebAssembly::GLOBAL_GET_I32;
228}
229
230unsigned
232 return MF.getSubtarget<WebAssemblySubtarget>().hasAddr64()
233 ? WebAssembly::GLOBAL_SET_I64
234 : WebAssembly::GLOBAL_SET_I32;
235}
236
238 unsigned SrcReg, MachineFunction &MF, MachineBasicBlock &MBB,
239 MachineBasicBlock::iterator &InsertStore, const DebugLoc &DL) const {
240 const auto *TII = MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
241
243 const char *ES = "__wasm_set_stack_pointer";
244 auto *SPSymbol = MF.createExternalSymbolName(ES);
245 BuildMI(MBB, InsertStore, DL, TII->get(WebAssembly::CALL))
246 .addExternalSymbol(SPSymbol)
247 .addReg(SrcReg);
248 } else {
249 const char *ES = "__stack_pointer";
250 auto *SPSymbol = MF.createExternalSymbolName(ES);
251
252 BuildMI(MBB, InsertStore, DL, TII->get(getOpcGlobSet(MF)))
253 .addExternalSymbol(SPSymbol)
254 .addReg(SrcReg);
255 }
256}
257
262 assert(!I->getOperand(0).getImm() && (hasFP(MF) || hasBP(MF)) &&
263 "Call frame pseudos should only be used for dynamic stack adjustment");
264 auto &ST = MF.getSubtarget<WebAssemblySubtarget>();
265 const auto *TII = ST.getInstrInfo();
266 if (I->getOpcode() == TII->getCallFrameDestroyOpcode() &&
267 needsSPWriteback(MF)) {
268 DebugLoc DL = I->getDebugLoc();
269 writeBackSP(getSPReg(MF), MF, MBB, I, DL);
270 }
271 return MBB.erase(I);
272}
273
275 MachineBasicBlock &MBB) const {
276 // TODO: Do ".setMIFlag(MachineInstr::FrameSetup)" on emitted instructions
277 auto &MFI = MF.getFrameInfo();
278 assert(MFI.getCalleeSavedInfo().empty() &&
279 "WebAssembly should not have callee-saved registers");
280
281 if (!needsSP(MF))
282 return;
283 uint64_t StackSize = MFI.getStackSize();
284
285 auto &ST = MF.getSubtarget<WebAssemblySubtarget>();
286 const auto *TII = ST.getInstrInfo();
287 auto &MRI = MF.getRegInfo();
288
289 auto InsertPt = MBB.begin();
290 while (InsertPt != MBB.end() &&
291 WebAssembly::isArgument(InsertPt->getOpcode()))
292 ++InsertPt;
293 DebugLoc DL;
294
295 const TargetRegisterClass *PtrRC =
296 MRI.getTargetRegisterInfo()->getPointerRegClass();
297 unsigned SPReg = getSPReg(MF);
298 if (StackSize)
299 SPReg = MRI.createVirtualRegister(PtrRC);
300
301 if (ST.hasLibcallThreadContext()) {
302 const char *ES = "__wasm_get_stack_pointer";
303 auto *SPSymbol = MF.createExternalSymbolName(ES);
304 BuildMI(MBB, InsertPt, DL, TII->get(WebAssembly::CALL), SPReg)
305 .addExternalSymbol(SPSymbol);
306 } else {
307 const char *ES = "__stack_pointer";
308 auto *SPSymbol = MF.createExternalSymbolName(ES);
309 BuildMI(MBB, InsertPt, DL, TII->get(getOpcGlobGet(MF)), SPReg)
310 .addExternalSymbol(SPSymbol);
311 }
312
313 bool HasBP = hasBP(MF);
314 if (HasBP) {
315 auto FI = MF.getInfo<WebAssemblyFunctionInfo>();
316 Register BasePtr = MRI.createVirtualRegister(PtrRC);
317 FI->setBasePointerVreg(BasePtr);
318 BuildMI(MBB, InsertPt, DL, TII->get(WebAssembly::COPY), BasePtr)
319 .addReg(SPReg);
320 }
321 if (StackSize) {
322 // Subtract the frame size
323 Register OffsetReg = MRI.createVirtualRegister(PtrRC);
324 BuildMI(MBB, InsertPt, DL, TII->get(getOpcConst(MF)), OffsetReg)
325 .addImm(StackSize);
326 BuildMI(MBB, InsertPt, DL, TII->get(getOpcSub(MF)), getSPReg(MF))
327 .addReg(SPReg)
328 .addReg(OffsetReg);
329 }
330 if (HasBP) {
331 Register BitmaskReg = MRI.createVirtualRegister(PtrRC);
332 Align Alignment = MFI.getMaxAlign();
333 BuildMI(MBB, InsertPt, DL, TII->get(getOpcConst(MF)), BitmaskReg)
334 .addImm((int64_t) ~(Alignment.value() - 1));
335 BuildMI(MBB, InsertPt, DL, TII->get(getOpcAnd(MF)), getSPReg(MF))
336 .addReg(getSPReg(MF))
337 .addReg(BitmaskReg);
338 }
339 if (hasFP(MF)) {
340 // Unlike most conventional targets (where FP points to the saved FP),
341 // FP points to the bottom of the fixed-size locals, so we can use positive
342 // offsets in load/store instructions.
343 BuildMI(MBB, InsertPt, DL, TII->get(WebAssembly::COPY), getFPReg(MF))
344 .addReg(getSPReg(MF));
345 }
346 if (StackSize && needsSPWriteback(MF)) {
347 writeBackSP(getSPReg(MF), MF, MBB, InsertPt, DL);
348 }
349}
350
352 MachineBasicBlock &MBB) const {
353 uint64_t StackSize = MF.getFrameInfo().getStackSize();
354 if (!needsSP(MF) || !needsSPWriteback(MF))
355 return;
356 auto &ST = MF.getSubtarget<WebAssemblySubtarget>();
357 const auto *TII = ST.getInstrInfo();
358 auto &MRI = MF.getRegInfo();
359 auto InsertPt = MBB.getFirstTerminator();
360 DebugLoc DL;
361
362 if (InsertPt != MBB.end())
363 DL = InsertPt->getDebugLoc();
364
365 // Restore the stack pointer. If we had fixed-size locals, add the offset
366 // subtracted in the prolog.
367 unsigned SPReg = 0;
368 unsigned SPFPReg = hasFP(MF) ? getFPReg(MF) : getSPReg(MF);
369 if (hasBP(MF)) {
370 auto FI = MF.getInfo<WebAssemblyFunctionInfo>();
371 SPReg = FI->getBasePointerVreg();
372 } else if (StackSize) {
373 const TargetRegisterClass *PtrRC =
374 MRI.getTargetRegisterInfo()->getPointerRegClass();
375 Register OffsetReg = MRI.createVirtualRegister(PtrRC);
376 BuildMI(MBB, InsertPt, DL, TII->get(getOpcConst(MF)), OffsetReg)
377 .addImm(StackSize);
378 // In the epilog we don't need to write the result back to the SP32/64
379 // physreg because it won't be used again. We can use a stackified register
380 // instead.
381 SPReg = MRI.createVirtualRegister(PtrRC);
382 BuildMI(MBB, InsertPt, DL, TII->get(getOpcAdd(MF)), SPReg)
383 .addReg(SPFPReg)
384 .addReg(OffsetReg);
385 } else {
386 SPReg = SPFPReg;
387 }
388
389 writeBackSP(SPReg, MF, MBB, InsertPt, DL);
390}
391
393 TargetStackID::Value ID) const {
394 // Use the Object stack for WebAssembly locals which can only be accessed
395 // by name, not via an address in linear memory.
397 return true;
398
400}
401
405 Loc.Kind = DwarfFrameBase::WasmFrameBase;
407 if (needsSP(MF) && MFI.isFrameBaseVirtual()) {
408 unsigned LocalNum = MFI.getFrameBaseLocal();
409 Loc.Location.WasmLoc = {WebAssembly::TI_LOCAL, LocalNum};
410 } else {
411 // There is no __stack_pointer global in libcall thread context mode, so
412 // TI_GLOBAL_RELOC would produce a bogus relocation. We take care to ensure
413 // that this code is not reached in that case, but assert here to be sure.
415
416 // TODO: This should work on a breakpoint at a function with no frame,
417 // but probably won't work for traversing up the stack.
418 Loc.Location.WasmLoc = {WebAssembly::TI_GLOBAL_RELOC, 0};
419 }
420 return Loc;
421}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock & MBB
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
const HexagonInstrInfo * TII
#define I(x, y, z)
Definition MD5.cpp:57
static constexpr MCPhysReg SPReg
This class implements WebAssembly-specific bits of TargetFrameLowering class.
This file contains the WebAssembly implementation of the TargetInstrInfo class.
This file provides WebAssembly-specific target descriptions.
This file declares WebAssembly-specific per-machine-function information.
This file declares the WebAssembly-specific subclass of TargetSubtarget.
This file declares the WebAssembly-specific subclass of TargetMachine.
This file contains the declaration of the WebAssembly-specific type parsing utility functions.
This file contains the entry points for global functions defined in the LLVM WebAssembly back-end.
an instruction to allocate memory on the stack
Type * getAllocatedType() const
Return the type that is being allocated by the instruction.
unsigned getAddressSpace() const
Return the address space for the allocation.
A debug info location.
Definition DebugLoc.h:123
DISubprogram * getSubprogram() const
Get the attached subprogram.
bool hasPersonalityFn() const
Check whether this function has a personality function.
Definition Function.h:905
bool hasFnAttribute(Attribute::AttrKind Kind) const
Return true if the function has the attribute.
Definition Function.cpp:728
ExceptionHandling getExceptionHandlingType() const
Definition MCAsmInfo.h:659
MachineInstrBundleIterator< MachineInstr > iterator
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
bool hasVarSizedObjects() const
This method may be called any time after instruction selection is complete to determine if the stack ...
uint64_t getStackSize() const
Return the number of bytes that must be allocated to hold all of the fixed size frame objects.
const AllocaInst * getObjectAllocation(int ObjectIdx) const
Return the underlying Alloca of the specified stack object if it exists.
bool hasCalls() const
Return true if the current function has any function calls.
bool isFrameAddressTaken() const
This method may be called any time after instruction selection is complete to determine if there is a...
void setObjectOffset(int ObjectIdx, int64_t SPOffset)
Set the stack frame offset of the specified object.
bool hasPatchPoint() const
This method may be called any time after instruction selection is complete to determine if there is a...
void setObjectSize(int ObjectIdx, int64_t Size)
Change the size of the specified stack object.
void setStackID(int ObjectIdx, uint8_t ID)
bool hasStackMap() const
This method may be called any time after instruction selection is complete to determine if there is a...
uint8_t getStackID(int ObjectIdx) const
int64_t getObjectOffset(int ObjectIdx) const
Return the assigned stack offset of the specified object from the incoming stack pointer.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
const char * createExternalSymbolName(StringRef Name)
Allocate a string and populate it with the given external symbol name.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
const DataLayout & getDataLayout() const
Return the DataLayout attached to the Module associated to this MF.
Function & getFunction()
Return the LLVM function that this machine code represents.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const TargetMachine & getTarget() const
getTarget - Return the target machine this machine code is compiled with
const MachineInstrBuilder & addExternalSymbol(const char *FnName, unsigned TargetFlags=0) const
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
MachineOperand class - Representation of each machine instruction operand.
Wrapper class representing virtual and physical registers.
Definition Register.h:20
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
bool hasFP(const MachineFunction &MF) const
hasFP - Return true if the specified function should have a dedicated frame pointer register.
virtual bool isSupportedStackID(TargetStackID::Value ID) const
const MCAsmInfo & getMCAsmInfo() const
Return target specific asm information.
static unsigned getOpcAdd(const MachineFunction &MF)
static unsigned getFPReg(const MachineFunction &MF)
bool needsPrologForEH(const MachineFunction &MF) const
static unsigned getOpcGlobSet(const MachineFunction &MF)
bool hasReservedCallFrame(const MachineFunction &MF) const override
Under normal circumstances, when a frame pointer is not required, we reserve argument space for call ...
static unsigned getOpcAnd(const MachineFunction &MF)
static unsigned getSPReg(const MachineFunction &MF)
static unsigned getOpcGlobGet(const MachineFunction &MF)
DwarfFrameBase getDwarfFrameBase(const MachineFunction &MF) const override
Return the frame base information to be encoded in the DWARF subprogram debug info.
void writeBackSP(unsigned SrcReg, MachineFunction &MF, MachineBasicBlock &MBB, MachineBasicBlock::iterator &InsertStore, const DebugLoc &DL) const
Write SP back to __stack_pointer global, or call __wasm_set_stack_pointer.
bool isSupportedStackID(TargetStackID::Value ID) const override
MachineBasicBlock::iterator eliminateCallFramePseudoInstr(MachineFunction &MF, MachineBasicBlock &MBB, MachineBasicBlock::iterator I) const override
This method is called during prolog/epilog code insertion to eliminate call frame setup and destroy p...
static unsigned getOpcConst(const MachineFunction &MF)
bool hasFPImpl(const MachineFunction &MF) const override
Return true if the specified function should have a dedicated frame pointer register.
void emitPrologue(MachineFunction &MF, MachineBasicBlock &MBB) const override
These methods insert prolog and epilog code into the function.
void emitEpilogue(MachineFunction &MF, MachineBasicBlock &MBB) const override
static const size_t RedZoneSize
Size of the red zone for the user stack (leaf functions can use this much space below the stack point...
static std::optional< unsigned > getLocalForStackObject(MachineFunction &MF, int FrameIndex)
static unsigned getOpcSub(const MachineFunction &MF)
This class is derived from MachineFunctionInfo and contains private WebAssembly-specific information ...
const std::vector< MVT > & getLocals() const
const std::vector< MVT > & getParams() const
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
bool isArgument(unsigned Opc)
bool isWasmVarAddressSpace(unsigned AS)
This is an optimization pass for GlobalISel generic memory operations.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
LLVM_ABI void ComputeValueVTs(const TargetLowering &TLI, const DataLayout &DL, Type *Ty, SmallVectorImpl< EVT > &ValueVTs, SmallVectorImpl< EVT > *MemVTs=nullptr, SmallVectorImpl< TypeSize > *Offsets=nullptr, TypeSize StartingOffset=TypeSize::getZero())
ComputeValueVTs - Given an LLVM IR type, compute a sequence of EVTs that represent all the individual...
Definition Analysis.cpp:119
@ Wasm
WebAssembly Exception Handling.
Definition CodeGen.h:59
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
Definition STLExtras.h:1745
This struct is a compact representation of a valid (non-zero power of two) alignment.
Definition Alignment.h:39
constexpr uint64_t value() const
This is a hole in the type system and should not be abused.
Definition Alignment.h:77
Extended Value Type.
Definition ValueTypes.h:35