Cortex-M 하드폴트 분석 — Stacked Frame·CFSR 읽기
#한 줄 요약
“HardFault는 우연이 아니라 기록을 남기고 떨어집니다.” Stacked PC와 CFSR 두 값만 정확히 읽어내면 90%의 hardfault는 5분 안에 원인이 잡힙니다.
#사례 — “그냥 멈춰요”
QA에서 “보드가 5초쯤 동작하다 멈춥니다”라는 보고가 옵니다. UART 출력도 없고 LED 토글도 멈춥니다. JTAG로 halt해 보면 PC가 HardFault_Handler 안입니다. 여기서부터 시작입니다.
(gdb) backtrace#0 HardFault_Handler () at startup.s:142#1 <signal handler called>#2 0x????????
(gdb) info registersr0 0x12345678...PC 0x08000142 ← HardFault_HandlerLR 0xfffffff9 ← EXC_RETURN — main stack 리턴PSR 0x01000003backtrace는 도움이 안 됩니다. 진짜 fault가 난 PC는 stack frame 안에 있습니다.
#핵심 개념 — Stacked Frame
Cortex-M은 예외 진입 시 다음 8 word를 자동으로 stack에 push합니다.
SP → R0 R1 R2 R3 R12 LR ← fault 직전 함수의 return address PC ← fault 발생 명령 주소 xPSREXC_RETURN의 LR (0xFFFFFFF9, 0xFFFFFFFD 등)으로 어느 stack에 push되었는지를 알 수 있습니다.
LR 값 Mode Stack0xFFFFFFF1 Handler MSP0xFFFFFFF9 Thread, no FP MSP0xFFFFFFFD Thread, no FP PSP0xFFFFFFE1/E9 Thread, FP + FP regs#Step 1 — Handler에서 SP 잡기
__attribute__((naked))void HardFault_Handler(void) { __asm volatile ( "tst lr, #4 \n" // bit 2: 0=MSP, 1=PSP "ite eq \n" "mrseq r0, msp \n" "mrsne r0, psp \n" "b hardfault_report \n" );}
void hardfault_report(uint32_t *sp) { uint32_t r0 = sp[0]; uint32_t r1 = sp[1]; uint32_t r2 = sp[2]; uint32_t r3 = sp[3]; uint32_t r12 = sp[4]; uint32_t lr = sp[5]; uint32_t pc = sp[6]; // ← fault 명령 주소 uint32_t psr = sp[7];
uint32_t cfsr = *(volatile uint32_t*)0xE000ED28; uint32_t hfsr = *(volatile uint32_t*)0xE000ED2C; uint32_t mmar = *(volatile uint32_t*)0xE000ED34; uint32_t bfar = *(volatile uint32_t*)0xE000ED38;
printf("HARDFAULT\n"); printf("PC = 0x%08lx\n", pc); printf("LR = 0x%08lx\n", lr); printf("CFSR = 0x%08lx\n", cfsr); printf("HFSR = 0x%08lx\n", hfsr); printf("MMAR = 0x%08lx\n", mmar); printf("BFAR = 0x%08lx\n", bfar);
while (1);}이 핸들러를 startup 코드의 weak HardFault_Handler를 override해서 넣어 둡니다.
#Step 2 — CFSR 비트 해석
CFSR(Configurable Fault Status Register, 0xE000ED28) 32비트는 세 sub-register로 나뉩니다.
[31:16] UFSR — Usage Fault bit 16 UNDEFINSTR 정의되지 않은 명령 bit 17 INVSTATE Thumb 비트 0 bit 18 INVPC 예외 return 시 잘못된 EXC_RETURN bit 19 NOCP 코프로세서 없음 bit 24 UNALIGNED unaligned access bit 25 DIVBYZERO 0으로 나눔
[15:8] BFSR — Bus Fault bit 8 IBUSERR 명령 fetch bit 9 PRECISERR precise data bus error (BFAR 유효) bit 10 IMPRECISERR imprecise (BFAR 무효, 위치 부정확) bit 11 UNSTKERR exception entry stacking bit 12 STKERR exception exit unstacking bit 15 BFARVALID BFAR에 주소 적힘
[7:0] MMFSR — MemManage Fault (MPU) bit 0 IACCVIOL 명령 fetch MPU violation bit 1 DACCVIOL data access MPU violation bit 7 MMARVALID MMAR에 주소 적힘가장 흔한 두 패턴.
CFSR = 0x00008200 → BFSR bit 9 (PRECISERR) + bit 15 (BFARVALID) → BFAR 주소에 *유효한 fault 위치* 있음 → 보통: null pointer dereference, peripheral 미초기화
CFSR = 0x00020000 → UFSR bit 17 (INVSTATE) → 함수 포인터 호출 시 LSB가 0 (Thumb 비트 누락) → 보통: function pointer 깨짐#Step 3 — PC를 source line으로
arm-none-eabi-addr2line -e firmware.elf -f -C 0x08001234process_packetsrc/packet.c:87또는 GDB에서 직접.
(gdb) list *0x080012340x8001234 is in process_packet (packet.c:87).82 if (hdr->len > MAX_LEN) return -1;83 uint8_t *dst = buffers[hdr->channel];8485 // hdr->channel이 음수 (channel = -1) → buffers[-1] dereference86 // → BFAR = 0x20001ff8 (buffers 직전 주소)87 memcpy(dst, payload, hdr->len);이 한 줄에서 채널 음수가 들어온 path를 따라가면 원인이 잡힙니다.
#사례 마무리
처음 보드의 CFSR/BFAR이 다음이었습니다.
CFSR = 0x00008200 BFAR = 0x00000004PC = 0x08002A18 LR = 0x0800291C0x00000004는 null pointer + 4. 누군가 NULL->next 같은 dereference를 했다는 신호입니다.
addr2line -e firmware.elf 0x08002A18queue_popsrc/queue.c:42// queue.c:42QueueItem *item = q->head;q->head = item->next; // ← item이 NULL일 때q->head가 비어 있을 때 queue_pop을 호출한 caller가 범인이었습니다. ISR과 main이 같은 queue를 다루며 빈 큐 검사를 ISR이 빠뜨렸습니다. 5초 후 발생은 ISR이 처음 empty queue를 만났을 때였습니다.
#Imprecise BFSR — 가장 까다로움
CFSR = 0x00000400 BFSR bit 10 IMPRECISERRBFAR 무효Write buffer 때문에 fault PC가 실제 명령보다 몇 사이클 늦게 보고됩니다. 다음을 시도합니다.
// SCB->ACTLR에서 write buffer disable (디버깅 시에만)*(volatile uint32_t*)0xE000E008 |= (1 << 1); // DISDEFWBUF이러면 fault가 정확한 PC에 떨어집니다. Production에서는 해제합니다.
#사용 권장 패턴 — fault 정보 NVRAM 저장
typedef struct { uint32_t magic; // 0xFA17DEAD uint32_t pc, lr, psr; uint32_t cfsr, hfsr; uint32_t bfar, mmar; uint32_t reason; // 자체 정의 code} fault_record_t;
void hardfault_report(uint32_t *sp) { fault_record_t *fr = (void*)BACKUP_SRAM_BASE; fr->magic = 0xFA17DEAD; fr->pc = sp[6]; fr->lr = sp[5]; fr->cfsr = *(volatile uint32_t*)0xE000ED28; /* ... */ NVIC_SystemReset();}
void boot_check_fault(void) { fault_record_t *fr = (void*)BACKUP_SRAM_BASE; if (fr->magic == 0xFA17DEAD) { printf("Last boot fault: PC=0x%08lx CFSR=0x%08lx\n", fr->pc, fr->cfsr); fr->magic = 0; }}Field 환경에서 boot 후 이전 reset 원인을 알 수 있습니다. STM32 RTC backup, RP2040 watchdog scratch, NRF52 GPREGRET 모두 reset에서 살아남는 영역을 제공합니다.
#CFSR 비트별 흔한 원인
| CFSR | 의미 | 흔한 원인 |
|---|---|---|
| BFSR PRECISERR + BFAR | precise bus error | null pointer, 미초기화 peripheral, RCC clock off |
| BFSR IMPRECISERR | imprecise | write buffer로 인한 지연 — disable 후 재현 |
| UFSR UNDEFINSTR | undefined inst | flash 깨짐, JTAG 잘못 flash |
| UFSR INVSTATE | thumb 비트 0 | function pointer 깨짐, code memory 손상 |
| UFSR UNALIGNED | unaligned access | __packed 구조체 → 정렬 안 된 word load/store |
| MMFSR DACCVIOL + MMAR | MPU violation | task stack overflow가 guard region 침범 |
#자주 보는 함정
Handler에서 printf 호출
UART buffer DMA가 fault로 멈춘 상태일 수 있습니다. HAL_UART_Transmit polling 모드로 하거나, ITM/SWO로 출력합니다.
Stack pointer가 깨진 채로 handler 진입
SP = 0xdeadbeef ← stack 자체가 무효sp[6] 읽기 → 또 fault → "lockup"NVIC가 stacked frame을 못 push해서 무한 fault에 빠집니다. HFSR.FORCED가 떨어집니다. Stack overflow가 원인입니다. MPU stack guard로 먼저 잡아야 합니다.
잘못된 EXC_RETURN
__asm volatile ("bx %0" :: "r"(0xFFFFFFF8)); // ← reservedEXC_RETURN을 인라인 어셈블리로 만들 때 reserved 값을 쓰면 UFSR INVPC가 떨어집니다.
-Os빌드에서 PC와 source의 미스매치
addr2line이 함수 이름은 맞지만 줄이 어긋날 수 있습니다. -Og로 빌드하거나, objdump -dS firmware.elf | less로 어셈블리·source 인터리브를 봅니다.
CFSR을 읽고 안 clear
uint32_t cfsr = *(volatile uint32_t*)0xE000ED28;// CFSR은 W1C — 다음 fault 추적을 위해 clear*(volatile uint32_t*)0xE000ED28 = cfsr;Reset 없이 fault 재현 디버깅을 할 때 누적된 bit 때문에 헷갈립니다.
#정리
- HardFault는 stacked frame과 CFSR에 모든 단서를 남깁니다.
- LR (EXC_RETURN) 비트 2로 MSP/PSP를 골라 stacked frame을 잡습니다.
- Stacked PC가 fault 발생 명령.
addr2line으로 source line. - CFSR을 BFSR/UFSR/MMFSR로 나눠 읽고 BFAR/MMAR을 함께 봅니다.
- Imprecise BFSR은 write buffer disable로 precise하게 만듭니다.
- Field에서는 fault record를 NVRAM에 남기고 reset합니다.
- Stack overflow가 lockup의 가장 흔한 원인. MPU guard로 먼저 잡습니다.
다음 편은 UART 안 찍힐 때입니다.
#관련 항목
Modern Embedded Recipes · 115 of 152
- 1Modern Embedded Recipes — 모던 임베디드 실전 레시피 시리즈 소개
- 2디지털 신호 기초 — Voltage Level·Edge·Setup/Hold 분석
- 3임베디드 클럭과 타이밍 — Skew·Jitter·PLL·MMCM 분석
- 4GPIO 내부 구조 분해 — Push-Pull·Open-Drain·Schmitt Trigger
- 5UART 하드웨어 동작 분석 — Baud Rate·Framing·FIFO
- 6SPI 하드웨어 분석 — Clock Mode·MOSI/MISO·Chip Select
- 7I2C 하드웨어 분석 — Open-Drain·Clock Stretching·Arbitration
- 8ADC 동작 원리 — SAR·Sigma-Delta·Pipelined 비교
- 9DAC 동작 원리 — R-2R Ladder·Sigma-Delta·Settling Time
- 10PWM 신호 생성 분석 — Duty·Frequency·Dead Time·Center-Aligned
- 11CAN 버스 전기적 특성 — Differential·Termination·Dominant/Recessive
- 12RS-485·RS-422 차동 신호 분석 — Termination·Biasing·Topology
- 13LVDS 차동 신호 분석 — Common-Mode·Impedance·Eye Pattern
- 14ARM Cortex-M 시리즈 비교 — M0·M3·M4·M7·M33·M55 분석
- 15ARM Cortex-A 시리즈 비교 — A53·A55·A72·A78·X1 분석
- 16ARM 레지스터 구조 분석 — R0~R15·CPSR·SPSR·Banked Registers
- 17Cortex-M 예외 처리 — Vector Table·NVIC·Tail-Chaining 추적
- 18ARM 메모리 맵 분석 — Normal·Device·Strongly-Ordered Region
- 19ARM L1·L2 캐시 분석 — Set Associative·Inclusive·Maintenance
- 20ARM MPU 활용 — Region·Attribute·Privilege Separation
- 21ARM MMU 기초 분석 — Translation Table·TLB·ASID
- 22ARM TrustZone-M 기초 — Secure/Non-Secure·NSC·MPC
- 23ARM Memory Barrier 실전 — DMB·DSB·ISB·DMA·MMIO
- 24임베디드 크로스 컴파일러 분석 — GCC·Clang·Sysroot 구성
- 25C 컴파일 4단계 — Preprocess·Compile·Assemble·Link 추적
- 26ELF 파일 구조 분석 — Section·Segment·Symbol Table·DWARF
- 27링커 스크립트 기초 — SECTIONS·MEMORY·entry point
- 28링커 스크립트 고급 — Overlay·BSS·init_array·LMA/VMA
- 29임베디드 스타트업 코드 분석 — Reset_Handler·Vector Table·SystemInit
- 30C 런타임 crt0 분석 — Stack·BSS Zero·Data Copy·atexit
- 31임베디드 메모리 레이아웃 — .text·.rodata·.data·.bss·.heap·.stack
- 32임베디드 컴파일러 최적화 분석 — -O0~-O3·-Os·-LTO 비교
- 33Map 파일 분석 — Symbol·Section·Size 추적으로 코드 크기 진단
- 34Make·CMake 크로스 컴파일 — Toolchain File·Sysroot 통합
- 35임베디드 Bootloader 체인 — BootROM·SPL·U-Boot·Kernel·Secure Boot
- 36첫 bare-metal 프로그램 작성 — Linker·Startup·main의 최소 구성
- 37MMIO 레지스터 직접 접근 — volatile·Memory Map·Aliasing 분석
- 38GPIO 드라이버 직접 구현 — STM32 HAL 없이 레지스터로
- 39임베디드 클럭 설정 분석 — HSE·PLL·SYSCLK·AHB/APB 분주
- 40Cortex-M 인터럽트 핸들링 — NVIC·Priority·Vector·EXTI
- 41SysTick 타이머 활용 — 24-bit Counter·1ms Tick·delay 구현
- 42UART 드라이버 구현 — polling·interrupt·DMA 3가지 방식 비교
- 43SPI 드라이버 구현 — Master·Slave·CRC·DMA
- 44I2C 드라이버 구현 — Master·7-bit/10-bit·Clock Stretching 처리
- 45임베디드 DMA 기초 — Memory-to-Memory·Peripheral·Circular Mode
- 46저전력 모드 분석 — Sleep·Stop·Standby·Wake-up Source
- 47IWDG·WWDG 워치독 구현 — Independent vs Window 비교
- 48임베디드 Flash 프로그래밍 — Erase·Program·Read While Write
- 49DDR 초기화 실패 진단 — Timing·Calibration·Walking Bit Test
- 50PWM 출력 실전 — LED 밝기·모터 속도 제어
- 51DC 모터 제어 — H-Bridge·PWM Duty·Encoder Feedback
- 52스테퍼 모터 제어 — Full Step·Half Step·Microstepping
- 53서보 모터 제어 — PWM 1ms~2ms·Closed Loop·PID
- 54Character LCD 제어 — HD44780·4-bit Mode·Custom Char
- 55SPI OLED 제어 — SSD1306·Frame Buffer·Page 단위 갱신
- 56TFT 디스플레이 구동 — RGB565·FSMC·LTDC·DMA2D
- 57환경 센서 활용 — BME280 온습압·SHT3x·BMP180 비교
- 58IMU 센서 활용 — MPU6050·LSM6DSO·Sensor Fusion
- 59CAN 통신 구현 — bxCAN·Filter·Mailbox·CAN-FD
- 60USB Device 기초 — Descriptor·Enumeration·Endpoint·HID/CDC
- 61Ethernet MAC+PHY 통합 — RMII·lwIP·DMA Descriptor
- 62SD Card + FatFs 구현 — SPI/SDIO 모드·CSD/CID·Wear
- 63RTC 활용 — Calendar·Alarm·Wake-up Timer·Backup Domain
- 64RTOS 도입 결정 분석 — Super Loop vs RTOS 트레이드오프
- 65RTOS Task 설계 패턴 — 우선순위·스택·State Machine
- 66RTOS Scheduler 동작 분석 — Tick·Context Switch·Yield
- 67RTOS Semaphore 활용 — Binary·Counting·ISR Give
- 68RTOS Mutex 활용 — Recursive·Priority Inheritance 적용
- 69RTOS Queue 활용 — By-Value·By-Reference·Timeout 패턴
- 70RTOS Event Group 활용 — Bit Wait·Sync·Notify
- 71RTOS Software Timer 활용 — One-shot·Auto-reload·Daemon Task
- 72ISR-Safe API 설계 — Reentrant·Atomic·Defer 패턴
- 73Priority Inversion 진단·예방 — Mars Pathfinder Lesson 추적
- 74Timer Wheel 분석 — Hashed·Hierarchical·O(1) Tick
- 75RTOS 디버깅 기법 — Tracealyzer·SystemView·Stack 추적
- 76임베디드 Linux 부팅 흐름 분석 — BootROM·U-Boot·Kernel·init
- 77U-Boot 활용 — bootcmd·env·tftp·boot.scr 분석
- 78Device Tree 실전 — DTS·DTB·Overlay·Phandle 추적
- 79Device Tree Overlay 적용 — Runtime fragment·dtoverlay
- 80임베디드 커널 빌드 — defconfig·menuconfig·Image·zImage
- 81커널 모듈 기초 — init/exit·Parameter·KBuild·DKMS
- 82캐릭터 드라이버 작성 — file_operations·cdev·register_chrdev
- 83Platform 드라이버 작성 — probe·remove·of_match·DT 바인딩
- 84mmap 4가지 모드 — Anonymous·File·Shared·Huge Page
- 85epoll 실전 — LT·ET·ONESHOT·EXCLUSIVE 비교
- 86UIO·VFIO 분석 — User-Space Driver와 IOMMU 격리
- 87sysfs·configfs 활용 — kobject 기반 User 인터페이스
- 88IRQ Affinity 튜닝 — smp_affinity·isolcpus·irqbalance
- 89루트 파일시스템 구축 — Buildroot 기초·Package·Toolchain
- 90임베디드 동적 메모리 — malloc 위험·결정성·대안 분석
- 91메모리 정렬과 패딩 분석 — Natural·Strict Alignment·Trap
- 92Cache Line Alignment — alignas·Padding·SoA 적용
- 93DMA-Friendly Allocator — dma_alloc_coherent·IOMMU·Pool
- 94Zero-Copy Pipeline — DMA-BUF·sendfile·io_uring·splice
- 95NUMA Memory Topology — numactl·numa_alloc·HBM 적용
- 96SIMD 활용 분석 — Intrinsics·Auto-Vectorization·OpenMP SIMD
- 97ARM NEON 심화 — Matrix Multiply·FFT·Image Filter 적용
- 98임베디드 스택 분석 — high-water·overflow 탐지
- 99임베디드 코드 크기 최적화 — -Os·LTO·Section Garbage Collection
- 100임베디드 전력 최적화 — Sleep Mode·Clock Gating·DVFS
- 101WCET 분석 기법 — Static·Measurement·Hybrid 방법론
- 102Lock-Free Ring Buffer 구현 — SPSC·Power-of-2·Memory Order
- 103Wait-Free Signaling — Atomic Flag·Sequence·Latest-Value
- 104RCU (Read-Copy-Update) 기초 — Quiescent State·Grace Period
- 105Hazard Pointer 분석 — Lock-Free Memory Reclamation
- 106Compare-And-Swap 패턴 — Stack·Counter·Linked List 적용
- 107Atomic Operation 비용 분석 — Fence·Cache Line·Contention
- 108Spinlock vs Mutex 결정 가이드 — Context Switch·Hold Time
- 109ABA 문제 회피 — Tagged Pointer·Hazard·Generation Counter
- 110False Sharing 해결 — Cache Line Padding·SoA 적용
- 111MPMC Queue 구현 — Multi-producer Multi-consumer Lock-Free
- 112임베디드 디버깅 마인드셋 — 가설·격리·재현·이분탐색
- 113JTAG·SWD 안 붙을 때 — 핀·전압·속도·세션 진단
- 114GDB 원격 디버깅 — OpenOCD·J-Link·target remote 구성
- 115Cortex-M 하드폴트 분석 — Stacked Frame·CFSR 읽기
- 116UART 안 찍힐 때 — Bare-metal 체크리스트
- 117임베디드 부팅 실패 진단 — 단계별 Isolation
- 118인터럽트 누락·중복 진단 — Priority·Pending·Re-entry 추적
- 119메모리 오버플로우·오염 진단 — Canary·MPU·Pattern 분석
- 120타이밍·Race 진단 — Heisenbug 잡는 법
- 121통신 프로토콜 분석 — Logic Analyzer와 Protocol Decoder
- 122임베디드 로깅 시스템 설계 — 레벨·버퍼·SWO·Deferred
- 123임베디드 포스트모템 분석 — Core Dump와 Field Crash
- 124FPGA 기초 분석 — LUT·FF·BRAM·DSP 자원 구조
- 125Vivado 사용법 — Project·Constraint·Synth·Impl·Bitstream
- 126PCIe BAR 매핑 분석 — Config Space·Enumeration·MMIO 접근
- 127AXI 인터페이스 — AXI4·AXI4-Lite·AXI-Stream 비교
- 128Zynq PS-PL 통신 — GP·HP·ACP 인터페이스 선택
- 129Mailbox Protocol 분석 — Host와 Accelerator를 잇는 Doorbell
- 130Command Queue·Submission Queue — NVMe·XDMA 공통 패턴
- 131DMA Completion 메커니즘 — Interrupt·Polling·Completion Ring
- 132PCIe Streaming 분석 — BAR Type·MSI-X·Kernel Bypass
- 133Vitis HLS 분석 — Pragma·Pipeline II·Dataflow 실전 감각
- 134HLS 최적화 기법 — Pipeline·Unroll·Partition·Dataflow
- 135Vitis AI 분석 — DPU·xmodel·VART
- 136OpenCL on FPGA — Kernel·Channel·Burst Memory 분석
- 137Intel Quartus 사용법 — Platform Designer·Nios II·HLS
- 138Edge Inference 분석 — Cloud vs Edge·Latency·Privacy
- 139NPU 아키텍처 분석 — Ethos·Hexagon·Systolic Array 비교
- 140딥러닝 Quantization 분석 — PTQ·QAT·INT8·INT4·Calibration
- 141TensorRT 분석 — ONNX→Engine·FP16·INT8·DLA·Multi-Stream
- 142TFLite Micro 분석 — Op Resolver·Tensor Arena·Cortex-M
- 143ONNX Runtime 분석 — Execution Provider와 Cross-Platform 배포
- 144Edge Thermal Management — Throttling·DVFS·Fan Curve·Sustained
- 145NVIDIA Jetson 분석 — Nano·Xavier·Orin·Thor·JetPack·DLA·VPI
- 146Zero-Copy Camera Pipeline — V4L2·DMA-BUF·GPU Import·NPU 직결
- 147온디바이스 LLM 추론 — llama.cpp·GGUF·MLX·KV Cache·NPU Backend
- 148Cortex-M33 TF-M·TrustZone — Secure Firmware·PSA·MCUboot
- 149Matter·Thread 분석 — IoT 통합 표준·Commissioning·Multi-Fabric
- 150PCIe → CXL 진화 — 같은 PHY 위 cache-coherent 프로토콜 추가
- 151QEMU CXL Type 3 디바이스 에뮬레이션 — 노트북에서 CXL 개발 환경 구축
- 152Linux CXL 드라이버 분석 — cxl_pci·cxl_core·region·DAX
관련 글
TFLite Micro 분석 — Op Resolver·Tensor Arena·Cortex-M
MCU용 TensorFlow Lite Micro의 구조, op resolver·tensor arena·CMSIS-NN integration·Ethos-U delegate.
임베디드 포스트모템 분석 — Core Dump와 Field Crash
Linux coredump·gdb 분석부터 MCU 환경의 mini-dump(Memfault)·last-gasp logging·field debug 패턴까지.
임베디드 로깅 시스템 설계 — 레벨·버퍼·SWO·Deferred
임베디드 환경에서 overhead를 최소화한 로깅. 레벨 분리·circular buffer·SWO/RTT·deferred 처리 패턴.
이 글을 참조하는 글 (7)
- 임베디드 포스트모템 분석 — Core Dump와 Field Crash— Modern Embedded Recipes
- 임베디드 로깅 시스템 설계 — 레벨·버퍼·SWO·Deferred— Modern Embedded Recipes
- 메모리 오버플로우·오염 진단 — Canary·MPU·Pattern 분석— Modern Embedded Recipes
- 인터럽트 누락·중복 진단 — Priority·Pending·Re-entry 추적— Modern Embedded Recipes
- 임베디드 부팅 실패 진단 — 단계별 Isolation— Modern Embedded Recipes
- GDB 원격 디버깅 — OpenOCD·J-Link·target remote 구성— Modern Embedded Recipes
- 임베디드 디버깅 마인드셋 — 가설·격리·재현·이분탐색— Modern Embedded Recipes