2018-08-03 09:19:12 +00:00
|
|
|
#include "debugging.hpp"
|
|
|
|
|
2020-12-04 10:17:49 +00:00
|
|
|
#include <chrono>
|
2024-06-15 11:34:13 +00:00
|
|
|
#include <deque>
|
2022-05-25 18:29:02 +00:00
|
|
|
#include <fstream>
|
2023-01-30 08:17:56 +00:00
|
|
|
#include <iostream>
|
2020-12-08 21:23:11 +00:00
|
|
|
#include <memory>
|
2022-06-08 21:25:50 +00:00
|
|
|
|
2024-03-14 23:39:19 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
// TODO: why is this necessary? this has /external:I
|
|
|
|
#pragma warning(push)
|
|
|
|
#pragma warning(disable : 4702)
|
|
|
|
#endif
|
2022-06-08 21:25:50 +00:00
|
|
|
#include <boost/iostreams/stream.hpp>
|
2024-03-14 23:39:19 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(pop)
|
|
|
|
#endif
|
2020-12-04 10:17:49 +00:00
|
|
|
|
2018-08-14 07:17:05 +00:00
|
|
|
#include <components/crashcatcher/crashcatcher.hpp>
|
2022-07-02 22:02:29 +00:00
|
|
|
#include <components/files/conversion.hpp>
|
2023-03-18 09:30:48 +00:00
|
|
|
#include <components/misc/strings/conversion.hpp>
|
2023-01-30 08:17:56 +00:00
|
|
|
#include <components/misc/strings/lower.hpp>
|
|
|
|
|
2018-12-11 23:12:13 +00:00
|
|
|
#ifdef _WIN32
|
2022-04-23 17:00:47 +00:00
|
|
|
#include <components/crashcatcher/windows_crashcatcher.hpp>
|
2022-07-02 22:02:29 +00:00
|
|
|
#include <components/files/conversion.hpp>
|
2023-10-15 14:44:24 +00:00
|
|
|
#include <components/misc/windows.hpp>
|
2023-07-05 19:25:22 +00:00
|
|
|
|
|
|
|
#include <Knownfolders.h>
|
|
|
|
|
|
|
|
#pragma push_macro("FAR")
|
|
|
|
#pragma push_macro("NEAR")
|
|
|
|
#undef FAR
|
|
|
|
#define FAR
|
|
|
|
#undef NEAR
|
|
|
|
#define NEAR
|
|
|
|
#include <Shlobj.h>
|
|
|
|
#pragma pop_macro("NEAR")
|
|
|
|
#pragma pop_macro("FAR")
|
|
|
|
|
2018-12-11 23:12:13 +00:00
|
|
|
#endif
|
|
|
|
|
2022-06-06 22:22:43 +00:00
|
|
|
#include <SDL_messagebox.h>
|
|
|
|
|
2018-08-03 09:19:12 +00:00
|
|
|
namespace Debug
|
|
|
|
{
|
2018-12-11 22:58:46 +00:00
|
|
|
#ifdef _WIN32
|
Only reroute stdout etc. to new console if not already redirected
This should fix the issue where Windows Release builds (compiled as
/SUBSYSTEM:WINDOWS instead of /SUBSYSTEM:CONSOLE) can't have their
output redirected.
Basically, a console application creates a console if not given one, so
you get a console window behind OpenMW while it's running. It was
decided that this was ugly, so we set Release builds to be windows
applications, which don't get an automatic console and don't
automatically connect to a console if given one anyway.
Of course, we still wanted to actually be able to print to a console if
given one, so we manually attach to the parent process' console if it
exists, then reopen the standard streams connected to CON, the Windows
pseudo-file representing the current console.
This is a little like connecting a second wire into a dumb terminal in
that you're pumping characters into the display rather than onto a
pipeline, so output can't be redirected.
It turns out, though, that if a /SUBSYSTEM:WINDOWS application has its
standard streams redirected by the calling process, it still gets its
handles as normal, so everything starts off connected just how we want
it and we were clobbering this good setup with the straight-to-console
fix.
All we need to do to fix that is check if we've got valid standard
handles and that they go somewhere useful, and if so, avoid reopening
them once the console is attached. Simples.
2020-11-14 02:04:46 +00:00
|
|
|
bool isRedirected(DWORD nStdHandle)
|
|
|
|
{
|
|
|
|
DWORD fileType = GetFileType(GetStdHandle(nStdHandle));
|
|
|
|
|
|
|
|
return (fileType == FILE_TYPE_DISK) || (fileType == FILE_TYPE_PIPE);
|
|
|
|
}
|
|
|
|
|
2018-12-11 22:58:46 +00:00
|
|
|
bool attachParentConsole()
|
|
|
|
{
|
|
|
|
if (GetConsoleWindow() != nullptr)
|
|
|
|
return true;
|
|
|
|
|
Only reroute stdout etc. to new console if not already redirected
This should fix the issue where Windows Release builds (compiled as
/SUBSYSTEM:WINDOWS instead of /SUBSYSTEM:CONSOLE) can't have their
output redirected.
Basically, a console application creates a console if not given one, so
you get a console window behind OpenMW while it's running. It was
decided that this was ugly, so we set Release builds to be windows
applications, which don't get an automatic console and don't
automatically connect to a console if given one anyway.
Of course, we still wanted to actually be able to print to a console if
given one, so we manually attach to the parent process' console if it
exists, then reopen the standard streams connected to CON, the Windows
pseudo-file representing the current console.
This is a little like connecting a second wire into a dumb terminal in
that you're pumping characters into the display rather than onto a
pipeline, so output can't be redirected.
It turns out, though, that if a /SUBSYSTEM:WINDOWS application has its
standard streams redirected by the calling process, it still gets its
handles as normal, so everything starts off connected just how we want
it and we were clobbering this good setup with the straight-to-console
fix.
All we need to do to fix that is check if we've got valid standard
handles and that they go somewhere useful, and if so, avoid reopening
them once the console is attached. Simples.
2020-11-14 02:04:46 +00:00
|
|
|
bool inRedirected = isRedirected(STD_INPUT_HANDLE);
|
|
|
|
bool outRedirected = isRedirected(STD_OUTPUT_HANDLE);
|
|
|
|
bool errRedirected = isRedirected(STD_ERROR_HANDLE);
|
|
|
|
|
2024-04-16 00:14:20 +00:00
|
|
|
// Note: Do not spend three days reinvestigating this PowerShell bug thinking its our bug.
|
|
|
|
// https://gitlab.com/OpenMW/openmw/-/merge_requests/408#note_447467393
|
2024-04-16 12:14:36 +00:00
|
|
|
// The handles look valid, but GetFinalPathNameByHandleA can't tell what files they go to and writing to them
|
|
|
|
// doesn't work.
|
2024-04-16 00:14:20 +00:00
|
|
|
|
2018-12-11 22:58:46 +00:00
|
|
|
if (AttachConsole(ATTACH_PARENT_PROCESS))
|
|
|
|
{
|
|
|
|
fflush(stdout);
|
|
|
|
fflush(stderr);
|
|
|
|
std::cout.flush();
|
|
|
|
std::cerr.flush();
|
|
|
|
|
|
|
|
// this looks dubious but is really the right way
|
Only reroute stdout etc. to new console if not already redirected
This should fix the issue where Windows Release builds (compiled as
/SUBSYSTEM:WINDOWS instead of /SUBSYSTEM:CONSOLE) can't have their
output redirected.
Basically, a console application creates a console if not given one, so
you get a console window behind OpenMW while it's running. It was
decided that this was ugly, so we set Release builds to be windows
applications, which don't get an automatic console and don't
automatically connect to a console if given one anyway.
Of course, we still wanted to actually be able to print to a console if
given one, so we manually attach to the parent process' console if it
exists, then reopen the standard streams connected to CON, the Windows
pseudo-file representing the current console.
This is a little like connecting a second wire into a dumb terminal in
that you're pumping characters into the display rather than onto a
pipeline, so output can't be redirected.
It turns out, though, that if a /SUBSYSTEM:WINDOWS application has its
standard streams redirected by the calling process, it still gets its
handles as normal, so everything starts off connected just how we want
it and we were clobbering this good setup with the straight-to-console
fix.
All we need to do to fix that is check if we've got valid standard
handles and that they go somewhere useful, and if so, avoid reopening
them once the console is attached. Simples.
2020-11-14 02:04:46 +00:00
|
|
|
if (!inRedirected)
|
|
|
|
{
|
|
|
|
_wfreopen(L"CON", L"r", stdin);
|
|
|
|
freopen("CON", "r", stdin);
|
2024-04-16 00:10:39 +00:00
|
|
|
std::cin.clear();
|
Only reroute stdout etc. to new console if not already redirected
This should fix the issue where Windows Release builds (compiled as
/SUBSYSTEM:WINDOWS instead of /SUBSYSTEM:CONSOLE) can't have their
output redirected.
Basically, a console application creates a console if not given one, so
you get a console window behind OpenMW while it's running. It was
decided that this was ugly, so we set Release builds to be windows
applications, which don't get an automatic console and don't
automatically connect to a console if given one anyway.
Of course, we still wanted to actually be able to print to a console if
given one, so we manually attach to the parent process' console if it
exists, then reopen the standard streams connected to CON, the Windows
pseudo-file representing the current console.
This is a little like connecting a second wire into a dumb terminal in
that you're pumping characters into the display rather than onto a
pipeline, so output can't be redirected.
It turns out, though, that if a /SUBSYSTEM:WINDOWS application has its
standard streams redirected by the calling process, it still gets its
handles as normal, so everything starts off connected just how we want
it and we were clobbering this good setup with the straight-to-console
fix.
All we need to do to fix that is check if we've got valid standard
handles and that they go somewhere useful, and if so, avoid reopening
them once the console is attached. Simples.
2020-11-14 02:04:46 +00:00
|
|
|
}
|
|
|
|
if (!outRedirected)
|
|
|
|
{
|
|
|
|
_wfreopen(L"CON", L"w", stdout);
|
|
|
|
freopen("CON", "w", stdout);
|
2024-04-16 00:10:39 +00:00
|
|
|
std::cout.clear();
|
Only reroute stdout etc. to new console if not already redirected
This should fix the issue where Windows Release builds (compiled as
/SUBSYSTEM:WINDOWS instead of /SUBSYSTEM:CONSOLE) can't have their
output redirected.
Basically, a console application creates a console if not given one, so
you get a console window behind OpenMW while it's running. It was
decided that this was ugly, so we set Release builds to be windows
applications, which don't get an automatic console and don't
automatically connect to a console if given one anyway.
Of course, we still wanted to actually be able to print to a console if
given one, so we manually attach to the parent process' console if it
exists, then reopen the standard streams connected to CON, the Windows
pseudo-file representing the current console.
This is a little like connecting a second wire into a dumb terminal in
that you're pumping characters into the display rather than onto a
pipeline, so output can't be redirected.
It turns out, though, that if a /SUBSYSTEM:WINDOWS application has its
standard streams redirected by the calling process, it still gets its
handles as normal, so everything starts off connected just how we want
it and we were clobbering this good setup with the straight-to-console
fix.
All we need to do to fix that is check if we've got valid standard
handles and that they go somewhere useful, and if so, avoid reopening
them once the console is attached. Simples.
2020-11-14 02:04:46 +00:00
|
|
|
}
|
|
|
|
if (!errRedirected)
|
|
|
|
{
|
|
|
|
_wfreopen(L"CON", L"w", stderr);
|
|
|
|
freopen("CON", "w", stderr);
|
2024-04-16 00:10:39 +00:00
|
|
|
std::cerr.clear();
|
Only reroute stdout etc. to new console if not already redirected
This should fix the issue where Windows Release builds (compiled as
/SUBSYSTEM:WINDOWS instead of /SUBSYSTEM:CONSOLE) can't have their
output redirected.
Basically, a console application creates a console if not given one, so
you get a console window behind OpenMW while it's running. It was
decided that this was ugly, so we set Release builds to be windows
applications, which don't get an automatic console and don't
automatically connect to a console if given one anyway.
Of course, we still wanted to actually be able to print to a console if
given one, so we manually attach to the parent process' console if it
exists, then reopen the standard streams connected to CON, the Windows
pseudo-file representing the current console.
This is a little like connecting a second wire into a dumb terminal in
that you're pumping characters into the display rather than onto a
pipeline, so output can't be redirected.
It turns out, though, that if a /SUBSYSTEM:WINDOWS application has its
standard streams redirected by the calling process, it still gets its
handles as normal, so everything starts off connected just how we want
it and we were clobbering this good setup with the straight-to-console
fix.
All we need to do to fix that is check if we've got valid standard
handles and that they go somewhere useful, and if so, avoid reopening
them once the console is attached. Simples.
2020-11-14 02:04:46 +00:00
|
|
|
}
|
2018-12-11 22:58:46 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-05-08 20:43:30 +00:00
|
|
|
static LogListener logListener;
|
|
|
|
void setLogListener(LogListener listener)
|
|
|
|
{
|
|
|
|
logListener = std::move(listener);
|
|
|
|
}
|
|
|
|
|
2022-06-06 22:22:43 +00:00
|
|
|
class DebugOutputBase : public boost::iostreams::sink
|
2018-08-03 09:19:12 +00:00
|
|
|
{
|
2022-06-06 22:22:43 +00:00
|
|
|
public:
|
|
|
|
virtual std::streamsize write(const char* str, std::streamsize size)
|
|
|
|
{
|
|
|
|
if (size <= 0)
|
|
|
|
return size;
|
2024-06-15 11:30:16 +00:00
|
|
|
std::string_view msg{ str, static_cast<size_t>(size) };
|
2022-06-06 22:22:43 +00:00
|
|
|
|
|
|
|
// Skip debug level marker
|
2024-06-15 11:30:16 +00:00
|
|
|
Level level = All;
|
|
|
|
if (Log::sWriteLevel)
|
|
|
|
{
|
|
|
|
level = getLevelMarker(msg[0]);
|
2022-06-06 22:22:43 +00:00
|
|
|
msg = msg.substr(1);
|
2024-06-15 11:30:16 +00:00
|
|
|
}
|
2022-06-06 22:22:43 +00:00
|
|
|
|
|
|
|
char prefix[32];
|
2024-03-14 23:39:19 +00:00
|
|
|
std::size_t prefixSize;
|
2022-06-06 22:22:43 +00:00
|
|
|
{
|
|
|
|
prefix[0] = '[';
|
|
|
|
const auto now = std::chrono::system_clock::now();
|
|
|
|
const auto time = std::chrono::system_clock::to_time_t(now);
|
2022-09-24 14:20:42 +00:00
|
|
|
tm time_info{};
|
|
|
|
#ifdef _WIN32
|
|
|
|
(void)localtime_s(&time_info, &time);
|
|
|
|
#else
|
|
|
|
(void)localtime_r(&time, &time_info);
|
|
|
|
#endif
|
|
|
|
prefixSize = std::strftime(prefix + 1, sizeof(prefix) - 1, "%T", &time_info) + 1;
|
2022-06-06 22:22:43 +00:00
|
|
|
char levelLetter = " EWIVD*"[int(level)];
|
|
|
|
const auto ms = std::chrono::duration_cast<std::chrono::milliseconds>(now.time_since_epoch()).count();
|
|
|
|
prefixSize += snprintf(prefix + prefixSize, sizeof(prefix) - prefixSize, ".%03u %c] ",
|
|
|
|
static_cast<unsigned>(ms % 1000), levelLetter);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!msg.empty())
|
|
|
|
{
|
|
|
|
if (msg[0] == 0)
|
|
|
|
break;
|
|
|
|
size_t lineSize = 1;
|
|
|
|
while (lineSize < msg.size() && msg[lineSize - 1] != '\n')
|
|
|
|
lineSize++;
|
|
|
|
writeImpl(prefix, prefixSize, level);
|
|
|
|
writeImpl(msg.data(), lineSize, level);
|
|
|
|
if (logListener)
|
|
|
|
logListener(level, std::string_view(prefix, prefixSize), std::string_view(msg.data(), lineSize));
|
|
|
|
msg = msg.substr(lineSize);
|
|
|
|
}
|
|
|
|
|
2020-12-04 10:17:49 +00:00
|
|
|
return size;
|
2022-06-06 22:22:43 +00:00
|
|
|
}
|
2020-12-04 10:17:49 +00:00
|
|
|
|
2022-06-06 22:22:43 +00:00
|
|
|
virtual ~DebugOutputBase() = default;
|
2020-12-04 10:17:49 +00:00
|
|
|
|
2022-06-06 22:22:43 +00:00
|
|
|
protected:
|
2024-06-15 11:30:16 +00:00
|
|
|
static Level getLevelMarker(char marker)
|
2018-08-03 09:19:12 +00:00
|
|
|
{
|
2024-06-15 11:30:16 +00:00
|
|
|
if (0 <= marker && static_cast<unsigned>(marker) < static_cast<unsigned>(All))
|
|
|
|
return static_cast<Level>(marker);
|
|
|
|
return All;
|
2020-12-04 10:17:49 +00:00
|
|
|
}
|
|
|
|
|
2024-03-15 00:11:19 +00:00
|
|
|
virtual std::streamsize writeImpl(const char* str, std::streamsize size, Level debugLevel)
|
|
|
|
{
|
|
|
|
return size;
|
|
|
|
}
|
2022-06-06 22:22:43 +00:00
|
|
|
};
|
2018-08-03 09:19:12 +00:00
|
|
|
|
2022-06-06 22:22:43 +00:00
|
|
|
#if defined _WIN32 && defined _DEBUG
|
|
|
|
class DebugOutput : public DebugOutputBase
|
2018-08-03 09:19:12 +00:00
|
|
|
{
|
2022-06-06 22:22:43 +00:00
|
|
|
public:
|
|
|
|
std::streamsize writeImpl(const char* str, std::streamsize size, Level debugLevel)
|
2018-08-03 09:19:12 +00:00
|
|
|
{
|
2022-06-06 22:22:43 +00:00
|
|
|
// Make a copy for null termination
|
|
|
|
std::string tmp(str, static_cast<unsigned int>(size));
|
|
|
|
// Write string to Visual Studio Debug output
|
|
|
|
OutputDebugString(tmp.c_str());
|
|
|
|
return size;
|
2018-08-03 09:19:12 +00:00
|
|
|
}
|
|
|
|
|
2022-06-06 22:22:43 +00:00
|
|
|
virtual ~DebugOutput() = default;
|
|
|
|
};
|
|
|
|
#else
|
2018-08-03 09:19:12 +00:00
|
|
|
|
2024-05-12 22:40:22 +00:00
|
|
|
namespace
|
2018-08-03 09:19:12 +00:00
|
|
|
{
|
2024-05-12 23:08:05 +00:00
|
|
|
struct Record
|
|
|
|
{
|
|
|
|
std::string mValue;
|
|
|
|
Level mLevel;
|
|
|
|
};
|
|
|
|
|
2024-06-15 11:34:13 +00:00
|
|
|
std::deque<Record> globalBuffer;
|
2024-05-12 23:08:05 +00:00
|
|
|
|
2024-05-12 22:40:22 +00:00
|
|
|
Color getColor(Level level)
|
2018-08-03 09:19:12 +00:00
|
|
|
{
|
2024-05-12 22:40:22 +00:00
|
|
|
switch (level)
|
2022-06-06 22:22:43 +00:00
|
|
|
{
|
2024-05-12 22:40:22 +00:00
|
|
|
case Error:
|
|
|
|
return Red;
|
|
|
|
case Warning:
|
|
|
|
return Yellow;
|
|
|
|
case Info:
|
|
|
|
return Reset;
|
|
|
|
case Verbose:
|
|
|
|
return DarkGray;
|
|
|
|
case Debug:
|
|
|
|
return DarkGray;
|
2024-06-15 11:30:16 +00:00
|
|
|
case All:
|
2024-05-12 22:40:22 +00:00
|
|
|
return Reset;
|
2022-06-06 22:22:43 +00:00
|
|
|
}
|
2024-05-12 22:40:22 +00:00
|
|
|
return Reset;
|
2022-06-06 22:22:43 +00:00
|
|
|
}
|
|
|
|
|
2024-05-12 22:40:22 +00:00
|
|
|
bool useColoredOutput()
|
2022-06-06 22:22:43 +00:00
|
|
|
{
|
|
|
|
#if defined(_WIN32)
|
2024-05-12 22:40:22 +00:00
|
|
|
if (std::getenv("NO_COLOR") != nullptr)
|
2024-04-13 16:09:48 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
DWORD mode;
|
|
|
|
if (GetConsoleMode(GetStdHandle(STD_ERROR_HANDLE), &mode) && mode & ENABLE_VIRTUAL_TERMINAL_PROCESSING)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// some console emulators may not use the Win32 API, so try the Unixy approach
|
2024-05-12 22:40:22 +00:00
|
|
|
return std::getenv("TERM") != nullptr && GetFileType(GetStdHandle(STD_ERROR_HANDLE)) == FILE_TYPE_CHAR;
|
2022-06-06 22:22:43 +00:00
|
|
|
#else
|
2024-05-12 22:40:22 +00:00
|
|
|
return std::getenv("TERM") != nullptr && std::getenv("NO_COLOR") == nullptr && isatty(fileno(stderr));
|
2022-06-06 22:22:43 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2024-05-12 22:40:22 +00:00
|
|
|
class Identity
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit Identity(std::ostream& stream)
|
|
|
|
: mStream(stream)
|
|
|
|
{
|
|
|
|
}
|
2022-06-06 22:22:43 +00:00
|
|
|
|
2024-05-12 22:40:22 +00:00
|
|
|
void write(const char* str, std::streamsize size, Level /*level*/)
|
|
|
|
{
|
|
|
|
mStream.write(str, size);
|
|
|
|
mStream.flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::ostream& mStream;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Coloured
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit Coloured(std::ostream& stream)
|
|
|
|
: mStream(stream)
|
|
|
|
// TODO: check which stream is stderr?
|
|
|
|
, mUseColor(useColoredOutput())
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const char* str, std::streamsize size, Level level)
|
|
|
|
{
|
|
|
|
if (mUseColor)
|
|
|
|
mStream << "\033[0;" << getColor(level) << 'm';
|
|
|
|
mStream.write(str, size);
|
|
|
|
if (mUseColor)
|
|
|
|
mStream << "\033[0;" << Reset << 'm';
|
|
|
|
mStream.flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::ostream& mStream;
|
|
|
|
bool mUseColor;
|
|
|
|
};
|
|
|
|
|
2024-05-12 23:08:05 +00:00
|
|
|
class Buffer
|
|
|
|
{
|
|
|
|
public:
|
2024-06-15 11:34:13 +00:00
|
|
|
explicit Buffer(std::size_t capacity, std::deque<Record>& buffer)
|
|
|
|
: mCapacity(capacity)
|
|
|
|
, mBuffer(buffer)
|
2024-05-12 23:08:05 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const char* str, std::streamsize size, Level debugLevel)
|
|
|
|
{
|
2024-06-15 11:34:13 +00:00
|
|
|
while (mBuffer.size() >= mCapacity)
|
|
|
|
mBuffer.pop_front();
|
2024-05-12 23:08:05 +00:00
|
|
|
mBuffer.push_back(Record{ std::string(str, size), debugLevel });
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2024-06-15 11:34:13 +00:00
|
|
|
std::size_t mCapacity;
|
|
|
|
std::deque<Record>& mBuffer;
|
2024-05-12 23:08:05 +00:00
|
|
|
};
|
|
|
|
|
2024-05-12 22:40:22 +00:00
|
|
|
template <class First, class Second>
|
|
|
|
class Tee : public DebugOutputBase
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit Tee(First first, Second second)
|
|
|
|
: mFirst(first)
|
|
|
|
, mSecond(second)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
std::streamsize writeImpl(const char* str, std::streamsize size, Level debugLevel) override
|
|
|
|
{
|
|
|
|
mFirst.write(str, size, debugLevel);
|
|
|
|
mSecond.write(str, size, debugLevel);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
First mFirst;
|
|
|
|
Second mSecond;
|
|
|
|
};
|
|
|
|
}
|
2022-06-06 22:22:43 +00:00
|
|
|
#endif
|
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
static std::unique_ptr<std::ostream> rawStdout = nullptr;
|
|
|
|
static std::unique_ptr<std::ostream> rawStderr = nullptr;
|
|
|
|
static std::unique_ptr<std::mutex> rawStderrMutex = nullptr;
|
|
|
|
static std::ofstream logfile;
|
2022-01-12 23:20:16 +00:00
|
|
|
|
|
|
|
#if defined(_WIN32) && defined(_DEBUG)
|
2024-06-15 10:42:50 +00:00
|
|
|
static boost::iostreams::stream_buffer<DebugOutput> sb;
|
2022-01-12 23:20:16 +00:00
|
|
|
#else
|
2024-06-15 10:42:50 +00:00
|
|
|
static boost::iostreams::stream_buffer<Tee<Identity, Coloured>> standardOut;
|
|
|
|
static boost::iostreams::stream_buffer<Tee<Identity, Coloured>> standardErr;
|
|
|
|
static boost::iostreams::stream_buffer<Tee<Buffer, Coloured>> bufferedOut;
|
|
|
|
static boost::iostreams::stream_buffer<Tee<Buffer, Coloured>> bufferedErr;
|
2022-01-12 23:20:16 +00:00
|
|
|
#endif
|
2020-12-08 21:23:11 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
std::ostream& getRawStdout()
|
|
|
|
{
|
|
|
|
return rawStdout ? *rawStdout : std::cout;
|
|
|
|
}
|
2020-12-08 21:23:11 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
std::ostream& getRawStderr()
|
|
|
|
{
|
|
|
|
return rawStderr ? *rawStderr : std::cerr;
|
|
|
|
}
|
2021-12-19 21:49:41 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
Misc::Locked<std::ostream&> getLockedRawStderr()
|
|
|
|
{
|
|
|
|
return Misc::Locked<std::ostream&>(*rawStderrMutex, getRawStderr());
|
|
|
|
}
|
2021-12-19 21:49:41 +00:00
|
|
|
|
2024-06-15 11:08:13 +00:00
|
|
|
Level getDebugLevel()
|
|
|
|
{
|
|
|
|
if (const char* env = getenv("OPENMW_DEBUG_LEVEL"))
|
|
|
|
{
|
|
|
|
const std::string_view value(env);
|
|
|
|
if (value == "ERROR")
|
|
|
|
return Error;
|
|
|
|
if (value == "WARNING")
|
|
|
|
return Warning;
|
|
|
|
if (value == "INFO")
|
|
|
|
return Info;
|
|
|
|
if (value == "VERBOSE")
|
|
|
|
return Verbose;
|
|
|
|
if (value == "DEBUG")
|
|
|
|
return Debug;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Verbose;
|
|
|
|
}
|
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
void setupLogging(const std::filesystem::path& logDir, std::string_view appName)
|
|
|
|
{
|
2024-06-15 11:30:16 +00:00
|
|
|
Log::sMinDebugLevel = getDebugLevel();
|
|
|
|
Log::sWriteLevel = true;
|
2024-06-15 11:08:13 +00:00
|
|
|
|
2024-05-12 23:08:05 +00:00
|
|
|
#if !(defined(_WIN32) && defined(_DEBUG))
|
2024-06-15 10:42:50 +00:00
|
|
|
const std::string logName = Misc::StringUtils::lowerCase(appName) + ".log";
|
|
|
|
logfile.open(logDir / logName, std::ios::out);
|
2022-01-12 23:20:16 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
Identity log(logfile);
|
2022-01-12 23:20:16 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
for (const Record& v : globalBuffer)
|
|
|
|
log.write(v.mValue.data(), v.mValue.size(), v.mLevel);
|
2024-05-12 23:08:05 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
globalBuffer.clear();
|
2024-05-12 23:08:05 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
standardOut.open(Tee(log, Coloured(*rawStdout)));
|
|
|
|
standardErr.open(Tee(log, Coloured(*rawStderr)));
|
2024-05-12 23:08:05 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
std::cout.rdbuf(&standardOut);
|
|
|
|
std::cerr.rdbuf(&standardErr);
|
2018-08-14 06:30:27 +00:00
|
|
|
#endif
|
2023-07-05 19:38:46 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
2024-06-15 10:42:50 +00:00
|
|
|
if (Crash::CrashCatcher::instance())
|
|
|
|
{
|
|
|
|
Crash::CrashCatcher::instance()->updateDumpPath(logDir);
|
|
|
|
}
|
2023-07-05 19:38:46 +00:00
|
|
|
#endif
|
2024-06-15 10:42:50 +00:00
|
|
|
}
|
2018-08-14 06:30:27 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
int wrapApplication(
|
|
|
|
int (*innerApplication)(int argc, char* argv[]), int argc, char* argv[], std::string_view appName)
|
|
|
|
{
|
2022-01-12 23:20:16 +00:00
|
|
|
#if defined _WIN32
|
2024-06-15 10:42:50 +00:00
|
|
|
(void)attachParentConsole();
|
2022-01-12 23:20:16 +00:00
|
|
|
#endif
|
2024-06-15 10:42:50 +00:00
|
|
|
rawStdout = std::make_unique<std::ostream>(std::cout.rdbuf());
|
|
|
|
rawStderr = std::make_unique<std::ostream>(std::cerr.rdbuf());
|
|
|
|
rawStderrMutex = std::make_unique<std::mutex>();
|
2018-08-14 06:30:27 +00:00
|
|
|
|
2024-05-12 23:08:05 +00:00
|
|
|
#if defined(_WIN32) && defined(_DEBUG)
|
2024-06-15 10:42:50 +00:00
|
|
|
// Redirect cout and cerr to VS debug output when running in debug mode
|
|
|
|
sb.open(DebugOutput());
|
|
|
|
std::cout.rdbuf(&sb);
|
|
|
|
std::cerr.rdbuf(&sb);
|
2024-05-12 23:08:05 +00:00
|
|
|
#else
|
2024-06-15 11:34:13 +00:00
|
|
|
constexpr std::size_t bufferCapacity = 1024;
|
|
|
|
|
|
|
|
bufferedOut.open(Tee(Buffer(bufferCapacity, globalBuffer), Coloured(*rawStdout)));
|
|
|
|
bufferedErr.open(Tee(Buffer(bufferCapacity, globalBuffer), Coloured(*rawStderr)));
|
2024-05-12 23:08:05 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
std::cout.rdbuf(&bufferedOut);
|
|
|
|
std::cerr.rdbuf(&bufferedErr);
|
2024-05-12 23:08:05 +00:00
|
|
|
#endif
|
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
int ret = 0;
|
|
|
|
try
|
2022-07-01 20:25:42 +00:00
|
|
|
{
|
2024-06-15 10:42:50 +00:00
|
|
|
if (const auto env = std::getenv("OPENMW_DISABLE_CRASH_CATCHER");
|
|
|
|
env == nullptr || Misc::StringUtils::toNumeric<int>(env, 0) == 0)
|
2023-07-05 19:25:22 +00:00
|
|
|
{
|
2024-06-15 10:42:50 +00:00
|
|
|
#if defined(_WIN32)
|
|
|
|
const std::string crashDumpName = Misc::StringUtils::lowerCase(appName) + "-crash.dmp";
|
|
|
|
const std::string freezeDumpName = Misc::StringUtils::lowerCase(appName) + "-freeze.dmp";
|
|
|
|
std::filesystem::path dumpDirectory = std::filesystem::temp_directory_path();
|
|
|
|
PWSTR userProfile = nullptr;
|
|
|
|
if (SUCCEEDED(SHGetKnownFolderPath(FOLDERID_Profile, 0, nullptr, &userProfile)))
|
|
|
|
{
|
|
|
|
dumpDirectory = userProfile;
|
|
|
|
}
|
|
|
|
CoTaskMemFree(userProfile);
|
|
|
|
Crash::CrashCatcher crashy(argc, argv, dumpDirectory, crashDumpName, freezeDumpName);
|
2020-12-12 16:29:29 +00:00
|
|
|
#else
|
2024-06-15 10:42:50 +00:00
|
|
|
const std::string crashLogName = Misc::StringUtils::lowerCase(appName) + "-crash.log";
|
|
|
|
// install the crash handler as soon as possible.
|
|
|
|
crashCatcherInstall(argc, argv, std::filesystem::temp_directory_path() / crashLogName);
|
2020-12-12 16:29:29 +00:00
|
|
|
#endif
|
2024-06-15 10:42:50 +00:00
|
|
|
ret = innerApplication(argc, argv);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ret = innerApplication(argc, argv);
|
2022-07-01 20:25:42 +00:00
|
|
|
}
|
2024-06-15 10:42:50 +00:00
|
|
|
catch (const std::exception& e)
|
|
|
|
{
|
2018-08-03 09:19:12 +00:00
|
|
|
#if (defined(__APPLE__) || defined(__linux) || defined(__unix) || defined(__posix))
|
2024-06-15 10:42:50 +00:00
|
|
|
if (!isatty(fileno(stdin)))
|
2018-08-03 09:19:12 +00:00
|
|
|
#endif
|
2024-06-15 10:42:50 +00:00
|
|
|
SDL_ShowSimpleMessageBox(0, (std::string(appName) + ": Fatal error").c_str(), e.what(), nullptr);
|
2018-08-03 09:19:12 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
Log(Debug::Error) << "Fatal error: " << e.what();
|
2018-08-03 09:19:12 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
ret = 1;
|
|
|
|
}
|
2018-08-03 09:19:12 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
// Restore cout and cerr
|
|
|
|
std::cout.rdbuf(rawStdout->rdbuf());
|
|
|
|
std::cerr.rdbuf(rawStderr->rdbuf());
|
2024-06-15 11:30:16 +00:00
|
|
|
|
|
|
|
Log::sMinDebugLevel = All;
|
|
|
|
Log::sWriteLevel = false;
|
2018-08-03 09:19:12 +00:00
|
|
|
|
2024-06-15 10:42:50 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2018-08-03 09:19:12 +00:00
|
|
|
}
|