Discover Embedded Debugging Techniques¶
This tutorial describes the available tools provided to developers to debug an application. It also presents debugging methods applied to two concrete uses cases (GUI application freeze and HardFault).
Intended Audience¶
The audience for this document is engineers who want to learn about the tools available to debug embedded applications.
In addition, the Use Case 1: Debugging a GUI Application Freeze is particularly relevant for Application engineers. Whereas the Use Case 2: Debugging a HardFault is more relevant for Firmware engineers.
Debugging Tools¶
This section presents an overview of the main tools available to debug an embedded application. Please refer to the developer guides for the complete reference (Application Developer Guide, VEE Porting Guide, Kernel Developer Guide).
- Events Tracing and Logging
- Runtime State Dump
- Memory Inspection
- Platform Qualification
- Simulator Debugger
- Static Analysis Tools
- GUI Debugging Tools
Events Tracing and Logging¶
When an application has issues, the first step is to understand what is happening inside the system.
The Trace Library is a real-time event recording library. Use it to trace the beginning and the ending of events.
private static final int EVENT_ID = 0; public static void switchState(ApplicationState newState) { tracer.recordEvent(EVENT_ID); previousState = currentState; currentState = newState; tracer.recordEventEnd(EVENT_ID); }
This API is most useful with the SystemView Event tracer to visualize the timeline of events.
The Message Library is a small RAM/ROM/CPU footprint API to log errors, warnings, and misc information.
private static final String LOG_CATEGORY = "Application"; private static final int LOG_ID = 2; public static void switchState(ApplicationState newState) { previousState = currentState; currentState = newState; BasicMessageLogger.INSTANCE.log(Level.INFO, LOG_CATEGORY, LOG_ID, previousState, currentState); }
The Logging Library implements a subset of the standard Java
java.util.logging
.public static void switchState(ApplicationState newState) { previousState = currentState; currentState = newState; Logger logger = Logger.getLogger(Main.class.getName()); logger.log(Level.INFO, "The application state has changed from " + previousState.toString() + " to " + currentState.toString() + "."); }
Please refer to the Code Instrumentation for Logging documentation for a comparison of these libraries.
Runtime State Dump¶
Output information on the standard output System.out and use the Stack Trace Reader to read and decode the MicroEJ stack traces.
The Core Engine VM dump is a low-level API to display the state of the MicroEJ Runtime and the MicroEJ threads (name, priority, stack trace, etc. )
=================================== VM Dump ==================================== Java threads count: 3 Peak java threads count: 3 Total created java threads: 3 Last executed native function: 0x90035E3D Last executed external hook function: 0x00000000 State: running -------------------------------------------------------------------------------- Java Thread[1026] name="main" prio=5 state=RUNNING max_java_stack=456 current_java_stack=184 java.lang.MainThread@0xC0083C7C: at (native) [0x90003F65] at com.microej.demo.widget.main.MainPage.getContentWidget(MainPage.java:95) Object References: - com.microej.demo.widget.main.MainPage@0xC00834E0 - com.microej.demo.widget.main.MainPage$1@0xC0082184 - java.lang.Thread@0xC0082194 - java.lang.Thread@0xC0082194 at com.microej.demo.widget.common.Navigation.createRootWidget(Navigation.java:104) Object References: - com.microej.demo.widget.main.MainPage@0xC00834E0 at com.microej.demo.widget.common.Navigation.createDesktop(Navigation.java:88) Object References: - com.microej.demo.widget.main.MainPage@0xC00834E0 - ej.mwt.stylesheet.CachedStylesheet@0xC00821DC at com.microej.demo.widget.common.Navigation.main(Navigation.java:40) Object References: - com.microej.demo.widget.main.MainPage@0xC00834E0 at java.lang.MainThread.run(Thread.java:855) Object References: - java.lang.MainThread@0xC0083C7C at java.lang.Thread.runWrapper(Thread.java:464) Object References: - java.lang.MainThread@0xC0083C7C at java.lang.Thread.callWrapper(Thread.java:449) -------------------------------------------------------------------------------- Java Thread[1281] name="UIPump" prio=5 state=WAITING timeout(ms)=INF max_java_stack=120 current_java_stack=117 external event: status=waiting java.lang.Thread@0xC0083628: at ej.microui.MicroUIPump.read(Unknown Source) Object References: - ej.microui.display.DisplayPump@0xC0083640 at ej.microui.MicroUIPump.run(MicroUIPump.java:176) Object References: - ej.microui.display.DisplayPump@0xC0083640 at java.lang.Thread.run(Thread.java:311) Object References: - java.lang.Thread@0xC0083628 at java.lang.Thread.runWrapper(Thread.java:464) Object References: - java.lang.Thread@0xC0083628 at java.lang.Thread.callWrapper(Thread.java:449) -------------------------------------------------------------------------------- Java Thread[1536] name="Thread1" prio=5 state=READY max_java_stack=60 current_java_stack=57 java.lang.Thread@0xC0082194: at java.lang.Thread.runWrapper(Unknown Source) Object References: - java.lang.Thread@0xC0082194 at java.lang.Thread.callWrapper(Thread.java:449) ================================================================================ ============================== Garbage Collector =============================== State: Stopped Last analyzed object: null Total memory: 15500 Current allocated memory: 7068 Current free memory: 8432 Allocated memory after last GC: 0 Free memory after last GC: 15500 ================================================================================ =============================== Native Resources =============================== Id CloseFunc Owner Description -------------------------------------------------------------------------------- ================================================================================
Memory Inspection¶
Memory issues such as memory corruption and memory leaks can be hard to troubleshoot. However, the following tools are available to address these issues:
Check the internal structure integrity of the MicroJvm virtual machine with the LLMJVM_checkIntegrity API to detect memory corruptions in native functions.
Use the Heap Usage Monitoring Tool to estimate the heap requirements of an application.
The Heap Dumper & Heap Analyzer tools analyze the content of the heap. They are helpful to detect memory leaks and look for optimization of the heap usage.
Platform Qualification¶
The Platform Qualification Tools (PQT) project provides the tools required to validate each component of a MicroEJ Platform. After porting or adding a feature to a MicroEJ Platform, it is necessary to validate its integration.
The project is available on GitHub: https://github.com/MicroEJ/VEEPortQualificationTools
Please refer to the VEE Port Qualification documentation for more information.
Simulator Debugger¶
Debug an Application on Simulator, add breakpoints, inspect stack frame, use step-by-step, etc.
Configure the libraries’ sources location to View library as sources in the debugger.
Static Analysis Tools¶
Static analysis tools are helpful allies to prevent several classes of bugs.
SonarQube™ provides reports on duplicated code, coding standards, unit tests, code coverage, code complexity, potential bugs, comments, and architecture.
Use the Null Analysis tool to detect and prevent NullPointerException, one of the most common causes of runtime failure of Java programs.
GUI Debugging Tools¶
The Widget Library provides several Debug Utilities to investigate and troubleshoot GUI applications. For example, it is possible to print the type and bounds of each widget in the hierarchy of a widget:
Scroll: 0,0 480x272 (absolute: 0,0) +--ScrollableList: 0,0 480x272 (absolute: 0,0) | +--Label: 0,0 480x50 (absolute: 0,0) | +--Dock: 0,50 480x50 (absolute: 0,50) | | +--ImageWidget: 0,0 70x50 (absolute: 0,50) | | +--Label: 70,0 202x50 (absolute: 70,50) | +--Label: 0,100 480x50 (absolute: 0,100)
MicroUI Event Buffer provides an API to store and dump the events received:
============================== MicroUI FIFO Dump =============================== ---------------------------------- Old Events ---------------------------------- [27: 0x00000000] garbage [28: 0x00000000] garbage [...] [99: 0x00000000] garbage [00: 0x08000000] Display SHOW Displayable (Displayable index = 0) [01: 0x00000008] Command HELP (event generator 0) [02: 0x0d000000] Display REPAINT Displayable (Displayable index = 0) [03: 0x07030000] Input event: Pointer pressed (event generator 3) [04: 0x009f0063] at 159,99 (absolute) [05: 0x07030600] Input event: Pointer moved (event generator 3) [06: 0x00aa0064] at 170,100 (absolute) [07: 0x02030700] Pointer dragged (event generator 3) [08: 0x0d000000] Display REPAINT Displayable (Displayable index = 0) [09: 0x07030600] Input event: Pointer moved (event generator 3) [10: 0x00b30066] at 179,102 (absolute) [11: 0x02030700] Pointer dragged (event generator 3) [12: 0x0d000000] Display REPAINT Displayable (Displayable index = 0) [13: 0x07030600] Input event: Pointer moved (event generator 3) [14: 0x00c50067] at 197,103 (absolute) [15: 0x02030700] Pointer dragged (event generator 3) [16: 0x0d000000] Display REPAINT Displayable (Displayable index = 0) [17: 0x07030600] Input event: Pointer moved (event generator 3) [18: 0x00d00066] at 208,102 (absolute) [19: 0x02030700] Pointer dragged (event generator 3) [20: 0x0d000000] Display REPAINT Displayable (Displayable index = 0) [21: 0x07030100] Input event: Pointer released (event generator 3) [22: 0x00000000] at 0,0 (absolute) [23: 0x00000008] Command HELP (event generator 0) ---------------------------------- New Events ---------------------------------- [24: 0x0d000000] Display REPAINT Displayable (Displayable index = 0) [25: 0x07030000] Input event: Pointer pressed (event generator 3) [26: 0x002a0029] at 42,41 (absolute) --------------------------- New Events' Java objects --------------------------- [java/lang/Object[2]@0xC000FD1C [0] com/microej/examples/microui/mvc/MVCDisplayable@0xC000BAC0 [1] null ================================================================================
MicroUI can log several actions, which can be viewed in SystemView. Please refer to Debug Traces for more information.
Make sure to understand MWT Concepts, especially the relations between the rendering, the lay-out the event dispatch and the states of desktop and widget.
For UI2 and former versions, please refer to MicroUI and multithreading for a description of the threading model.
Use Case 1: Debugging a GUI Application Freeze¶
When an application User Interface freezes and becomes unresponsive, in most cases, one of the following conditions applies:
- An unrecoverable system failure occurred, like a HardFault, and the RTOS tasks are not scheduled anymore.
- The RTOS task that runs the Core Engine is never given CPU time (suspended or blocked).
- The RTOS task that runs the Core Engine is executing never-ending native code (infinite loop in native implementation for example).
- A Java method executes a long-running operation in the MicroUI thread (also called Display Pump thread).
- The application code is unable to receive or process user input events.
The following sections explain how to instrument the code to locate the issue when the UI freeze occurs. The steps followed are:
- Check if the RTOS properly schedules the Core Engine task.
- Check if the Core Engine properly schedules all Java threads.
- Check if the Core Engine properly schedules the MicroUI thread.
- Check if Input Events are properly processed.
Note
- The checks of the schedulers are possible with SystemView and MicroUI Debug Traces.
- The Input Events check is possible with the LLUI_INPUT_dump API.
Check RTOS Tasks Scheduling¶
Let’s start at low level by figuring out if the RTOS is scheduling tasks correctly. If possible, use a debugger or SystemView; if not, use the heartbeat task described below.
The following flow chart summarizes the investigation steps with a heartbeat task:
(1) Make one of the RTOS tasks acts like a heartbeat: create a dedicated task and make it report in some way at a regular pace (print a message on standard output, blink an LED, use SystemView, etc.). Set the heartbeat task priority to the same priority as the Core Engine task.
(2) In this configuration, if the heartbeat is still running when the UI freeze occurs, we can go a step further and check whether the Core Engine is still scheduling Java threads or not. See next section Check Java Threads Scheduling.
(3) If the heartbeat doesn’t run when the UI freeze occurs, set the heartbeat task priority to the maximum priority.
Warning
Some RTOS use a task to schedule the RTOS timers. The heartbeat task priority must be lower than the RTOS timers priority.
(4) In this configuration, if the heartbeat is still running when the UI freeze occurs, then an RTOS task with a priority higher than the Core Engine task keeps using the CPU. Use the RTOS specific tools to identify what is the faulty task.
(5) If the heartbeat doesn’t run when the UI freeze occurs, then the RTOS scheduler is not scheduling anything. This can be caused by an RTOS timer task or an interrupt handler that never returns, or a crash of the RTOS scheduler.
Check Java Threads Scheduling¶
As a reminder, the threading model implemented by Core Engine is called green thread: it defines a multi-threaded environment without relying on any native RTOS capabilities.
Therefore, all Java threads run in a single RTOS task.
For more details, please refer to the MicroEJ Core Engine section.
A quick way to check if the Java threads are scheduled correctly is, here again, to make one of the threads print a heartbeat message. Copy/paste the following snippet in the main()
method of the application:
TimerTask task = new TimerTask() {
@Override
public void run() {
System.out.println("Alive");
}
};
Timer timer = new Timer();
timer.schedule(task, 10_000, 10_000);
This code creates a new Java thread that will print the message Alive
on the standard output every 10 seconds.
Assuming no one canceled the Timer
, if the Alive
printouts stop when the UI freeze occurs, then it can mean that:
- The Core Engine stopped scheduling the Java threads.
- Or that one or more threads with a higher priority prevent the threads with a lower priority from running.
Here are a few suggestions:
Ensure no Java threads with a high priority prevent the scheduling of the other Java threads. For example, convert the above example with a dedicated thread with the highest priority:
Thread thread = new Thread(new Runnable() { @Override public void run() { while (true) { try { Thread.sleep(10_000); System.out.println("Alive"); } catch (InterruptedException e) { e.printStackTrace(); } } } }); thread.setPriority(Thread.MAX_PRIORITY); thread.start();
The RTOS task that runs the Core Engine might be suspended or blocked. Check if some API call is suspending the task or if a shared resource could be blocking it.
When a Java native method is called, it calls its C counterpart function in the RTOS task that runs the Core Engine. While the C function is running, no other Java methods can run because the Core Engine waits for the C function to finish. Consequently, no Java thread can ever run again if the C function never returns. Therefore, spot any suspect native functions and trace every entry/exit to detect faulty code.
Please refer to Implementation Details if you encounter issues when implementing the heartbeat.
Check UI Thread Liveness¶
Now, what if the Alive
heartbeat runs while the UI is frozen?
Java threads are getting scheduled, but the UI thread (also called
Display Pump thread) does not process display events.
Let’s make the heartbeat snippet above execute in the UI
thread. Simply wraps the System.out.println("Alive")
with a
callSerially():
TimerTask task = new TimerTask() {
@Override
public void run() {
System.out.println("TimerTask Alive");
MicroUI.callSerially(new Runnable() {
@Override
public void run() {
System.out.println("UI Alive");
}
});
}
@Override
public void uncaughtException(Timer timer, Throwable e) {
// Default implementation of this method would cancel the task.
// Let's just ignore uncaught exceptions for debug purposes.
e.printStackTrace();
}
};
Timer timer = new Timer();
timer.schedule(task, 10_000, 10_000);
In case this snippet prints TimerTask Alive
but not UI alive
when
the freeze occurs, then there are a few options:
The application might be processing a long operation in the UI thread, for example:
- infinite/indeterminate loops
- network/database access
- heavy computations
- Thread.sleep()/Object.wait()
SNI_suspendCurrentJavaThread()
in native call
When doing so, any other UI-related operation will not be processed until completion, leading the display to be unresponsive. Any code that runs in the UI thread might be responsible. Look for code executed as a result of calls to:
repaint()
: code in renderContent()revalidate()
/revalidateSubTree()
: code invalidateContent()
andsetBoundsContent()
- handleEvent()
- callSerially(): code wrapped in such calls will be executed in the UI thread
The UI thread has terminated.
As a general rule, avoid running extended operations in the UI thread, follow the general pattern and use a dedicated thread/executor instead:
ExecutorService executorService = ServiceLoaderFactory.getServiceLoader().getService(ExecutorService.class, SingleThreadExecutor.class);
executorService.execute(new Runnable() {
@Override
public void run() {
// (... long non-UI operation ...)
// optional: update the UI upon completion
Display.getDefaultDisplay().callSerially(new Runnable() {
@Override
public void run() {
// update display code (will be executed in UI thread)
}
});
}
});
Check Input Events Processing¶
Another case worth looking at is whether the application is processing user input events as it should. The UI may look “frozen” only because it doesn’t react to input events. Replace the desktop instance with the one below to log all user inputs.
Desktop desktop = new Desktop() {
@Override
public EventHandler getController() {
EventHandler controller = super.getController();
return new EventHandler() {
@Override
public boolean handleEvent(int event) {
System.out.println("Desktop.handleEvent() received event of type " + Event.getType(event));
return controller.handleEvent(event);
}
};
}
};
It is also possible to display the content of MicroUI Event Buffer with the LLUI_INPUT_IMPL_log_XXX
API.
Please refer to the Event Buffer documentation for more information.
Implementation Details¶
Java Threads Creation¶
The number of threads in the MicroEJ Application must be sufficient to support the creation of additional threads when using Timer and Thread. The number of available threads can be updated in the launch configuration of the application (see Option(text): Number of threads).
If it is not possible to increase the number of available threads (for example, because the memory is full), try to reuse another thread but not the UI thread.
UART Not Available¶
If the UART output is not available, use another method to signal that the heartbeat task is running (e.g., blink an LED, use SystemView).
Use Case 2: Debugging a HardFault¶
When the application crashes, it can result from a HardFault triggered by the MCU.
The following sections explain:
- What are exceptions, HardFaults, and the exception handler.
- What to do in case of Memory Corruptions.
- What to do when a HardFault occurs.
Useful Resources¶
- IAR System: Debugging a HardFault on Cortex-M https://www.iar.com/knowledge/support/technical-notes/debugger/debugging-a-hardfault-on-cortex-m/
- ESP-IDF Programming Guide: Fatal Errors https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-guides/fatal-errors.html
- Using Cortex-M3/M4/M7 Fault Exceptions MDK Tutorial http://www.keil.com/appnotes/files/apnt209.pdf
Exceptions, HardFaults And Exception Handler¶
From ARM Architecture Reference Manual
An exception causes the processor to suspend program execution to handle an event, such as an externally generated interrupt or an attempt to execute an undefined instruction. Exceptions can be generated by internal and external sources. Normally, when an exception is taken, the processor state is preserved immediately, before handling the exception. This means that, when the event has been handled, the original state can be restored and program execution resumed from the point where the exception was taken.
For example, an IRQ request is an exception that can be recovered by handling the hardware request properly. On the other hand, an Undefined Instruction exception suggests a more severe system failure that might not be recoverable.
The exceptions that cannot be recovered are named HardFaults.
From ARM Architecture Reference Manual
When an exception is taken, processor execution is forced to an address that corresponds to the type of exception. This address is called the exception vector for that exception.
The code pointed by the exception vector is named exception handler. Therefore, a dedicated exception handler can be configured for all exceptions, including HardFaults.
Possible exceptions can be:
- Data Abort exception (access to unknown address)
- Undefined Instruction exception (execute code that is not valid)
- …
Check the hardware documentation for the complete list of exceptions.
What To Do In Exception Handlers?¶
For all HardFault handlers, the following data are available and must be printed:
- Name and value of all registers available
- Name of the handler
- Address of the failing instruction
Optionally:
- Content of the stack
- Call function
LLMJVM_dump
(fromLLMJVM.h
) to display the VM state (see Dump the States of the Core Engine)
Refer to the architecture documentation for how to configure the exception interrupt vector.
Memory Protection Unit (MPU)¶
A Memory Protection Unit (MPU) is a hardware unit that provides memory protection. An MPU allows privileged software to define memory regions and their policy. The policy describes who can access the memory.
For example, configure the heap and stack of a task to be accessible from the task itself only. The MPU generates an exception if another task or a device driver attempts to access the memory region.
If applicable, configure the MPU should to protect the application.
Check the RTOS documentation if it supports MPU.
For example, FreeRTOS includes FreeRTOS-MPU https://www.freertos.org/FreeRTOS-MPU-memory-protection-unit.html.
Configure the MPU to configure the access to the JVM heap and stack to prevent any other native threads from altering this area. Refer to this section for the list of section names defined by the MicroEJ Core Engine.
Memory Corruption¶
Memory corruption can result in the following symptoms:
- The address of the failing instruction is in a data section.
- The trace is incomplete or incorrect.
- The address of the failing instruction is located in the Garbage Collector (GC).
The cause(s) of a memory corruption can be:
- A native (C) function has a bug and writes to an incorrect memory location
- A native stack overflow
- A native heap overflow
- A device incorrectly initialized or misconfigured.
- …
When the HardFault occurs in the MicroJVM task, the VM task heap or stack may be corrupted.
Add LLMJVM_checkIntegrity
call in checkpoints of the BSP code to identify the timeslot of the memory corruption.
Typically, you can check a native with:
void Java_com_mycompany_MyClass_myNativeFunction(void) {
int32_t crcBefore = LLMJVM_checkIntegrity();
myNativeFunctionDo();
int32_t crcAfter = LLMJVM_checkIntegrity();
if(crcBefore != crcAfter){
// Corrupted memory in MicroJVM virtual machine internal structures
while(1);
}
}
Investigation¶
Determine which memory regions are affected and determine which components are responsible for the corruption.
List all the memories available and their specifics:
- Access mode (addressable, DMA, …)
- Cache mechanism? L1, L2
Is low-power enabled for CPU and peripherals? Is the memory disabled/changed to save power?
Get the memory layout of the project:
- What are the code sections for the BSP and the Application?
- Where are the BSP stack and heap? What about the Application stack and heap?
- Where is the Java immortals heap?
- Where are the Java strings?
- Where is the MicroEJ UI buffer?
- Besides the Java immortals, what are the other intersection point between the Java application and the BSP? (e.g., a temporary RAM buffer for JPEG decoder).
- Please refer to the Core Engine Link section to locate the Application sections, and to the Standalone Application Options for their sizes.
Implement a CRC of the hot sections when entering/leaving all natives. Hot Sections are memory sections used by both Java code and native code (e.g., C or ASM).
Move the C stack at the beginning of the memory to trigger a crash when it overflows (instead of corrupting the memory).
When a HardFault Occurs¶
Extract Information and Coredump¶
Attach an embedded debugger and get the following information:
stack traces and registers information for each stack frame
memory information
the whole memory, if possible
otherwise, get the hot sections
- BSP and Java heap and stack
- UI buffer
- immortals heap
- sections where the Java application and BSP are working together
Check which function is located at the address inside the PC register.
- It can be done either in Debug mode or by searching inside the generated .map file.
Memory Dump Analysis¶
- Run the Heap Dumper to check the application heap has not been corrupted.
- Make sure the native stack is not full (usually, there shall have the remaining initialization patterns in memory on top of the stack, such as
0xDEADBEEF
)
Trigger a VM Dump¶
LLMJVM_dump
function is provided by LLMJVM.h
.
This function prints the VM state.
Data printed in the VM state are:
- List of Java threads
- Stack trace for each thread
See this section to learn more about LLMJVM_dump
.