2011-10-16 11:45:00 +02:00
|
|
|
/*
|
2013-02-24 18:40:43 +01:00
|
|
|
Minetest
|
2013-02-24 19:38:45 +01:00
|
|
|
Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2011-10-16 11:45:00 +02:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
2012-06-05 16:56:56 +02:00
|
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
2011-10-16 11:45:00 +02:00
|
|
|
(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
|
2012-06-05 16:56:56 +02:00
|
|
|
GNU Lesser General Public License for more details.
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2012-06-05 16:56:56 +02:00
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
2011-10-16 11:45:00 +02:00
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef LOG_HEADER
|
|
|
|
#define LOG_HEADER
|
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
#include <map>
|
|
|
|
#include <queue>
|
2011-10-16 11:45:00 +02:00
|
|
|
#include <string>
|
2015-10-13 09:57:44 +02:00
|
|
|
#include <fstream>
|
|
|
|
#include "threads.h"
|
|
|
|
|
|
|
|
class ILogOutput;
|
|
|
|
|
|
|
|
enum LogLevel {
|
|
|
|
LL_NONE, // Special level that is always printed
|
|
|
|
LL_ERROR,
|
|
|
|
LL_WARNING,
|
|
|
|
LL_ACTION, // In-game actions
|
|
|
|
LL_INFO,
|
|
|
|
LL_VERBOSE,
|
|
|
|
LL_MAX,
|
|
|
|
};
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
class Logger {
|
|
|
|
public:
|
|
|
|
void addOutput(ILogOutput *out);
|
|
|
|
void addOutput(ILogOutput *out, LogLevel lev);
|
|
|
|
void addOutputMaxLevel(ILogOutput *out, LogLevel lev);
|
|
|
|
void removeOutput(ILogOutput *out);
|
|
|
|
void setLevelSilenced(LogLevel lev, bool silenced);
|
2014-12-12 21:12:31 +01:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
void registerThread(const std::string &name);
|
|
|
|
void deregisterThread();
|
|
|
|
|
|
|
|
void log(LogLevel lev, const std::string &text);
|
|
|
|
// Logs without a prefix
|
|
|
|
void logRaw(LogLevel lev, const std::string &text);
|
|
|
|
|
|
|
|
void setTraceEnabled(bool enable) { m_trace_enabled = enable; }
|
|
|
|
bool getTraceEnabled() { return m_trace_enabled; }
|
|
|
|
|
|
|
|
static LogLevel stringToLevel(const std::string &name);
|
|
|
|
|
|
|
|
private:
|
|
|
|
void logToSystem(LogLevel, const std::string &text);
|
|
|
|
void logToOutputs(LogLevel, const std::string &text);
|
|
|
|
|
|
|
|
const std::string getLevelLabel(LogLevel lev);
|
|
|
|
const std::string getThreadName();
|
|
|
|
|
|
|
|
std::vector<ILogOutput *> m_outputs[LL_MAX];
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
// Should implement atomic loads and stores (even though it's only
|
|
|
|
// written to when one thread has access currently).
|
|
|
|
// Works on all known architectures (x86, ARM, MIPS).
|
|
|
|
volatile bool m_silenced_levels[LL_MAX];
|
|
|
|
std::map<threadid_t, std::string> m_thread_names;
|
|
|
|
mutable Mutex m_mutex;
|
|
|
|
bool m_trace_enabled;
|
2011-10-16 11:45:00 +02:00
|
|
|
};
|
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
class ILogOutput {
|
2011-10-16 11:45:00 +02:00
|
|
|
public:
|
2015-10-13 09:57:44 +02:00
|
|
|
virtual void log(const std::string &line) = 0;
|
2011-10-16 11:45:00 +02:00
|
|
|
};
|
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
class StreamLogOutput : public ILogOutput {
|
|
|
|
public:
|
|
|
|
StreamLogOutput(std::ostream &stream) :
|
|
|
|
stream(stream)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void log(const std::string &line)
|
|
|
|
{
|
|
|
|
stream << line << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::ostream &stream;
|
|
|
|
};
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
class FileLogOutput : public ILogOutput {
|
|
|
|
public:
|
|
|
|
void open(const std::string &filename);
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
void log(const std::string &line)
|
|
|
|
{
|
|
|
|
stream << line << std::endl;
|
|
|
|
}
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
private:
|
|
|
|
std::ofstream stream;
|
|
|
|
};
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
class LogOutputBuffer : public ILogOutput {
|
|
|
|
public:
|
|
|
|
LogOutputBuffer(Logger &logger, LogLevel lev) :
|
|
|
|
logger(logger)
|
|
|
|
{
|
|
|
|
logger.addOutput(this, lev);
|
|
|
|
}
|
|
|
|
|
|
|
|
~LogOutputBuffer()
|
|
|
|
{
|
|
|
|
logger.removeOutput(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void log(const std::string &line)
|
|
|
|
{
|
|
|
|
buffer.push(line);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool empty()
|
|
|
|
{
|
|
|
|
return buffer.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string get()
|
|
|
|
{
|
|
|
|
if (empty())
|
|
|
|
return "";
|
|
|
|
std::string s = buffer.front();
|
|
|
|
buffer.pop();
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::queue<std::string> buffer;
|
|
|
|
Logger &logger;
|
|
|
|
};
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2012-05-20 17:30:30 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
extern StreamLogOutput stdout_output;
|
|
|
|
extern StreamLogOutput stderr_output;
|
|
|
|
extern std::ostream null_stream;
|
2012-05-20 17:30:30 +02:00
|
|
|
|
2015-04-01 15:01:28 +02:00
|
|
|
extern std::ostream *dout_con_ptr;
|
|
|
|
extern std::ostream *derr_con_ptr;
|
|
|
|
extern std::ostream *dout_server_ptr;
|
|
|
|
extern std::ostream *derr_server_ptr;
|
2015-10-13 09:57:44 +02:00
|
|
|
|
|
|
|
#ifndef SERVER
|
|
|
|
extern std::ostream *dout_client_ptr;
|
|
|
|
extern std::ostream *derr_client_ptr;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
extern Logger g_logger;
|
|
|
|
|
|
|
|
// Writes directly to all LL_NONE log outputs for g_logger with no prefix.
|
|
|
|
extern std::ostream rawstream;
|
|
|
|
|
|
|
|
extern std::ostream errorstream;
|
|
|
|
extern std::ostream warningstream;
|
|
|
|
extern std::ostream actionstream;
|
|
|
|
extern std::ostream infostream;
|
|
|
|
extern std::ostream verbosestream;
|
|
|
|
extern std::ostream dstream;
|
|
|
|
|
|
|
|
#define TRACEDO(x) do { \
|
|
|
|
if (g_logger.getTraceEnabled()) { \
|
|
|
|
x; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define TRACESTREAM(x) TRACEDO(verbosestream x)
|
|
|
|
|
2015-04-01 15:01:28 +02:00
|
|
|
#define dout_con (*dout_con_ptr)
|
|
|
|
#define derr_con (*derr_con_ptr)
|
|
|
|
#define dout_server (*dout_server_ptr)
|
|
|
|
#define derr_server (*derr_server_ptr)
|
|
|
|
|
|
|
|
#ifndef SERVER
|
2015-10-13 09:57:44 +02:00
|
|
|
#define dout_client (*dout_client_ptr)
|
|
|
|
#define derr_client (*derr_client_ptr)
|
2015-04-01 15:01:28 +02:00
|
|
|
#endif
|
|
|
|
|
2011-10-16 11:45:00 +02:00
|
|
|
|
2015-10-13 09:57:44 +02:00
|
|
|
#endif
|