Module Usage#
Right now, pw_snapshot just dictates a format. That means there is no provided system information collection integration, underlying storage, or transport mechanism to fetch a snapshot from a device. These must be set up independently by your project.
Building a Snapshot#
Even though a Snapshot is just a proto message, the potential size of the proto makes it important to consider the encoder.
Nanopb is a popular encoder for embedded devices, itβs impractical to use with the pw_snapshot proto. Nanopb works by generating in-memory structs that represent the protobuf message. Repeated, optional, and variable-length fields increase the size of the in-memory struct. The struct representation of snapshot-like protos can quickly near 10KB in size. Allocating 10KB
Pigweedβs pw_protobuf is a better choice as its design is centered around incrementally writing a proto directly to the final wire format. If you only write a few fields in a snapshot, you can do so with minimal memory overhead.
#include "pw_bytes/span.h"
#include "pw_protobuf/encoder.h"
#include "pw_snapshot_protos/snapshot.pwpb.h"
#include "pw_status/status.h"
#include "pw_stream/stream.h"
pw::Status EncodeSnapshot(pw::stream::Writer& writer,
pw::ByteSpan submessage_encode_buffer,
const CrashInfo &crash_info) {
// Create a snapshot proto encoder.
pw::snapshot::Snapshot::StreamEncoder snapshot_encoder(
writer, submessage_encode_buffer);
{ // This scope is required to handle RAII behavior of the submessage.
// Start writing the Metadata submessage.
pw::snapshot::Metadata::StreamEncoder metadata_encoder =
snapshot_encoder.GetMetadataEncoder();
metadata_encoder.WriteReason(EncodeReasonLog(crash_info));
metadata_encoder.WriteFatal(true);
metadata_encoder.WriteProjectName(pw::as_bytes(pw::span("smart-shoe")));
metadata_encoder.WriteDeviceName(
pw::as_bytes(pw::span("smart-shoe-p1")));
}
return proto_encoder.status();
}
Custom Project Data#
There are two main ways to add custom project-specific data to a snapshot. Tags are the simplest way to capture small snippets of information that require no or minimal post-processing. For more complex data, itβs usually more practical to extend the Snapshot proto.
Extending the Proto#
Extending the Snapshot proto relies on proto behavior details that are explained in the Snapshot Proto Format. Extending the snapshot proto is as simple as defining a proto message that only declares fields with numbers that are reserved by the Snapshot proto for downstream projects. When encoding your snapshot, you can then write both the upstream Snapshot proto and your projectβs custom extension proto message to the same proto encoder.
The upstream snapshot tooling will ignore any project-specific proto data, the proto data can be decoded a second time using a project-specific proto. At that point, any handling logic of the project-specific data would have to be done as part of project-specific tooling.
Analyzing Snapshots#
Snapshots can be processed for analysis using the pw_snapshot.process
python
tool. This tool turns a binary snapshot proto into human readable, actionable
information. As some snapshot fields may optionally be tokenized, a
pw_tokenizer database or ELF file with embedded pw_tokenizer tokens may
optionally be passed to the tool to detokenize applicable fields.
# Example invocation, which dumps to stdout by default.
$ python -m pw_snapshot.processor path/to/serialized_snapshot.bin
____ _ __ _____ _ _____ ____ _____ __ ______ ______
/ __ \ | / / / ___// | / / | / __ \/ ___// / / / __ \/_ __/
/ /_/ / | /| / / \__ \/ |/ / /| | / /_/ /\__ \/ /_/ / / / / / /
/ ____/| |/ |/ / ___/ / /| / ___ |/ ____/___/ / __ / /_/ / / /
/_/ |__/|__/____/____/_/ |_/_/ |_/_/ /____/_/ /_/\____/ /_/
/_____/
βͺβββ βββΒ· βββββ βββΒ· β Β·
ββββββ ββ β’ ββ ββ ββ β
β βͺ βββββ β. βββββ β
ββ .ββ βͺββ βͺββΒ·ββ βͺββββ
β β β Β· β β β .ββ
Device crash cause:
../examples/example_rpc.cc: Assert failed: 1+1 == 42
Project name: gShoe
Device: GSHOE-QUANTUM_CORE-REV_0.1
Device FW version: QUANTUM_CORE-0.1.325-e4a84b1a
FW build UUID: ad2d39258c1bc487f07ca7e04991a836fdf7d0a0
Snapshot UUID: 8481bb12a162164f5c74855f6d94ea1a
Thread State
2 threads running, Main Stack (Handler Mode) active at the time of capture.
~~~~~~~~~~~~~~~~~~~~~~~~~
Thread (INTERRUPT_HANDLER): Main Stack (Handler Mode) <-- [ACTIVE]
Est CPU usage: unknown
Stack info
Stack used: 0x2001b000 - 0x2001ae20 (480 bytes)
Stack limits: 0x2001b000 - 0x???????? (size unknown)
Raw Stack
00caadde
Thread (RUNNING): Idle
Est CPU usage: unknown
Stack info
Stack used: 0x2001ac00 - 0x2001ab0c (244 bytes, 47.66%)
Stack limits: 0x2001ac00 - 0x2001aa00 (512 bytes)
Symbolizing Addresses#
The snapshot processor tool has built-in support for symbolization of some data
embedded into snapshots. Taking advantage of this requires the use of a
project-provided SymbolizerMatcher
callback. This is used by the snapshot
processor to understand which ELF file should be used to symbolize which
snapshot in cases where a snapshot has related snapshots embedded inside of it.
Hereβs an example implementation that uses the device name:
# Given a firmware bundle directory, determine the ELF file associated with
# the provided snapshot.
def _snapshot_symbolizer_matcher(fw_bundle_dir: Path,
snapshot: snapshot_pb2.Snapshot
) -> Symbolizer:
metadata = MetadataProcessor(snapshot.metadata, DETOKENIZER)
if metadata.device_name().startswith('GSHOE_MAIN_CORE'):
return LlvmSymbolizer(fw_bundle_dir / 'main.elf')
if metadata.device_name().startswith('GSHOE_SENSOR_CORE'):
return LlvmSymbolizer(fw_bundle_dir / 'sensors.elf')
return LlvmSymbolizer()
# A project specific wrapper to decode snapshots that provides a detokenizer
# and ElfMatcher.
def decode_snapshots(snapshot: bytes, fw_bundle_dir: Path) -> str:
# This is the actual ElfMatcher, which wraps the helper in a lambda that
# captures the passed firmware artifacts directory.
matcher: processor.SymbolizerMatcher = (
lambda snapshot: _snapshot_symbolizer_matcher(
fw_bundle_dir, snapshot))
return processor.process_snapshots(snapshot, DETOKENIZER, matcher)
C++ Utilities#
UUID utilities#
Snapshot UUIDs are used to uniquely identify snapshots. Pigweed strongly
recommends using randomly generated data as a snapshot UUID. The
more entropy and random bits, the lower the probability that two devices will
produce the same UUID for a snapshot. 16 bytes should be sufficient for most
projects, so this module provides UuidSpan
and ConstUuidSpan
types that
can be helpful for referring to UUID-sized byte spans.
Reading a snapshotβs UUID#
An in-memory snapshotβs UUID may be read using ReadUuidFromSnapshot()
.
void NotifyNewSnapshot(ConstByteSpan snapshot) {
std::array<std::byte, pw::snapshot::kUuidSizeBytes> uuid;
pw::Result<pw::ConstByteSpan> result =
pw::snapshot::ReadUuidFromSnapshot(snapshot, uuid);
if (!result.ok()) {
PW_LOG_ERROR("Failed to read UUID from new snapshot, error code %d",
static_cast<int>(result.status().code()));
return;
}
LogNewSnapshotUuid(result.value());
}