bootstate.cpp 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. #include "bootstate.h"
  2. #include "Config.h"
  3. #include "esp_attr.h"
  4. #include "esp_log.h"
  5. #include "esp_ota_ops.h"
  6. #include "esp_spi_flash.h"
  7. #include "messaging.h"
  8. #include "tools.h"
  9. static const char* TAG = "bootstate";
  10. RTC_NOINIT_ATTR uint32_t RebootCounter;
  11. RTC_NOINIT_ATTR uint32_t RecoveryRebootCounter;
  12. RTC_NOINIT_ATTR uint16_t ColdBootIndicatorFlag;
  13. EXT_RAM_ATTR bool is_recovery_running = false;
  14. EXT_RAM_ATTR bool cold_boot = true;
  15. EXT_RAM_ATTR esp_reset_reason_t xReason = ESP_RST_UNKNOWN;
  16. EXT_RAM_ATTR static bool restarting = false;
  17. uint32_t bootstate_read_counter(void) { return RebootCounter; }
  18. uint32_t bootstate_uptate_counter(int32_t xValue) {
  19. if (RebootCounter > 100) {
  20. RebootCounter = 0;
  21. RecoveryRebootCounter = 0;
  22. }
  23. RebootCounter = (xValue != 0) ? (RebootCounter + xValue) : 0;
  24. RecoveryRebootCounter = (xValue != 0) && is_recovery_running ? (RecoveryRebootCounter + xValue) : 0;
  25. return RebootCounter;
  26. }
  27. void bootstate_handle_boot() {
  28. if (ColdBootIndicatorFlag != 0xFACE) {
  29. ESP_LOGI(TAG, "System is booting from power on.");
  30. cold_boot = true;
  31. ColdBootIndicatorFlag = 0xFACE;
  32. } else {
  33. cold_boot = false;
  34. }
  35. const esp_partition_t* running = esp_ota_get_running_partition();
  36. xReason = esp_reset_reason();
  37. ESP_LOGI(TAG, "Reset reason is: %u. Running from partition %s type %s ", xReason, running->label,
  38. running->subtype == ESP_PARTITION_SUBTYPE_APP_FACTORY ? "Factory" : "Application");
  39. is_recovery_running = (running->subtype == ESP_PARTITION_SUBTYPE_APP_FACTORY);
  40. if (!is_recovery_running) {
  41. /* unscheduled restart (HW, Watchdog or similar) thus increment dynamic
  42. * counter then log current boot statistics as a warning */
  43. uint32_t Counter = bootstate_uptate_counter(1); // increment counter
  44. ESP_LOGI(TAG, "Reboot counter=%u\n", Counter);
  45. if (Counter == 5) {
  46. guided_factory();
  47. }
  48. } else {
  49. uint32_t Counter = bootstate_uptate_counter(1); // increment counter
  50. if (RecoveryRebootCounter == 1 && Counter >= 5) {
  51. // First time we are rebooting in recovery after crashing
  52. messaging_post_message(MESSAGING_ERROR, MESSAGING_CLASS_SYSTEM,
  53. "System was forced into recovery mode after crash likely caused by some bad "
  54. "configuration\n");
  55. }
  56. ESP_LOGI(TAG, "Recovery Reboot counter=%u\n", Counter);
  57. if (RecoveryRebootCounter == 5) {
  58. ESP_LOGW(TAG, "System rebooted too many times. This could be an indication that "
  59. "configuration is corrupted. Erasing config.");
  60. if (config_erase_config()) {
  61. config_raise_changed(true);
  62. guided_factory();
  63. } else {
  64. ESP_LOGE(TAG, "Error erasing configuration");
  65. }
  66. }
  67. if (RecoveryRebootCounter > 5) {
  68. messaging_post_message(MESSAGING_ERROR, MESSAGING_CLASS_SYSTEM,
  69. "System was forced into recovery mode after crash likely caused by some bad "
  70. "configuration. Configuration was reset to factory.\n");
  71. }
  72. }
  73. }
  74. esp_err_t guided_boot(esp_partition_subtype_t partition_subtype) {
  75. if (is_recovery_running) {
  76. if (partition_subtype == ESP_PARTITION_SUBTYPE_APP_FACTORY) {
  77. simple_restart();
  78. }
  79. } else {
  80. if (partition_subtype != ESP_PARTITION_SUBTYPE_APP_FACTORY) {
  81. simple_restart();
  82. }
  83. }
  84. esp_err_t err = ESP_OK;
  85. const esp_partition_t* partition;
  86. esp_partition_iterator_t it = esp_partition_find(ESP_PARTITION_TYPE_APP, partition_subtype, NULL);
  87. if (it == NULL) {
  88. log_send_messaging(MESSAGING_ERROR, "Reboot failed. Partitions error");
  89. } else {
  90. ESP_LOGD(TAG, "Found partition. Getting info.");
  91. partition = (esp_partition_t*)esp_partition_get(it);
  92. ESP_LOGD(TAG, "Releasing partition iterator");
  93. esp_partition_iterator_release(it);
  94. if (partition != NULL) {
  95. log_send_messaging(MESSAGING_INFO, "Rebooting to %s", partition->label);
  96. err = esp_ota_set_boot_partition(partition);
  97. if (err != ESP_OK) {
  98. log_send_messaging(MESSAGING_ERROR, "Unable to select partition for reboot: %s", esp_err_to_name(err));
  99. }
  100. } else {
  101. log_send_messaging(MESSAGING_ERROR, "partition type %u not found! Unable to reboot to recovery.", partition_subtype);
  102. }
  103. ESP_LOGD(TAG, "Yielding to other processes");
  104. taskYIELD();
  105. simple_restart();
  106. }
  107. return ESP_OK;
  108. }
  109. esp_err_t guided_restart_ota() {
  110. log_send_messaging(MESSAGING_WARNING, "Booting to Squeezelite");
  111. guided_boot(ESP_PARTITION_SUBTYPE_APP_OTA_0);
  112. return ESP_FAIL; // return fail. This should never return... we're rebooting!
  113. }
  114. esp_err_t guided_factory() {
  115. log_send_messaging(MESSAGING_WARNING, "Booting to recovery");
  116. guided_boot(ESP_PARTITION_SUBTYPE_APP_FACTORY);
  117. return ESP_FAIL; // return fail. This should never return... we're rebooting!
  118. }
  119. void simple_restart() {
  120. restarting = true;
  121. log_send_messaging(MESSAGING_WARNING, "Rebooting.");
  122. TimerHandle_t timer = xTimerCreate("reboot", 1, pdFALSE, nullptr, [](TimerHandle_t xTimer) {
  123. if (!config_waitcommit()) {
  124. log_send_messaging(MESSAGING_WARNING, "Waiting for configuration to commit ");
  125. ESP_LOGD(TAG,"Queuing restart asynchronously to ensure all events are flushed.");
  126. network_async_reboot(RESTART);
  127. return;
  128. }
  129. vTaskDelay(750 / portTICK_PERIOD_MS);
  130. esp_restart();
  131. xTimerDelete(xTimer, portMAX_DELAY);
  132. });
  133. xTimerStart(timer, portMAX_DELAY);
  134. }
  135. bool is_restarting() { return restarting; }