/** * Furnace Tracker - multi-system chiptune tracker * Copyright (C) 2021-2022 tildearrow and contributors * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "ta-log.h" #include #include #ifdef IS_MOBILE int logLevel=LOGLEVEL_TRACE; #else int logLevel=LOGLEVEL_INFO; #endif FILE* logFile; char* logFileBuf; unsigned int logFilePosI; unsigned int logFilePosO; std::thread* logFileThread; std::mutex logFileLock; std::mutex logFileLockI; std::condition_variable logFileNotify; bool logFileAvail=false; std::atomic logPosition; LogEntry logEntries[TA_LOG_SIZE]; static constexpr unsigned int TA_LOG_MASK=TA_LOG_SIZE-1; static constexpr unsigned int TA_LOGFILE_BUF_MASK=TA_LOGFILE_BUF_SIZE-1; const char* logTypes[5]={ "ERROR", "warning", "info", "debug", "trace" }; void appendLogBuf(const LogEntry& entry) { logFileLockI.lock(); std::string toWrite=fmt::sprintf( "%02d:%02d:%02d [%s] %s\n", entry.time.tm_hour, entry.time.tm_min, entry.time.tm_sec, logTypes[entry.loglevel], entry.text ); const char* msg=toWrite.c_str(); size_t len=toWrite.size(); int remaining=(logFilePosO-logFilePosI-1)&TA_LOGFILE_BUF_SIZE; if (len>=(unsigned int)remaining) { printf("line too long to fit in log buffer!\n"); logFileLockI.unlock(); return; } if ((logFilePosI+len)>=TA_LOGFILE_BUF_SIZE) { size_t firstWrite=TA_LOGFILE_BUF_SIZE-logFilePosI; memcpy(logFileBuf+logFilePosI,msg,firstWrite); memcpy(logFileBuf,msg+firstWrite,len-firstWrite); } else { memcpy(logFileBuf+logFilePosI,msg,len); } logFilePosI=(logFilePosI+len)&TA_LOGFILE_BUF_MASK; logFileLockI.unlock(); } int writeLog(int level, const char* msg, fmt::printf_args args) { time_t thisMakesNoSense=time(NULL); int pos=(logPosition.fetch_add(1))&TA_LOG_MASK; logEntries[pos].text.assign(fmt::vsprintf(msg,args)); // why do I have to pass a pointer // can't I just pass the time_t directly?! #ifdef _WIN32 struct tm* tempTM=localtime(&thisMakesNoSense); if (tempTM==NULL) { memset(&logEntries[pos].time,0,sizeof(struct tm)); } else { memcpy(&logEntries[pos].time,tempTM,sizeof(struct tm)); } #else if (localtime_r(&thisMakesNoSense,&logEntries[pos].time)==NULL) { memset(&logEntries[pos].time,0,sizeof(struct tm)); } #endif logEntries[pos].loglevel=level; logEntries[pos].ready=true; // write to log file if (logFileAvail) { appendLogBuf(logEntries[pos]); logFileNotify.notify_one(); } if (logLevel lock(logFileLock); while (true) { unsigned int logFilePosICopy=logFilePosI; if (logFilePosICopy!=logFilePosO) { // write if (logFilePosO>logFilePosICopy) { fwrite(logFileBuf+logFilePosO,1,TA_LOGFILE_BUF_SIZE-logFilePosO,logFile); logFilePosO=0; } else { fwrite(logFileBuf+logFilePosO,1,logFilePosICopy-logFilePosO,logFile); logFilePosO=logFilePosICopy&TA_LOGFILE_BUF_MASK; } } else { // wait if (!logFileAvail) break; fflush(logFile); logFileNotify.wait(lock); } } } bool startLogFile(const char* path) { if (logFileAvail) return true; // rotate log file if possible // open log file if ((logFile=fopen(path,"w+"))==NULL) { logFileAvail=false; logW("could not open log file! (%s)",strerror(errno)); return false; } logFileBuf=new char[TA_LOGFILE_BUF_SIZE]; logFilePosI=0; logFilePosO=0; logFileAvail=true; logFileThread=new std::thread(_logFileThread); return true; } bool finishLogFile() { if (!logFileAvail) return false; logFileAvail=false; // flush logFileLockI.lock(); logFileNotify.notify_one(); logFileThread->join(); logFileLockI.unlock(); fclose(logFile); return true; }