What a mess to merge from Github/KjellKod/g3log to BitBucket/KjellKod/g3log

This commit is contained in:
KjellKod 2015-02-02 00:31:43 -07:00
parent 1f844f5ea1
commit 938fc55971
27 changed files with 1509 additions and 768 deletions

View File

@ -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;

View File

@ -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)

View File

@ -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
View 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)

View File

@ -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);

View File

@ -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)

View File

@ -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) {

View File

@ -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();
} }

View File

@ -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

View File

@ -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

View 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

View File

@ -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

View File

@ -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__)

View File

@ -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

View File

@ -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);
} }

View File

@ -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);
} }

View File

@ -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;
}; };

View File

@ -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;
}
}

View File

@ -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

View File

@ -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
View 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(&current_context, 0, sizeof(CONTEXT));
RtlCaptureContext(&current_context);
return stackdump(&current_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

View 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

View File

@ -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

View File

@ -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)
# #

View 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

View File

@ -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();