2014-06-17 15:44:03 +00:00
|
|
|
#pragma once
|
|
|
|
#include "Utilities/MTRingbuffer.h"
|
|
|
|
|
|
|
|
//#define BUFFERED_LOGGING 1
|
|
|
|
|
|
|
|
//first parameter is of type Log::LogType and text is of type std::string
|
|
|
|
|
2015-05-22 12:16:30 +00:00
|
|
|
#define LOG_SUCCESS(logType, text, ...) log_message(logType, Log::LogSeveritySuccess, text, ##__VA_ARGS__)
|
|
|
|
#define LOG_NOTICE(logType, text, ...) log_message(logType, Log::LogSeverityNotice, text, ##__VA_ARGS__)
|
|
|
|
#define LOG_WARNING(logType, text, ...) log_message(logType, Log::LogSeverityWarning, text, ##__VA_ARGS__)
|
|
|
|
#define LOG_ERROR(logType, text, ...) log_message(logType, Log::LogSeverityError, text, ##__VA_ARGS__)
|
2014-06-17 15:44:03 +00:00
|
|
|
|
|
|
|
namespace Log
|
|
|
|
{
|
|
|
|
const unsigned int MAX_LOG_BUFFER_LENGTH = 1024*1024;
|
|
|
|
const unsigned int gBuffSize = 1000;
|
|
|
|
|
|
|
|
enum LogType : u32
|
|
|
|
{
|
|
|
|
GENERAL = 0,
|
|
|
|
LOADER,
|
|
|
|
MEMORY,
|
|
|
|
RSX,
|
|
|
|
HLE,
|
|
|
|
PPU,
|
|
|
|
SPU,
|
2014-11-02 15:50:20 +00:00
|
|
|
ARMv7,
|
2014-06-17 15:44:03 +00:00
|
|
|
TTY,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct LogTypeName
|
|
|
|
{
|
|
|
|
LogType mType;
|
|
|
|
std::string mName;
|
|
|
|
};
|
|
|
|
|
|
|
|
//well I'd love make_array() but alas manually counting is not the end of the world
|
2014-11-02 15:50:20 +00:00
|
|
|
static const std::array<LogTypeName, 9> gTypeNameTable = { {
|
2014-06-17 15:44:03 +00:00
|
|
|
{ GENERAL, "G: " },
|
|
|
|
{ LOADER, "LDR: " },
|
|
|
|
{ MEMORY, "MEM: " },
|
|
|
|
{ RSX, "RSX: " },
|
|
|
|
{ HLE, "HLE: " },
|
|
|
|
{ PPU, "PPU: " },
|
|
|
|
{ SPU, "SPU: " },
|
2014-11-02 15:50:20 +00:00
|
|
|
{ ARMv7, "ARM: " },
|
2014-06-17 15:44:03 +00:00
|
|
|
{ TTY, "TTY: " }
|
|
|
|
} };
|
|
|
|
|
|
|
|
enum LogSeverity : u32
|
|
|
|
{
|
2015-05-22 12:16:30 +00:00
|
|
|
LogSeverityNotice = 0,
|
|
|
|
LogSeverityWarning,
|
|
|
|
LogSeveritySuccess,
|
|
|
|
LogSeverityError,
|
2014-06-17 15:44:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct LogMessage
|
|
|
|
{
|
|
|
|
using size_type = u32;
|
|
|
|
LogType mType;
|
|
|
|
LogSeverity mServerity;
|
|
|
|
std::string mText;
|
|
|
|
|
2014-09-27 20:56:27 +00:00
|
|
|
u32 size() const;
|
|
|
|
void serialize(char *output) const;
|
2014-06-17 15:44:03 +00:00
|
|
|
static LogMessage deserialize(char *input, u32* size_out=nullptr);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LogListener
|
|
|
|
{
|
|
|
|
virtual ~LogListener() {};
|
2014-09-27 20:56:27 +00:00
|
|
|
virtual void log(const LogMessage &msg) = 0;
|
2014-06-17 15:44:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct LogChannel
|
|
|
|
{
|
|
|
|
LogChannel();
|
|
|
|
LogChannel(const std::string& name);
|
|
|
|
LogChannel(LogChannel& other) = delete;
|
|
|
|
LogChannel& operator = (LogChannel& other) = delete;
|
2014-09-27 20:56:27 +00:00
|
|
|
void log(const LogMessage &msg);
|
2014-06-17 15:44:03 +00:00
|
|
|
void addListener(std::shared_ptr<LogListener> listener);
|
|
|
|
void removeListener(std::shared_ptr<LogListener> listener);
|
|
|
|
std::string name;
|
|
|
|
private:
|
|
|
|
bool mEnabled;
|
|
|
|
LogSeverity mLogLevel;
|
|
|
|
std::mutex mListenerLock;
|
|
|
|
std::set<std::shared_ptr<LogListener>> mListeners;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LogManager
|
|
|
|
{
|
|
|
|
LogManager();
|
|
|
|
~LogManager();
|
|
|
|
static LogManager& getInstance();
|
|
|
|
LogChannel& getChannel(LogType type);
|
|
|
|
void log(LogMessage msg);
|
2014-06-27 07:22:00 +00:00
|
|
|
void addListener(std::shared_ptr<LogListener> listener);
|
|
|
|
void removeListener(std::shared_ptr<LogListener> listener);
|
2014-06-17 15:44:03 +00:00
|
|
|
#ifdef BUFFERED_LOGGING
|
|
|
|
void consumeLog();
|
|
|
|
#endif
|
|
|
|
private:
|
|
|
|
#ifdef BUFFERED_LOGGING
|
|
|
|
MTRingbuffer<char, MAX_LOG_BUFFER_LENGTH> mBuffer;
|
|
|
|
std::condition_variable mBufferReady;
|
|
|
|
std::mutex mStatusMut;
|
|
|
|
std::atomic<bool> mExiting;
|
|
|
|
std::thread mLogConsumer;
|
|
|
|
#endif
|
|
|
|
std::array<LogChannel, std::tuple_size<decltype(gTypeNameTable)>::value> mChannels;
|
|
|
|
//std::array<LogChannel,gTypeNameTable.size()> mChannels; //TODO: use this once Microsoft sorts their shit out
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::GENERAL; } } GENERAL;
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::LOADER; } } LOADER;
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::MEMORY; } } MEMORY;
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::RSX; } } RSX;
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::HLE; } } HLE;
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::PPU; } } PPU;
|
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::SPU; } } SPU;
|
2014-11-02 15:50:20 +00:00
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::ARMv7; } } ARMv7;
|
2014-06-17 15:44:03 +00:00
|
|
|
static struct { inline operator Log::LogType() { return Log::LogType::TTY; } } TTY;
|
2014-06-27 13:26:46 +00:00
|
|
|
|
2015-01-18 22:54:56 +00:00
|
|
|
void log_message(Log::LogType type, Log::LogSeverity sev, const char* text);
|
2015-01-19 13:31:02 +00:00
|
|
|
void log_message(Log::LogType type, Log::LogSeverity sev, std::string text);
|
2015-01-13 14:54:36 +00:00
|
|
|
|
2015-01-18 22:54:56 +00:00
|
|
|
template<typename... Targs>
|
|
|
|
__noinline void log_message(Log::LogType type, Log::LogSeverity sev, const char* fmt, Targs... args)
|
2015-01-13 14:54:36 +00:00
|
|
|
{
|
2015-03-07 13:39:07 +00:00
|
|
|
log_message(type, sev, fmt::Format(fmt, fmt::do_unveil(args)...));
|
2014-06-27 13:26:46 +00:00
|
|
|
}
|