1*9712c20fSFrederick Mayle // Copyright 2010 Google LLC
2*9712c20fSFrederick Mayle //
3*9712c20fSFrederick Mayle // Redistribution and use in source and binary forms, with or without
4*9712c20fSFrederick Mayle // modification, are permitted provided that the following conditions are
5*9712c20fSFrederick Mayle // met:
6*9712c20fSFrederick Mayle //
7*9712c20fSFrederick Mayle // * Redistributions of source code must retain the above copyright
8*9712c20fSFrederick Mayle // notice, this list of conditions and the following disclaimer.
9*9712c20fSFrederick Mayle // * Redistributions in binary form must reproduce the above
10*9712c20fSFrederick Mayle // copyright notice, this list of conditions and the following disclaimer
11*9712c20fSFrederick Mayle // in the documentation and/or other materials provided with the
12*9712c20fSFrederick Mayle // distribution.
13*9712c20fSFrederick Mayle // * Neither the name of Google LLC nor the names of its
14*9712c20fSFrederick Mayle // contributors may be used to endorse or promote products derived from
15*9712c20fSFrederick Mayle // this software without specific prior written permission.
16*9712c20fSFrederick Mayle //
17*9712c20fSFrederick Mayle // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*9712c20fSFrederick Mayle // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*9712c20fSFrederick Mayle // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20*9712c20fSFrederick Mayle // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21*9712c20fSFrederick Mayle // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22*9712c20fSFrederick Mayle // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23*9712c20fSFrederick Mayle // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*9712c20fSFrederick Mayle // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*9712c20fSFrederick Mayle // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*9712c20fSFrederick Mayle // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27*9712c20fSFrederick Mayle // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*9712c20fSFrederick Mayle
29*9712c20fSFrederick Mayle // Original author: Jim Blandy <[email protected]> <[email protected]>
30*9712c20fSFrederick Mayle
31*9712c20fSFrederick Mayle // cfi_assembler.cc: Implementation of google_breakpad::CFISection class.
32*9712c20fSFrederick Mayle // See cfi_assembler.h for details.
33*9712c20fSFrederick Mayle
34*9712c20fSFrederick Mayle #ifdef HAVE_CONFIG_H
35*9712c20fSFrederick Mayle #include <config.h> // Must come first
36*9712c20fSFrederick Mayle #endif
37*9712c20fSFrederick Mayle
38*9712c20fSFrederick Mayle #include "common/dwarf/cfi_assembler.h"
39*9712c20fSFrederick Mayle
40*9712c20fSFrederick Mayle #include <assert.h>
41*9712c20fSFrederick Mayle #include <stdlib.h>
42*9712c20fSFrederick Mayle
43*9712c20fSFrederick Mayle namespace google_breakpad {
44*9712c20fSFrederick Mayle
CIEHeader(uint64_t code_alignment_factor,int data_alignment_factor,unsigned return_address_register,uint8_t version,const string & augmentation,bool dwarf64,uint8_t address_size,uint8_t segment_size)45*9712c20fSFrederick Mayle CFISection& CFISection::CIEHeader(uint64_t code_alignment_factor,
46*9712c20fSFrederick Mayle int data_alignment_factor,
47*9712c20fSFrederick Mayle unsigned return_address_register,
48*9712c20fSFrederick Mayle uint8_t version,
49*9712c20fSFrederick Mayle const string& augmentation,
50*9712c20fSFrederick Mayle bool dwarf64,
51*9712c20fSFrederick Mayle uint8_t address_size,
52*9712c20fSFrederick Mayle uint8_t segment_size) {
53*9712c20fSFrederick Mayle assert(!entry_length_);
54*9712c20fSFrederick Mayle entry_length_ = new PendingLength();
55*9712c20fSFrederick Mayle in_fde_ = false;
56*9712c20fSFrederick Mayle
57*9712c20fSFrederick Mayle if (dwarf64) {
58*9712c20fSFrederick Mayle D32(kDwarf64InitialLengthMarker);
59*9712c20fSFrederick Mayle D64(entry_length_->length);
60*9712c20fSFrederick Mayle entry_length_->start = Here();
61*9712c20fSFrederick Mayle D64(eh_frame_ ? kEHFrame64CIEIdentifier : kDwarf64CIEIdentifier);
62*9712c20fSFrederick Mayle } else {
63*9712c20fSFrederick Mayle D32(entry_length_->length);
64*9712c20fSFrederick Mayle entry_length_->start = Here();
65*9712c20fSFrederick Mayle D32(eh_frame_ ? kEHFrame32CIEIdentifier : kDwarf32CIEIdentifier);
66*9712c20fSFrederick Mayle }
67*9712c20fSFrederick Mayle D8(version);
68*9712c20fSFrederick Mayle AppendCString(augmentation);
69*9712c20fSFrederick Mayle if (version >= 4) {
70*9712c20fSFrederick Mayle D8(address_size);
71*9712c20fSFrederick Mayle D8(segment_size);
72*9712c20fSFrederick Mayle }
73*9712c20fSFrederick Mayle ULEB128(code_alignment_factor);
74*9712c20fSFrederick Mayle LEB128(data_alignment_factor);
75*9712c20fSFrederick Mayle if (version == 1)
76*9712c20fSFrederick Mayle D8(return_address_register);
77*9712c20fSFrederick Mayle else
78*9712c20fSFrederick Mayle ULEB128(return_address_register);
79*9712c20fSFrederick Mayle return *this;
80*9712c20fSFrederick Mayle }
81*9712c20fSFrederick Mayle
FDEHeader(Label cie_pointer,uint64_t initial_location,uint64_t address_range,bool dwarf64)82*9712c20fSFrederick Mayle CFISection& CFISection::FDEHeader(Label cie_pointer,
83*9712c20fSFrederick Mayle uint64_t initial_location,
84*9712c20fSFrederick Mayle uint64_t address_range,
85*9712c20fSFrederick Mayle bool dwarf64) {
86*9712c20fSFrederick Mayle assert(!entry_length_);
87*9712c20fSFrederick Mayle entry_length_ = new PendingLength();
88*9712c20fSFrederick Mayle in_fde_ = true;
89*9712c20fSFrederick Mayle fde_start_address_ = initial_location;
90*9712c20fSFrederick Mayle
91*9712c20fSFrederick Mayle if (dwarf64) {
92*9712c20fSFrederick Mayle D32(0xffffffff);
93*9712c20fSFrederick Mayle D64(entry_length_->length);
94*9712c20fSFrederick Mayle entry_length_->start = Here();
95*9712c20fSFrederick Mayle if (eh_frame_)
96*9712c20fSFrederick Mayle D64(Here() - cie_pointer);
97*9712c20fSFrederick Mayle else
98*9712c20fSFrederick Mayle D64(cie_pointer);
99*9712c20fSFrederick Mayle } else {
100*9712c20fSFrederick Mayle D32(entry_length_->length);
101*9712c20fSFrederick Mayle entry_length_->start = Here();
102*9712c20fSFrederick Mayle if (eh_frame_)
103*9712c20fSFrederick Mayle D32(Here() - cie_pointer);
104*9712c20fSFrederick Mayle else
105*9712c20fSFrederick Mayle D32(cie_pointer);
106*9712c20fSFrederick Mayle }
107*9712c20fSFrederick Mayle EncodedPointer(initial_location);
108*9712c20fSFrederick Mayle // The FDE length in an .eh_frame section uses the same encoding as the
109*9712c20fSFrederick Mayle // initial location, but ignores the base address (selected by the upper
110*9712c20fSFrederick Mayle // nybble of the encoding), as it's a length, not an address that can be
111*9712c20fSFrederick Mayle // made relative.
112*9712c20fSFrederick Mayle EncodedPointer(address_range,
113*9712c20fSFrederick Mayle DwarfPointerEncoding(pointer_encoding_ & 0x0f));
114*9712c20fSFrederick Mayle return *this;
115*9712c20fSFrederick Mayle }
116*9712c20fSFrederick Mayle
FinishEntry()117*9712c20fSFrederick Mayle CFISection& CFISection::FinishEntry() {
118*9712c20fSFrederick Mayle assert(entry_length_);
119*9712c20fSFrederick Mayle Align(address_size_, DW_CFA_nop);
120*9712c20fSFrederick Mayle entry_length_->length = Here() - entry_length_->start;
121*9712c20fSFrederick Mayle delete entry_length_;
122*9712c20fSFrederick Mayle entry_length_ = NULL;
123*9712c20fSFrederick Mayle in_fde_ = false;
124*9712c20fSFrederick Mayle return *this;
125*9712c20fSFrederick Mayle }
126*9712c20fSFrederick Mayle
EncodedPointer(uint64_t address,DwarfPointerEncoding encoding,const EncodedPointerBases & bases)127*9712c20fSFrederick Mayle CFISection& CFISection::EncodedPointer(uint64_t address,
128*9712c20fSFrederick Mayle DwarfPointerEncoding encoding,
129*9712c20fSFrederick Mayle const EncodedPointerBases& bases) {
130*9712c20fSFrederick Mayle // Omitted data is extremely easy to emit.
131*9712c20fSFrederick Mayle if (encoding == DW_EH_PE_omit)
132*9712c20fSFrederick Mayle return *this;
133*9712c20fSFrederick Mayle
134*9712c20fSFrederick Mayle // If (encoding & DW_EH_PE_indirect) != 0, then we assume
135*9712c20fSFrederick Mayle // that ADDRESS is the address at which the pointer is stored --- in
136*9712c20fSFrederick Mayle // other words, that bit has no effect on how we write the pointer.
137*9712c20fSFrederick Mayle encoding = DwarfPointerEncoding(encoding & ~DW_EH_PE_indirect);
138*9712c20fSFrederick Mayle
139*9712c20fSFrederick Mayle // Find the base address to which this pointer is relative. The upper
140*9712c20fSFrederick Mayle // nybble of the encoding specifies this.
141*9712c20fSFrederick Mayle uint64_t base;
142*9712c20fSFrederick Mayle switch (encoding & 0xf0) {
143*9712c20fSFrederick Mayle case DW_EH_PE_absptr: base = 0; break;
144*9712c20fSFrederick Mayle case DW_EH_PE_pcrel: base = bases.cfi + Size(); break;
145*9712c20fSFrederick Mayle case DW_EH_PE_textrel: base = bases.text; break;
146*9712c20fSFrederick Mayle case DW_EH_PE_datarel: base = bases.data; break;
147*9712c20fSFrederick Mayle case DW_EH_PE_funcrel: base = fde_start_address_; break;
148*9712c20fSFrederick Mayle case DW_EH_PE_aligned: base = 0; break;
149*9712c20fSFrederick Mayle default: abort();
150*9712c20fSFrederick Mayle };
151*9712c20fSFrederick Mayle
152*9712c20fSFrederick Mayle // Make ADDRESS relative. Yes, this is appropriate even for "absptr"
153*9712c20fSFrederick Mayle // values; see gcc/unwind-pe.h.
154*9712c20fSFrederick Mayle address -= base;
155*9712c20fSFrederick Mayle
156*9712c20fSFrederick Mayle // Align the pointer, if required.
157*9712c20fSFrederick Mayle if ((encoding & 0xf0) == DW_EH_PE_aligned)
158*9712c20fSFrederick Mayle Align(AddressSize());
159*9712c20fSFrederick Mayle
160*9712c20fSFrederick Mayle // Append ADDRESS to this section in the appropriate form. For the
161*9712c20fSFrederick Mayle // fixed-width forms, we don't need to differentiate between signed and
162*9712c20fSFrederick Mayle // unsigned encodings, because ADDRESS has already been extended to 64
163*9712c20fSFrederick Mayle // bits before it was passed to us.
164*9712c20fSFrederick Mayle switch (encoding & 0x0f) {
165*9712c20fSFrederick Mayle case DW_EH_PE_absptr:
166*9712c20fSFrederick Mayle Address(address);
167*9712c20fSFrederick Mayle break;
168*9712c20fSFrederick Mayle
169*9712c20fSFrederick Mayle case DW_EH_PE_uleb128:
170*9712c20fSFrederick Mayle ULEB128(address);
171*9712c20fSFrederick Mayle break;
172*9712c20fSFrederick Mayle
173*9712c20fSFrederick Mayle case DW_EH_PE_sleb128:
174*9712c20fSFrederick Mayle LEB128(address);
175*9712c20fSFrederick Mayle break;
176*9712c20fSFrederick Mayle
177*9712c20fSFrederick Mayle case DW_EH_PE_udata2:
178*9712c20fSFrederick Mayle case DW_EH_PE_sdata2:
179*9712c20fSFrederick Mayle D16(address);
180*9712c20fSFrederick Mayle break;
181*9712c20fSFrederick Mayle
182*9712c20fSFrederick Mayle case DW_EH_PE_udata4:
183*9712c20fSFrederick Mayle case DW_EH_PE_sdata4:
184*9712c20fSFrederick Mayle D32(address);
185*9712c20fSFrederick Mayle break;
186*9712c20fSFrederick Mayle
187*9712c20fSFrederick Mayle case DW_EH_PE_udata8:
188*9712c20fSFrederick Mayle case DW_EH_PE_sdata8:
189*9712c20fSFrederick Mayle D64(address);
190*9712c20fSFrederick Mayle break;
191*9712c20fSFrederick Mayle
192*9712c20fSFrederick Mayle default:
193*9712c20fSFrederick Mayle abort();
194*9712c20fSFrederick Mayle }
195*9712c20fSFrederick Mayle
196*9712c20fSFrederick Mayle return *this;
197*9712c20fSFrederick Mayle };
198*9712c20fSFrederick Mayle
199*9712c20fSFrederick Mayle const uint32_t CFISection::kDwarf64InitialLengthMarker;
200*9712c20fSFrederick Mayle const uint32_t CFISection::kDwarf32CIEIdentifier;
201*9712c20fSFrederick Mayle const uint64_t CFISection::kDwarf64CIEIdentifier;
202*9712c20fSFrederick Mayle const uint32_t CFISection::kEHFrame32CIEIdentifier;
203*9712c20fSFrederick Mayle const uint64_t CFISection::kEHFrame64CIEIdentifier;
204*9712c20fSFrederick Mayle
205*9712c20fSFrederick Mayle } // namespace google_breakpad
206