/* Copyright (c) 2013 yvt This file is part of OpenSpades. OpenSpades 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 3 of the License, or (at your option) any later version. OpenSpades 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 OpenSpades. If not, see . */ #include "Debug.h" #include "../Core/Debug.h" #include #include "../Imports/SDL.h" #include #include "IStream.h" #include "FileManager.h" #include #include #define SPADES_USE_TLS 1 #if SPADES_USE_TLS #include "ThreadLocalStorage.h" #endif // SPADES_USE_TLS namespace spades { namespace reflection { Function::Function(const char *n, const char *f, int l): name(n), file(f), line(l){ } BacktraceEntryAdder::BacktraceEntryAdder(const BacktraceEntry& entry){ bt = Backtrace::GetGlobalBacktrace(); if(bt) bt->Push(entry); } BacktraceEntryAdder::~BacktraceEntryAdder() { if(bt) bt->Pop(); } #if SPADES_USE_TLS static AutoDeletedThreadLocalStorage backtraceTls("backtraceTls"); // TLS is initialized as global constructor and // some constructors are called before initialization of TLS. // this prevents backtrace to be used until all constructors are // called. static bool backtraceStarted = false; Backtrace *Backtrace::GetGlobalBacktrace() { if(!backtraceStarted) return NULL; Backtrace *b = backtraceTls; if(!b){ b = new Backtrace; backtraceTls = b; } return b; } void Backtrace::ThreadExiting() { } void Backtrace::StartBacktrace(){ backtraceStarted = true; } #else static std::map globalBacktrace; static Uint32 firstThread = 0; static Backtrace firstThreadBacktrace; Backtrace *Backtrace::GetGlobalBacktrace() { Uint32 thread = SDL_ThreadID(); if(firstThread == 0) firstThread = thread; if(thread == firstThread){ return &firstThreadBacktrace; } std::map::iterator it = globalBacktrace.find(thread); if(it == globalBacktrace.end()){ Backtrace *t = new Backtrace(); globalBacktrace[thread] = t; return t; }else{ return it->second; } } void Backtrace::ThreadExiting() { Uint32 thread = SDL_ThreadID(); if(thread == firstThread) return; std::map::iterator it = globalBacktrace.find(thread); if(it != globalBacktrace.end()){ delete it->second; globalBacktrace.erase(it); } } void Backtrace::StartBacktrace(){ } #endif void Backtrace::Push(const spades::reflection::BacktraceEntry &entry) { entries.push_back(entry); } void Backtrace::Pop() { SPAssert(entries.size() > 0); entries.pop_back(); } std::vector Backtrace::GetAllEntries() { return entries; } std::string Backtrace::ToString() const { std::string message; char buf[1024]; if(entries.empty()){ message += "(none)"; }else{ for(size_t i = 0; i < entries.size(); i++){ const reflection::BacktraceEntry& ent = entries[entries.size() - 1 - i]; const reflection::Function& func = ent.GetFunction(); std::string fn = func.GetFileName(); size_t ind = fn.find_last_of('/'); if(ind != std::string::npos){ fn = fn.substr(ind + 1); } sprintf(buf, "%s at %s:%d\n", func.GetName(), fn.c_str(), func.GetLineNumber()); message += buf; } } return message; } } #pragma mark - static IStream *logStream = NULL; static bool attemptedToInitializeLog = false; static std::string accumlatedLog; void StartLog() { attemptedToInitializeLog = true; logStream = FileManager::OpenForWriting("SystemMessages.log"); logStream->Write(accumlatedLog); accumlatedLog.clear(); } void LogMessage(const char *file, int line, const char *format, ...) { char buf[4096]; va_list va; va_start(va, format); vsprintf(buf, format, va); va_end(va); std::string str = buf; std::string fn = file; if(fn.rfind('/') != std::string::npos) fn=fn.substr(fn.rfind('/')+1); time_t t; struct tm tm; time(&t); tm = *localtime(&t); std::string timeStr = asctime(&tm); // remove '\n' in the end of the result of asctime(). timeStr.resize(timeStr.size()-1); //lm: using \r\n instead of \n so that some shitty windows editors (notepad f.e.) can parse this file aswell (all decent editors should ignore it anyway) sprintf(buf, "%s [%s:%d] %s\r\n", timeStr.c_str(), fn.c_str(), line, str.c_str()); printf("%s", buf); if(logStream || !attemptedToInitializeLog) { if(attemptedToInitializeLog){ logStream->Write(buf); logStream->Flush(); }else accumlatedLog += buf; } } }