2018-08-03 13:19:12 +04:00
|
|
|
#include "debugging.hpp"
|
|
|
|
|
2020-12-04 11:17:49 +01:00
|
|
|
#include <chrono>
|
2020-12-08 22:23:11 +01:00
|
|
|
#include <memory>
|
2020-12-12 17:29:29 +01:00
|
|
|
#include <functional>
|
2020-12-04 11:17:49 +01:00
|
|
|
|
2018-08-14 11:17:05 +04:00
|
|
|
#include <components/crashcatcher/crashcatcher.hpp>
|
|
|
|
|
2018-12-12 00:12:13 +01:00
|
|
|
#ifdef _WIN32
|
2020-12-12 17:29:29 +01:00
|
|
|
# include <components/crashcatcher/windows_crashcatcher.hpp>
|
2018-12-12 00:12:13 +01:00
|
|
|
# undef WIN32_LEAN_AND_MEAN
|
|
|
|
# define WIN32_LEAN_AND_MEAN
|
|
|
|
# include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
2018-08-03 13:19:12 +04:00
|
|
|
namespace Debug
|
|
|
|
{
|
2018-12-11 23:58:46 +01: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 23:58:46 +01: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);
|
|
|
|
|
2018-12-11 23:58:46 +01: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);
|
|
|
|
}
|
|
|
|
if (!outRedirected)
|
|
|
|
{
|
|
|
|
_wfreopen(L"CON", L"w", stdout);
|
|
|
|
freopen("CON", "w", stdout);
|
|
|
|
}
|
|
|
|
if (!errRedirected)
|
|
|
|
{
|
|
|
|
_wfreopen(L"CON", L"w", stderr);
|
|
|
|
freopen("CON", "w", stderr);
|
|
|
|
}
|
2018-12-11 23:58:46 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-08-03 13:19:12 +04:00
|
|
|
std::streamsize DebugOutputBase::write(const char *str, std::streamsize size)
|
|
|
|
{
|
2020-12-04 11:17:49 +01:00
|
|
|
if (size <= 0)
|
|
|
|
return size;
|
|
|
|
std::string_view msg{str, size_t(size)};
|
|
|
|
|
2018-08-03 13:19:12 +04:00
|
|
|
// Skip debug level marker
|
|
|
|
Level level = getLevelMarker(str);
|
|
|
|
if (level != NoLevel)
|
2020-12-04 11:17:49 +01:00
|
|
|
msg = msg.substr(1);
|
|
|
|
|
|
|
|
char prefix[32];
|
|
|
|
int prefixSize;
|
2018-08-03 13:19:12 +04:00
|
|
|
{
|
2020-12-04 11:17:49 +01:00
|
|
|
prefix[0] = '[';
|
|
|
|
uint64_t ms = std::chrono::duration_cast<std::chrono::milliseconds>(
|
|
|
|
std::chrono::high_resolution_clock::now().time_since_epoch()).count();
|
|
|
|
std::time_t t = ms / 1000;
|
|
|
|
prefixSize = std::strftime(prefix + 1, sizeof(prefix) - 1, "%T", std::localtime(&t)) + 1;
|
|
|
|
char levelLetter = " EWIVD*"[int(level)];
|
|
|
|
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);
|
|
|
|
msg = msg.substr(lineSize);
|
2018-08-03 13:19:12 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
Level DebugOutputBase::getLevelMarker(const char *str)
|
|
|
|
{
|
|
|
|
if (unsigned(*str) <= unsigned(Marker))
|
|
|
|
{
|
|
|
|
return Level(*str);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NoLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugOutputBase::fillCurrentDebugLevel()
|
|
|
|
{
|
|
|
|
const char* env = getenv("OPENMW_DEBUG_LEVEL");
|
|
|
|
if (env)
|
|
|
|
{
|
|
|
|
std::string value(env);
|
|
|
|
if (value == "ERROR")
|
|
|
|
CurrentDebugLevel = Error;
|
|
|
|
else if (value == "WARNING")
|
|
|
|
CurrentDebugLevel = Warning;
|
|
|
|
else if (value == "INFO")
|
|
|
|
CurrentDebugLevel = Info;
|
|
|
|
else if (value == "VERBOSE")
|
|
|
|
CurrentDebugLevel = Verbose;
|
2018-11-01 14:08:33 +04:00
|
|
|
else if (value == "DEBUG")
|
|
|
|
CurrentDebugLevel = Debug;
|
2018-08-03 13:19:12 +04:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
CurrentDebugLevel = Verbose;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-08 22:23:11 +01:00
|
|
|
static std::unique_ptr<std::ostream> rawStdout = nullptr;
|
2022-01-13 00:20:16 +01:00
|
|
|
static std::unique_ptr<std::ostream> rawStderr = nullptr;
|
|
|
|
static boost::filesystem::ofstream logfile;
|
|
|
|
|
|
|
|
#if defined(_WIN32) && defined(_DEBUG)
|
|
|
|
static boost::iostreams::stream_buffer<Debug::DebugOutput> sb;
|
|
|
|
#else
|
|
|
|
static boost::iostreams::stream_buffer<Debug::Tee> coutsb;
|
|
|
|
static boost::iostreams::stream_buffer<Debug::Tee> cerrsb;
|
|
|
|
#endif
|
2020-12-08 22:23:11 +01:00
|
|
|
|
|
|
|
std::ostream& getRawStdout()
|
|
|
|
{
|
|
|
|
return rawStdout ? *rawStdout : std::cout;
|
|
|
|
}
|
|
|
|
|
2022-01-13 00:20:16 +01:00
|
|
|
// Redirect cout and cerr to the log file
|
|
|
|
void setupLogging(const std::string& logDir, const std::string& appName, std::ios_base::openmode mode)
|
2018-08-03 13:19:12 +04:00
|
|
|
{
|
2019-09-28 11:16:14 +02:00
|
|
|
#if defined(_WIN32) && defined(_DEBUG)
|
2022-01-13 00:20:16 +01:00
|
|
|
// Redirect cout and cerr to VS debug output when running in debug mode
|
|
|
|
sb.open(Debug::DebugOutput());
|
|
|
|
std::cout.rdbuf(&sb);
|
|
|
|
std::cerr.rdbuf(&sb);
|
2019-09-28 11:16:14 +02:00
|
|
|
#else
|
2022-01-13 00:20:16 +01:00
|
|
|
const std::string logName = Misc::StringUtils::lowerCase(appName) + ".log";
|
|
|
|
logfile.open(boost::filesystem::path(logDir) / logName, mode);
|
|
|
|
|
|
|
|
coutsb.open(Debug::Tee(logfile, *rawStdout));
|
|
|
|
cerrsb.open(Debug::Tee(logfile, *rawStderr));
|
|
|
|
|
|
|
|
std::cout.rdbuf(&coutsb);
|
|
|
|
std::cerr.rdbuf(&cerrsb);
|
2018-08-14 10:30:27 +04:00
|
|
|
#endif
|
2022-01-13 00:20:16 +01:00
|
|
|
}
|
2018-08-14 10:30:27 +04:00
|
|
|
|
2022-01-13 00:20:16 +01:00
|
|
|
int wrapApplication(int (*innerApplication)(int argc, char *argv[]), int argc, char *argv[],
|
|
|
|
const std::string& appName, bool autoSetupLogging)
|
|
|
|
{
|
|
|
|
#if defined _WIN32
|
|
|
|
(void)Debug::attachParentConsole();
|
|
|
|
#endif
|
|
|
|
rawStdout = std::make_unique<std::ostream>(std::cout.rdbuf());
|
|
|
|
rawStderr = std::make_unique<std::ostream>(std::cerr.rdbuf());
|
2018-08-14 10:30:27 +04:00
|
|
|
|
2018-08-03 13:19:12 +04:00
|
|
|
int ret = 0;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
Files::ConfigurationManager cfgMgr;
|
2018-08-14 11:17:05 +04:00
|
|
|
|
2022-01-13 00:20:16 +01:00
|
|
|
if (autoSetupLogging)
|
|
|
|
{
|
|
|
|
std::ios_base::openmode mode = std::ios::out;
|
2018-08-03 13:19:12 +04:00
|
|
|
|
2022-01-13 00:20:16 +01:00
|
|
|
// If we are collecting a stack trace, append to existing log file
|
|
|
|
if (argc == 2 && strcmp(argv[1], crash_switch) == 0)
|
|
|
|
mode |= std::ios::app;
|
2018-08-03 13:19:12 +04:00
|
|
|
|
2022-01-13 00:20:16 +01:00
|
|
|
setupLogging(cfgMgr.getLogPath().string(), appName, mode);
|
|
|
|
}
|
2018-08-14 11:17:05 +04:00
|
|
|
|
2020-12-12 17:29:29 +01:00
|
|
|
#if defined(_WIN32)
|
2020-12-13 14:13:07 +01:00
|
|
|
const std::string crashLogName = Misc::StringUtils::lowerCase(appName) + "-crash.dmp";
|
2020-12-12 17:29:29 +01:00
|
|
|
Crash::CrashCatcher crashy(argc, argv, (cfgMgr.getLogPath() / crashLogName).make_preferred().string());
|
|
|
|
#else
|
2020-12-13 14:13:07 +01:00
|
|
|
const std::string crashLogName = Misc::StringUtils::lowerCase(appName) + "-crash.log";
|
2018-08-14 11:17:05 +04:00
|
|
|
// install the crash handler as soon as possible. note that the log path
|
|
|
|
// does not depend on config being read.
|
|
|
|
crashCatcherInstall(argc, argv, (cfgMgr.getLogPath() / crashLogName).string());
|
2020-12-12 17:29:29 +01:00
|
|
|
#endif
|
2018-08-03 13:19:12 +04:00
|
|
|
ret = innerApplication(argc, argv);
|
|
|
|
}
|
2020-12-12 17:29:29 +01:00
|
|
|
catch (const std::exception& e)
|
2018-08-03 13:19:12 +04:00
|
|
|
{
|
|
|
|
#if (defined(__APPLE__) || defined(__linux) || defined(__unix) || defined(__posix))
|
|
|
|
if (!isatty(fileno(stdin)))
|
|
|
|
#endif
|
2018-10-09 10:21:12 +04:00
|
|
|
SDL_ShowSimpleMessageBox(0, (appName + ": Fatal error").c_str(), e.what(), nullptr);
|
2018-08-03 13:19:12 +04:00
|
|
|
|
2018-08-14 10:30:27 +04:00
|
|
|
Log(Debug::Error) << "Error: " << e.what();
|
2018-08-03 13:19:12 +04:00
|
|
|
|
|
|
|
ret = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Restore cout and cerr
|
2022-01-13 00:20:16 +01:00
|
|
|
std::cout.rdbuf(rawStdout->rdbuf());
|
|
|
|
std::cerr.rdbuf(rawStderr->rdbuf());
|
2020-12-04 11:17:49 +01:00
|
|
|
Debug::CurrentDebugLevel = Debug::NoLevel;
|
2018-08-03 13:19:12 +04:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|