ESPHome 2026.3.0-dev
Loading...
Searching...
No Matches
task_log_buffer_zephyr.cpp
Go to the documentation of this file.
1#ifdef USE_ZEPHYR
2
4
5namespace esphome::logger {
6
7__thread bool non_main_task_recursion_guard_; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
8
9#ifdef USE_ESPHOME_TASK_LOG_BUFFER
10
11static inline uint32_t total_size_in_32bit_words(uint16_t text_length) {
12 // Calculate total size in 32-bit words needed (header + text length + null terminator + 3(4 bytes alignment)
13 return (sizeof(TaskLogBuffer::LogMessage) + text_length + 1 + 3) / sizeof(uint32_t);
14}
15
16static inline uint32_t get_wlen(const mpsc_pbuf_generic *item) {
17 return total_size_in_32bit_words(reinterpret_cast<const TaskLogBuffer::LogMessage *>(item)->text_length);
18}
19
20TaskLogBuffer::TaskLogBuffer(size_t total_buffer_size) {
21 // alignment to 4 bytes
22 total_buffer_size = (total_buffer_size + 3) / sizeof(uint32_t);
23 this->mpsc_config_.buf = new uint32_t[total_buffer_size];
24 this->mpsc_config_.size = total_buffer_size;
25 this->mpsc_config_.flags = MPSC_PBUF_MODE_OVERWRITE;
26 this->mpsc_config_.get_wlen = get_wlen,
27
28 mpsc_pbuf_init(&this->log_buffer_, &this->mpsc_config_);
29}
30
31TaskLogBuffer::~TaskLogBuffer() { delete[] this->mpsc_config_.buf; }
32
33bool TaskLogBuffer::send_message_thread_safe(uint8_t level, const char *tag, uint16_t line, const char *thread_name,
34 const char *format, va_list args) {
35 // First, calculate the exact length needed using a null buffer (no actual writing)
36 va_list args_copy;
37 va_copy(args_copy, args);
38 int ret = vsnprintf(nullptr, 0, format, args_copy);
39 va_end(args_copy);
40
41 if (ret <= 0) {
42 return false; // Formatting error or empty message
43 }
44
45 // Calculate actual text length (capped to maximum size)
46 static constexpr size_t MAX_TEXT_SIZE = 255;
47 size_t text_length = (static_cast<size_t>(ret) > MAX_TEXT_SIZE) ? MAX_TEXT_SIZE : ret;
48 size_t total_size = total_size_in_32bit_words(text_length);
49 auto *msg = reinterpret_cast<LogMessage *>(mpsc_pbuf_alloc(&this->log_buffer_, total_size, K_NO_WAIT));
50 if (msg == nullptr) {
51 return false;
52 }
53 msg->level = level;
54 msg->tag = tag;
55 msg->line = line;
56 strncpy(msg->thread_name, thread_name, sizeof(msg->thread_name) - 1);
57 msg->thread_name[sizeof(msg->thread_name) - 1] = '\0'; // Ensure null termination
58
59 // Format the message text directly into the acquired memory
60 // We add 1 to text_length to ensure space for null terminator during formatting
61 char *text_area = msg->text_data();
62 ret = vsnprintf(text_area, text_length + 1, format, args);
63
64 // Handle unexpected formatting error (ret < 0 is encoding error; ret == 0 is valid empty output)
65 if (ret < 0) {
66 // this should not happen, vsnprintf was called already once
67 // fill with '\n' to not call mpsc_pbuf_free from producer
68 // it will be trimmed anyway
69 for (size_t i = 0; i < text_length; ++i) {
70 text_area[i] = '\n';
71 }
72 text_area[text_length] = 0;
73 // do not return false to free the buffer from main thread
74 }
75
76 msg->text_length = text_length;
77
78 mpsc_pbuf_commit(&this->log_buffer_, reinterpret_cast<mpsc_pbuf_generic *>(msg));
79 return true;
80}
81
82bool TaskLogBuffer::borrow_message_main_loop(LogMessage *&message, uint16_t &text_length) {
83 if (this->current_token_) {
84 return false;
85 }
86
87 this->current_token_ = mpsc_pbuf_claim(&this->log_buffer_);
88
89 if (this->current_token_ == nullptr) {
90 return false;
91 }
92
93 // we claimed buffer already, const_cast is safe here
94 message = const_cast<LogMessage *>(reinterpret_cast<const LogMessage *>(this->current_token_));
95
96 text_length = message->text_length;
97 // Remove trailing newlines
98 while (text_length > 0 && message->text_data()[text_length - 1] == '\n') {
99 text_length--;
100 }
101
102 return true;
103}
104
106 if (this->current_token_ == nullptr) {
107 return;
108 }
109 mpsc_pbuf_free(&this->log_buffer_, this->current_token_);
110 this->current_token_ = nullptr;
111}
112#endif // USE_ESPHOME_TASK_LOG_BUFFER
113
114} // namespace esphome::logger
115
116#endif // USE_ZEPHYR
TaskLogBuffer(size_t total_buffer_size)
bool borrow_message_main_loop(LogMessage *&message, uint16_t &text_length)
bool send_message_thread_safe(uint8_t level, const char *tag, uint16_t line, const char *thread_name, const char *format, va_list args)
const char * message
Definition component.cpp:38
__thread bool non_main_task_recursion_guard_
va_end(args)