ESPHome 2026.6.4
Loading...
Searching...
No Matches
hal.h
Go to the documentation of this file.
1#pragma once
2
3#ifdef USE_LIBRETINY
4
5#include <cstdint>
6
7// For the inline millis() fast paths (xTaskGetTickCount, portTICK_PERIOD_MS).
8#include <FreeRTOS.h>
9#include <task.h>
10
12
13// IRAM_ATTR places a function in executable RAM so it is callable from an
14// ISR even while flash is busy (XIP stall, OTA, logger flash write). All
15// LibreTiny families that need it share the same .sram.text input section
16// name; how that section is routed into RAM differs per family:
17// RTL8720C: stock linker consumes *(.sram.text*) into .ram.code_text.
18// RTL8710B: patch_linker.py.script injects KEEP(*(.sram.text*)) at the
19// top of .ram_image2.data (which IS in ltchiptool's
20// sections_ram). The stock linker has KEEP(*(.image2.ram.text*))
21// in .ram_image2.text but that output section is NOT in
22// ltchiptool's AmebaZ elf2bin sections_ram list, so code routed
23// there is dropped from the flashed binary.
24// LN882H: patch_linker.py.script injects KEEP(*(.sram.text*)) into
25// .flash_copysection (> RAM0 AT> FLASH), after KEEP(*(.vectors))
26// so the Cortex-M4 vector table stays 512-byte-aligned for VTOR.
27//
28// BK72xx (all variants) are left as a no-op: their SDK wraps flash
29// operations in GLOBAL_INT_DISABLE() which masks FIQ + IRQ at the CPU for
30// the duration of every write, so no ISR fires while flash is stalled and
31// the race IRAM_ATTR guards against cannot occur. The trade-off is that
32// interrupts are delayed (not dropped) by up to ~20 ms during a sector
33// erase, but that is an SDK-level choice and cannot be changed from this
34// layer.
35#if defined(USE_BK72XX)
36#define IRAM_ATTR
37#else
38#define IRAM_ATTR __attribute__((noinline, section(".sram.text")))
39#endif
40#define PROGMEM
41
42#ifdef USE_BK72XX
43// Declared in the Beken FreeRTOS port (portmacro.h) and built in ARM mode so
44// it is callable from Thumb code via interworking. The MRS CPSR instruction
45// is ARM-only and user code here may be built in Thumb, so in_isr_context()
46// defers to this port helper on BK72xx instead of reading CPSR inline.
48#endif
49
50// Forward decls from Arduino's <Arduino.h> for the inline wrappers below.
51// NOLINTBEGIN(google-runtime-int,readability-identifier-naming,readability-redundant-declaration)
52extern "C" void yield(void);
53extern "C" void delay(unsigned long ms);
54extern "C" unsigned long micros(void);
55extern "C" unsigned long millis(void);
56extern "C" void delayMicroseconds(unsigned int us);
57// NOLINTEND(google-runtime-int,readability-identifier-naming,readability-redundant-declaration)
58
59// Forward decls from libretiny's <lt_api.h> family for the inline arch_*
60// wrappers below. Pulling the full header would drag in the rest of the
61// LibreTiny C API.
62extern "C" void lt_wdt_feed(void);
64extern "C" uint32_t lt_cpu_get_freq(void);
65
66namespace esphome::libretiny {}
67
68namespace esphome {
69
71__attribute__((always_inline)) inline bool in_isr_context() {
72#if defined(USE_BK72XX)
73 // BK72xx is ARM968E-S (ARM9); see extern declaration above.
75#else
76 // Cortex-M (AmebaZ, AmebaZ2, LN882H). IPSR is the active exception number;
77 // non-zero means we're in a handler.
78 uint32_t ipsr;
79 __asm__ volatile("mrs %0, ipsr" : "=r"(ipsr));
80 return ipsr != 0;
81#endif
82}
83
84__attribute__((always_inline)) inline void yield() { ::yield(); }
85__attribute__((always_inline)) inline void delay(uint32_t ms) { ::delay(ms); }
86__attribute__((always_inline)) inline uint32_t micros() { return static_cast<uint32_t>(::micros()); }
87
88// Per-variant millis() fast path — matches MillisInternal::get().
89#if defined(USE_RTL87XX) || defined(USE_LN882X)
90static_assert(configTICK_RATE_HZ == 1000, "millis() fast path requires 1 kHz FreeRTOS tick");
91__attribute__((always_inline)) inline uint32_t millis() {
92 // xTaskGetTickCountFromISR is mandatory in interrupt context per the FreeRTOS API contract.
93 return in_isr_context() ? xTaskGetTickCountFromISR() : xTaskGetTickCount();
94}
95#elif defined(USE_BK72XX)
96static_assert(configTICK_RATE_HZ == 500, "BK72xx millis() fast path assumes 500 Hz FreeRTOS tick");
97__attribute__((always_inline)) inline uint32_t millis() { return xTaskGetTickCount() * portTICK_PERIOD_MS; }
98#else
99__attribute__((always_inline)) inline uint32_t millis() { return static_cast<uint32_t>(::millis()); }
100#endif
101__attribute__((always_inline)) inline uint64_t millis_64() { return Millis64Impl::compute(millis()); }
102
103// NOLINTNEXTLINE(readability-identifier-naming)
104__attribute__((always_inline)) inline void delayMicroseconds(uint32_t us) { ::delayMicroseconds(us); }
105__attribute__((hot, always_inline)) inline void arch_feed_wdt() { lt_wdt_feed(); }
106__attribute__((always_inline)) inline uint32_t arch_get_cpu_cycle_count() { return lt_cpu_get_cycle_count(); }
107__attribute__((always_inline)) inline uint32_t arch_get_cpu_freq_hz() { return lt_cpu_get_freq(); }
108
109void arch_init();
110
111} // namespace esphome
112
113#endif // USE_LIBRETINY
struct @65::@66 __attribute__
Wake the main loop task from an ISR. ISR-safe.
Definition main_task.h:32
unsigned long millis(void)
void delay(unsigned long ms)
unsigned long micros(void)
void yield(void)
uint32_t platform_is_in_interrupt_context(void)
uint32_t lt_cpu_get_freq(void)
void delayMicroseconds(unsigned int us)
uint32_t lt_cpu_get_cycle_count(void)
void lt_wdt_feed(void)
uint32_t arch_get_cpu_cycle_count()
Definition hal.cpp:71
void arch_init()
Definition hal.cpp:47
void IRAM_ATTR HOT delayMicroseconds(uint32_t us)
Definition hal.cpp:48
uint32_t arch_get_cpu_freq_hz()
Definition hal.cpp:63
uint64_t millis_64()
Definition hal.cpp:29
uint32_t IRAM_ATTR HOT micros()
Definition hal.cpp:43
void arch_feed_wdt()
Definition hal.cpp:53
void HOT delay(uint32_t ms)
Definition hal.cpp:85
uint32_t IRAM_ATTR HOT millis()
Definition hal.cpp:28
static void uint32_t