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.

198 lines
5.1 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
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. ESP.restart();
  79. });
  80. Embedis::command( F("FACTORY.RESET"), [](Embedis* e) {
  81. settingsFactoryReset();
  82. e->response(Embedis::OK);
  83. });
  84. Embedis::command( F("HEAP"), [](Embedis* e) {
  85. e->stream->printf("Free HEAP: %d bytes\n", ESP.getFreeHeap());
  86. e->response(Embedis::OK);
  87. });
  88. Embedis::command( F("RELAY"), [](Embedis* e) {
  89. if (e->argc < 2) {
  90. return e->response(Embedis::ARGS_ERROR);
  91. }
  92. int id = String(e->argv[1]).toInt();
  93. if (e->argc > 2) {
  94. int value = String(e->argv[2]).toInt();
  95. if (value == 2) {
  96. relayToggle(id);
  97. } else {
  98. relayStatus(id, value == 1);
  99. }
  100. }
  101. e->stream->printf("Status: %s\n", relayStatus(id) ? "true" : "false");
  102. e->response(Embedis::OK);
  103. });
  104. #if LIGHT_PROVIDER != LIGHT_PROVIDER_NONE
  105. Embedis::command( F("COLOR"), [](Embedis* e) {
  106. if (e->argc > 1) {
  107. String color = String(e->argv[1]);
  108. lightColor(color.c_str(), true, true);
  109. }
  110. e->stream->printf("Color: %s\n", lightColor().c_str());
  111. e->response(Embedis::OK);
  112. });
  113. #endif
  114. Embedis::command( F("EEPROM"), [](Embedis* e) {
  115. unsigned long freeEEPROM = SPI_FLASH_SEC_SIZE - settingsSize();
  116. e->stream->printf("Number of keys: %d\n", settingsKeyCount());
  117. e->stream->printf("Free EEPROM: %d bytes (%d%%)\n", freeEEPROM, 100 * freeEEPROM / SPI_FLASH_SEC_SIZE);
  118. e->response(Embedis::OK);
  119. });
  120. Embedis::command( F("DUMP"), [](Embedis* e) {
  121. unsigned int size = settingsKeyCount();
  122. for (unsigned int i=0; i<size; i++) {
  123. String key = settingsKeyName(i);
  124. String value = getSetting(key);
  125. e->stream->printf("+%s => %s\n", key.c_str(), value.c_str());
  126. }
  127. e->response(Embedis::OK);
  128. });
  129. Embedis::command( F("DUMP.RAW"), [](Embedis* e) {
  130. for (unsigned int i = 0; i < SPI_FLASH_SEC_SIZE; i++) {
  131. if (i % 16 == 0) e->stream->printf("\n[%04X] ", i);
  132. e->stream->printf("%02X ", EEPROM.read(i));
  133. }
  134. e->stream->printf("\n");
  135. e->response(Embedis::OK);
  136. });
  137. DEBUG_MSG("[SETTINGS] EEPROM size: %d bytes\n", SPI_FLASH_SEC_SIZE);
  138. DEBUG_MSG("[SETTINGS] Settings size: %d bytes\n", settingsSize());
  139. }
  140. void settingsLoop() {
  141. embedis.process();
  142. }
  143. template<typename T> String getSetting(const String& key, T defaultValue) {
  144. String value;
  145. if (!Embedis::get(key, value)) value = String(defaultValue);
  146. return value;
  147. }
  148. String getSetting(const String& key) {
  149. return getSetting(key, "");
  150. }
  151. template<typename T> bool setSetting(const String& key, T value) {
  152. return Embedis::set(key, String(value));
  153. }
  154. bool delSetting(const String& key) {
  155. return Embedis::del(key);
  156. }
  157. void saveSettings() {
  158. DEBUG_MSG("[SETTINGS] Saving\n");
  159. #if not AUTO_SAVE
  160. EEPROM.commit();
  161. #endif
  162. }