mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-16 19:59:55 +00:00
b6718ecb10
Acquire log lock only when logger should log See merge request OpenMW/openmw!1363
68 lines
1.4 KiB
C++
68 lines
1.4 KiB
C++
#ifndef DEBUG_LOG_H
|
|
#define DEBUG_LOG_H
|
|
|
|
#include <mutex>
|
|
#include <iostream>
|
|
|
|
namespace Debug
|
|
{
|
|
enum Level
|
|
{
|
|
Error = 1,
|
|
Warning = 2,
|
|
Info = 3,
|
|
Verbose = 4,
|
|
Debug = 5,
|
|
Marker = Debug,
|
|
|
|
NoLevel = 6 // Do not filter messages in this case
|
|
};
|
|
|
|
extern Level CurrentDebugLevel;
|
|
}
|
|
|
|
class Log
|
|
{
|
|
static std::mutex sLock;
|
|
|
|
std::unique_lock<std::mutex> mLock;
|
|
public:
|
|
explicit Log(Debug::Level level)
|
|
: mShouldLog(level <= Debug::CurrentDebugLevel)
|
|
{
|
|
// No need to hold the lock if there will be no logging anyway
|
|
if (!mShouldLog)
|
|
return;
|
|
|
|
// Locks a global lock while the object is alive
|
|
mLock = std::unique_lock<std::mutex>(sLock);
|
|
|
|
// If the app has no logging system enabled, log level is not specified.
|
|
// Show all messages without marker - we just use the plain cout in this case.
|
|
if (Debug::CurrentDebugLevel == Debug::NoLevel)
|
|
return;
|
|
|
|
std::cout << static_cast<unsigned char>(level);
|
|
}
|
|
|
|
// Perfect forwarding wrappers to give the chain of objects to cout
|
|
template<typename T>
|
|
Log& operator<<(T&& rhs)
|
|
{
|
|
if (mShouldLog)
|
|
std::cout << std::forward<T>(rhs);
|
|
|
|
return *this;
|
|
}
|
|
|
|
~Log()
|
|
{
|
|
if (mShouldLog)
|
|
std::cout << std::endl;
|
|
}
|
|
|
|
private:
|
|
const bool mShouldLog;
|
|
};
|
|
|
|
#endif
|