CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutSign UpSign In
hrydgard

CoCalc provides the best real-time collaborative environment for Jupyter Notebooks, LaTeX documents, and SageMath, scalable from individual users to large groups and classes!

GitHub Repository: hrydgard/ppsspp
Path: blob/master/Common/Log/LogManager.h
Views: 1401
1
// Copyright (C) 2003 Dolphin Project.
2
3
// This program is free software: you can redistribute it and/or modify
4
// it under the terms of the GNU General Public License as published by
5
// the Free Software Foundation, version 2.0 or later versions.
6
7
// This program is distributed in the hope that it will be useful,
8
// but WITHOUT ANY WARRANTY; without even the implied warranty of
9
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
// GNU General Public License 2.0 for more details.
11
12
// A copy of the GPL 2.0 should have been included with the program.
13
// If not, see http://www.gnu.org/licenses/
14
15
// Official SVN repository and contact information can be found at
16
// http://code.google.com/p/dolphin-emu/
17
18
#pragma once
19
20
#include "ppsspp_config.h"
21
22
#include <mutex>
23
#include <vector>
24
#include <cstdarg>
25
#include <cstdio>
26
27
#include "Common/Data/Format/IniFile.h"
28
#include "Common/CommonFuncs.h"
29
#include "Common/Log.h"
30
31
#define MAX_MESSAGES 8000
32
33
extern const char *hleCurrentThreadName;
34
35
// Struct that listeners can output how they want. For example, on Android we don't want to add
36
// timestamp or write the level as a string, those already exist.
37
struct LogMessage {
38
char timestamp[16];
39
char header[64]; // Filename/thread/etc. in front.
40
LogLevel level;
41
const char *log;
42
std::string msg; // The actual log message.
43
};
44
45
// pure virtual interface
46
class LogListener {
47
public:
48
virtual ~LogListener() {}
49
50
virtual void Log(const LogMessage &msg) = 0;
51
};
52
53
class FileLogListener : public LogListener {
54
public:
55
FileLogListener(const char *filename);
56
~FileLogListener();
57
58
void Log(const LogMessage &msg) override;
59
60
bool IsValid() { if (!fp_) return false; else return true; }
61
bool IsEnabled() const { return m_enable; }
62
void SetEnabled(bool enable) { m_enable = enable; }
63
64
const char *GetName() const { return "file"; }
65
66
private:
67
std::mutex m_log_lock;
68
FILE *fp_ = nullptr;
69
bool m_enable;
70
};
71
72
class OutputDebugStringLogListener : public LogListener {
73
public:
74
void Log(const LogMessage &msg) override;
75
};
76
77
class RingbufferLogListener : public LogListener {
78
public:
79
void Log(const LogMessage &msg) override;
80
81
bool IsEnabled() const { return enabled_; }
82
void SetEnabled(bool enable) { enabled_ = enable; }
83
84
int GetCount() const { return count_ < MAX_LOGS ? count_ : MAX_LOGS; }
85
const char *TextAt(int i) const { return messages_[(curMessage_ - i - 1) & (MAX_LOGS - 1)].msg.c_str(); }
86
LogLevel LevelAt(int i) const { return messages_[(curMessage_ - i - 1) & (MAX_LOGS - 1)].level; }
87
88
private:
89
enum { MAX_LOGS = 128 };
90
LogMessage messages_[MAX_LOGS];
91
int curMessage_ = 0;
92
int count_ = 0;
93
bool enabled_ = false;
94
};
95
96
// TODO: A simple buffered log that can be used to display the log in-window
97
// on Android etc.
98
// class BufferedLogListener { ... }
99
100
struct LogChannel {
101
char m_shortName[32]{};
102
LogLevel level;
103
bool enabled;
104
};
105
106
class ConsoleListener;
107
class StdioListener;
108
109
class LogManager {
110
private:
111
LogManager(bool *enabledSetting);
112
~LogManager();
113
114
// Prevent copies.
115
LogManager(const LogManager &) = delete;
116
void operator=(const LogManager &) = delete;
117
118
LogChannel log_[(size_t)Log::NUMBER_OF_LOGS];
119
FileLogListener *fileLog_ = nullptr;
120
#if PPSSPP_PLATFORM(WINDOWS)
121
ConsoleListener *consoleLog_ = nullptr;
122
#endif
123
StdioListener *stdioLog_ = nullptr;
124
OutputDebugStringLogListener *debuggerLog_ = nullptr;
125
RingbufferLogListener *ringLog_ = nullptr;
126
static LogManager *logManager_; // Singleton. Ugh.
127
128
std::mutex listeners_lock_;
129
std::vector<LogListener*> listeners_;
130
131
public:
132
void AddListener(LogListener *listener);
133
void RemoveListener(LogListener *listener);
134
135
static u32 GetMaxLevel() { return (u32)MAX_LOGLEVEL; }
136
static int GetNumChannels() { return (int)Log::NUMBER_OF_LOGS; }
137
138
void LogLine(LogLevel level, Log type,
139
const char *file, int line, const char *fmt, va_list args);
140
bool IsEnabled(LogLevel level, Log type);
141
142
LogChannel *GetLogChannel(Log type) {
143
return &log_[(size_t)type];
144
}
145
146
void SetLogLevel(Log type, LogLevel level) {
147
log_[(size_t)type].level = level;
148
}
149
150
void SetAllLogLevels(LogLevel level) {
151
for (int i = 0; i < (int)Log::NUMBER_OF_LOGS; ++i) {
152
log_[i].level = level;
153
}
154
}
155
156
void SetEnabled(Log type, bool enable) {
157
log_[(size_t)type].enabled = enable;
158
}
159
160
LogLevel GetLogLevel(Log type) {
161
return log_[(size_t)type].level;
162
}
163
164
#if PPSSPP_PLATFORM(WINDOWS)
165
ConsoleListener *GetConsoleListener() const {
166
return consoleLog_;
167
}
168
#endif
169
170
StdioListener *GetStdioListener() const {
171
return stdioLog_;
172
}
173
174
OutputDebugStringLogListener *GetDebuggerListener() const {
175
return debuggerLog_;
176
}
177
178
RingbufferLogListener *GetRingbufferListener() const {
179
return ringLog_;
180
}
181
182
static inline LogManager* GetInstance() {
183
return logManager_;
184
}
185
186
static void SetInstance(LogManager *logManager) {
187
logManager_ = logManager;
188
}
189
190
static void Init(bool *enabledSetting);
191
static void Shutdown();
192
193
void ChangeFileLog(const char *filename);
194
195
void SaveConfig(Section *section);
196
void LoadConfig(const Section *section, bool debugDefaults);
197
};
198
199