
Security News
NVD Quietly Sweeps 100K+ CVEs Into a “Deferred” Black Hole
NVD now marks all pre-2018 CVEs as "Deferred," signaling it will no longer enrich older vulnerabilities, further eroding trust in its data.
Documentation https://cs01.github.io/pygdbmi
Source Code https://github.com/cs01/pygdbmi
Python (py) gdb machine interface (mi)
GDB/MI is a line based machine oriented text interface to GDB and is activated by specifying using the --interpreter command line option (see Mode Options). It is specifically intended to support the development of systems which use the debugger as just one small component of a larger system.
To get machine interface output from gdb, run gdb with the --interpreter=mi2
flag like so:
gdb --interpreter=mi2
pip install pygdbmi
Cross platform support for Linux, macOS and Windows
Linux/Unix
Ubuntu 14.04 and 16.04 have been tested to work. Other versions likely work as well.
macOS
Note: the error please check gdb is codesigned - see taskgated(8)
can be fixed by codesigning gdb with these instructions. If the error is not fixed, please create an issue in github.
Windows
Windows 10 has been tested to work with MinGW and cygwin.
gdb mi defines a syntax for its output that is suitable for machine readability and scripting: example output:
-> -break-insert main
<- ^done,bkpt={number="1",type="breakpoint",disp="keep",
enabled="y",addr="0x08048564",func="main",file="myprog.c",
fullname="/home/myprog.c",line="68",thread-groups=["i1"],
times="0"}
<- (gdb)
Use pygdbmi.gdbmiparser.parse_response
to turn that string output into a JSON serializable dictionary
from pygdbmi import gdbmiparser
from pprint import pprint
response = gdbmiparser.parse_response('^done,bkpt={number="1",type="breakpoint",disp="keep", enabled="y",addr="0x08048564",func="main",file="myprog.c",fullname="/home/myprog.c",line="68",thread-groups=["i1"],times="0"')
pprint(response)
pprint(response)
# Prints:
# {'message': 'done',
# 'payload': {'bkpt': {'addr': '0x08048564',
# 'disp': 'keep',
# 'enabled': 'y',
# 'file': 'myprog.c',
# 'fullname': '/home/myprog.c',
# 'func': 'main',
# 'line': '68',
# 'number': '1',
# 'thread-groups': ['i1'],
# 'times': '0',
# 'type': 'breakpoint'}},
# 'token': None,
# 'type': 'result'}
But how do you get the gdb output into Python in the first place? If you want, pygdbmi
also has a class to control gdb as subprocess. You can write commands, and get structured output back:
from pygdbmi.gdbcontroller import GdbController
from pprint import pprint
# Start gdb process
gdbmi = GdbController()
print(gdbmi.command) # print actual command run as subprocess
# Load binary a.out and get structured response
response = gdbmi.write('-file-exec-file a.out')
pprint(response)
# Prints:
# [{'message': 'thread-group-added',
# 'payload': {'id': 'i1'},
# 'stream': 'stdout',
# 'token': None,
# 'type': 'notify'},
# {'message': 'done',
# 'payload': None,
# 'stream': 'stdout',
# 'token': None,
# 'type': 'result'}]
Now do whatever you want with gdb. All gdb commands, as well as gdb machine interface commands are acceptable. gdb mi commands give better structured output that is machine readable, rather than gdb console output. mi commands begin with a -
.
response = gdbmi.write('-break-insert main') # machine interface (MI) commands start with a '-'
response = gdbmi.write('break main') # normal gdb commands work too, but the return value is slightly different
response = gdbmi.write('-exec-run')
response = gdbmi.write('run')
response = gdbmi.write('-exec-next', timeout_sec=0.1) # the wait time can be modified from the default of 1 second
response = gdbmi.write('next')
response = gdbmi.write('next', raise_error_on_timeout=False)
response = gdbmi.write('next', raise_error_on_timeout=True, timeout_sec=0.01)
response = gdbmi.write('-exec-continue')
response = gdbmi.send_signal_to_gdb('SIGKILL') # name of signal is okay
response = gdbmi.send_signal_to_gdb(2) # value of signal is okay too
response = gdbmi.interrupt_gdb() # sends SIGINT to gdb
response = gdbmi.write('continue')
response = gdbmi.exit()
Each parsed gdb response consists of a list of dictionaries. Each dictionary has keys message
, payload
, token
, and type
.
message
contains a textual message from gdb, which is not always present. When missing, this is None
.payload
contains the content of gdb's output, which can contain any of the following: dictionary
, list
, string
. This too is not always present, and can be None
depending on the response.token
If an input command was prefixed with a (optional) token then the corresponding output for that command will also be prefixed by that same token. This field is only present for pygdbmi output types nofity
and result
. When missing, this is None
.The type
is defined based on gdb's various mi output record types, and can be
result
- the result of a gdb command, such as done
, running
, error
, etc.notify
- additional async changes that have occurred, such as breakpoint modifiedconsole
- textual responses to cli commandslog
- debugging messages from gdb's internalsoutput
- output from targettarget
- output from remote targetdone
- when gdb has finished its outputDocumentation fixes, bug fixes, performance improvements, and functional improvements are welcome. You may want to create an issue before beginning work to make sure I am interested in merging it to the master branch.
pygdbmi uses nox for automation.
See available tasks with
nox -l
Run tests and lint with
nox -s tests
nox -s lint
Positional arguments passed to nox -s tests
are passed directly to pytest
. For instance, to run only the parse tests use
nox -s tests -- tests/test_gdbmiparser.py
See pytest
's documentation for more details on how to run tests.
To format code using the correct settings use
nox -s format
Or, to format only specified files, use
nox -s format -- example.py pygdbmi/IoManager.py
Only maintainers of the pygdbmi package on PyPi can make a release.
In the following steps, replace these strings with the correct values:
<REMOTE>
is the name of the remote for the main pygdbmi repository (for instance, origin
)<VERSION>
is the version number chosen in step 2.To make a release:
Checkout the master
branch and pull from the main repository with git pull <REMOTE> master
Decide the version number for the new release; we follow
Semantic Versioning but prefixing the version with 0.
: given a version
number 0.SECOND.THIRD.FOURTH, increment the:
Update CHANGELOG.md
to list the chosen version number instead of ## dev
Update __version__
in pygdbmi/__init__.py
to the chosen version number
Create a branch, for instance using git checkout -b before-release-<VERSION>
Commit your changes, for instance using git commit -a -m 'Bump version to <VERSION> for release'
Check that the docs look fine by serving them locally with nox -s serve_docs
Push the branch, for instance with git push --set-upstream <REMOTE> before-release-<VERSION>
If tests pass on the PR you created, you can merge into master
Go to the new release page and prepare the release:
v<VERSION>
(for example v0.1.2.3
)pygdbmi v<VERSION>
(for example pygdbmi v0.1.2.3
)CHANGELOG.md
excluding the line
with the version numberPublish the release to PyPI with nox -s publish
Publish the docs with nox -s publish_docs
Verify that the PyPi page for pygdbmi looks correct
Verify that the published docs look correct
Prepare for changes for the next release by adding something like this above the previous
entries in CHANGELOG.md
(where <VERSION+1>
is <VERSION>
with the last digit increaded
by 1):
## <VERSION+1>.dev0
- *Replace this line with new entries*
Create a branch for the changes with git checkout -b after-release-<VERSION>
Commit the change with git commit -m 'Prepare for work on the next release' CHANGELOG.md
Push the branch with git push --set-upstream <REMOTE> after-release-<VERSION>
If tests pass, merge into master
FAQs
Parse gdb machine interface output with Python
We found that pygdbmi demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 2 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
NVD now marks all pre-2018 CVEs as "Deferred," signaling it will no longer enrich older vulnerabilities, further eroding trust in its data.
Research
Security News
Lazarus-linked threat actors expand their npm malware campaign with new RAT loaders, hex obfuscation, and over 5,600 downloads across 11 packages.
Security News
Safari 18.4 adds support for Iterator Helpers and two other TC39 JavaScript features, bringing full cross-browser coverage to key parts of the ECMAScript spec.