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.

225 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. lightColor(color.c_str(), true, true);
  110. }
  111. e->stream->printf("Color: %s\n", lightColor().c_str());
  112. e->response(Embedis::OK);
  113. });
  114. #endif
  115. Embedis::command( F("EEPROM"), [](Embedis* e) {
  116. unsigned long freeEEPROM = SPI_FLASH_SEC_SIZE - settingsSize();
  117. e->stream->printf("Number of keys: %d\n", settingsKeyCount());
  118. e->stream->printf("Free EEPROM: %d bytes (%d%%)\n", freeEEPROM, 100 * freeEEPROM / SPI_FLASH_SEC_SIZE);
  119. e->response(Embedis::OK);
  120. });
  121. Embedis::command( F("DUMP"), [](Embedis* e) {
  122. unsigned int size = settingsKeyCount();
  123. for (unsigned int i=0; i<size; i++) {
  124. String key = settingsKeyName(i);
  125. String value = getSetting(key);
  126. e->stream->printf("+%s => %s\n", key.c_str(), value.c_str());
  127. }
  128. e->response(Embedis::OK);
  129. });
  130. Embedis::command( F("DUMP.RAW"), [](Embedis* e) {
  131. for (unsigned int i = 0; i < SPI_FLASH_SEC_SIZE; i++) {
  132. if (i % 16 == 0) e->stream->printf("\n[%04X] ", i);
  133. e->stream->printf("%02X ", EEPROM.read(i));
  134. }
  135. e->stream->printf("\n");
  136. e->response(Embedis::OK);
  137. });
  138. DEBUG_MSG_P(PSTR("[SETTINGS] EEPROM size: %d bytes\n"), SPI_FLASH_SEC_SIZE);
  139. DEBUG_MSG_P(PSTR("[SETTINGS] Settings size: %d bytes\n"), settingsSize());
  140. }
  141. void settingsLoop() {
  142. embedis.process();
  143. }
  144. void moveSetting(const char * from, const char * to) {
  145. String value = getSetting(from);
  146. if (value.length() > 0) setSetting(to, value);
  147. delSetting(from);
  148. }
  149. template<typename T> String getSetting(const String& key, T defaultValue) {
  150. String value;
  151. if (!Embedis::get(key, value)) value = String(defaultValue);
  152. return value;
  153. }
  154. template<typename T> String getSetting(const String& key, unsigned int index, T defaultValue) {
  155. return getSetting(key + String(index), defaultValue);
  156. }
  157. String getSetting(const String& key) {
  158. return getSetting(key, "");
  159. }
  160. template<typename T> bool setSetting(const String& key, T value) {
  161. return Embedis::set(key, String(value));
  162. }
  163. template<typename T> bool setSetting(const String& key, unsigned int index, T value) {
  164. return setSetting(key + String(index), value);
  165. }
  166. bool delSetting(const String& key) {
  167. return Embedis::del(key);
  168. }
  169. bool delSetting(const String& key, unsigned int index) {
  170. return delSetting(key + String(index));
  171. }
  172. bool hasSetting(const String& key) {
  173. return getSetting(key).length() != 0;
  174. }
  175. bool hasSetting(const String& key, unsigned int index) {
  176. return getSetting(key, index, "").length() != 0;
  177. }
  178. void saveSettings() {
  179. DEBUG_MSG_P(PSTR("[SETTINGS] Saving\n"));
  180. #if not AUTO_SAVE
  181. EEPROM.commit();
  182. #endif
  183. }