improve logging MT robustness
This commit is contained in:
parent
6ef75422d8
commit
b3f4a11a8e
|
@ -28,6 +28,7 @@ void handleLeds(void *param){
|
||||||
GwLog *logger=api->getLogger();
|
GwLog *logger=api->getLogger();
|
||||||
#ifndef GWLED_FASTLED
|
#ifndef GWLED_FASTLED
|
||||||
LOG_DEBUG(GwLog::LOG,"currently only fastled handling");
|
LOG_DEBUG(GwLog::LOG,"currently only fastled handling");
|
||||||
|
delay(50);
|
||||||
vTaskDelete(NULL);
|
vTaskDelete(NULL);
|
||||||
return;
|
return;
|
||||||
#else
|
#else
|
||||||
|
|
|
@ -34,7 +34,11 @@ void GwLog::logDebug(int level,const char *fmt,...){
|
||||||
va_start(args,fmt);
|
va_start(args,fmt);
|
||||||
xSemaphoreTake(locker, portMAX_DELAY);
|
xSemaphoreTake(locker, portMAX_DELAY);
|
||||||
vsnprintf(buffer,99,fmt,args);
|
vsnprintf(buffer,99,fmt,args);
|
||||||
if (! writer) return;
|
buffer[99]=0;
|
||||||
|
if (! writer) {
|
||||||
|
xSemaphoreGive(locker);
|
||||||
|
return;
|
||||||
|
}
|
||||||
writer->write(prefix.c_str());
|
writer->write(prefix.c_str());
|
||||||
writer->write(buffer);
|
writer->write(buffer);
|
||||||
writer->write("\n");
|
writer->write("\n");
|
||||||
|
@ -46,3 +50,13 @@ void GwLog::setWriter(GwLogWriter *writer){
|
||||||
this->writer=writer;
|
this->writer=writer;
|
||||||
xSemaphoreGive(locker);
|
xSemaphoreGive(locker);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GwLog::flush(){
|
||||||
|
xSemaphoreTake(locker, portMAX_DELAY);
|
||||||
|
if (! this->writer) {
|
||||||
|
xSemaphoreGive(locker);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this->writer->flush();
|
||||||
|
xSemaphoreGive(locker);
|
||||||
|
}
|
||||||
|
|
|
@ -6,6 +6,7 @@ class GwLogWriter{
|
||||||
public:
|
public:
|
||||||
virtual ~GwLogWriter(){}
|
virtual ~GwLogWriter(){}
|
||||||
virtual void write(const char *data)=0;
|
virtual void write(const char *data)=0;
|
||||||
|
virtual void flush(){};
|
||||||
};
|
};
|
||||||
class GwLog{
|
class GwLog{
|
||||||
private:
|
private:
|
||||||
|
@ -25,6 +26,7 @@ class GwLog{
|
||||||
void logString(const char *fmt,...);
|
void logString(const char *fmt,...);
|
||||||
void logDebug(int level, const char *fmt,...);
|
void logDebug(int level, const char *fmt,...);
|
||||||
int isActive(int level){return level <= logLevel;};
|
int isActive(int level){return level <= logLevel;};
|
||||||
|
void flush();
|
||||||
};
|
};
|
||||||
#define LOG_DEBUG(level,...){ if (logger != NULL && logger->isActive(level)) logger->logDebug(level,__VA_ARGS__);}
|
#define LOG_DEBUG(level,...){ if (logger != NULL && logger->isActive(level)) logger->logDebug(level,__VA_ARGS__);}
|
||||||
|
|
||||||
|
|
|
@ -46,23 +46,24 @@ int GwSerial::setup(int baud, int rxpin, int txpin)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
bool GwSerial::isInitialized() { return initialized; }
|
bool GwSerial::isInitialized() { return initialized; }
|
||||||
size_t GwSerial::enqueue(const uint8_t *data, size_t len)
|
size_t GwSerial::enqueue(const uint8_t *data, size_t len, bool partial)
|
||||||
{
|
{
|
||||||
if (! isInitialized()) return 0;
|
if (! isInitialized()) return 0;
|
||||||
return buffer->addData(data, len);
|
return buffer->addData(data, len,partial);
|
||||||
}
|
}
|
||||||
GwBuffer::WriteStatus GwSerial::write(){
|
GwBuffer::WriteStatus GwSerial::write(){
|
||||||
if (! isInitialized()) return GwBuffer::ERROR;
|
if (! isInitialized()) return GwBuffer::ERROR;
|
||||||
return buffer->fetchData(writer,-1,false);
|
return buffer->fetchData(writer,-1,false);
|
||||||
}
|
}
|
||||||
void GwSerial::sendToClients(const char *buf,int sourceId){
|
size_t GwSerial::sendToClients(const char *buf,int sourceId,bool partial){
|
||||||
if ( sourceId == id) return;
|
if ( sourceId == id) return 0;
|
||||||
size_t len=strlen(buf);
|
size_t len=strlen(buf);
|
||||||
size_t enqueued=enqueue((const uint8_t*)buf,len);
|
size_t enqueued=enqueue((const uint8_t*)buf,len,partial);
|
||||||
if (enqueued != len){
|
if (enqueued != len && ! partial){
|
||||||
LOG_DEBUG(GwLog::DEBUG,"GwSerial overflow on channel %d",id);
|
LOG_DEBUG(GwLog::DEBUG,"GwSerial overflow on channel %d",id);
|
||||||
overflows++;
|
overflows++;
|
||||||
}
|
}
|
||||||
|
return enqueued;
|
||||||
}
|
}
|
||||||
void GwSerial::loop(bool handleRead){
|
void GwSerial::loop(bool handleRead){
|
||||||
write();
|
write();
|
||||||
|
|
|
@ -16,7 +16,7 @@ class GwSerial{
|
||||||
GwBuffer::WriteStatus write();
|
GwBuffer::WriteStatus write();
|
||||||
int id=-1;
|
int id=-1;
|
||||||
int overflows=0;
|
int overflows=0;
|
||||||
size_t enqueue(const uint8_t *data, size_t len);
|
size_t enqueue(const uint8_t *data, size_t len,bool partial=false);
|
||||||
HardwareSerial *serial;
|
HardwareSerial *serial;
|
||||||
public:
|
public:
|
||||||
static const int bufferSize=200;
|
static const int bufferSize=200;
|
||||||
|
@ -24,7 +24,7 @@ class GwSerial{
|
||||||
~GwSerial();
|
~GwSerial();
|
||||||
int setup(int baud,int rxpin,int txpin);
|
int setup(int baud,int rxpin,int txpin);
|
||||||
bool isInitialized();
|
bool isInitialized();
|
||||||
void sendToClients(const char *buf,int sourceId);
|
size_t sendToClients(const char *buf,int sourceId,bool partial=false);
|
||||||
void loop(bool handleRead=true);
|
void loop(bool handleRead=true);
|
||||||
bool readMessages(GwBufferWriter *writer);
|
bool readMessages(GwBufferWriter *writer);
|
||||||
void flush();
|
void flush();
|
||||||
|
|
44
src/main.cpp
44
src/main.cpp
|
@ -171,10 +171,31 @@ GwSerial *serial1=NULL;
|
||||||
|
|
||||||
|
|
||||||
class GwSerialLog : public GwLogWriter{
|
class GwSerialLog : public GwLogWriter{
|
||||||
|
static const size_t bufferSize=4096;
|
||||||
|
char *logBuffer=NULL;
|
||||||
|
int wp=0;
|
||||||
public:
|
public:
|
||||||
|
GwSerialLog(){
|
||||||
|
logBuffer=new char[bufferSize];
|
||||||
|
wp=0;
|
||||||
|
}
|
||||||
virtual ~GwSerialLog(){}
|
virtual ~GwSerialLog(){}
|
||||||
virtual void write(const char *data){
|
virtual void write(const char *data){
|
||||||
usbSerial->sendToClients(data,-1); //ignore any errors
|
int len=strlen(data);
|
||||||
|
if ((wp+len) >= (bufferSize-1)) return;
|
||||||
|
strncpy(logBuffer+wp,data,len);
|
||||||
|
wp+=len;
|
||||||
|
logBuffer[wp]=0;
|
||||||
|
}
|
||||||
|
virtual void flush(){
|
||||||
|
size_t handled=0;
|
||||||
|
while (handled < wp){
|
||||||
|
usbSerial->flush();
|
||||||
|
size_t rt=usbSerial->sendToClients(logBuffer+handled,-1,true);
|
||||||
|
handled+=rt;
|
||||||
|
}
|
||||||
|
wp=0;
|
||||||
|
logBuffer[0]=0;
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
@ -214,12 +235,14 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
void delayedRestart(){
|
bool delayedRestart(){
|
||||||
xTaskCreate([](void *p){
|
return xTaskCreate([](void *p){
|
||||||
|
GwLog *logRef=(GwLog *)p;
|
||||||
|
logRef->logDebug(GwLog::LOG,"delayed reset started");
|
||||||
delay(500);
|
delay(500);
|
||||||
ESP.restart();
|
ESP.restart();
|
||||||
vTaskDelete(NULL);
|
vTaskDelete(NULL);
|
||||||
},"reset",1000,NULL,0,NULL);
|
},"reset",1000,&logger,0,NULL) == pdPASS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -243,7 +266,9 @@ protected:
|
||||||
{
|
{
|
||||||
logger.logDebug(GwLog::LOG, "Reset Button");
|
logger.logDebug(GwLog::LOG, "Reset Button");
|
||||||
result = JSON_OK;
|
result = JSON_OK;
|
||||||
delayedRestart();
|
if (!delayedRestart()){
|
||||||
|
logger.logDebug(GwLog::ERROR,"cannot initiate restart");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -463,10 +488,10 @@ void setup() {
|
||||||
|
|
||||||
// Start TCP server
|
// Start TCP server
|
||||||
socketServer.begin();
|
socketServer.begin();
|
||||||
usbSerial->flush();
|
logger.flush();
|
||||||
|
|
||||||
logger.logDebug(GwLog::LOG,"usbRead: %s", usbReadFilter.toString().c_str());
|
logger.logDebug(GwLog::LOG,"usbRead: %s", usbReadFilter.toString().c_str());
|
||||||
usbSerial->flush();
|
logger.flush();
|
||||||
|
|
||||||
webserver.registerMainHandler("/api/reset", [](AsyncWebServerRequest *request)->GwRequestMessage *{
|
webserver.registerMainHandler("/api/reset", [](AsyncWebServerRequest *request)->GwRequestMessage *{
|
||||||
return new ResetRequest();
|
return new ResetRequest();
|
||||||
|
@ -540,7 +565,7 @@ void setup() {
|
||||||
preferences.end();
|
preferences.end();
|
||||||
|
|
||||||
logger.logDebug(GwLog::LOG,"NodeAddress=%d", NodeAddress);
|
logger.logDebug(GwLog::LOG,"NodeAddress=%d", NodeAddress);
|
||||||
usbSerial->flush();
|
logger.flush();
|
||||||
NMEA2000.SetMode(tNMEA2000::N2km_ListenAndNode, NodeAddress);
|
NMEA2000.SetMode(tNMEA2000::N2km_ListenAndNode, NodeAddress);
|
||||||
NMEA2000.SetForwardOwnMessages(false);
|
NMEA2000.SetForwardOwnMessages(false);
|
||||||
// Set the information for other bus devices, which messages we support
|
// Set the information for other bus devices, which messages we support
|
||||||
|
@ -549,7 +574,7 @@ void setup() {
|
||||||
unsigned long *op=pgns;
|
unsigned long *op=pgns;
|
||||||
while (*op != 0){
|
while (*op != 0){
|
||||||
logger.logDebug(GwLog::DEBUG,"add transmit pgn %ld",(long)(*op));
|
logger.logDebug(GwLog::DEBUG,"add transmit pgn %ld",(long)(*op));
|
||||||
usbSerial->flush();
|
logger.flush();
|
||||||
op++;
|
op++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -666,6 +691,7 @@ class NMEAMessageReceiver : public GwBufferWriter{
|
||||||
NMEAMessageReceiver receiver;
|
NMEAMessageReceiver receiver;
|
||||||
unsigned long lastHeapReport=0;
|
unsigned long lastHeapReport=0;
|
||||||
void loop() {
|
void loop() {
|
||||||
|
logger.flush();
|
||||||
gwWifi.loop();
|
gwWifi.loop();
|
||||||
unsigned long now=millis();
|
unsigned long now=millis();
|
||||||
if (HEAP_REPORT_TIME > 0 && now > (lastHeapReport+HEAP_REPORT_TIME)){
|
if (HEAP_REPORT_TIME > 0 && now > (lastHeapReport+HEAP_REPORT_TIME)){
|
||||||
|
|
Loading…
Reference in New Issue