Task watchdog got triggered - The tasks did not reset the watchdog in time
Asked Answered
N

3

11

I am trying to write a little asynchronous WebServer. Let me briefly describe the scenario:

My ESP32 as well is a router. So if I connect with my mobile phone into the WiFi the ESP32 is spreading and call the ip address and a special path with a browser, a WebSite is delivered. Here a button is displayed. Until this point it works pretty well. Now, if I click on that button, a HTTPS Web Request (method: GET) is sent to a special machine. This machine answers and returns a JSON. This could last a couple of seconds. After extracting a value out of the JSON string, this value shall be displayed.

To achieve this, I am using the following libraries:

I know (by another sketch) that the last three ones work without any problems.

Unfortunately, when I click the button, the following output appears onto my serial monitor:

Starting connection to server...
[HTTPS] begin... Path: https://192.168.4.101/api/unlock/generate_pin
[HTTPS] GET...
E (137906) task_wdt: Task watchdog got triggered. The following tasks did not reset the watchdog in time:
E (137906) task_wdt: - async_tcp (CPU 0/1)
E (137906) task_wdt: Tasks currently running:
E (137906) task_wdt: CPU 0: IDLE0
E (137906) task_wdt: CPU 1: loopTask
E (137906) task_wdt: Aborting.
abort() was called at PC 0x400e08af on core 0

Backtrace: 0x4008cc18:0x3ffbe170 0x4008ce49:0x3ffbe190 0x400e08af:0x3ffbe1b0 0x40084f21:0x3ffbe1d0 0x4016581b:0x3ffbc120 0x400e1c66:0x3ffbc140 0x4008ab21:0x3ffbc160 0x4008932d:0x3ffbc180

Rebooting...
ets Jun 8 2016 00:22:57

rst:0xc (SW_CPU_RESET),boot:0x17 (SPI_FAST_FLASH_BOOT)
configsip: 0, SPIWP:0xee
clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00
mode:DIO, clock div:1
load:0x3fff0018,len:4
load:0x3fff001c,len:1044
load:0x40078000,len:8896
load:0x40080400,len:5816
entry 0x400806ac
Serial initial done

Anyone any idea what is happening and how to fix this? So that the GET request is sent correctly / the answer is received?

I am using a Heltec WiFi Kit 32.

Would be very happy about every answer, thanks in advance.

Best regards

P.S.: Please let me finally add my code:

#include <heltec.h>
#include "WiFi.h"
#include "ESPAsyncWebServer.h"

#include <WiFiClientSecure.h>
#include <HTTPClient.h>
 
const char* ssid = "MyWiFiSSID";
const char* password =  "MyWiFiPW";
 
AsyncWebServer server(80);

void setup() {

  Heltec.begin(true, false, true, true, 470E6);

  WiFi.softAP(ssid, password);
  
  IPAddress IP = WiFi.softAPIP();
  Serial.print("AccessPoint IP address: ");
  Serial.println(IP);
  
  server.on("/hello", HTTP_GET, [](AsyncWebServerRequest *request){
    
    request->send(200, "text/html", "<!DOCTYPE html><html><head><meta name=\"viewport\" content=\"width=device-width, initial-scale=1\" charset=\"UTF-8\"><link rel=\"icon\" href=\"data:,\"><style>html { font-family: Helvetica; display: inline-block; margin: 0px auto; text-align: center;}.button { background-color: #4CAF50; border: none; color: white; padding: 16px 40px; text-decoration: none; font-size: 30px; margin: 2px; cursor: pointer;}</style></head><body><h1>Welcome to the Landing Page of the Web Server</h1><p><a href=\"/get_unlock_pin\"><button class=\"button\">Click Me</button></a></p></body></html>");
  });

  server.on("/get_unlock_pin", HTTP_GET, [](AsyncWebServerRequest *request){

    String firstpartofrawhtmlcode = "<!DOCTYPE html><html><head><meta name=\"viewport\" content=\"width=device-width, initial-scale=1\" charset=\"UTF-8\"><link rel=\"icon\" href=\"data:,\"><style>html { font-family: Helvetica; display: inline-block; margin: 0px auto; text-align: center;}.button { background-color: #4CAF50; border: none; color: white; padding: 16px 40px; text-decoration: none; font-size: 30px; margin: 2px; cursor: pointer;}</style></head><body><h2>Received Pin: </h2><h2 style=\"color: #FF0000\">";
    String receivedPin = getPin("192.168.4.101");
    String secondpartofrawhtmlcode = "</h2></body></html>";
    String fullrawhtmlcode;
    firstpartofrawhtmlcode = firstpartofrawhtmlcode.concat(receivedPin);
    fullrawhtmlcode = firstpartofrawhtmlcode.concat(secondpartofrawhtmlcode);
    request->send(200, "text/html", fullrawhtmlcode);
  });
 
  server.begin();
}

void loop() {

}

String getPin(String ip){
    Serial.println("\nStarting connection to server...");  
    WiFiClientSecure *wificlient = new WiFiClientSecure;

    HTTPClient https;
    https.setAuthorization("MyUserName", "MyPassword");

    String path = "https://" + ip + "/api/unlock/generate_pin";
      
    Serial.print("[HTTPS] begin... Path: " + path + "\n");
    if (https.begin(*wificlient, path)) { 
        Serial.print("[HTTPS] GET...\n");
        int httpCode = https.GET();
        if (httpCode > 0) {
          Serial.printf("[HTTPS] GET... code: %d\n", httpCode);
  
          if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY) {
            String payload = https.getString();
            Serial.println(payload);
            //Extract Pin from JSON
            String tmp = payload.substring(payload.indexOf(':'), payload.indexOf('}'));
            String tmp2 = tmp.substring(tmp.indexOf('"')+1,tmp.lastIndexOf('"')); 
            if(tmp2.substring(0,1) == "-"){
               return "-";
            }else{
               return tmp2;
            }              
          }
        } else {
               Serial.printf("[HTTPS] GET... failed, error: %s\n", https.errorToString(httpCode).c_str());
        }  
        https.end();
    } else {
        Serial.printf("[HTTPS] Unable to connect\n");
    }
}
Nunnery answered 19/2, 2021 at 13:16 Comment(0)
W
6

ESPAsyncWebServer callbacks prevent the watchdog timer from being reset while they run. This means they are not meant for doing any actual processing. Register the request and defer processing to the main loop (or some other thread). Have a look at this question for details.

Wainscoting answered 19/2, 2021 at 14:21 Comment(0)
C
4

After some digging into this problem; I found a fairly simple solution for this;

Just add: #include "soc/rtc_wdt.h" into the library.

Then do:

rtc_wdt_protect_off();    // Turns off the automatic wdt service
rtc_wdt_enable();         // Turn it on manually
rtc_wdt_set_time(RTC_WDT_STAGE0, 20000);  // Define how long you desire to let dog wait.

Then you need to do rtc_wdt_feed(); to feed the dog in some place you think it indeed runs faster than the dog waiting time.

Coryden answered 4/5, 2023 at 20:25 Comment(1)
ESP-IDF config has a set of CONFIG_ESP_TASK_WDT_xxx defines. You can also control all the logic using those.Cauchy
L
1

I found this in menuconfig: CONFIG_ESP_INT_WDT_TIMEOUT_MS, along with several other relevant items as some have already mentioned, but this one caught my attention because of its description:

The timeout of the watchdog, in miliseconds. Make this higher than the FreeRTOS tick rate.

The FreeRTOS tick rate (FREERTOS_HZ):

The tick rate at which FreeRTOS does pre-emptive context switching.

is set to 100Hz by default in menuconfig, but you will be unable to flash with anything less than 1000Hz.

That being the case, the 300msec default value for CONFIG_ESP_INT_WDT_TIMEOUT_MS isn't making much sense. I changed it to 1300msec and also changed CONFIG_ESP_TASK_WDT_TIMEOUT_S to 10 seconds.

These changes have made the situation much better for me, but it'll still abort if a task blocks for too long.

Lovelady answered 24/12, 2023 at 2:57 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.