/****************************************************************************** Copyright (C) 2013 by Hugh Bailey This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. This program 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . ******************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include "qt-wrappers.hpp" #include "obs-app.hpp" #include "window-basic-main.hpp" #include "window-basic-settings.hpp" #include "window-license-agreement.hpp" #include "crash-report.hpp" #include "platform.hpp" #include #ifdef _WIN32 #include #else #include #endif using namespace std; static log_handler_t def_log_handler; static string currentLogFile; static string lastLogFile; static bool portable_mode = false; QObject *CreateShortcutFilter() { return new OBSEventFilter([](QObject *obj, QEvent *event) { auto mouse_event = [](QMouseEvent &event) { obs_key_combination_t hotkey = {0, OBS_KEY_NONE}; bool pressed = event.type() == QEvent::MouseButtonPress; switch (event.button()) { case Qt::NoButton: case Qt::LeftButton: case Qt::RightButton: case Qt::AllButtons: case Qt::MouseButtonMask: return false; case Qt::MidButton: hotkey.key = OBS_KEY_MOUSE3; break; #define MAP_BUTTON(i, j) case Qt::ExtraButton ## i: \ hotkey.key = OBS_KEY_MOUSE ## j; break; MAP_BUTTON( 1, 4); MAP_BUTTON( 2, 5); MAP_BUTTON( 3, 6); MAP_BUTTON( 4, 7); MAP_BUTTON( 5, 8); MAP_BUTTON( 6, 9); MAP_BUTTON( 7, 10); MAP_BUTTON( 8, 11); MAP_BUTTON( 9, 12); MAP_BUTTON(10, 13); MAP_BUTTON(11, 14); MAP_BUTTON(12, 15); MAP_BUTTON(13, 16); MAP_BUTTON(14, 17); MAP_BUTTON(15, 18); MAP_BUTTON(16, 19); MAP_BUTTON(17, 20); MAP_BUTTON(18, 21); MAP_BUTTON(19, 22); MAP_BUTTON(20, 23); MAP_BUTTON(21, 24); MAP_BUTTON(22, 25); MAP_BUTTON(23, 26); MAP_BUTTON(24, 27); #undef MAP_BUTTON } hotkey.modifiers = TranslateQtKeyboardEventModifiers( event.modifiers()); obs_hotkey_inject_event(hotkey, pressed); return true; }; auto key_event = [&](QKeyEvent *event) { QDialog *dialog = qobject_cast(obj); obs_key_combination_t hotkey = {0, OBS_KEY_NONE}; bool pressed = event->type() == QEvent::KeyPress; switch (event->key()) { case Qt::Key_Shift: case Qt::Key_Control: case Qt::Key_Alt: case Qt::Key_Meta: break; #ifdef __APPLE__ case Qt::Key_CapsLock: // kVK_CapsLock == 57 hotkey.key = obs_key_from_virtual_key(57); pressed = true; break; #endif case Qt::Key_Enter: case Qt::Key_Escape: case Qt::Key_Return: if (dialog && pressed) return false; default: hotkey.key = obs_key_from_virtual_key( event->nativeVirtualKey()); } hotkey.modifiers = TranslateQtKeyboardEventModifiers( event->modifiers()); obs_hotkey_inject_event(hotkey, pressed); return true; }; switch (event->type()) { case QEvent::MouseButtonPress: case QEvent::MouseButtonRelease: return mouse_event(*static_cast(event)); /*case QEvent::MouseButtonDblClick: case QEvent::Wheel:*/ case QEvent::KeyPress: case QEvent::KeyRelease: return key_event(static_cast(event)); default: return false; } }); } string CurrentTimeString() { time_t now = time(0); struct tm tstruct; char buf[80]; tstruct = *localtime(&now); strftime(buf, sizeof(buf), "%X", &tstruct); return buf; } string CurrentDateTimeString() { time_t now = time(0); struct tm tstruct; char buf[80]; tstruct = *localtime(&now); strftime(buf, sizeof(buf), "%Y-%m-%d, %X", &tstruct); return buf; } static void do_log(int log_level, const char *msg, va_list args, void *param) { fstream &logFile = *static_cast(param); char str[4096]; #ifndef _WIN32 va_list args2; va_copy(args2, args); #endif vsnprintf(str, 4095, msg, args); #ifdef _WIN32 OutputDebugStringA(str); OutputDebugStringA("\n"); #else def_log_handler(log_level, msg, args2, nullptr); #endif if (log_level <= LOG_INFO) logFile << CurrentTimeString() << ": " << str << endl; #ifdef _WIN32 if (log_level <= LOG_ERROR && IsDebuggerPresent()) __debugbreak(); #endif } #define DEFAULT_LANG "en-US" bool OBSApp::InitGlobalConfigDefaults() { config_set_default_string(globalConfig, "General", "Language", DEFAULT_LANG); config_set_default_uint(globalConfig, "General", "MaxLogs", 10); #if _WIN32 config_set_default_string(globalConfig, "Video", "Renderer", "Direct3D 11"); #else config_set_default_string(globalConfig, "Video", "Renderer", "OpenGL"); #endif config_set_default_bool(globalConfig, "BasicWindow", "PreviewEnabled", true); return true; } static bool do_mkdir(const char *path) { if (os_mkdir(path) == MKDIR_ERROR) { OBSErrorBox(NULL, "Failed to create directory %s", path); return false; } return true; } static bool MakeUserDirs() { char path[512]; if (portable_mode) { if (GetConfigPath(path, sizeof(path), "") <= 0) return false; if (!do_mkdir(path)) return false; } if (GetConfigPath(path, sizeof(path), "obs-studio") <= 0) return false; if (!do_mkdir(path)) return false; if (GetConfigPath(path, sizeof(path), "obs-studio/basic") <= 0) return false; if (!do_mkdir(path)) return false; if (GetConfigPath(path, sizeof(path), "obs-studio/logs") <= 0) return false; if (!do_mkdir(path)) return false; #ifdef _WIN32 if (GetConfigPath(path, sizeof(path), "obs-studio/crashes") <= 0) return false; if (!do_mkdir(path)) return false; #endif return true; } static bool MakeUserProfileDirs() { char path[512]; if (GetConfigPath(path, sizeof(path), "obs-studio/basic/profiles") <= 0) return false; if (!do_mkdir(path)) return false; if (GetConfigPath(path, sizeof(path), "obs-studio/basic/scenes") <= 0) return false; if (!do_mkdir(path)) return false; return true; } bool OBSApp::InitGlobalConfig() { char path[512]; int len = GetConfigPath(path, sizeof(path), "obs-studio/global.ini"); if (len <= 0) { return false; } int errorcode = globalConfig.Open(path, CONFIG_OPEN_ALWAYS); if (errorcode != CONFIG_SUCCESS) { OBSErrorBox(NULL, "Failed to open global.ini: %d", errorcode); return false; } return InitGlobalConfigDefaults(); } bool OBSApp::InitLocale() { const char *lang = config_get_string(globalConfig, "General", "Language"); locale = lang; string englishPath; if (!GetDataFilePath("locale/" DEFAULT_LANG ".ini", englishPath)) { OBSErrorBox(NULL, "Failed to find locale/" DEFAULT_LANG ".ini"); return false; } textLookup = text_lookup_create(englishPath.c_str()); if (!textLookup) { OBSErrorBox(NULL, "Failed to create locale from file '%s'", englishPath.c_str()); return false; } bool userLocale = config_has_user_value(globalConfig, "General", "Language"); bool defaultLang = astrcmpi(lang, DEFAULT_LANG) == 0; if (userLocale && defaultLang) return true; if (!userLocale && defaultLang) { for (auto &locale_ : GetPreferredLocales()) { if (locale_ == lang) return true; stringstream file; file << "locale/" << locale_ << ".ini"; string path; if (!GetDataFilePath(file.str().c_str(), path)) continue; if (!text_lookup_add(textLookup, path.c_str())) continue; blog(LOG_INFO, "Using preferred locale '%s'", locale_.c_str()); locale = locale_; return true; } return true; } stringstream file; file << "locale/" << lang << ".ini"; string path; if (GetDataFilePath(file.str().c_str(), path)) { if (!text_lookup_add(textLookup, path.c_str())) blog(LOG_ERROR, "Failed to add locale file '%s'", path.c_str()); } else { blog(LOG_ERROR, "Could not find locale file '%s'", file.str().c_str()); } return true; } bool OBSApp::SetTheme(std::string name, std::string path) { theme = name; /* Check user dir first, then preinstalled themes. */ if (path == "") { char userDir[512]; name = "themes/" + name + ".qss"; string temp = "obs-studio/" + name; int ret = GetConfigPath(userDir, sizeof(userDir), temp.c_str()); if (ret > 0 && QFile::exists(userDir)) { path = string(userDir); } else if (!GetDataFilePath(name.c_str(), path)) { OBSErrorBox(NULL, "Failed to find %s.", name.c_str()); return false; } } QString mpath = QString("file:///") + path.c_str(); setStyleSheet(mpath); return true; } bool OBSApp::InitTheme() { const char *themeName = config_get_string(globalConfig, "General", "Theme"); if (!themeName) themeName = "Default"; stringstream t; t << themeName; return SetTheme(t.str()); } OBSApp::OBSApp(int &argc, char **argv) : QApplication(argc, argv) {} static void move_basic_to_profiles(void) { char path[512]; char new_path[512]; os_glob_t *glob; /* if not first time use */ if (GetConfigPath(path, 512, "obs-studio/basic") <= 0) return; if (!os_file_exists(path)) return; /* if the profiles directory doesn't already exist */ if (GetConfigPath(new_path, 512, "obs-studio/basic/profiles") <= 0) return; if (os_file_exists(new_path)) return; if (os_mkdir(new_path) == MKDIR_ERROR) return; strcat(new_path, "/"); strcat(new_path, Str("Untitled")); if (os_mkdir(new_path) == MKDIR_ERROR) return; strcat(path, "/*.*"); if (os_glob(path, 0, &glob) != 0) return; strcpy(path, new_path); for (size_t i = 0; i < glob->gl_pathc; i++) { struct os_globent ent = glob->gl_pathv[i]; char *file; if (ent.directory) continue; file = strrchr(ent.path, '/'); if (!file++) continue; if (astrcmpi(file, "scenes.json") == 0) continue; strcpy(new_path, path); strcat(new_path, "/"); strcat(new_path, file); os_rename(ent.path, new_path); } os_globfree(glob); } static void move_basic_to_scene_collections(void) { char path[512]; char new_path[512]; if (GetConfigPath(path, 512, "obs-studio/basic") <= 0) return; if (!os_file_exists(path)) return; if (GetConfigPath(new_path, 512, "obs-studio/basic/scenes") <= 0) return; if (os_file_exists(new_path)) return; if (os_mkdir(new_path) == MKDIR_ERROR) return; strcat(path, "/scenes.json"); strcat(new_path, "/"); strcat(new_path, Str("Untitled")); strcat(new_path, ".json"); os_rename(path, new_path); } void OBSApp::AppInit() { if (!InitApplicationBundle()) throw "Failed to initialize application bundle"; if (!MakeUserDirs()) throw "Failed to create required user directories"; if (!InitGlobalConfig()) throw "Failed to initialize global config"; if (!InitLocale()) throw "Failed to load locale"; if (!InitTheme()) throw "Failed to load theme"; config_set_default_string(globalConfig, "Basic", "Profile", Str("Untitled")); config_set_default_string(globalConfig, "Basic", "ProfileDir", Str("Untitled")); config_set_default_string(globalConfig, "Basic", "SceneCollection", Str("Untitled")); config_set_default_string(globalConfig, "Basic", "SceneCollectionFile", Str("Untitled")); move_basic_to_profiles(); move_basic_to_scene_collections(); if (!MakeUserProfileDirs()) throw "Failed to create profile directories"; } const char *OBSApp::GetRenderModule() const { const char *renderer = config_get_string(globalConfig, "Video", "Renderer"); return (astrcmpi(renderer, "Direct3D 11") == 0) ? DL_D3D11 : DL_OPENGL; } bool OBSApp::OBSInit() { bool licenseAccepted = config_get_bool(globalConfig, "General", "LicenseAccepted"); OBSLicenseAgreement agreement(nullptr); if (licenseAccepted || agreement.exec() == QDialog::Accepted) { if (!licenseAccepted) { config_set_bool(globalConfig, "General", "LicenseAccepted", true); config_save(globalConfig); } mainWindow = new OBSBasic(); mainWindow->setAttribute(Qt::WA_DeleteOnClose, true); connect(mainWindow, SIGNAL(destroyed()), this, SLOT(quit())); mainWindow->OBSInit(); connect(this, &QGuiApplication::applicationStateChanged, [](Qt::ApplicationState state) { obs_hotkey_enable_background_press( state != Qt::ApplicationActive); }); obs_hotkey_enable_background_press( applicationState() != Qt::ApplicationActive); return true; } else { return false; } } string OBSApp::GetVersionString() const { stringstream ver; #ifdef HAVE_OBSCONFIG_H ver << OBS_VERSION; #else ver << LIBOBS_API_MAJOR_VER << "." << LIBOBS_API_MINOR_VER << "." << LIBOBS_API_PATCH_VER; #endif ver << " ("; #ifdef _WIN32 if (sizeof(void*) == 8) ver << "64bit, "; ver << "windows)"; #elif __APPLE__ ver << "mac)"; #elif __FreeBSD__ ver << "freebsd)"; #else /* assume linux for the time being */ ver << "linux)"; #endif return ver.str(); } #ifdef __APPLE__ #define INPUT_AUDIO_SOURCE "coreaudio_input_capture" #define OUTPUT_AUDIO_SOURCE "coreaudio_output_capture" #elif _WIN32 #define INPUT_AUDIO_SOURCE "wasapi_input_capture" #define OUTPUT_AUDIO_SOURCE "wasapi_output_capture" #else #define INPUT_AUDIO_SOURCE "pulse_input_capture" #define OUTPUT_AUDIO_SOURCE "pulse_output_capture" #endif const char *OBSApp::InputAudioSource() const { return INPUT_AUDIO_SOURCE; } const char *OBSApp::OutputAudioSource() const { return OUTPUT_AUDIO_SOURCE; } const char *OBSApp::GetLastLog() const { return lastLogFile.c_str(); } const char *OBSApp::GetCurrentLog() const { return currentLogFile.c_str(); } QString OBSTranslator::translate(const char *context, const char *sourceText, const char *disambiguation, int n) const { const char *out = nullptr; if (!text_lookup_getstr(App()->GetTextLookup(), sourceText, &out)) return QString(); UNUSED_PARAMETER(context); UNUSED_PARAMETER(disambiguation); UNUSED_PARAMETER(n); return QT_UTF8(out); } static bool get_token(lexer *lex, string &str, base_token_type type) { base_token token; if (!lexer_getbasetoken(lex, &token, IGNORE_WHITESPACE)) return false; if (token.type != type) return false; str.assign(token.text.array, token.text.len); return true; } static bool expect_token(lexer *lex, const char *str, base_token_type type) { base_token token; if (!lexer_getbasetoken(lex, &token, IGNORE_WHITESPACE)) return false; if (token.type != type) return false; return strref_cmp(&token.text, str) == 0; } static uint64_t convert_log_name(const char *name) { BaseLexer lex; string year, month, day, hour, minute, second; lexer_start(lex, name); if (!get_token(lex, year, BASETOKEN_DIGIT)) return 0; if (!expect_token(lex, "-", BASETOKEN_OTHER)) return 0; if (!get_token(lex, month, BASETOKEN_DIGIT)) return 0; if (!expect_token(lex, "-", BASETOKEN_OTHER)) return 0; if (!get_token(lex, day, BASETOKEN_DIGIT)) return 0; if (!get_token(lex, hour, BASETOKEN_DIGIT)) return 0; if (!expect_token(lex, "-", BASETOKEN_OTHER)) return 0; if (!get_token(lex, minute, BASETOKEN_DIGIT)) return 0; if (!expect_token(lex, "-", BASETOKEN_OTHER)) return 0; if (!get_token(lex, second, BASETOKEN_DIGIT)) return 0; stringstream timestring; timestring << year << month << day << hour << minute << second; return std::stoull(timestring.str()); } static void delete_oldest_file(const char *location) { BPtr logDir(GetConfigPathPtr(location)); string oldestLog; uint64_t oldest_ts = (uint64_t)-1; struct os_dirent *entry; unsigned int maxLogs = (unsigned int)config_get_uint( App()->GlobalConfig(), "General", "MaxLogs"); os_dir_t *dir = os_opendir(logDir); if (dir) { unsigned int count = 0; while ((entry = os_readdir(dir)) != NULL) { if (entry->directory || *entry->d_name == '.') continue; uint64_t ts = convert_log_name(entry->d_name); if (ts) { if (ts < oldest_ts) { oldestLog = entry->d_name; oldest_ts = ts; } count++; } } os_closedir(dir); if (count > maxLogs) { stringstream delPath; delPath << logDir << "/" << oldestLog; os_unlink(delPath.str().c_str()); } } } static void get_last_log(void) { BPtr logDir(GetConfigPathPtr("obs-studio/logs")); struct os_dirent *entry; os_dir_t *dir = os_opendir(logDir); uint64_t highest_ts = 0; if (dir) { while ((entry = os_readdir(dir)) != NULL) { if (entry->directory || *entry->d_name == '.') continue; uint64_t ts = convert_log_name(entry->d_name); if (ts > highest_ts) { lastLogFile = entry->d_name; highest_ts = ts; } } os_closedir(dir); } } string GenerateTimeDateFilename(const char *extension) { time_t now = time(0); char file[256] = {}; struct tm *cur_time; cur_time = localtime(&now); snprintf(file, sizeof(file), "%d-%02d-%02d %02d-%02d-%02d.%s", cur_time->tm_year+1900, cur_time->tm_mon+1, cur_time->tm_mday, cur_time->tm_hour, cur_time->tm_min, cur_time->tm_sec, extension); return string(file); } vector> GetLocaleNames() { string path; if (!GetDataFilePath("locale.ini", path)) throw "Could not find locale.ini path"; ConfigFile ini; if (ini.Open(path.c_str(), CONFIG_OPEN_EXISTING) != 0) throw "Could not open locale.ini"; size_t sections = config_num_sections(ini); vector> names; names.reserve(sections); for (size_t i = 0; i < sections; i++) { const char *tag = config_get_section(ini, i); const char *name = config_get_string(ini, tag, "Name"); names.emplace_back(tag, name); } return names; } static void create_log_file(fstream &logFile) { stringstream dst; get_last_log(); currentLogFile = GenerateTimeDateFilename("txt"); dst << "obs-studio/logs/" << currentLogFile.c_str(); BPtr path(GetConfigPathPtr(dst.str().c_str())); logFile.open(path, ios_base::in | ios_base::out | ios_base::trunc); if (logFile.is_open()) { delete_oldest_file("obs-studio/logs"); base_set_log_handler(do_log, &logFile); } else { blog(LOG_ERROR, "Failed to open log file"); } } static int run_program(fstream &logFile, int argc, char *argv[]) { int ret = -1; QCoreApplication::addLibraryPath("."); OBSApp program(argc, argv); try { program.AppInit(); OBSTranslator translator; create_log_file(logFile); program.installTranslator(&translator); ret = program.OBSInit() ? program.exec() : 0; } catch (const char *error) { blog(LOG_ERROR, "%s", error); OBSErrorBox(nullptr, "%s", error); } return ret; } #define MAX_CRASH_REPORT_SIZE (50 * 1024) #ifdef _WIN32 #define CRASH_MESSAGE \ "Woops, OBS has crashed!\n\nWould you like to copy the crash log " \ "to the clipboard? (Crash logs will still be saved to the " \ "%appdata%\\obs-studio\\crashes directory)" static void main_crash_handler(const char *format, va_list args, void *param) { char *text = new char[MAX_CRASH_REPORT_SIZE]; vsnprintf(text, MAX_CRASH_REPORT_SIZE, format, args); delete_oldest_file("obs-studio/crashes"); string name = "obs-studio/crashes/Crash "; name += GenerateTimeDateFilename("txt"); BPtr path(GetConfigPathPtr(name.c_str())); fstream file; file.open(path, ios_base::in | ios_base::out | ios_base::trunc); file << text; file.close(); int ret = MessageBoxA(NULL, CRASH_MESSAGE, "OBS has crashed!", MB_YESNO | MB_ICONERROR | MB_TASKMODAL); if (ret == IDYES) { size_t len = strlen(text); HGLOBAL mem = GlobalAlloc(GMEM_MOVEABLE, len); memcpy(GlobalLock(mem), text, len); GlobalUnlock(mem); OpenClipboard(0); EmptyClipboard(); SetClipboardData(CF_TEXT, mem); CloseClipboard(); } exit(-1); UNUSED_PARAMETER(param); } static void load_debug_privilege(void) { const DWORD flags = TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY; TOKEN_PRIVILEGES tp; HANDLE token; LUID val; if (!OpenProcessToken(GetCurrentProcess(), flags, &token)) { return; } if (!!LookupPrivilegeValue(NULL, SE_DEBUG_NAME, &val)) { tp.PrivilegeCount = 1; tp.Privileges[0].Luid = val; tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED; AdjustTokenPrivileges(token, false, &tp, sizeof(tp), NULL, NULL); } CloseHandle(token); } #endif #ifdef __APPLE__ #define BASE_PATH ".." #else #define BASE_PATH "../.." #endif #define CONFIG_PATH BASE_PATH "/config" #ifndef OBS_UNIX_STRUCTURE #define OBS_UNIX_STRUCTURE 0 #endif int GetConfigPath(char *path, size_t size, const char *name) { if (!OBS_UNIX_STRUCTURE && portable_mode) { if (name && *name) { return snprintf(path, size, CONFIG_PATH "/%s", name); } else { return snprintf(path, size, CONFIG_PATH); } } else { return os_get_config_path(path, size, name); } } char *GetConfigPathPtr(const char *name) { if (!OBS_UNIX_STRUCTURE && portable_mode) { char path[512]; if (snprintf(path, sizeof(path), CONFIG_PATH "/%s", name) > 0) { return bstrdup(path); } else { return NULL; } } else { return os_get_config_path_ptr(name); } } bool GetFileSafeName(const char *name, std::string &file) { size_t base_len = strlen(name); size_t len = os_utf8_to_wcs(name, base_len, nullptr, 0); std::wstring wfile; if (!len) return false; wfile.resize(len); os_utf8_to_wcs(name, base_len, &wfile[0], len); for (size_t i = wfile.size(); i > 0; i--) { size_t im1 = i - 1; if (iswspace(wfile[im1])) { wfile[im1] = '_'; } else if (wfile[im1] != '_' && !iswalnum(wfile[im1])) { wfile.erase(im1, 1); } } if (wfile.size() == 0) wfile = L"characters_only"; len = os_wcs_to_utf8(wfile.c_str(), wfile.size(), nullptr, 0); if (!len) return false; file.resize(len); os_wcs_to_utf8(wfile.c_str(), wfile.size(), &file[0], len); return true; } bool GetClosestUnusedFileName(std::string &path, const char *extension) { size_t len = path.size(); if (extension) { path += "."; path += extension; } if (!os_file_exists(path.c_str())) return true; int index = 1; do { path.resize(len); path += std::to_string(++index); if (extension) { path += "."; path += extension; } } while (os_file_exists(path.c_str())); return true; } static inline bool arg_is(const char *arg, const char *long_form, const char *short_form) { return (long_form && strcmp(arg, long_form) == 0) || (short_form && strcmp(arg, short_form) == 0); } int main(int argc, char *argv[]) { #ifndef _WIN32 signal(SIGPIPE, SIG_IGN); #endif #ifdef _WIN32 load_debug_privilege(); base_set_crash_handler(main_crash_handler, nullptr); #endif base_get_log_handler(&def_log_handler, nullptr); for (int i = 1; i < argc; i++) { if (arg_is(argv[i], "--portable", "-p")) { portable_mode = true; } } #if !OBS_UNIX_STRUCTURE if (!portable_mode) { portable_mode = os_file_exists(BASE_PATH "/portable_mode") || os_file_exists(BASE_PATH "/obs_portable_mode") || os_file_exists(BASE_PATH "/portable_mode.txt") || os_file_exists(BASE_PATH "/obs_portable_mode.txt"); } #endif fstream logFile; int ret = run_program(logFile, argc, argv); blog(LOG_INFO, "Number of memory leaks: %ld", bnum_allocs()); base_set_log_handler(nullptr, nullptr); return ret; }