1#!/usr/bin/env python3
2#
3# Copyright (c) 2022 Huawei Technologies SASU
4#
5# SPDX-License-Identifier: Apache-2.0
6
7import binascii
8import logging
9import struct
10
11from gdbstubs.gdbstub import GdbStub
12
13
14logger = logging.getLogger("gdbstub")
15
16
17class RegNum():
18    X0 = 0          # X0-X29 - 30 GP registers
19    X1 = 1
20    X2 = 2
21    X3 = 3
22    X4 = 4
23    X5 = 5
24    X6 = 6
25    X7 = 7
26    X8 = 8
27    X9 = 9
28    X10 = 10
29    X11 = 11
30    X12 = 12
31    X13 = 13
32    X14 = 14
33    X15 = 15
34    X16 = 16
35    X17 = 17
36    X18 = 18
37    X19 = 19
38    X20 = 20
39    X21 = 21
40    X22 = 22
41    X23 = 23
42    X24 = 24
43    X25 = 25
44    X26 = 26
45    X27 = 27
46    X28 = 28
47    X29 = 29         # Frame pointer register
48    LR      = 30     # X30 Link Register(LR)
49    SP_EL0  = 31     # Stack pointer EL0 (SP_EL0)
50    PC      = 32     # Program Counter (PC)
51
52
53class GdbStub_ARM64(GdbStub):
54    ARCH_DATA_BLK_STRUCT    = "<QQQQQQQQQQQQQQQQQQQQQQ"
55
56    # Default signal used by all other script, just using the same
57    GDB_SIGNAL_DEFAULT = 7
58
59    # The number of registers expected by GDB
60    GDB_G_PKT_NUM_REGS = 33
61
62
63    def __init__(self, logfile, elffile):
64        super().__init__(logfile=logfile, elffile=elffile)
65        self.registers = None
66        self.gdb_signal = self.GDB_SIGNAL_DEFAULT
67
68        self.parse_arch_data_block()
69
70    def parse_arch_data_block(self):
71
72        arch_data_blk = self.logfile.get_arch_data()['data']
73
74        tu = struct.unpack(self.ARCH_DATA_BLK_STRUCT, arch_data_blk)
75
76        self.registers = dict()
77
78        self.registers[RegNum.X0]   = tu[0]
79        self.registers[RegNum.X1]   = tu[1]
80        self.registers[RegNum.X2]   = tu[2]
81        self.registers[RegNum.X3]   = tu[3]
82        self.registers[RegNum.X4]   = tu[4]
83        self.registers[RegNum.X5]   = tu[5]
84        self.registers[RegNum.X6]   = tu[6]
85        self.registers[RegNum.X7]   = tu[7]
86        self.registers[RegNum.X8]   = tu[8]
87        self.registers[RegNum.X9]   = tu[9]
88        self.registers[RegNum.X10]  = tu[10]
89        self.registers[RegNum.X11]  = tu[11]
90        self.registers[RegNum.X12]  = tu[12]
91        self.registers[RegNum.X13]  = tu[13]
92        self.registers[RegNum.X14]  = tu[14]
93        self.registers[RegNum.X15]  = tu[15]
94        self.registers[RegNum.X16]  = tu[16]
95        self.registers[RegNum.X17]  = tu[17]
96        self.registers[RegNum.X18]  = tu[18]
97
98        # Callee saved registers are not provided in arch_esf structure
99        # So they will be omitted (set to undefined) when stub generates the
100        # packet in handle_register_group_read_packet.
101
102        self.registers[RegNum.LR]       = tu[19]
103        self.registers[RegNum.SP_EL0]   = tu[20]
104        self.registers[RegNum.PC]       = tu[21]
105
106
107    def handle_register_group_read_packet(self):
108        reg_fmt = "<Q"
109
110        idx = 0
111        pkt = b''
112
113        while idx < self.GDB_G_PKT_NUM_REGS:
114            if idx in self.registers:
115                bval = struct.pack(reg_fmt, self.registers[idx])
116                pkt += binascii.hexlify(bval)
117            else:
118                # Register not in coredump -> unknown value
119                # Send in "xxxxxxxx"
120                pkt += b'x' * 16
121
122            idx += 1
123
124        self.put_gdb_packet(pkt)
125
126    def handle_register_single_read_packet(self, pkt):
127        # Mark registers as "<unavailable>".
128        # 'p' packets are usually used for registers
129        # other than the general ones (e.g. eax, ebx)
130        # so we can safely reply "xxxxxxxx" here.
131        self.put_gdb_packet(b'x' * 16)
132