2022-05-23 15:30:13 +02:00
|
|
|
# SPDX-FileCopyrightText: 2022 Espressif Systems (Shanghai) CO LTD
|
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
2022-02-17 15:27:34 +08:00
|
|
|
import io
|
2021-05-14 11:20:38 +02:00
|
|
|
import os
|
|
|
|
import queue
|
|
|
|
import tempfile
|
2022-02-17 15:27:34 +08:00
|
|
|
from contextlib import contextmanager, redirect_stdout
|
2021-05-14 11:20:38 +02:00
|
|
|
from typing import Generator
|
|
|
|
|
2022-02-17 15:27:34 +08:00
|
|
|
from .constants import TAG_KEY
|
2021-05-14 11:20:38 +02:00
|
|
|
from .logger import Logger
|
|
|
|
from .output_helpers import yellow_print
|
2021-05-14 11:20:38 +02:00
|
|
|
from .web_socket_client import WebSocketClient
|
|
|
|
|
|
|
|
# coredump related messages
|
|
|
|
COREDUMP_UART_START = b'================= CORE DUMP START ================='
|
|
|
|
COREDUMP_UART_END = b'================= CORE DUMP END ================='
|
|
|
|
COREDUMP_UART_PROMPT = b'Press Enter to print core dump to UART...'
|
|
|
|
|
|
|
|
# coredump states
|
|
|
|
COREDUMP_IDLE = 0
|
|
|
|
COREDUMP_READING = 1
|
|
|
|
COREDUMP_DONE = 2
|
|
|
|
|
|
|
|
# coredump decoding options
|
|
|
|
COREDUMP_DECODE_DISABLE = 'disable'
|
|
|
|
COREDUMP_DECODE_INFO = 'info'
|
|
|
|
|
|
|
|
|
|
|
|
class CoreDump:
|
|
|
|
def __init__(self, decode_coredumps, event_queue, logger, websocket_client, elf_file):
|
|
|
|
# type: (str, queue.Queue, Logger, WebSocketClient, str) -> None
|
|
|
|
|
|
|
|
self._coredump_buffer = b''
|
|
|
|
self._decode_coredumps = decode_coredumps
|
|
|
|
self.event_queue = event_queue
|
|
|
|
self._reading_coredump = COREDUMP_IDLE
|
|
|
|
self.logger = logger
|
|
|
|
self.websocket_client = websocket_client
|
|
|
|
self.elf_file = elf_file
|
|
|
|
|
2022-02-22 17:13:45 +01:00
|
|
|
@property
|
|
|
|
def in_progress(self) -> bool:
|
|
|
|
return bool(self._coredump_buffer)
|
|
|
|
|
2021-05-14 11:20:38 +02:00
|
|
|
def _process_coredump(self): # type: () -> None
|
|
|
|
if self._decode_coredumps != COREDUMP_DECODE_INFO:
|
|
|
|
raise NotImplementedError('process_coredump: %s not implemented' % self._decode_coredumps)
|
|
|
|
coredump_file = None
|
2022-02-17 15:27:34 +08:00
|
|
|
# On Windows, the temporary file can't be read unless it is closed.
|
|
|
|
# Set delete=False and delete the file manually later.
|
|
|
|
with tempfile.NamedTemporaryFile(mode='wb', delete=False) as coredump_file:
|
|
|
|
coredump_file.write(self._coredump_buffer)
|
|
|
|
coredump_file.flush()
|
2021-05-14 11:20:38 +02:00
|
|
|
|
2022-02-17 15:27:34 +08:00
|
|
|
if self.websocket_client:
|
|
|
|
self.logger.output_enabled = True
|
|
|
|
yellow_print('Communicating through WebSocket')
|
|
|
|
self.websocket_client.send({'event': 'coredump',
|
|
|
|
'file': coredump_file.name,
|
|
|
|
'prog': self.elf_file})
|
|
|
|
yellow_print('Waiting for debug finished event')
|
|
|
|
self.websocket_client.wait([('event', 'debug_finished')])
|
|
|
|
yellow_print('Communications through WebSocket is finished')
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
import esp_coredump
|
|
|
|
except ImportError as e:
|
|
|
|
yellow_print('Failed to parse core dump info: '
|
|
|
|
'Module {} is not installed \n\n'.format(e.name))
|
2021-05-14 11:20:38 +02:00
|
|
|
self.logger.output_enabled = True
|
2022-02-17 15:27:34 +08:00
|
|
|
self.logger.print(COREDUMP_UART_START + b'\n')
|
|
|
|
self.logger.print(self._coredump_buffer)
|
|
|
|
# end line will be printed in handle_serial_input
|
2021-05-14 11:20:38 +02:00
|
|
|
else:
|
2022-02-17 15:27:34 +08:00
|
|
|
coredump = esp_coredump.CoreDump(core=coredump_file.name, core_format='b64', prog=self.elf_file)
|
|
|
|
f = io.StringIO()
|
|
|
|
with redirect_stdout(f):
|
|
|
|
coredump.info_corefile()
|
|
|
|
output = f.getvalue()
|
2021-05-14 11:20:38 +02:00
|
|
|
self.logger.output_enabled = True
|
2022-02-17 15:27:34 +08:00
|
|
|
self.logger.print(output.encode('utf-8'))
|
2021-05-14 11:20:38 +02:00
|
|
|
self.logger.output_enabled = False # Will be reenabled in check_coredump_trigger_after_print
|
2022-02-17 15:27:34 +08:00
|
|
|
if coredump_file is not None:
|
|
|
|
try:
|
|
|
|
os.unlink(coredump_file.name)
|
|
|
|
except OSError as e:
|
|
|
|
yellow_print('Couldn\'t remote temporary core dump file ({})'.format(e))
|
2021-05-14 11:20:38 +02:00
|
|
|
|
|
|
|
def _check_coredump_trigger_before_print(self, line): # type: (bytes) -> None
|
|
|
|
if self._decode_coredumps == COREDUMP_DECODE_DISABLE:
|
|
|
|
return
|
|
|
|
if COREDUMP_UART_PROMPT in line:
|
|
|
|
yellow_print('Initiating core dump!')
|
|
|
|
self.event_queue.put((TAG_KEY, '\n'))
|
|
|
|
return
|
|
|
|
if COREDUMP_UART_START in line:
|
|
|
|
yellow_print('Core dump started (further output muted)')
|
|
|
|
self._reading_coredump = COREDUMP_READING
|
|
|
|
self._coredump_buffer = b''
|
|
|
|
self.logger.output_enabled = False
|
|
|
|
return
|
|
|
|
if COREDUMP_UART_END in line:
|
|
|
|
self._reading_coredump = COREDUMP_DONE
|
|
|
|
yellow_print('\nCore dump finished!')
|
|
|
|
self._process_coredump()
|
|
|
|
return
|
|
|
|
if self._reading_coredump == COREDUMP_READING:
|
|
|
|
kb = 1024
|
|
|
|
buffer_len_kb = len(self._coredump_buffer) // kb
|
|
|
|
self._coredump_buffer += line.replace(b'\r', b'') + b'\n'
|
|
|
|
new_buffer_len_kb = len(self._coredump_buffer) // kb
|
|
|
|
if new_buffer_len_kb > buffer_len_kb:
|
|
|
|
yellow_print('Received %3d kB...' % new_buffer_len_kb, newline='\r')
|
|
|
|
|
|
|
|
def _check_coredump_trigger_after_print(self): # type: () -> None
|
|
|
|
if self._decode_coredumps == COREDUMP_DECODE_DISABLE:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Re-enable output after the last line of core dump has been consumed
|
|
|
|
if not self.logger.output_enabled and self._reading_coredump == COREDUMP_DONE:
|
|
|
|
self._reading_coredump = COREDUMP_IDLE
|
|
|
|
self.logger.output_enabled = True
|
|
|
|
self._coredump_buffer = b''
|
|
|
|
|
|
|
|
@contextmanager
|
|
|
|
def check(self, line): # type: (bytes) -> Generator
|
|
|
|
self._check_coredump_trigger_before_print(line)
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
finally:
|
|
|
|
self._check_coredump_trigger_after_print()
|