]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - source/Plugins/SymbolFile/DWARF/DWARFDebugRanges.cpp
Vendor import of lldb trunk r290819:
[FreeBSD/FreeBSD.git] / source / Plugins / SymbolFile / DWARF / DWARFDebugRanges.cpp
1 //===-- DWARFDebugRanges.cpp ------------------------------------*- C++ -*-===//
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
10 #include "DWARFDebugRanges.h"
11 #include "SymbolFileDWARF.h"
12 #include "lldb/Core/Stream.h"
13 #include <assert.h>
14
15 using namespace lldb_private;
16 using namespace std;
17
18 DWARFDebugRanges::DWARFDebugRanges() : m_range_map() {}
19
20 DWARFDebugRanges::~DWARFDebugRanges() {}
21
22 void DWARFDebugRanges::Extract(SymbolFileDWARF *dwarf2Data) {
23   DWARFRangeList range_list;
24   lldb::offset_t offset = 0;
25   dw_offset_t debug_ranges_offset = offset;
26   while (Extract(dwarf2Data, &offset, range_list)) {
27     range_list.Sort();
28     m_range_map[debug_ranges_offset] = range_list;
29     debug_ranges_offset = offset;
30   }
31 }
32
33 bool DWARFDebugRanges::Extract(SymbolFileDWARF *dwarf2Data,
34                                lldb::offset_t *offset_ptr,
35                                DWARFRangeList &range_list) {
36   range_list.Clear();
37
38   lldb::offset_t range_offset = *offset_ptr;
39   const DWARFDataExtractor &debug_ranges_data =
40       dwarf2Data->get_debug_ranges_data();
41   uint32_t addr_size = debug_ranges_data.GetAddressByteSize();
42
43   while (
44       debug_ranges_data.ValidOffsetForDataOfSize(*offset_ptr, 2 * addr_size)) {
45     dw_addr_t begin = debug_ranges_data.GetMaxU64(offset_ptr, addr_size);
46     dw_addr_t end = debug_ranges_data.GetMaxU64(offset_ptr, addr_size);
47     if (!begin && !end) {
48       // End of range list
49       break;
50     }
51     // Extend 4 byte addresses that consists of 32 bits of 1's to be 64 bits
52     // of ones
53     switch (addr_size) {
54     case 2:
55       if (begin == 0xFFFFull)
56         begin = LLDB_INVALID_ADDRESS;
57       break;
58
59     case 4:
60       if (begin == 0xFFFFFFFFull)
61         begin = LLDB_INVALID_ADDRESS;
62       break;
63
64     case 8:
65       break;
66
67     default:
68       assert(!"DWARFRangeList::Extract() unsupported address size.");
69       break;
70     }
71
72     // Filter out empty ranges
73     if (begin < end)
74       range_list.Append(DWARFRangeList::Entry(begin, end - begin));
75   }
76
77   // Make sure we consumed at least something
78   return range_offset != *offset_ptr;
79 }
80
81 void DWARFDebugRanges::Dump(Stream &s,
82                             const DWARFDataExtractor &debug_ranges_data,
83                             lldb::offset_t *offset_ptr,
84                             dw_addr_t cu_base_addr) {
85   uint32_t addr_size = s.GetAddressByteSize();
86   bool verbose = s.GetVerbose();
87
88   dw_addr_t base_addr = cu_base_addr;
89   while (
90       debug_ranges_data.ValidOffsetForDataOfSize(*offset_ptr, 2 * addr_size)) {
91     dw_addr_t begin = debug_ranges_data.GetMaxU64(offset_ptr, addr_size);
92     dw_addr_t end = debug_ranges_data.GetMaxU64(offset_ptr, addr_size);
93     // Extend 4 byte addresses that consists of 32 bits of 1's to be 64 bits
94     // of ones
95     if (begin == 0xFFFFFFFFull && addr_size == 4)
96       begin = LLDB_INVALID_ADDRESS;
97
98     s.Indent();
99     if (verbose) {
100       s.AddressRange(begin, end, sizeof(dw_addr_t), " offsets = ");
101     }
102
103     if (begin == 0 && end == 0) {
104       s.PutCString(" End");
105       break;
106     } else if (begin == LLDB_INVALID_ADDRESS) {
107       // A base address selection entry
108       base_addr = end;
109       s.Address(base_addr, sizeof(dw_addr_t), " Base address = ");
110     } else {
111       // Convert from offset to an address
112       dw_addr_t begin_addr = begin + base_addr;
113       dw_addr_t end_addr = end + base_addr;
114
115       s.AddressRange(begin_addr, end_addr, sizeof(dw_addr_t),
116                      verbose ? " ==> addrs = " : NULL);
117     }
118   }
119 }
120
121 bool DWARFDebugRanges::FindRanges(dw_addr_t debug_ranges_base,
122                                   dw_offset_t debug_ranges_offset,
123                                   DWARFRangeList &range_list) const {
124   dw_addr_t debug_ranges_address = debug_ranges_base + debug_ranges_offset;
125   range_map_const_iterator pos = m_range_map.find(debug_ranges_address);
126   if (pos != m_range_map.end()) {
127     range_list = pos->second;
128     return true;
129   }
130   return false;
131 }