Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 1 | # Copyright 2014 The Chromium OS Authors. All rights reserved. |
| 2 | # Use of this source code is governed by a BSD-style license that can be |
| 3 | # found in the LICENSE file. |
| 4 | |
| 5 | """Connect to a DUT in firmware via remote GDB, install custom GDB commands.""" |
| 6 | |
| 7 | import errno |
Raul E Rangel | eb8aadc | 2018-05-17 09:10:27 -0600 | [diff] [blame] | 8 | import glob |
Chris McDonald | 59650c3 | 2021-07-20 15:29:28 -0600 | [diff] [blame] | 9 | import logging |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 10 | import os |
| 11 | import re |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 12 | import socket |
| 13 | import time |
| 14 | |
Chris McDonald | 59650c3 | 2021-07-20 15:29:28 -0600 | [diff] [blame] | 15 | from chromite.third_party.pyelftools.elftools.elf.elffile import ELFFile |
| 16 | |
Mike Frysinger | 06a51c8 | 2021-04-06 11:39:17 -0400 | [diff] [blame] | 17 | from chromite.lib import build_target_lib |
Aviv Keshet | b7519e1 | 2016-10-04 00:50:00 -0700 | [diff] [blame] | 18 | from chromite.lib import constants |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 19 | from chromite.lib import cros_build_lib |
| 20 | from chromite.lib import timeout_util |
Chris McDonald | 59650c3 | 2021-07-20 15:29:28 -0600 | [diff] [blame] | 21 | |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 22 | |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 23 | # Need to do this before Servo import |
| 24 | cros_build_lib.AssertInsideChroot() |
| 25 | |
Mike Frysinger | 92bdef5 | 2019-08-21 21:05:13 -0400 | [diff] [blame] | 26 | # pylint: disable=import-error,wrong-import-position |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 27 | from servo import client |
Ruben Rodriguez Buchillon | 15bdd45 | 2018-10-09 08:31:41 +0800 | [diff] [blame] | 28 | from servo import servo_parsing |
Aseda Aboagye | 6e21d3f | 2016-10-21 11:37:56 -0700 | [diff] [blame] | 29 | from servo import terminal_freezer |
Chris McDonald | 59650c3 | 2021-07-20 15:29:28 -0600 | [diff] [blame] | 30 | |
| 31 | |
Mike Frysinger | 92bdef5 | 2019-08-21 21:05:13 -0400 | [diff] [blame] | 32 | # pylint: enable=import-error,wrong-import-position |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 33 | |
Ralph Nathan | 91874ca | 2015-03-19 13:29:41 -0700 | [diff] [blame] | 34 | |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 35 | _SRC_ROOT = os.path.join(constants.CHROOT_SOURCE_ROOT, 'src') |
| 36 | _SRC_DC = os.path.join(_SRC_ROOT, 'platform/depthcharge') |
| 37 | _SRC_VB = os.path.join(_SRC_ROOT, 'platform/vboot_reference') |
| 38 | _SRC_LP = os.path.join(_SRC_ROOT, 'third_party/coreboot/payloads/libpayload') |
| 39 | |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 40 | _PTRN_GDB = b'Ready for GDB connection' |
Douglas Anderson | 5a1056b | 2021-08-26 14:10:08 -0700 | [diff] [blame] | 41 | _PTRN_BOARD = ( |
| 42 | b'Starting(?: read-only| read/write)? depthcharge on ([A-Za-z_]+)...') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 43 | |
| 44 | |
Julius Werner | 149fe78 | 2018-10-10 15:18:14 -0700 | [diff] [blame] | 45 | def GetGdbForElf(elf): |
| 46 | """Return the correct C compiler prefix for the target ELF file.""" |
Mike Frysinger | 17844a0 | 2019-08-24 18:21:02 -0400 | [diff] [blame] | 47 | with open(elf, 'rb') as fp: |
Julius Werner | 149fe78 | 2018-10-10 15:18:14 -0700 | [diff] [blame] | 48 | return { |
| 49 | 'EM_386': 'x86_64-cros-linux-gnu-gdb', |
| 50 | 'EM_X86_64': 'x86_64-cros-linux-gnu-gdb', |
| 51 | 'EM_ARM': 'armv7a-cros-linux-gnueabihf-gdb', |
| 52 | 'EM_AARCH64': 'aarch64-cros-linux-gnu-gdb', |
Mike Frysinger | 17844a0 | 2019-08-24 18:21:02 -0400 | [diff] [blame] | 53 | }[ELFFile(fp).header.e_machine] |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 54 | |
| 55 | |
| 56 | def ParseArgs(argv): |
| 57 | """Parse and validate command line arguments.""" |
Raul E Rangel | b54ec04 | 2020-02-06 09:25:12 -0700 | [diff] [blame] | 58 | description = 'Debug depthcharge using GDB' |
| 59 | parser = servo_parsing.ServodClientParser(description=description) |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 60 | parser.add_argument('-b', '--board', |
| 61 | help='The board overlay name (auto-detect by default)') |
Raul E Rangel | 72f3271 | 2018-05-29 11:42:59 -0600 | [diff] [blame] | 62 | parser.add_argument('-c', '--cgdb', action='store_true', |
| 63 | help='Use cgdb curses interface rather than plain gdb') |
Raul E Rangel | b54ec04 | 2020-02-06 09:25:12 -0700 | [diff] [blame] | 64 | parser.add_argument('-y', '--symbols', |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 65 | help='Root directory or complete path to symbolized ELF ' |
| 66 | '(defaults to /build/<BOARD>/firmware)') |
| 67 | parser.add_argument('-r', '--reboot', choices=['yes', 'no', 'auto'], |
| 68 | help='Reboot the DUT before connect (default: reboot if ' |
| 69 | 'the remote and is unreachable)', default='auto') |
| 70 | parser.add_argument('-e', '--execute', action='append', default=[], |
| 71 | help='GDB command to run after connect (can be supplied ' |
| 72 | 'multiple times)') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 73 | parser.add_argument('-t', '--tty', |
| 74 | help='TTY file to connect to (defaults to cpu_uart_pty)') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 75 | opts = parser.parse_args(argv) |
Ruben Rodriguez Buchillon | d7c65bd | 2018-12-10 10:10:19 +0800 | [diff] [blame] | 76 | |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 77 | return opts |
| 78 | |
| 79 | |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 80 | def FindSymbols(firmware_dir, board): |
Douglas Anderson | 69e5e70 | 2021-08-13 11:10:19 -0700 | [diff] [blame] | 81 | """Find the symbolized depthcharge ELF (may be supplied by -y flag).""" |
Raul E Rangel | eb8aadc | 2018-05-17 09:10:27 -0600 | [diff] [blame] | 82 | |
| 83 | # Allow overriding the file directly just in case our detection screws up. |
| 84 | if firmware_dir and firmware_dir.endswith('.elf'): |
| 85 | return firmware_dir |
| 86 | |
| 87 | if not firmware_dir: |
| 88 | # Unified builds have the format |
| 89 | # /build/<board|family>/firmware/<build_target|model>/. The board in |
| 90 | # depthcharge corresponds to the build_target in unified builds. For this |
| 91 | # reason we need to glob all boards to find the correct build_target. |
| 92 | unified_build_dirs = glob.glob('/build/*/firmware/%s' % board) |
| 93 | if len(unified_build_dirs) == 1: |
| 94 | firmware_dir = unified_build_dirs[0] |
| 95 | elif len(unified_build_dirs) > 1: |
| 96 | raise ValueError( |
Douglas Anderson | 69e5e70 | 2021-08-13 11:10:19 -0700 | [diff] [blame] | 97 | 'Multiple boards were found (%s). Use -y to specify manually' % |
Raul E Rangel | eb8aadc | 2018-05-17 09:10:27 -0600 | [diff] [blame] | 98 | (', '.join(unified_build_dirs))) |
| 99 | |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 100 | if not firmware_dir: |
Mike Frysinger | 06a51c8 | 2021-04-06 11:39:17 -0400 | [diff] [blame] | 101 | firmware_dir = os.path.join( |
| 102 | build_target_lib.get_default_sysroot_path(board), 'firmware') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 103 | |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 104 | # Very old firmware you might still find on GoldenEye had dev.ro.elf. |
| 105 | basenames = ['dev.elf', 'dev.ro.elf'] |
| 106 | for basename in basenames: |
| 107 | path = os.path.join(firmware_dir, 'depthcharge', basename) |
| 108 | if not os.path.exists(path): |
| 109 | path = os.path.join(firmware_dir, basename) |
| 110 | if os.path.exists(path): |
| 111 | logging.warning('Auto-detected symbol file at %s... make sure that this' |
| 112 | ' matches the image on your DUT!', path) |
| 113 | return path |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 114 | |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 115 | raise ValueError('Could not find depthcharge symbol file (dev.elf)! ' |
| 116 | '(You can use -s to supply it manually.)') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 117 | |
| 118 | |
| 119 | # TODO(jwerner): Fine tune |wait| delay or maybe even make it configurable if |
| 120 | # this causes problems due to load on the host. The callers where this is |
| 121 | # critical should all have their own timeouts now, though, so it's questionable |
| 122 | # whether the delay here is even needed at all anymore. |
| 123 | def ReadAll(fd, wait=0.03): |
| 124 | """Read from |fd| until no more data has come for at least |wait| seconds.""" |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 125 | data = [] |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 126 | try: |
| 127 | while True: |
| 128 | time.sleep(wait) |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 129 | new_data = os.read(fd, 4096) |
| 130 | if not new_data: |
| 131 | break |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 132 | data.append(new_data) |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 133 | except OSError as e: |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 134 | if e.errno != errno.EAGAIN: |
| 135 | raise |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 136 | data = b''.join(data) |
Julius Werner | bb98c22 | 2022-06-23 16:08:35 -0700 | [diff] [blame] | 137 | if data: |
| 138 | logging.debug(data.decode('ascii', errors='replace')) |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 139 | return data |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 140 | |
| 141 | |
| 142 | def GdbChecksum(message): |
| 143 | """Calculate a remote-GDB style checksum.""" |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 144 | chksum = sum(message) |
| 145 | return (b'%.2x' % chksum)[-2:] |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 146 | |
| 147 | |
| 148 | def TestConnection(fd): |
| 149 | """Return True iff there is a resposive GDB stub on the other end of 'fd'.""" |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 150 | cmd = b'vUnknownCommand' |
Mike Frysinger | 79cca96 | 2019-06-13 15:26:53 -0400 | [diff] [blame] | 151 | for _ in range(3): |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 152 | os.write(fd, b'$%s#%s\n' % (cmd, GdbChecksum(cmd))) |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 153 | reply = ReadAll(fd) |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 154 | if b'+$#00' in reply: |
| 155 | os.write(fd, b'+') |
Mike Frysinger | 2403af4 | 2015-04-30 06:25:03 -0400 | [diff] [blame] | 156 | logging.info('TestConnection: Could successfully connect to remote end.') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 157 | return True |
Mike Frysinger | 2403af4 | 2015-04-30 06:25:03 -0400 | [diff] [blame] | 158 | logging.info('TestConnection: Remote end does not respond.') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 159 | return False |
| 160 | |
| 161 | |
| 162 | def main(argv): |
| 163 | opts = ParseArgs(argv) |
Raul E Rangel | b54ec04 | 2020-02-06 09:25:12 -0700 | [diff] [blame] | 164 | servo = client.ServoClient(host=opts.host, port=opts.port) |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 165 | |
| 166 | if not opts.tty: |
| 167 | try: |
| 168 | opts.tty = servo.get('cpu_uart_pty') |
| 169 | except (client.ServoClientError, socket.error): |
Mike Frysinger | 2403af4 | 2015-04-30 06:25:03 -0400 | [diff] [blame] | 170 | logging.error('Cannot auto-detect TTY file without servod. Use the --tty ' |
| 171 | 'option.') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 172 | raise |
Aseda Aboagye | 6e21d3f | 2016-10-21 11:37:56 -0700 | [diff] [blame] | 173 | with terminal_freezer.TerminalFreezer(opts.tty): |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 174 | fd = os.open(opts.tty, os.O_RDWR | os.O_NONBLOCK) |
| 175 | |
| 176 | data = ReadAll(fd) |
| 177 | if opts.reboot == 'auto': |
| 178 | if TestConnection(fd): |
| 179 | opts.reboot = 'no' |
| 180 | else: |
| 181 | opts.reboot = 'yes' |
| 182 | |
| 183 | if opts.reboot == 'yes': |
Mike Frysinger | 2403af4 | 2015-04-30 06:25:03 -0400 | [diff] [blame] | 184 | logging.info('Rebooting DUT...') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 185 | try: |
| 186 | servo.set('warm_reset', 'on') |
| 187 | time.sleep(0.1) |
| 188 | servo.set('warm_reset', 'off') |
| 189 | except (client.ServoClientError, socket.error): |
Mike Frysinger | 2403af4 | 2015-04-30 06:25:03 -0400 | [diff] [blame] | 190 | logging.error('Cannot reboot without a Servo board. You have to boot ' |
| 191 | 'into developer mode and press CTRL+G manually before ' |
| 192 | 'running fwgdb.') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 193 | raise |
| 194 | |
| 195 | # Throw away old data to avoid confusion from messages before the reboot |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 196 | data = b'' |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 197 | msg = ('Could not reboot into depthcharge!') |
Mike Frysinger | d6e2df0 | 2014-11-26 02:55:04 -0500 | [diff] [blame] | 198 | with timeout_util.Timeout(10, msg): |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 199 | while not re.search(_PTRN_BOARD, data): |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 200 | data += ReadAll(fd) |
| 201 | |
Mike Frysinger | d6e2df0 | 2014-11-26 02:55:04 -0500 | [diff] [blame] | 202 | msg = ('Could not enter GDB mode with CTRL+G! ' |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 203 | '(Confirm that you flashed an "image.dev.bin" image to this DUT, ' |
| 204 | 'and that you have GBB_FLAG_FORCE_DEV_SWITCH_ON (0x8) set.)') |
| 205 | with timeout_util.Timeout(5, msg): |
| 206 | while not re.search(_PTRN_GDB, data): |
Julius Werner | bb98c22 | 2022-06-23 16:08:35 -0700 | [diff] [blame] | 207 | # Some delay to avoid spamming the console too hard while not being |
| 208 | # long enough to cause a user-visible slowdown. |
| 209 | time.sleep(0.5) |
Julius Werner | 193c914 | 2018-05-09 16:25:38 -0700 | [diff] [blame] | 210 | # Send a CTRL+G to tell depthcharge to trap into GDB. |
| 211 | logging.debug('[Ctrl+G]') |
Douglas Anderson | 62dd87e | 2021-08-13 11:08:54 -0700 | [diff] [blame] | 212 | os.write(fd, b'\x07') |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 213 | data += ReadAll(fd) |
| 214 | |
| 215 | if not opts.board: |
| 216 | matches = re.findall(_PTRN_BOARD, data) |
| 217 | if not matches: |
| 218 | raise ValueError('Could not auto-detect board! Please use -b option.') |
| 219 | opts.board = matches[-1] |
Mike Frysinger | 2403af4 | 2015-04-30 06:25:03 -0400 | [diff] [blame] | 220 | logging.info('Auto-detected board as %s from DUT console output.', |
| 221 | opts.board) |
Julius Werner | 634ccac | 2014-06-24 13:59:18 -0700 | [diff] [blame] | 222 | |
| 223 | if not TestConnection(fd): |
| 224 | raise IOError('Could not connect to remote end! Confirm that your DUT is ' |
| 225 | 'running in GDB mode on %s.' % opts.tty) |
| 226 | |
| 227 | # Eat up leftover data or it will spill back to terminal |
| 228 | ReadAll(fd) |
| 229 | os.close(fd) |
| 230 | |
| 231 | opts.execute.insert(0, 'target remote %s' % opts.tty) |
| 232 | ex_args = sum([['--ex', cmd] for cmd in opts.execute], []) |
| 233 | |
Julius Werner | 149fe78 | 2018-10-10 15:18:14 -0700 | [diff] [blame] | 234 | elf = FindSymbols(opts.symbols, opts.board) |
| 235 | gdb_cmd = GetGdbForElf(elf) |
Raul E Rangel | 72f3271 | 2018-05-29 11:42:59 -0600 | [diff] [blame] | 236 | |
Raul E Rangel | 72f3271 | 2018-05-29 11:42:59 -0600 | [diff] [blame] | 237 | gdb_args = [ |
Julius Werner | 149fe78 | 2018-10-10 15:18:14 -0700 | [diff] [blame] | 238 | '--symbols', elf, |
Raul E Rangel | 72f3271 | 2018-05-29 11:42:59 -0600 | [diff] [blame] | 239 | '--directory', _SRC_DC, |
| 240 | '--directory', _SRC_VB, |
| 241 | '--directory', _SRC_LP, |
| 242 | ] + ex_args |
| 243 | |
| 244 | if opts.cgdb: |
| 245 | full_cmd = ['cgdb', '-d', gdb_cmd, '--'] + gdb_args |
| 246 | else: |
| 247 | full_cmd = [gdb_cmd] + gdb_args |
| 248 | |
Julius Werner | 149fe78 | 2018-10-10 15:18:14 -0700 | [diff] [blame] | 249 | logging.info('Launching GDB...') |
Mike Frysinger | 45602c7 | 2019-09-22 02:15:11 -0400 | [diff] [blame] | 250 | cros_build_lib.run( |
Raul E Rangel | 72f3271 | 2018-05-29 11:42:59 -0600 | [diff] [blame] | 251 | full_cmd, ignore_sigint=True, debug_level=logging.WARNING) |