[PowerPC] Collect some CallLowering arguments into a struct. [NFC]
[llvm-project.git] / libunwind / src / DwarfInstructions.hpp
blobee98f538d437e46a1f86d8df85857124b7432017
1 //===-------------------------- DwarfInstructions.hpp ---------------------===//
2 //
3 // 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
6 //
7 //
8 // Processor specific interpretation of DWARF unwind info.
9 //
10 //===----------------------------------------------------------------------===//
12 #ifndef __DWARF_INSTRUCTIONS_HPP__
13 #define __DWARF_INSTRUCTIONS_HPP__
15 #include <stdint.h>
16 #include <stdio.h>
17 #include <stdlib.h>
19 #include "dwarf2.h"
20 #include "Registers.hpp"
21 #include "DwarfParser.hpp"
22 #include "config.h"
25 namespace libunwind {
28 /// DwarfInstructions maps abtract DWARF unwind instructions to a particular
29 /// architecture
30 template <typename A, typename R>
31 class DwarfInstructions {
32 public:
33 typedef typename A::pint_t pint_t;
34 typedef typename A::sint_t sint_t;
36 static int stepWithDwarf(A &addressSpace, pint_t pc, pint_t fdeStart,
37 R &registers, bool &isSignalFrame);
39 private:
41 enum {
42 DW_X86_64_RET_ADDR = 16
45 enum {
46 DW_X86_RET_ADDR = 8
49 typedef typename CFI_Parser<A>::RegisterLocation RegisterLocation;
50 typedef typename CFI_Parser<A>::PrologInfo PrologInfo;
51 typedef typename CFI_Parser<A>::FDE_Info FDE_Info;
52 typedef typename CFI_Parser<A>::CIE_Info CIE_Info;
54 static pint_t evaluateExpression(pint_t expression, A &addressSpace,
55 const R &registers,
56 pint_t initialStackValue);
57 static pint_t getSavedRegister(A &addressSpace, const R &registers,
58 pint_t cfa, const RegisterLocation &savedReg);
59 static double getSavedFloatRegister(A &addressSpace, const R &registers,
60 pint_t cfa, const RegisterLocation &savedReg);
61 static v128 getSavedVectorRegister(A &addressSpace, const R &registers,
62 pint_t cfa, const RegisterLocation &savedReg);
64 static pint_t getCFA(A &addressSpace, const PrologInfo &prolog,
65 const R &registers) {
66 if (prolog.cfaRegister != 0)
67 return (pint_t)((sint_t)registers.getRegister((int)prolog.cfaRegister) +
68 prolog.cfaRegisterOffset);
69 if (prolog.cfaExpression != 0)
70 return evaluateExpression((pint_t)prolog.cfaExpression, addressSpace,
71 registers, 0);
72 assert(0 && "getCFA(): unknown location");
73 __builtin_unreachable();
78 template <typename A, typename R>
79 typename A::pint_t DwarfInstructions<A, R>::getSavedRegister(
80 A &addressSpace, const R &registers, pint_t cfa,
81 const RegisterLocation &savedReg) {
82 switch (savedReg.location) {
83 case CFI_Parser<A>::kRegisterInCFA:
84 return (pint_t)addressSpace.getRegister(cfa + (pint_t)savedReg.value);
86 case CFI_Parser<A>::kRegisterAtExpression:
87 return (pint_t)addressSpace.getRegister(evaluateExpression(
88 (pint_t)savedReg.value, addressSpace, registers, cfa));
90 case CFI_Parser<A>::kRegisterIsExpression:
91 return evaluateExpression((pint_t)savedReg.value, addressSpace,
92 registers, cfa);
94 case CFI_Parser<A>::kRegisterInRegister:
95 return registers.getRegister((int)savedReg.value);
97 case CFI_Parser<A>::kRegisterUnused:
98 case CFI_Parser<A>::kRegisterOffsetFromCFA:
99 // FIX ME
100 break;
102 _LIBUNWIND_ABORT("unsupported restore location for register");
105 template <typename A, typename R>
106 double DwarfInstructions<A, R>::getSavedFloatRegister(
107 A &addressSpace, const R &registers, pint_t cfa,
108 const RegisterLocation &savedReg) {
109 switch (savedReg.location) {
110 case CFI_Parser<A>::kRegisterInCFA:
111 return addressSpace.getDouble(cfa + (pint_t)savedReg.value);
113 case CFI_Parser<A>::kRegisterAtExpression:
114 return addressSpace.getDouble(
115 evaluateExpression((pint_t)savedReg.value, addressSpace,
116 registers, cfa));
118 case CFI_Parser<A>::kRegisterIsExpression:
119 case CFI_Parser<A>::kRegisterUnused:
120 case CFI_Parser<A>::kRegisterOffsetFromCFA:
121 case CFI_Parser<A>::kRegisterInRegister:
122 // FIX ME
123 break;
125 _LIBUNWIND_ABORT("unsupported restore location for float register");
128 template <typename A, typename R>
129 v128 DwarfInstructions<A, R>::getSavedVectorRegister(
130 A &addressSpace, const R &registers, pint_t cfa,
131 const RegisterLocation &savedReg) {
132 switch (savedReg.location) {
133 case CFI_Parser<A>::kRegisterInCFA:
134 return addressSpace.getVector(cfa + (pint_t)savedReg.value);
136 case CFI_Parser<A>::kRegisterAtExpression:
137 return addressSpace.getVector(
138 evaluateExpression((pint_t)savedReg.value, addressSpace,
139 registers, cfa));
141 case CFI_Parser<A>::kRegisterIsExpression:
142 case CFI_Parser<A>::kRegisterUnused:
143 case CFI_Parser<A>::kRegisterOffsetFromCFA:
144 case CFI_Parser<A>::kRegisterInRegister:
145 // FIX ME
146 break;
148 _LIBUNWIND_ABORT("unsupported restore location for vector register");
151 template <typename A, typename R>
152 int DwarfInstructions<A, R>::stepWithDwarf(A &addressSpace, pint_t pc,
153 pint_t fdeStart, R &registers,
154 bool &isSignalFrame) {
155 FDE_Info fdeInfo;
156 CIE_Info cieInfo;
157 if (CFI_Parser<A>::decodeFDE(addressSpace, fdeStart, &fdeInfo,
158 &cieInfo) == NULL) {
159 PrologInfo prolog;
160 if (CFI_Parser<A>::parseFDEInstructions(addressSpace, fdeInfo, cieInfo, pc,
161 R::getArch(), &prolog)) {
162 // get pointer to cfa (architecture specific)
163 pint_t cfa = getCFA(addressSpace, prolog, registers);
165 // restore registers that DWARF says were saved
166 R newRegisters = registers;
167 pint_t returnAddress = 0;
168 const int lastReg = R::lastDwarfRegNum();
169 assert(static_cast<int>(CFI_Parser<A>::kMaxRegisterNumber) >= lastReg &&
170 "register range too large");
171 assert(lastReg >= (int)cieInfo.returnAddressRegister &&
172 "register range does not contain return address register");
173 for (int i = 0; i <= lastReg; ++i) {
174 if (prolog.savedRegisters[i].location !=
175 CFI_Parser<A>::kRegisterUnused) {
176 if (registers.validFloatRegister(i))
177 newRegisters.setFloatRegister(
178 i, getSavedFloatRegister(addressSpace, registers, cfa,
179 prolog.savedRegisters[i]));
180 else if (registers.validVectorRegister(i))
181 newRegisters.setVectorRegister(
182 i, getSavedVectorRegister(addressSpace, registers, cfa,
183 prolog.savedRegisters[i]));
184 else if (i == (int)cieInfo.returnAddressRegister)
185 returnAddress = getSavedRegister(addressSpace, registers, cfa,
186 prolog.savedRegisters[i]);
187 else if (registers.validRegister(i))
188 newRegisters.setRegister(
189 i, getSavedRegister(addressSpace, registers, cfa,
190 prolog.savedRegisters[i]));
191 else
192 return UNW_EBADREG;
196 // By definition, the CFA is the stack pointer at the call site, so
197 // restoring SP means setting it to CFA.
198 newRegisters.setSP(cfa);
200 isSignalFrame = cieInfo.isSignalFrame;
202 #if defined(_LIBUNWIND_TARGET_AARCH64)
203 // If the target is aarch64 then the return address may have been signed
204 // using the v8.3 pointer authentication extensions. The original
205 // return address needs to be authenticated before the return address is
206 // restored. autia1716 is used instead of autia as autia1716 assembles
207 // to a NOP on pre-v8.3a architectures.
208 if ((R::getArch() == REGISTERS_ARM64) &&
209 prolog.savedRegisters[UNW_ARM64_RA_SIGN_STATE].value) {
210 #if !defined(_LIBUNWIND_IS_NATIVE_ONLY)
211 return UNW_ECROSSRASIGNING;
212 #else
213 register unsigned long long x17 __asm("x17") = returnAddress;
214 register unsigned long long x16 __asm("x16") = cfa;
216 // These are the autia1716/autib1716 instructions. The hint instructions
217 // are used here as gcc does not assemble autia1716/autib1716 for pre
218 // armv8.3a targets.
219 if (cieInfo.addressesSignedWithBKey)
220 asm("hint 0xe" : "+r"(x17) : "r"(x16)); // autib1716
221 else
222 asm("hint 0xc" : "+r"(x17) : "r"(x16)); // autia1716
223 returnAddress = x17;
224 #endif
226 #endif
228 #if defined(_LIBUNWIND_TARGET_SPARC)
229 if (R::getArch() == REGISTERS_SPARC) {
230 // Skip call site instruction and delay slot
231 returnAddress += 8;
232 // Skip unimp instruction if function returns a struct
233 if ((addressSpace.get32(returnAddress) & 0xC1C00000) == 0)
234 returnAddress += 4;
236 #endif
238 #if defined(_LIBUNWIND_TARGET_PPC64)
239 #define PPC64_ELFV1_R2_LOAD_INST_ENCODING 0xe8410028u // ld r2,40(r1)
240 #define PPC64_ELFV1_R2_OFFSET 40
241 #define PPC64_ELFV2_R2_LOAD_INST_ENCODING 0xe8410018u // ld r2,24(r1)
242 #define PPC64_ELFV2_R2_OFFSET 24
243 // If the instruction at return address is a TOC (r2) restore,
244 // then r2 was saved and needs to be restored.
245 // ELFv2 ABI specifies that the TOC Pointer must be saved at SP + 24,
246 // while in ELFv1 ABI it is saved at SP + 40.
247 if (R::getArch() == REGISTERS_PPC64 && returnAddress != 0) {
248 pint_t sp = newRegisters.getRegister(UNW_REG_SP);
249 pint_t r2 = 0;
250 switch (addressSpace.get32(returnAddress)) {
251 case PPC64_ELFV1_R2_LOAD_INST_ENCODING:
252 r2 = addressSpace.get64(sp + PPC64_ELFV1_R2_OFFSET);
253 break;
254 case PPC64_ELFV2_R2_LOAD_INST_ENCODING:
255 r2 = addressSpace.get64(sp + PPC64_ELFV2_R2_OFFSET);
256 break;
258 if (r2)
259 newRegisters.setRegister(UNW_PPC64_R2, r2);
261 #endif
263 // Return address is address after call site instruction, so setting IP to
264 // that does simualates a return.
265 newRegisters.setIP(returnAddress);
267 // Simulate the step by replacing the register set with the new ones.
268 registers = newRegisters;
270 return UNW_STEP_SUCCESS;
273 return UNW_EBADFRAME;
276 template <typename A, typename R>
277 typename A::pint_t
278 DwarfInstructions<A, R>::evaluateExpression(pint_t expression, A &addressSpace,
279 const R &registers,
280 pint_t initialStackValue) {
281 const bool log = false;
282 pint_t p = expression;
283 pint_t expressionEnd = expression + 20; // temp, until len read
284 pint_t length = (pint_t)addressSpace.getULEB128(p, expressionEnd);
285 expressionEnd = p + length;
286 if (log)
287 fprintf(stderr, "evaluateExpression(): length=%" PRIu64 "\n",
288 (uint64_t)length);
289 pint_t stack[100];
290 pint_t *sp = stack;
291 *(++sp) = initialStackValue;
293 while (p < expressionEnd) {
294 if (log) {
295 for (pint_t *t = sp; t > stack; --t) {
296 fprintf(stderr, "sp[] = 0x%" PRIx64 "\n", (uint64_t)(*t));
299 uint8_t opcode = addressSpace.get8(p++);
300 sint_t svalue, svalue2;
301 pint_t value;
302 uint32_t reg;
303 switch (opcode) {
304 case DW_OP_addr:
305 // push immediate address sized value
306 value = addressSpace.getP(p);
307 p += sizeof(pint_t);
308 *(++sp) = value;
309 if (log)
310 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
311 break;
313 case DW_OP_deref:
314 // pop stack, dereference, push result
315 value = *sp--;
316 *(++sp) = addressSpace.getP(value);
317 if (log)
318 fprintf(stderr, "dereference 0x%" PRIx64 "\n", (uint64_t)value);
319 break;
321 case DW_OP_const1u:
322 // push immediate 1 byte value
323 value = addressSpace.get8(p);
324 p += 1;
325 *(++sp) = value;
326 if (log)
327 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
328 break;
330 case DW_OP_const1s:
331 // push immediate 1 byte signed value
332 svalue = (int8_t) addressSpace.get8(p);
333 p += 1;
334 *(++sp) = (pint_t)svalue;
335 if (log)
336 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
337 break;
339 case DW_OP_const2u:
340 // push immediate 2 byte value
341 value = addressSpace.get16(p);
342 p += 2;
343 *(++sp) = value;
344 if (log)
345 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
346 break;
348 case DW_OP_const2s:
349 // push immediate 2 byte signed value
350 svalue = (int16_t) addressSpace.get16(p);
351 p += 2;
352 *(++sp) = (pint_t)svalue;
353 if (log)
354 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
355 break;
357 case DW_OP_const4u:
358 // push immediate 4 byte value
359 value = addressSpace.get32(p);
360 p += 4;
361 *(++sp) = value;
362 if (log)
363 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
364 break;
366 case DW_OP_const4s:
367 // push immediate 4 byte signed value
368 svalue = (int32_t)addressSpace.get32(p);
369 p += 4;
370 *(++sp) = (pint_t)svalue;
371 if (log)
372 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
373 break;
375 case DW_OP_const8u:
376 // push immediate 8 byte value
377 value = (pint_t)addressSpace.get64(p);
378 p += 8;
379 *(++sp) = value;
380 if (log)
381 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
382 break;
384 case DW_OP_const8s:
385 // push immediate 8 byte signed value
386 value = (pint_t)addressSpace.get64(p);
387 p += 8;
388 *(++sp) = value;
389 if (log)
390 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
391 break;
393 case DW_OP_constu:
394 // push immediate ULEB128 value
395 value = (pint_t)addressSpace.getULEB128(p, expressionEnd);
396 *(++sp) = value;
397 if (log)
398 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)value);
399 break;
401 case DW_OP_consts:
402 // push immediate SLEB128 value
403 svalue = (sint_t)addressSpace.getSLEB128(p, expressionEnd);
404 *(++sp) = (pint_t)svalue;
405 if (log)
406 fprintf(stderr, "push 0x%" PRIx64 "\n", (uint64_t)svalue);
407 break;
409 case DW_OP_dup:
410 // push top of stack
411 value = *sp;
412 *(++sp) = value;
413 if (log)
414 fprintf(stderr, "duplicate top of stack\n");
415 break;
417 case DW_OP_drop:
418 // pop
419 --sp;
420 if (log)
421 fprintf(stderr, "pop top of stack\n");
422 break;
424 case DW_OP_over:
425 // dup second
426 value = sp[-1];
427 *(++sp) = value;
428 if (log)
429 fprintf(stderr, "duplicate second in stack\n");
430 break;
432 case DW_OP_pick:
433 // pick from
434 reg = addressSpace.get8(p);
435 p += 1;
436 value = sp[-(int)reg];
437 *(++sp) = value;
438 if (log)
439 fprintf(stderr, "duplicate %d in stack\n", reg);
440 break;
442 case DW_OP_swap:
443 // swap top two
444 value = sp[0];
445 sp[0] = sp[-1];
446 sp[-1] = value;
447 if (log)
448 fprintf(stderr, "swap top of stack\n");
449 break;
451 case DW_OP_rot:
452 // rotate top three
453 value = sp[0];
454 sp[0] = sp[-1];
455 sp[-1] = sp[-2];
456 sp[-2] = value;
457 if (log)
458 fprintf(stderr, "rotate top three of stack\n");
459 break;
461 case DW_OP_xderef:
462 // pop stack, dereference, push result
463 value = *sp--;
464 *sp = *((pint_t*)value);
465 if (log)
466 fprintf(stderr, "x-dereference 0x%" PRIx64 "\n", (uint64_t)value);
467 break;
469 case DW_OP_abs:
470 svalue = (sint_t)*sp;
471 if (svalue < 0)
472 *sp = (pint_t)(-svalue);
473 if (log)
474 fprintf(stderr, "abs\n");
475 break;
477 case DW_OP_and:
478 value = *sp--;
479 *sp &= value;
480 if (log)
481 fprintf(stderr, "and\n");
482 break;
484 case DW_OP_div:
485 svalue = (sint_t)(*sp--);
486 svalue2 = (sint_t)*sp;
487 *sp = (pint_t)(svalue2 / svalue);
488 if (log)
489 fprintf(stderr, "div\n");
490 break;
492 case DW_OP_minus:
493 value = *sp--;
494 *sp = *sp - value;
495 if (log)
496 fprintf(stderr, "minus\n");
497 break;
499 case DW_OP_mod:
500 svalue = (sint_t)(*sp--);
501 svalue2 = (sint_t)*sp;
502 *sp = (pint_t)(svalue2 % svalue);
503 if (log)
504 fprintf(stderr, "module\n");
505 break;
507 case DW_OP_mul:
508 svalue = (sint_t)(*sp--);
509 svalue2 = (sint_t)*sp;
510 *sp = (pint_t)(svalue2 * svalue);
511 if (log)
512 fprintf(stderr, "mul\n");
513 break;
515 case DW_OP_neg:
516 *sp = 0 - *sp;
517 if (log)
518 fprintf(stderr, "neg\n");
519 break;
521 case DW_OP_not:
522 svalue = (sint_t)(*sp);
523 *sp = (pint_t)(~svalue);
524 if (log)
525 fprintf(stderr, "not\n");
526 break;
528 case DW_OP_or:
529 value = *sp--;
530 *sp |= value;
531 if (log)
532 fprintf(stderr, "or\n");
533 break;
535 case DW_OP_plus:
536 value = *sp--;
537 *sp += value;
538 if (log)
539 fprintf(stderr, "plus\n");
540 break;
542 case DW_OP_plus_uconst:
543 // pop stack, add uelb128 constant, push result
544 *sp += static_cast<pint_t>(addressSpace.getULEB128(p, expressionEnd));
545 if (log)
546 fprintf(stderr, "add constant\n");
547 break;
549 case DW_OP_shl:
550 value = *sp--;
551 *sp = *sp << value;
552 if (log)
553 fprintf(stderr, "shift left\n");
554 break;
556 case DW_OP_shr:
557 value = *sp--;
558 *sp = *sp >> value;
559 if (log)
560 fprintf(stderr, "shift left\n");
561 break;
563 case DW_OP_shra:
564 value = *sp--;
565 svalue = (sint_t)*sp;
566 *sp = (pint_t)(svalue >> value);
567 if (log)
568 fprintf(stderr, "shift left arithmetric\n");
569 break;
571 case DW_OP_xor:
572 value = *sp--;
573 *sp ^= value;
574 if (log)
575 fprintf(stderr, "xor\n");
576 break;
578 case DW_OP_skip:
579 svalue = (int16_t) addressSpace.get16(p);
580 p += 2;
581 p = (pint_t)((sint_t)p + svalue);
582 if (log)
583 fprintf(stderr, "skip %" PRIu64 "\n", (uint64_t)svalue);
584 break;
586 case DW_OP_bra:
587 svalue = (int16_t) addressSpace.get16(p);
588 p += 2;
589 if (*sp--)
590 p = (pint_t)((sint_t)p + svalue);
591 if (log)
592 fprintf(stderr, "bra %" PRIu64 "\n", (uint64_t)svalue);
593 break;
595 case DW_OP_eq:
596 value = *sp--;
597 *sp = (*sp == value);
598 if (log)
599 fprintf(stderr, "eq\n");
600 break;
602 case DW_OP_ge:
603 value = *sp--;
604 *sp = (*sp >= value);
605 if (log)
606 fprintf(stderr, "ge\n");
607 break;
609 case DW_OP_gt:
610 value = *sp--;
611 *sp = (*sp > value);
612 if (log)
613 fprintf(stderr, "gt\n");
614 break;
616 case DW_OP_le:
617 value = *sp--;
618 *sp = (*sp <= value);
619 if (log)
620 fprintf(stderr, "le\n");
621 break;
623 case DW_OP_lt:
624 value = *sp--;
625 *sp = (*sp < value);
626 if (log)
627 fprintf(stderr, "lt\n");
628 break;
630 case DW_OP_ne:
631 value = *sp--;
632 *sp = (*sp != value);
633 if (log)
634 fprintf(stderr, "ne\n");
635 break;
637 case DW_OP_lit0:
638 case DW_OP_lit1:
639 case DW_OP_lit2:
640 case DW_OP_lit3:
641 case DW_OP_lit4:
642 case DW_OP_lit5:
643 case DW_OP_lit6:
644 case DW_OP_lit7:
645 case DW_OP_lit8:
646 case DW_OP_lit9:
647 case DW_OP_lit10:
648 case DW_OP_lit11:
649 case DW_OP_lit12:
650 case DW_OP_lit13:
651 case DW_OP_lit14:
652 case DW_OP_lit15:
653 case DW_OP_lit16:
654 case DW_OP_lit17:
655 case DW_OP_lit18:
656 case DW_OP_lit19:
657 case DW_OP_lit20:
658 case DW_OP_lit21:
659 case DW_OP_lit22:
660 case DW_OP_lit23:
661 case DW_OP_lit24:
662 case DW_OP_lit25:
663 case DW_OP_lit26:
664 case DW_OP_lit27:
665 case DW_OP_lit28:
666 case DW_OP_lit29:
667 case DW_OP_lit30:
668 case DW_OP_lit31:
669 value = static_cast<pint_t>(opcode - DW_OP_lit0);
670 *(++sp) = value;
671 if (log)
672 fprintf(stderr, "push literal 0x%" PRIx64 "\n", (uint64_t)value);
673 break;
675 case DW_OP_reg0:
676 case DW_OP_reg1:
677 case DW_OP_reg2:
678 case DW_OP_reg3:
679 case DW_OP_reg4:
680 case DW_OP_reg5:
681 case DW_OP_reg6:
682 case DW_OP_reg7:
683 case DW_OP_reg8:
684 case DW_OP_reg9:
685 case DW_OP_reg10:
686 case DW_OP_reg11:
687 case DW_OP_reg12:
688 case DW_OP_reg13:
689 case DW_OP_reg14:
690 case DW_OP_reg15:
691 case DW_OP_reg16:
692 case DW_OP_reg17:
693 case DW_OP_reg18:
694 case DW_OP_reg19:
695 case DW_OP_reg20:
696 case DW_OP_reg21:
697 case DW_OP_reg22:
698 case DW_OP_reg23:
699 case DW_OP_reg24:
700 case DW_OP_reg25:
701 case DW_OP_reg26:
702 case DW_OP_reg27:
703 case DW_OP_reg28:
704 case DW_OP_reg29:
705 case DW_OP_reg30:
706 case DW_OP_reg31:
707 reg = static_cast<uint32_t>(opcode - DW_OP_reg0);
708 *(++sp) = registers.getRegister((int)reg);
709 if (log)
710 fprintf(stderr, "push reg %d\n", reg);
711 break;
713 case DW_OP_regx:
714 reg = static_cast<uint32_t>(addressSpace.getULEB128(p, expressionEnd));
715 *(++sp) = registers.getRegister((int)reg);
716 if (log)
717 fprintf(stderr, "push reg %d + 0x%" PRIx64 "\n", reg, (uint64_t)svalue);
718 break;
720 case DW_OP_breg0:
721 case DW_OP_breg1:
722 case DW_OP_breg2:
723 case DW_OP_breg3:
724 case DW_OP_breg4:
725 case DW_OP_breg5:
726 case DW_OP_breg6:
727 case DW_OP_breg7:
728 case DW_OP_breg8:
729 case DW_OP_breg9:
730 case DW_OP_breg10:
731 case DW_OP_breg11:
732 case DW_OP_breg12:
733 case DW_OP_breg13:
734 case DW_OP_breg14:
735 case DW_OP_breg15:
736 case DW_OP_breg16:
737 case DW_OP_breg17:
738 case DW_OP_breg18:
739 case DW_OP_breg19:
740 case DW_OP_breg20:
741 case DW_OP_breg21:
742 case DW_OP_breg22:
743 case DW_OP_breg23:
744 case DW_OP_breg24:
745 case DW_OP_breg25:
746 case DW_OP_breg26:
747 case DW_OP_breg27:
748 case DW_OP_breg28:
749 case DW_OP_breg29:
750 case DW_OP_breg30:
751 case DW_OP_breg31:
752 reg = static_cast<uint32_t>(opcode - DW_OP_breg0);
753 svalue = (sint_t)addressSpace.getSLEB128(p, expressionEnd);
754 svalue += static_cast<sint_t>(registers.getRegister((int)reg));
755 *(++sp) = (pint_t)(svalue);
756 if (log)
757 fprintf(stderr, "push reg %d + 0x%" PRIx64 "\n", reg, (uint64_t)svalue);
758 break;
760 case DW_OP_bregx:
761 reg = static_cast<uint32_t>(addressSpace.getULEB128(p, expressionEnd));
762 svalue = (sint_t)addressSpace.getSLEB128(p, expressionEnd);
763 svalue += static_cast<sint_t>(registers.getRegister((int)reg));
764 *(++sp) = (pint_t)(svalue);
765 if (log)
766 fprintf(stderr, "push reg %d + 0x%" PRIx64 "\n", reg, (uint64_t)svalue);
767 break;
769 case DW_OP_fbreg:
770 _LIBUNWIND_ABORT("DW_OP_fbreg not implemented");
771 break;
773 case DW_OP_piece:
774 _LIBUNWIND_ABORT("DW_OP_piece not implemented");
775 break;
777 case DW_OP_deref_size:
778 // pop stack, dereference, push result
779 value = *sp--;
780 switch (addressSpace.get8(p++)) {
781 case 1:
782 value = addressSpace.get8(value);
783 break;
784 case 2:
785 value = addressSpace.get16(value);
786 break;
787 case 4:
788 value = addressSpace.get32(value);
789 break;
790 case 8:
791 value = (pint_t)addressSpace.get64(value);
792 break;
793 default:
794 _LIBUNWIND_ABORT("DW_OP_deref_size with bad size");
796 *(++sp) = value;
797 if (log)
798 fprintf(stderr, "sized dereference 0x%" PRIx64 "\n", (uint64_t)value);
799 break;
801 case DW_OP_xderef_size:
802 case DW_OP_nop:
803 case DW_OP_push_object_addres:
804 case DW_OP_call2:
805 case DW_OP_call4:
806 case DW_OP_call_ref:
807 default:
808 _LIBUNWIND_ABORT("DWARF opcode not implemented");
812 if (log)
813 fprintf(stderr, "expression evaluates to 0x%" PRIx64 "\n", (uint64_t)*sp);
814 return *sp;
819 } // namespace libunwind
821 #endif // __DWARF_INSTRUCTIONS_HPP__