Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 1 | # |
Patrick Williams | 92b42cb | 2022-09-03 06:53:57 -0500 | [diff] [blame] | 2 | # Copyright OpenEmbedded Contributors |
| 3 | # |
Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 4 | # SPDX-License-Identifier: MIT |
| 5 | # |
| 6 | |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 7 | import os |
| 8 | import sys |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 9 | import json |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 10 | import errno |
| 11 | import datetime |
| 12 | import itertools |
Patrick Williams | c0f7c04 | 2017-02-23 20:41:17 -0600 | [diff] [blame] | 13 | from .commands import runCmd |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 14 | |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 15 | class BaseDumper(object): |
| 16 | """ Base class to dump commands from host/target """ |
| 17 | |
| 18 | def __init__(self, cmds, parent_dir): |
| 19 | self.cmds = [] |
Patrick Williams | f1e5d69 | 2016-03-30 15:21:19 -0500 | [diff] [blame] | 20 | # Some testing doesn't inherit testimage, so it is needed |
| 21 | # to set some defaults. |
Brad Bishop | 977dc1a | 2019-02-06 16:01:43 -0500 | [diff] [blame] | 22 | self.parent_dir = parent_dir |
Andrew Geissler | 5f35090 | 2021-07-23 13:09:54 -0400 | [diff] [blame] | 23 | self.dump_dir = parent_dir |
Patrick Williams | f1e5d69 | 2016-03-30 15:21:19 -0500 | [diff] [blame] | 24 | dft_cmds = """ top -bn1 |
| 25 | iostat -x -z -N -d -p ALL 20 2 |
| 26 | ps -ef |
| 27 | free |
| 28 | df |
| 29 | memstat |
| 30 | dmesg |
| 31 | ip -s link |
| 32 | netstat -an""" |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 33 | if not cmds: |
Patrick Williams | f1e5d69 | 2016-03-30 15:21:19 -0500 | [diff] [blame] | 34 | cmds = dft_cmds |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 35 | for cmd in cmds.split('\n'): |
| 36 | cmd = cmd.lstrip() |
| 37 | if not cmd or cmd[0] == '#': |
| 38 | continue |
| 39 | self.cmds.append(cmd) |
| 40 | |
| 41 | def create_dir(self, dir_suffix): |
| 42 | dump_subdir = ("%s_%s" % ( |
| 43 | datetime.datetime.now().strftime('%Y%m%d%H%M'), |
| 44 | dir_suffix)) |
| 45 | dump_dir = os.path.join(self.parent_dir, dump_subdir) |
| 46 | try: |
| 47 | os.makedirs(dump_dir) |
| 48 | except OSError as err: |
| 49 | if err.errno != errno.EEXIST: |
| 50 | raise err |
| 51 | self.dump_dir = dump_dir |
| 52 | |
Andrew Geissler | 5f35090 | 2021-07-23 13:09:54 -0400 | [diff] [blame] | 53 | def _construct_filename(self, command): |
Andrew Geissler | 8f84068 | 2023-07-21 09:09:43 -0500 | [diff] [blame] | 54 | if isinstance(self, TargetDumper): |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 55 | prefix = "target" |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 56 | elif isinstance(self, MonitorDumper): |
| 57 | prefix = "qmp" |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 58 | else: |
| 59 | prefix = "unknown" |
| 60 | for i in itertools.count(): |
| 61 | filename = "%s_%02d_%s" % (prefix, i, command) |
| 62 | fullname = os.path.join(self.dump_dir, filename) |
| 63 | if not os.path.exists(fullname): |
| 64 | break |
Andrew Geissler | 5f35090 | 2021-07-23 13:09:54 -0400 | [diff] [blame] | 65 | return fullname |
| 66 | |
| 67 | def _write_dump(self, command, output): |
| 68 | fullname = self._construct_filename(command) |
Andrew Geissler | 7e0e3c0 | 2022-02-25 20:34:39 +0000 | [diff] [blame] | 69 | os.makedirs(os.path.dirname(fullname), exist_ok=True) |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 70 | if isinstance(self, MonitorDumper): |
| 71 | with open(fullname, 'w') as json_file: |
| 72 | json.dump(output, json_file, indent=4) |
| 73 | else: |
| 74 | with open(fullname, 'w') as dump_file: |
| 75 | dump_file.write(output) |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 76 | |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 77 | class TargetDumper(BaseDumper): |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 78 | """ Class to get dumps from target, it only works with QemuRunner. |
| 79 | Will give up permanently after 5 errors from running commands over |
| 80 | serial console. This helps to end testing when target is really dead, hanging |
| 81 | or unresponsive. |
| 82 | """ |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 83 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 84 | def __init__(self, cmds, parent_dir, runner): |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 85 | super(TargetDumper, self).__init__(cmds, parent_dir) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 86 | self.runner = runner |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 87 | self.errors = 0 |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 88 | |
| 89 | def dump_target(self, dump_dir=""): |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 90 | if self.errors >= 5: |
| 91 | print("Too many errors when dumping data from target, assuming it is dead! Will not dump data anymore!") |
| 92 | return |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 93 | if dump_dir: |
| 94 | self.dump_dir = dump_dir |
| 95 | for cmd in self.cmds: |
| 96 | # We can continue with the testing if serial commands fail |
| 97 | try: |
| 98 | (status, output) = self.runner.run_serial(cmd) |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 99 | if status == 0: |
| 100 | self.errors = self.errors + 1 |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 101 | self._write_dump(cmd.split()[0], output) |
| 102 | except: |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 103 | self.errors = self.errors + 1 |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 104 | print("Tried to dump info from target but " |
| 105 | "serial console failed") |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 106 | print("Failed CMD: %s" % (cmd)) |
| 107 | |
| 108 | class MonitorDumper(BaseDumper): |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 109 | """ Class to get dumps via the Qemu Monitor, it only works with QemuRunner |
| 110 | Will stop completely if there are more than 5 errors when dumping monitor data. |
| 111 | This helps to end testing when target is really dead, hanging or unresponsive. |
| 112 | """ |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 113 | |
| 114 | def __init__(self, cmds, parent_dir, runner): |
| 115 | super(MonitorDumper, self).__init__(cmds, parent_dir) |
| 116 | self.runner = runner |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 117 | self.errors = 0 |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 118 | |
| 119 | def dump_monitor(self, dump_dir=""): |
| 120 | if self.runner is None: |
| 121 | return |
| 122 | if dump_dir: |
| 123 | self.dump_dir = dump_dir |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 124 | if self.errors >= 5: |
| 125 | print("Too many errors when dumping data from qemu monitor, assuming it is dead! Will not dump data anymore!") |
| 126 | return |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 127 | for cmd in self.cmds: |
Andrew Geissler | 5f35090 | 2021-07-23 13:09:54 -0400 | [diff] [blame] | 128 | cmd_name = cmd.split()[0] |
Andrew Geissler | c926e17 | 2021-05-07 16:11:35 -0500 | [diff] [blame] | 129 | try: |
Andrew Geissler | 5f35090 | 2021-07-23 13:09:54 -0400 | [diff] [blame] | 130 | if len(cmd.split()) > 1: |
| 131 | cmd_args = cmd.split()[1] |
| 132 | if "%s" in cmd_args: |
| 133 | filename = self._construct_filename(cmd_name) |
| 134 | cmd_data = json.loads(cmd_args % (filename)) |
| 135 | output = self.runner.run_monitor(cmd_name, cmd_data) |
| 136 | else: |
| 137 | output = self.runner.run_monitor(cmd_name) |
| 138 | self._write_dump(cmd_name, output) |
| 139 | except Exception as e: |
Andrew Geissler | 6aa7eec | 2023-03-03 12:41:14 -0600 | [diff] [blame] | 140 | self.errors = self.errors + 1 |
Andrew Geissler | 595f630 | 2022-01-24 19:11:47 +0000 | [diff] [blame] | 141 | print("Failed to dump QMP CMD: %s with\nException: %s" % (cmd_name, e)) |