Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 1 | //===- DWARFFormValue.cpp -------------------------------------------------===// |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | |
Adrian Prantl | 0c36a75 | 2015-01-06 16:50:25 +0000 | [diff] [blame] | 10 | #include "SyntaxHighlighting.h" |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 11 | #include "llvm/ADT/ArrayRef.h" |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 12 | #include "llvm/ADT/None.h" |
| 13 | #include "llvm/ADT/Optional.h" |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 14 | #include "llvm/ADT/StringRef.h" |
Zachary Turner | 82af943 | 2015-01-30 18:07:45 +0000 | [diff] [blame] | 15 | #include "llvm/DebugInfo/DWARF/DWARFContext.h" |
| 16 | #include "llvm/DebugInfo/DWARF/DWARFFormValue.h" |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 17 | #include "llvm/DebugInfo/DWARF/DWARFRelocMap.h" |
| 18 | #include "llvm/DebugInfo/DWARF/DWARFUnit.h" |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 19 | #include "llvm/Support/Dwarf.h" |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 20 | #include "llvm/Support/ErrorHandling.h" |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 21 | #include "llvm/Support/Format.h" |
| 22 | #include "llvm/Support/raw_ostream.h" |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 23 | #include <cinttypes> |
| 24 | #include <cstdint> |
Matt Arsenault | 45cbfa5 | 2015-10-21 21:10:12 +0000 | [diff] [blame] | 25 | #include <limits> |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 26 | |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 27 | using namespace llvm; |
| 28 | using namespace dwarf; |
Adrian Prantl | 0c36a75 | 2015-01-06 16:50:25 +0000 | [diff] [blame] | 29 | using namespace syntax; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 30 | |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 31 | static const DWARFFormValue::FormClass DWARF4FormClasses[] = { |
| 32 | DWARFFormValue::FC_Unknown, // 0x0 |
| 33 | DWARFFormValue::FC_Address, // 0x01 DW_FORM_addr |
| 34 | DWARFFormValue::FC_Unknown, // 0x02 unused |
| 35 | DWARFFormValue::FC_Block, // 0x03 DW_FORM_block2 |
| 36 | DWARFFormValue::FC_Block, // 0x04 DW_FORM_block4 |
| 37 | DWARFFormValue::FC_Constant, // 0x05 DW_FORM_data2 |
| 38 | // --- These can be FC_SectionOffset in DWARF3 and below: |
| 39 | DWARFFormValue::FC_Constant, // 0x06 DW_FORM_data4 |
| 40 | DWARFFormValue::FC_Constant, // 0x07 DW_FORM_data8 |
| 41 | // --- |
| 42 | DWARFFormValue::FC_String, // 0x08 DW_FORM_string |
| 43 | DWARFFormValue::FC_Block, // 0x09 DW_FORM_block |
| 44 | DWARFFormValue::FC_Block, // 0x0a DW_FORM_block1 |
| 45 | DWARFFormValue::FC_Constant, // 0x0b DW_FORM_data1 |
| 46 | DWARFFormValue::FC_Flag, // 0x0c DW_FORM_flag |
| 47 | DWARFFormValue::FC_Constant, // 0x0d DW_FORM_sdata |
| 48 | DWARFFormValue::FC_String, // 0x0e DW_FORM_strp |
| 49 | DWARFFormValue::FC_Constant, // 0x0f DW_FORM_udata |
| 50 | DWARFFormValue::FC_Reference, // 0x10 DW_FORM_ref_addr |
| 51 | DWARFFormValue::FC_Reference, // 0x11 DW_FORM_ref1 |
| 52 | DWARFFormValue::FC_Reference, // 0x12 DW_FORM_ref2 |
| 53 | DWARFFormValue::FC_Reference, // 0x13 DW_FORM_ref4 |
| 54 | DWARFFormValue::FC_Reference, // 0x14 DW_FORM_ref8 |
| 55 | DWARFFormValue::FC_Reference, // 0x15 DW_FORM_ref_udata |
| 56 | DWARFFormValue::FC_Indirect, // 0x16 DW_FORM_indirect |
| 57 | DWARFFormValue::FC_SectionOffset, // 0x17 DW_FORM_sec_offset |
| 58 | DWARFFormValue::FC_Exprloc, // 0x18 DW_FORM_exprloc |
| 59 | DWARFFormValue::FC_Flag, // 0x19 DW_FORM_flag_present |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 60 | }; |
| 61 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 62 | namespace { |
| 63 | |
| 64 | /// A helper class that can be used in DWARFFormValue.cpp functions that need |
| 65 | /// to know the byte size of DW_FORM values that vary in size depending on the |
| 66 | /// DWARF version, address byte size, or DWARF32 or DWARF64. |
| 67 | class FormSizeHelper { |
| 68 | uint16_t Version; |
| 69 | uint8_t AddrSize; |
| 70 | llvm::dwarf::DwarfFormat Format; |
| 71 | |
| 72 | public: |
| 73 | FormSizeHelper(uint16_t V, uint8_t A, llvm::dwarf::DwarfFormat F) |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 74 | : Version(V), AddrSize(A), Format(F) {} |
| 75 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 76 | uint8_t getAddressByteSize() const { return AddrSize; } |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 77 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 78 | uint8_t getRefAddrByteSize() const { |
| 79 | if (Version == 2) |
| 80 | return AddrSize; |
| 81 | return getDwarfOffsetByteSize(); |
| 82 | } |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 83 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 84 | uint8_t getDwarfOffsetByteSize() const { |
| 85 | switch (Format) { |
| 86 | case dwarf::DwarfFormat::DWARF32: |
| 87 | return 4; |
| 88 | case dwarf::DwarfFormat::DWARF64: |
| 89 | return 8; |
| 90 | } |
| 91 | llvm_unreachable("Invalid Format value"); |
| 92 | } |
| 93 | }; |
| 94 | |
| 95 | } // end anonymous namespace |
| 96 | |
| 97 | template <class T> |
| 98 | static Optional<uint8_t> getFixedByteSize(dwarf::Form Form, const T *U) { |
| 99 | switch (Form) { |
| 100 | case DW_FORM_addr: |
| 101 | if (U) |
| 102 | return U->getAddressByteSize(); |
| 103 | return None; |
| 104 | |
| 105 | case DW_FORM_block: // ULEB128 length L followed by L bytes. |
| 106 | case DW_FORM_block1: // 1 byte length L followed by L bytes. |
| 107 | case DW_FORM_block2: // 2 byte length L followed by L bytes. |
| 108 | case DW_FORM_block4: // 4 byte length L followed by L bytes. |
| 109 | case DW_FORM_string: // C-string with null terminator. |
| 110 | case DW_FORM_sdata: // SLEB128. |
| 111 | case DW_FORM_udata: // ULEB128. |
| 112 | case DW_FORM_ref_udata: // ULEB128. |
| 113 | case DW_FORM_indirect: // ULEB128. |
| 114 | case DW_FORM_exprloc: // ULEB128 length L followed by L bytes. |
| 115 | case DW_FORM_strx: // ULEB128. |
| 116 | case DW_FORM_addrx: // ULEB128. |
| 117 | case DW_FORM_loclistx: // ULEB128. |
| 118 | case DW_FORM_rnglistx: // ULEB128. |
| 119 | case DW_FORM_GNU_addr_index: // ULEB128. |
| 120 | case DW_FORM_GNU_str_index: // ULEB128. |
| 121 | return None; |
| 122 | |
| 123 | case DW_FORM_ref_addr: |
| 124 | if (U) |
| 125 | return U->getRefAddrByteSize(); |
| 126 | return None; |
| 127 | |
| 128 | case DW_FORM_flag: |
| 129 | case DW_FORM_data1: |
| 130 | case DW_FORM_ref1: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 131 | case DW_FORM_strx1: |
| 132 | case DW_FORM_addrx1: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 133 | return 1; |
| 134 | |
| 135 | case DW_FORM_data2: |
| 136 | case DW_FORM_ref2: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 137 | case DW_FORM_strx2: |
| 138 | case DW_FORM_addrx2: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 139 | return 2; |
| 140 | |
| 141 | case DW_FORM_data4: |
| 142 | case DW_FORM_ref4: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 143 | case DW_FORM_ref_sup4: |
| 144 | case DW_FORM_strx4: |
| 145 | case DW_FORM_addrx4: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 146 | return 4; |
| 147 | |
Greg Clayton | 04c1928 | 2016-11-11 17:38:14 +0000 | [diff] [blame] | 148 | case DW_FORM_strp: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 149 | case DW_FORM_GNU_ref_alt: |
| 150 | case DW_FORM_GNU_strp_alt: |
| 151 | case DW_FORM_line_strp: |
| 152 | case DW_FORM_sec_offset: |
| 153 | case DW_FORM_strp_sup: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 154 | if (U) |
| 155 | return U->getDwarfOffsetByteSize(); |
| 156 | return None; |
| 157 | |
| 158 | case DW_FORM_data8: |
| 159 | case DW_FORM_ref8: |
| 160 | case DW_FORM_ref_sig8: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 161 | case DW_FORM_ref_sup8: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 162 | return 8; |
| 163 | |
| 164 | case DW_FORM_flag_present: |
| 165 | return 0; |
| 166 | |
| 167 | case DW_FORM_data16: |
| 168 | return 16; |
| 169 | |
| 170 | case DW_FORM_implicit_const: |
Victor Leschuk | cbddae7 | 2017-01-10 21:18:26 +0000 | [diff] [blame] | 171 | // The implicit value is stored in the abbreviation as a SLEB128, and |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 172 | // there no data in debug info. |
| 173 | return 0; |
| 174 | |
| 175 | default: |
| 176 | llvm_unreachable("Handle this form in this switch statement"); |
| 177 | } |
| 178 | return None; |
| 179 | } |
| 180 | |
| 181 | template <class T> |
| 182 | static bool skipFormValue(dwarf::Form Form, const DataExtractor &DebugInfoData, |
| 183 | uint32_t *OffsetPtr, const T *U) { |
| 184 | bool Indirect = false; |
| 185 | do { |
| 186 | switch (Form) { |
Greg Clayton | 04c1928 | 2016-11-11 17:38:14 +0000 | [diff] [blame] | 187 | // Blocks of inlined data that have a length field and the data bytes |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 188 | // inlined in the .debug_info. |
| 189 | case DW_FORM_exprloc: |
| 190 | case DW_FORM_block: { |
| 191 | uint64_t size = DebugInfoData.getULEB128(OffsetPtr); |
| 192 | *OffsetPtr += size; |
| 193 | return true; |
| 194 | } |
| 195 | case DW_FORM_block1: { |
| 196 | uint8_t size = DebugInfoData.getU8(OffsetPtr); |
| 197 | *OffsetPtr += size; |
| 198 | return true; |
| 199 | } |
| 200 | case DW_FORM_block2: { |
| 201 | uint16_t size = DebugInfoData.getU16(OffsetPtr); |
| 202 | *OffsetPtr += size; |
| 203 | return true; |
| 204 | } |
| 205 | case DW_FORM_block4: { |
| 206 | uint32_t size = DebugInfoData.getU32(OffsetPtr); |
| 207 | *OffsetPtr += size; |
| 208 | return true; |
| 209 | } |
| 210 | |
| 211 | // Inlined NULL terminated C-strings. |
| 212 | case DW_FORM_string: |
| 213 | DebugInfoData.getCStr(OffsetPtr); |
| 214 | return true; |
| 215 | |
| 216 | case DW_FORM_addr: |
| 217 | case DW_FORM_ref_addr: |
| 218 | case DW_FORM_flag_present: |
| 219 | case DW_FORM_data1: |
| 220 | case DW_FORM_data2: |
| 221 | case DW_FORM_data4: |
| 222 | case DW_FORM_data8: |
| 223 | case DW_FORM_flag: |
| 224 | case DW_FORM_ref1: |
| 225 | case DW_FORM_ref2: |
| 226 | case DW_FORM_ref4: |
| 227 | case DW_FORM_ref8: |
| 228 | case DW_FORM_ref_sig8: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 229 | case DW_FORM_ref_sup4: |
| 230 | case DW_FORM_ref_sup8: |
| 231 | case DW_FORM_strx1: |
| 232 | case DW_FORM_strx2: |
| 233 | case DW_FORM_strx4: |
| 234 | case DW_FORM_addrx1: |
| 235 | case DW_FORM_addrx2: |
| 236 | case DW_FORM_addrx4: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 237 | case DW_FORM_sec_offset: |
| 238 | case DW_FORM_strp: |
Greg Clayton | 04c1928 | 2016-11-11 17:38:14 +0000 | [diff] [blame] | 239 | case DW_FORM_strp_sup: |
| 240 | case DW_FORM_line_strp: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 241 | case DW_FORM_GNU_ref_alt: |
| 242 | case DW_FORM_GNU_strp_alt: |
| 243 | if (Optional<uint8_t> FixedSize = ::getFixedByteSize(Form, U)) { |
| 244 | *OffsetPtr += *FixedSize; |
| 245 | return true; |
| 246 | } |
| 247 | return false; |
| 248 | |
| 249 | // signed or unsigned LEB 128 values. |
| 250 | case DW_FORM_sdata: |
| 251 | DebugInfoData.getSLEB128(OffsetPtr); |
| 252 | return true; |
| 253 | |
| 254 | case DW_FORM_udata: |
| 255 | case DW_FORM_ref_udata: |
| 256 | case DW_FORM_strx: |
| 257 | case DW_FORM_addrx: |
| 258 | case DW_FORM_loclistx: |
| 259 | case DW_FORM_rnglistx: |
| 260 | case DW_FORM_GNU_addr_index: |
| 261 | case DW_FORM_GNU_str_index: |
| 262 | DebugInfoData.getULEB128(OffsetPtr); |
| 263 | return true; |
| 264 | |
| 265 | case DW_FORM_indirect: |
| 266 | Indirect = true; |
| 267 | Form = static_cast<dwarf::Form>(DebugInfoData.getULEB128(OffsetPtr)); |
| 268 | break; |
| 269 | |
| 270 | default: |
| 271 | return false; |
| 272 | } |
| 273 | } while (Indirect); |
| 274 | return true; |
| 275 | } |
| 276 | |
| 277 | Optional<uint8_t> DWARFFormValue::getFixedByteSize(dwarf::Form Form, |
| 278 | const DWARFUnit *U) { |
| 279 | return ::getFixedByteSize(Form, U); |
| 280 | } |
| 281 | |
| 282 | Optional<uint8_t> |
| 283 | DWARFFormValue::getFixedByteSize(dwarf::Form Form, uint16_t Version, |
| 284 | uint8_t AddrSize, |
| 285 | llvm::dwarf::DwarfFormat Format) { |
| 286 | FormSizeHelper FSH(Version, AddrSize, Format); |
| 287 | return ::getFixedByteSize(Form, &FSH); |
| 288 | } |
| 289 | |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 290 | bool DWARFFormValue::isFormClass(DWARFFormValue::FormClass FC) const { |
| 291 | // First, check DWARF4 form classes. |
Craig Topper | 0013be1 | 2015-09-21 05:32:41 +0000 | [diff] [blame] | 292 | if (Form < makeArrayRef(DWARF4FormClasses).size() && |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 293 | DWARF4FormClasses[Form] == FC) |
| 294 | return true; |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 295 | // Check more forms from DWARF4 and DWARF5 proposals. |
| 296 | switch (Form) { |
| 297 | case DW_FORM_ref_sig8: |
| 298 | case DW_FORM_GNU_ref_alt: |
Alexey Samsonov | cbd806a | 2013-10-29 16:32:19 +0000 | [diff] [blame] | 299 | return (FC == FC_Reference); |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 300 | case DW_FORM_GNU_addr_index: |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 301 | return (FC == FC_Address); |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 302 | case DW_FORM_GNU_str_index: |
| 303 | case DW_FORM_GNU_strp_alt: |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 304 | return (FC == FC_String); |
Victor Leschuk | cbddae7 | 2017-01-10 21:18:26 +0000 | [diff] [blame] | 305 | case DW_FORM_implicit_const: |
| 306 | return (FC == FC_Constant); |
Greg Clayton | 6c27376 | 2016-10-27 16:32:04 +0000 | [diff] [blame] | 307 | default: |
| 308 | break; |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 309 | } |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 310 | // In DWARF3 DW_FORM_data4 and DW_FORM_data8 served also as a section offset. |
| 311 | // Don't check for DWARF version here, as some producers may still do this |
Greg Clayton | 48432cf | 2017-05-01 22:07:02 +0000 | [diff] [blame^] | 312 | // by mistake. Also accept DW_FORM_strp since this is .debug_str section |
| 313 | // offset. |
| 314 | return (Form == DW_FORM_data4 || Form == DW_FORM_data8 || |
| 315 | Form == DW_FORM_strp) && |
Benjamin Kramer | 68a2956 | 2015-05-25 13:28:03 +0000 | [diff] [blame] | 316 | FC == FC_SectionOffset; |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 317 | } |
| 318 | |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 319 | bool DWARFFormValue::extractValue(const DataExtractor &data, |
| 320 | uint32_t *offset_ptr, |
David Blaikie | 07e2244 | 2013-09-23 22:44:40 +0000 | [diff] [blame] | 321 | const DWARFUnit *cu) { |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 322 | U = cu; |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 323 | bool indirect = false; |
| 324 | bool is_block = false; |
Craig Topper | 2617dcc | 2014-04-15 06:32:26 +0000 | [diff] [blame] | 325 | Value.data = nullptr; |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 326 | // Read the value for the form into value and follow and DW_FORM_indirect |
| 327 | // instances we run into |
| 328 | do { |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 329 | indirect = false; |
| 330 | switch (Form) { |
| 331 | case DW_FORM_addr: |
Eric Christopher | 7c678de | 2012-11-07 23:22:07 +0000 | [diff] [blame] | 332 | case DW_FORM_ref_addr: { |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 333 | if (!U) |
Frederic Riss | e4576d2 | 2014-11-12 23:48:04 +0000 | [diff] [blame] | 334 | return false; |
Alexey Samsonov | d60859b | 2013-04-09 14:09:42 +0000 | [diff] [blame] | 335 | uint16_t AddrSize = |
| 336 | (Form == DW_FORM_addr) |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 337 | ? U->getAddressByteSize() |
| 338 | : U->getRefAddrByteSize(); |
George Rimar | f8a9642 | 2017-04-21 09:12:18 +0000 | [diff] [blame] | 339 | Value.uval = |
| 340 | getRelocatedValue(data, AddrSize, offset_ptr, U->getRelocMap()); |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 341 | break; |
Alexey Samsonov | 9cb13d5 | 2012-11-12 14:25:36 +0000 | [diff] [blame] | 342 | } |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 343 | case DW_FORM_exprloc: |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 344 | case DW_FORM_block: |
| 345 | Value.uval = data.getULEB128(offset_ptr); |
| 346 | is_block = true; |
| 347 | break; |
| 348 | case DW_FORM_block1: |
| 349 | Value.uval = data.getU8(offset_ptr); |
| 350 | is_block = true; |
| 351 | break; |
| 352 | case DW_FORM_block2: |
| 353 | Value.uval = data.getU16(offset_ptr); |
| 354 | is_block = true; |
| 355 | break; |
| 356 | case DW_FORM_block4: |
| 357 | Value.uval = data.getU32(offset_ptr); |
| 358 | is_block = true; |
| 359 | break; |
| 360 | case DW_FORM_data1: |
| 361 | case DW_FORM_ref1: |
| 362 | case DW_FORM_flag: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 363 | case DW_FORM_strx1: |
| 364 | case DW_FORM_addrx1: |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 365 | Value.uval = data.getU8(offset_ptr); |
| 366 | break; |
| 367 | case DW_FORM_data2: |
| 368 | case DW_FORM_ref2: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 369 | case DW_FORM_strx2: |
| 370 | case DW_FORM_addrx2: |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 371 | Value.uval = data.getU16(offset_ptr); |
| 372 | break; |
| 373 | case DW_FORM_data4: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 374 | case DW_FORM_ref4: |
| 375 | case DW_FORM_ref_sup4: |
| 376 | case DW_FORM_strx4: |
| 377 | case DW_FORM_addrx4: { |
George Rimar | f8a9642 | 2017-04-21 09:12:18 +0000 | [diff] [blame] | 378 | const RelocAddrMap* RelocMap = U ? U->getRelocMap() : nullptr; |
| 379 | Value.uval = getRelocatedValue(data, 4, offset_ptr, RelocMap); |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 380 | break; |
David Blaikie | 18e7350 | 2013-06-19 21:37:13 +0000 | [diff] [blame] | 381 | } |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 382 | case DW_FORM_data8: |
| 383 | case DW_FORM_ref8: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 384 | case DW_FORM_ref_sup8: |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 385 | Value.uval = data.getU64(offset_ptr); |
| 386 | break; |
| 387 | case DW_FORM_sdata: |
| 388 | Value.sval = data.getSLEB128(offset_ptr); |
| 389 | break; |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 390 | case DW_FORM_udata: |
| 391 | case DW_FORM_ref_udata: |
| 392 | Value.uval = data.getULEB128(offset_ptr); |
| 393 | break; |
| 394 | case DW_FORM_string: |
| 395 | Value.cstr = data.getCStr(offset_ptr); |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 396 | break; |
| 397 | case DW_FORM_indirect: |
Greg Clayton | 6c27376 | 2016-10-27 16:32:04 +0000 | [diff] [blame] | 398 | Form = static_cast<dwarf::Form>(data.getULEB128(offset_ptr)); |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 399 | indirect = true; |
| 400 | break; |
Greg Clayton | 04c1928 | 2016-11-11 17:38:14 +0000 | [diff] [blame] | 401 | case DW_FORM_strp: |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 402 | case DW_FORM_sec_offset: |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 403 | case DW_FORM_GNU_ref_alt: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 404 | case DW_FORM_GNU_strp_alt: |
| 405 | case DW_FORM_line_strp: |
Paul Robinson | f96e21a | 2017-03-06 22:20:03 +0000 | [diff] [blame] | 406 | case DW_FORM_strp_sup: { |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 407 | if (!U) |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 408 | return false; |
George Rimar | f8a9642 | 2017-04-21 09:12:18 +0000 | [diff] [blame] | 409 | Value.uval = getRelocatedValue(data, U->getDwarfOffsetByteSize(), |
| 410 | offset_ptr, U->getRelocMap()); |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 411 | break; |
Eric Christopher | 55863be | 2013-04-07 03:43:09 +0000 | [diff] [blame] | 412 | } |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 413 | case DW_FORM_flag_present: |
| 414 | Value.uval = 1; |
| 415 | break; |
| 416 | case DW_FORM_ref_sig8: |
| 417 | Value.uval = data.getU64(offset_ptr); |
| 418 | break; |
Eric Christopher | 59c53c2 | 2012-11-16 23:44:11 +0000 | [diff] [blame] | 419 | case DW_FORM_GNU_addr_index: |
Eric Christopher | 59c53c2 | 2012-11-16 23:44:11 +0000 | [diff] [blame] | 420 | case DW_FORM_GNU_str_index: |
| 421 | Value.uval = data.getULEB128(offset_ptr); |
| 422 | break; |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 423 | default: |
Greg Clayton | 0e62ee7 | 2017-01-13 00:13:42 +0000 | [diff] [blame] | 424 | // DWARFFormValue::skipValue() will have caught this and caused all |
| 425 | // DWARF DIEs to fail to be parsed, so this code is not be reachable. |
| 426 | llvm_unreachable("unsupported form"); |
Benjamin Kramer | 123bfbb | 2011-09-15 03:11:09 +0000 | [diff] [blame] | 427 | } |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 428 | } while (indirect); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 429 | |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 430 | if (is_block) { |
| 431 | StringRef str = data.getData().substr(*offset_ptr, Value.uval); |
Craig Topper | 2617dcc | 2014-04-15 06:32:26 +0000 | [diff] [blame] | 432 | Value.data = nullptr; |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 433 | if (!str.empty()) { |
| 434 | Value.data = reinterpret_cast<const uint8_t *>(str.data()); |
| 435 | *offset_ptr += Value.uval; |
| 436 | } |
| 437 | } |
| 438 | |
| 439 | return true; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 440 | } |
| 441 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 442 | bool DWARFFormValue::skipValue(DataExtractor DebugInfoData, |
| 443 | uint32_t *offset_ptr, const DWARFUnit *U) const { |
| 444 | return DWARFFormValue::skipValue(Form, DebugInfoData, offset_ptr, U); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 445 | } |
| 446 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 447 | bool DWARFFormValue::skipValue(dwarf::Form form, DataExtractor DebugInfoData, |
| 448 | uint32_t *offset_ptr, const DWARFUnit *U) { |
| 449 | return skipFormValue(form, DebugInfoData, offset_ptr, U); |
David Blaikie | ad07b5d | 2015-12-04 17:20:04 +0000 | [diff] [blame] | 450 | } |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 451 | |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 452 | bool DWARFFormValue::skipValue(dwarf::Form form, DataExtractor DebugInfoData, |
| 453 | uint32_t *offset_ptr, uint16_t Version, |
| 454 | uint8_t AddrSize, |
| 455 | llvm::dwarf::DwarfFormat Format) { |
| 456 | FormSizeHelper FSH(Version, AddrSize, Format); |
| 457 | return skipFormValue(form, DebugInfoData, offset_ptr, &FSH); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 458 | } |
| 459 | |
| 460 | void |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 461 | DWARFFormValue::dump(raw_ostream &OS) const { |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 462 | uint64_t uvalue = Value.uval; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 463 | bool cu_relative_offset = false; |
| 464 | |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 465 | switch (Form) { |
Benjamin Kramer | 79730ad | 2011-11-05 16:01:13 +0000 | [diff] [blame] | 466 | case DW_FORM_addr: OS << format("0x%016" PRIx64, uvalue); break; |
Eric Christopher | 962c908 | 2013-01-15 23:56:56 +0000 | [diff] [blame] | 467 | case DW_FORM_GNU_addr_index: { |
Eric Christopher | 962c908 | 2013-01-15 23:56:56 +0000 | [diff] [blame] | 468 | OS << format(" indexed (%8.8x) address = ", (uint32_t)uvalue); |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 469 | uint64_t Address; |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 470 | if (U == nullptr) |
| 471 | OS << "<invalid dwarf unit>"; |
| 472 | else if (U->getAddrOffsetSectionItem(uvalue, Address)) |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 473 | OS << format("0x%016" PRIx64, Address); |
| 474 | else |
Eric Christopher | 962c908 | 2013-01-15 23:56:56 +0000 | [diff] [blame] | 475 | OS << "<no .debug_addr section>"; |
| 476 | break; |
| 477 | } |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 478 | case DW_FORM_flag_present: OS << "true"; break; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 479 | case DW_FORM_flag: |
Benjamin Kramer | 79730ad | 2011-11-05 16:01:13 +0000 | [diff] [blame] | 480 | case DW_FORM_data1: OS << format("0x%02x", (uint8_t)uvalue); break; |
| 481 | case DW_FORM_data2: OS << format("0x%04x", (uint16_t)uvalue); break; |
| 482 | case DW_FORM_data4: OS << format("0x%08x", (uint32_t)uvalue); break; |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 483 | case DW_FORM_ref_sig8: |
Benjamin Kramer | 79730ad | 2011-11-05 16:01:13 +0000 | [diff] [blame] | 484 | case DW_FORM_data8: OS << format("0x%016" PRIx64, uvalue); break; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 485 | case DW_FORM_string: |
| 486 | OS << '"'; |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 487 | OS.write_escaped(Value.cstr); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 488 | OS << '"'; |
| 489 | break; |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 490 | case DW_FORM_exprloc: |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 491 | case DW_FORM_block: |
| 492 | case DW_FORM_block1: |
| 493 | case DW_FORM_block2: |
| 494 | case DW_FORM_block4: |
| 495 | if (uvalue > 0) { |
| 496 | switch (Form) { |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 497 | case DW_FORM_exprloc: |
Benjamin Kramer | f3da529 | 2011-11-05 08:57:40 +0000 | [diff] [blame] | 498 | case DW_FORM_block: OS << format("<0x%" PRIx64 "> ", uvalue); break; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 499 | case DW_FORM_block1: OS << format("<0x%2.2x> ", (uint8_t)uvalue); break; |
| 500 | case DW_FORM_block2: OS << format("<0x%4.4x> ", (uint16_t)uvalue); break; |
| 501 | case DW_FORM_block4: OS << format("<0x%8.8x> ", (uint32_t)uvalue); break; |
| 502 | default: break; |
| 503 | } |
| 504 | |
| 505 | const uint8_t* data_ptr = Value.data; |
| 506 | if (data_ptr) { |
| 507 | // uvalue contains size of block |
| 508 | const uint8_t* end_data_ptr = data_ptr + uvalue; |
| 509 | while (data_ptr < end_data_ptr) { |
| 510 | OS << format("%2.2x ", *data_ptr); |
| 511 | ++data_ptr; |
| 512 | } |
| 513 | } |
| 514 | else |
| 515 | OS << "NULL"; |
| 516 | } |
| 517 | break; |
| 518 | |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 519 | case DW_FORM_sdata: OS << Value.sval; break; |
| 520 | case DW_FORM_udata: OS << Value.uval; break; |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 521 | case DW_FORM_strp: |
Benjamin Kramer | 07d4b1c | 2011-09-15 16:57:13 +0000 | [diff] [blame] | 522 | OS << format(" .debug_str[0x%8.8x] = ", (uint32_t)uvalue); |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 523 | dumpString(OS); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 524 | break; |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 525 | case DW_FORM_GNU_str_index: |
Eric Christopher | 2cbd576 | 2013-01-07 19:32:41 +0000 | [diff] [blame] | 526 | OS << format(" indexed (%8.8x) string = ", (uint32_t)uvalue); |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 527 | dumpString(OS); |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 528 | break; |
Eugene Zelenko | e94042c | 2017-02-27 23:43:14 +0000 | [diff] [blame] | 529 | case DW_FORM_GNU_strp_alt: |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 530 | OS << format("alt indirect string, offset: 0x%" PRIx64 "", uvalue); |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 531 | dumpString(OS); |
Eric Christopher | 2cbd576 | 2013-01-07 19:32:41 +0000 | [diff] [blame] | 532 | break; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 533 | case DW_FORM_ref_addr: |
Benjamin Kramer | 79730ad | 2011-11-05 16:01:13 +0000 | [diff] [blame] | 534 | OS << format("0x%016" PRIx64, uvalue); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 535 | break; |
| 536 | case DW_FORM_ref1: |
| 537 | cu_relative_offset = true; |
| 538 | OS << format("cu + 0x%2.2x", (uint8_t)uvalue); |
| 539 | break; |
| 540 | case DW_FORM_ref2: |
| 541 | cu_relative_offset = true; |
| 542 | OS << format("cu + 0x%4.4x", (uint16_t)uvalue); |
| 543 | break; |
| 544 | case DW_FORM_ref4: |
| 545 | cu_relative_offset = true; |
| 546 | OS << format("cu + 0x%4.4x", (uint32_t)uvalue); |
| 547 | break; |
| 548 | case DW_FORM_ref8: |
| 549 | cu_relative_offset = true; |
Benjamin Kramer | f3da529 | 2011-11-05 08:57:40 +0000 | [diff] [blame] | 550 | OS << format("cu + 0x%8.8" PRIx64, uvalue); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 551 | break; |
| 552 | case DW_FORM_ref_udata: |
| 553 | cu_relative_offset = true; |
Benjamin Kramer | f3da529 | 2011-11-05 08:57:40 +0000 | [diff] [blame] | 554 | OS << format("cu + 0x%" PRIx64, uvalue); |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 555 | break; |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 556 | case DW_FORM_GNU_ref_alt: |
| 557 | OS << format("<alt 0x%" PRIx64 ">", uvalue); |
| 558 | break; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 559 | |
| 560 | // All DW_FORM_indirect attributes should be resolved prior to calling |
| 561 | // this function |
| 562 | case DW_FORM_indirect: |
| 563 | OS << "DW_FORM_indirect"; |
| 564 | break; |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 565 | |
Eric Christopher | 4c7765f | 2013-01-17 03:00:04 +0000 | [diff] [blame] | 566 | // Should be formatted to 64-bit for DWARF64. |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 567 | case DW_FORM_sec_offset: |
Eric Christopher | 4c7765f | 2013-01-17 03:00:04 +0000 | [diff] [blame] | 568 | OS << format("0x%08x", (uint32_t)uvalue); |
Eric Christopher | d999bb7 | 2012-08-24 01:14:23 +0000 | [diff] [blame] | 569 | break; |
Eric Christopher | b2120fd | 2013-01-07 22:40:48 +0000 | [diff] [blame] | 570 | |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 571 | default: |
| 572 | OS << format("DW_FORM(0x%4.4x)", Form); |
| 573 | break; |
| 574 | } |
| 575 | |
Adrian Prantl | 0c36a75 | 2015-01-06 16:50:25 +0000 | [diff] [blame] | 576 | if (cu_relative_offset) { |
| 577 | OS << " => {"; |
| 578 | WithColor(OS, syntax::Address).get() |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 579 | << format("0x%8.8" PRIx64, uvalue + (U ? U->getOffset() : 0)); |
Adrian Prantl | 0c36a75 | 2015-01-06 16:50:25 +0000 | [diff] [blame] | 580 | OS << "}"; |
| 581 | } |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 582 | } |
| 583 | |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 584 | void DWARFFormValue::dumpString(raw_ostream &OS) const { |
| 585 | Optional<const char *> DbgStr = getAsCString(); |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 586 | if (DbgStr.hasValue()) { |
| 587 | raw_ostream &COS = WithColor(OS, syntax::String); |
| 588 | COS << '"'; |
| 589 | COS.write_escaped(DbgStr.getValue()); |
| 590 | COS << '"'; |
| 591 | } |
| 592 | } |
| 593 | |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 594 | Optional<const char *> DWARFFormValue::getAsCString() const { |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 595 | if (!isFormClass(FC_String)) |
| 596 | return None; |
| 597 | if (Form == DW_FORM_string) |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 598 | return Value.cstr; |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 599 | // FIXME: Add support for DW_FORM_GNU_strp_alt |
| 600 | if (Form == DW_FORM_GNU_strp_alt || U == nullptr) |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 601 | return None; |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 602 | uint32_t Offset = Value.uval; |
Alexey Samsonov | 742e6b8 | 2013-10-18 07:13:32 +0000 | [diff] [blame] | 603 | if (Form == DW_FORM_GNU_str_index) { |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 604 | uint32_t StrOffset; |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 605 | if (!U->getStringOffsetSectionItem(Offset, StrOffset)) |
| 606 | return None; |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 607 | Offset = StrOffset; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 608 | } |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 609 | if (const char *Str = U->getStringExtractor().getCStr(&Offset)) { |
| 610 | return Str; |
| 611 | } |
| 612 | return None; |
Eric Christopher | 2cbd576 | 2013-01-07 19:32:41 +0000 | [diff] [blame] | 613 | } |
| 614 | |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 615 | Optional<uint64_t> DWARFFormValue::getAsAddress() const { |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 616 | if (!isFormClass(FC_Address)) |
| 617 | return None; |
Alexey Samsonov | 742e6b8 | 2013-10-18 07:13:32 +0000 | [diff] [blame] | 618 | if (Form == DW_FORM_GNU_addr_index) { |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 619 | uint32_t Index = Value.uval; |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 620 | uint64_t Result; |
Craig Topper | 2617dcc | 2014-04-15 06:32:26 +0000 | [diff] [blame] | 621 | if (!U || !U->getAddrOffsetSectionItem(Index, Result)) |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 622 | return None; |
| 623 | return Result; |
Alexey Samsonov | e3ba81b | 2013-08-27 09:20:22 +0000 | [diff] [blame] | 624 | } |
| 625 | return Value.uval; |
Eric Christopher | 962c908 | 2013-01-15 23:56:56 +0000 | [diff] [blame] | 626 | } |
| 627 | |
Greg Clayton | cddab27 | 2016-10-31 16:46:02 +0000 | [diff] [blame] | 628 | Optional<uint64_t> DWARFFormValue::getAsReference() const { |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 629 | if (!isFormClass(FC_Reference)) |
| 630 | return None; |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 631 | switch (Form) { |
| 632 | case DW_FORM_ref1: |
| 633 | case DW_FORM_ref2: |
| 634 | case DW_FORM_ref4: |
| 635 | case DW_FORM_ref8: |
| 636 | case DW_FORM_ref_udata: |
Craig Topper | 2617dcc | 2014-04-15 06:32:26 +0000 | [diff] [blame] | 637 | if (!U) |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 638 | return None; |
| 639 | return Value.uval + U->getOffset(); |
| 640 | case DW_FORM_ref_addr: |
Greg Clayton | 82f12b1 | 2016-11-11 16:21:37 +0000 | [diff] [blame] | 641 | case DW_FORM_ref_sig8: |
| 642 | case DW_FORM_GNU_ref_alt: |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 643 | return Value.uval; |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 644 | default: |
Alexey Samsonov | bf19a57 | 2015-05-19 20:29:28 +0000 | [diff] [blame] | 645 | return None; |
Benjamin Kramer | eaa7433 | 2011-09-13 21:47:32 +0000 | [diff] [blame] | 646 | } |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 647 | } |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 648 | |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 649 | Optional<uint64_t> DWARFFormValue::getAsSectionOffset() const { |
| 650 | if (!isFormClass(FC_SectionOffset)) |
| 651 | return None; |
| 652 | return Value.uval; |
| 653 | } |
| 654 | |
| 655 | Optional<uint64_t> DWARFFormValue::getAsUnsignedConstant() const { |
Frederic Riss | f3549a2 | 2014-09-04 06:14:35 +0000 | [diff] [blame] | 656 | if ((!isFormClass(FC_Constant) && !isFormClass(FC_Flag)) |
| 657 | || Form == DW_FORM_sdata) |
Alexey Samsonov | 48cbda5 | 2013-10-28 23:01:48 +0000 | [diff] [blame] | 658 | return None; |
| 659 | return Value.uval; |
Benjamin Kramer | aa2f78f | 2011-09-13 19:42:23 +0000 | [diff] [blame] | 660 | } |
Frederic Riss | f3549a2 | 2014-09-04 06:14:35 +0000 | [diff] [blame] | 661 | |
Frederic Riss | 77f850e | 2015-03-04 22:07:41 +0000 | [diff] [blame] | 662 | Optional<int64_t> DWARFFormValue::getAsSignedConstant() const { |
| 663 | if ((!isFormClass(FC_Constant) && !isFormClass(FC_Flag)) || |
Matt Arsenault | 45cbfa5 | 2015-10-21 21:10:12 +0000 | [diff] [blame] | 664 | (Form == DW_FORM_udata && uint64_t(std::numeric_limits<int64_t>::max()) < Value.uval)) |
Frederic Riss | 77f850e | 2015-03-04 22:07:41 +0000 | [diff] [blame] | 665 | return None; |
| 666 | switch (Form) { |
| 667 | case DW_FORM_data4: |
| 668 | return int32_t(Value.uval); |
| 669 | case DW_FORM_data2: |
| 670 | return int16_t(Value.uval); |
| 671 | case DW_FORM_data1: |
| 672 | return int8_t(Value.uval); |
| 673 | case DW_FORM_sdata: |
| 674 | case DW_FORM_data8: |
| 675 | default: |
| 676 | return Value.sval; |
| 677 | } |
| 678 | } |
| 679 | |
Frederic Riss | eab1777 | 2014-09-04 06:35:09 +0000 | [diff] [blame] | 680 | Optional<ArrayRef<uint8_t>> DWARFFormValue::getAsBlock() const { |
Frederic Riss | f3549a2 | 2014-09-04 06:14:35 +0000 | [diff] [blame] | 681 | if (!isFormClass(FC_Block) && !isFormClass(FC_Exprloc)) |
| 682 | return None; |
Craig Topper | 0013be1 | 2015-09-21 05:32:41 +0000 | [diff] [blame] | 683 | return makeArrayRef(Value.data, Value.uval); |
Frederic Riss | f3549a2 | 2014-09-04 06:14:35 +0000 | [diff] [blame] | 684 | } |
| 685 | |
Chris Bieneman | e0e451d | 2016-12-22 22:44:27 +0000 | [diff] [blame] | 686 | Optional<uint64_t> DWARFFormValue::getAsCStringOffset() const { |
| 687 | if (!isFormClass(FC_String) && Form == DW_FORM_string) |
| 688 | return None; |
| 689 | return Value.uval; |
| 690 | } |
| 691 | |
| 692 | Optional<uint64_t> DWARFFormValue::getAsReferenceUVal() const { |
| 693 | if (!isFormClass(FC_Reference)) |
| 694 | return None; |
| 695 | return Value.uval; |
| 696 | } |