mirror of
https://github.com/KjellKod/g3log.git
synced 2025-01-07 09:48:06 +01:00
What a mess to merge from Github/KjellKod/g3log to BitBucket/KjellKod/g3log
This commit is contained in:
parent
1f844f5ea1
commit
938fc55971
@ -7,6 +7,12 @@ compiler:
|
|||||||
- gcc
|
- gcc
|
||||||
#- clang
|
#- clang
|
||||||
|
|
||||||
|
# whitelist
|
||||||
|
branches:
|
||||||
|
only:
|
||||||
|
- master
|
||||||
|
|
||||||
|
|
||||||
before_install:
|
before_install:
|
||||||
# use http://lint.travis-ci.org/ to validate changes
|
# use http://lint.travis-ci.org/ to validate changes
|
||||||
# sudo add-apt-repository -y ppa:h-rayflood/llvm;
|
# sudo add-apt-repository -y ppa:h-rayflood/llvm;
|
||||||
|
@ -32,6 +32,7 @@ ELSEIF("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU")
|
|||||||
|
|
||||||
|
|
||||||
ELSEIF(MSVC)
|
ELSEIF(MSVC)
|
||||||
|
set(PLATFORM_LINK_LIBRIES dbghelp)
|
||||||
# VC11 bug: http://code.google.com/p/googletest/issues/detail?id=408
|
# VC11 bug: http://code.google.com/p/googletest/issues/detail?id=408
|
||||||
# add_definition(-D_VARIADIC_MAX=10)
|
# add_definition(-D_VARIADIC_MAX=10)
|
||||||
# https://github.com/anhstudios/swganh/pull/186/files
|
# https://github.com/anhstudios/swganh/pull/186/files
|
||||||
@ -59,7 +60,7 @@ ENDIF()
|
|||||||
IF (MSVC OR MINGW)
|
IF (MSVC OR MINGW)
|
||||||
list(REMOVE_ITEM SRC_FILES ${LOG_SRC}/crashhandler_unix.cpp)
|
list(REMOVE_ITEM SRC_FILES ${LOG_SRC}/crashhandler_unix.cpp)
|
||||||
ELSE()
|
ELSE()
|
||||||
list(REMOVE_ITEM SRC_FILES ${LOG_SRC}/crashhandler_win.cpp)
|
list(REMOVE_ITEM SRC_FILES ${LOG_SRC}/crashhandler_windows.cpp ${LOG_SRC}/stacktrace_windows.hpp ${LOG_SRC}/stacktrace_windows.cpp)
|
||||||
ENDIF (MSVC OR MINGW)
|
ENDIF (MSVC OR MINGW)
|
||||||
|
|
||||||
set(SRC_FILES ${SRC_FILES} ${SRC_PLATFORM_SPECIFIC})
|
set(SRC_FILES ${SRC_FILES} ${SRC_PLATFORM_SPECIFIC})
|
||||||
@ -69,8 +70,11 @@ ENDIF()
|
|||||||
#MESSAGE(" g3logger files: [${SRC_FILES}]")
|
#MESSAGE(" g3logger files: [${SRC_FILES}]")
|
||||||
add_library(g3logger ${SRC_FILES})
|
add_library(g3logger ${SRC_FILES})
|
||||||
set_target_properties(g3logger PROPERTIES LINKER_LANGUAGE CXX)
|
set_target_properties(g3logger PROPERTIES LINKER_LANGUAGE CXX)
|
||||||
|
target_link_libraries(g3logger ${PLATFORM_LINK_LIBRIES})
|
||||||
|
|
||||||
add_library(g3logger_shared SHARED ${SRC_FILES})
|
add_library(g3logger_shared SHARED ${SRC_FILES})
|
||||||
set_target_properties(g3logger_shared PROPERTIES LINKER_LANGUAGE CXX)
|
set_target_properties(g3logger_shared PROPERTIES LINKER_LANGUAGE CXX)
|
||||||
|
target_link_libraries(g3logger_shared ${PLATFORM_LINK_LIBRIES})
|
||||||
|
|
||||||
SET(G3LOG_SHARED_LIBRARY g3logger_shared)
|
SET(G3LOG_SHARED_LIBRARY g3logger_shared)
|
||||||
SET(G3LOG_LIBRARY g3logger)
|
SET(G3LOG_LIBRARY g3logger)
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
# mkdir build
|
# mkdir build
|
||||||
# cd build;
|
# cd build;
|
||||||
# 3. cmake -DCMAKE_BUILD_TYPE=Release -G "Visual Studio XXX" ..
|
# 3. cmake -DCMAKE_BUILD_TYPE=Release -G "Visual Studio XXX" ..
|
||||||
|
# (cmake -DCMAKE_BUILD_TYPE=Release -G "Visual Studio 12")
|
||||||
# (XXX is the Visual Studio version you are running)
|
# (XXX is the Visual Studio version you are running)
|
||||||
# 4. msbuild g3log.sln /p:Configuration=Release
|
# 4. msbuild g3log.sln /p:Configuration=Release
|
||||||
#
|
#
|
||||||
@ -75,17 +76,19 @@ endif()
|
|||||||
|
|
||||||
|
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
# G3LOG DYNAMIC LEVELS OPTIONS
|
# G3LOG OPTIONAL FEATURES
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
# ENABLE WITH: -USE_DYNAMIC_LOGGING_LEVELS=ON : run-type turn on/off levels
|
# Dynamic logging: ENABLE WITH: -DUSE_DYNAMIC_LOGGING_LEVELS=ON : run-type turn on/off levels
|
||||||
INCLUDE (${g3log_SOURCE_DIR}/Dynamic.cmake)
|
# DBUG instead of DEBUG logging level: ENABLE WITH: -DCHANGE_G3LOG_DEBUG_TO_DBUG=ON
|
||||||
|
INCLUDE (${g3log_SOURCE_DIR}/Options.cmake)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
# EXAMPLE OPTIONS: By defauls is ON. This will create 'g3log-FATAL-* examples'
|
# EXAMPLE OPTIONS: By defauls is ON. This will create 'g3log-FATAL-* examples'
|
||||||
# ============================================================================
|
# ============================================================================
|
||||||
# DISABLE WITH: -DUSE_SIMPLE_EXAMPLE=OFF
|
# DISABLE WITH: -DUSE_FATAL_EXAMPLE=OFF
|
||||||
INCLUDE (${g3log_SOURCE_DIR}/example/Example.cmake)
|
INCLUDE (${g3log_SOURCE_DIR}/example/Example.cmake)
|
||||||
|
|
||||||
|
|
||||||
|
32
Options.cmake
Normal file
32
Options.cmake
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# -DUSE_DYNAMIC_LOGGING_LEVELS=ON : run-type turn on/off levels
|
||||||
|
option (USE_DYNAMIC_LOGGING_LEVELS
|
||||||
|
"Turn ON/OFF log levels. An disabled level will not push logs of that level to the sink. By default dynamic logging is disabled" OFF)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
IF(USE_DYNAMIC_LOGGING_LEVELS)
|
||||||
|
add_definitions(-DG2_DYNAMIC_LOGGING)
|
||||||
|
MESSAGE("-DUSE_DYNAMIC_LOGGING_LEVELS=ON
|
||||||
|
\tDynamic logging levels can be turned on. Make sure to have \n\t\t[#define G2_DYNAMIC_LOGGING 1] in your source code")
|
||||||
|
MESSAGE("\t\tUse [g2::setLogLevel(LEVEL boolean)] to enable/disable logging on specified levels")
|
||||||
|
ELSE()
|
||||||
|
MESSAGE("-DUSE_DYNAMIC_LOGGING_LEVELS=OFF")
|
||||||
|
ENDIF(USE_DYNAMIC_LOGGING_LEVELS)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# -DCHANGE_G3LOG_DEBUG_TO_DBUG=ON : change the DEBUG logging level to be DBUG to avoid clash with other libraries that might have
|
||||||
|
# predefined DEBUG for their own purposes
|
||||||
|
option (CHANGE_G3LOG_DEBUG_TO_DBUG
|
||||||
|
"Use DBUG logging level instead of DEBUG. By default DEBUG is the debugging level" OFF)
|
||||||
|
|
||||||
|
IF(CHANGE_G3LOG_DEBUG_TO_DBUG)
|
||||||
|
add_definitions(-DCHANGE_G3LOG_DEBUG_TO_DBUG)
|
||||||
|
MESSAGE("-DCHANGE_G3LOG_DEBUG_TO_DBUG=ON
|
||||||
|
\tDBUG instead of DEBUG logging level can be used. Make sure to have \n\t\t[#define CHANGE_G3LOG_DEBUG_TO_DBUG 1] in your source code")
|
||||||
|
ELSE()
|
||||||
|
MESSAGE("-DCHANGE_G3LOG_DEBUG_TO_DBUG=OFF. Debuggin logging level is 'DEBUG'")
|
||||||
|
ENDIF(CHANGE_G3LOG_DEBUG_TO_DBUG)
|
||||||
|
|
||||||
|
|
@ -76,7 +76,7 @@ The code is given for free as public domain. This gives the option to change, us
|
|||||||
|
|
||||||
The *std::string* comes pre-formatted. The *g2::LogMessageMover* is a wrapped struct that contains the raw data for custom handling in your own sink.
|
The *std::string* comes pre-formatted. The *g2::LogMessageMover* is a wrapped struct that contains the raw data for custom handling in your own sink.
|
||||||
|
|
||||||
It is easy to either use pre-made sinks or make your own. When a sink is set to be used by the G3log it is added to the logger inside a ```std::unique_ptr```. The sink can be called though its public API through a *handler* which will asynchronously forward the call to the receiving sink.
|
A sink is *owned* by the G3log and is added to the logger inside a ```std::unique_ptr```. The sink can be called though its public API through a *handler* which will asynchronously forward the call to the receiving sink.
|
||||||
```
|
```
|
||||||
auto sinkHandle = logworker->addSink(std2::make_unique<CustomSink>(),
|
auto sinkHandle = logworker->addSink(std2::make_unique<CustomSink>(),
|
||||||
&CustomSink::ReceiveLogMessage);
|
&CustomSink::ReceiveLogMessage);
|
||||||
|
@ -10,21 +10,21 @@
|
|||||||
|
|
||||||
|
|
||||||
set(DIR_EXAMPLE ${g3log_SOURCE_DIR}/example)
|
set(DIR_EXAMPLE ${g3log_SOURCE_DIR}/example)
|
||||||
option (USE_SIMPLE_EXAMPLE "Simple (fatal-crash/contract) examples " ON)
|
option (USE_FATAL_EXAMPLE "Fatal (fatal-crashes/contract) examples " ON)
|
||||||
|
|
||||||
|
|
||||||
IF (USE_SIMPLE_EXAMPLE)
|
IF (USE_FATAL_EXAMPLE)
|
||||||
|
|
||||||
MESSAGE("-DUSE_SIMPLE_EXAMPLE=ON")
|
MESSAGE("-DUSE_FATAL_EXAMPLE=ON")
|
||||||
MESSAGE("\tg3log-FATAL-contract and g3log-FATAL-sigsegv shows fatal examples of when g3log comes in handy")
|
MESSAGE("\tg3log-FATAL- [contract][sigsegv][choice] are examples of when g3log comes in handy")
|
||||||
include_directories (${DIR_EXAMPLE})
|
include_directories (${DIR_EXAMPLE})
|
||||||
add_executable(g3log-FATAL-contract ${DIR_EXAMPLE}/main_contract.cpp)
|
add_executable(g3log-FATAL-contract ${DIR_EXAMPLE}/main_contract.cpp)
|
||||||
add_executable(g3log-FATAL-sigsegv ${DIR_EXAMPLE}/main_sigsegv.cpp)
|
add_executable(g3log-FATAL-sigsegv ${DIR_EXAMPLE}/main_sigsegv.cpp)
|
||||||
add_executable(g3log-FATAL-choice ${DIR_EXAMPLE}/main_fatal_choice.cpp)
|
add_executable(g3log-FATAL-choice ${DIR_EXAMPLE}/main_fatal_choice.cpp)
|
||||||
|
|
||||||
target_link_libraries(g3log-FATAL-contract ${G3LOG_LIBRARY} ${PLATFORM_LINK_LIBRIES})
|
target_link_libraries(g3log-FATAL-contract ${G3LOG_LIBRARY})
|
||||||
target_link_libraries(g3log-FATAL-sigsegv ${G3LOG_LIBRARY} ${PLATFORM_LINK_LIBRIES})
|
target_link_libraries(g3log-FATAL-sigsegv ${G3LOG_LIBRARY})
|
||||||
target_link_libraries(g3log-FATAL-choice ${G3LOG_LIBRARY} ${PLATFORM_LINK_LIBRIES})
|
target_link_libraries(g3log-FATAL-choice ${G3LOG_LIBRARY})
|
||||||
ELSE()
|
ELSE()
|
||||||
MESSAGE("-DUSE_SIMPLE_EXAMPLE=OFF")
|
MESSAGE("-DUSE_SIMPLE_EXAMPLE=OFF")
|
||||||
ENDIF (USE_SIMPLE_EXAMPLE)
|
ENDIF (USE_FATAL_EXAMPLE)
|
@ -32,41 +32,36 @@ void ToLower(std::string &str)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void sleep_for(size_t seconds) {
|
|
||||||
std::this_thread::sleep_for(std::chrono::seconds(seconds));
|
|
||||||
}
|
|
||||||
|
|
||||||
void RaiseSIGABRT() {
|
void RaiseSIGABRT() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
|
||||||
raise(SIGABRT);
|
raise(SIGABRT);
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
|
|
||||||
void RaiseSIGFPE() {
|
void RaiseSIGFPE() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
LOGF_IF(INFO, (false != true), "Exiting %s SIGFPE", "by");
|
||||||
raise(SIGFPE);
|
raise(SIGFPE);
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
|
|
||||||
void RaiseSIGSEGV() {
|
void RaiseSIGSEGV() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
LOG(DEBUG) << "Exit by SIGSEGV";
|
||||||
raise(SIGSEGV);
|
raise(SIGSEGV);
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
|
|
||||||
void RaiseSIGILL() {
|
void RaiseSIGILL() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
LOGF(DEBUG, "Exit by %s", "SIGILL");
|
||||||
raise(SIGILL);
|
raise(SIGILL);
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
|
|
||||||
void RAiseSIGTERM() {
|
void RAiseSIGTERM() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
LOGF_IF(INFO, (false != true), "Exiting %s SIGFPE", "by");
|
||||||
raise(SIGTERM);
|
raise(SIGTERM);
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
@ -75,23 +70,21 @@ int gShouldBeZero = 1;
|
|||||||
void DivisionByZero() {
|
void DivisionByZero() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
std::cout << "Executing DivisionByZero: gShouldBeZero: " << gShouldBeZero << std::endl;
|
std::cout << "Executing DivisionByZero: gShouldBeZero: " << gShouldBeZero << std::endl;
|
||||||
sleep_for(2);
|
LOG(INFO) << "Division by zero is a big no-no";
|
||||||
int value = 3;
|
int value = 3;
|
||||||
auto test = value / gShouldBeZero;
|
auto test = value / gShouldBeZero;
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now..., test value: " << test;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IllegalPrintf() {
|
void IllegalPrintf() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
LOG(DEBUG) << "Impending doom due to illeteracy";
|
||||||
printf("ILLEGAL PRINTF_SYNTAX %d EXAMPLE. %s %s", "hello", 1);
|
|
||||||
LOGF(INFO, "2nd attempt at ILLEGAL PRINTF_SYNTAX %d EXAMPLE. %s %s", "hello", 1);
|
LOGF(INFO, "2nd attempt at ILLEGAL PRINTF_SYNTAX %d EXAMPLE. %s %s", "hello", 1);
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
|
|
||||||
void OutOfBoundsArrayIndexing() {
|
void OutOfBoundsArrayIndexing() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
|
||||||
std::vector<int> v;
|
std::vector<int> v;
|
||||||
v[0] = 5;
|
v[0] = 5;
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
@ -100,8 +93,8 @@ void OutOfBoundsArrayIndexing() {
|
|||||||
|
|
||||||
void AccessViolation() {
|
void AccessViolation() {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
sleep_for(2);
|
|
||||||
char* ptr = 0;
|
char* ptr = 0;
|
||||||
|
LOG(INFO) << "Death by access violation is imminent";
|
||||||
*ptr = 0;
|
*ptr = 0;
|
||||||
LOG(WARNING) << "Expected to have died by now...";
|
LOG(WARNING) << "Expected to have died by now...";
|
||||||
}
|
}
|
||||||
@ -111,6 +104,23 @@ void NoExitFunction() {
|
|||||||
CHECK(false) << "This function should never be called";
|
CHECK(false) << "This function should never be called";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void RaiseSIGABRTAndAccessViolation() {
|
||||||
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
|
|
||||||
|
auto f1 = std::async(std::launch::async, &RaiseSIGABRT);
|
||||||
|
auto f2 = std::async(std::launch::async, &AccessViolation);
|
||||||
|
f1.wait();
|
||||||
|
f2.wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void CallActualExitFunction(std::function<void()> fatal_function) {
|
||||||
|
fatal_function();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CallExitFunction(std::function<void()> fatal_function) {
|
||||||
|
CallActualExitFunction(fatal_function);
|
||||||
|
}
|
||||||
|
|
||||||
void ExecuteDeathFunction(const bool runInNewThread, int fatalChoice) {
|
void ExecuteDeathFunction(const bool runInNewThread, int fatalChoice) {
|
||||||
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
std::cout << "Calling :" << __FUNCTION__ << " Line: " << __LINE__ << std::endl << std::flush;
|
||||||
@ -121,17 +131,18 @@ void ExecuteDeathFunction(const bool runInNewThread, int fatalChoice) {
|
|||||||
case 3: exitFunction = &RaiseSIGSEGV; break;
|
case 3: exitFunction = &RaiseSIGSEGV; break;
|
||||||
case 4: exitFunction = &RaiseSIGILL; break;
|
case 4: exitFunction = &RaiseSIGILL; break;
|
||||||
case 5: exitFunction = &RAiseSIGTERM; break;
|
case 5: exitFunction = &RAiseSIGTERM; break;
|
||||||
case 6: exitFunction = &DivisionByZero; gShouldBeZero = 0; break;
|
case 6: exitFunction = &DivisionByZero; gShouldBeZero = 0; DivisionByZero(); break;
|
||||||
case 7: exitFunction = &IllegalPrintf; break;
|
case 7: exitFunction = &IllegalPrintf; break;
|
||||||
case 8: exitFunction = &OutOfBoundsArrayIndexing; break;
|
case 8: exitFunction = &OutOfBoundsArrayIndexing; break;
|
||||||
case 9: exitFunction = &AccessViolation; break;
|
case 9: exitFunction = &AccessViolation; break;
|
||||||
|
case 10: exitFunction = &RaiseSIGABRTAndAccessViolation; break;
|
||||||
default: break;
|
default: break;
|
||||||
}
|
}
|
||||||
if (runInNewThread) {
|
if (runInNewThread) {
|
||||||
auto dieInNearFuture = std::async(std::launch::async, exitFunction);
|
auto dieInNearFuture = std::async(std::launch::async, CallExitFunction, exitFunction);
|
||||||
dieInNearFuture.wait();
|
dieInNearFuture.wait();
|
||||||
} else {
|
} else {
|
||||||
exitFunction();
|
CallExitFunction(exitFunction);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string unexpected = "Expected to exit by FATAL event. That did not happen (printf choice in Windows?).";
|
std::string unexpected = "Expected to exit by FATAL event. That did not happen (printf choice in Windows?).";
|
||||||
@ -178,13 +189,14 @@ int ChoiceOfFatalExit() {
|
|||||||
std::cout << "[6] Division By Zero" << std::endl;
|
std::cout << "[6] Division By Zero" << std::endl;
|
||||||
std::cout << "[7] Illegal printf" << std::endl;
|
std::cout << "[7] Illegal printf" << std::endl;
|
||||||
std::cout << "[8] Out of bounds array indexing " << std::endl;
|
std::cout << "[8] Out of bounds array indexing " << std::endl;
|
||||||
std::cout << "[9] Access violation \n\n" << std::endl;
|
std::cout << "[9] Access violation" << std::endl;
|
||||||
|
std::cout << "[10] Rasing SIGABRT + Access Violation in two separate threads" << std::endl;
|
||||||
std::cout << std::flush;
|
std::cout << std::flush;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
std::getline(std::cin, option);
|
std::getline(std::cin, option);
|
||||||
choice = std::stoi(option);
|
choice = std::stoi(option);
|
||||||
if (choice <= 0 || choice > 9) {
|
if (choice <= 0 || choice > 10) {
|
||||||
std::cout << "Invalid choice: [" << option << "\n\n";
|
std::cout << "Invalid choice: [" << option << "\n\n";
|
||||||
} else {
|
} else {
|
||||||
return choice;
|
return choice;
|
||||||
@ -195,11 +207,14 @@ int ChoiceOfFatalExit() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ForwardChoiceForFatalExit(bool runInNewThread, int fatalChoice) {
|
||||||
|
ExecuteDeathFunction(runInNewThread, fatalChoice);
|
||||||
|
}
|
||||||
|
|
||||||
void ChooseFatalExit() {
|
void ChooseFatalExit() {
|
||||||
const bool runInNewThread = AskForAsyncDeath();
|
const bool runInNewThread = AskForAsyncDeath();
|
||||||
const int choiceOfFatalExit = ChoiceOfFatalExit();
|
const int exitChoice = ChoiceOfFatalExit();
|
||||||
ExecuteDeathFunction(runInNewThread, choiceOfFatalExit);
|
ForwardChoiceForFatalExit(runInNewThread, exitChoice);
|
||||||
}
|
}
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
@ -214,7 +229,7 @@ int main(int argc, char **argv)
|
|||||||
<< "The logfile is generated at: [" << log_file_name.get() << "]\n\n" << std::endl;
|
<< "The logfile is generated at: [" << log_file_name.get() << "]\n\n" << std::endl;
|
||||||
|
|
||||||
|
|
||||||
LOGF(INFO, "Fatal exit example starts now, it's as easy as %d", 123);
|
LOGF(DEBUG, "Fatal exit example starts now, it's as easy as %d", 123);
|
||||||
LOG(INFO) << "Feel free to read the source code also in g3log/example/main_fatal_choice.cpp";
|
LOG(INFO) << "Feel free to read the source code also in g3log/example/main_fatal_choice.cpp";
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
@ -9,22 +9,44 @@
|
|||||||
* ============================================================================*/
|
* ============================================================================*/
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <csignal>
|
#include <csignal>
|
||||||
|
#include "g2loglevels.hpp"
|
||||||
|
|
||||||
// kjell. Separera på crashhandler.hpp och crashhanlder_internal.hpp
|
// kjell. Separera på crashhandler.hpp och crashhanlder_internal.hpp
|
||||||
// implementationsfilen kan vara den samma
|
// implementationsfilen kan vara den samma
|
||||||
|
|
||||||
namespace g2 {
|
namespace g2 {
|
||||||
|
|
||||||
|
|
||||||
|
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
|
typedef unsigned long SignalType;
|
||||||
|
/// SIGFPE, SIGILL, and SIGSEGV handling must be installed per thread
|
||||||
|
/// on Windows. This is automatically done if you do at least one LOG(...) call
|
||||||
|
/// you can also use this function call, per thread so make sure these three
|
||||||
|
/// fatal signals are covered in your thread (even if you don't do a LOG(...) call
|
||||||
|
void installSignalHandlerForThread();
|
||||||
|
#else
|
||||||
|
typedef int SignalType;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
namespace internal {
|
namespace internal {
|
||||||
/** \return signal_name. Ref: signum.hpp and \ref installSignalHandler */
|
/** return whether or any fatal handling is still ongoing
|
||||||
std::string signalName(int signal_number);
|
* this is used by g2log::fatalCallToLogger
|
||||||
|
* only in the case of Windows exceptions (not fatal signals)
|
||||||
|
* are we interested in changing this from false to true to
|
||||||
|
* help any other exceptions handler work with 'EXCEPTION_CONTINUE_SEARCH'*/
|
||||||
|
bool blockForFatalHandling();
|
||||||
|
|
||||||
|
/** \return signal_name Ref: signum.hpp and \ref installSignalHandler
|
||||||
|
* or for Windows exception name */
|
||||||
|
std::string exitReasonName(const LEVELS& level, g2::SignalType signal_number);
|
||||||
|
|
||||||
/** return calling thread's stackdump*/
|
/** return calling thread's stackdump*/
|
||||||
std::string stackdump();
|
std::string stackdump(const char* dump = nullptr);
|
||||||
|
|
||||||
/** Re-"throw" a fatal signal, previously caught. This will exit the application
|
/** Re-"throw" a fatal signal, previously caught. This will exit the application
|
||||||
* This is an internal only function. Do not use it elsewhere. It is triggered
|
* This is an internal only function. Do not use it elsewhere. It is triggered
|
||||||
* from g2log, g2LogWorker after flushing messages to file */
|
* from g2log, g2LogWorker after flushing messages to file */
|
||||||
void exitWithDefaultSignalHandler(int signal_number);
|
void exitWithDefaultSignalHandler(const LEVELS& level, g2::SignalType signal_number);
|
||||||
} // end g2::internal
|
} // end g2::internal
|
||||||
|
|
||||||
|
|
||||||
@ -37,5 +59,5 @@ void exitWithDefaultSignalHandler(int signal_number);
|
|||||||
SIGILL ILlegal instruction (ANSI)
|
SIGILL ILlegal instruction (ANSI)
|
||||||
SIGSEGV Segmentation violation i.e. illegal memory reference
|
SIGSEGV Segmentation violation i.e. illegal memory reference
|
||||||
SIGTERM TERMINATION (ANSI) */
|
SIGTERM TERMINATION (ANSI) */
|
||||||
void installSignalHandler();
|
void installCrashHandler();
|
||||||
}
|
}
|
||||||
|
@ -37,20 +37,16 @@ namespace {
|
|||||||
// Dump of stack,. then exit through g2log background worker
|
// Dump of stack,. then exit through g2log background worker
|
||||||
// ALL thanks to this thread at StackOverflow. Pretty much borrowed from:
|
// ALL thanks to this thread at StackOverflow. Pretty much borrowed from:
|
||||||
// Ref: http://stackoverflow.com/questions/77005/how-to-generate-a-stacktrace-when-my-gcc-c-app-crashes
|
// Ref: http://stackoverflow.com/questions/77005/how-to-generate-a-stacktrace-when-my-gcc-c-app-crashes
|
||||||
|
void signalHandler(int signal_number, siginfo_t *info, void *unused_context) {
|
||||||
void crashHandler(int signal_number, siginfo_t *info, void *unused_context) {
|
|
||||||
using namespace g2::internal;
|
using namespace g2::internal;
|
||||||
|
{
|
||||||
std::ostringstream oss;
|
const auto dump = stackdump();
|
||||||
oss << "Received fatal signal: " << g2::internal::signalName(signal_number);
|
|
||||||
oss << "(" << signal_number << ")\tPID: " << getpid() << std::endl;
|
|
||||||
//oss << stackdump();
|
|
||||||
|
|
||||||
{ // Local scope, trigger send
|
|
||||||
std::ostringstream fatal_stream;
|
std::ostringstream fatal_stream;
|
||||||
fatal_stream << oss.str() << std::endl;
|
const auto fatal_reason = exitReasonName(g2::internal::FATAL_SIGNAL, signal_number);
|
||||||
fatal_stream << "\n***** SIGNAL " << signalName(signal_number) << "(" << signal_number << ")" << std::endl;
|
fatal_stream << "Received fatal signal: " << fatal_reason;
|
||||||
LogCapture trigger(FATAL_SIGNAL, signal_number);
|
fatal_stream << "(" << signal_number << ")\tPID: " << getpid() << std::endl;
|
||||||
|
fatal_stream << "\n***** SIGNAL " << fatal_reason << "(" << signal_number << ")" << std::endl;
|
||||||
|
LogCapture trigger(FATAL_SIGNAL, static_cast<g2::SignalType>(signal_number), dump.c_str());
|
||||||
trigger.stream() << fatal_stream.str();
|
trigger.stream() << fatal_stream.str();
|
||||||
} // message sent to g2LogWorker
|
} // message sent to g2LogWorker
|
||||||
// wait to die
|
// wait to die
|
||||||
@ -78,7 +74,17 @@ namespace g2 {
|
|||||||
// http://stackoverflow.com/questions/6878546/why-doesnt-parent-process-return-to-the-exact-location-after-handling-signal_number
|
// http://stackoverflow.com/questions/6878546/why-doesnt-parent-process-return-to-the-exact-location-after-handling-signal_number
|
||||||
namespace internal {
|
namespace internal {
|
||||||
|
|
||||||
std::string stackdump() {
|
bool blockForFatalHandling() {
|
||||||
|
return true; // For windows we will after fatal processing change it to false
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Generate stackdump. Or in case a stackdump was pre-generated and non-empty just use that one
|
||||||
|
/// i.e. the latter case is only for Windows and test purposes
|
||||||
|
std::string stackdump(const char *rawdump) {
|
||||||
|
if (nullptr != rawdump && !std::string(rawdump).empty()) {
|
||||||
|
return {rawdump};
|
||||||
|
}
|
||||||
|
|
||||||
const size_t max_dump_size = 50;
|
const size_t max_dump_size = 50;
|
||||||
void *dump[max_dump_size];
|
void *dump[max_dump_size];
|
||||||
size_t size = backtrace(dump, max_dump_size);
|
size_t size = backtrace(dump, max_dump_size);
|
||||||
@ -128,7 +134,12 @@ namespace g2 {
|
|||||||
return oss.str();
|
return oss.str();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string signalName(int signal_number) {
|
|
||||||
|
|
||||||
|
/// string representation of signal ID
|
||||||
|
std::string exitReasonName(const LEVELS& level, g2::SignalType fatal_id) {
|
||||||
|
|
||||||
|
int signal_number = static_cast<int>(fatal_id);
|
||||||
switch (signal_number) {
|
switch (signal_number) {
|
||||||
case SIGABRT: return "SIGABRT";
|
case SIGABRT: return "SIGABRT";
|
||||||
break;
|
break;
|
||||||
@ -142,17 +153,36 @@ namespace g2 {
|
|||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
std::ostringstream oss;
|
std::ostringstream oss;
|
||||||
oss << "UNKNOWN SIGNAL(" << signal_number << ")";
|
oss << "UNKNOWN SIGNAL(" << signal_number << ") for " << level.text;
|
||||||
return oss.str();
|
return oss.str();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// KJELL : TODO. The Fatal Message can contain a callback function that depending on OS and test scenario does
|
||||||
|
// different things.
|
||||||
|
// exitWithDefaultSignalHandler is called from g2logworke::bgFatal AFTER all the logging sinks have been cleared
|
||||||
|
// I.e. saving a function that has the value already encapsulated within.
|
||||||
|
// FatalMessagePtr msgPtr
|
||||||
|
// Linux/OSX --> msgPtr.get()->ContinueWithFatalExit(); --> exitWithDefaultSignalHandler(int signal_number);
|
||||||
|
// Windows ..... (if signal) --> exitWithDefaultSignalHandler(int signal_number);
|
||||||
|
// (if exception) ....
|
||||||
|
// the calling thread that is in a never-ending loop should break out of that loop
|
||||||
|
// i.e. an atomic flag should be set
|
||||||
|
// the next step should then be to re-throw the same exception
|
||||||
|
// i.e. just call the next exception handler
|
||||||
|
// we should make sure that 1) g2log exception handler is called BEFORE widows
|
||||||
|
// it should continue and then be caught in Visual Studios exception handler
|
||||||
|
//
|
||||||
|
//
|
||||||
|
|
||||||
// Triggered by g2log->g2LogWorker after receiving a FATAL trigger
|
// Triggered by g2log->g2LogWorker after receiving a FATAL trigger
|
||||||
// which is LOG(FATAL), CHECK(false) or a fatal signal our signalhandler caught.
|
// which is LOG(FATAL), CHECK(false) or a fatal signal our signalhandler caught.
|
||||||
// --- If LOG(FATAL) or CHECK(false) the signal_number will be SIGABRT
|
// --- If LOG(FATAL) or CHECK(false) the signal_number will be SIGABRT
|
||||||
|
void exitWithDefaultSignalHandler(const LEVELS& level, g2::SignalType fatal_signal_id) {
|
||||||
void exitWithDefaultSignalHandler(int signal_number) {
|
const int signal_number = static_cast<int>(fatal_signal_id);
|
||||||
std::cerr << "Exiting - FATAL SIGNAL: " << signal_number << " " << std::flush;
|
std::cerr << "Exiting due to " << level.text << ", " << signal_number << " " << std::flush;
|
||||||
struct sigaction action;
|
struct sigaction action;
|
||||||
memset(&action, 0, sizeof (action)); //
|
memset(&action, 0, sizeof (action)); //
|
||||||
sigemptyset(&action.sa_mask);
|
sigemptyset(&action.sa_mask);
|
||||||
@ -163,11 +193,15 @@ namespace g2 {
|
|||||||
}
|
}
|
||||||
} // end g2::internal
|
} // end g2::internal
|
||||||
|
|
||||||
|
|
||||||
|
//
|
||||||
|
// Installs FATAL signal handler that is enough to handle most fatal events
|
||||||
|
// on *NIX systems
|
||||||
void installSignalHandler() {
|
void installSignalHandler() {
|
||||||
struct sigaction action;
|
struct sigaction action;
|
||||||
memset(&action, 0, sizeof (action));
|
memset(&action, 0, sizeof (action));
|
||||||
sigemptyset(&action.sa_mask);
|
sigemptyset(&action.sa_mask);
|
||||||
action.sa_sigaction = &crashHandler; // callback to crashHandler for fatal signals
|
action.sa_sigaction = &signalHandler; // callback to crashHandler for fatal signals
|
||||||
// sigaction to use sa_sigaction file. ref: http://www.linuxprogrammingblog.com/code-examples/sigaction
|
// sigaction to use sa_sigaction file. ref: http://www.linuxprogrammingblog.com/code-examples/sigaction
|
||||||
action.sa_flags = SA_SIGINFO;
|
action.sa_flags = SA_SIGINFO;
|
||||||
|
|
||||||
@ -183,4 +217,13 @@ namespace g2 {
|
|||||||
if (sigaction(SIGTERM, &action, NULL) < 0)
|
if (sigaction(SIGTERM, &action, NULL) < 0)
|
||||||
perror("sigaction - SIGTERM");
|
perror("sigaction - SIGTERM");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void installCrashHandler() {
|
||||||
|
installSignalHandler();
|
||||||
|
} // namespace g2::internal
|
||||||
|
|
||||||
|
|
||||||
} // end namespace g2
|
} // end namespace g2
|
||||||
|
|
||||||
|
@ -1,99 +0,0 @@
|
|||||||
/** ==========================================================================
|
|
||||||
* 2011 by KjellKod.cc. This is PUBLIC DOMAIN to use at your own risk and comes
|
|
||||||
* with no warranties. This code is yours to share, use and modify with no
|
|
||||||
* strings attached and no restrictions or obligations.
|
|
||||||
*
|
|
||||||
* For more information see g3log/LICENSE or refer refer to http://unlicense.org
|
|
||||||
* ============================================================================*/
|
|
||||||
|
|
||||||
#include "crashhandler.hpp"
|
|
||||||
#include "g2logmessage.hpp"
|
|
||||||
#include "g2logmessagecapture.hpp"
|
|
||||||
|
|
||||||
#include <csignal>
|
|
||||||
#include <cstring>
|
|
||||||
#include <cstdlib>
|
|
||||||
#include <sstream>
|
|
||||||
#if !(defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
|
||||||
#error "crashhandler_win.cpp used but not on a windows system"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <process.h> // getpid
|
|
||||||
#define getpid _getpid
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
void crashHandler(int signal_number) {
|
|
||||||
using namespace g2::internal;
|
|
||||||
std::ostringstream fatal_stream;
|
|
||||||
fatal_stream << "\n\n***** FATAL TRIGGER RECEIVED ******* " << std::endl;
|
|
||||||
fatal_stream << "\n***** RETHROWING SIGNAL " << signalName(signal_number) << "(" << signal_number << ")" << std::endl;
|
|
||||||
|
|
||||||
LogCapture trigger(FATAL_SIGNAL, signal_number);
|
|
||||||
trigger.stream() << fatal_stream.str();
|
|
||||||
} // scope exit - message sent to LogWorker, wait to die...
|
|
||||||
} // end anonymous namespace
|
|
||||||
|
|
||||||
|
|
||||||
namespace g2 {
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
std::string stackdump() {
|
|
||||||
std::string temp;
|
|
||||||
return temp;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string signalName(int signal_number) {
|
|
||||||
switch (signal_number) {
|
|
||||||
case SIGABRT: return "SIGABRT";
|
|
||||||
break;
|
|
||||||
case SIGFPE: return "SIGFPE";
|
|
||||||
break;
|
|
||||||
case SIGSEGV: return "SIGSEGV";
|
|
||||||
break;
|
|
||||||
case SIGILL: return "SIGILL";
|
|
||||||
break;
|
|
||||||
case SIGTERM: return "SIGTERM";
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
std::ostringstream oss;
|
|
||||||
oss << "UNKNOWN SIGNAL(" << signal_number << ")";
|
|
||||||
return oss.str();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Triggered by g2log::LogWorker after receiving a FATAL trigger
|
|
||||||
// which is LOG(FATAL), CHECK(false) or a fatal signal our signalhandler caught.
|
|
||||||
// --- If LOG(FATAL) or CHECK(false) the signal_number will be SIGABRT
|
|
||||||
|
|
||||||
void exitWithDefaultSignalHandler(int signal_number) {
|
|
||||||
// Restore our signalhandling to default
|
|
||||||
if (SIG_ERR == signal(SIGABRT, SIG_DFL))
|
|
||||||
perror("signal - SIGABRT");
|
|
||||||
if (SIG_ERR == signal(SIGFPE, SIG_DFL))
|
|
||||||
perror("signal - SIGABRT");
|
|
||||||
if (SIG_ERR == signal(SIGSEGV, SIG_DFL))
|
|
||||||
perror("signal - SIGABRT");
|
|
||||||
if (SIG_ERR == signal(SIGILL, SIG_DFL))
|
|
||||||
perror("signal - SIGABRT");
|
|
||||||
if (SIG_ERR == signal(SIGTERM, SIG_DFL))
|
|
||||||
perror("signal - SIGABRT");
|
|
||||||
|
|
||||||
raise(signal_number);
|
|
||||||
}
|
|
||||||
} // end g2::internal
|
|
||||||
|
|
||||||
void installSignalHandler() {
|
|
||||||
if (SIG_ERR == signal(SIGABRT, crashHandler))
|
|
||||||
perror("signal - SIGABRT");
|
|
||||||
if (SIG_ERR == signal(SIGFPE, crashHandler))
|
|
||||||
perror("signal - SIGFPE");
|
|
||||||
if (SIG_ERR == signal(SIGSEGV, crashHandler))
|
|
||||||
perror("signal - SIGSEGV");
|
|
||||||
if (SIG_ERR == signal(SIGILL, crashHandler))
|
|
||||||
perror("signal - SIGILL");
|
|
||||||
if (SIG_ERR == signal(SIGTERM, crashHandler))
|
|
||||||
perror("signal - SIGTERM");
|
|
||||||
}
|
|
||||||
} // end namespace g2
|
|
215
src/crashhandler_windows.cpp
Normal file
215
src/crashhandler_windows.cpp
Normal file
@ -0,0 +1,215 @@
|
|||||||
|
/** ==========================================================================
|
||||||
|
* 2011 by KjellKod.cc. This is PUBLIC DOMAIN to use at your own risk and comes
|
||||||
|
* with no warranties. This code is yours to share, use and modify with no
|
||||||
|
* strings attached and no restrictions or obligations.
|
||||||
|
*
|
||||||
|
* For more information see g3log/LICENSE or refer refer to http://unlicense.org
|
||||||
|
* ============================================================================*/
|
||||||
|
|
||||||
|
#if !(defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
|
#error "crashhandler_windows.cpp used but not on a windows system"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
#include <csignal>
|
||||||
|
#include <cstring>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <sstream>
|
||||||
|
#include <atomic>
|
||||||
|
#include <process.h> // getpid
|
||||||
|
#define getpid _getpid
|
||||||
|
|
||||||
|
#include "crashhandler.hpp"
|
||||||
|
#include "stacktrace_windows.hpp"
|
||||||
|
#include "g2logmessage.hpp"
|
||||||
|
#include "g2logmessagecapture.hpp"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
std::atomic<bool> gBlockForFatal {true};
|
||||||
|
void* g_vector_exception_handler = nullptr;
|
||||||
|
LPTOP_LEVEL_EXCEPTION_FILTER g_previous_unexpected_exception_handler = nullptr;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Restore back to default fatal event handling
|
||||||
|
void ReverseToOriginalFatalHandling() {
|
||||||
|
SetUnhandledExceptionFilter (g_previous_unexpected_exception_handler);
|
||||||
|
RemoveVectoredExceptionHandler (g_vector_exception_handler);
|
||||||
|
|
||||||
|
if (SIG_ERR == signal(SIGABRT, SIG_DFL))
|
||||||
|
perror("signal - SIGABRT");
|
||||||
|
|
||||||
|
if (SIG_ERR == signal(SIGFPE, SIG_DFL))
|
||||||
|
perror("signal - SIGABRT");
|
||||||
|
|
||||||
|
if (SIG_ERR == signal(SIGSEGV, SIG_DFL))
|
||||||
|
perror("signal - SIGABRT");
|
||||||
|
|
||||||
|
if (SIG_ERR == signal(SIGILL, SIG_DFL))
|
||||||
|
perror("signal - SIGABRT");
|
||||||
|
|
||||||
|
if (SIG_ERR == signal(SIGTERM, SIG_DFL))
|
||||||
|
perror("signal - SIGABRT");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// called for fatal signals SIGABRT, SIGFPE, SIGSEGV, SIGILL, SIGTERM
|
||||||
|
void signalHandler(int signal_number) {
|
||||||
|
using namespace g2::internal;
|
||||||
|
std::string dump = stacktrace::stackdump();
|
||||||
|
|
||||||
|
std::ostringstream fatal_stream;
|
||||||
|
fatal_stream << "\n***** Received fatal signal " << g2::internal::exitReasonName(g2::internal::FATAL_SIGNAL, signal_number);
|
||||||
|
fatal_stream << "(" << signal_number << ")\tPID: " << getpid() << std::endl;
|
||||||
|
|
||||||
|
|
||||||
|
LogCapture trigger(FATAL_SIGNAL, static_cast<g2::SignalType>(signal_number), dump.c_str());
|
||||||
|
trigger.stream() << fatal_stream.str();
|
||||||
|
} // scope exit - message sent to LogWorker, wait to die...
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Unhandled exception catching
|
||||||
|
LONG WINAPI exceptionHandling(EXCEPTION_POINTERS* info) {
|
||||||
|
std::string dump = stacktrace::stackdump();
|
||||||
|
|
||||||
|
std::ostringstream fatal_stream;
|
||||||
|
const g2::SignalType exception_code = info->ExceptionRecord->ExceptionCode;
|
||||||
|
fatal_stream << "\n***** Received fatal exception " << g2::internal::exitReasonName(g2::internal::FATAL_EXCEPTION, exception_code);
|
||||||
|
fatal_stream << "\tPID: " << getpid() << std::endl;
|
||||||
|
|
||||||
|
const auto fatal_id = static_cast<g2::SignalType>(exception_code);
|
||||||
|
LogCapture trigger(g2::internal::FATAL_EXCEPTION, fatal_id, dump.c_str());
|
||||||
|
trigger.stream() << fatal_stream.str();
|
||||||
|
return EXCEPTION_CONTINUE_SEARCH; //EXCEPTION_EXECUTE_HANDLER;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Unhandled exception catching
|
||||||
|
LONG WINAPI unexpectedExceptionHandling(EXCEPTION_POINTERS* info) {
|
||||||
|
ReverseToOriginalFatalHandling();
|
||||||
|
return exceptionHandling(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// Setup through (Windows API) AddVectoredExceptionHandler
|
||||||
|
/// Ref: http://blogs.msdn.com/b/zhanli/archive/2010/06/25/c-tips-addvectoredexceptionhandler-addvectoredcontinuehandler-and-setunhandledexceptionfilter.aspx
|
||||||
|
LONG WINAPI vectorExceptionHandling(PEXCEPTION_POINTERS p) {
|
||||||
|
ReverseToOriginalFatalHandling();
|
||||||
|
return exceptionHandling(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
} // end anonymous namespace
|
||||||
|
|
||||||
|
|
||||||
|
namespace g2 {
|
||||||
|
namespace internal {
|
||||||
|
|
||||||
|
|
||||||
|
// For windows exceptions this might ONCE be set to false, in case of a
|
||||||
|
// windows exceptions and not a signal
|
||||||
|
bool blockForFatalHandling() {
|
||||||
|
return gBlockForFatal;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// Generate stackdump. Or in case a stackdump was pre-generated and
|
||||||
|
/// non-empty just use that one. i.e. the latter case is only for
|
||||||
|
/// Windows and test purposes
|
||||||
|
std::string stackdump(const char* dump) {
|
||||||
|
if (nullptr != dump && !std::string(dump).empty()) {
|
||||||
|
return {dump};
|
||||||
|
}
|
||||||
|
|
||||||
|
return stacktrace::stackdump();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/// string representation of signal ID or Windows exception id
|
||||||
|
std::string exitReasonName(const LEVELS& level, g2::SignalType fatal_id) {
|
||||||
|
if (level == g2::internal::FATAL_EXCEPTION) {
|
||||||
|
return stacktrace::exceptionIdToText(fatal_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (fatal_id) {
|
||||||
|
case SIGABRT: return "SIGABRT"; break;
|
||||||
|
case SIGFPE: return "SIGFPE"; break;
|
||||||
|
case SIGSEGV: return "SIGSEGV"; break;
|
||||||
|
case SIGILL: return "SIGILL"; break;
|
||||||
|
case SIGTERM: return "SIGTERM"; break;
|
||||||
|
default:
|
||||||
|
std::ostringstream oss;
|
||||||
|
oss << "UNKNOWN SIGNAL(" << fatal_id << ")";
|
||||||
|
return oss.str();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Triggered by g2log::LogWorker after receiving a FATAL trigger
|
||||||
|
// which is LOG(FATAL), CHECK(false) or a fatal signal our signalhandler caught.
|
||||||
|
// --- If LOG(FATAL) or CHECK(false) the signal_number will be SIGABRT
|
||||||
|
void exitWithDefaultSignalHandler(const LEVELS& level, g2::SignalType fatal_signal_id) {
|
||||||
|
|
||||||
|
ReverseToOriginalFatalHandling();
|
||||||
|
// For windows exceptions we want to continue the possibility of
|
||||||
|
// exception handling now when the log and stacktrace are flushed
|
||||||
|
// to sinks. We therefore avoid to kill the preocess here. Instead
|
||||||
|
// it will be the exceptionHandling functions above that
|
||||||
|
// will let exception handling continue with: EXCEPTION_CONTINUE_SEARCH
|
||||||
|
if (g2::internal::FATAL_EXCEPTION == level) {
|
||||||
|
gBlockForFatal = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// for a sigal however, we exit through that fatal signal
|
||||||
|
const int signal_number = static_cast<int>(fatal_signal_id);
|
||||||
|
raise(signal_number);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void installSignalHandler() {
|
||||||
|
g2::installSignalHandlerForThread();
|
||||||
|
if (SIG_ERR == signal(SIGABRT, signalHandler))
|
||||||
|
perror("signal - SIGABRT");
|
||||||
|
|
||||||
|
if (SIG_ERR == signal(SIGTERM, signalHandler))
|
||||||
|
perror("signal - SIGTERM");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
} // end g2::internal
|
||||||
|
|
||||||
|
|
||||||
|
/// SIGFPE, SIGILL, and SIGSEGV handling must be installed per thread
|
||||||
|
/// on Windows. This is automatically done if you do at least one LOG(...) call
|
||||||
|
/// you can also use this function call, per thread so make sure these three
|
||||||
|
/// fatal signals are covered in your thread (even if you don't do a LOG(...) call
|
||||||
|
void installSignalHandlerForThread() {
|
||||||
|
if (SIG_ERR == signal(SIGFPE, signalHandler))
|
||||||
|
perror("signal - SIGFPE");
|
||||||
|
if (SIG_ERR == signal(SIGSEGV, signalHandler))
|
||||||
|
perror("signal - SIGSEGV");
|
||||||
|
if (SIG_ERR == signal(SIGILL, signalHandler))
|
||||||
|
perror("signal - SIGILL");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void installCrashHandler() {
|
||||||
|
internal::installSignalHandler();
|
||||||
|
//const size_t kFirstExceptionHandler = 1; // Kept here for documentational purposes. last exception seems more what we want
|
||||||
|
const size_t kLastExceptionHandler = 0;
|
||||||
|
g_vector_exception_handler = AddVectoredExceptionHandler(kLastExceptionHandler, vectorExceptionHandling);
|
||||||
|
g_previous_unexpected_exception_handler = SetUnhandledExceptionFilter(unexpectedExceptionHandling);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // end namespace g2
|
@ -53,7 +53,8 @@ namespace g2 {
|
|||||||
|
|
||||||
void initializeLogging(LogWorker* bgworker) {
|
void initializeLogging(LogWorker* bgworker) {
|
||||||
std::call_once(g_initialize_flag, []() {
|
std::call_once(g_initialize_flag, []() {
|
||||||
installSignalHandler(); });
|
installCrashHandler();
|
||||||
|
});
|
||||||
std::lock_guard<std::mutex> lock(g_logging_init_mutex);
|
std::lock_guard<std::mutex> lock(g_logging_init_mutex);
|
||||||
CHECK(!internal::isLoggingInitialized());
|
CHECK(!internal::isLoggingInitialized());
|
||||||
CHECK(bgworker != nullptr);
|
CHECK(bgworker != nullptr);
|
||||||
@ -107,9 +108,8 @@ namespace g2 {
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
// explicits copy of all input. This is makes it possibly to use g3log across dynamically loaded libraries
|
/** explicits copy of all input. This is makes it possibly to use g3log across dynamically loaded libraries
|
||||||
// i.e. (dlopen + dlsym)
|
* i.e. (dlopen + dlsym) */
|
||||||
|
|
||||||
void saveMessage(const char* entry, const char* file, int line, const char* function, const LEVELS& level,
|
void saveMessage(const char* entry, const char* file, int line, const char* function, const LEVELS& level,
|
||||||
const char* boolean_expression, int fatal_signal, const char* stack_trace) {
|
const char* boolean_expression, int fatal_signal, const char* stack_trace) {
|
||||||
LEVELS msgLevel {level};
|
LEVELS msgLevel {level};
|
||||||
@ -162,24 +162,24 @@ namespace g2 {
|
|||||||
* will sleep forever (i.e. until the background thread catches up, saves the fatal
|
* will sleep forever (i.e. until the background thread catches up, saves the fatal
|
||||||
* message and kills the software with the fatal signal.
|
* message and kills the software with the fatal signal.
|
||||||
*/
|
*/
|
||||||
void fatalCallToLogger(FatalMessagePtr message) {
|
void pushFatalMessageToLogger(FatalMessagePtr message) {
|
||||||
if (!isLoggingInitialized()) {
|
if (!isLoggingInitialized()) {
|
||||||
std::ostringstream error;
|
std::ostringstream error;
|
||||||
error << "FATAL CALL but logger is NOT initialized\n"
|
error << "FATAL CALL but logger is NOT initialized\n"
|
||||||
<< "SIGNAL: " << message.get()->signal()
|
<< "CAUSE: " << message.get()->reason()
|
||||||
<< "\nMessage: \n" << message.get()->toString() << std::flush;
|
<< "\nMessage: \n" << message.get()->toString() << std::flush;
|
||||||
std::cerr << error.str() << std::flush;
|
std::cerr << error.str() << std::flush;
|
||||||
internal::exitWithDefaultSignalHandler(message.get()->_signal_id);
|
internal::exitWithDefaultSignalHandler(message.get()->_level, message.get()->_signal_id);
|
||||||
}
|
}
|
||||||
g_logger_instance->fatal(message);
|
g_logger_instance->fatal(message);
|
||||||
while (true) {
|
while (blockForFatalHandling()) {
|
||||||
std::this_thread::sleep_for(std::chrono::seconds(1));
|
std::this_thread::sleep_for(std::chrono::seconds(1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// By default this function pointer goes to \ref fatalCallToLogger;
|
// By default this function pointer goes to \ref pushFatalMessageToLogger;
|
||||||
std::function<void(FatalMessagePtr) > g_fatal_to_g2logworker_function_ptr = fatalCallToLogger;
|
std::function<void(FatalMessagePtr) > g_fatal_to_g2logworker_function_ptr = pushFatalMessageToLogger;
|
||||||
|
|
||||||
/** The default, initial, handling to send a 'fatal' event to g2logworker
|
/** The default, initial, handling to send a 'fatal' event to g2logworker
|
||||||
* the caller will stay here, eternally, until the software is aborted
|
* the caller will stay here, eternally, until the software is aborted
|
||||||
@ -194,7 +194,7 @@ namespace g2 {
|
|||||||
* This function switches the function pointer so that only
|
* This function switches the function pointer so that only
|
||||||
* 'unitTest' mock-fatal calls are made.
|
* 'unitTest' mock-fatal calls are made.
|
||||||
* */
|
* */
|
||||||
void changeFatalInitHandlerForUnitTesting(std::function<void(FatalMessagePtr) > fatal_call) {
|
void setFatalExitHandler(std::function<void(FatalMessagePtr) > fatal_call) {
|
||||||
g_fatal_to_g2logworker_function_ptr = fatal_call;
|
g_fatal_to_g2logworker_function_ptr = fatal_call;
|
||||||
}
|
}
|
||||||
} // internal
|
} // internal
|
||||||
|
@ -64,10 +64,26 @@ namespace g2 {
|
|||||||
void saveMessage(const char* message, const char* file, int line, const char* function, const LEVELS& level,
|
void saveMessage(const char* message, const char* file, int line, const char* function, const LEVELS& level,
|
||||||
const char* boolean_expression, int fatal_signal, const char* stack_trace);
|
const char* boolean_expression, int fatal_signal, const char* stack_trace);
|
||||||
|
|
||||||
|
// forwards the message to all sinks
|
||||||
void pushMessageToLogger(LogMessagePtr log_entry);
|
void pushMessageToLogger(LogMessagePtr log_entry);
|
||||||
|
|
||||||
|
|
||||||
|
// forwards a FATAL message to all sinks,. after which the g2logworker
|
||||||
|
// will trigger crashhandler / g2::internal::exitWithDefaultSignalHandler
|
||||||
|
//
|
||||||
|
// By default the "fatalCall" will forward a Fatalessageptr to this function
|
||||||
|
// this behaviour can be changed if you set a different fatal handler through
|
||||||
|
// "setFatalExitHandler"
|
||||||
|
void pushFatalMessageToLogger(FatalMessagePtr message);
|
||||||
|
|
||||||
|
|
||||||
// Save the created FatalMessage to any existing sinks and exit with
|
// Save the created FatalMessage to any existing sinks and exit with
|
||||||
// the originating fatal signal,. or SIGABRT if it originated from a broken contract
|
// the originating fatal signal,. or SIGABRT if it originated from a broken contract
|
||||||
|
// By default forwards to: pushFatalMessageToLogger, see "setFatalExitHandler" to override
|
||||||
|
//
|
||||||
|
// If you override it then you probably want to call "pushFatalMessageToLogger" after your
|
||||||
|
// custom fatal handler is done. This will make sure that the fatal message the pushed
|
||||||
|
// to sinks as well as shutting down the process
|
||||||
void fatalCall(FatalMessagePtr message);
|
void fatalCall(FatalMessagePtr message);
|
||||||
|
|
||||||
|
|
||||||
@ -77,8 +93,6 @@ namespace g2 {
|
|||||||
// Shutdown logging, but ONLY if the active logger corresponds to the one currently initialized
|
// Shutdown logging, but ONLY if the active logger corresponds to the one currently initialized
|
||||||
bool shutDownLoggingForActiveOnly(LogWorker* active);
|
bool shutDownLoggingForActiveOnly(LogWorker* active);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/** By default the g2log will call g2LogWorker::fatal(...) which will
|
/** By default the g2log will call g2LogWorker::fatal(...) which will
|
||||||
* abort() the system after flushing the logs to file. This makes unit
|
* abort() the system after flushing the logs to file. This makes unit
|
||||||
* test of FATAL level cumbersome. A work around is to change the
|
* test of FATAL level cumbersome. A work around is to change the
|
||||||
@ -87,7 +101,10 @@ namespace g2 {
|
|||||||
* The bool return values in the fatal_call is whether or not the fatal_call should
|
* The bool return values in the fatal_call is whether or not the fatal_call should
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
void changeFatalInitHandlerForUnitTesting(std::function<void(FatalMessagePtr) > fatal_call);
|
void setFatalExitHandler(std::function<void(FatalMessagePtr)> fatal_call);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
} // internal
|
} // internal
|
||||||
} // g2
|
} // g2
|
||||||
|
|
||||||
@ -175,3 +192,4 @@ And here is possible output
|
|||||||
if (false == (boolean_expression)) INTERNAL_CONTRACT_MESSAGE(#boolean_expression).capturef(printf_like_message, ##__VA_ARGS__)
|
if (false == (boolean_expression)) INTERNAL_CONTRACT_MESSAGE(#boolean_expression).capturef(printf_like_message, ##__VA_ARGS__)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -15,6 +15,22 @@
|
|||||||
#include "g2log.hpp"
|
#include "g2log.hpp"
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
void checkLevel(const int level) {
|
||||||
|
#if (defined(CHANGE_G3LOG_DEBUG_TO_DBUG))
|
||||||
|
CHECK((level >= DBUG.value) && (level <= FATAL.value));
|
||||||
|
#else
|
||||||
|
CHECK((level >= DEBUG.value) && (level <= FATAL.value));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
} // anonymous namespace
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
namespace g2 {
|
namespace g2 {
|
||||||
namespace internal {
|
namespace internal {
|
||||||
bool wasFatal(const LEVELS& level) {
|
bool wasFatal(const LEVELS& level) {
|
||||||
@ -29,11 +45,10 @@ namespace g2 {
|
|||||||
} // internal
|
} // internal
|
||||||
|
|
||||||
#ifdef G2_DYNAMIC_LOGGING
|
#ifdef G2_DYNAMIC_LOGGING
|
||||||
|
|
||||||
void setLogLevel(LEVELS log_level, bool enabled) {
|
void setLogLevel(LEVELS log_level, bool enabled) {
|
||||||
assert(internal::g_level_size == 4 && "Mismatch between number of logging levels and their use");
|
assert(internal::g_level_size == 4 && "Mismatch between number of logging levels and their use");
|
||||||
int level = log_level.value;
|
int level = log_level.value;
|
||||||
CHECK((level >= DEBUG.value) && (level <= FATAL.value));
|
CHECK((level >= g2::kDebugVaulue) && (level <= FATAL.value));
|
||||||
internal::g_log_level_status[level].store(enabled, std::memory_order_release);
|
internal::g_log_level_status[level].store(enabled, std::memory_order_release);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -41,7 +56,7 @@ namespace g2 {
|
|||||||
bool logLevel(LEVELS log_level) {
|
bool logLevel(LEVELS log_level) {
|
||||||
#ifdef G2_DYNAMIC_LOGGING
|
#ifdef G2_DYNAMIC_LOGGING
|
||||||
int level = log_level.value;
|
int level = log_level.value;
|
||||||
CHECK((level >= DEBUG.value) && (level <= FATAL.value));
|
CHECK((level >= g2::kDebugVaulue) && (level <= FATAL.value));
|
||||||
bool status = (internal::g_log_level_status[level].load(std::memory_order_acquire));
|
bool status = (internal::g_log_level_status[level].load(std::memory_order_acquire));
|
||||||
return status;
|
return status;
|
||||||
#endif
|
#endif
|
||||||
|
@ -13,8 +13,23 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
|
||||||
|
// Users of Juce or other libraries might have a define DEBUG which clashes with
|
||||||
|
// the DEBUG logging level for G3log. In that case they can instead use the define
|
||||||
|
// "CHANGE_G3LOG_DEBUG_TO_DBUG" and G3log's logging level DEBUG is changed to be DBUG
|
||||||
|
#if (defined(CHANGE_G3LOG_DEBUG_TO_DBUG))
|
||||||
|
#if (defined(DBUG))
|
||||||
|
#error "DEBUG is already defined elsewhere which clashes with G3Log's log level DEBUG"
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
#if (defined(DEBUG))
|
||||||
|
#error "DEBUG is already defined elsewhere which clashes with G3Log's log level DEBUG"
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
|
|
||||||
// Levels for logging, made so that it would be easy to change, remove, add levels -- KjellKod
|
// Levels for logging, made so that it would be easy to change, remove, add levels -- KjellKod
|
||||||
struct LEVELS {
|
struct LEVELS {
|
||||||
// force internal copy of the const char*. This is a simple safeguard for when g3log is used in a
|
// force internal copy of the const char*. This is a simple safeguard for when g3log is used in a
|
||||||
@ -24,11 +39,31 @@ struct LEVELS {
|
|||||||
: value(other.value), text(other.text.c_str()) {}
|
: value(other.value), text(other.text.c_str()) {}
|
||||||
|
|
||||||
LEVELS(int id, const char* idtext) : value(id), text(idtext) {}
|
LEVELS(int id, const char* idtext) : value(id), text(idtext) {}
|
||||||
|
|
||||||
|
friend bool operator==(const LEVELS& lhs, const LEVELS& rhs) {
|
||||||
|
return (lhs.value == rhs.value && lhs.text == rhs.text);
|
||||||
|
}
|
||||||
|
|
||||||
const int value;
|
const int value;
|
||||||
const std::string text;
|
const std::string text;
|
||||||
};
|
};
|
||||||
|
|
||||||
const LEVELS DEBUG{0, {"DEBUG"}}, INFO{DEBUG.value + 1, {"INFO"}},
|
|
||||||
|
|
||||||
|
namespace g2 {
|
||||||
|
static const int kDebugVaulue = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if (defined(CHANGE_G3LOG_DEBUG_TO_DBUG))
|
||||||
|
const LEVELS DBUG {
|
||||||
|
g2::kDebugVaulue, {"DEBUG"}
|
||||||
|
},
|
||||||
|
#else
|
||||||
|
const LEVELS DEBUG {
|
||||||
|
g2::kDebugVaulue, {"DEBUG"}
|
||||||
|
},
|
||||||
|
#endif
|
||||||
|
INFO {g2::kDebugVaulue + 1, {"INFO"}},
|
||||||
WARNING {INFO.value + 1, {"WARNING"}},
|
WARNING {INFO.value + 1, {"WARNING"}},
|
||||||
// Insert here *any* extra logging levels that is needed
|
// Insert here *any* extra logging levels that is needed
|
||||||
// 1) Remember to update the FATAL initialization below
|
// 1) Remember to update the FATAL initialization below
|
||||||
@ -38,7 +73,10 @@ FATAL{WARNING.value + 1, {"FATAL"}};
|
|||||||
|
|
||||||
namespace g2 {
|
namespace g2 {
|
||||||
namespace internal {
|
namespace internal {
|
||||||
const LEVELS CONTRACT{100, {"CONTRACT"}}, FATAL_SIGNAL{101, {"FATAL_SIGNAL"}};
|
const LEVELS CONTRACT {
|
||||||
|
100, {"CONTRACT"}
|
||||||
|
}, FATAL_SIGNAL {101, {"FATAL_SIGNAL"}},
|
||||||
|
FATAL_EXCEPTION {102, {"FATAL_EXCEPTION"}};
|
||||||
bool wasFatal(const LEVELS& level);
|
bool wasFatal(const LEVELS& level);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,7 +22,9 @@ namespace {
|
|||||||
std::chrono::steady_clock::time_point g_start_time;
|
std::chrono::steady_clock::time_point g_start_time;
|
||||||
|
|
||||||
int64_t microsecondsCounter() {
|
int64_t microsecondsCounter() {
|
||||||
std::call_once(g_start_time_flag, []() { g_start_time = std::chrono::steady_clock::now(); });
|
std::call_once(g_start_time_flag, []() {
|
||||||
|
g_start_time = std::chrono::steady_clock::now();
|
||||||
|
});
|
||||||
auto now = std::chrono::steady_clock::now();
|
auto now = std::chrono::steady_clock::now();
|
||||||
return std::chrono::duration_cast<std::chrono::microseconds>(now - g_start_time).count();
|
return std::chrono::duration_cast<std::chrono::microseconds>(now - g_start_time).count();
|
||||||
}
|
}
|
||||||
@ -38,46 +40,100 @@ namespace {
|
|||||||
|
|
||||||
namespace g2 {
|
namespace g2 {
|
||||||
|
|
||||||
std::string LogMessage::toString() const {
|
|
||||||
|
// helper for setting the normal log details in an entry
|
||||||
|
std::string LogDetailsToString(const LogMessage& msg) {
|
||||||
std::string out;
|
std::string out;
|
||||||
out.append("\n" + timestamp() + "." + microseconds() + "\t"
|
out.append("\n" + msg.timestamp() + "." + msg.microseconds() + "\t"
|
||||||
+ level() + " [" + file() + " L: " + line() + "]\t");
|
+ msg.level() + " [" + msg.file() + " L: " + msg.line() + "]\t");
|
||||||
|
|
||||||
// Non-fatal Log Message
|
|
||||||
if (false == wasFatal()) {
|
|
||||||
out.append('"' + message() + '"');
|
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (internal::FATAL_SIGNAL.value == _level.value) {
|
|
||||||
out.clear(); // clear any previous text and formatting
|
// helper for normal
|
||||||
out.append("\n" + timestamp() + "." + microseconds()
|
std::string normalToString(const LogMessage& msg) {
|
||||||
|
auto out = LogDetailsToString(msg);
|
||||||
|
out.append('"' + msg.message() + '"');
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
// helper for fatal signal
|
||||||
|
std::string fatalSignalToString(const LogMessage &msg) {
|
||||||
|
std::string out; // clear any previous text and formatting
|
||||||
|
out.append("\n" + msg.timestamp() + "." + msg.microseconds()
|
||||||
+ "\n\n***** FATAL SIGNAL RECEIVED ******* \n"
|
+ "\n\n***** FATAL SIGNAL RECEIVED ******* \n"
|
||||||
+ '"' + message() + '"');
|
+ '"' + msg.message() + '"');
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not crash scenario but LOG or CONTRACT
|
|
||||||
auto level_value = _level.value;
|
// helper for fatal exception (windows only)
|
||||||
if (FATAL.value == level_value) {
|
std::string fatalExceptionToString(const LogMessage &msg) {
|
||||||
|
std::string out; // clear any previous text and formatting
|
||||||
|
out.append("\n" + msg.timestamp() + "." + msg.microseconds()
|
||||||
|
+ "\n\n***** FATAL EXCEPTION RECEIVED ******* \n"
|
||||||
|
+ '"' + msg.message() + '"');
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// helper for fatal LOG
|
||||||
|
std::string fatalLogToString(const LogMessage& msg) {
|
||||||
|
auto out = LogDetailsToString(msg);
|
||||||
static const std::string fatalExitReason = {"EXIT trigger caused by LOG(FATAL) entry: "};
|
static const std::string fatalExitReason = {"EXIT trigger caused by LOG(FATAL) entry: "};
|
||||||
out.append("\n\t*******\t " + fatalExitReason + "\n\t" + '"' + message() + '"');
|
out.append("\n\t*******\t " + fatalExitReason + "\n\t" + '"' + msg.message() + '"');
|
||||||
} else if (internal::CONTRACT.value == level_value) {
|
|
||||||
static const std::string contractExitReason = {"EXIT trigger caused by broken Contract:"};
|
|
||||||
out.append("\n\t*******\t " + contractExitReason + " CHECK(" + _expression + ")\n\t"
|
|
||||||
+ '"' + message() + '"');
|
|
||||||
} else {
|
|
||||||
static const std::string errorUnknown = {"UNKNOWN Log Message Type"};
|
|
||||||
out.append("\n\t*******" + errorUnknown + "\t\n" + '"' + message() + '"');
|
|
||||||
}
|
|
||||||
|
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// helper for fatal CHECK
|
||||||
|
std::string fatalCheckToString(const LogMessage& msg) {
|
||||||
|
auto out = LogDetailsToString(msg);
|
||||||
|
static const std::string contractExitReason = {"EXIT trigger caused by broken Contract:"};
|
||||||
|
out.append("\n\t*******\t " + contractExitReason + " CHECK(" + msg.expression() + ")\n\t"
|
||||||
|
+ '"' +msg. message() + '"');
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Format the log message according to it's type
|
||||||
|
std::string LogMessage::toString() const {
|
||||||
|
if (false == wasFatal()) {
|
||||||
|
return normalToString(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto level_value = _level.value;
|
||||||
|
if (internal::FATAL_SIGNAL.value == _level.value) {
|
||||||
|
return fatalSignalToString(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (internal::FATAL_EXCEPTION.value == _level.value) {
|
||||||
|
return fatalExceptionToString(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (FATAL.value == _level.value) {
|
||||||
|
return fatalLogToString(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (internal::CONTRACT.value == level_value) {
|
||||||
|
return fatalCheckToString(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
// What? Did we hit a custom made level?
|
||||||
|
auto out = LogDetailsToString(*this);
|
||||||
|
static const std::string errorUnknown = {"UNKNOWN or Custom made Log Message Type"};
|
||||||
|
out.append("\n\t*******" + errorUnknown + "\t\n" + '"' + message() + '"');
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
std::string LogMessage::timestamp(const std::string &time_look) const {
|
std::string LogMessage::timestamp(const std::string &time_look) const {
|
||||||
return localtime_formatted(_timestamp, time_look);
|
return localtime_formatted(_timestamp, time_look);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
LogMessage::LogMessage(const std::string &file, const int line,
|
LogMessage::LogMessage(const std::string &file, const int line,
|
||||||
const std::string &function, const LEVELS &level)
|
const std::string &function, const LEVELS &level)
|
||||||
: _timestamp(g2::systemtime_now())
|
: _timestamp(g2::systemtime_now())
|
||||||
@ -128,7 +184,7 @@ namespace g2 {
|
|||||||
return oss.str();
|
return oss.str();
|
||||||
}
|
}
|
||||||
|
|
||||||
FatalMessage::FatalMessage(const LogMessage& details, int signal_id)
|
FatalMessage::FatalMessage(const LogMessage &details, g2::SignalType signal_id)
|
||||||
: LogMessage(details), _signal_id(signal_id) { }
|
: LogMessage(details), _signal_id(signal_id) { }
|
||||||
|
|
||||||
|
|
||||||
@ -141,8 +197,8 @@ namespace g2 {
|
|||||||
return LogMessage(*this);
|
return LogMessage(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string FatalMessage::signal() const{
|
std::string FatalMessage::reason() const {
|
||||||
return internal::signalName(_signal_id);
|
return internal::exitReasonName(_level, _signal_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -22,8 +22,8 @@
|
|||||||
#include "g2loglevels.hpp"
|
#include "g2loglevels.hpp"
|
||||||
#include "g2time.hpp"
|
#include "g2time.hpp"
|
||||||
#include "g2moveoncopy.hpp"
|
#include "g2moveoncopy.hpp"
|
||||||
|
#include "crashhandler.hpp"
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
namespace g2 {
|
namespace g2 {
|
||||||
|
|
||||||
/** LogMessage contains all the data collected from the LOG(...) call.
|
/** LogMessage contains all the data collected from the LOG(...) call.
|
||||||
@ -86,14 +86,14 @@ namespace g2 {
|
|||||||
* A thread that causes a FatalMessage will sleep forever until the
|
* A thread that causes a FatalMessage will sleep forever until the
|
||||||
* application has exited (after message flush) */
|
* application has exited (after message flush) */
|
||||||
struct FatalMessage : public LogMessage {
|
struct FatalMessage : public LogMessage {
|
||||||
FatalMessage(const LogMessage& details, int signal_id);
|
FatalMessage(const LogMessage& details, g2::SignalType signal_id);
|
||||||
FatalMessage(const FatalMessage&);
|
FatalMessage(const FatalMessage&);
|
||||||
virtual ~FatalMessage(){};
|
virtual ~FatalMessage(){};
|
||||||
|
|
||||||
LogMessage copyToLogMessage() const;
|
LogMessage copyToLogMessage() const;
|
||||||
std::string signal() const;
|
std::string reason() const;
|
||||||
|
|
||||||
const int _signal_id;
|
const SignalType _signal_id;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -7,8 +7,82 @@
|
|||||||
* ============================================================================*/
|
* ============================================================================*/
|
||||||
|
|
||||||
#include "g2logmessagecapture.hpp"
|
#include "g2logmessagecapture.hpp"
|
||||||
|
#include "crashhandler.hpp"
|
||||||
|
|
||||||
|
|
||||||
|
// For Windows we need force a thread_local install per thread of three
|
||||||
|
// signals that must have a signal handler instealled per thread-basis
|
||||||
|
// It is really a royal pain. Seriously Microsoft? Seriously?
|
||||||
|
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
|
#define SIGNAL_HANDLER_VERIFY() g2::installSignalHandlerForThread()
|
||||||
|
#else
|
||||||
|
// Does nothing --- enforces that semicolon must be written
|
||||||
|
#define SIGNAL_HANDLER_VERIFY() do {} while(0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/** logCapture is a simple struct for capturing log/fatal entries. At destruction the
|
||||||
|
* captured message is forwarded to background worker.
|
||||||
|
* As a safety precaution: No memory allocated here will be moved into the background
|
||||||
|
* worker in case of dynamic loaded library reasons instead the arguments are copied
|
||||||
|
* inside of g2log.cpp::saveMessage*/
|
||||||
LogCapture::~LogCapture() {
|
LogCapture::~LogCapture() {
|
||||||
using namespace g2::internal;
|
using namespace g2::internal;
|
||||||
|
SIGNAL_HANDLER_VERIFY();
|
||||||
saveMessage(_stream.str().c_str(), _file, _line, _function, _level, _expression, _fatal_signal, _stack_trace.c_str());
|
saveMessage(_stream.str().c_str(), _file, _line, _function, _level, _expression, _fatal_signal, _stack_trace.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// Called from crash handler when a fatal signal has occurred (SIGSEGV etc)
|
||||||
|
LogCapture::LogCapture(const LEVELS &level, g2::SignalType fatal_signal, const char* dump)
|
||||||
|
: LogCapture("", 0, "", level, "", fatal_signal, dump) {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @file, line, function are given in g2log.hpp from macros
|
||||||
|
* @level INFO/DEBUG/WARNING/FATAL
|
||||||
|
* @expression for CHECK calls
|
||||||
|
* @fatal_signal for failed CHECK:SIGABRT or fatal signal caught in the signal handler
|
||||||
|
*/
|
||||||
|
LogCapture::LogCapture(const char* file, const int line, const char* function, const LEVELS &level,
|
||||||
|
const char* expression, g2::SignalType fatal_signal, const char* dump)
|
||||||
|
: _file(file), _line(line), _function(function), _level(level), _expression(expression), _fatal_signal(fatal_signal) {
|
||||||
|
|
||||||
|
if (g2::internal::wasFatal(level)) {
|
||||||
|
_stack_trace = {"\n*******\tSTACKDUMP *******\n"};
|
||||||
|
_stack_trace.append(g2::internal::stackdump(dump));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* capturef, used for "printf" like API in CHECKF, LOGF, LOGF_IF
|
||||||
|
* See also for the attribute formatting ref: http://www.codemaestro.com/reviews/18
|
||||||
|
*/
|
||||||
|
void LogCapture::capturef(const char* printf_like_message, ...) {
|
||||||
|
static const int kMaxMessageSize = 2048;
|
||||||
|
static const std::string kTruncatedWarningText = "[...truncated...]";
|
||||||
|
char finished_message[kMaxMessageSize];
|
||||||
|
va_list arglist;
|
||||||
|
va_start(arglist, printf_like_message);
|
||||||
|
|
||||||
|
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__) && !defined(__GNUC__))
|
||||||
|
const int nbrcharacters = vsnprintf_s(finished_message, _countof(finished_message), _TRUNCATE, printf_like_message, arglist);
|
||||||
|
#else
|
||||||
|
const int nbrcharacters = vsnprintf(finished_message, sizeof (finished_message), printf_like_message, arglist);
|
||||||
|
#endif
|
||||||
|
va_end(arglist);
|
||||||
|
|
||||||
|
if (nbrcharacters <= 0) {
|
||||||
|
stream() << "\n\tERROR LOG MSG NOTIFICATION: Failure to parse successfully the message";
|
||||||
|
stream() << '"' << printf_like_message << '"' << std::endl;
|
||||||
|
} else if (nbrcharacters > kMaxMessageSize) {
|
||||||
|
stream() << finished_message << kTruncatedWarningText;
|
||||||
|
} else {
|
||||||
|
stream() << finished_message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -10,40 +10,36 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <cstdarg>
|
#include <cstdarg>
|
||||||
|
#include <csignal>
|
||||||
#include "g2loglevels.hpp"
|
#include "g2loglevels.hpp"
|
||||||
#include "crashhandler.hpp"
|
|
||||||
#include "g2log.hpp"
|
#include "g2log.hpp"
|
||||||
|
#include "crashhandler.hpp"
|
||||||
struct LogCapture {
|
|
||||||
/// Called from crash handler when a fatal signal has occurred (SIGSEGV etc)
|
/**
|
||||||
LogCapture(const LEVELS& level, int fatal_signal)
|
* Simple struct for capturing log/fatal entries. At destruction the captured message is
|
||||||
: LogCapture("", 0, "", level, "", fatal_signal) {
|
* forwarded to background worker.
|
||||||
}
|
* As a safety precaution: No memory allocated here will be moved into the background
|
||||||
|
* worker in case of dynamic loaded library reasons
|
||||||
|
*/
|
||||||
|
struct LogCapture {
|
||||||
|
/// Called from crash handler when a fatal signal has occurred (SIGSEGV etc)
|
||||||
|
LogCapture(const LEVELS& level, g2::SignalType fatal_signal, const char* dump = nullptr);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simple struct for capturing log/fatal entries. At destruction the captured message is forwarded to background worker.
|
|
||||||
* -- As a safety precaution: No memory allocated here will be moved into the background worker in case of dynamic loaded library reasons
|
|
||||||
* -- instead the arguments are copied inside of g2log.cpp::saveMessage
|
|
||||||
* @file, line, function are given in g2log.hpp from macros
|
* @file, line, function are given in g2log.hpp from macros
|
||||||
* @level INFO/DEBUG/WARNING/FATAL
|
* @level INFO/DEBUG/WARNING/FATAL
|
||||||
* @expression for CHECK calls
|
* @expression for CHECK calls
|
||||||
* @fatal_signal for failed CHECK:SIGABRT or fatal signal caught in the signal handler
|
* @fatal_signal for failed CHECK:SIGABRT or fatal signal caught in the signal handler
|
||||||
*/
|
*/
|
||||||
LogCapture(const char* file, const int line, const char* function, const LEVELS& level, const char* expression = "", int fatal_signal = SIGABRT)
|
LogCapture(const char* file, const int line, const char* function, const LEVELS& level,
|
||||||
: _file(file), _line(line), _function(function), _level(level), _expression(expression), _fatal_signal(fatal_signal) {
|
const char* expression = "", g2::SignalType fatal_signal = SIGABRT, const char* dump = nullptr);
|
||||||
|
|
||||||
if (g2::internal::wasFatal(level)) {
|
|
||||||
_stack_trace = {"\n*******\tSTACKDUMP *******\n"};
|
|
||||||
_stack_trace.append(g2::internal::stackdump());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// At destruction the message will be forwarded to the g2log worker.
|
// At destruction the message will be forwarded to the g2log worker.
|
||||||
// in case of dynamically (at runtime) loaded libraries the important thing to know is that
|
// in case of dynamically (at runtime) loaded libraries the important thing to know is that
|
||||||
// all strings are copied so the original are not destroyed at the receiving end, only the copy
|
// all strings are copied so the original are not destroyed at the receiving end, only the copy
|
||||||
~LogCapture();
|
virtual ~LogCapture();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -53,32 +49,10 @@ struct LogCapture {
|
|||||||
#ifndef __GNUC__
|
#ifndef __GNUC__
|
||||||
#define __attribute__(x) // Disable 'attributes' if compiler does not support 'em
|
#define __attribute__(x) // Disable 'attributes' if compiler does not support 'em
|
||||||
#endif
|
#endif
|
||||||
|
void capturef(const char* printf_like_message, ...) __attribute__((format(printf, 2, 3))); // 2,3 ref: http://www.codemaestro.com/reviews/18
|
||||||
|
|
||||||
void capturef(const char *printf_like_message, ...) __attribute__((format(printf, 2, 3))) // ref: http://www.codemaestro.com/reviews/18
|
|
||||||
{
|
|
||||||
static const int kMaxMessageSize = 2048;
|
|
||||||
static const std::string kTruncatedWarningText = "[...truncated...]";
|
|
||||||
char finished_message[kMaxMessageSize];
|
|
||||||
va_list arglist;
|
|
||||||
va_start(arglist, printf_like_message);
|
|
||||||
|
|
||||||
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__) && !defined(__GNUC__))
|
|
||||||
const int nbrcharacters = vsnprintf_s(finished_message, _countof(finished_message), _TRUNCATE, printf_like_message, arglist);
|
|
||||||
#else
|
|
||||||
const int nbrcharacters = vsnprintf(finished_message, sizeof (finished_message), printf_like_message, arglist);
|
|
||||||
#endif
|
|
||||||
va_end(arglist);
|
|
||||||
|
|
||||||
if (nbrcharacters <= 0) {
|
|
||||||
stream() << "\n\tERROR LOG MSG NOTIFICATION: Failure to parse successfully the message";
|
|
||||||
stream() << '"' << printf_like_message << '"' << std::endl;
|
|
||||||
} else if (nbrcharacters > kMaxMessageSize) {
|
|
||||||
stream() << finished_message << kTruncatedWarningText;
|
|
||||||
} else {
|
|
||||||
stream() << finished_message;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
/// prettifying API for this completely open struct
|
||||||
std::ostringstream& stream() {
|
std::ostringstream& stream() {
|
||||||
return _stream;
|
return _stream;
|
||||||
}
|
}
|
||||||
@ -92,7 +66,7 @@ struct LogCapture {
|
|||||||
const char* _function;
|
const char* _function;
|
||||||
const LEVELS& _level;
|
const LEVELS& _level;
|
||||||
const char* _expression;
|
const char* _expression;
|
||||||
const int _fatal_signal;
|
const g2::SignalType _fatal_signal;
|
||||||
|
|
||||||
};
|
};
|
||||||
//} // g2
|
//} // g2
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
* ********************************************* */
|
* ********************************************* */
|
||||||
|
|
||||||
#include "g2logworker.hpp"
|
#include "g2logworker.hpp"
|
||||||
|
#include "g2logmessage.hpp"
|
||||||
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
@ -46,12 +47,19 @@ namespace g2 {
|
|||||||
// safe to shutdown logging now
|
// safe to shutdown logging now
|
||||||
g2::internal::shutDownLogging();
|
g2::internal::shutDownLogging();
|
||||||
|
|
||||||
std::string signal = msgPtr.get()->signal();
|
std::string reason = msgPtr.get()->reason();
|
||||||
auto fatal_signal_id = msgPtr.get()->_signal_id;
|
const auto level = msgPtr.get()->_level;
|
||||||
|
const auto fatal_id = msgPtr.get()->_signal_id;
|
||||||
|
|
||||||
|
|
||||||
std::unique_ptr<LogMessage> uniqueMsg(std::move(msgPtr.get()));
|
std::unique_ptr<LogMessage> uniqueMsg(std::move(msgPtr.get()));
|
||||||
uniqueMsg->write().append("\nExiting after fatal event (").append(uniqueMsg->level());
|
uniqueMsg->write().append("\nExiting after fatal event (").append(uniqueMsg->level());
|
||||||
uniqueMsg->write().append("). Exiting with signal: ").append(signal)
|
|
||||||
|
|
||||||
|
// Change output in case of a fatal signal (or windows exception)
|
||||||
|
std::string exiting = {"Fatal type: "};
|
||||||
|
|
||||||
|
uniqueMsg->write().append("). ").append(exiting).append(" ").append(reason)
|
||||||
.append("\nLog content flushed flushed sucessfully to sink\n\n");
|
.append("\nLog content flushed flushed sucessfully to sink\n\n");
|
||||||
|
|
||||||
std::cerr << uniqueMsg->message() << std::flush;
|
std::cerr << uniqueMsg->message() << std::flush;
|
||||||
@ -64,7 +72,7 @@ namespace g2 {
|
|||||||
// This clear is absolutely necessary
|
// This clear is absolutely necessary
|
||||||
// All sinks are forced to receive the fatal message above before we continue
|
// All sinks are forced to receive the fatal message above before we continue
|
||||||
_sinks.clear(); // flush all queues
|
_sinks.clear(); // flush all queues
|
||||||
internal::exitWithDefaultSignalHandler(fatal_signal_id);
|
internal::exitWithDefaultSignalHandler(level, fatal_id);
|
||||||
|
|
||||||
// should never reach this point
|
// should never reach this point
|
||||||
perror("g2log exited after receiving FATAL trigger. Flush message status: ");
|
perror("g2log exited after receiving FATAL trigger. Flush message status: ");
|
||||||
|
218
src/stacktrace_windows.cpp
Normal file
218
src/stacktrace_windows.cpp
Normal file
@ -0,0 +1,218 @@
|
|||||||
|
/** ==========================================================================
|
||||||
|
* Original code made by Robert Engeln. Given as a PUBLIC DOMAIN dedication for
|
||||||
|
* the benefit of g3log. It was originally published at:
|
||||||
|
* http://code-freeze.blogspot.com/2012/01/generating-stack-traces-from-c.html
|
||||||
|
|
||||||
|
* 2014-2015: adapted for g3log by Kjell Hedstrom (KjellKod).
|
||||||
|
*
|
||||||
|
* This is PUBLIC DOMAIN to use at your own risk and comes
|
||||||
|
* with no warranties. This code is yours to share, use and modify with no
|
||||||
|
* strings attached and no restrictions or obligations.
|
||||||
|
*
|
||||||
|
* For more information see g3log/LICENSE or refer refer to http://unlicense.org
|
||||||
|
* ============================================================================*/
|
||||||
|
|
||||||
|
#include "stacktrace_windows.hpp"
|
||||||
|
#include <windows.h>
|
||||||
|
#include <DbgHelp.h>
|
||||||
|
#include <map>
|
||||||
|
#include <memory>
|
||||||
|
#include <atomic>
|
||||||
|
#include <cassert>
|
||||||
|
#include <vector>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#if !(defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
|
#error "stacktrace_win.cpp used but not on a windows system"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#define g2_MAP_PAIR_STRINGIFY(x) {x, #x}
|
||||||
|
#define thread_local __declspec(thread)
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
thread_local bool g_thread_local_recursive_crash_check = false;
|
||||||
|
|
||||||
|
const std::map<g2::SignalType, std::string> kExceptionsAsText = {
|
||||||
|
g2_MAP_PAIR_STRINGIFY(EXCEPTION_ACCESS_VIOLATION)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_ARRAY_BOUNDS_EXCEEDED)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_BREAKPOINT)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_DATATYPE_MISALIGNMENT)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_DENORMAL_OPERAND)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_DIVIDE_BY_ZERO)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_INEXACT_RESULT)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_INEXACT_RESULT)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_INVALID_OPERATION)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_OVERFLOW)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_STACK_CHECK)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_FLT_UNDERFLOW)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_ILLEGAL_INSTRUCTION)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_IN_PAGE_ERROR)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_INT_DIVIDE_BY_ZERO)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_INT_OVERFLOW)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_INVALID_DISPOSITION)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_NONCONTINUABLE_EXCEPTION)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_PRIV_INSTRUCTION)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_SINGLE_STEP)
|
||||||
|
, g2_MAP_PAIR_STRINGIFY(EXCEPTION_STACK_OVERFLOW)
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// Using the given context, fill in all the stack frames.
|
||||||
|
// Which then later can be interpreted to human readable text
|
||||||
|
void captureStackTrace(CONTEXT* context, std::vector<uint64_t>& frame_pointers) {
|
||||||
|
DWORD machine_type = 0;
|
||||||
|
STACKFRAME64 frame = {}; // force zeroeing
|
||||||
|
frame.AddrPC.Mode = AddrModeFlat;
|
||||||
|
frame.AddrFrame.Mode = AddrModeFlat;
|
||||||
|
frame.AddrStack.Mode = AddrModeFlat;
|
||||||
|
#ifdef _M_X64
|
||||||
|
frame.AddrPC.Offset = context->Rip;
|
||||||
|
frame.AddrFrame.Offset = context->Rbp;
|
||||||
|
frame.AddrStack.Offset = context->Rsp;
|
||||||
|
machine_type = IMAGE_FILE_MACHINE_AMD64;
|
||||||
|
#else
|
||||||
|
frame.AddrPC.Offset = context->Eip;
|
||||||
|
frame.AddrPC.Offset = context->Ebp;
|
||||||
|
frame.AddrPC.Offset = context->Esp;
|
||||||
|
machine_type = IMAGE_FILE_MACHINE_I386;
|
||||||
|
#endif
|
||||||
|
auto noErrors = TRUE; // Why, Oh Why have Microsoft redefined bool?
|
||||||
|
for (size_t index = 0; index < frame_pointers.size(); ++index)
|
||||||
|
{
|
||||||
|
if (StackWalk64(machine_type,
|
||||||
|
GetCurrentProcess(),
|
||||||
|
GetCurrentThread(),
|
||||||
|
&frame,
|
||||||
|
context,
|
||||||
|
NULL,
|
||||||
|
SymFunctionTableAccess64,
|
||||||
|
SymGetModuleBase64,
|
||||||
|
NULL)) {
|
||||||
|
frame_pointers[index] = frame.AddrPC.Offset;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// extract readable text from a given stack frame. All thanks to
|
||||||
|
// using SymFromAddr and SymGetLineFromAddr64 with the stack pointer
|
||||||
|
std::string getSymbolInformation(const size_t index, const std::vector<uint64_t>& frame_pointers) {
|
||||||
|
auto addr = frame_pointers[index];
|
||||||
|
std::string frame_dump = "stack dump [" + std::to_string(index) + "]\t";
|
||||||
|
|
||||||
|
DWORD64 displacement64;
|
||||||
|
DWORD displacement;
|
||||||
|
char symbol_buffer[sizeof(SYMBOL_INFO) + 256];
|
||||||
|
SYMBOL_INFO* symbol = reinterpret_cast<SYMBOL_INFO*>(symbol_buffer);
|
||||||
|
symbol->SizeOfStruct = sizeof(SYMBOL_INFO);
|
||||||
|
symbol->MaxNameLen = MAX_SYM_NAME;
|
||||||
|
|
||||||
|
IMAGEHLP_LINE64 line;
|
||||||
|
line.SizeOfStruct = sizeof(IMAGEHLP_LINE64);
|
||||||
|
std::string lineInformation;
|
||||||
|
std::string callInformation;
|
||||||
|
if (SymFromAddr(GetCurrentProcess(), addr, &displacement64, symbol)) {
|
||||||
|
callInformation.append(" ").append({symbol->Name, symbol->NameLen});
|
||||||
|
if (SymGetLineFromAddr64(GetCurrentProcess(), addr, &displacement, &line)) {
|
||||||
|
lineInformation.append("\t").append(line.FileName).append(" L: ");
|
||||||
|
lineInformation.append(std::to_string(line.LineNumber));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
frame_dump.append(lineInformation).append(callInformation);
|
||||||
|
return frame_dump;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Retrieves all the symbols for the stack frames, fills them witin a text representation and returns it
|
||||||
|
std::string convertFramesToText(std::vector<uint64_t>& frame_pointers) {
|
||||||
|
std::string dump; // slightly more efficient than ostringstream
|
||||||
|
const size_t kSize = frame_pointers.size();
|
||||||
|
for (size_t index = 0; index < kSize && frame_pointers[index]; ++index) {
|
||||||
|
dump += getSymbolInformation(index, frame_pointers);
|
||||||
|
dump += "\n";
|
||||||
|
}
|
||||||
|
return dump;
|
||||||
|
}
|
||||||
|
} // anonymous
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
namespace stacktrace {
|
||||||
|
/// return the text description of a Windows exception code
|
||||||
|
/// From MSDN GetExceptionCode http://msdn.microsoft.com/en-us/library/windows/desktop/ms679356(v=vs.85).aspx
|
||||||
|
std::string exceptionIdToText(g2::SignalType id) {
|
||||||
|
const auto iter = kExceptionsAsText.find(id);
|
||||||
|
if ( iter == kExceptionsAsText.end()) {
|
||||||
|
std::string unknown {"Unknown/" + std::to_string(id)};
|
||||||
|
return unknown;
|
||||||
|
}
|
||||||
|
|
||||||
|
return iter->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/// helper function: retrieve stackdump from no excisting exception pointer
|
||||||
|
std::string stackdump() {
|
||||||
|
CONTEXT current_context;
|
||||||
|
memset(¤t_context, 0, sizeof(CONTEXT));
|
||||||
|
RtlCaptureContext(¤t_context);
|
||||||
|
return stackdump(¤t_context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// helper function: retrieve stackdump, starting from an exception pointer
|
||||||
|
std::string stackdump(EXCEPTION_POINTERS* info) {
|
||||||
|
auto context = info->ContextRecord;
|
||||||
|
return stackdump(context);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// main stackdump function. retrieve stackdump, from the given context
|
||||||
|
std::string stackdump(CONTEXT* context) {
|
||||||
|
|
||||||
|
if (g_thread_local_recursive_crash_check) {
|
||||||
|
std::string recursive_crash = {"\n\n\n***** Recursive crash detected"};
|
||||||
|
recursive_crash.append(", cannot continue stackdump traversal. *****\n\n\n");
|
||||||
|
return recursive_crash;
|
||||||
|
}
|
||||||
|
g_thread_local_recursive_crash_check = true;
|
||||||
|
|
||||||
|
static std::mutex m;
|
||||||
|
std::lock_guard<std::mutex> lock(m);
|
||||||
|
|
||||||
|
const BOOL kLoadSymModules = TRUE;
|
||||||
|
const auto initialized = SymInitialize(GetCurrentProcess(), nullptr, kLoadSymModules);
|
||||||
|
if (TRUE != initialized) {
|
||||||
|
return {"Error: Cannot call SymInitialize(...) for retrieving symbols in stack"};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
std::shared_ptr<void> RaiiSymCleaner(nullptr, [&](void*) {
|
||||||
|
SymCleanup(GetCurrentProcess());
|
||||||
|
}); // Raii sym cleanup
|
||||||
|
|
||||||
|
|
||||||
|
const size_t kmax_frame_dump_size = 64;
|
||||||
|
std::vector<uint64_t> frame_pointers(kmax_frame_dump_size);
|
||||||
|
// C++11: size set and values are zeroed
|
||||||
|
|
||||||
|
assert(frame_pointers.size() == kmax_frame_dump_size);
|
||||||
|
captureStackTrace(context, frame_pointers);
|
||||||
|
return convertFramesToText(frame_pointers);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
} // stacktrace
|
||||||
|
|
||||||
|
|
||||||
|
|
37
src/stacktrace_windows.hpp
Normal file
37
src/stacktrace_windows.hpp
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
/** ==========================================================================
|
||||||
|
* 2014 by KjellKod.cc AND Robert Engeln.
|
||||||
|
* The stacktrace code was given as a public domain dedication by Robert Engeln
|
||||||
|
* It was originally published at: http://code-freeze.blogspot.com/2012/01/generating-stack-traces-from-c.html
|
||||||
|
* It was (here) modified for g3log purposes.
|
||||||
|
*
|
||||||
|
* This is PUBLIC DOMAIN to use at your own risk and comes
|
||||||
|
* with no warranties. This code is yours to share, use and modify with no
|
||||||
|
* strings attached and no restrictions or obligations.
|
||||||
|
*
|
||||||
|
* For more information see g3log/LICENSE or refer refer to http://unlicense.org
|
||||||
|
* ============================================================================*/
|
||||||
|
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#if !(defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
|
#error "stacktrace_win.cpp used but not on a windows system"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <windows.h>
|
||||||
|
#include "crashhandler.hpp"
|
||||||
|
|
||||||
|
namespace stacktrace {
|
||||||
|
/// return the text description of a Windows exception code
|
||||||
|
std::string exceptionIdToText(g2::SignalType id);
|
||||||
|
|
||||||
|
/// helper function: retrieve stackdump from no excisting exception pointer
|
||||||
|
std::string stackdump();
|
||||||
|
|
||||||
|
/// helper function: retrieve stackdump, starting from an exception pointer
|
||||||
|
std::string stackdump(EXCEPTION_POINTERS* info);
|
||||||
|
|
||||||
|
/// main stackdump function. retrieve stackdump, from the given context
|
||||||
|
std::string stackdump(CONTEXT* context);
|
||||||
|
|
||||||
|
} // stacktrace
|
@ -16,18 +16,26 @@
|
|||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
|
||||||
#if defined(G2LOG_PERFORMANCE)
|
#if defined(G2LOG_PERFORMANCE)
|
||||||
const std::string title{"G2LOG"};
|
const std::string title {
|
||||||
|
"G2LOG"
|
||||||
|
};
|
||||||
#elif defined(GOOGLE_GLOG_PERFORMANCE)
|
#elif defined(GOOGLE_GLOG_PERFORMANCE)
|
||||||
const std::string title{"GOOGLE__GLOG"};
|
const std::string title {
|
||||||
|
"GOOGLE__GLOG"
|
||||||
|
};
|
||||||
#else
|
#else
|
||||||
#error G2LOG_PERFORMANCE or GOOGLE_GLOG_PERFORMANCE was not defined
|
#error G2LOG_PERFORMANCE or GOOGLE_GLOG_PERFORMANCE was not defined
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
const std::string g_path{"./"};
|
const std::string g_path {
|
||||||
|
"./"
|
||||||
|
};
|
||||||
#else
|
#else
|
||||||
const std::string g_path{"/tmp/"};
|
const std::string g_path {
|
||||||
|
"/tmp/"
|
||||||
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
@ -57,14 +65,20 @@ int main(int argc, char** argv)
|
|||||||
const std::string g_prefix_log_name = title + "-performance-" + thread_count_oss.str() + "threads-WORST_LOG";
|
const std::string g_prefix_log_name = title + "-performance-" + thread_count_oss.str() + "threads-WORST_LOG";
|
||||||
const std::string g_measurement_dump = g_path + g_prefix_log_name + "_RESULT.txt";
|
const std::string g_measurement_dump = g_path + g_prefix_log_name + "_RESULT.txt";
|
||||||
const std::string g_measurement_bucket_dump = g_path + g_prefix_log_name + "_RESULT_buckets.txt";
|
const std::string g_measurement_bucket_dump = g_path + g_prefix_log_name + "_RESULT_buckets.txt";
|
||||||
const uint64_t us_to_ms{1000};
|
const uint64_t us_to_ms {
|
||||||
const uint64_t us_to_s{1000000};
|
1000
|
||||||
|
};
|
||||||
|
const uint64_t us_to_s {
|
||||||
|
1000000
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
std::ostringstream oss;
|
std::ostringstream oss;
|
||||||
oss << "\n\n" << title << " performance " << number_of_threads << " threads WORST (PEAK) times\n";
|
oss << "\n\n" << title << " performance " << number_of_threads << " threads WORST (PEAK) times\n";
|
||||||
oss << "Each thread running #: " << g_loop << " * " << g_iterations << " iterations of log entries" << std::endl; // worst mean case is about 10us per log entry
|
oss << "Each thread running #: " << g_loop << " * " << g_iterations << " iterations of log entries" << std::endl; // worst mean case is about 10us per log entry
|
||||||
const uint64_t xtra_margin{2};
|
const uint64_t xtra_margin {
|
||||||
|
2
|
||||||
|
};
|
||||||
oss << "*** It can take som time. Please wait: Approximate wait time on MY PC was: " << number_of_threads * (uint64_t)(g_iterations * 10 * xtra_margin / us_to_s) << " seconds" << std::endl;
|
oss << "*** It can take som time. Please wait: Approximate wait time on MY PC was: " << number_of_threads * (uint64_t)(g_iterations * 10 * xtra_margin / us_to_s) << " seconds" << std::endl;
|
||||||
writeTextToFile(g_measurement_dump, oss.str(), kAppend);
|
writeTextToFile(g_measurement_dump, oss.str(), kAppend);
|
||||||
oss.str(""); // clear the stream
|
oss.str(""); // clear the stream
|
||||||
|
@ -39,7 +39,11 @@ option (USE_G3LOG_UNIT_TEST
|
|||||||
# and this: http://stackoverflow.com/questions/2257464/google-test-and-visual-studio-2010-rc
|
# and this: http://stackoverflow.com/questions/2257464/google-test-and-visual-studio-2010-rc
|
||||||
|
|
||||||
|
|
||||||
SET(tests_to_run test_filechange test_io test_configuration test_concept_sink test_sink)
|
IF (MSVC OR MINGW)
|
||||||
|
SET(OS_SPECIFIC_TEST test_crashhandler_windows)
|
||||||
|
ENDIF(MSVC OR MINGW)
|
||||||
|
|
||||||
|
SET(tests_to_run test_filechange test_io test_configuration test_concept_sink test_sink ${OS_SPECIFIC_TEST})
|
||||||
SET(helper ${DIR_UNIT_TEST}/testing_helpers.h ${DIR_UNIT_TEST}/testing_helpers.cpp)
|
SET(helper ${DIR_UNIT_TEST}/testing_helpers.h ${DIR_UNIT_TEST}/testing_helpers.cpp)
|
||||||
include_directories(${DIR_UNIT_TEST})
|
include_directories(${DIR_UNIT_TEST})
|
||||||
|
|
||||||
@ -56,7 +60,7 @@ option (USE_G3LOG_UNIT_TEST
|
|||||||
IF( NOT(MSVC))
|
IF( NOT(MSVC))
|
||||||
set_target_properties(${test} PROPERTIES COMPILE_FLAGS "-isystem -pthread ")
|
set_target_properties(${test} PROPERTIES COMPILE_FLAGS "-isystem -pthread ")
|
||||||
ENDIF( NOT(MSVC))
|
ENDIF( NOT(MSVC))
|
||||||
target_link_libraries(${test} g3logger gtest_170_lib ${PLATFORM_LINK_LIBRIES})
|
target_link_libraries(${test} g3logger gtest_170_lib)
|
||||||
ENDFOREACH(test)
|
ENDFOREACH(test)
|
||||||
|
|
||||||
#
|
#
|
||||||
|
44
test_unit/test_crashhandler_windows.cpp
Normal file
44
test_unit/test_crashhandler_windows.cpp
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
/** ==========================================================================
|
||||||
|
* 2014 by KjellKod.cc. This is PUBLIC DOMAIN to use at your own risk and comes
|
||||||
|
* with no warranties. This code is yours to share, use and modify with no
|
||||||
|
* strings attached and no restrictions or obligations.
|
||||||
|
*
|
||||||
|
* For more information see g3log/LICENSE or refer refer to http://unlicense.org
|
||||||
|
* ============================================================================*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
#if (defined(WIN32) || defined(_WIN32) || defined(__WIN32__))
|
||||||
|
#include "stacktrace_windows.hpp"
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
|
||||||
|
TEST(CrashHandler_Windows, ExceptionType) {
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(123), "Unknown/123");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(1), "Unknown/1");
|
||||||
|
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_ACCESS_VIOLATION), "EXCEPTION_ACCESS_VIOLATION");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_ARRAY_BOUNDS_EXCEEDED), "EXCEPTION_ARRAY_BOUNDS_EXCEEDED");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_BREAKPOINT),"EXCEPTION_BREAKPOINT");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_DATATYPE_MISALIGNMENT),"EXCEPTION_DATATYPE_MISALIGNMENT");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_DENORMAL_OPERAND),"EXCEPTION_FLT_DENORMAL_OPERAND");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_DIVIDE_BY_ZERO),"EXCEPTION_FLT_DIVIDE_BY_ZERO");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_INEXACT_RESULT),"EXCEPTION_FLT_INEXACT_RESULT");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_INEXACT_RESULT),"EXCEPTION_FLT_INEXACT_RESULT");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_INVALID_OPERATION),"EXCEPTION_FLT_INVALID_OPERATION");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_OVERFLOW),"EXCEPTION_FLT_OVERFLOW");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_STACK_CHECK),"EXCEPTION_FLT_STACK_CHECK");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_FLT_UNDERFLOW),"EXCEPTION_FLT_UNDERFLOW");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_ILLEGAL_INSTRUCTION),"EXCEPTION_ILLEGAL_INSTRUCTION");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_IN_PAGE_ERROR),"EXCEPTION_IN_PAGE_ERROR");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_INT_DIVIDE_BY_ZERO),"EXCEPTION_INT_DIVIDE_BY_ZERO");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_INT_OVERFLOW),"EXCEPTION_INT_OVERFLOW");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_INVALID_DISPOSITION),"EXCEPTION_INVALID_DISPOSITION");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_NONCONTINUABLE_EXCEPTION),"EXCEPTION_NONCONTINUABLE_EXCEPTION");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_PRIV_INSTRUCTION),"EXCEPTION_PRIV_INSTRUCTION");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_SINGLE_STEP),"EXCEPTION_SINGLE_STEP");
|
||||||
|
EXPECT_EQ(stacktrace::exceptionIdToText(EXCEPTION_STACK_OVERFLOW),"EXCEPTION_STACK_OVERFLOW");
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // defined WIN32
|
@ -111,7 +111,7 @@ namespace testing_helpers {
|
|||||||
using namespace g2;
|
using namespace g2;
|
||||||
g2::initializeLogging(_scope->_currentWorker.get());
|
g2::initializeLogging(_scope->_currentWorker.get());
|
||||||
clearMockFatal();
|
clearMockFatal();
|
||||||
internal::changeFatalInitHandlerForUnitTesting(&mockFatalCall);
|
internal::setFatalExitHandler(&mockFatalCall);
|
||||||
|
|
||||||
auto filename = _handle->call(&FileSink::fileName);
|
auto filename = _handle->call(&FileSink::fileName);
|
||||||
if (!filename.valid()) ADD_FAILURE();
|
if (!filename.valid()) ADD_FAILURE();
|
||||||
|
Loading…
Reference in New Issue
Block a user