blob: f81f96ce5a36d36a58c5e8c5256c5d99561a8a40 [file] [log] [blame]
Louis Dionne7f068e52021-11-17 16:25:01 -05001//===----------------------------------------------------------------------===//
Saleem Abdulrasool17552662015-04-24 19:39:17 +00002//
Chandler Carruth61860a52019-01-19 10:56:40 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Saleem Abdulrasool17552662015-04-24 19:39:17 +00006//
7//
Ed Maste4c43c3d2016-07-19 17:15:50 +00008// Processor specific interpretation of DWARF unwind info.
Saleem Abdulrasool17552662015-04-24 19:39:17 +00009//
10//===----------------------------------------------------------------------===//
11
12#ifndef __DWARF_INSTRUCTIONS_HPP__
13#define __DWARF_INSTRUCTIONS_HPP__
14
15#include <stdint.h>
16#include <stdio.h>
17#include <stdlib.h>
18
Saleem Abdulrasool17552662015-04-24 19:39:17 +000019#include "DwarfParser.hpp"
Ryan Prichard60a480e2022-09-07 17:27:57 -040020#include "Registers.hpp"
Saleem Abdulrasool17552662015-04-24 19:39:17 +000021#include "config.h"
Ryan Prichard60a480e2022-09-07 17:27:57 -040022#include "dwarf2.h"
23#include "libunwind_ext.h"
Saleem Abdulrasool17552662015-04-24 19:39:17 +000024
25
26namespace libunwind {
27
28
Gabriel Ravier42aa6de2022-08-20 18:09:03 -070029/// DwarfInstructions maps abstract DWARF unwind instructions to a particular
Saleem Abdulrasool17552662015-04-24 19:39:17 +000030/// architecture
31template <typename A, typename R>
32class DwarfInstructions {
33public:
34 typedef typename A::pint_t pint_t;
35 typedef typename A::sint_t sint_t;
36
37 static int stepWithDwarf(A &addressSpace, pint_t pc, pint_t fdeStart,
Sterling Augustineb6a66392019-10-31 12:45:20 -070038 R &registers, bool &isSignalFrame);
Saleem Abdulrasool17552662015-04-24 19:39:17 +000039
40private:
41
42 enum {
43 DW_X86_64_RET_ADDR = 16
44 };
45
46 enum {
47 DW_X86_RET_ADDR = 8
48 };
49
50 typedef typename CFI_Parser<A>::RegisterLocation RegisterLocation;
51 typedef typename CFI_Parser<A>::PrologInfo PrologInfo;
52 typedef typename CFI_Parser<A>::FDE_Info FDE_Info;
53 typedef typename CFI_Parser<A>::CIE_Info CIE_Info;
54
55 static pint_t evaluateExpression(pint_t expression, A &addressSpace,
56 const R &registers,
57 pint_t initialStackValue);
58 static pint_t getSavedRegister(A &addressSpace, const R &registers,
59 pint_t cfa, const RegisterLocation &savedReg);
60 static double getSavedFloatRegister(A &addressSpace, const R &registers,
61 pint_t cfa, const RegisterLocation &savedReg);
62 static v128 getSavedVectorRegister(A &addressSpace, const R &registers,
63 pint_t cfa, const RegisterLocation &savedReg);
64
65 static pint_t getCFA(A &addressSpace, const PrologInfo &prolog,
66 const R &registers) {
67 if (prolog.cfaRegister != 0)
68 return (pint_t)((sint_t)registers.getRegister((int)prolog.cfaRegister) +
69 prolog.cfaRegisterOffset);
70 if (prolog.cfaExpression != 0)
71 return evaluateExpression((pint_t)prolog.cfaExpression, addressSpace,
72 registers, 0);
73 assert(0 && "getCFA(): unknown location");
74 __builtin_unreachable();
75 }
Daniel Kiss837a94e2022-05-13 09:12:07 +020076#if defined(_LIBUNWIND_TARGET_AARCH64)
77 static bool getRA_SIGN_STATE(A &addressSpace, R registers, pint_t cfa,
78 PrologInfo &prolog);
79#endif
Saleem Abdulrasool17552662015-04-24 19:39:17 +000080};
81
Koakumaf2ef96e2022-02-05 13:08:26 -080082template <typename R>
83auto getSparcWCookie(const R &r, int) -> decltype(r.getWCookie()) {
84 return r.getWCookie();
85}
86template <typename R> uint64_t getSparcWCookie(const R &, long) {
87 return 0;
88}
Saleem Abdulrasool17552662015-04-24 19:39:17 +000089
90template <typename A, typename R>
91typename A::pint_t DwarfInstructions<A, R>::getSavedRegister(
92 A &addressSpace, const R &registers, pint_t cfa,
93 const RegisterLocation &savedReg) {
94 switch (savedReg.location) {
95 case CFI_Parser<A>::kRegisterInCFA:
Martin Storsjo688e5942019-01-22 20:50:42 +000096 return (pint_t)addressSpace.getRegister(cfa + (pint_t)savedReg.value);
Saleem Abdulrasool17552662015-04-24 19:39:17 +000097
Koakumaf2ef96e2022-02-05 13:08:26 -080098 case CFI_Parser<A>::kRegisterInCFADecrypt: // sparc64 specific
Martin Storsjö2b08f992022-02-09 19:36:58 +020099 return (pint_t)(addressSpace.getP(cfa + (pint_t)savedReg.value) ^
100 getSparcWCookie(registers, 0));
Koakumaf2ef96e2022-02-05 13:08:26 -0800101
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000102 case CFI_Parser<A>::kRegisterAtExpression:
Martin Storsjo688e5942019-01-22 20:50:42 +0000103 return (pint_t)addressSpace.getRegister(evaluateExpression(
104 (pint_t)savedReg.value, addressSpace, registers, cfa));
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000105
106 case CFI_Parser<A>::kRegisterIsExpression:
107 return evaluateExpression((pint_t)savedReg.value, addressSpace,
108 registers, cfa);
109
110 case CFI_Parser<A>::kRegisterInRegister:
111 return registers.getRegister((int)savedReg.value);
Daniel Kiss163101b2020-09-16 23:03:19 +0200112 case CFI_Parser<A>::kRegisterUndefined:
113 return 0;
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000114 case CFI_Parser<A>::kRegisterUnused:
115 case CFI_Parser<A>::kRegisterOffsetFromCFA:
116 // FIX ME
117 break;
118 }
119 _LIBUNWIND_ABORT("unsupported restore location for register");
120}
121
122template <typename A, typename R>
123double DwarfInstructions<A, R>::getSavedFloatRegister(
124 A &addressSpace, const R &registers, pint_t cfa,
125 const RegisterLocation &savedReg) {
126 switch (savedReg.location) {
127 case CFI_Parser<A>::kRegisterInCFA:
128 return addressSpace.getDouble(cfa + (pint_t)savedReg.value);
129
130 case CFI_Parser<A>::kRegisterAtExpression:
131 return addressSpace.getDouble(
132 evaluateExpression((pint_t)savedReg.value, addressSpace,
133 registers, cfa));
Daniel Kissa002c722021-09-27 12:01:35 +0200134 case CFI_Parser<A>::kRegisterUndefined:
135 return 0.0;
Daniel Kiss10eb9982021-10-01 16:49:37 +0200136 case CFI_Parser<A>::kRegisterInRegister:
137#ifndef _LIBUNWIND_TARGET_ARM
138 return registers.getFloatRegister((int)savedReg.value);
139#endif
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000140 case CFI_Parser<A>::kRegisterIsExpression:
141 case CFI_Parser<A>::kRegisterUnused:
142 case CFI_Parser<A>::kRegisterOffsetFromCFA:
Koakumaf2ef96e2022-02-05 13:08:26 -0800143 case CFI_Parser<A>::kRegisterInCFADecrypt:
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000144 // FIX ME
145 break;
146 }
147 _LIBUNWIND_ABORT("unsupported restore location for float register");
148}
149
150template <typename A, typename R>
151v128 DwarfInstructions<A, R>::getSavedVectorRegister(
152 A &addressSpace, const R &registers, pint_t cfa,
153 const RegisterLocation &savedReg) {
154 switch (savedReg.location) {
155 case CFI_Parser<A>::kRegisterInCFA:
156 return addressSpace.getVector(cfa + (pint_t)savedReg.value);
157
158 case CFI_Parser<A>::kRegisterAtExpression:
159 return addressSpace.getVector(
160 evaluateExpression((pint_t)savedReg.value, addressSpace,
161 registers, cfa));
162
163 case CFI_Parser<A>::kRegisterIsExpression:
164 case CFI_Parser<A>::kRegisterUnused:
Daniel Kiss163101b2020-09-16 23:03:19 +0200165 case CFI_Parser<A>::kRegisterUndefined:
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000166 case CFI_Parser<A>::kRegisterOffsetFromCFA:
167 case CFI_Parser<A>::kRegisterInRegister:
Koakumaf2ef96e2022-02-05 13:08:26 -0800168 case CFI_Parser<A>::kRegisterInCFADecrypt:
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000169 // FIX ME
170 break;
171 }
172 _LIBUNWIND_ABORT("unsupported restore location for vector register");
173}
Daniel Kiss837a94e2022-05-13 09:12:07 +0200174#if defined(_LIBUNWIND_TARGET_AARCH64)
175template <typename A, typename R>
176bool DwarfInstructions<A, R>::getRA_SIGN_STATE(A &addressSpace, R registers,
177 pint_t cfa, PrologInfo &prolog) {
178 pint_t raSignState;
179 auto regloc = prolog.savedRegisters[UNW_AARCH64_RA_SIGN_STATE];
180 if (regloc.location == CFI_Parser<A>::kRegisterUnused)
Daniel Kiss1e26dd02022-05-19 09:38:30 +0200181 raSignState = static_cast<pint_t>(regloc.value);
Daniel Kiss837a94e2022-05-13 09:12:07 +0200182 else
183 raSignState = getSavedRegister(addressSpace, registers, cfa, regloc);
184
185 // Only bit[0] is meaningful.
186 return raSignState & 0x01;
187}
188#endif
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000189
190template <typename A, typename R>
191int DwarfInstructions<A, R>::stepWithDwarf(A &addressSpace, pint_t pc,
Sterling Augustineb6a66392019-10-31 12:45:20 -0700192 pint_t fdeStart, R &registers,
193 bool &isSignalFrame) {
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000194 FDE_Info fdeInfo;
195 CIE_Info cieInfo;
196 if (CFI_Parser<A>::decodeFDE(addressSpace, fdeStart, &fdeInfo,
197 &cieInfo) == NULL) {
198 PrologInfo prolog;
199 if (CFI_Parser<A>::parseFDEInstructions(addressSpace, fdeInfo, cieInfo, pc,
Daniel Cederman9f2f07a2019-01-14 10:15:20 +0000200 R::getArch(), &prolog)) {
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000201 // get pointer to cfa (architecture specific)
202 pint_t cfa = getCFA(addressSpace, prolog, registers);
203
Ed Maste4c43c3d2016-07-19 17:15:50 +0000204 // restore registers that DWARF says were saved
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000205 R newRegisters = registers;
Marco Vanottiea7e5f62021-07-22 17:58:23 -0700206
207 // Typically, the CFA is the stack pointer at the call site in
208 // the previous frame. However, there are scenarios in which this is not
209 // true. For example, if we switched to a new stack. In that case, the
210 // value of the previous SP might be indicated by a CFI directive.
211 //
212 // We set the SP here to the CFA, allowing for it to be overridden
213 // by a CFI directive later on.
214 newRegisters.setSP(cfa);
215
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000216 pint_t returnAddress = 0;
Florian Mayer2c497062022-06-03 11:45:04 -0700217 constexpr int lastReg = R::lastDwarfRegNum();
218 static_assert(static_cast<int>(CFI_Parser<A>::kMaxRegisterNumber) >=
219 lastReg,
220 "register range too large");
Ed Mastec14579b2015-08-13 13:45:45 +0000221 assert(lastReg >= (int)cieInfo.returnAddressRegister &&
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000222 "register range does not contain return address register");
223 for (int i = 0; i <= lastReg; ++i) {
Logan Chien0112edf2015-05-30 14:00:39 +0000224 if (prolog.savedRegisters[i].location !=
225 CFI_Parser<A>::kRegisterUnused) {
226 if (registers.validFloatRegister(i))
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000227 newRegisters.setFloatRegister(
228 i, getSavedFloatRegister(addressSpace, registers, cfa,
229 prolog.savedRegisters[i]));
230 else if (registers.validVectorRegister(i))
231 newRegisters.setVectorRegister(
232 i, getSavedVectorRegister(addressSpace, registers, cfa,
233 prolog.savedRegisters[i]));
234 else if (i == (int)cieInfo.returnAddressRegister)
235 returnAddress = getSavedRegister(addressSpace, registers, cfa,
236 prolog.savedRegisters[i]);
237 else if (registers.validRegister(i))
238 newRegisters.setRegister(
239 i, getSavedRegister(addressSpace, registers, cfa,
240 prolog.savedRegisters[i]));
241 else
242 return UNW_EBADREG;
Daniel Kiss163101b2020-09-16 23:03:19 +0200243 } else if (i == (int)cieInfo.returnAddressRegister) {
244 // Leaf function keeps the return address in register and there is no
Gabriel Ravier42aa6de2022-08-20 18:09:03 -0700245 // explicit instructions how to restore it.
Daniel Kiss163101b2020-09-16 23:03:19 +0200246 returnAddress = registers.getRegister(cieInfo.returnAddressRegister);
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000247 }
248 }
249
Sterling Augustineb6a66392019-10-31 12:45:20 -0700250 isSignalFrame = cieInfo.isSignalFrame;
251
Luke Cheeseman4d8e4312018-12-14 11:30:12 +0000252#if defined(_LIBUNWIND_TARGET_AARCH64)
253 // If the target is aarch64 then the return address may have been signed
254 // using the v8.3 pointer authentication extensions. The original
255 // return address needs to be authenticated before the return address is
256 // restored. autia1716 is used instead of autia as autia1716 assembles
257 // to a NOP on pre-v8.3a architectures.
Daniel Cederman9f2f07a2019-01-14 10:15:20 +0000258 if ((R::getArch() == REGISTERS_ARM64) &&
Daniel Kiss837a94e2022-05-13 09:12:07 +0200259 getRA_SIGN_STATE(addressSpace, registers, cfa, prolog) &&
Peter Collingbournec6243922021-02-11 16:16:51 -0800260 returnAddress != 0) {
Luke Cheeseman4d8e4312018-12-14 11:30:12 +0000261#if !defined(_LIBUNWIND_IS_NATIVE_ONLY)
262 return UNW_ECROSSRASIGNING;
263#else
264 register unsigned long long x17 __asm("x17") = returnAddress;
265 register unsigned long long x16 __asm("x16") = cfa;
266
Luke Cheesemancba83c32018-12-17 11:43:24 +0000267 // These are the autia1716/autib1716 instructions. The hint instructions
268 // are used here as gcc does not assemble autia1716/autib1716 for pre
269 // armv8.3a targets.
270 if (cieInfo.addressesSignedWithBKey)
271 asm("hint 0xe" : "+r"(x17) : "r"(x16)); // autib1716
272 else
273 asm("hint 0xc" : "+r"(x17) : "r"(x16)); // autia1716
Luke Cheeseman4d8e4312018-12-14 11:30:12 +0000274 returnAddress = x17;
275#endif
276 }
277#endif
278
Ties Stuijc8c0ec92021-12-08 09:44:45 +0000279#if defined(_LIBUNWIND_IS_NATIVE_ONLY) && defined(_LIBUNWIND_TARGET_ARM) && \
280 defined(__ARM_FEATURE_PAUTH)
281 if ((R::getArch() == REGISTERS_ARM) &&
282 prolog.savedRegisters[UNW_ARM_RA_AUTH_CODE].value) {
283 pint_t pac =
284 getSavedRegister(addressSpace, registers, cfa,
285 prolog.savedRegisters[UNW_ARM_RA_AUTH_CODE]);
286 __asm__ __volatile__("autg %0, %1, %2"
287 :
288 : "r"(pac), "r"(returnAddress), "r"(cfa)
289 :);
290 }
291#endif
292
Daniel Cederman9f2f07a2019-01-14 10:15:20 +0000293#if defined(_LIBUNWIND_TARGET_SPARC)
294 if (R::getArch() == REGISTERS_SPARC) {
295 // Skip call site instruction and delay slot
296 returnAddress += 8;
297 // Skip unimp instruction if function returns a struct
298 if ((addressSpace.get32(returnAddress) & 0xC1C00000) == 0)
299 returnAddress += 4;
300 }
301#endif
302
Koakumaf2ef96e2022-02-05 13:08:26 -0800303#if defined(_LIBUNWIND_TARGET_SPARC64)
304 // Skip call site instruction and delay slot.
305 if (R::getArch() == REGISTERS_SPARC64)
306 returnAddress += 8;
307#endif
308
Martin Storsjo8a6fc692019-05-16 06:49:13 +0000309#if defined(_LIBUNWIND_TARGET_PPC64)
310#define PPC64_ELFV1_R2_LOAD_INST_ENCODING 0xe8410028u // ld r2,40(r1)
311#define PPC64_ELFV1_R2_OFFSET 40
312#define PPC64_ELFV2_R2_LOAD_INST_ENCODING 0xe8410018u // ld r2,24(r1)
313#define PPC64_ELFV2_R2_OFFSET 24
314 // If the instruction at return address is a TOC (r2) restore,
315 // then r2 was saved and needs to be restored.
316 // ELFv2 ABI specifies that the TOC Pointer must be saved at SP + 24,
317 // while in ELFv1 ABI it is saved at SP + 40.
318 if (R::getArch() == REGISTERS_PPC64 && returnAddress != 0) {
319 pint_t sp = newRegisters.getRegister(UNW_REG_SP);
320 pint_t r2 = 0;
321 switch (addressSpace.get32(returnAddress)) {
322 case PPC64_ELFV1_R2_LOAD_INST_ENCODING:
323 r2 = addressSpace.get64(sp + PPC64_ELFV1_R2_OFFSET);
324 break;
325 case PPC64_ELFV2_R2_LOAD_INST_ENCODING:
326 r2 = addressSpace.get64(sp + PPC64_ELFV2_R2_OFFSET);
327 break;
328 }
329 if (r2)
330 newRegisters.setRegister(UNW_PPC64_R2, r2);
331 }
332#endif
333
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000334 // Return address is address after call site instruction, so setting IP to
Gabriel Ravier42aa6de2022-08-20 18:09:03 -0700335 // that does simulates a return.
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000336 newRegisters.setIP(returnAddress);
337
338 // Simulate the step by replacing the register set with the new ones.
339 registers = newRegisters;
340
341 return UNW_STEP_SUCCESS;
342 }
343 }
344 return UNW_EBADFRAME;
345}
346
347template <typename A, typename R>
348typename A::pint_t
349DwarfInstructions<A, R>::evaluateExpression(pint_t expression, A &addressSpace,
350 const R &registers,
351 pint_t initialStackValue) {
352 const bool log = false;
353 pint_t p = expression;
354 pint_t expressionEnd = expression + 20; // temp, until len read
355 pint_t length = (pint_t)addressSpace.getULEB128(p, expressionEnd);
356 expressionEnd = p + length;
357 if (log)
358 fprintf(stderr, "evaluateExpression(): length=%" PRIu64 "\n",
359 (uint64_t)length);
360 pint_t stack[100];
361 pint_t *sp = stack;
362 *(++sp) = initialStackValue;
363
364 while (p < expressionEnd) {
365 if (log) {
366 for (pint_t *t = sp; t > stack; --t) {
367 fprintf(stderr, "sp[] = 0x%" PRIx64 "\n", (uint64_t)(*t));
368 }
369 }
370 uint8_t opcode = addressSpace.get8(p++);
371 sint_t svalue, svalue2;
372 pint_t value;
373 uint32_t reg;
374 switch (opcode) {
375 case DW_OP_addr:
376 // push immediate address sized value
377 value = addressSpace.getP(p);
378 p += sizeof(pint_t);
379 *(++sp) = value;
380 if (log)
381 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
382 break;
383
384 case DW_OP_deref:
385 // pop stack, dereference, push result
386 value = *sp--;
387 *(++sp) = addressSpace.getP(value);
388 if (log)
389 fprintf(stderr, "dereference 0x%" PRIx64 "\n", (uint64_t)value);
390 break;
391
392 case DW_OP_const1u:
393 // push immediate 1 byte value
394 value = addressSpace.get8(p);
395 p += 1;
396 *(++sp) = value;
397 if (log)
398 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
399 break;
400
401 case DW_OP_const1s:
402 // push immediate 1 byte signed value
403 svalue = (int8_t) addressSpace.get8(p);
404 p += 1;
405 *(++sp) = (pint_t)svalue;
406 if (log)
407 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
408 break;
409
410 case DW_OP_const2u:
411 // push immediate 2 byte value
412 value = addressSpace.get16(p);
413 p += 2;
414 *(++sp) = value;
415 if (log)
416 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
417 break;
418
419 case DW_OP_const2s:
420 // push immediate 2 byte signed value
421 svalue = (int16_t) addressSpace.get16(p);
422 p += 2;
423 *(++sp) = (pint_t)svalue;
424 if (log)
425 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
426 break;
427
428 case DW_OP_const4u:
429 // push immediate 4 byte value
430 value = addressSpace.get32(p);
431 p += 4;
432 *(++sp) = value;
433 if (log)
434 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
435 break;
436
437 case DW_OP_const4s:
438 // push immediate 4 byte signed value
439 svalue = (int32_t)addressSpace.get32(p);
440 p += 4;
441 *(++sp) = (pint_t)svalue;
442 if (log)
443 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
444 break;
445
446 case DW_OP_const8u:
447 // push immediate 8 byte value
448 value = (pint_t)addressSpace.get64(p);
449 p += 8;
450 *(++sp) = value;
451 if (log)
452 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
453 break;
454
455 case DW_OP_const8s:
456 // push immediate 8 byte signed value
457 value = (pint_t)addressSpace.get64(p);
458 p += 8;
459 *(++sp) = value;
460 if (log)
461 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
462 break;
463
464 case DW_OP_constu:
465 // push immediate ULEB128 value
466 value = (pint_t)addressSpace.getULEB128(p, expressionEnd);
467 *(++sp) = value;
468 if (log)
469 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
470 break;
471
472 case DW_OP_consts:
473 // push immediate SLEB128 value
474 svalue = (sint_t)addressSpace.getSLEB128(p, expressionEnd);
475 *(++sp) = (pint_t)svalue;
476 if (log)
477 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
478 break;
479
480 case DW_OP_dup:
481 // push top of stack
482 value = *sp;
483 *(++sp) = value;
484 if (log)
485 fprintf(stderr, "duplicate top of stack\n");
486 break;
487
488 case DW_OP_drop:
489 // pop
490 --sp;
491 if (log)
492 fprintf(stderr, "pop top of stack\n");
493 break;
494
495 case DW_OP_over:
496 // dup second
497 value = sp[-1];
498 *(++sp) = value;
499 if (log)
500 fprintf(stderr, "duplicate second in stack\n");
501 break;
502
503 case DW_OP_pick:
504 // pick from
505 reg = addressSpace.get8(p);
506 p += 1;
Steven Wue12b2482019-12-18 12:22:21 -0800507 value = sp[-(int)reg];
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000508 *(++sp) = value;
509 if (log)
510 fprintf(stderr, "duplicate %d in stack\n", reg);
511 break;
512
513 case DW_OP_swap:
514 // swap top two
515 value = sp[0];
516 sp[0] = sp[-1];
517 sp[-1] = value;
518 if (log)
519 fprintf(stderr, "swap top of stack\n");
520 break;
521
522 case DW_OP_rot:
523 // rotate top three
524 value = sp[0];
525 sp[0] = sp[-1];
526 sp[-1] = sp[-2];
527 sp[-2] = value;
528 if (log)
529 fprintf(stderr, "rotate top three of stack\n");
530 break;
531
532 case DW_OP_xderef:
533 // pop stack, dereference, push result
534 value = *sp--;
535 *sp = *((pint_t*)value);
536 if (log)
537 fprintf(stderr, "x-dereference 0x%" PRIx64 "\n", (uint64_t)value);
538 break;
539
540 case DW_OP_abs:
541 svalue = (sint_t)*sp;
542 if (svalue < 0)
543 *sp = (pint_t)(-svalue);
544 if (log)
545 fprintf(stderr, "abs\n");
546 break;
547
548 case DW_OP_and:
549 value = *sp--;
550 *sp &= value;
551 if (log)
552 fprintf(stderr, "and\n");
553 break;
554
555 case DW_OP_div:
556 svalue = (sint_t)(*sp--);
557 svalue2 = (sint_t)*sp;
558 *sp = (pint_t)(svalue2 / svalue);
559 if (log)
560 fprintf(stderr, "div\n");
561 break;
562
563 case DW_OP_minus:
564 value = *sp--;
565 *sp = *sp - value;
566 if (log)
567 fprintf(stderr, "minus\n");
568 break;
569
570 case DW_OP_mod:
571 svalue = (sint_t)(*sp--);
572 svalue2 = (sint_t)*sp;
573 *sp = (pint_t)(svalue2 % svalue);
574 if (log)
575 fprintf(stderr, "module\n");
576 break;
577
578 case DW_OP_mul:
579 svalue = (sint_t)(*sp--);
580 svalue2 = (sint_t)*sp;
581 *sp = (pint_t)(svalue2 * svalue);
582 if (log)
583 fprintf(stderr, "mul\n");
584 break;
585
586 case DW_OP_neg:
587 *sp = 0 - *sp;
588 if (log)
589 fprintf(stderr, "neg\n");
590 break;
591
592 case DW_OP_not:
593 svalue = (sint_t)(*sp);
594 *sp = (pint_t)(~svalue);
595 if (log)
596 fprintf(stderr, "not\n");
597 break;
598
599 case DW_OP_or:
600 value = *sp--;
601 *sp |= value;
602 if (log)
603 fprintf(stderr, "or\n");
604 break;
605
606 case DW_OP_plus:
607 value = *sp--;
608 *sp += value;
609 if (log)
610 fprintf(stderr, "plus\n");
611 break;
612
613 case DW_OP_plus_uconst:
614 // pop stack, add uelb128 constant, push result
Saleem Abdulrasool92acbf12017-03-08 16:03:27 +0000615 *sp += static_cast<pint_t>(addressSpace.getULEB128(p, expressionEnd));
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000616 if (log)
617 fprintf(stderr, "add constant\n");
618 break;
619
620 case DW_OP_shl:
621 value = *sp--;
622 *sp = *sp << value;
623 if (log)
624 fprintf(stderr, "shift left\n");
625 break;
626
627 case DW_OP_shr:
628 value = *sp--;
629 *sp = *sp >> value;
630 if (log)
631 fprintf(stderr, "shift left\n");
632 break;
633
634 case DW_OP_shra:
635 value = *sp--;
636 svalue = (sint_t)*sp;
637 *sp = (pint_t)(svalue >> value);
638 if (log)
Gabriel Ravier42aa6de2022-08-20 18:09:03 -0700639 fprintf(stderr, "shift left arithmetic\n");
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000640 break;
641
642 case DW_OP_xor:
643 value = *sp--;
644 *sp ^= value;
645 if (log)
646 fprintf(stderr, "xor\n");
647 break;
648
649 case DW_OP_skip:
650 svalue = (int16_t) addressSpace.get16(p);
651 p += 2;
652 p = (pint_t)((sint_t)p + svalue);
653 if (log)
654 fprintf(stderr, "skip %" PRIu64 "\n", (uint64_t)svalue);
655 break;
656
657 case DW_OP_bra:
658 svalue = (int16_t) addressSpace.get16(p);
659 p += 2;
660 if (*sp--)
661 p = (pint_t)((sint_t)p + svalue);
662 if (log)
663 fprintf(stderr, "bra %" PRIu64 "\n", (uint64_t)svalue);
664 break;
665
666 case DW_OP_eq:
667 value = *sp--;
668 *sp = (*sp == value);
669 if (log)
670 fprintf(stderr, "eq\n");
671 break;
672
673 case DW_OP_ge:
674 value = *sp--;
675 *sp = (*sp >= value);
676 if (log)
677 fprintf(stderr, "ge\n");
678 break;
679
680 case DW_OP_gt:
681 value = *sp--;
682 *sp = (*sp > value);
683 if (log)
684 fprintf(stderr, "gt\n");
685 break;
686
687 case DW_OP_le:
688 value = *sp--;
689 *sp = (*sp <= value);
690 if (log)
691 fprintf(stderr, "le\n");
692 break;
693
694 case DW_OP_lt:
695 value = *sp--;
696 *sp = (*sp < value);
697 if (log)
698 fprintf(stderr, "lt\n");
699 break;
700
701 case DW_OP_ne:
702 value = *sp--;
703 *sp = (*sp != value);
704 if (log)
705 fprintf(stderr, "ne\n");
706 break;
707
708 case DW_OP_lit0:
709 case DW_OP_lit1:
710 case DW_OP_lit2:
711 case DW_OP_lit3:
712 case DW_OP_lit4:
713 case DW_OP_lit5:
714 case DW_OP_lit6:
715 case DW_OP_lit7:
716 case DW_OP_lit8:
717 case DW_OP_lit9:
718 case DW_OP_lit10:
719 case DW_OP_lit11:
720 case DW_OP_lit12:
721 case DW_OP_lit13:
722 case DW_OP_lit14:
723 case DW_OP_lit15:
724 case DW_OP_lit16:
725 case DW_OP_lit17:
726 case DW_OP_lit18:
727 case DW_OP_lit19:
728 case DW_OP_lit20:
729 case DW_OP_lit21:
730 case DW_OP_lit22:
731 case DW_OP_lit23:
732 case DW_OP_lit24:
733 case DW_OP_lit25:
734 case DW_OP_lit26:
735 case DW_OP_lit27:
736 case DW_OP_lit28:
737 case DW_OP_lit29:
738 case DW_OP_lit30:
739 case DW_OP_lit31:
740 value = static_cast<pint_t>(opcode - DW_OP_lit0);
741 *(++sp) = value;
742 if (log)
743 fprintf(stderr, "push literal 0x%" PRIx64 "\n", (uint64_t)value);
744 break;
745
746 case DW_OP_reg0:
747 case DW_OP_reg1:
748 case DW_OP_reg2:
749 case DW_OP_reg3:
750 case DW_OP_reg4:
751 case DW_OP_reg5:
752 case DW_OP_reg6:
753 case DW_OP_reg7:
754 case DW_OP_reg8:
755 case DW_OP_reg9:
756 case DW_OP_reg10:
757 case DW_OP_reg11:
758 case DW_OP_reg12:
759 case DW_OP_reg13:
760 case DW_OP_reg14:
761 case DW_OP_reg15:
762 case DW_OP_reg16:
763 case DW_OP_reg17:
764 case DW_OP_reg18:
765 case DW_OP_reg19:
766 case DW_OP_reg20:
767 case DW_OP_reg21:
768 case DW_OP_reg22:
769 case DW_OP_reg23:
770 case DW_OP_reg24:
771 case DW_OP_reg25:
772 case DW_OP_reg26:
773 case DW_OP_reg27:
774 case DW_OP_reg28:
775 case DW_OP_reg29:
776 case DW_OP_reg30:
777 case DW_OP_reg31:
778 reg = static_cast<uint32_t>(opcode - DW_OP_reg0);
779 *(++sp) = registers.getRegister((int)reg);
780 if (log)
781 fprintf(stderr, "push reg %d\n", reg);
782 break;
783
784 case DW_OP_regx:
785 reg = static_cast<uint32_t>(addressSpace.getULEB128(p, expressionEnd));
786 *(++sp) = registers.getRegister((int)reg);
787 if (log)
788 fprintf(stderr, "push reg %d + 0x%" PRIx64 "\n", reg, (uint64_t)svalue);
789 break;
790
791 case DW_OP_breg0:
792 case DW_OP_breg1:
793 case DW_OP_breg2:
794 case DW_OP_breg3:
795 case DW_OP_breg4:
796 case DW_OP_breg5:
797 case DW_OP_breg6:
798 case DW_OP_breg7:
799 case DW_OP_breg8:
800 case DW_OP_breg9:
801 case DW_OP_breg10:
802 case DW_OP_breg11:
803 case DW_OP_breg12:
804 case DW_OP_breg13:
805 case DW_OP_breg14:
806 case DW_OP_breg15:
807 case DW_OP_breg16:
808 case DW_OP_breg17:
809 case DW_OP_breg18:
810 case DW_OP_breg19:
811 case DW_OP_breg20:
812 case DW_OP_breg21:
813 case DW_OP_breg22:
814 case DW_OP_breg23:
815 case DW_OP_breg24:
816 case DW_OP_breg25:
817 case DW_OP_breg26:
818 case DW_OP_breg27:
819 case DW_OP_breg28:
820 case DW_OP_breg29:
821 case DW_OP_breg30:
822 case DW_OP_breg31:
823 reg = static_cast<uint32_t>(opcode - DW_OP_breg0);
824 svalue = (sint_t)addressSpace.getSLEB128(p, expressionEnd);
825 svalue += static_cast<sint_t>(registers.getRegister((int)reg));
826 *(++sp) = (pint_t)(svalue);
827 if (log)
828 fprintf(stderr, "push reg %d + 0x%" PRIx64 "\n", reg, (uint64_t)svalue);
829 break;
830
831 case DW_OP_bregx:
832 reg = static_cast<uint32_t>(addressSpace.getULEB128(p, expressionEnd));
833 svalue = (sint_t)addressSpace.getSLEB128(p, expressionEnd);
834 svalue += static_cast<sint_t>(registers.getRegister((int)reg));
835 *(++sp) = (pint_t)(svalue);
836 if (log)
837 fprintf(stderr, "push reg %d + 0x%" PRIx64 "\n", reg, (uint64_t)svalue);
838 break;
839
840 case DW_OP_fbreg:
841 _LIBUNWIND_ABORT("DW_OP_fbreg not implemented");
842 break;
843
844 case DW_OP_piece:
845 _LIBUNWIND_ABORT("DW_OP_piece not implemented");
846 break;
847
848 case DW_OP_deref_size:
849 // pop stack, dereference, push result
850 value = *sp--;
851 switch (addressSpace.get8(p++)) {
852 case 1:
853 value = addressSpace.get8(value);
854 break;
855 case 2:
856 value = addressSpace.get16(value);
857 break;
858 case 4:
859 value = addressSpace.get32(value);
860 break;
861 case 8:
862 value = (pint_t)addressSpace.get64(value);
863 break;
864 default:
865 _LIBUNWIND_ABORT("DW_OP_deref_size with bad size");
866 }
867 *(++sp) = value;
868 if (log)
869 fprintf(stderr, "sized dereference 0x%" PRIx64 "\n", (uint64_t)value);
870 break;
871
872 case DW_OP_xderef_size:
873 case DW_OP_nop:
874 case DW_OP_push_object_addres:
875 case DW_OP_call2:
876 case DW_OP_call4:
877 case DW_OP_call_ref:
878 default:
Ed Maste4c43c3d2016-07-19 17:15:50 +0000879 _LIBUNWIND_ABORT("DWARF opcode not implemented");
Saleem Abdulrasool17552662015-04-24 19:39:17 +0000880 }
881
882 }
883 if (log)
884 fprintf(stderr, "expression evaluates to 0x%" PRIx64 "\n", (uint64_t)*sp);
885 return *sp;
886}
887
888
889
890} // namespace libunwind
891
892#endif // __DWARF_INSTRUCTIONS_HPP__