15. Handle crashes and view snapshots#
When a system crashes it can be difficult to debug the cause of the crash. To help with this, pw_system provides a crash handler built on pw_cpu_exception which is invoked when a CPU exception is triggered.
The pw_system crash handler will automatically create a crash snapshot on exception and reboot the system. This snapshot can then be downloaded onto a host system for analysis.
Generate a crash#
If you’re not already connected to your Pico via
pw_console
, do so now:In Bazel Targets expand //apps/production, then right-click :rp2040_console (native binary), then select Run target.
In Bazel Targets expand //apps/production, then right-click :rp2350_console (native binary), then select Run target.
Run the terminal-based console:
bazelisk run //apps/production:rp2040_console
bazelisk run //apps/production:rp2350_console
pw_system
provides an RPC to crash the system by triggering a
HardFault.
To invoke this RPC, type the following into the Python Repl (bottom-left pane) of pw_console
:
>>> device.rpcs.pw.system.proto.DeviceService.Crash()
If you’re using the full setup, on system restart the presence of a crash snapshot will be detected, and the following will be in the system logs:
INF pw_system RpcDevice 00:00:00.000 pw_system main
ERR pw_system RpcDevice 00:00:00.000 ==========================
ERR pw_system RpcDevice 00:00:00.000 ======CRASH DETECTED======
ERR pw_system RpcDevice 00:00:00.000 ==========================
ERR pw_system RpcDevice 00:00:00.000 Crash snapshots available.
ERR pw_system RpcDevice 00:00:00.000 Run `device.get_crash_snapshots()` to download and clear the snapshots.
INF pw_system RpcDevice 00:00:00.000 System init
On a “Pico and Enviro+” or “Pico only”
setup, the snapshot will still be generated, but logs won’t be visible in the console. When
the system restarts, the connection to the console is broken so logs
will not be displayed. After invoking the Crash()
RPC, exit the console
and start it again to
re-establish the connection.
View a crash snapshot#
The crash snapshot contains relevant information to debug crashes, such as register state thread backtraces and un-flushed logs. If there is a crash snapshot on the system, it can be downloaded to the host with the following RPC.
>>> device.get_crash_snapshots()
This RPC will download the snapshot, decode it and save it in a temporary directory, the location of which will be printed to the console as follows:
INF Wrote crash snapshot to: /var/folders/2j/sjk9390d5rxc3c9ycwcf3mdh0103lh/T/crash_0.txt
It’s also possible to specify the path as part of the RPC call:
>>> device.get_crash_snapshots("/path/")
The decoded text file should look similar to this truncated example:
Device crash cause:
pw_system/device_service_pwpb.cc:38 Crash: RPC triggered crash
Reason token: 0x735f7770
CPU Arch: ARMV8M
Exception caused by a usage fault.
Active Crash Fault Status Register (CFSR) fields:
UNDEFINSTR Undefined Instruction UsageFault.
The processor has attempted to execute an undefined
instruction. When this bit is set to 1, the PC value stacked
for the exception return points to the undefined instruction.
An undefined instruction is an instruction that the processor
cannot decode.
All registers:
pc 0x10000f0a pw::system::DeviceServicePwpb::Crash(pw::system::proto::pwpb::CrashRequest::Message const&, pw::system::proto::pwpb::CrashResponse::Message&) (/b/pw_system/device_service_pwpb.cc:38)
lr 0x10012787 pw::StringBuilder::FormatVaList(char const*, std::__va_list) (/build/pw_string/string_builder.cc:102)
psr 0x41000000
msp 0x20081fe0 __scratch_y_end__ (??:?)
psp 0x2000a100 pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
exc_return 0xfffffffd
cfsr 0x00010000
msplim 0x00000000
psplim 0x20002288
mmfar 0xe000ed34 __scratch_y_end__ (??:?)
bfar 0xe000ed38 __scratch_y_end__ (??:?)
icsr 0x00400806
hfsr 0x00000000
shcsr 0x00070008
control 0x00000000
r0 0x2000a0e0 pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r1 0x0000003e pw_assert_basic_HandleFailure (/b/pw_assert_basic/basic_handler.cc:74)
r2 0x0000002b pw_assert_basic_HandleFailure (/b/pw_assert_basic/basic_handler.cc:74)
r3 0x2000a100 pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r4 0x10019596
r5 0x2000a178 pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r6 0x10019eec pw::system::proto::pw_rpc::pwpb::DeviceService::Service<pw::system::DeviceServicePwpb>::kPwRpcMethods (??:?)
r7 0x2000a108 pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r8 0x2000a118 pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r9 0x2000a16e pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r10 0x2000b4f0 pw::system::(anonymous namespace)::server (hdlc_rpc_server.cc:0)
r11 0x2000a22c pw::system::rpc_thread_context (freertos_target_hooks.cc:0)
r12 0x00000008 pw_assert_HandleFailure (/b/pw_assert_basic/assert_basic.cc:20)
Thread State
6 threads running, RpcThread active at the time of capture.
~~~~~~~~~
Thread (RUNNING): RpcThread <-- [ACTIVE]
Est CPU usage: unknown
Stack info
Current usage: 0x2000a288 - 0x2000a100 (392 bytes, 1.20%)
Est peak usage: 944 bytes, 2.88%
Stack limits: 0x2000a288 - 0x2000228c (32764 bytes)
Stack Trace (most recent call first):
1: at void pw::rpc::internal::PwpbMethod::CallSynchronousUnary<pw::system::proto::pwpb::RebootRequest::Message, pw::system::proto::pwpb::RebootResponse::Message>(pw::rpc::internal::CallContext const&, pw::rpc::internal::Packet const&, pw::system::proto::pwpb::RebootRequest::Message&, pw::system::proto::pwpb::RebootResponse::Message&) const (0x10000F59)
in /build/pw_rpc/pwpb/public/pw_rpc/pwpb/internal/method.h:258
2: at void pw::rpc::internal::PwpbMethod::CallSynchronousUnary<pw::system::proto::pwpb::CrashRequest::Message, pw::system::proto::pwpb::CrashResponse::Message>(pw::rpc::internal::CallContext const&, pw::rpc::internal::Packet const&, pw::system::proto::pwpb::CrashRequest::Message&, pw::system::proto::pwpb::CrashResponse::Message&) const (0x10001137)
in /build/pw_rpc/pwpb/public/pw_rpc/pwpb/internal/method.h:267
3: at xQueueSemaphoreTake (0x10013049)
in /build/external/freertos+/queue.c:1555
4: at void pw::rpc::internal::PwpbMethod::SynchronousUnaryInvoker<pw::system::proto::pwpb::CrashRequest::Message, pw::system::proto::pwpb::CrashResponse::Message>(pw::rpc::internal::CallContext const&, pw::rpc::internal::Packet const&) (0x10000F4F)
in /build/pw_rpc/pwpb/public/pw_rpc/pwpb/internal/method.h:322
5: at pw::rpc::Server::ProcessPacket(pw::rpc::internal::Packet) (0x1000EA9D)
in /build/pw_rpc/public/pw_rpc/internal/method.h:0
6: at pw::rpc::Server::ProcessPacket(pw::span<std::byte const, 4294967295u>) (0x1000E9CD)
in /build/pw_rpc/server.cc:40
7: at pw::system::RpcDispatchThread::Run() (0x10008625)
in /build/pw_system/hdlc_rpc_server.cc:127
8: at pw::thread::freertos::Context::ThreadEntryPoint(void*) (0x1000EFA5)
in /build/third_party/fuchsia/repo/sdk/lib/fit/include/lib/fit/internal/function.h:362
9: at prvTaskExitError (0x100137C9)
in /build/external/freertos+/portable/GCC/ARM_CM33_NTZ/non_secure/port.c:634
...
Device Logs:
[RpcDevice] pw_system 0 pw_system main targets/rp2040/boot.cc:56
[RpcDevice] pw_system 0 System init pw_system/init.cc:65
[RpcDevice] pw_system 0 Registering RPC services pw_system/init.cc:75
...
Summary#
On this page, we met pw_cpu_exception
, the CPU exception handler entry point.
We also learned how to generate crashes and download the resulting crash snapshot.
Next, head over to 16. Wrapping up to wrap up your tour of Pigweed.