Работа с EEPROM памятью
Вот и добрались мы до третьего типа памяти, доступного на Arduino: EEPROM (англ. Electrically Erasable Programmable Read-Only Memory - электрически стираемое перепрограммируемое постоянное запоминающее устройство (ЭСППЗУ)), она же энергонезависимая память. Вспомним остальные типы памяти, Flash и SRAM, и их возможности по хранению данных:
Тип | Чтение из программы | Запись из программы | Очистка при перезагрузке |
Flash | Да, PROGMEM | Можно, но сложно | Нет |
SRAM | Да | Да | Да |
EEPROM | Да | Да | Нет |
Простыми словами: EEPROM - память, к которой мы имеем полный доступ из выполняющейся программы, т.е. можем во время выполнения читать и писать туда данные, и эти данные не сбрасываются при перезагрузке МК. Круто? Круто. Зачем?
- Хранение настроек, изменяющихся "из меню" устройства, без перепрошивки;
- Калибровка, сохранение калибровочных данных;
- Использование как дополнительной SRAM памяти в случае её нехватки;
- "Чёрный ящик" - постоянная запись показаний с датчиков для дальнейшей расшифровки сбоев;
- Запись состояния рабочего процесса для восстановления работы после внезапной перезагрузки.
Единственный важный момент: EEPROM имеет ресурс по количеству перезаписи ячеек. Производитель гарантирует 100 000 циклов записи каждой ячейки (AVR Arduino), по факту это количество зависит от конкретного чипа и температурных условий, независимые тесты показали 3-6 миллионов циклов перезаписи при комнатной температуре до появления первой ошибки, т.е. заявленные 100 000 взяты с очень большим запасом. Но есть небольшое уточнение - при заявленных 100 000 циклах перезаписи гарантируется сохранность записанных данных в течение 100 лет при температуре 24°C, если перезаписывать по миллиону - данные испортятся быстрее. В то же время количество чтений каждой ячейки неограниченно.
EEPROM представляет собой область памяти, состоящую из элементарных ячеек с размером в один байт (как SRAM). Объём EEPROM разный у разных моделей МК:
- ATmega328 (Arduino UNO, Nano, Pro Mini): 1 кБ
- ATmega2560 (Arduino Mega): 4 кБ
- ATtiny85 (Digispark): 512 Б
- ESP8266 / ESP32: 4096 Б
Основная задача при работе с EEPROM - не напутать с адресами, потому что каждый байт имеет свой адрес. Если вы пишете двухбайтные данные, то они займут два байта, и следующие данные нужно будет писать по адресу как минимум +2 к предыдущему, иначе они "перемешаются". Рассмотрим пример хранения набора данных разного типа, расположенных в памяти последовательно друг за другом (в скобках я пишу размер текущего типа данных, на размер которого увеличится адрес для следующего "блока"):
- byte - адрес 0 (+1)
- byte - адрес 1 (+1)
- int - адрес 2 (+2) (+4 для esp8266)
- byte - адрес 4 (+1)
- float - адрес 5 (+4)
- int - адрес 9 (+2)
- и так далее
Важный момент: все ячейки имеют значение по умолчанию (у нового чипа) 255.
Скорость работы с EEPROM (время не зависит от частоты системного клока):
- Запись одного байта занимает ~3.3 мс (миллисекунды)
- Чтение одного байта занимает ~0.4 мкс (микросекунды)
Возможны искажения при записи данных в EEPROM при слишком низком VCC (напряжении питания), настоятельно рекомендуется использовать BOD или вручную мониторить напряжение перед записью.
При использовании внутреннего тактового генератора на 8 МГц, его отклонение не должно быть выше 10% (7.2-8.8 МГц), иначе запись в EEPROM или FLASH скорее всего будет производиться с ошибками. Соответственно все разгоны внутреннего клока недопустимы при записи EEPROM или FLASH.
Для работы с EEPROM в среде Arduino у нас есть целых две библиотеки, вторая является более удобной "оболочкой" для первой. Рассмотрим их обе, потому что в "чужом скетче" может встретиться всё что угодно, да и совместное использование этих двух библиотек делает работу с EEPROM невероятно удобной.
Библиотека avr/eeprom.h
Описание к этой библиотеке я спрятал под спойлер, потому что она не очень актуальна и знать о ней необязательно. Также она не работает на esp8266/32 по понятным причинам.
Библиотека EEPROM.h
Библиотека EEPROM.h идёт в комплекте с ядром Arduino и является стандартной библиотекой. По сути EEPROM.h - это удобная оболочка для avr/eeprom.h, чуть расширяющая её возможности и упрощающая использование.
Рассмотрим инструменты, которые нам предлагает библиотека:
EEPROM.write(адрес, данные)
- пишет данные (только byte!) по адресуEEPROM.update(адрес, данные)
- обновляет (та же запись, но лучше) байт данных, находящийся по адресу. Не реализована для esp8266/32!EEPROM.read(адрес)
- читает и возвращает байт данных, находящийся по адресуEEPROM.put(адрес, данные)
- записывает (по факту - обновляет, update) данные любого типа (типа переданной переменной) по адресуEEPROM.get(адрес, данные)
- читает данные по адресу и сам записывает их в данные - указанную переменнуюEEPROM[]
- библиотека позволяет работать с EEPROM памятью как с обычным массивом типа byte (uint8_t
)
У esp8266 и esp32 есть отличия:
- Перед началом работы нужно вызвать
EEPROM.begin(размер)
с указанием максимального объёма памяти: 4.. 4096 Байт. - Для применения записи нужно вызвать
EEPROM.commit()
: например несколько раз делается write(), put(), и в завершение - commit() - В некоторых версиях SDK отсутствует
EEPROM.update()
В отличие от avr/eeprom.h у нас нет отдельных инструментов для работы с конкретными типами данных, отличными от byte
, и сделать write/update/read для float/long/int мы не можем. Но зато у нас есть всеядные put и get, которые очень удобно использовать! Рассмотрим пример с чтением/записью байтов:
#include <EEPROM.h> void setup() { Serial.begin(9600); //EEPROM.begin(100); // для esp8266/esp32 // пишем 200 по адресу 10 EEPROM.update(10, 200); //EEPROM.commit(); // для esp8266/esp32 Serial.println(EEPROM.read(10)); // выведет 200 Serial.println(EEPROM[10]); // выведет 200 } void loop() {}
Логика работы с адресами такая же, как в предыдущем пункте урока! Обратите внимание на работу с EEPROM как с массивом, можно читать, писать, сравнивать, и даже использовать составные операторы, например EEPROM[0] += 10
, но это работает только для элементарных ячеек, байтов.
Теперь посмотрим, как работает put()
и get()
:
#include <EEPROM.h> void setup() { Serial.begin(9600); //EEPROM.begin(100); // для esp8266/esp32 // объявляем переменные, которые будем писать float dataF = 3.14; int16_t dataI = -634; byte dataArray[] = {10, 20, 30, 40}; EEPROM.put(0, dataF); EEPROM.put(4, dataI); EEPROM.put(6, dataArray); //EEPROM.commit(); // для esp8266/esp32 // объявляем переменные, куда будем читать float dataF_read = 0; int16_t dataI_read = 0; byte dataArray_read[4]; // читаем точно так же, как писали EEPROM.get(0, dataF_read); EEPROM.get(4, dataI_read); EEPROM.get(6, dataArray_read); // проверяем Serial.println(dataF_read); Serial.println(dataI_read); Serial.println(dataArray_read[0]); Serial.println(dataArray_read[1]); Serial.println(dataArray_read[2]); Serial.println(dataArray_read[3]); } void loop() {}
put()
и get()
сами определяют тип данных и считают размер блока данных, использовать их очень приятно. Они работают как с массивами, так и со структурами.
EEPROM.h + структуры
Самый удобный хранить набор данных в EEPROM - структура (разбирали в уроке про типы данных). Структура позволяет объединить любые данные под одним именем, и одной строчкой загонять их в EEPROM и так же читать обратно. А также не придётся думать об адресации! Пример:
#include <EEPROM.h> struct Data { byte bright = 0; int counter = 0; float fvalue = 0; }; // глобальный экземпляр для личного использования Data data; void setup() { EEPROM.get(0, data); // прочитать из адреса 0 // меняем data.bright = 10; data.counter = 1234; data.fvalue = 3.14; EEPROM.put(0, data); // поместить в EEPROM по адресу 0 } void loop() {}
EEPROM.h + avr/eeprom.h
Пример не очень актуален, используй EEPROM + структуры
Реальный пример
Рассмотрим пример, в котором происходит следующее: две кнопки управляют яркостью светодиода, подключенного к ШИМ пину. Установленная яркость сохраняется в EEPROM, т.е. при перезапуске устройства будет включена яркость, установленная последний раз. Для опроса кнопок используется библиотека GyverButton. Для начала посмотрите на первоначальную программу, где установленная яркость не сохраняется. Программу можно чуть оптимизировать, но это не является целью данного урока.
В этот код нам нужно добавить:
- Подключить библиотеку EEPROM.h
- При запуске: чтение яркости из EEPROM и включение светодиода
- При клике: запись актуального значения в EEPROM
Итак, теперь при запуске у нас восстанавливается последняя настроенная яркость, и при изменении она записывается. Напомню, что EEPROM изнашивается от перезаписи. Конечно для того, чтобы "накликать" яркость несколько миллионов раз и убить ячейку, у вас уйдёт очень много времени, но процесс записи нового значения можно и нужно оптимизировать, особенно в более серьёзных проектах, ниже поговорим об этом подробнее. Также в нашем коде есть ещё один неприятный момент: при самом первом запуске после прошивки EEPROM не инициализирована, каждая ячейка хранит в себе число 255, и именно такое значение примет переменная LEDbright после первого запуска, при так называемом "первом чтении". Здесь это не имеет значения, но в более серьёзном устройстве нужно будет задать нужные значения по умолчанию в EEPROM при первом запуске, об этом мы тоже поговорим ниже. Иначе представьте, какие "настройки по умолчанию" получит ваше устройство для яркости/скорости/громкости/номера режима/прочее!
Запись и чтение строк
У нас есть два типа строк: массивы символов и String-строки. С массивом символов всё более-менее понятно: это массив, он имеет фиксированный размер, его можно записать при помощи put()
и прочитать при помощи get()
. Также такая строка может входить в структуру, что очень удобно. В этом случае нужно объявить структуру с указанием максимальной длины строки, которая может там храниться. Например для какого-нибудь проекта с WiFi нам хочется хранить логин и пароль от роутера и режим работы:
struct Cfg { char ssid[16]; char pass[16]; byte mode; };
Структуру очень просто прочитать и записать, а работать со строками в ней можно при помощи стандартных строковых функций.
А как записать и прочитать динамические данные, такие как String-строки? Можно рассмотреть два способа: с массивом ограниченной длины (как в примере выше) и полностью динамическое хранение.
Будем считать, что максимальная длина строки - 20 символов. Простой пример:
#include <EEPROM.h> #define STR_ADDR 0 // адрес хранения строки в EEPROM void setup() { Serial.begin(115200); // читаем char str[20]; EEPROM.get(STR_ADDR, str); // выводим Serial.print("Read text: "); Serial.println(str); } void loop() { // читаем строку из порта if (Serial.available()) { char str[20]; int len = Serial.readBytes(str, 20); // завершающий символ, добавляем вручную str[len] = 0; // записываем EEPROM.put(STR_ADDR, str); Serial.print("Save text: "); Serial.println(str); } }
При динамическом хранении мы будем сохранять также длину строки, в первой ячейке от которой идёт счёт. А уже дальше - саму строку. Писать и читать будем посимвольно, по другому тут уже не получится:
#include <EEPROM.h> #define STR_ADDR 0 // адрес хранения строки в EEPROM void setup() { Serial.begin(115200); String str; int len = EEPROM.read(STR_ADDR); // читаем длину строки str.reserve(len); // резервируем место (для оптимальности) // читаем for (int i = 0; i < len; i++) { str += (char)EEPROM.read(STR_ADDR + 1 + i); } // выводим Serial.print("Read text: "); Serial.println(str); } void loop() { // читаем строку из порта if (Serial.available()) { String inc = Serial.readString(); Serial.print("Save text: "); Serial.println(inc); int len = inc.length(); // длина строки EEPROM.write(STR_ADDR, len); // записываем её // и далее саму строку посимвольно for (int i = 0; i < len; i++) { EEPROM.write(STR_ADDR + 1 + i, inc[i]); } } }
Полезные трюки
Инициализация
Под инициализацией я имею в виду установку значений ячеек в EEPROM "по умолчанию" во время первого запуска устройства. В рассмотренном выше примере мы действовали в таком порядке:
- Чтение из EEPROM в переменную
- Использование переменной по назначению
При первом запуске кода (и при всех дальнейших, в которых в ячейку ничего нового не пишется) переменная получит значение, которое было в EEPROM по умолчанию. В большинстве случаев это значение не подойдёт устройству, например ячейка хранит номер режима, по задумке разработчика - от 0 до 5, а из EEPROM мы прочитаем 255. Непорядок! При первом запуске нужно инициализировать EEPROM так, чтобы устройство работало корректно, для этого нужно определить этот самый первый запуск. Можно сделать это вручную, прошив программу, которая забьёт EEPROM нужными данными. Далее прошить уже рабочую программу. При разработке программы это очень неудобно, ведь количество сохраняемых данных может меняться в процессе разработки, поэтому можно использовать следующий алгоритм:
- Резервируем какую-нибудь ячейку (например, последнюю) под хранение "ключа" первого запуска
- Читаем ячейку, если её содержимое не совпадает с ключом - это первый запуск!
- В обработчике первого запуска пишем в ячейку нужный ключ
- Пишем в остальные ячейки необходимые значения по умолчанию
- И после этого уже читаем данные во все нужные переменные
Рассмотрим на всё том же примере со светодиодом и кнопками:
Теперь при первом запуске мы получим инициализацию нужных ячеек. Если нужно переинициализировать EEPROM, например в случае добавления новых данных, достаточно изменить наш ключ на любое другое значение в пределах одного байта (0-254). Я пишу именно до 254, потому что 255 является значением ячейки по умолчанию и наш трюк не сработает.Сброс до "заводских"
Чтобы вернуть настройки к изначально заданным в программе, нужно "спровоцировать" инициализацию. Очевидный способ сделать это - изменить ключ инициализации, который мы назвали INIT_KEY
. Либо можно просто вызвать EEPROM.put(адрес, базовые настройки)
в нужном месте программы.
Скорость
Как я писал выше, скорость работы с EEPROM составляет:
- Запись/обновление одного байта занимает ~3.3 мс (миллисекунды)
- Чтение одного байта занимает ~0.4 мкс (микросекунды)
При большом желании можно использовать ячейку вместо переменной, т.е. выше мы с вами рассматривали пример, в котором EEPROM читался в переменную в программе, и дальнейшая работа происходила уже с ней. При сильной нехватке оперативной памяти можно читать значение напрямую из EEPROM, ведь это занимает ничтожно мало времени. А вот с записью всё гораздо хуже, там целых 3.3 мс. Например так:
analogWrite(LED_PIN, EEPROM.read(0)); // изменили яркость
Для изменения значения придётся прочитать ячейку, выполнить нужные операции, и снова в неё записать. Ещё один удобный хак: можно ввести макросы на чтение и запись определённых значений, например:
#define GET_MODE EEPROM.read(0) // получить номер режима #define GET_BRIGHT EEPROM.read(1) // получить яркость #define SET_MODE(x) EEPROM.write(0, (x)) // запомнить режим #define SET_BRIGHT(x) EEPROM.put(1, (x)) // запомнить яркость
Получим удобные макросы, с которыми писать код будет чуть быстрее и удобнее, т.е. строка SET_MODE(3) запишет 3 в ячейку 0
Уменьшение износа
Важная тема: уменьшение износа ячеек частыми перезаписями. Ситуаций может быть много, интересных решений для них - тоже. Рассмотрим простейший пример - всё тот же код со светодиодом и кнопкой. Делать будем следующее: записывать новое значение будем только в том случае, если после последнего нажатия на кнопку прошло какое-то время. То есть нам понадобится таймер (воспользуемся таймером на millis), при нажатии на кнопку таймер будет сбрасываться, а при срабатывании таймера будем писать актуальное значение в EEPROM. Также понадобится флаг, который будет сигнализировать о записи и позволит записать именно один раз. Алгоритм такой:
- При нажатии на кнопку:
- Если флаг опущен - поднять флаг
- Сбросить таймер
- Если сработал таймер и флаг поднят:
- Опустить флаг
- Записать значения в EEPROM
Посмотрим на всё том же примере:
Вот таким нехитрым способом мы многократно снизили износ EEPROM, я очень часто использую этот "алгоритм" работы с настройками в своих устройствах. Есть другие задачи, в которых данные в EEPROM пишутся не когда пользователь что-то изменит, а постоянно, т.е. память работает в режиме чёрного ящика и постоянно записывает значения. Это может быть например контроллер печи, который держит температурный режим по специальному закону, и после внезапной перезагрузки должен вернуться к тому месту в процессе, на котором прервался. Тут есть глобально два варианта:
- Ёмкий конденсатор по питанию микроконтроллера, позволяющий сохранить работу МК после отключения питания на время, достаточное для записи в EEPROM (~3.3 мс). Также МК должен знать о том, что общее питание отключилось: если это высокое напряжение (выше 5 Вольт), то это может быть делитель напряжения на аналоговый пин. Если это 5 Вольт - можно измерять напряжение МК, и момент отключения (разрядка конденсатора) тоже можно отловить и записать нужные данные. Можно взвести прерывание, которое сработает при падении напряжения питания ниже опасного уровня. Можно 5 Вольт завести напрямую на цифровой пин, а сам МК питать через диод и поставить конденсатор - тогда напряжение на измеряющем пине пропадёт до того, как отключится МК, он будет работать от конденсатора. Вот схема:
- Можно писать данные (необязательно один байт, можно целую структуру) хитро, размазывая их по всему EEPROM. Тут глобально два варианта:
- Писать данные каждый раз в следующую ячейку, и закольцевать переход на первую. Также понадобится хранить где-то счётчик, указывающий на адрес текущей ячейки, и этот счётчик тоже придётся хранить хитро, чтобы он не износил ячейку. Например счётчик - это структура, состоящая из счётчика перезаписей этой структуры и счётчика адреса для большой структуры.
- Писать данные, пока не достигнут лимит количества перезаписей, количество текущих перезаписей хранить например в этой же структуре. Скажем структура занимает 30 байт, то есть в перспективе мы можем найти эту структуру по адресу, кратному 30. Программа работает, счётчик считает количество перезаписей, при достижении опасного количества вся структура "переезжает" на следующие 30 адресов.
Вариантов уменьшения износа ячеек EEPROM можно придумать много, уникально под свою ситуацию. Есть даже библиотеки готовые, например EEPROMWearLevel. Есть очень интересная статья на Хабре, там рассмотрено ещё несколько хороших алгоритмов и даны ссылки на ещё большее их количество.
Библиотека EEManager
Я часто использую EEPROM в своих проектах, поэтому обернул все рассмотренные выше конструкции в библиотеку, изучить и скачать можно здесь. Библиотека подходит для всех архитектур, в которых есть стандартная EEPROM.h. В библиотеке реализовано:
- Работа с данными любого типа
- Чтение и запись в указанную переменную
- Функция "ключа первого запуска" для задания начальных значений
- Отложенное обновление по тайм-ауту для уменьшения износа
Я надеюсь вы полностью разобрались с самым последним примером с кнопкой и светодиодом, поэтому покажу работу EEManager на его основе:
Таким образом вся работа с еепром по чтению, записи, обеспечению корректного первого запуска и уменьшению износа памяти свелась к трём строчкам кода. Пользуйтесь!
Видео
Полезные страницы
- Набор GyverKIT – большой стартовый набор Arduino моей разработки, продаётся в России
- Каталог ссылок на дешёвые Ардуины, датчики, модули и прочие железки с AliExpress у проверенных продавцов
- Подборка библиотек для Arduino, самых интересных и полезных, официальных и не очень
- Полная документация по языку Ардуино, все встроенные функции и макросы, все доступные типы данных
- Сборник полезных алгоритмов для написания скетчей: структура кода, таймеры, фильтры, парсинг данных
- Видео уроки по программированию Arduino с канала “Заметки Ардуинщика” – одни из самых подробных в рунете
- Поддержать автора за работу над уроками
- Обратная связь – сообщить об ошибке в уроке или предложить дополнение по тексту ([email protected])