mirror of https://github.com/BOINC/boinc.git
1016 lines
29 KiB
C++
1016 lines
29 KiB
C++
// This file is part of BOINC.
|
|
// http://boinc.berkeley.edu
|
|
// Copyright (C) 2023 University of California
|
|
//
|
|
// BOINC is free software; you can redistribute it and/or modify it
|
|
// under the terms of the GNU Lesser General Public License
|
|
// as published by the Free Software Foundation,
|
|
// either version 3 of the License, or (at your option) any later version.
|
|
//
|
|
// BOINC is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
// See the GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
// Stuff related to stderr/stdout direction and exception handling;
|
|
// used by both core client and by apps
|
|
|
|
#if defined(_WIN32)
|
|
#include "boinc_win.h"
|
|
#endif
|
|
|
|
#ifdef __EMX__
|
|
#include <sys/stat.h>
|
|
#endif
|
|
|
|
#ifndef _WIN32
|
|
#include "config.h"
|
|
#include <cstring>
|
|
#include <cstdio>
|
|
#include <cstdarg>
|
|
#include <cstdlib>
|
|
#include <unistd.h>
|
|
#include <time.h>
|
|
#endif
|
|
|
|
#include "boinc_stdio.h"
|
|
|
|
#ifdef __APPLE__
|
|
#include "mac_backtrace.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_EXECINFO_H
|
|
#include <execinfo.h>
|
|
#endif
|
|
|
|
#include "app_ipc.h"
|
|
#include "error_numbers.h"
|
|
#include "filesys.h"
|
|
#include "util.h"
|
|
#include "str_replace.h"
|
|
#include "parse.h"
|
|
#include "str_replace.h"
|
|
|
|
|
|
#include "diagnostics.h"
|
|
|
|
bool main_exited; // set at end of main()
|
|
|
|
#ifdef ANDROID_VOODOO
|
|
// for signal handler backtrace
|
|
unwind_backtrace_signal_arch_t unwind_backtrace_signal_arch;
|
|
acquire_my_map_info_list_t acquire_my_map_info_list;
|
|
release_my_map_info_list_t release_my_map_info_list;
|
|
get_backtrace_symbols_t get_backtrace_symbols;
|
|
free_backtrace_symbols_t free_backtrace_symbols;
|
|
load_symbol_table_t load_symbol_table;
|
|
free_symbol_table_t free_symbol_table;
|
|
find_symbol_t find_symbol;
|
|
format_backtrace_line_t format_backtrace_line;
|
|
#endif
|
|
|
|
|
|
#if defined(_WIN32) && defined(_MSC_VER)
|
|
|
|
static _CrtMemState start_snapshot;
|
|
static _CrtMemState finish_snapshot;
|
|
static _CrtMemState difference_snapshot;
|
|
|
|
#endif
|
|
|
|
static int diagnostics_initialized = false;
|
|
static int flags;
|
|
static char stdout_log[MAXPATHLEN];
|
|
static char stdout_archive[MAXPATHLEN];
|
|
static FILE* stdout_file;
|
|
static char stderr_log[MAXPATHLEN];
|
|
static char stderr_archive[MAXPATHLEN];
|
|
static FILE* stderr_file;
|
|
static char boinc_dir[MAXPATHLEN];
|
|
static char boinc_install_dir[MAXPATHLEN];
|
|
static int boinc_proxy_enabled;
|
|
static char boinc_proxy[256];
|
|
static char symstore[256];
|
|
static int aborted_via_gui;
|
|
static double stderr_file_size = 0;
|
|
static double max_stderr_file_size = 2048*1024;
|
|
static double stdout_file_size = 0;
|
|
static double max_stdout_file_size = 2048*1024;
|
|
|
|
#ifdef ANDROID_VOODOO
|
|
static void* libhandle;
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
|
|
// Starting with Visual Studio 2005 the C Runtime Library has really started to
|
|
// enforce parameter validation. Problem is that the parameter validation code
|
|
// uses its own structured exception handler and terminates without writing
|
|
// any useful output to stderr. Microsoft has created a hook an application
|
|
// developer can use to get more debugging information which is the purpose
|
|
// of this function. When an invalid parameter is passed to the C Runtime
|
|
// library this function will write whatever trace information it can and
|
|
// then throw a breakpoint exception to dump all the rest of the useful
|
|
// information.
|
|
static void boinc_catch_signal_invalid_parameter(
|
|
const wchar_t* expression, const wchar_t* function, const wchar_t* file, unsigned int line, uintptr_t /* pReserved */
|
|
) {
|
|
if (function && file && expression) {
|
|
fprintf(
|
|
stderr,
|
|
"ERROR: Invalid parameter detected in function %S. File: %S Line: %d\n",
|
|
function,
|
|
file,
|
|
line
|
|
);
|
|
fprintf(
|
|
stderr,
|
|
"ERROR: Expression: %S\n",
|
|
expression
|
|
);
|
|
} else {
|
|
fputs("ERROR: Invalid parameter detected in CRT function\n", stderr);
|
|
}
|
|
|
|
// Cause a Debug Breakpoint.
|
|
DebugBreak();
|
|
}
|
|
|
|
// Override default terminate and abort functions, call DebugBreak instead.
|
|
//
|
|
static void boinc_term_func() {
|
|
|
|
// Cause a Debug Breakpoint.
|
|
DebugBreak();
|
|
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
|
|
// Trap ASSERTs and TRACEs from the CRT and spew them to stderr.
|
|
//
|
|
#if defined(wxUSE_GUI)
|
|
int __cdecl boinc_message_reporting(int, char *, int *retVal){
|
|
// in wxWidgets, we don't know if main has returned
|
|
(*retVal) = 0;
|
|
return 0;
|
|
}
|
|
#else
|
|
int __cdecl boinc_message_reporting(int reportType, char *szMsg, int *retVal){
|
|
int n;
|
|
(*retVal) = 0;
|
|
|
|
// can't call CRT functions after main returns
|
|
//
|
|
if (main_exited) return 0;
|
|
|
|
|
|
switch(reportType){
|
|
|
|
case _CRT_WARN:
|
|
case _CRT_ERROR:
|
|
|
|
if (flags & BOINC_DIAG_TRACETOSTDERR) {
|
|
n = fprintf(stderr, "%s", szMsg);
|
|
if (n > 0) stderr_file_size += n;
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_TRACETOSTDOUT) {
|
|
n = fprintf(stdout, "%s", szMsg);
|
|
if (n > 0) stdout_file_size += n;
|
|
}
|
|
|
|
break;
|
|
case _CRT_ASSERT:
|
|
|
|
n = fprintf(stderr, "ASSERT: %s\n", szMsg);
|
|
if (n > 0) stderr_file_size += n;
|
|
|
|
(*retVal) = 1;
|
|
break;
|
|
|
|
}
|
|
return(TRUE);
|
|
}
|
|
#endif
|
|
|
|
#endif // _DEBUG
|
|
#endif // _WIN32
|
|
|
|
|
|
// initialize the app diagnostic environment.
|
|
//
|
|
int boinc_init_diagnostics(int _flags) {
|
|
return diagnostics_init(
|
|
BOINC_DIAG_BOINCAPPLICATION | _flags,
|
|
BOINC_DIAG_STDOUT, BOINC_DIAG_STDERR
|
|
);
|
|
}
|
|
|
|
|
|
// initialize the graphic diagnostic environment.
|
|
//
|
|
int boinc_init_graphics_diagnostics(int _flags) {
|
|
return diagnostics_init(
|
|
BOINC_DIAG_BOINCAPPLICATION | _flags,
|
|
BOINC_DIAG_GFX_STDOUT, BOINC_DIAG_GFX_STDERR
|
|
);
|
|
}
|
|
|
|
|
|
// Used to cleanup the diagnostics environment.
|
|
//
|
|
int boinc_finish_diag() {
|
|
return diagnostics_finish();
|
|
}
|
|
|
|
int boinc_install_signal_handlers() {
|
|
#ifdef _WIN32
|
|
SetUnhandledExceptionFilter(boinc_catch_signal);
|
|
#if defined(_MSC_VER) && (_MSC_VER >= 1400)
|
|
_set_invalid_parameter_handler(boinc_catch_signal_invalid_parameter);
|
|
#endif
|
|
#else //_WIN32
|
|
|
|
// register handlers for fatal internal signals
|
|
// so that they get reported in stderr.txt
|
|
// Do NOT catch SIGQUIT because core client uses that to kill app
|
|
//
|
|
boinc_set_signal_handler(SIGILL, boinc_catch_signal);
|
|
boinc_set_signal_handler(SIGABRT, boinc_catch_signal);
|
|
boinc_set_signal_handler(SIGBUS, boinc_catch_signal);
|
|
boinc_set_signal_handler(SIGSEGV, boinc_catch_signal);
|
|
boinc_set_signal_handler(SIGSYS, boinc_catch_signal);
|
|
boinc_set_signal_handler(SIGPIPE, boinc_catch_signal);
|
|
#endif //_WIN32
|
|
return 0;
|
|
}
|
|
|
|
|
|
// initialize the diagnostics environment.
|
|
//
|
|
int diagnostics_init(
|
|
int _flags, const char* stdout_prefix, const char* stderr_prefix
|
|
) {
|
|
// Check to see if we have already been called
|
|
//
|
|
if (diagnostics_initialized) {
|
|
return ERR_INVALID_PARAM;
|
|
}
|
|
diagnostics_initialized = true;
|
|
|
|
// Setup initial values
|
|
//
|
|
flags = _flags;
|
|
safe_strcpy(stdout_log, "");
|
|
safe_strcpy(stdout_archive, "");
|
|
safe_strcpy(stderr_log, "");
|
|
safe_strcpy(stderr_archive, "");
|
|
safe_strcpy(boinc_dir, "");
|
|
safe_strcpy(boinc_install_dir, "");
|
|
boinc_proxy_enabled = 0;
|
|
safe_strcpy(boinc_proxy, "");
|
|
safe_strcpy(symstore, "");
|
|
|
|
|
|
// Check for invalid parameter combinations
|
|
//
|
|
if ((flags & BOINC_DIAG_REDIRECTSTDERR) && (flags & BOINC_DIAG_REDIRECTSTDERROVERWRITE)) {
|
|
return ERR_INVALID_PARAM;
|
|
}
|
|
|
|
if ((flags & BOINC_DIAG_REDIRECTSTDOUT) && (flags & BOINC_DIAG_REDIRECTSTDOUTOVERWRITE)) {
|
|
return ERR_INVALID_PARAM;
|
|
}
|
|
|
|
|
|
// Determine where the log files are to be stored
|
|
//
|
|
if (flags & BOINC_DIAG_PERUSERLOGFILES) {
|
|
char user_dir[MAXPATHLEN];
|
|
|
|
#if defined(_WIN32)
|
|
snprintf(user_dir, sizeof(user_dir), "%s", getenv("APPDATA"));
|
|
safe_strcat(user_dir, "/BOINC");
|
|
#elif defined(__APPLE__)
|
|
snprintf(user_dir, sizeof(user_dir), "%s", getenv("HOME"));
|
|
safe_strcat(user_dir, "/Library/Application Support/BOINC");
|
|
#else
|
|
snprintf(user_dir, sizeof(user_dir), "%s", getenv("HOME"));
|
|
safe_strcat(user_dir, "/.BOINC");
|
|
#endif
|
|
|
|
// Check to see if the directory exists
|
|
if (!is_dir(user_dir)) {
|
|
boinc_mkdir(user_dir);
|
|
}
|
|
|
|
snprintf(stdout_log, sizeof(stdout_log), "%.*s/%.*s.txt", DIR_LEN, user_dir, FILE_LEN, stdout_prefix);
|
|
snprintf(stdout_archive, sizeof(stdout_archive), "%.*s/%.*s.old", DIR_LEN, user_dir, FILE_LEN, stdout_prefix);
|
|
snprintf(stderr_log, sizeof(stderr_log), "%.*s/%.*s.txt", DIR_LEN, user_dir, FILE_LEN, stderr_prefix);
|
|
snprintf(stderr_archive, sizeof(stderr_archive), "%.*s/%.*s.old", DIR_LEN, user_dir, FILE_LEN, stderr_prefix);
|
|
|
|
} else {
|
|
|
|
snprintf(stdout_log, sizeof(stdout_log), "%s.txt", stdout_prefix);
|
|
snprintf(stdout_archive, sizeof(stdout_archive), "%s.old", stdout_prefix);
|
|
snprintf(stderr_log, sizeof(stderr_log), "%s.txt", stderr_prefix);
|
|
snprintf(stderr_archive, sizeof(stderr_archive), "%s.old", stderr_prefix);
|
|
|
|
}
|
|
|
|
// Archive any old stderr.txt and stdout.txt files, if requested
|
|
//
|
|
if (flags & BOINC_DIAG_ARCHIVESTDERR) {
|
|
boinc_copy(stderr_log, stderr_archive);
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_ARCHIVESTDOUT) {
|
|
boinc_copy(stdout_log, stdout_archive);
|
|
}
|
|
|
|
// Redirect stderr and/or stdout, if requested
|
|
//
|
|
if (flags & BOINC_DIAG_REDIRECTSTDERR) {
|
|
file_size(stderr_log, stderr_file_size);
|
|
#ifdef _WIN32
|
|
stderr_file = freopen(stderr_log, "ac", stderr);
|
|
#else
|
|
stderr_file = freopen(stderr_log, "a", stderr);
|
|
#endif
|
|
if (!stderr_file) {
|
|
return ERR_FOPEN;
|
|
}
|
|
setbuf(stderr_file, 0);
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_REDIRECTSTDERROVERWRITE) {
|
|
#ifdef _WIN32
|
|
stderr_file = freopen(stderr_log, "wc", stderr);
|
|
#else
|
|
stderr_file = freopen(stderr_log, "w", stderr);
|
|
#endif
|
|
if (!stderr_file) {
|
|
return ERR_FOPEN;
|
|
}
|
|
setbuf(stderr_file, 0);
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_REDIRECTSTDOUT) {
|
|
file_size(stdout_log, stdout_file_size);
|
|
stdout_file = freopen(stdout_log, "a", stdout);
|
|
if (!stdout_file) {
|
|
return ERR_FOPEN;
|
|
}
|
|
setvbuf(stdout_file, NULL, _IOLBF, BUFSIZ);
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_REDIRECTSTDOUTOVERWRITE) {
|
|
stdout_file = freopen(stdout_log, "w", stdout);
|
|
if (!stdout_file) {
|
|
return ERR_FOPEN;
|
|
}
|
|
setvbuf(stdout_file, NULL, _IOLBF, BUFSIZ);
|
|
}
|
|
|
|
|
|
#if defined(_WIN32)
|
|
|
|
//_set_abort_behavior(NULL, _WRITE_ABORT_MSG);
|
|
#ifdef __MINGW32__
|
|
std::set_terminate(boinc_term_func);
|
|
std::set_unexpected(boinc_term_func);
|
|
#else
|
|
set_terminate(boinc_term_func);
|
|
set_unexpected(boinc_term_func);
|
|
#endif
|
|
|
|
#if defined(_DEBUG)
|
|
|
|
_CrtSetReportHook(boinc_message_reporting);
|
|
|
|
if (flags & BOINC_DIAG_MEMORYLEAKCHECKENABLED) {
|
|
SET_CRT_DEBUG_FIELD(_CRTDBG_LEAK_CHECK_DF);
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_HEAPCHECKENABLED) {
|
|
if (flags & BOINC_DIAG_HEAPCHECKEVERYALLOC) {
|
|
SET_CRT_DEBUG_FIELD(_CRTDBG_CHECK_ALWAYS_DF);
|
|
} else {
|
|
SET_CRT_DEBUG_FIELD(_CRTDBG_CHECK_EVERY_1024_DF);
|
|
}
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_BOINCAPPLICATION) {
|
|
if (flags & BOINC_DIAG_MEMORYLEAKCHECKENABLED) {
|
|
_CrtMemCheckpoint(&start_snapshot);
|
|
}
|
|
}
|
|
|
|
#endif // defined(_DEBUG)
|
|
|
|
// Initialize the thread list structure
|
|
// The data for this structure should be set by
|
|
// boinc_init or boinc_init_graphics.
|
|
diagnostics_init_thread_list();
|
|
|
|
diagnostics_init_unhandled_exception_monitor();
|
|
|
|
diagnostics_init_message_monitor();
|
|
|
|
#endif // defined(_WIN32)
|
|
|
|
#ifdef ANDROID_VOODOO
|
|
#define resolve_func(l,x) \
|
|
x=(x##_t)dlsym(l,#x); \
|
|
if (!x) {\
|
|
fprintf(stderr,"Unable to resolve function %s\n",#x); \
|
|
unwind_backtrace_signal_arch=NULL; \
|
|
}
|
|
|
|
if ((libhandle=dlopen("libcorkscrew.so",RTLD_NOW|RTLD_GLOBAL))) {
|
|
resolve_func(libhandle,unwind_backtrace_signal_arch);
|
|
resolve_func(libhandle,acquire_my_map_info_list);
|
|
resolve_func(libhandle,release_my_map_info_list);
|
|
resolve_func(libhandle,get_backtrace_symbols);
|
|
resolve_func(libhandle,free_backtrace_symbols);
|
|
resolve_func(libhandle,format_backtrace_line);
|
|
resolve_func(libhandle,load_symbol_table);
|
|
resolve_func(libhandle,free_symbol_table);
|
|
resolve_func(libhandle,find_symbol);
|
|
} else {
|
|
fprintf(stderr,"stackdumps unavailable\n");
|
|
}
|
|
#endif // ANDROID_VOODOO
|
|
|
|
// Our PrintBactrace() won't work in MacOS screensaver so let
|
|
// the MacOS handle signals and write backtrace to stderr.
|
|
#if !(defined (__APPLE__) && defined(SCREENSAVER))
|
|
// Install unhandled exception filters and signal traps.
|
|
if (BOINC_SUCCESS != boinc_install_signal_handlers()) {
|
|
return ERR_SIGNAL_OP;
|
|
}
|
|
#endif
|
|
|
|
// Store various pieces of inforation for future use.
|
|
if (flags & BOINC_DIAG_BOINCAPPLICATION) {
|
|
char buf[256];
|
|
char proxy_address[256];
|
|
int proxy_port;
|
|
MIOFILE mf;
|
|
FILE* p;
|
|
#ifdef _WIN32
|
|
LONG lReturnValue;
|
|
HKEY hkSetupHive;
|
|
DWORD dwSize = 0;
|
|
#endif
|
|
|
|
safe_strcpy(buf, "");
|
|
safe_strcpy(proxy_address, "");
|
|
proxy_port = 0;
|
|
|
|
#ifndef _USING_FCGI_
|
|
p = fopen(INIT_DATA_FILE, "r");
|
|
#else
|
|
p = FCGI::fopen(INIT_DATA_FILE, "r");
|
|
#endif
|
|
|
|
if (p) {
|
|
mf.init_file(p);
|
|
while(mf.fgets(buf, sizeof(buf))) {
|
|
if (match_tag(buf, "</app_init_data>")) break;
|
|
else if (parse_str(buf, "<boinc_dir>", boinc_dir, sizeof(boinc_dir))) continue;
|
|
else if (parse_str(buf, "<symstore>", symstore, sizeof(symstore))) ;
|
|
else if (match_tag(buf, "<use_http_proxy/>")) {
|
|
boinc_proxy_enabled = true;
|
|
continue;
|
|
}
|
|
else if (parse_str(buf, "<http_server_name>", proxy_address, sizeof(proxy_address))) continue;
|
|
else if (parse_int(buf, "<http_server_port>", proxy_port)) continue;
|
|
}
|
|
fclose(p);
|
|
}
|
|
|
|
if (boinc_proxy_enabled) {
|
|
int buffer_used = snprintf(boinc_proxy, sizeof(boinc_proxy), "%s:%d", proxy_address, proxy_port);
|
|
if ((sizeof(boinc_proxy) == buffer_used) || (-1 == buffer_used)) {
|
|
boinc_proxy[sizeof(boinc_proxy)-1] = '\0';
|
|
}
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
// Lookup the location of where BOINC was installed to and store
|
|
// that for future use.
|
|
lReturnValue = RegOpenKeyEx(
|
|
HKEY_LOCAL_MACHINE,
|
|
_T("SOFTWARE\\Space Sciences Laboratory, U.C. Berkeley\\BOINC Setup"),
|
|
0,
|
|
KEY_READ,
|
|
&hkSetupHive
|
|
);
|
|
if (lReturnValue == ERROR_SUCCESS) {
|
|
// How large does our buffer need to be?
|
|
dwSize = sizeof(boinc_install_dir);
|
|
|
|
lReturnValue = RegQueryValueEx(
|
|
hkSetupHive,
|
|
_T("INSTALLDIR"),
|
|
NULL,
|
|
NULL,
|
|
(LPBYTE)&boinc_install_dir,
|
|
&dwSize
|
|
);
|
|
}
|
|
|
|
if (hkSetupHive) RegCloseKey(hkSetupHive);
|
|
#endif
|
|
}
|
|
|
|
return BOINC_SUCCESS;
|
|
}
|
|
|
|
int diagnostics_thread_init() {
|
|
// Install unhandled exception filters and signal traps.
|
|
if (BOINC_SUCCESS != boinc_install_signal_handlers()) {
|
|
return ERR_SIGNAL_OP;
|
|
}
|
|
|
|
return BOINC_SUCCESS;
|
|
}
|
|
|
|
// Cleanup the diagnostic framework before dumping any memory leaks.
|
|
//
|
|
int diagnostics_finish() {
|
|
|
|
#ifdef _WIN32
|
|
|
|
// Shutdown the message monitor thread and handles
|
|
diagnostics_finish_message_monitor();
|
|
|
|
// Shutdown the unhandled exception filter thread and handles
|
|
diagnostics_finish_unhandled_exception_monitor();
|
|
|
|
// Cleanup internal thread list structures and free up any
|
|
// used memory.
|
|
diagnostics_finish_thread_list();
|
|
|
|
#ifdef _DEBUG
|
|
|
|
// Only perform the memory leak dump if it is a boinc application
|
|
// and not the BOINC Manager, BOINC Core Client, or BOINC
|
|
// Screen saver since they'll check on close.
|
|
if (flags & BOINC_DIAG_BOINCAPPLICATION) {
|
|
if (flags & BOINC_DIAG_MEMORYLEAKCHECKENABLED) {
|
|
_CrtMemCheckpoint(&finish_snapshot);
|
|
if (_CrtMemDifference(&difference_snapshot, &start_snapshot, &finish_snapshot)) {
|
|
fprintf(stdout, "\n\n");
|
|
fprintf(stdout, "**********\n");
|
|
fprintf(stdout, "**********\n");
|
|
fprintf(stdout, "\n");
|
|
fprintf(stdout, "Memory Leaks Detected!!!\n");
|
|
fprintf(stdout, "\n");
|
|
fprintf(stdout, "Memory Statistics:\n");
|
|
_CrtMemDumpStatistics(&difference_snapshot);
|
|
fprintf(stdout, "\n");
|
|
_CrtMemDumpAllObjectsSince(&difference_snapshot);
|
|
fprintf(stdout, "\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif // defined(_DEBUG)
|
|
#endif // defined(_WIN32)
|
|
|
|
#ifdef ANDROID_VOODOO
|
|
if (libhandle) {
|
|
dlclose(libhandle);
|
|
}
|
|
#endif
|
|
|
|
// Set initialization flag to false.
|
|
diagnostics_initialized = false;
|
|
|
|
return BOINC_SUCCESS;
|
|
}
|
|
|
|
// has the diagnostics library been initialized?
|
|
//
|
|
int diagnostics_is_initialized(){
|
|
return diagnostics_initialized;
|
|
}
|
|
|
|
|
|
// return true if the specified flag is set.
|
|
//
|
|
int diagnostics_is_flag_set(int _flags) {
|
|
return flags & _flags;
|
|
}
|
|
|
|
|
|
// return the location of the BOINC directory.
|
|
//
|
|
char* diagnostics_get_boinc_dir() {
|
|
return boinc_dir;
|
|
}
|
|
|
|
|
|
// return the location of the BOINC install directory.
|
|
//
|
|
char* diagnostics_get_boinc_install_dir() {
|
|
return boinc_install_dir;
|
|
}
|
|
|
|
|
|
// return the location of the symbol store.
|
|
//
|
|
char* diagnostics_get_symstore() {
|
|
return symstore;
|
|
}
|
|
|
|
|
|
// store the location of the symbol store.
|
|
//
|
|
int diagnostics_set_symstore(const char* project_symstore) {
|
|
safe_strcpy(symstore, project_symstore);
|
|
return 0;
|
|
}
|
|
|
|
|
|
// do we need to worry about a proxy server?
|
|
//
|
|
int diagnostics_is_proxy_enabled() {
|
|
return boinc_proxy_enabled;
|
|
}
|
|
|
|
|
|
// proxy server address and port
|
|
//
|
|
char* diagnostics_get_proxy() {
|
|
return boinc_proxy;
|
|
}
|
|
|
|
|
|
// Set the value of the flag
|
|
int diagnostics_set_aborted_via_gui() {
|
|
aborted_via_gui = 1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
// Return the value of he flag
|
|
int diagnostics_is_aborted_via_gui() {
|
|
return aborted_via_gui;
|
|
}
|
|
|
|
// Cycle the log files at regular events.
|
|
//
|
|
int diagnostics_cycle_logs() {
|
|
// If the stderr.txt or stdout.txt files are too big, cycle them
|
|
//
|
|
if (flags & BOINC_DIAG_REDIRECTSTDERR) {
|
|
if (stderr_file_size > max_stderr_file_size) {
|
|
if (NULL == stderr_file) return ERR_FOPEN;
|
|
fclose(stderr_file);
|
|
boinc_copy(stderr_log, stderr_archive);
|
|
stderr_file_size = 0;
|
|
stderr_file = freopen(stderr_log, "w", stderr);
|
|
if (NULL == stderr_file) return ERR_FOPEN;
|
|
setbuf(stderr_file, 0);
|
|
}
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_REDIRECTSTDOUT) {
|
|
if (stdout_file_size > max_stdout_file_size) {
|
|
if (NULL == stdout_file) return ERR_FOPEN;
|
|
fclose(stdout_file);
|
|
stdout_file_size = 0;
|
|
boinc_copy(stdout_log, stdout_archive);
|
|
stdout_file = freopen(stdout_log, "w", stdout);
|
|
if (NULL == stdout_file) return ERR_FOPEN;
|
|
setvbuf(stdout_file, NULL, _IOLBF, BUFSIZ);
|
|
}
|
|
}
|
|
return BOINC_SUCCESS;
|
|
}
|
|
|
|
|
|
// Diagnostics for POSIX Compatible systems.
|
|
//
|
|
#if defined(HAVE_SIGNAL_H) && !defined(_WIN32)
|
|
|
|
// Set a signal handler only if it is not currently ignored
|
|
//
|
|
extern "C" void boinc_set_signal_handler(int sig, handler_t handler) {
|
|
#if HAVE_SIGACTION
|
|
struct sigaction temp;
|
|
sigaction(sig, NULL, &temp);
|
|
if (temp.sa_handler != SIG_IGN) {
|
|
temp.sa_sigaction = handler;
|
|
sigaction(sig, &temp, NULL);
|
|
}
|
|
#else
|
|
void (*temp)(int);
|
|
temp = signal(sig, boinc_catch_signal);
|
|
if (temp == SIG_IGN) {
|
|
signal(sig, SIG_IGN);
|
|
}
|
|
#endif /* HAVE_SIGACTION */
|
|
}
|
|
|
|
|
|
// Set a signal handler even if it is currently ignored
|
|
//
|
|
void boinc_set_signal_handler_force(int sig, void(*handler)(int)) {
|
|
#if HAVE_SIGACTION
|
|
struct sigaction temp;
|
|
sigaction(sig, NULL, &temp);
|
|
temp.sa_handler = handler;
|
|
sigaction(sig, &temp, NULL);
|
|
#else
|
|
void (*temp)(int);
|
|
temp = signal(sig, boinc_catch_signal);
|
|
signal(sig, SIG_IGN);
|
|
#endif /* HAVE_SIGACTION */
|
|
}
|
|
|
|
// exit code to use if signalled; can be changed
|
|
static int signal_exit_code = EXIT_SIGNAL;
|
|
|
|
void set_signal_exit_code(int x) {
|
|
signal_exit_code = x;
|
|
}
|
|
|
|
#ifdef ANDROID_VOODOO
|
|
const char *argv0;
|
|
|
|
static char *xtoa(size_t x) {
|
|
static char buf[20];
|
|
static char hex[]="0123456789abcdef";
|
|
int n;
|
|
buf[19]=0;
|
|
n=18;
|
|
while (x) {
|
|
buf[n--]=hex[x&0xf];
|
|
x/=0x10;
|
|
}
|
|
buf[n--]='x';
|
|
buf[n]='0';
|
|
return buf+n;
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SIGACTION
|
|
#ifdef ANDROID_VOODOO
|
|
void boinc_catch_signal(int signal, struct siginfo *siginfo, void *sigcontext) {
|
|
#else
|
|
void boinc_catch_signal(int signal, siginfo_t*, void *) {
|
|
#endif // ANDROID
|
|
#else
|
|
void boinc_catch_signal(int signal) {
|
|
#endif // HAVE_SIGACTION
|
|
switch(signal) {
|
|
case SIGHUP: fprintf(stderr, "SIGHUP: terminal line hangup\n");
|
|
return;
|
|
case SIGINT: fprintf(stderr, "SIGINT: interrupt program\n"); break;
|
|
case SIGILL: fprintf(stderr, "SIGILL: illegal instruction\n"); break;
|
|
case SIGABRT: fprintf(stderr, "SIGABRT: abort called\n"); break;
|
|
#if SIGBUS != SIGSEGV
|
|
// in case SIGBUS == SIGSEGV (e.g., Haiku)
|
|
case SIGBUS: fprintf(stderr, "SIGBUS: bus error\n"); break;
|
|
#endif
|
|
case SIGSEGV: fprintf(stderr, "SIGSEGV: segmentation violation\n"); break;
|
|
case SIGSYS: fprintf(stderr, "SIGSYS: system call given invalid argument\n"); break;
|
|
case SIGPIPE: fprintf(stderr, "SIGPIPE: write on a pipe with no reader\n");
|
|
return;
|
|
default: fprintf(stderr, "unknown signal %d\n", signal); break;
|
|
}
|
|
|
|
#ifdef HAVE_EXECINFO_H
|
|
void *array[64];
|
|
size_t size;
|
|
size = backtrace (array, 64);
|
|
// Anything that calls malloc here (i.e *printf()) will probably fail
|
|
// so we'll do it the hard way.
|
|
int retval = write(fileno(stderr),"Stack trace (",strlen("Stack trace ("));
|
|
char mbuf[10];
|
|
char *p=mbuf+9;
|
|
int i=size;
|
|
*(p--)=0;
|
|
while (i) {
|
|
*(p--)=i%10+'0';
|
|
i/=10;
|
|
}
|
|
retval = write(fileno(stderr),p+1,strlen(p+1));
|
|
retval = write(fileno(stderr)," frames):",strlen(" frames):"));
|
|
mbuf[0]=10;
|
|
retval = write(fileno(stderr),mbuf,1);
|
|
backtrace_symbols_fd(array, size, fileno(stderr));
|
|
if (retval) {}
|
|
#endif
|
|
|
|
#ifdef __APPLE__
|
|
PrintBacktrace();
|
|
#endif
|
|
|
|
#ifdef ANDROID_VOODOO
|
|
// this is some dark undocumented Android voodoo that uses libcorkscrew.so.
|
|
// Minimal use of library functions because they may not work in a signal
|
|
// handler.
|
|
//
|
|
#define DUMP_LINE_LEN 256
|
|
static backtrace_frame_t backtrace[64];
|
|
static backtrace_symbol_t backtrace_symbols[64];
|
|
if (unwind_backtrace_signal_arch != NULL) {
|
|
map_info_t *map_info = acquire_my_map_info_list();
|
|
ssize_t size = unwind_backtrace_signal_arch(
|
|
siginfo, sigcontext, map_info, backtrace, 0, 64
|
|
);
|
|
get_backtrace_symbols(backtrace, size, backtrace_symbols);
|
|
char line[DUMP_LINE_LEN];
|
|
for (int i=0; i<size; i++) {
|
|
format_backtrace_line(
|
|
i, &backtrace[i], &backtrace_symbols[i], line, DUMP_LINE_LEN
|
|
);
|
|
line[DUMP_LINE_LEN-1] = 0;
|
|
if (backtrace_symbols[i].symbol_name) {
|
|
strlcat(line," ",DUMP_LINE_LEN);
|
|
if (backtrace_symbols[i].demangled_name) {
|
|
strlcat(
|
|
line, backtrace_symbols[i].demangled_name,
|
|
DUMP_LINE_LEN
|
|
);
|
|
}
|
|
} else {
|
|
symbol_table_t* symbols = NULL;
|
|
if (backtrace_symbols[i].map_name) {
|
|
symbols = load_symbol_table(backtrace_symbols[i].map_name);
|
|
} else {
|
|
symbols = load_symbol_table(argv0);
|
|
}
|
|
symbol_t* symbol = NULL;
|
|
if (symbols) {
|
|
symbol = find_symbol(symbols, backtrace[i].absolute_pc);
|
|
}
|
|
if (symbol) {
|
|
int offset = backtrace[i].absolute_pc - symbol->start;
|
|
strlcat(line, " (", DUMP_LINE_LEN);
|
|
strlcat(line, symbol->name, DUMP_LINE_LEN);
|
|
strlcat(line, "+", DUMP_LINE_LEN);
|
|
strlcat(line, xtoa(offset), DUMP_LINE_LEN);
|
|
strlcat(line, ")", DUMP_LINE_LEN);
|
|
line[DUMP_LINE_LEN-1] = 0;
|
|
} else {
|
|
strlcat(line, " (\?\?\?)", DUMP_LINE_LEN);
|
|
}
|
|
if (symbols) free_symbol_table(symbols);
|
|
}
|
|
if (backtrace[i].absolute_pc) {
|
|
strlcat(line, " [", DUMP_LINE_LEN);
|
|
strlcat(
|
|
line,
|
|
xtoa(*reinterpret_cast<unsigned int *>(backtrace[i].absolute_pc)),
|
|
DUMP_LINE_LEN
|
|
);
|
|
strlcat(line, "]", DUMP_LINE_LEN);
|
|
}
|
|
strlcat(line, "\n", DUMP_LINE_LEN);
|
|
write(fileno(stderr),line, strlen(line));
|
|
fflush(stderr);
|
|
}
|
|
}
|
|
#endif // ANDROID_VOODOO
|
|
|
|
fprintf(stderr, "\nExiting...\n");
|
|
_exit(signal_exit_code);
|
|
}
|
|
|
|
#endif
|
|
|
|
//
|
|
// Diagnostics Routines common to all Platforms
|
|
//
|
|
|
|
// Converts the BOINCTRACE macro into a single string and report it
|
|
// to the CRT so it can be reported via the normal means.
|
|
//
|
|
void boinc_trace(const char *pszFormat, ...) {
|
|
static char szBuffer[4096];
|
|
static char szDate[64];
|
|
static char szTime[64];
|
|
int n;
|
|
|
|
// Trace messages should only be reported if running as a standalone
|
|
// application or told too.
|
|
if ((flags & BOINC_DIAG_TRACETOSTDERR) ||
|
|
(flags & BOINC_DIAG_TRACETOSTDOUT)) {
|
|
|
|
memset(szBuffer, 0, sizeof(szBuffer));
|
|
memset(szDate, 0, sizeof(szDate));
|
|
memset(szTime, 0, sizeof(szTime));
|
|
|
|
#ifdef _WIN32
|
|
strdate(szDate);
|
|
strtime(szTime);
|
|
#else
|
|
time_t t;
|
|
char *theCR;
|
|
|
|
time(&t);
|
|
safe_strcpy(szTime, asctime(localtime(&t)));
|
|
theCR = strrchr(szTime, '\n');
|
|
if (theCR) *theCR = '\0';
|
|
theCR = strrchr(szTime, '\r');
|
|
if (theCR) *theCR = '\0';
|
|
#endif
|
|
|
|
va_list ptr;
|
|
va_start(ptr, pszFormat);
|
|
|
|
vsnprintf(szBuffer, sizeof(szBuffer), pszFormat, ptr);
|
|
|
|
va_end(ptr);
|
|
|
|
#if defined(_WIN32) && defined(_DEBUG)
|
|
n = _CrtDbgReport(_CRT_WARN, NULL, NULL, NULL, "[%s %s] TRACE [%lu]: %s", szDate, szTime, GetCurrentThreadId(), szBuffer);
|
|
#else
|
|
if (flags & BOINC_DIAG_TRACETOSTDERR) {
|
|
#ifdef _WIN32
|
|
n = fprintf(stderr, "[%s %s] TRACE [%lu]: %s\n", szDate, szTime, GetCurrentThreadId(), szBuffer);
|
|
#else
|
|
n = fprintf(stderr, "[%s] TRACE: %s\n", szTime, szBuffer);
|
|
#endif
|
|
if (n > 0) stderr_file_size += n;
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_TRACETOSTDOUT) {
|
|
#ifdef _WIN32
|
|
n = fprintf(stdout, "[%s %s] TRACE [%lu]: %s\n", szDate, szTime, GetCurrentThreadId(), szBuffer);
|
|
#else
|
|
n = fprintf(stdout, "[%s] TRACE: %s\n", szTime, szBuffer);
|
|
#endif
|
|
if (n > 0) stdout_file_size += n;
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
// Converts the BOINCINFO macro into a single string and report it
|
|
// to stderr so it can be reported via the normal means.
|
|
//
|
|
#ifndef BOINC_INFOMSGS
|
|
void boinc_info(const char* /*pszFormat*/, ... ){ return; }
|
|
#else
|
|
void boinc_info(const char* pszFormat, ...){
|
|
static char szBuffer[4096];
|
|
static char szDate[64];
|
|
static char szTime[64];
|
|
int n;
|
|
|
|
memset(szBuffer, 0, sizeof(szBuffer));
|
|
memset(szDate, 0, sizeof(szDate));
|
|
memset(szTime, 0, sizeof(szTime));
|
|
|
|
strdate(szDate);
|
|
strtime(szTime);
|
|
|
|
va_list ptr;
|
|
va_start(ptr, pszFormat);
|
|
|
|
vsnprintf(szBuffer, sizeof(szBuffer), pszFormat, ptr);
|
|
|
|
va_end(ptr);
|
|
|
|
#if defined(_WIN32) && defined(_DEBUG)
|
|
_CrtDbgReport(_CRT_WARN, NULL, NULL, NULL, "[%s %s] BOINCMSG: %s\n", szDate, szTime, szBuffer);
|
|
#else
|
|
if (flags & BOINC_DIAG_TRACETOSTDERR) {
|
|
n = fprintf(stderr, "[%s %s] BOINCMSG: %s\n", szDate, szTime, szBuffer);
|
|
if (n > 0) stderr_file_size += n;
|
|
}
|
|
|
|
if (flags & BOINC_DIAG_TRACETOSTDOUT) {
|
|
n = fprintf(stdout, "[%s %s] BOINCMSG: %s\n", szDate, szTime, szBuffer);
|
|
if (n > 0) stdout_file_size += n;
|
|
}
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
void diagnostics_set_max_file_sizes(double stdout_size, double stderr_size) {
|
|
if (stdout_size) max_stdout_file_size = stdout_size;
|
|
if (stderr_size) max_stderr_file_size = stderr_size;
|
|
}
|
|
|
|
// Dump string to whatever the platform debuggers
|
|
//
|
|
#ifndef _WIN32
|
|
int diagnostics_trace_to_debugger(const char*) {
|
|
return 0;
|
|
}
|
|
#endif
|