ESPHome 2026.6.0-dev
Loading...
Searching...
No Matches
hal.cpp
Go to the documentation of this file.
1#ifdef USE_ESP8266
2
3#include "esphome/core/hal.h"
5
6#include <Arduino.h>
7#include <core_esp8266_features.h>
8#include <coredecls.h>
9
10extern "C" {
11#include <user_interface.h>
12}
13
14// Empty esp8266 namespace block to satisfy ci-custom's lint_namespace check.
15// HAL functions live in namespace esphome (root) — they are not part of the
16// esp8266 component's API.
17namespace esphome::esp8266 {} // namespace esphome::esp8266
18
19namespace esphome {
20
21// yield(), micros(), millis_64(), delayMicroseconds(), arch_feed_wdt(),
22// progmem_read_*() are inlined in components/esp8266/hal.h.
23//
24// Fast accumulator replacement for Arduino's millis() (~3.3 μs via 4× 64-bit
25// multiplies on the LX106). Tracks a running ms counter from 32-bit
26// system_get_time() deltas using pure 32-bit ops. Installed as __wrap_millis
27// (via -Wl,--wrap=millis) so Arduino libs and IRAM_ATTR ISR handlers (e.g.
28// Wiegand, ZyAura) also get the fast version. xt_rsil(15) guards the static
29// state against ISR re-entry; the critical section is bounded (≤10 while-loop
30// iterations, ~100 ns on the common path, or a constant-time /1000 ~2.5 μs on
31// the rare path — well under WiFi's ~10 μs ISR latency budget). NMIs (level
32// >15) are not masked, but the ESP8266 SDK's NMI handlers don't call millis().
33//
34// system_get_time() wraps every ~71.6 min; unsigned (now_us - last_us) handles
35// one wrap. The main loop calls millis() at 60+ Hz, so delta stays tiny — a
36// >71 min block would trip the watchdog long before it could matter here.
37static constexpr uint32_t MILLIS_RARE_PATH_THRESHOLD_US = 10000;
38static constexpr uint32_t US_PER_MS = 1000;
39
40uint32_t IRAM_ATTR HOT millis() {
41 // Struct packs the three statics so the compiler loads one base address
42 // instead of three separate literal pool entries (saves ~8 bytes IRAM).
43 static struct {
44 uint32_t cache;
45 uint32_t remainder;
46 uint32_t last_us;
47 } state = {0, 0, 0};
48 uint32_t ps = xt_rsil(15);
49 uint32_t now_us = system_get_time();
50 uint32_t delta = now_us - state.last_us;
51 state.last_us = now_us;
52 state.remainder += delta;
53 if (state.remainder >= MILLIS_RARE_PATH_THRESHOLD_US) {
54 // Rare path: large gap (WiFi scan, boot, long block). Constant-time
55 // conversion keeps the critical section bounded.
56 uint32_t ms = state.remainder / US_PER_MS;
57 state.cache += ms;
58 // Reuse ms instead of `remainder %= US_PER_MS` — `%` would compile to a
59 // second __umodsi3 call on the LX106 (no hardware divide).
60 state.remainder -= ms * US_PER_MS;
61 } else {
62 // Common path: small gap. At most ~10 iterations since remainder was
63 // < threshold (10 ms) on entry and delta adds at most one more threshold
64 // before exiting this branch.
65 while (state.remainder >= US_PER_MS) {
66 state.cache++;
67 state.remainder -= US_PER_MS;
68 }
69 }
70 uint32_t result = state.cache;
71 xt_wsr_ps(ps);
72 return result;
73}
74
75// Delegate to Arduino's 1-arg esp_delay(), which uses os_timer + esp_suspend to
76// suspend the cont task for `ms` milliseconds without polling millis(). This
77// matches pre-2026.5.0 behavior (when esphome::delay() forwarded to ::delay())
78// and lets the SDK run freely while we wait, which timing-sensitive
79// interrupt-driven code (e.g. ESP8266 software-serial RX in components like
80// fingerprint_grow) depends on. The poll-based busy-wait that this replaced
81// rarely yielded inside short waits like delay(1), starving WiFi/SDK tasks and
82// extending interrupt latency. Unlike ::delay(), esp_delay()'s 1-arg form does
83// not call millis(), so the slow Arduino millis() body is not pulled into IRAM
84// by this path (the --wrap=millis goal of #15662 is preserved).
85void HOT delay(uint32_t ms) {
86 if (ms == 0) {
87 optimistic_yield(1000);
88 return;
89 }
90 esp_delay(ms);
91}
92
93void arch_restart() {
94 system_restart();
95 // restart() doesn't always end execution
96 while (true) { // NOLINT(clang-diagnostic-unreachable-code)
97 yield();
98 }
99}
100
101} // namespace esphome
102
103// Linker wrap: redirect all ::millis() calls (Arduino libs, ISRs) to our accumulator.
104// Requires -Wl,--wrap=millis in build flags (added by __init__.py).
105// NOLINTNEXTLINE(bugprone-reserved-identifier,cert-dcl37-c,cert-dcl51-cpp,readability-identifier-naming)
106extern "C" uint32_t IRAM_ATTR __wrap_millis() { return esphome::millis(); }
107// Note: Arduino's init() registers a 60-second overflow timer for micros64().
108// We leave it running — wrapping init() as a no-op would break micros64()'s
109// overflow tracking, and the timer's cost is negligible (~3 μs per 60 s).
110
111#endif // USE_ESP8266
void yield(void)
uint32_t IRAM_ATTR __wrap_millis()
Definition hal.cpp:106
bool state
Definition fan.h:2
void HOT delay(uint32_t ms)
Definition hal.cpp:85
uint32_t IRAM_ATTR HOT millis()
Definition hal.cpp:28
void arch_restart()
Definition hal.cpp:39
static void uint32_t