9#ifndef LLVM_DEBUGINFO_DWARF_DWARFTYPEPRINTER_H
10#define LLVM_DEBUGINFO_DWARF_DWARFTYPEPRINTER_H
46 std::string *OriginalFullName =
nullptr);
49 bool SkipFirstParamIfArtificial =
false);
60 std::string *OriginalFullName =
nullptr);
63 bool SkipFirstParamIfArtificial,
bool Const,
71 case dwarf::DW_TAG_structure_type:
72 case dwarf::DW_TAG_class_type:
73 case dwarf::DW_TAG_union_type:
74 case dwarf::DW_TAG_namespace:
75 case dwarf::DW_TAG_enumeration_type:
76 case dwarf::DW_TAG_typedef:
86 template <
typename FormValueType>
87 void appendCastedValue(
const FormValueType &FormValue, DieType Cast,
91template <
typename DieType>
94 static constexpr StringRef Prefix =
"DW_TAG_";
95 static constexpr StringRef Suffix =
"_type";
99 TagStr.
size() - (Prefix.size() + Suffix.
size()))
103template <
typename DieType>
105 for (
const DieType &
C :
D.children()) {
106 if (
C.getTag() != dwarf::DW_TAG_subrange_type)
108 std::optional<uint64_t> LB;
109 std::optional<uint64_t>
Count;
110 std::optional<uint64_t> UB;
111 std::optional<unsigned> DefaultLB;
112 if (std::optional<typename DieType::DWARFFormValue> L =
113 C.find(dwarf::DW_AT_lower_bound))
114 LB = L->getAsUnsignedConstant();
115 if (std::optional<typename DieType::DWARFFormValue> CountV =
116 C.find(dwarf::DW_AT_count))
117 Count = CountV->getAsUnsignedConstant();
118 if (std::optional<typename DieType::DWARFFormValue> UpperV =
119 C.find(dwarf::DW_AT_upper_bound))
120 UB = UpperV->getAsUnsignedConstant();
121 if (std::optional<uint64_t> LV =
D.getLanguage())
124 if (LB && *LB == *DefaultLB)
126 if (!LB && !
Count && !UB)
128 else if (!LB && (
Count || UB) && DefaultLB)
129 OS <<
'[' << (
Count ? *
Count : *UB - *DefaultLB + 1) <<
']';
153template <
typename DieType>
156 return D.resolveReferencedType(Attr);
158template <
typename DieType>
160 return D.resolveReferencedType(
F);
162template <
typename DWARFFormValueType>
163const char *
toString(std::optional<DWARFFormValueType>
F) {
179 auto TypeAttr =
D.find(dwarf::DW_AT_type);
184 if (!Unwrapped || Unwrapped.getTag() != dwarf::DW_TAG_typedef)
187 if (!Visited.
insert(Unwrapped.getOffset()).second)
195template <
typename DieType>
197 while (
D && (
D.getTag() == dwarf::DW_TAG_const_type ||
198 D.getTag() == dwarf::DW_TAG_volatile_type))
203template <
typename DieType>
206 return D && (
D.getTag() == dwarf::DW_TAG_subroutine_type ||
207 D.getTag() == dwarf::DW_TAG_array_type);
210template <
typename DieType>
224template <
typename DieType>
226 DieType
D, std::string *OriginalFullName) {
236 case dwarf::DW_TAG_pointer_type: {
240 case dwarf::DW_TAG_subroutine_type: {
248 case dwarf::DW_TAG_array_type: {
252 case dwarf::DW_TAG_reference_type:
255 case dwarf::DW_TAG_rvalue_reference_type:
258 case dwarf::DW_TAG_ptr_to_member_type: {
274 case dwarf::DW_TAG_LLVM_ptrauth_type:
277 case dwarf::DW_TAG_const_type:
278 case dwarf::DW_TAG_volatile_type:
281 case dwarf::DW_TAG_namespace: {
285 OS <<
"(anonymous namespace)";
288 case dwarf::DW_TAG_unspecified_type: {
290 if (TypeName ==
"decltype(nullptr)")
291 TypeName =
"std::nullptr_t";
311 static constexpr StringRef MangledPrefix =
"_STN|";
312 if (Name.consume_front(MangledPrefix)) {
313 auto Separator = Name.find(
'|');
315 StringRef BaseName = Name.substr(0, Separator);
317 if (OriginalFullName)
326 if (Name.ends_with(
">"))
342template <
typename DieType>
344 DieType
D, DieType Inner,
bool SkipFirstParamIfArtificial) {
347 switch (
D.getTag()) {
348 case dwarf::DW_TAG_subroutine_type: {
353 case dwarf::DW_TAG_array_type: {
357 case dwarf::DW_TAG_const_type:
358 case dwarf::DW_TAG_volatile_type:
361 case dwarf::DW_TAG_ptr_to_member_type:
362 case dwarf::DW_TAG_reference_type:
363 case dwarf::DW_TAG_rvalue_reference_type:
364 case dwarf::DW_TAG_pointer_type: {
369 dwarf::DW_TAG_ptr_to_member_type);
372 case dwarf::DW_TAG_LLVM_ptrauth_type: {
374 if (
auto Form =
D.find(Attr))
375 return *Form->getAsUnsignedConstant();
379 if (getValOrNull(dwarf::DW_AT_LLVM_ptrauth_isa_pointer))
381 if (getValOrNull(dwarf::DW_AT_LLVM_ptrauth_authenticates_null_values))
382 optionsVec.
push_back(
"authenticates-null-values");
383 if (
auto AuthenticationMode =
384 D.find(dwarf::DW_AT_LLVM_ptrauth_authentication_mode)) {
385 switch (*AuthenticationMode->getAsUnsignedConstant()) {
399 for (
const auto *option : optionsVec) {
405 options =
", \"" + options +
"\"";
406 std::string PtrauthString;
409 <<
"__ptrauth(" << getValOrNull(dwarf::DW_AT_LLVM_ptrauth_key) <<
", "
410 << getValOrNull(dwarf::DW_AT_LLVM_ptrauth_address_discriminated)
413 getValOrNull(dwarf::DW_AT_LLVM_ptrauth_extra_discriminator),
416 OS << PtrauthStream.
str();
431template <
typename DieType>
433 if (
D && scopedTAGs(
D.getTag()))
438template <
typename DieType>
440 if (
D && scopedTAGs(
D.getTag()))
445template <
typename DieType>
446template <
typename FormValueType>
447void DWARFTypePrinter<DieType>::appendCastedValue(
448 const FormValueType &FormValue, DieType Cast,
bool IsUnsigned) {
451 std::optional<uint64_t> UVal = FormValue.getAsUnsignedConstant();
455 ValStr = std::to_string(*UVal);
457 std::optional<int64_t> SVal = FormValue.getAsSignedConstant();
461 ValStr = std::to_string(*SVal);
465 appendQualifiedName(Cast);
467 OS << std::move(ValStr);
470template <
typename DieType>
472 bool *FirstParameter) {
473 bool FirstParameterValue =
true;
474 bool IsTemplate =
false;
476 FirstParameter = &FirstParameterValue;
477 for (
const DieType &
C :
D) {
485 *FirstParameter =
false;
487 if (
C.getTag() == dwarf::DW_TAG_GNU_template_parameter_pack) {
491 if (
C.getTag() == dwarf::DW_TAG_template_value_parameter) {
494 if (
T.getTag() == dwarf::DW_TAG_enumeration_type) {
495 auto V =
C.find(dwarf::DW_AT_const_value);
496 appendCastedValue(*V,
T,
false);
503 if (
T.getTag() == dwarf::DW_TAG_pointer_type ||
504 T.getTag() == dwarf::DW_TAG_reference_type ||
505 T.getTag() == dwarf::DW_TAG_ptr_to_member_type)
510 auto V =
C.find(dwarf::DW_AT_const_value);
511 bool IsQualifiedChar =
false;
512 if (Name ==
"bool") {
513 OS << (*V->getAsUnsignedConstant() ?
"true" :
"false");
514 }
else if (Name ==
"short") {
516 OS << std::to_string(*V->getAsSignedConstant());
517 }
else if (Name ==
"unsigned short") {
518 OS <<
"(unsigned short)";
519 OS << std::to_string(*V->getAsSignedConstant());
520 }
else if (Name ==
"int")
521 OS << std::to_string(*V->getAsSignedConstant());
522 else if (Name ==
"long") {
523 OS << std::to_string(*V->getAsSignedConstant());
525 }
else if (Name ==
"long long") {
526 OS << std::to_string(*V->getAsSignedConstant());
528 }
else if (Name ==
"unsigned int") {
529 OS << std::to_string(*V->getAsUnsignedConstant());
531 }
else if (Name ==
"unsigned long") {
532 OS << std::to_string(*V->getAsUnsignedConstant());
534 }
else if (Name ==
"unsigned long long") {
535 OS << std::to_string(*V->getAsUnsignedConstant());
537 }
else if (Name ==
"char" ||
539 (Name ==
"unsigned char" || Name ==
"signed char"))) {
542 auto Val = *V->getAsSignedConstant();
547 if (IsQualifiedChar) {
582 if ((Val & ~0xFFu) == ~0xFFu)
584 if (Val < 127 && Val >= 32) {
588 }
else if (Val < 256)
590 else if (Val <= 0xFFFF)
597 }
else if (Name.starts_with(
"_BitInt")) {
598 appendCastedValue(*V,
T,
false);
599 }
else if (Name.starts_with(
"unsigned _BitInt")) {
600 appendCastedValue(*V,
T,
true);
604 if (
C.getTag() == dwarf::DW_TAG_GNU_template_template_param) {
605 const char *RawName =
613 if (
C.getTag() != dwarf::DW_TAG_template_type_parameter)
619 if (IsTemplate && *FirstParameter && FirstParameter == &FirstParameterValue) {
626template <
typename DieType>
640template <
typename DieType>
642 DieType &
C, DieType &V) {
643 (
N.getTag() == dwarf::DW_TAG_const_type ?
C : V) =
N;
646 auto Tag =
T.getTag();
647 if (
Tag == dwarf::DW_TAG_const_type) {
650 }
else if (
Tag == dwarf::DW_TAG_volatile_type) {
657template <
typename DieType>
663 if (
T &&
T.getTag() == dwarf::DW_TAG_subroutine_type)
665 static_cast<bool>(
C),
static_cast<bool>(V));
670template <
typename DieType>
676 bool Subroutine =
T &&
T.getTag() == dwarf::DW_TAG_subroutine_type;
678 while (
A &&
A.getTag() == dwarf::DW_TAG_array_type)
681 (!
A || (
A.getTag() != dwarf::DW_TAG_pointer_type &&
682 A.getTag() != llvm::dwarf::DW_TAG_ptr_to_member_type)) &&
691 if (!Leading && !Subroutine) {
703template <
typename DieType>
705 DieType
D, std::string *OriginalFullName) {
712template <
typename DieType>
714 DieType
D, DieType Inner,
bool SkipFirstParamIfArtificial,
bool Const,
716 DieType FirstParamIfArtificial;
720 bool RealFirst =
true;
721 for (DieType
P :
D) {
722 if (
P.getTag() != dwarf::DW_TAG_formal_parameter &&
723 P.getTag() != dwarf::DW_TAG_unspecified_parameters)
726 if (SkipFirstParamIfArtificial && RealFirst &&
727 P.find(dwarf::DW_AT_artificial)) {
728 FirstParamIfArtificial =
T;
733 if (
P.getTag() == dwarf::DW_TAG_unspecified_parameters)
740 if (FirstParamIfArtificial) {
741 if (DieType
P = FirstParamIfArtificial) {
742 if (
P.getTag() == dwarf::DW_TAG_pointer_type) {
743 auto CVStep = [&](DieType CV) {
745 Const |= U.getTag() == dwarf::DW_TAG_const_type;
746 Volatile |= U.getTag() == dwarf::DW_TAG_volatile_type;
751 if (DieType CV = CVStep(
P)) {
758 if (
auto CC =
D.find(dwarf::DW_AT_calling_convention)) {
759 switch (*CC->getAsUnsignedConstant()) {
760 case dwarf::CallingConvention::DW_CC_BORLAND_stdcall:
761 OS <<
" __attribute__((stdcall))";
763 case dwarf::CallingConvention::DW_CC_BORLAND_msfastcall:
764 OS <<
" __attribute__((fastcall))";
766 case dwarf::CallingConvention::DW_CC_BORLAND_thiscall:
767 OS <<
" __attribute__((thiscall))";
769 case dwarf::CallingConvention::DW_CC_LLVM_vectorcall:
770 OS <<
" __attribute__((vectorcall))";
772 case dwarf::CallingConvention::DW_CC_BORLAND_pascal:
773 OS <<
" __attribute__((pascal))";
775 case dwarf::CallingConvention::DW_CC_LLVM_Win64:
776 OS <<
" __attribute__((ms_abi))";
778 case dwarf::CallingConvention::DW_CC_LLVM_X86_64SysV:
779 OS <<
" __attribute__((sysv_abi))";
781 case dwarf::CallingConvention::DW_CC_LLVM_AAPCS:
783 OS <<
" __attribute__((pcs(\"aapcs\")))";
785 case dwarf::CallingConvention::DW_CC_LLVM_AAPCS_VFP:
786 OS <<
" __attribute__((pcs(\"aapcs-vfp\")))";
788 case dwarf::CallingConvention::DW_CC_LLVM_IntelOclBicc:
789 OS <<
" __attribute__((intel_ocl_bicc))";
791 case dwarf::CallingConvention::DW_CC_LLVM_SpirFunction:
798 case dwarf::CallingConvention::DW_CC_LLVM_DeviceKernel:
799 OS <<
" __attribute__((device_kernel))";
801 case dwarf::CallingConvention::DW_CC_LLVM_Swift:
803 OS <<
" __attribute__((swiftcall))";
805 case dwarf::CallingConvention::DW_CC_LLVM_PreserveMost:
806 OS <<
" __attribute__((preserve_most))";
808 case dwarf::CallingConvention::DW_CC_LLVM_PreserveAll:
809 OS <<
" __attribute__((preserve_all))";
811 case dwarf::CallingConvention::DW_CC_LLVM_PreserveNone:
812 OS <<
" __attribute__((preserve_none))";
814 case dwarf::CallingConvention::DW_CC_LLVM_X86RegCall:
815 OS <<
" __attribute__((regcall))";
817 case dwarf::CallingConvention::DW_CC_LLVM_M68kRTD:
818 OS <<
" __attribute__((m68k_rtd))";
827 if (
D.find(dwarf::DW_AT_reference))
829 if (
D.find(dwarf::DW_AT_rvalue_reference))
835template <
typename DieType>
837 if (
D.getTag() == dwarf::DW_TAG_compile_unit)
839 if (
D.getTag() == dwarf::DW_TAG_type_unit)
841 if (
D.getTag() == dwarf::DW_TAG_skeleton_unit)
843 if (
D.getTag() == dwarf::DW_TAG_subprogram)
845 if (
D.getTag() == dwarf::DW_TAG_lexical_block)
847 D =
D.resolveTypeUnitReference();
848 if (DieType
P =
D.getParent())
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
This file contains constants used for implementing Dwarf debug support.
This file defines the SmallSet class.
static dwarf::Attribute TypeAttr[]
Tagged union holding either a T or a Error.
A helper class to return the specified delimiter string after the first invocation of operator String...
SmallSet - This maintains a set of unique values, optimizing for the case when the set is small (less...
std::pair< const_iterator, bool > insert(const T &V)
insert - Insert an element into the set if it isn't already there.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Represent a constant reference to a string, i.e.
static constexpr size_t npos
constexpr StringRef substr(size_t Start, size_t N=npos) const
Return a reference to the substring from [Start, Start + N).
bool starts_with(StringRef Prefix) const
Check if this string starts with the given Prefix.
constexpr size_t size() const
Get the string size.
bool ends_with(StringRef Suffix) const
Check if this string ends with the given Suffix.
This class implements an extremely fast bulk output stream that can only output to a stream.
A raw_ostream that writes to an std::string.
std::string & str()
Returns the string's reference.
@ C
The default llvm calling convention, compatible with C.
const char * toString(std::optional< DWARFFormValueType > F)
DieType resolveReferencedType(DieType D, dwarf::Attribute Attr=dwarf::DW_AT_type)
DieType unwrapReferencedTypedefType(DieType D)
Resolve the DW_AT_type of D until we reach a DIE that is not a DW_TAG_typedef.
This is an optimization pass for GlobalISel generic memory operations.
std::string utohexstr(uint64_t X, bool LowerCase=false, unsigned Width=0)
FunctionAddr VTableAddr Count
format_object< Ts... > format(const char *Fmt, const Ts &... Vals)
These are helper functions used to produce formatted output.
void consumeError(Error Err)
Consume a Error without doing anything.
void appendTypeTagName(dwarf::Tag T)
Dump the name encoded in the type tag.
bool needsParens(DieType D)
void appendArrayType(const DieType &D)
DieType appendQualifiedNameBefore(DieType D)
void appendQualifiedName(DieType D)
void decomposeConstVolatile(DieType &N, DieType &T, DieType &C, DieType &V)
DieType skipQualifiers(DieType D)
void appendUnqualifiedName(DieType D, std::string *OriginalFullName=nullptr)
Recursively append the DIE type name when applicable.
void appendConstVolatileQualifierBefore(DieType N)
void appendScopes(DieType D)
DWARFTypePrinter(raw_ostream &OS)
void appendSubroutineNameAfter(DieType D, DieType Inner, bool SkipFirstParamIfArtificial, bool Const, bool Volatile)
bool appendTemplateParameters(DieType D, bool *FirstParameter=nullptr)
void appendConstVolatileQualifierAfter(DieType N)
void appendAndTerminateTemplateParameters(DieType D)
void appendPointerLikeTypeBefore(DieType D, DieType Inner, StringRef Ptr)
void appendUnqualifiedNameAfter(DieType D, DieType Inner, bool SkipFirstParamIfArtificial=false)
DieType appendUnqualifiedNameBefore(DieType D, std::string *OriginalFullName=nullptr)