2024-03-28 11:32:18 +00:00
|
|
|
# pylint: disable=line-too-long,invalid-name,multiple-statements,missing-function-docstring,missing-class-docstring,missing-module-docstring,no-else-return,too-few-public-methods,unused-argument
|
|
|
|
# DESCRIPTION: Verilator: GDB startup file with useful define
|
|
|
|
#
|
|
|
|
# Copyright 2023 by Wilson Snyder. This program is free software; you
|
|
|
|
# can redistribute it and/or modify the Verilator internals under the terms
|
|
|
|
# of either the GNU Lesser General Public License Version 3 or the Perl
|
|
|
|
# Artistic License Version 2.0.
|
|
|
|
#
|
|
|
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
import tempfile
|
|
|
|
import gdb # pylint: disable=import-error
|
|
|
|
|
|
|
|
|
|
|
|
def _vltgdb_get_dump(node):
|
|
|
|
gdb.execute(f'set $_gdb_dump_json_str = AstNode::dumpTreeJsonGdb({node})')
|
|
|
|
dump = gdb.execute('printf "%s", $_gdb_dump_json_str', to_string=True)
|
|
|
|
gdb.execute('call free($_gdb_dump_json_str)')
|
|
|
|
return dump
|
|
|
|
|
|
|
|
|
|
|
|
def _vltgdb_tmpfile():
|
|
|
|
return tempfile.NamedTemporaryFile(mode="wt") # write, text mode
|
|
|
|
|
|
|
|
|
|
|
|
def _vltgdb_fwrite(file, s):
|
|
|
|
"""Write to file and flush buffer before passing the file to astsee"""
|
|
|
|
file.write(s)
|
|
|
|
file.flush()
|
|
|
|
|
|
|
|
|
|
|
|
class AstseeCmd(gdb.Command):
|
|
|
|
"""Verilator: Pretty print or diff nodes using `astsee`. A node can be:
|
|
|
|
* an pointer identifier,
|
|
|
|
* an address literal,
|
|
|
|
* a GDB value (like `$1`) that stores a dump previously done by the `jstash` command.
|
|
|
|
Apart from not taking input from a file, it works exactly like `verilator_astsee`:
|
|
|
|
* passing one node gives you a pretty print,
|
|
|
|
* passing two nodes gives you a diff,
|
|
|
|
* for more options see `astsee` readme/help.
|
|
|
|
For examples see internals.rst
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
super().__init__("jtree", gdb.COMMAND_USER, gdb.COMPLETE_EXPRESSION)
|
|
|
|
|
|
|
|
def _null_check(self, old, new):
|
|
|
|
err = ""
|
2024-08-27 01:43:34 +00:00
|
|
|
if old == "<nullptr>\n":
|
|
|
|
err += "old == <nullptr>\n"
|
|
|
|
if new == "<nullptr>\n":
|
|
|
|
err += "new == <nullptr>"
|
|
|
|
if err:
|
|
|
|
raise gdb.GdbError(err.strip("\n"))
|
2024-03-28 11:32:18 +00:00
|
|
|
|
|
|
|
def invoke(self, arg_str, from_tty):
|
|
|
|
from astsee import verilator_cli as astsee # pylint: disable=import-error,import-outside-toplevel
|
|
|
|
# we import it here so "no module X" error would occur on typing `jtree` rather than on every gdb start
|
|
|
|
|
|
|
|
# We hack `astsee_verilator`'s arg parser to find arguments with nodes
|
|
|
|
# After finding them, we replace them with proper files
|
|
|
|
astsee_args = astsee.parser.parse_args(gdb.string_to_argv(arg_str))
|
2024-08-27 01:43:34 +00:00
|
|
|
with _vltgdb_tmpfile() as oldfile, _vltgdb_tmpfile() as newfile, _vltgdb_tmpfile(
|
|
|
|
) as metafile:
|
2024-03-28 11:32:18 +00:00
|
|
|
if astsee_args.file:
|
|
|
|
_vltgdb_fwrite(oldfile, _vltgdb_get_dump(astsee_args.file))
|
|
|
|
astsee_args.file = oldfile.name
|
|
|
|
if astsee_args.newfile:
|
|
|
|
_vltgdb_fwrite(newfile, _vltgdb_get_dump(astsee_args.newfile))
|
|
|
|
astsee_args.newfile = newfile.name
|
|
|
|
if astsee_args.meta is None:
|
|
|
|
# pass
|
2024-08-27 01:43:34 +00:00
|
|
|
gdb.execute(f'call AstNode::dumpJsonMetaFileGdb("{metafile.name}")')
|
2024-03-28 11:32:18 +00:00
|
|
|
astsee_args.meta = metafile.name
|
|
|
|
try:
|
|
|
|
astsee.main(astsee_args)
|
|
|
|
except SystemExit as e: # astsee prints nice errmsgs on exit(), so rethrow it as GdbError to suppress cryptic python trace
|
|
|
|
raise gdb.GdbError("astsee exited with non-zero code") from e
|
|
|
|
|
|
|
|
|
|
|
|
AstseeCmd()
|