34#include <esp_heap_caps.h>
38#include <freertos/FreeRTOS.h>
39#include <freertos/semphr.h>
40#elif defined(USE_LIBRETINY)
49#define HOT __attribute__((hot))
50#define ESPDEPRECATED(msg, when) __attribute__((deprecated(msg)))
51#define ESPHOME_ALWAYS_INLINE __attribute__((always_inline))
52#define PACKED __attribute__((packed))
64using std::is_trivially_copyable;
65using std::make_unique;
66using std::enable_if_t;
68using std::is_invocable;
69#if __cpp_lib_bit_cast >= 201806
74 typename To,
typename From,
75 enable_if_t<
sizeof(To) ==
sizeof(From) && is_trivially_copyable<From>::value && is_trivially_copyable<To>::value,
79 memcpy(&dst, &src,
sizeof(To));
85inline float lerp(
float completion,
float start,
float end) =
delete;
89template<
typename T>
constexpr T
byteswap(T n) {
91 for (
size_t i = 0; i <
sizeof(T); i++)
92 reinterpret_cast<uint8_t *
>(&
m)[i] =
reinterpret_cast<uint8_t *
>(&n)[
sizeof(T) - 1 - i];
95template<>
constexpr uint8_t
byteswap(uint8_t n) {
return n; }
98template<>
inline uint16_t
byteswap(uint16_t n) {
return __builtin_bswap16(n); }
100template<>
inline uint64_t
byteswap(uint64_t n) {
return __builtin_bswap64(n); }
101template<>
inline int8_t
byteswap(int8_t n) {
return n; }
102template<>
inline int16_t
byteswap(int16_t n) {
return __builtin_bswap16(n); }
103template<>
inline int32_t
byteswap(int32_t n) {
return __builtin_bswap32(n); }
104template<>
inline int64_t
byteswap(int64_t n) {
return __builtin_bswap64(n); }
106template<>
constexpr uint16_t
byteswap(uint16_t n) {
return __builtin_bswap16(n); }
108template<>
constexpr uint64_t
byteswap(uint64_t n) {
return __builtin_bswap64(n); }
109template<>
constexpr int8_t
byteswap(int8_t n) {
return n; }
110template<>
constexpr int16_t
byteswap(int16_t n) {
return __builtin_bswap16(n); }
111template<>
constexpr int32_t
byteswap(int32_t n) {
return __builtin_bswap32(n); }
112template<>
constexpr int64_t
byteswap(int64_t n) {
return __builtin_bswap64(n); }
145 delete[] this->
heap_;
150 if (other.is_inline_()) {
151 memcpy(this->
inline_, other.inline_, this->len_);
153 this->
heap_ = other.heap_;
154 other.heap_ =
nullptr;
161 if (
this != &other) {
163 delete[] this->
heap_;
164 this->
len_ = other.len_;
165 if (other.is_inline_()) {
166 memcpy(this->
inline_, other.inline_, this->len_);
168 this->
heap_ = other.heap_;
169 other.heap_ =
nullptr;
184 delete[] this->
heap_;
185 this->
heap_ =
nullptr;
188 if (size > InlineSize && (this->
is_inline_() || size != this->len_)) {
213 using iterator =
typename std::array<T, N>::iterator;
219 std::array<T, N> data_;
228 while (first != last && count_ < N) {
229 data_[count_++] = *first++;
235 for (
const auto &
val : init) {
238 data_[count_++] =
val;
245 data_[count_++] = value;
253 template<
typename InputIt>
void assign(InputIt first, InputIt last) {
255 while (first != last && count_ < N) {
256 data_[count_++] = *first++;
267 return data_[count_++];
270 size_t size()
const {
return count_; }
271 bool empty()
const {
return count_ == 0; }
274 T *
data() {
return data_.data(); }
275 const T *
data()
const {
return data_.data(); }
293 operator std::span<T>() {
return std::span<T>(data_.data(), count_); }
294 operator std::span<const T>()
const {
return std::span<const T>(data_.data(), count_); }
305 using index_type = std::conditional_t<(N <= std::numeric_limits<uint8_t>::max()), uint8_t, uint16_t>;
383template<typename T, size_t MAX_CAPACITY = std::numeric_limits<uint16_t>::max()>
class FixedRingBuffer {
384 using index_type = std::conditional_t<
385 (MAX_CAPACITY <= std::numeric_limits<uint8_t>::max()), uint8_t,
386 std::conditional_t<(MAX_CAPACITY <= std::numeric_limits<uint16_t>::max()), uint16_t,
uint32_t>>;
421 if constexpr (std::is_trivially_copyable<T>::value && std::is_trivially_default_constructible<T>::value) {
422 ::operator
delete(this->
data_);
424 delete[] this->
data_;
434 if constexpr (std::is_trivially_copyable<T>::value && std::is_trivially_default_constructible<T>::value) {
437 this->
data_ =
static_cast<T *
>(::operator
new(
capacity *
sizeof(T)));
505template<
typename T,
size_t N>
inline void init_array_from(std::array<T, N> &dest, std::initializer_list<T> src) {
507 assert(src.size() == N);
509 if constexpr (std::is_trivially_copyable_v<T>) {
510 __builtin_memcpy(dest.data(), src.begin(), N *
sizeof(T));
513 for (
const auto &v : src) {
529 void destroy_elements_() {
531 if constexpr (!std::is_trivially_destructible<T>::value) {
532 for (
size_t i = 0; i < size_; i++) {
540 if (data_ !=
nullptr) {
543 ::operator
delete(data_);
555 void assign_from_initializer_list_(std::initializer_list<T> init_list) {
556 init(init_list.size());
558 for (
const auto &item : init_list) {
559 new (data_ + idx) T(item);
562 size_ = init_list.size();
570 FixedVector(std::initializer_list<T> init_list) { assign_from_initializer_list_(init_list); }
584 operator std::vector<T>()
const {
return {data_, data_ + size_}; }
587 if (
this != &other) {
593 capacity_ = other.capacity_;
605 assign_from_initializer_list_(init_list);
619 data_ =
static_cast<T *
>(::operator
new(n *
sizeof(T)));
640 if (size_ < capacity_) {
642 new (&data_[size_]) T(value);
651 if (size_ < capacity_) {
653 new (&data_[size_]) T(std::move(value));
664 new (&data_[size_]) T(std::forward<Args>(
args)...);
666 return data_[size_ - 1];
672 const T &
front()
const {
return data_[0]; }
676 T &
back() {
return data_[size_ - 1]; }
677 const T &
back()
const {
return data_[size_ - 1]; }
679 size_t size()
const {
return size_; }
680 bool empty()
const {
return size_ == 0; }
682 bool full()
const {
return size_ == capacity_; }
691 T &
at(
size_t i) {
return data_[i]; }
692 const T &
at(
size_t i)
const {
return data_[i]; }
696 T *
end() {
return data_ + size_; }
697 const T *
begin()
const {
return data_; }
698 const T *
end()
const {
return data_ + size_; }
709 if (
size <= STACK_SIZE) {
710 this->buffer_ = this->stack_buffer_;
712 this->heap_buffer_ =
new T[
size];
713 this->buffer_ = this->heap_buffer_;
724 T *
get() {
return this->buffer_; }
727 T stack_buffer_[STACK_SIZE];
728 T *heap_buffer_{
nullptr};
743 for (int8_t i = 0; i < exp; i++)
746 for (int8_t i = exp; i < 0; i++)
753template<
typename T,
typename U> T
remap(U value, U min, U max, T min_out, T max_out) {
754 return (value - min) * (max_out - min_out) / (max - min) + min_out;
758uint8_t
crc8(
const uint8_t *data, uint8_t
len, uint8_t crc = 0x00, uint8_t poly = 0x8C,
bool msb_first =
false);
761uint16_t
crc16(
const uint8_t *data, uint16_t
len, uint16_t crc = 0xffff, uint16_t reverse_poly = 0xa001,
762 bool refin =
false,
bool refout =
false);
763uint16_t
crc16be(
const uint8_t *data, uint16_t
len, uint16_t crc = 0, uint16_t poly = 0x1021,
bool refin =
false,
764 bool refout =
false);
778 using UnsignedT = std::make_unsigned_t<T>;
779 UnsignedT uvalue =
static_cast<UnsignedT
>(value);
780 for (
size_t i = 0; i <
sizeof(T); i++) {
782 hash ^= (uvalue >> (i * 8)) & 0xFF;
813 using UnsignedT = std::make_unsigned_t<T>;
814 UnsignedT uvalue =
static_cast<UnsignedT
>(value);
815 for (
size_t i = 0; i <
sizeof(T); i++) {
816 hash ^= (uvalue >> (i * 8)) & 0xFF;
848template<
typename ReturnT = u
int32_t>
inline constexpr ESPHOME_ALWAYS_INLINE ReturnT
micros_to_millis(uint64_t us) {
853 uint64_t
x = us >> 3;
860 return static_cast<ReturnT
>(hi) * q + (adj < lo ? (adj + r) / d + q : adj / d);
882 return (
static_cast<uint16_t
>(msb) << 8) | (
static_cast<uint16_t
>(lsb));
890 return (
static_cast<uint32_t>(byte1) << 24) | (
static_cast<uint32_t>(byte2) << 16) |
895template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
constexpr T
encode_value(
const uint8_t *bytes) {
897 for (
size_t i = 0; i <
sizeof(T); i++) {
904template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
909template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
911 std::array<uint8_t,
sizeof(T)> ret{};
912 for (
size_t i =
sizeof(T); i > 0; i--) {
913 ret[i - 1] =
val & 0xFF;
921 x = ((
x & 0xAA) >> 1) | ((
x & 0x55) << 1);
922 x = ((
x & 0xCC) >> 2) | ((
x & 0x33) << 2);
923 x = ((
x & 0xF0) >> 4) | ((
x & 0x0F) << 4);
932 return (
reverse_bits(
static_cast<uint16_t
>(
x & 0xFFFF)) << 16) |
938#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
947#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
969bool str_startswith(
const std::string &str,
const std::string &start);
988std::string
str_until(
const char *str,
char ch);
990std::string
str_until(
const std::string &str,
char ch);
999constexpr char to_snake_case_char(
char c) {
return (c ==
' ') ?
'_' : (c >=
'A' && c <=
'Z') ? c + (
'a' -
'A') : c; }
1006 return (c ==
'-' || c ==
'_' || (c >=
'0' && c <=
'9') || (c >=
'a' && c <=
'z') || (c >=
'A' && c <=
'Z')) ? c :
'_';
1018char *
str_sanitize_to(
char *buffer,
size_t buffer_size,
const char *str);
1035 for (
size_t i = 0; i <
len; i++) {
1073#define buf_append_printf(buf, size, pos, fmt, ...) buf_append_printf_p(buf, size, pos, PSTR(fmt), ##__VA_ARGS__)
1109 size_t remaining =
size -
pos - 1;
1110 size_t len = strlen(str);
1111 if (
len > remaining) {
1114 memcpy(buf +
pos, str,
len);
1128std::string
make_name_with_suffix(
const std::string &name,
char sep,
const char *suffix_ptr,
size_t suffix_len);
1138std::string
make_name_with_suffix(
const char *name,
size_t name_len,
char sep,
const char *suffix_ptr,
1151 const char *suffix_ptr,
size_t suffix_len);
1159template<
typename T, enable_if_t<(std::is_
integral<T>::value && std::is_
unsigned<T>::value),
int> = 0>
1161 char *
end =
nullptr;
1162 unsigned long value = ::strtoul(str, &
end, 10);
1163 if (
end == str || *
end !=
'\0' || value > std::numeric_limits<T>::max())
1168template<
typename T, enable_if_t<(std::is_
integral<T>::value && std::is_
unsigned<T>::value),
int> = 0>
1173template<
typename T, enable_if_t<(std::is_
integral<T>::value && std::is_
signed<T>::value),
int> = 0>
1175 char *
end =
nullptr;
1176 signed long value = ::strtol(str, &
end, 10);
1177 if (
end == str || *
end !=
'\0' || value < std::numeric_limits<T>::min() || value > std::numeric_limits<T>::max())
1182template<
typename T, enable_if_t<(std::is_
integral<T>::value && std::is_
signed<T>::value),
int> = 0>
1187template<
typename T, enable_if_t<(std::is_same<T,
float>::value),
int> = 0> optional<T>
parse_number(
const char *str) {
1188 char *
end =
nullptr;
1189 float value = ::strtof(str, &
end);
1190 if (
end == str || *
end !=
'\0' || value == HUGE_VALF)
1195template<
typename T, enable_if_t<(std::is_same<T,
float>::value),
int> = 0>
1211size_t parse_hex(
const char *str,
size_t len, uint8_t *data,
size_t count);
1213inline bool parse_hex(
const char *str, uint8_t *data,
size_t count) {
1214 return parse_hex(str, strlen(str), data, count) == 2 * count;
1217inline bool parse_hex(
const std::string &str, uint8_t *data,
size_t count) {
1218 return parse_hex(str.c_str(), str.length(), data, count) == 2 * count;
1221inline bool parse_hex(
const char *str, std::vector<uint8_t> &data,
size_t count) {
1223 return parse_hex(str, strlen(str), data.data(), count) == 2 * count;
1226inline bool parse_hex(
const std::string &str, std::vector<uint8_t> &data,
size_t count) {
1228 return parse_hex(str.c_str(), str.length(), data.data(), count) == 2 * count;
1235template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
1238 if (
len > 2 *
sizeof(T) ||
parse_hex(str,
len,
reinterpret_cast<uint8_t *
>(&
val),
sizeof(T)) == 0)
1243template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0> optional<T>
parse_hex(
const char *str) {
1247template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0> optional<T>
parse_hex(
const std::string &str) {
1253static constexpr uint8_t INVALID_HEX_CHAR = 255;
1256 if (c >=
'0' && c <=
'9')
1258 if (c >=
'A' && c <=
'F')
1259 return c -
'A' + 10;
1260 if (c >=
'a' && c <=
'f')
1261 return c -
'a' + 10;
1262 return INVALID_HEX_CHAR;
1266ESPHOME_ALWAYS_INLINE
inline char format_hex_char(uint8_t v,
char base) {
return v >= 10 ? base + (v - 10) :
'0' + v; }
1286 int32_t tens = v / 10;
1287 *buf++ =
'0' + tens;
1289 }
else if (v >= 10) {
1290 int32_t tens = v / 10;
1291 *buf++ =
'0' + tens;
1299static constexpr size_t UINT32_MAX_STR_SIZE = 11;
1310 return static_cast<size_t>(
end - buf.data());
1319 static_assert(N >= 3,
"Buffer must hold at least one hex byte (3 chars)");
1324template<size_t N, typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
1326 static_assert(N >=
sizeof(T) * 2 + 1,
"Buffer too small for type");
1328 return format_hex_to(buffer,
reinterpret_cast<const uint8_t *
>(&
val),
sizeof(T));
1332template<
size_t N>
inline char *
format_hex_to(
char (&buffer)[N],
const std::vector<uint8_t> &data) {
1337template<
size_t N,
size_t M>
inline char *
format_hex_to(
char (&buffer)[N],
const std::array<uint8_t, M> &data) {
1348template<size_t N, typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
1350 static_assert(N >=
sizeof(T) * 2 + 3,
"Buffer too small for prefixed hex");
1354 format_hex_to(buffer + 2, N - 2,
reinterpret_cast<const uint8_t *
>(&
val),
sizeof(T));
1360 static_assert(N >= 5,
"Buffer must hold at least '0x' + one hex byte + null");
1386 static_assert(N >= 3,
"Buffer must hold at least one hex byte");
1397template<
size_t N,
size_t M>
1423 static_assert(N >= 5,
"Buffer must hold at least one hex uint16_t");
1428static constexpr size_t MAC_ADDRESS_SIZE = 6;
1432static constexpr size_t MAC_ADDRESS_BUFFER_SIZE = MAC_ADDRESS_SIZE * 2 + 1;
1436 return format_hex_pretty_to(output, MAC_ADDRESS_PRETTY_BUFFER_SIZE, mac, MAC_ADDRESS_SIZE,
':');
1441 format_hex_to(output, MAC_ADDRESS_BUFFER_SIZE, mac, MAC_ADDRESS_SIZE);
1455std::string
format_hex(
const std::vector<uint8_t> &data);
1459template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0> std::string
format_hex(T
val) {
1461 return format_hex(
reinterpret_cast<uint8_t *
>(&
val),
sizeof(T));
1466template<std::
size_t N> std::string
format_hex(
const std::array<uint8_t, N> &data) {
1498std::string
format_hex_pretty(
const uint8_t *data,
size_t length,
char separator =
'.',
bool show_length =
true);
1523std::string
format_hex_pretty(
const uint16_t *data,
size_t length,
char separator =
'.',
bool show_length =
true);
1549std::string
format_hex_pretty(
const std::vector<uint8_t> &data,
char separator =
'.',
bool show_length =
true);
1574std::string
format_hex_pretty(
const std::vector<uint16_t> &data,
char separator =
'.',
bool show_length =
true);
1599std::string
format_hex_pretty(
const std::string &data,
char separator =
'.',
bool show_length =
true);
1627template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
1630 return format_hex_pretty(
reinterpret_cast<uint8_t *
>(&
val),
sizeof(T), separator, show_length);
1659 static_assert(N >= 9,
"Buffer must hold at least one binary byte (9 chars)");
1679template<size_t N, typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0>
1681 static_assert(N >=
sizeof(T) * 8 + 1,
"Buffer too small for type");
1683 return format_bin_to(buffer,
reinterpret_cast<const uint8_t *
>(&
val),
sizeof(T));
1693template<typename T, enable_if_t<std::is_unsigned<T>::value,
int> = 0> std::string
format_bin(T
val) {
1695 return format_bin(
reinterpret_cast<uint8_t *
>(&
val),
sizeof(T));
1709ESPDEPRECATED(
"Allocates heap memory. Use value_accuracy_to_buf() instead. Removed in 2026.7.0.",
"2026.1.0")
1713static constexpr
size_t VALUE_ACCURACY_MAX_LEN = 64;
1716size_t value_accuracy_to_buf(std::span<
char, VALUE_ACCURACY_MAX_LEN> buf,
float value, int8_t accuracy_decimals);
1719 int8_t accuracy_decimals, StringRef unit_of_measurement);
1724std::
string base64_encode(const uint8_t *buf,
size_t buf_len);
1727std::vector<uint8_t>
base64_decode(const std::
string &encoded_string);
1728size_t base64_decode(std::
string const &encoded_string, uint8_t *buf,
size_t buf_len);
1729size_t base64_decode(const uint8_t *encoded_data,
size_t encoded_len, uint8_t *buf,
size_t buf_len);
1744ESPDEPRECATED("Use LightState::gamma_correct_lut() instead. Removed in 2026.9.0.", "2026.3.0")
1748ESPDEPRECATED("Use LightState::gamma_uncorrect_lut() instead. Removed in 2026.9.0.", "2026.3.0")
1752void rgb_to_hsv(
float red,
float green,
float blue,
int &hue,
float &saturation,
float &value);
1754void hsv_to_rgb(
int hue,
float saturation,
float value,
float &red,
float &green,
float &blue);
1782 static_assert(
sizeof(
void *) ==
sizeof(std::uintptr_t),
"void* must be the same size as uintptr_t");
1784 void (*fn_)(
void *, Ts...){
nullptr};
1785 void *ctx_{
nullptr};
1793 using DecayF = std::decay_t<F>;
1794 if constexpr (
sizeof(DecayF) <=
sizeof(
void *) && std::is_trivially_copyable_v<DecayF>) {
1801 DecayF decayed = std::forward<F>(callable);
1802 __builtin_memcpy(&cb.ctx_, &decayed,
sizeof(DecayF));
1803 cb.fn_ = [](
void *c, Ts...
args) {
1804 alignas(DecayF)
char buf[
sizeof(DecayF)];
1805 __builtin_memcpy(buf, &c,
sizeof(DecayF));
1806 (*std::launder(
reinterpret_cast<DecayF *
>(buf)))(
args...);
1813 auto *stored =
new DecayF(std::forward<F>(callable));
1814 return {[](
void *c, Ts...
args) { (*
static_cast<DecayF *
>(c))(
args...); },
static_cast<void *
>(stored)};
1837 static_assert(std::is_trivially_copyable_v<CbType>,
"Callback must be trivially copyable");
1847 : data_(other.data_), size_(other.size_), capacity_(other.capacity_) {
1848 other.data_ =
nullptr;
1850 other.capacity_ = 0;
1853 std::swap(this->data_, other.data_);
1854 std::swap(this->size_, other.size_);
1855 std::swap(this->capacity_, other.capacity_);
1861 template<
typename F>
void add(F &&callback) { this->add_(CbType::create(std::forward<F>(callback))); }
1865 if (this->size_ != 0) {
1866 for (
auto *it = this->data_, *
end = it + this->size_; it !=
end; ++it) {
1871 uint16_t
size()
const {
return this->size_; }
1881 if (this->size_ == this->capacity_) {
1885 this->data_[this->size_++] = cb;
1889 uint16_t capacity_{0};
1906 template<
typename F>
void add(F &&callback) { this->add_(
Callback<
void(Ts...)>::create(std::forward<F>(callback))); }
1910 for (
auto &cb : this->callbacks_)
1913 size_t size()
const {
return this->callbacks_.size(); }
1955 template<
typename F>
void add(F &&callback) { this->add_(
Callback<
void(Ts...)>::create(std::forward<F>(callback))); }
1959 if (this->callbacks_) {
1960 this->callbacks_->call(
args...);
1965 size_t size()
const {
return this->callbacks_ ? this->callbacks_->size() : 0; }
1968 bool empty()
const {
return !this->callbacks_ || this->callbacks_->size() == 0; }
1976 if (!this->callbacks_) {
1979 this->callbacks_->add_(cb);
2041#if defined(USE_ESP8266) || defined(USE_RP2040)
2048#elif defined(USE_ESP32) || defined(USE_LIBRETINY)
2050 Mutex() { handle_ = xSemaphoreCreateMutex(); }
2052 void lock() { xSemaphoreTake(this->handle_, portMAX_DELAY); }
2053 bool try_lock() {
return xSemaphoreTake(this->handle_, 0) == pdTRUE; }
2054 void unlock() { xSemaphoreGive(this->handle_); }
2057 SemaphoreHandle_t handle_;
2110#if defined(USE_ESP8266) || defined(USE_RP2040) || defined(USE_ZEPHYR)
2128#if defined(USE_ESP32) || defined(USE_RP2040)
2232 size_t size = n * manual_size;
2236 ptr =
static_cast<T *
>(heap_caps_malloc(
size, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT));
2239 ptr =
static_cast<T *
>(heap_caps_malloc(
size, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT));
2243 ptr =
static_cast<T *
>(malloc(
size));
2251 size_t size = n * manual_size;
2255 ptr =
static_cast<T *
>(heap_caps_realloc(p,
size, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT));
2258 ptr =
static_cast<T *
>(heap_caps_realloc(p,
size, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT));
2262 ptr =
static_cast<T *
>(realloc(p,
size));
2276 return ESP.getFreeHeap();
2277#elif defined(USE_ESP32)
2279 this->flags_ &
ALLOC_INTERNAL ? heap_caps_get_free_size(MALLOC_CAP_8BIT | MALLOC_CAP_INTERNAL) : 0;
2281 this->flags_ &
ALLOC_EXTERNAL ? heap_caps_get_free_size(MALLOC_CAP_8BIT | MALLOC_CAP_SPIRAM) : 0;
2282 return max_internal + max_external;
2283#elif defined(USE_RP2040)
2284 return ::rp2040.getFreeHeap();
2285#elif defined(USE_LIBRETINY)
2286 return lt_heap_get_free();
2297 return ESP.getMaxFreeBlockSize();
2298#elif defined(USE_ESP32)
2300 this->flags_ &
ALLOC_INTERNAL ? heap_caps_get_largest_free_block(MALLOC_CAP_8BIT | MALLOC_CAP_INTERNAL) : 0;
2302 this->flags_ &
ALLOC_EXTERNAL ? heap_caps_get_largest_free_block(MALLOC_CAP_8BIT | MALLOC_CAP_SPIRAM) : 0;
2303 return std::max(max_internal, max_external);
2319template<
typename T,
typename U>
2321 { a > b } -> std::convertible_to<bool>;
2322 { a < b } -> std::convertible_to<bool>;
2325template<std::totally_ordered T, comparable_with<T> U> T
clamp_at_least(T value, U min) {
2330template<std::totally_ordered T, comparable_with<T> U> T
clamp_at_most(T value, U max) {
2343template<typename T, enable_if_t<!std::is_pointer<T>::value,
int> = 0> T
id(T value) {
return value; }
2348template<typename T, enable_if_t<std::is_pointer<T *>::value,
int> = 0> T &
id(T *value) {
return *value; }
CallbackManager & operator=(const CallbackManager &)=delete
CallbackManager & operator=(CallbackManager &&other) noexcept
friend class LazyCallbackManager
void operator()(Ts... args)
Call all callbacks in this manager.
CallbackManager()=default
void add(F &&callback)
Add any callable.
CallbackManager(CallbackManager &&other) noexcept
void add_(CbType cb)
Non-template core to avoid code duplication per lambda type.
CallbackManager(const CallbackManager &)=delete
void ESPHOME_ALWAYS_INLINE call(Ts... args)
Call all callbacks in this manager.
Lightweight read-only view over a const array stored in RODATA (will typically be in flash memory) Av...
const constexpr T & operator[](size_t i) const
constexpr bool empty() const
constexpr ConstVector(const T *data, size_t size)
constexpr size_t size() const
Helper class to deduplicate items in a series of values.
bool next(T value)
Feeds the next item in the series to the deduplicator and returns false if this is a duplicate.
bool has_value() const
Returns true if this deduplicator has processed any items.
bool next_unknown()
Returns true if the deduplicator's value was previously known.
const T & operator*() const
bool operator!=(const ConstIterator &other) const
ConstIterator & operator++()
ConstIterator(const FixedRingBuffer *buf, index_type pos)
bool operator!=(const Iterator &other) const
Iterator(FixedRingBuffer *buf, index_type pos)
Fixed-capacity circular buffer - allocates once at runtime, never reallocates.
FixedRingBuffer & operator=(const FixedRingBuffer &)=delete
ConstIterator begin() const
bool push(const T &value)
Push a value. Returns false if full.
index_type capacity() const
void push_overwrite(const T &value)
Push a value, overwriting the oldest if full.
void init(index_type capacity)
Allocate capacity - can only be called once.
void pop()
Remove the oldest element.
void clear()
Clear all elements (reset to empty, keep capacity)
FixedRingBuffer()=default
FixedRingBuffer(const FixedRingBuffer &)=delete
ConstIterator end() const
Fixed-capacity vector - allocates once at runtime, never reallocates This avoids std::vector template...
const T & at(size_t i) const
FixedVector(FixedVector &&other) noexcept
FixedVector(std::initializer_list< T > init_list)
Constructor from initializer list - allocates exact size needed This enables brace initialization: Fi...
FixedVector & operator=(std::initializer_list< T > init_list)
Assignment from initializer list - avoids temporary and move overhead This enables: FixedVector<int> ...
T & front()
Access first element (no bounds checking - matches std::vector behavior) Caller must ensure vector is...
const T & operator[](size_t i) const
T & operator[](size_t i)
Access element without bounds checking (matches std::vector behavior) Caller must ensure index is val...
T & back()
Access last element (no bounds checking - matches std::vector behavior) Caller must ensure vector is ...
FixedVector & operator=(const FixedVector &)=delete
FixedVector(const FixedVector &)=delete
void push_back(T &&value)
Add element by move without bounds checking Caller must ensure sufficient capacity was allocated via ...
T & emplace_back(Args &&...args)
Emplace element without bounds checking - constructs in-place with arguments Caller must ensure suffi...
FixedVector & operator=(FixedVector &&other) noexcept
T & at(size_t i)
Access element with bounds checking (matches std::vector behavior) Note: No exception thrown on out o...
void push_back(const T &value)
Add element without bounds checking Caller must ensure sufficient capacity was allocated via init() S...
Helper class to request loop() to be called as fast as possible.
static bool is_high_frequency()
Check whether the loop is running continuously.
void stop()
Stop running the loop continuously.
static uint8_t num_requests
void start()
Start running the loop continuously.
Helper class to disable interrupts.
LazyCallbackManager & operator=(const LazyCallbackManager &)=delete
LazyCallbackManager(const LazyCallbackManager &)=delete
size_t size() const
Return the number of registered callbacks.
LazyCallbackManager()=default
void add_(Callback< void(Ts...)> cb)
Non-template core to avoid code duplication per lambda type.
void add(F &&callback)
Add any callable. Allocates the underlying CallbackManager on first use.
void operator()(Ts... args)
Call all callbacks in this manager.
LazyCallbackManager & operator=(LazyCallbackManager &&)=delete
~LazyCallbackManager()
Destructor - clean up allocated CallbackManager if any.
void call(Ts... args)
Call all callbacks in this manager. No-op if no callbacks registered.
bool empty() const
Check if any callbacks are registered.
LazyCallbackManager(LazyCallbackManager &&)=delete
Helper class that wraps a mutex with a RAII-style API.
Helper class to lock the lwIP TCPIP core when making lwIP API calls from non-TCPIP threads.
LwIPLock(const LwIPLock &)=delete
LwIPLock & operator=(const LwIPLock &)=delete
Mutex implementation, with API based on the unavailable std::mutex.
Mutex(const Mutex &)=delete
Mutex & operator=(const Mutex &)=delete
Helper class to easily give an object a parent of type T.
T * get_parent() const
Get the parent of this object.
void set_parent(T *parent)
Set the parent of this object.
An STL allocator that uses SPI or internal RAM.
constexpr RAMAllocator(uint8_t flags)
T * reallocate(T *p, size_t n, size_t manual_size)
size_t get_free_heap_size() const
Return the total heap space available via this allocator.
T * reallocate(T *p, size_t n)
void deallocate(T *p, size_t n)
size_t get_max_free_block_size() const
Return the maximum size block this allocator could allocate.
constexpr RAMAllocator(const RAMAllocator< U > &other)
T * allocate(size_t n, size_t manual_size)
constexpr RAMAllocator()=default
Helper class for efficient buffer allocation - uses stack for small sizes, heap for large This is use...
SmallBufferWithHeapFallback(const SmallBufferWithHeapFallback &)=delete
SmallBufferWithHeapFallback & operator=(SmallBufferWithHeapFallback &&)=delete
~SmallBufferWithHeapFallback()
SmallBufferWithHeapFallback & operator=(const SmallBufferWithHeapFallback &)=delete
SmallBufferWithHeapFallback(size_t size)
SmallBufferWithHeapFallback(SmallBufferWithHeapFallback &&)=delete
Small buffer optimization - stores data inline when small, heap-allocates for large data This avoids ...
SmallInlineBuffer()=default
SmallInlineBuffer(const SmallInlineBuffer &)=delete
void set(const uint8_t *src, size_t size)
Set buffer contents, allocating heap if needed.
SmallInlineBuffer & operator=(const SmallInlineBuffer &)=delete
uint8_t inline_[InlineSize]
SmallInlineBuffer & operator=(SmallInlineBuffer &&other) noexcept
const uint8_t * data() const
SmallInlineBuffer(SmallInlineBuffer &&other) noexcept
void add(F &&callback)
Add any callable.
void call(Ts... args)
Call all callbacks in this manager.
void add_(Callback< void(Ts...)> cb)
Non-template core to avoid code duplication per lambda type.
StaticVector< Callback< void(Ts...)>, N > callbacks_
void operator()(Ts... args)
Call all callbacks in this manager.
CallbackManager backed by StaticVector for compile-time-known callback counts.
const T & operator*() const
ConstIterator(const StaticRingBuffer *buf, index_type pos)
ConstIterator & operator++()
bool operator!=(const ConstIterator &other) const
bool operator!=(const Iterator &other) const
Iterator(StaticRingBuffer *buf, index_type pos)
Fixed-size circular buffer with FIFO semantics and iteration support.
bool push(const T &value)
ConstIterator begin() const
ConstIterator end() const
void clear()
Clear all elements (reset to empty)
Minimal static vector - saves memory by avoiding std::vector overhead.
const_reverse_iterator rend() const
reverse_iterator rbegin()
const T & operator[](size_t i) const
void push_back(const T &value)
void assign(InputIt first, InputIt last)
const_reverse_iterator rbegin() const
std::reverse_iterator< const_iterator > const_reverse_iterator
typename std::array< T, N >::iterator iterator
typename std::array< T, N >::const_iterator const_iterator
std::reverse_iterator< iterator > reverse_iterator
const_iterator end() const
StaticVector(InputIt first, InputIt last)
StaticVector(std::initializer_list< T > init)
const_iterator begin() const
struct @65::@66 __attribute__
Wake the main loop task from an ISR. ISR-safe.
Functions to constrain the range of arithmetic values.
Providing packet encoding functions for exchanging data with a remote host.
T clamp_at_most(T value, U max)
bool random_bytes(uint8_t *data, size_t len)
Generate len random bytes using the platform's secure RNG (hardware RNG or OS CSPRNG).
size_t buf_append_str(char *buf, size_t size, size_t pos, const char *str)
Safely append a string to buffer without format parsing, returning new position (capped at size).
constexpr uint32_t fnv1a_hash_extend(uint32_t hash, const char *str)
Extend a FNV-1a hash with additional string data.
float random_float()
Return a random float between 0 and 1.
const char int const __FlashStringHelper * format
ESPHOME_ALWAYS_INLINE char format_hex_char(uint8_t v, char base)
Convert a nibble (0-15) to hex char with specified base ('a' for lowercase, 'A' for uppercase)
float gamma_uncorrect(float value, float gamma)
uint16_t crc16(const uint8_t *data, uint16_t len, uint16_t crc, uint16_t reverse_poly, bool refin, bool refout)
Calculate a CRC-16 checksum of data with size len.
size_t make_name_with_suffix_to(char *buffer, size_t buffer_size, const char *name, size_t name_len, char sep, const char *suffix_ptr, size_t suffix_len)
Zero-allocation version: format name + separator + suffix directly into buffer.
std::string value_accuracy_to_string(float value, int8_t accuracy_decimals)
constexpr T convert_big_endian(T val)
Convert a value between host byte order and big endian (most significant byte first) order.
float gamma_correct(float value, float gamma)
constexpr char to_sanitized_char(char c)
Sanitize a single char: keep alphanumerics, dashes, underscores; replace others with underscore.
bool mac_address_is_valid(const uint8_t *mac)
Check if the MAC address is not all zeros or all ones.
ESPHOME_ALWAYS_INLINE char format_hex_pretty_char(uint8_t v)
Convert a nibble (0-15) to uppercase hex char (used for pretty printing)
void format_mac_addr_lower_no_sep(const uint8_t *mac, char *output)
Format MAC address as xxxxxxxxxxxxxx (lowercase, no separators)
constexpr uint32_t FNV1_OFFSET_BASIS
FNV-1 32-bit offset basis.
void rgb_to_hsv(float red, float green, float blue, int &hue, float &saturation, float &value)
Convert red, green and blue (all 0-1) values to hue (0-360), saturation (0-1) and value (0-1).
std::string format_hex(const uint8_t *data, size_t length)
Format the byte array data of length len in lowercased hex.
uint16_t uint16_t size_t elem_size
size_t value_accuracy_to_buf(std::span< char, VALUE_ACCURACY_MAX_LEN > buf, float value, int8_t accuracy_decimals)
Format value with accuracy to buffer, returns chars written (excluding null)
std::string str_lower_case(const std::string &str)
Convert the string to lower case.
ParseOnOffState parse_on_off(const char *str, const char *on, const char *off)
Parse a string that contains either on, off or toggle.
const char int const __FlashStringHelper va_list args
std::string format_bin(const uint8_t *data, size_t length)
Format the byte array data of length len in binary.
constexpr T convert_little_endian(T val)
Convert a value between host byte order and little endian (least significant byte first) order.
std::string str_sanitize(const std::string &str)
Sanitizes the input string by removing all characters but alphanumerics, dashes and underscores.
constexpr uint32_t fnv1_hash_extend(uint32_t hash, T value)
Extend a FNV-1 hash with an integer (hashes each byte).
bool base64_decode_int32_vector(const std::string &base64, std::vector< int32_t > &out)
Decode base64/base64url string directly into vector of little-endian int32 values.
constexpr size_t format_hex_prefixed_size(size_t byte_count)
Calculate buffer size needed for format_hex_prefixed_to: "0xXXXXXXXX...\0" = bytes * 2 + 3.
constexpr uint32_t encode_uint24(uint8_t byte1, uint8_t byte2, uint8_t byte3)
Encode a 24-bit value given three bytes in most to least significant byte order.
char * format_hex_prefixed_to(char(&buffer)[N], T val)
Format an unsigned integer as "0x" prefixed lowercase hex to buffer.
bool has_custom_mac_address()
Check if a custom MAC address is set (ESP32 & variants)
size_t parse_hex(const char *str, size_t length, uint8_t *data, size_t count)
Parse bytes from a hex-encoded string into a byte array.
char * format_hex_pretty_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length, char separator)
Format byte array as uppercase hex to buffer (base implementation).
size_t uint32_to_str(std::span< char, UINT32_MAX_STR_SIZE > buf, uint32_t val)
Write unsigned 32-bit integer to buffer with compile-time size check.
uint32_t fnv1_hash_object_id(const char *str, size_t len)
Calculate FNV-1 hash of a string while applying snake_case + sanitize transformations.
uint32_t fnv1_hash(const char *str)
Calculate a FNV-1 hash of str.
T clamp_at_least(T value, U min)
optional< T > parse_number(const char *str)
Parse an unsigned decimal number from a null-terminated string.
std::string get_mac_address_pretty()
Get the device MAC address as a string, in colon-separated uppercase hex notation.
std::string str_snprintf(const char *fmt, size_t len,...)
void set_mac_address(uint8_t *mac)
Set the MAC address to use from the provided byte array (6 bytes).
int8_t step_to_accuracy_decimals(float step)
Derive accuracy in decimals from an increment step.
uint32_t random_uint32()
Return a random 32-bit unsigned integer.
const char * get_mac_address_pretty_into_buffer(std::span< char, MAC_ADDRESS_PRETTY_BUFFER_SIZE > buf)
Get the device MAC address into the given buffer, in colon-separated uppercase hex notation.
void IRAM_ATTR HOT delay_microseconds_safe(uint32_t us)
Delay for the given amount of microseconds, possibly yielding to other processes during the wait.
std::string str_upper_case(const std::string &str)
Convert the string to upper case.
std::string format_hex_pretty(const uint8_t *data, size_t length, char separator, bool show_length)
Format a byte array in pretty-printed, human-readable hex format.
constexpr size_t format_hex_size(size_t byte_count)
Calculate buffer size needed for format_hex_to: "XXXXXXXX...\0" = bytes * 2 + 1.
bool str_equals_case_insensitive(const std::string &a, const std::string &b)
Compare strings for equality in case-insensitive manner.
std::string str_until(const char *str, char ch)
Extract the part of the string until either the first occurrence of the specified character,...
bool str_endswith_ignore_case(const char *str, size_t str_len, const char *suffix, size_t suffix_len)
Case-insensitive check if string ends with suffix (no heap allocation).
constexpr ESPHOME_ALWAYS_INLINE ReturnT micros_to_millis(uint64_t us)
Convert a 64-bit microsecond count to milliseconds without calling __udivdi3 (software 64-bit divide,...
char * str_sanitize_to(char *buffer, size_t buffer_size, const char *str)
Sanitize a string to buffer, keeping only alphanumerics, dashes, and underscores.
void init_array_from(std::array< T, N > &dest, std::initializer_list< T > src)
Initialize a std::array from an initializer_list.
char * int8_to_str(char *buf, int8_t val)
Write int8 value to buffer without modulo operations.
std::string format_mac_address_pretty(const uint8_t *mac)
size_t value_accuracy_with_uom_to_buf(std::span< char, VALUE_ACCURACY_MAX_LEN > buf, float value, int8_t accuracy_decimals, StringRef unit_of_measurement)
Format value with accuracy and UOM to buffer, returns chars written (excluding null)
char * uint32_to_str_unchecked(char *buf, uint32_t val)
Write unsigned 32-bit integer to buffer (internal, no size check).
constexpr size_t format_hex_pretty_size(size_t byte_count)
Calculate buffer size needed for format_hex_pretty_to with separator: "XX:XX:...:XX\0".
std::string base64_encode(const std::vector< uint8_t > &buf)
constexpr uint32_t FNV1_PRIME
FNV-1 32-bit prime.
constexpr T encode_value(const uint8_t *bytes)
Encode a value from its constituent bytes (from most to least significant) in an array with length si...
void hsv_to_rgb(int hue, float saturation, float value, float &red, float &green, float &blue)
Convert hue (0-360), saturation (0-1) and value (0-1) to red, green and blue (all 0-1).
void get_mac_address_into_buffer(std::span< char, MAC_ADDRESS_BUFFER_SIZE > buf)
Get the device MAC address into the given buffer, in lowercase hex notation.
uint16_t crc16be(const uint8_t *data, uint16_t len, uint16_t crc, uint16_t poly, bool refin, bool refout)
constexpr uint32_t encode_uint32(uint8_t byte1, uint8_t byte2, uint8_t byte3, uint8_t byte4)
Encode a 32-bit value given four bytes in most to least significant byte order.
uint8_t crc8(const uint8_t *data, uint8_t len, uint8_t crc, uint8_t poly, bool msb_first)
Calculate a CRC-8 checksum of data with size len.
constexpr float celsius_to_fahrenheit(float value)
Convert degrees Celsius to degrees Fahrenheit.
std::string str_sprintf(const char *fmt,...)
size_t size_t const char va_start(args, fmt)
constexpr uint16_t encode_uint16(uint8_t msb, uint8_t lsb)
Encode a 16-bit value given the most and least significant byte.
void get_mac_address_raw(uint8_t *mac)
Get the device MAC address as raw bytes, written into the provided byte array (6 bytes).
size_t size_t const char * fmt
constexpr uint8_t parse_hex_char(char c)
bool str_startswith(const std::string &str, const std::string &start)
Check whether a string starts with a value.
constexpr std::array< uint8_t, sizeof(T)> decode_value(T val)
Decode a value into its constituent bytes (from most to least significant).
std::string get_mac_address()
Get the device MAC address as a string, in lowercase hex notation.
constexpr size_t format_bin_size(size_t byte_count)
Calculate buffer size needed for format_bin_to: "01234567...\0" = bytes * 8 + 1.
struct ESPDEPRECATED("Use std::index_sequence instead. Removed in 2026.6.0", "2025.12.0") seq
To bit_cast(const From &src)
Convert data between types, without aliasing issues or undefined behaviour.
constexpr char to_snake_case_char(char c)
Convert a single char to snake_case: lowercase and space to underscore.
constexpr float fahrenheit_to_celsius(float value)
Convert degrees Fahrenheit to degrees Celsius.
uint8_t reverse_bits(uint8_t x)
Reverse the order of 8 bits.
char * format_hex_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length)
Format byte array as lowercase hex to buffer (base implementation).
std::string str_snake_case(const std::string &str)
Convert the string to snake case (lowercase with underscores).
float lerp(float completion, float start, float end)=delete
constexpr size_t format_hex_pretty_uint16_size(size_t count)
Calculate buffer size needed for format_hex_pretty_to with uint16_t data: "XXXX:XXXX:....
T remap(U value, U min, U max, T min_out, T max_out)
Remap value from the range (min, max) to (min_out, max_out).
void * callback_manager_grow(void *data, uint16_t size, uint16_t &capacity, size_t elem_size)
Grow a CallbackManager's backing array to exactly size+1. Defined in helpers.cpp.
bool str_endswith(const std::string &str, const std::string &end)
Check whether a string ends with a value.
constexpr uint32_t fnv1a_hash(const char *str)
Calculate a FNV-1a hash of str.
std::string size_t std::string size_t buf_append_printf_p(char *buf, size_t size, size_t pos, PGM_P fmt,...)
Safely append formatted string to buffer, returning new position (capped at size).
size_t base64_decode(const std::string &encoded_string, uint8_t *buf, size_t buf_len)
uint16_t uint16_t & capacity
ParseOnOffState
Return values for parse_on_off().
float pow10_int(int8_t exp)
Compute 10^exp using iterative multiplication/division.
std::string make_name_with_suffix(const char *name, size_t name_len, char sep, const char *suffix_ptr, size_t suffix_len)
Optimized string concatenation: name + separator + suffix (const char* overload) Uses a fixed stack b...
char * format_bin_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length)
Format byte array as binary string to buffer.
char * format_mac_addr_upper(const uint8_t *mac, char *output)
Format MAC address as XX:XX:XX:XX:XX:XX (uppercase, colon separators)
std::string str_truncate(const std::string &str, size_t length)
Truncate a string to a specific length.
static Callback create(F &&callable)
Create from any callable.
void call(Ts... args) const
Invoke the callback. Only valid on Callbacks created via create(), never on default-constructed insta...
Lightweight type-erased callback (8 bytes on 32-bit) that avoids std::function overhead.