ESPHome 2026.1.0-dev
Loading...
Searching...
No Matches
preferences.cpp
Go to the documentation of this file.
1#ifdef USE_ESP32
2
4#include "esphome/core/log.h"
6#include <nvs_flash.h>
7#include <cinttypes>
8#include <cstring>
9#include <memory>
10#include <vector>
11
12namespace esphome {
13namespace esp32 {
14
15static const char *const TAG = "esp32.preferences";
16
17// Buffer size for converting uint32_t to string: max "4294967295" (10 chars) + null terminator + 1 padding
18static constexpr size_t KEY_BUFFER_SIZE = 12;
19
20struct NVSData {
21 uint32_t key;
22 std::unique_ptr<uint8_t[]> data;
23 size_t len;
24
25 void set_data(const uint8_t *src, size_t size) {
26 this->data = std::make_unique<uint8_t[]>(size);
27 memcpy(this->data.get(), src, size);
28 this->len = size;
29 }
30};
31
32static std::vector<NVSData> s_pending_save; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
33
34class ESP32PreferenceBackend : public ESPPreferenceBackend {
35 public:
36 uint32_t key;
37 uint32_t nvs_handle;
38 bool save(const uint8_t *data, size_t len) override {
39 // try find in pending saves and update that
40 for (auto &obj : s_pending_save) {
41 if (obj.key == this->key) {
42 obj.set_data(data, len);
43 return true;
44 }
45 }
46 NVSData save{};
47 save.key = this->key;
48 save.set_data(data, len);
49 s_pending_save.emplace_back(std::move(save));
50 ESP_LOGVV(TAG, "s_pending_save: key: %" PRIu32 ", len: %zu", this->key, len);
51 return true;
52 }
53 bool load(uint8_t *data, size_t len) override {
54 // try find in pending saves and load from that
55 for (auto &obj : s_pending_save) {
56 if (obj.key == this->key) {
57 if (obj.len != len) {
58 // size mismatch
59 return false;
60 }
61 memcpy(data, obj.data.get(), len);
62 return true;
63 }
64 }
65
66 char key_str[KEY_BUFFER_SIZE];
67 snprintf(key_str, sizeof(key_str), "%" PRIu32, this->key);
68 size_t actual_len;
69 esp_err_t err = nvs_get_blob(this->nvs_handle, key_str, nullptr, &actual_len);
70 if (err != 0) {
71 ESP_LOGV(TAG, "nvs_get_blob('%s'): %s - the key might not be set yet", key_str, esp_err_to_name(err));
72 return false;
73 }
74 if (actual_len != len) {
75 ESP_LOGVV(TAG, "NVS length does not match (%zu!=%zu)", actual_len, len);
76 return false;
77 }
78 err = nvs_get_blob(this->nvs_handle, key_str, data, &len);
79 if (err != 0) {
80 ESP_LOGV(TAG, "nvs_get_blob('%s') failed: %s", key_str, esp_err_to_name(err));
81 return false;
82 } else {
83 ESP_LOGVV(TAG, "nvs_get_blob: key: %s, len: %zu", key_str, len);
84 }
85 return true;
86 }
87};
88
89class ESP32Preferences : public ESPPreferences {
90 public:
91 uint32_t nvs_handle;
92
93 void open() {
94 nvs_flash_init();
95 esp_err_t err = nvs_open("esphome", NVS_READWRITE, &nvs_handle);
96 if (err == 0)
97 return;
98
99 ESP_LOGW(TAG, "nvs_open failed: %s - erasing NVS", esp_err_to_name(err));
100 nvs_flash_deinit();
101 nvs_flash_erase();
102 nvs_flash_init();
103
104 err = nvs_open("esphome", NVS_READWRITE, &nvs_handle);
105 if (err != 0) {
106 nvs_handle = 0;
107 }
108 }
109 ESPPreferenceObject make_preference(size_t length, uint32_t type, bool in_flash) override {
110 return this->make_preference(length, type);
111 }
112 ESPPreferenceObject make_preference(size_t length, uint32_t type) override {
113 auto *pref = new ESP32PreferenceBackend(); // NOLINT(cppcoreguidelines-owning-memory)
114 pref->nvs_handle = this->nvs_handle;
115 pref->key = type;
116
117 return ESPPreferenceObject(pref);
118 }
119
120 bool sync() override {
121 if (s_pending_save.empty())
122 return true;
123
124 ESP_LOGV(TAG, "Saving %zu items...", s_pending_save.size());
125 // goal try write all pending saves even if one fails
126 int cached = 0, written = 0, failed = 0;
127 esp_err_t last_err = ESP_OK;
128 uint32_t last_key = 0;
129
130 // go through vector from back to front (makes erase easier/more efficient)
131 for (ssize_t i = s_pending_save.size() - 1; i >= 0; i--) {
132 const auto &save = s_pending_save[i];
133 char key_str[KEY_BUFFER_SIZE];
134 snprintf(key_str, sizeof(key_str), "%" PRIu32, save.key);
135 ESP_LOGVV(TAG, "Checking if NVS data %s has changed", key_str);
136 if (this->is_changed_(this->nvs_handle, save, key_str)) {
137 esp_err_t err = nvs_set_blob(this->nvs_handle, key_str, save.data.get(), save.len);
138 ESP_LOGV(TAG, "sync: key: %s, len: %zu", key_str, save.len);
139 if (err != 0) {
140 ESP_LOGV(TAG, "nvs_set_blob('%s', len=%zu) failed: %s", key_str, save.len, esp_err_to_name(err));
141 failed++;
142 last_err = err;
143 last_key = save.key;
144 continue;
145 }
146 written++;
147 } else {
148 ESP_LOGV(TAG, "NVS data not changed skipping %" PRIu32 " len=%zu", save.key, save.len);
149 cached++;
150 }
151 s_pending_save.erase(s_pending_save.begin() + i);
152 }
153 ESP_LOGD(TAG, "Writing %d items: %d cached, %d written, %d failed", cached + written + failed, cached, written,
154 failed);
155 if (failed > 0) {
156 ESP_LOGE(TAG, "Writing %d items failed. Last error=%s for key=%" PRIu32, failed, esp_err_to_name(last_err),
157 last_key);
158 }
159
160 // note: commit on esp-idf currently is a no-op, nvs_set_blob always writes
161 esp_err_t err = nvs_commit(this->nvs_handle);
162 if (err != 0) {
163 ESP_LOGV(TAG, "nvs_commit() failed: %s", esp_err_to_name(err));
164 return false;
165 }
166
167 return failed == 0;
168 }
169
170 protected:
171 bool is_changed_(uint32_t nvs_handle, const NVSData &to_save, const char *key_str) {
172 size_t actual_len;
173 esp_err_t err = nvs_get_blob(nvs_handle, key_str, nullptr, &actual_len);
174 if (err != 0) {
175 ESP_LOGV(TAG, "nvs_get_blob('%s'): %s - the key might not be set yet", key_str, esp_err_to_name(err));
176 return true;
177 }
178 // Check size first before allocating memory
179 if (actual_len != to_save.len) {
180 return true;
181 }
182 auto stored_data = std::make_unique<uint8_t[]>(actual_len);
183 err = nvs_get_blob(nvs_handle, key_str, stored_data.get(), &actual_len);
184 if (err != 0) {
185 ESP_LOGV(TAG, "nvs_get_blob('%s') failed: %s", key_str, esp_err_to_name(err));
186 return true;
187 }
188 return memcmp(to_save.data.get(), stored_data.get(), to_save.len) != 0;
189 }
190
191 bool reset() override {
192 ESP_LOGD(TAG, "Erasing storage");
193 s_pending_save.clear();
194
195 nvs_flash_deinit();
196 nvs_flash_erase();
197 // Make the handle invalid to prevent any saves until restart
198 nvs_handle = 0;
199 return true;
200 }
201};
202
204 auto *prefs = new ESP32Preferences(); // NOLINT(cppcoreguidelines-owning-memory)
205 prefs->open();
206 global_preferences = prefs;
207}
208
209} // namespace esp32
210
211ESPPreferences *global_preferences; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
212
213} // namespace esphome
214
215#endif // USE_ESP32
uint16_t type
__int64 ssize_t
Definition httplib.h:178
void setup_preferences()
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:533
ESPPreferences * global_preferences
uint16_t length
Definition tt21100.cpp:0