xref: /aosp_15_r20/external/google-breakpad/src/common/dwarf/cfi_assembler.cc (revision 9712c20fc9bbfbac4935993a2ca0b3958c5adad2)
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