openspades/Sources/Core/Debug.cpp

218 lines
5.1 KiB
C++

/*
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 <http://www.gnu.org/licenses/>.
*/
#include "Debug.h"
#include "../Core/Debug.h"
#include <map>
#include "../Imports/SDL.h"
#include <string>
#include "IStream.h"
#include "FileManager.h"
#include <stdarg.h>
#include <time.h>
#include "Math.h"
#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<Backtrace> 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<Uint32, Backtrace *> 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<Uint32, Backtrace *>::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<Uint32, Backtrace *>::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<BacktraceEntry> 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);
//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, "%04d/%02d/%02d %02d:%02d:%02d [%s:%d] %s\r\n",
tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday, tm.tm_hour, tm.tm_min, tm.tm_sec,
fn.c_str(), line, str.c_str());
std::string outStr = EscapeControlCharacters(buf);
printf("%s", outStr.c_str());
if(logStream || !attemptedToInitializeLog) {
if(attemptedToInitializeLog){
logStream->Write(outStr);
logStream->Flush();
}else
accumlatedLog += buf;
}
}
}