Snapshots
Snapshots are a static type of save state in libdebug. They allow you to save the current state of the process in terms of registers, memory, and other process properties. Snapshots can be saved to disk as a file and loaded for future use. Finally, snapshots can be diffed to compare the differences between the state of the process at two different moments or executions.
Snapshots are static
Snapshots are static in the sense that they capture the state of the process at a single moment in time. They can be loaded and inspected at any time and across different architectures. They do not, however, allow to restore their state to the process.
There are three available levels of snapshots in libdebug, which differ in the amount of information they store:
Level | Registers | Memory Pages | Memory Contents |
---|---|---|---|
base |
|||
writable |
writable pages only | ||
full |
Since memory content snapshots can be large, the default level is base
.
You can create snapshots of single threads or the entire process.
API
-
Register Access
You can access a snapshot's registers using the
regs
attribute, just like you would when debugging the process. -
Memory Access
When the snapshot level is appropriate, you can access the memory of the process using the
memory
attribute. -
Memory Maps
Memory maps are always available. When the snapshot level is appropriate, you can access the contents as a bytes-like object.
-
Stack Trace
When the snapshot level is appropriate, you can access the backtrace of the process or thread.
Creating Snapshots
The function used to create a snapshot is create_snapshot()
. It behaves differently depending on the object it is called from.
Calling Object | Snapshot Type | Description |
---|---|---|
ThreadContext | ThreadSnapshot | Creates a snapshot of the specific thread. |
Debugger | ProcessSnapshot | Creates a snapshot of the entire process. This includes snapshots for all threads. |
The following is the signature of the function:
The following is an example usage of the function in both cases:
d = debugger("program")
my_thread = d.threads[1]
# Thread Snapshot
ts = my_thread.create_snapshot(level="full", name="cool snapshot") #(1)!
# Process Snapshot
ps = d.create_snapshot(level="writable", name="very cool snapshot") #(2)!
- This will create a full-level snapshot of the thread
my_thread
and name it "cool snapshot". - This will create a writable-level snapshot of the entire process and name it "very cool snapshot".
Naming Snapshots
When creating a snapshot, you can optionally specify a name for it. The name will be useful when comparing snapshots in diffs or when saving them to disk.
Saving and Loading Snapshots
You can save a snapshot to disk using the save()
method of the Snapshot object. The method will create a serializable version of the snapshot and export a json file to the specified path.
You can load a snapshot from disk using the load_snapshot()
method of the Debugger object. The method will read the json file from the specified path and create a Snapshot object from it.
The snapshot type will be inferred from the json file, so you can easily load both thread and process snapshots from the same method.
Resolving Diffs
Thanks to their static nature, snapshots can be easily compared to find differences in saved properties.
You can diff a snapshot against another using the diff()
method. The method will return a Diff object that represents the differences between the two snapshots. The diff will be of the lowest level of the two snapshots being compared in terms.
Example usage
ts1 = d.threads[1].create_snapshot(level="full")
[...] # (1)!
ts2 = d.threads[1].create_snapshot(level="full")
ts_diff = ts1.diff(ts2) # (2)!
- Do some operations that change the state of the process.
- Compute the diff between the two snapshots
Diffs have a rich and detailed API that allows you to inspect the differences in registers, memory, and other properties. Read more in the dedicated section.
Pretty Printing
Pretty Printing is a feature of some libdebug objects that allows you to print the contents of a snapshot in a colorful and eye-catching format. This is useful when you want to inspect the state of the process at a glance.
Pretty printing utilities of snapshots are "mirrors" of pretty pretting functions available for the Debugger and ThreadContext. Here is a list of available pretty printing functions and their equivalent for the running process:
Function | Description | Reference |
---|---|---|
pprint_registers() |
Prints the general-purpose registers of the snapshot. | API Reference |
pprint_registers_all() |
Prints all registers of the snapshot. | API Reference |
pprint_maps() |
Prints the memory of the snapshot. | API Reference |
pprint_backtrace() |
Prints the backtrace of the snapshot. | API Reference |
Attributes
Attribute | Data Type | Level | Description | Aliases |
---|---|---|---|---|
Common | ||||
name |
str (optional) |
All | The name of the snapshot. | |
arch |
str |
All | The ISA under which the snapshot process was running. | |
snapshot_id |
int | All | Progressive id counted from 0. Process and Thread snapshots have separate counters. | |
level |
str |
All | The snapshot level. | |
maps |
MemoryMapSnapshotList |
All | The memory maps of the process. Each map will also have the contents of the memory map under the appropriate snapshot level. | |
memory |
SnapshotMemoryView |
writable / full |
Interface to the memory of the process. | mem |
aslr_enabled |
bool |
All | Whether ASLR was enabled at the time of the snapshot. | |
Thread Snapshot | ||||
thread_id |
int |
All | The ID of the thread the snapshot was taken from. | tid |
regs |
SnapshotRegisters |
All | The register values of the thread. | registers |
Process Snapshot | ||||
process_id |
int |
All | The ID of the process the snapshot was taken from. | pid |
threads |
list[LightweightThreadSnapshot] |
All | Snapshots of all threads of the process. | |
regs |
SnapshotRegisters |
All | The register values of the main thread of the process. | registers |