27#define DEBUG_TYPE "dwarfdebug"
32 else if (
Value == std::numeric_limits<uint64_t>::max()) {
38 emitOp(dwarf::DW_OP_constu);
44 assert(DwarfReg >= 0 &&
"invalid negative dwarf register number");
46 "location description already locked down");
49 emitOp(dwarf::DW_OP_reg0 + DwarfReg, Comment);
51 emitOp(dwarf::DW_OP_regx, Comment);
57 assert(DwarfReg >= 0 &&
"invalid negative dwarf register number");
60 emitOp(dwarf::DW_OP_breg0 + DwarfReg);
62 emitOp(dwarf::DW_OP_bregx);
69 emitOp(dwarf::DW_OP_fbreg);
77 const unsigned SizeOfByte = 8;
79 emitOp(dwarf::DW_OP_bit_piece);
83 emitOp(dwarf::DW_OP_piece);
84 unsigned ByteSize = SizeInBits / SizeOfByte;
87 this->OffsetInBits += SizeInBits;
109 int64_t Reg =
TRI.getDwarfRegNumForVirtReg(MachineReg,
false);
117 int64_t Reg =
TRI.getDwarfRegNum(MachineReg,
false);
128 Reg =
TRI.getDwarfRegNum(SR,
false);
130 unsigned Idx =
TRI.getSubRegIndex(SR, MachineReg);
131 unsigned Size =
TRI.getSubRegIdxSize(Idx);
132 unsigned RegOffset =
TRI.getSubRegIdxOffset(Idx);
153 unsigned Idx =
TRI.getSubRegIndex(MachineReg, SR);
154 unsigned Size =
TRI.getSubRegIdxSize(Idx);
155 unsigned Offset =
TRI.getSubRegIdxOffset(Idx);
156 Reg =
TRI.getDwarfRegNum(SR,
false);
171 -1,
Offset - CurPos,
"no DWARF register encoding"));
176 Reg, std::min<unsigned>(
Size, MaxSize -
Offset),
"sub-register"));
188 -1,
RegSize - CurPos,
"no DWARF register encoding"));
194 emitOp(dwarf::DW_OP_stack_value);
201 emitOp(dwarf::DW_OP_lit0);
203 emitOp(dwarf::DW_OP_lit1);
209 emitOp(dwarf::DW_OP_consts);
247 "implicit value must be byte-sized");
249 emitOp(dwarf::DW_OP_implicit_value);
257 for (
unsigned I = 0;
I < NumBytes; ++
I)
265 if (NumBytes == 4 || NumBytes == 8 ) {
267 emitOp(dwarf::DW_OP_implicit_value);
276 for (
int i = 0; i < NumBytes; ++i) {
284 dbgs() <<
"Skipped DW_OP_implicit_value creation for ConstantFP of size: "
291 unsigned FragmentOffsetInBits) {
293 if (!
addMachineReg(
TRI, MachineReg, Fragment ? Fragment->SizeInBits : ~1U)) {
298 bool HasComplexExpression =
false;
299 auto Op = ExprCursor.
peek();
301 HasComplexExpression =
true;
332 if (Reg.DwarfRegNo >= 0)
333 addReg(Reg.DwarfRegNo, Reg.Comment);
335 if (
RegSize > FragmentInfo->SizeInBits)
347 emitOp(dwarf::DW_OP_stack_value);
353 auto NextOp = ExprCursor.
peek();
363 return Op.getOp() == dwarf::DW_OP_stack_value;
373 LLVM_DEBUG(
dbgs() <<
"TODO: giving up on debug information due to "
374 "multi-register usage.\n");
382 int SignedOffset = 0;
383 assert(!Reg.isSubRegister() &&
"full register expected");
387 if (
Op && (
Op->getOp() == dwarf::DW_OP_plus_uconst)) {
399 if (
Op &&
Op->getOp() == dwarf::DW_OP_constu) {
403 if (
N &&
N->getOp() == dwarf::DW_OP_plus &&
Offset <= IntMax) {
406 }
else if (
N &&
N->getOp() == dwarf::DW_OP_minus &&
408 SignedOffset = -
static_cast<int64_t
>(
Offset);
416 addBReg(Reg.DwarfRegNo, SignedOffset);
421 auto NextOp = ExprCursor.
peek();
431 if (
Loc.isIndirect())
437 if (
Loc.isIndirect())
446 auto Op = ExprCursor.
take();
451 "Can currently only emit entry values covering a single operation");
464 emitOp(
CU.getDwarf5OrGNULocationAtom(dwarf::DW_OP_entry_value));
485 "Began emitting entry value block before cancelling entry value");
495 unsigned I = 0, E =
CU.ExprRefedBaseTypes.size();
497 if (
CU.ExprRefedBaseTypes[
I].BitSize == BitSize &&
498 CU.ExprRefedBaseTypes[
I].Encoding == Encoding)
502 CU.ExprRefedBaseTypes.emplace_back(BitSize, Encoding);
510 auto Op = ExprCursor.
take();
511 switch (
Op->getOp()) {
512 case dwarf::DW_OP_deref:
536 std::optional<DIExpression::ExprOperand> PrevConvertOp;
539 auto Op = ExprCursor.take();
542 if (OpNum >= dwarf::DW_OP_reg0 && OpNum <= dwarf::DW_OP_reg31) {
545 }
else if (OpNum >= dwarf::DW_OP_breg0 && OpNum <= dwarf::DW_OP_breg31) {
546 addBReg(OpNum - dwarf::DW_OP_breg0,
Op->getArg(0));
552 if (!InsertArg(
Op->getArg(0), ExprCursor)) {
558 unsigned SizeInBits =
Op->getArg(1);
559 unsigned FragmentOffset =
Op->getArg(0);
589 unsigned SizeInBits =
Op->getArg(1);
590 unsigned BitOffset =
Op->getArg(0);
591 unsigned DerefSize = 0;
594 unsigned PtrSizeInBytes =
CU.getAsmPrinter()->MAI.getCodePointerSize();
600 DerefSize =
alignTo(BitOffset + SizeInBits, 8) / 8;
601 if (DerefSize == PtrSizeInBytes) {
602 emitOp(dwarf::DW_OP_deref);
604 emitOp(dwarf::DW_OP_deref_size);
613 if (8 * DerefSize == SizeInBits) {
617 emitOp(dwarf::DW_OP_constu);
625 unsigned LeftShift = PtrSizeInBytes * 8 - (SizeInBits + BitOffset);
626 unsigned RightShift = LeftShift + BitOffset;
628 emitOp(dwarf::DW_OP_constu);
633 emitOp(dwarf::DW_OP_constu);
646 case dwarf::DW_OP_plus_uconst:
648 emitOp(dwarf::DW_OP_plus_uconst);
651 case dwarf::DW_OP_plus:
652 case dwarf::DW_OP_minus:
653 case dwarf::DW_OP_mul:
654 case dwarf::DW_OP_div:
655 case dwarf::DW_OP_mod:
656 case dwarf::DW_OP_or:
657 case dwarf::DW_OP_and:
658 case dwarf::DW_OP_xor:
659 case dwarf::DW_OP_shl:
660 case dwarf::DW_OP_shr:
661 case dwarf::DW_OP_shra:
662 case dwarf::DW_OP_lit0:
663 case dwarf::DW_OP_not:
664 case dwarf::DW_OP_dup:
665 case dwarf::DW_OP_push_object_address:
666 case dwarf::DW_OP_over:
667 case dwarf::DW_OP_rot:
668 case dwarf::DW_OP_eq:
669 case dwarf::DW_OP_ne:
670 case dwarf::DW_OP_gt:
671 case dwarf::DW_OP_ge:
672 case dwarf::DW_OP_lt:
673 case dwarf::DW_OP_le:
674 case dwarf::DW_OP_neg:
675 case dwarf::DW_OP_abs:
678 case dwarf::DW_OP_deref:
685 emitOp(dwarf::DW_OP_deref);
687 case dwarf::DW_OP_constu:
691 case dwarf::DW_OP_consts:
693 emitOp(dwarf::DW_OP_consts);
697 unsigned BitSize =
Op->getArg(0);
700 emitOp(dwarf::DW_OP_convert);
709 if (PrevConvertOp && PrevConvertOp->getArg(0) < BitSize) {
710 if (Encoding == dwarf::DW_ATE_signed)
712 else if (Encoding == dwarf::DW_ATE_unsigned)
714 PrevConvertOp = std::nullopt;
721 case dwarf::DW_OP_stack_value:
724 case dwarf::DW_OP_swap:
726 emitOp(dwarf::DW_OP_swap);
728 case dwarf::DW_OP_xderef:
730 emitOp(dwarf::DW_OP_xderef);
732 case dwarf::DW_OP_deref_size:
733 emitOp(dwarf::DW_OP_deref_size);
739 case dwarf::DW_OP_regx:
740 emitOp(dwarf::DW_OP_regx);
743 case dwarf::DW_OP_bregx:
744 emitOp(dwarf::DW_OP_bregx);
792 "overlapping or duplicate fragments");
801 emitOp(dwarf::DW_OP_constu);
804 emitOp(dwarf::DW_OP_lit0);
807 emitOp(dwarf::DW_OP_constu);
816 if (FromBits / 7 < 1+1+1+1+1) {
818 emitOp(dwarf::DW_OP_constu);
826 emitOp(dwarf::DW_OP_lit1);
827 emitOp(dwarf::DW_OP_constu);
830 emitOp(dwarf::DW_OP_lit1);
831 emitOp(dwarf::DW_OP_minus);
837 emitOp(dwarf::DW_OP_WASM_location);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file implements a class to represent arbitrary precision integral constant values and operations...
static bool isMemoryLocation(DIExpressionCursor ExprCursor)
Assuming a well-formed expression, match "DW_OP_deref* DW_OP_LLVM_fragment?
This file contains constants used for implementing Dwarf debug support.
Register const TargetRegisterInfo * TRI
This file implements the SmallBitVector class.
APInt bitcastToAPInt() const
Class for arbitrary precision integers.
uint64_t getZExtValue() const
Get zero extended value.
unsigned getBitWidth() const
Return the number of bits in the APInt.
LLVM_ABI APInt byteSwap() const
LLVM_ABI APInt extractBits(unsigned numBits, unsigned bitPosition) const
Return an APInt with the extracted bits [bitPosition,bitPosition+numBits).
APInt lshr(unsigned shiftAmt) const
Logical right-shift function.
This class is intended to be used as a driving class for all asm writers.
const DataLayout & getDataLayout() const
Return information about data layout.
Holds a DIExpression and keeps track of how many operands have been consumed so far.
std::optional< DIExpression::ExprOperand > peekNext() const
Return the next operation.
std::optional< DIExpression::FragmentInfo > getFragmentInfo() const
Retrieve the fragment information, if any.
std::optional< DIExpression::ExprOperand > peek() const
Return the current operation.
void consume(unsigned N)
Consume N operations.
std::optional< DIExpression::ExprOperand > take()
Consume one operation.
A lightweight wrapper around an expression operand.
LLVM_ABI bool isEntryValue() const
Check if the expression consists of exactly one entry value operand.
bool isFragment() const
Return whether this is a piece of an aggregate variable.
static LLVM_ABI std::optional< FragmentInfo > getFragmentInfo(expr_op_iterator Start, expr_op_iterator End)
Retrieve the details of this fragment expression.
void addAnd(unsigned Mask)
Emit a bitwise and dwarf operation.
void setLocation(const MachineLocation &Loc, const DIExpression *DIExpr)
Set the location (Loc) and DIExpression (DIExpr) to describe.
virtual void emitOp(uint8_t Op, const char *Comment=nullptr)=0
Output a dwarf operand and an optional assembler comment.
virtual void disableTemporaryBuffer()=0
Disable emission to the temporary buffer.
bool isUnknownLocation() const
virtual unsigned getTemporaryBufferSize()=0
Return the emitted size, in number of bytes, for the data stored in the temporary buffer.
uint64_t OffsetInBits
Current Fragment Offset in Bits.
virtual bool isFrameRegister(const TargetRegisterInfo &TRI, llvm::Register MachineReg)=0
Return whether the given machine register is the frame register in the current function.
void finalize()
This needs to be called last to commit any pending changes.
void addFragmentOffset(const DIExpression *Expr)
If applicable, emit an empty DW_OP_piece / DW_OP_bit_piece to advance to the fragment described by Ex...
void emitLegacySExt(unsigned FromBits)
void cancelEntryValue()
Cancel the emission of an entry value.
bool isRegisterLocation() const
void setMemoryLocationKind()
Lock this down to become a memory location description.
virtual void emitBaseTypeRef(uint64_t Idx)=0
virtual void emitData1(uint8_t Value)=0
bool addMachineReg(const TargetRegisterInfo &TRI, llvm::Register MachineReg, unsigned MaxSize=~1U)
Emit a partial DWARF register operation.
unsigned SavedLocationKind
std::optional< uint8_t > TagOffset
bool isImplicitLocation() const
virtual void emitUnsigned(uint64_t Value)=0
Emit a raw unsigned value.
void addBooleanConstant(int64_t Value)
Emit a boolean constant.
void addConstantFP(const APFloat &Value, const AsmPrinter &AP)
Emit an floating point constant.
void maskSubRegister()
Add masking operations to stencil out a subregister.
SmallVector< Register, 2 > DwarfRegs
The register location, if any.
bool addMachineRegExpression(const TargetRegisterInfo &TRI, DIExpressionCursor &Expr, llvm::Register MachineReg, unsigned FragmentOffsetInBits=0)
Emit a machine register location.
void addStackValue()
Emit a DW_OP_stack_value, if supported.
void finalizeEntryValue()
Finalize an entry value by emitting its size operand, and committing the DWARF block which has been e...
bool isMemoryLocation() const
void addUnsignedConstant(uint64_t Value)
Emit an unsigned constant.
unsigned SubRegisterSizeInBits
Sometimes we need to add a DW_OP_bit_piece to describe a subregister.
void addFBReg(int64_t Offset)
Emit DW_OP_fbreg <Offset>.
void setSubRegisterPiece(unsigned SizeInBits, unsigned OffsetInBits)
Push a DW_OP_piece / DW_OP_bit_piece for emitting later, if one is needed to represent a subregister.
void addExpression(DIExpressionCursor &&Expr)
Emit all remaining operations in the DIExpressionCursor.
bool isEntryValue() const
unsigned getOrCreateBaseType(unsigned BitSize, dwarf::TypeKind Encoding)
Return the index of a base type with the given properties and create one if necessary.
void addImplicitValue(const APInt &Value, const AsmPrinter &AP)
Emit an implicit value.
void addSignedConstant(int64_t Value)
Emit a signed constant.
void emitLegacyZExt(unsigned FromBits)
bool IsEmittingEntryValue
Whether we are currently emitting an entry value operation.
virtual void emitSigned(int64_t Value)=0
Emit a raw signed value.
void addReg(int64_t DwarfReg, const char *Comment=nullptr)
Emit a DW_OP_reg operation.
unsigned SubRegisterOffsetInBits
void setEntryValueFlags(const MachineLocation &Loc)
Lock this down to become an entry value location.
virtual void commitTemporaryBuffer()=0
Commit the data stored in the temporary buffer to the main output.
void addShr(unsigned ShiftBy)
Emit a shift-right dwarf operation.
void addWasmLocation(unsigned Index, uint64_t Offset)
Emit location information expressed via WebAssembly location + offset The Index is an identifier for ...
virtual void enableTemporaryBuffer()=0
Start emitting data to the temporary buffer.
void emitConstu(uint64_t Value)
Emit a normalized unsigned constant.
void beginEntryValueExpression(DIExpressionCursor &ExprCursor)
Begin emission of an entry value dwarf operation.
void addOpPiece(unsigned SizeInBits, unsigned OffsetInBits=0)
Emit a DW_OP_piece or DW_OP_bit_piece operation for a variable fragment.
void addBReg(int64_t DwarfReg, int64_t Offset)
Emit a DW_OP_breg operation.
Wrapper class representing virtual and physical registers.
constexpr bool isPhysical() const
Return true if the specified register number is in the physical register namespace.
This is a 'bitvector' (really, a variable-sized bit array), optimized for the case when the array is ...
bool subsetOf(const SmallBitVector &RHS) const
Check if This is a subset of RHS.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
LLVM Value Representation.
An efficient, type-erasing, non-owning reference to a callable.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ DW_OP_LLVM_entry_value
Only used in LLVM metadata.
@ DW_OP_LLVM_implicit_pointer
Only used in LLVM metadata.
@ DW_OP_LLVM_extract_bits_zext
Only used in LLVM metadata.
@ DW_OP_LLVM_tag_offset
Only used in LLVM metadata.
@ DW_OP_LLVM_fragment
Only used in LLVM metadata.
@ DW_OP_LLVM_arg
Only used in LLVM metadata.
@ DW_OP_LLVM_convert
Only used in LLVM metadata.
@ DW_OP_LLVM_extract_bits_sext
Only used in LLVM metadata.
This is an optimization pass for GlobalISel generic memory operations.
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
constexpr uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
uint16_t MCPhysReg
An unsigned integer type large enough to represent all physical registers, but not necessarily virtua...
DWARFExpression::Operation Op
Holds information about all subregisters comprising a register location.
static Register createRegister(int64_t RegNo, const char *Comment)
Create a full register, no extra DW_OP_piece operators necessary.
static Register createSubRegister(int64_t RegNo, unsigned SizeInBits, const char *Comment)
Create a subregister that needs a DW_OP_piece operator with SizeInBits.