Fork of the espurna firmware for `mhsw` switches
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

226 lines
5.9 KiB

8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
  1. /*
  2. SETTINGS MODULE
  3. Copyright (C) 2016-2017 by Xose Pérez <xose dot perez at gmail dot com>
  4. */
  5. #include "Embedis.h"
  6. #include <EEPROM.h>
  7. #include "spi_flash.h"
  8. #include <StreamString.h>
  9. #define AUTO_SAVE 1
  10. Embedis embedis(Serial);
  11. // -----------------------------------------------------------------------------
  12. // Settings
  13. // -----------------------------------------------------------------------------
  14. unsigned long settingsSize() {
  15. unsigned pos = SPI_FLASH_SEC_SIZE - 1;
  16. while (size_t len = EEPROM.read(pos)) {
  17. pos = pos - len - 2;
  18. }
  19. return SPI_FLASH_SEC_SIZE - pos;
  20. }
  21. unsigned int settingsKeyCount() {
  22. unsigned count = 0;
  23. unsigned pos = SPI_FLASH_SEC_SIZE - 1;
  24. while (size_t len = EEPROM.read(pos)) {
  25. pos = pos - len - 2;
  26. count ++;
  27. }
  28. return count / 2;
  29. }
  30. String settingsKeyName(unsigned int index) {
  31. String s;
  32. unsigned count = 0;
  33. unsigned stop = index * 2 + 1;
  34. unsigned pos = SPI_FLASH_SEC_SIZE - 1;
  35. while (size_t len = EEPROM.read(pos)) {
  36. pos = pos - len - 2;
  37. count++;
  38. if (count == stop) {
  39. s.reserve(len);
  40. for (unsigned char i = 0 ; i < len; i++) {
  41. s += (char) EEPROM.read(pos + i + 1);
  42. }
  43. break;
  44. }
  45. }
  46. return s;
  47. }
  48. void settingsFactoryReset() {
  49. for (unsigned int i = 0; i < SPI_FLASH_SEC_SIZE; i++) {
  50. EEPROM.write(i, 0xFF);
  51. }
  52. EEPROM.commit();
  53. }
  54. void settingsSetup() {
  55. EEPROM.begin(SPI_FLASH_SEC_SIZE);
  56. Embedis::dictionary( F("EEPROM"),
  57. SPI_FLASH_SEC_SIZE,
  58. [](size_t pos) -> char { return EEPROM.read(pos); },
  59. [](size_t pos, char value) { EEPROM.write(pos, value); },
  60. #if AUTO_SAVE
  61. []() { EEPROM.commit(); }
  62. #else
  63. []() {}
  64. #endif
  65. );
  66. Embedis::hardware( F("WIFI"), [](Embedis* e) {
  67. StreamString s;
  68. WiFi.printDiag(s);
  69. e->response(s);
  70. }, 0);
  71. Embedis::command( F("RECONNECT"), [](Embedis* e) {
  72. wifiConfigure();
  73. wifiDisconnect();
  74. e->response(Embedis::OK);
  75. });
  76. Embedis::command( F("RESET"), [](Embedis* e) {
  77. e->response(Embedis::OK);
  78. customReset(CUSTOM_RESET_TERMINAL);
  79. ESP.restart();
  80. });
  81. Embedis::command( F("FACTORY.RESET"), [](Embedis* e) {
  82. settingsFactoryReset();
  83. e->response(Embedis::OK);
  84. });
  85. Embedis::command( F("HEAP"), [](Embedis* e) {
  86. e->stream->printf("Free HEAP: %d bytes\n", ESP.getFreeHeap());
  87. e->response(Embedis::OK);
  88. });
  89. Embedis::command( F("RELAY"), [](Embedis* e) {
  90. if (e->argc < 2) {
  91. return e->response(Embedis::ARGS_ERROR);
  92. }
  93. int id = String(e->argv[1]).toInt();
  94. if (e->argc > 2) {
  95. int value = String(e->argv[2]).toInt();
  96. if (value == 2) {
  97. relayToggle(id);
  98. } else {
  99. relayStatus(id, value == 1);
  100. }
  101. }
  102. e->stream->printf("Status: %s\n", relayStatus(id) ? "true" : "false");
  103. e->response(Embedis::OK);
  104. });
  105. #if LIGHT_PROVIDER != LIGHT_PROVIDER_NONE
  106. Embedis::command( F("COLOR"), [](Embedis* e) {
  107. if (e->argc > 1) {
  108. String color = String(e->argv[1]);
  109. parseColor(color.c_str());
  110. lightColor(true, true);
  111. }
  112. e->stream->printf("Color: %s\n", lightColor().c_str());
  113. e->response(Embedis::OK);
  114. });
  115. #endif
  116. Embedis::command( F("EEPROM"), [](Embedis* e) {
  117. unsigned long freeEEPROM = SPI_FLASH_SEC_SIZE - settingsSize();
  118. e->stream->printf("Number of keys: %d\n", settingsKeyCount());
  119. e->stream->printf("Free EEPROM: %d bytes (%d%%)\n", freeEEPROM, 100 * freeEEPROM / SPI_FLASH_SEC_SIZE);
  120. e->response(Embedis::OK);
  121. });
  122. Embedis::command( F("DUMP"), [](Embedis* e) {
  123. unsigned int size = settingsKeyCount();
  124. for (unsigned int i=0; i<size; i++) {
  125. String key = settingsKeyName(i);
  126. String value = getSetting(key);
  127. e->stream->printf("+%s => %s\n", key.c_str(), value.c_str());
  128. }
  129. e->response(Embedis::OK);
  130. });
  131. Embedis::command( F("DUMP.RAW"), [](Embedis* e) {
  132. for (unsigned int i = 0; i < SPI_FLASH_SEC_SIZE; i++) {
  133. if (i % 16 == 0) e->stream->printf("\n[%04X] ", i);
  134. e->stream->printf("%02X ", EEPROM.read(i));
  135. }
  136. e->stream->printf("\n");
  137. e->response(Embedis::OK);
  138. });
  139. DEBUG_MSG_P(PSTR("[SETTINGS] EEPROM size: %d bytes\n"), SPI_FLASH_SEC_SIZE);
  140. DEBUG_MSG_P(PSTR("[SETTINGS] Settings size: %d bytes\n"), settingsSize());
  141. }
  142. void settingsLoop() {
  143. embedis.process();
  144. }
  145. void moveSetting(const char * from, const char * to) {
  146. String value = getSetting(from);
  147. if (value.length() > 0) setSetting(to, value);
  148. delSetting(from);
  149. }
  150. template<typename T> String getSetting(const String& key, T defaultValue) {
  151. String value;
  152. if (!Embedis::get(key, value)) value = String(defaultValue);
  153. return value;
  154. }
  155. template<typename T> String getSetting(const String& key, unsigned int index, T defaultValue) {
  156. return getSetting(key + String(index), defaultValue);
  157. }
  158. String getSetting(const String& key) {
  159. return getSetting(key, "");
  160. }
  161. template<typename T> bool setSetting(const String& key, T value) {
  162. return Embedis::set(key, String(value));
  163. }
  164. template<typename T> bool setSetting(const String& key, unsigned int index, T value) {
  165. return setSetting(key + String(index), value);
  166. }
  167. bool delSetting(const String& key) {
  168. return Embedis::del(key);
  169. }
  170. bool delSetting(const String& key, unsigned int index) {
  171. return delSetting(key + String(index));
  172. }
  173. bool hasSetting(const String& key) {
  174. return getSetting(key).length() != 0;
  175. }
  176. bool hasSetting(const String& key, unsigned int index) {
  177. return getSetting(key, index, "").length() != 0;
  178. }
  179. void saveSettings() {
  180. DEBUG_MSG_P(PSTR("[SETTINGS] Saving\n"));
  181. #if not AUTO_SAVE
  182. EEPROM.commit();
  183. #endif
  184. }