/****************************************************************************** Copyright (C) 2013 by Hugh Bailey Copyright (C) 2014 by Zachary Lund 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 "obs-app.hpp" #include #include #include #include #include #include "platform.hpp" #ifdef __linux__ #include #include #include #include #include #endif #if defined(__FreeBSD__) || defined(__DragonFly__) #include #include #include #include #include #include #include #include #include #endif using namespace std; #ifdef __linux__ void RunningInstanceCheck(bool &already_running) { int uniq = socket(AF_LOCAL, SOCK_DGRAM | SOCK_CLOEXEC, 0); if (uniq == -1) { blog(LOG_ERROR, "Failed to check for running instance, socket: %d", errno); already_running = 0; return; } struct sockaddr_un bindInfo; memset(&bindInfo, 0, sizeof(sockaddr_un)); bindInfo.sun_family = AF_LOCAL; char *abstactSockName = NULL; asprintf(&abstactSockName, "%s %d %s", "/com/obsproject", getpid(), App()->GetVersionString().c_str()); memmove(bindInfo.sun_path + 1, abstactSockName, strlen(abstactSockName)); free(abstactSockName); int bindErr = bind(uniq, (struct sockaddr *)&bindInfo, sizeof(struct sockaddr_un)); already_running = bindErr == 0 ? 0 : 1; if (already_running) { return; } FILE *fp = fopen("/proc/net/unix", "re"); if (fp == NULL) { return; } char *line = NULL; size_t n = 0; int obsCnt = 0; while (getdelim(&line, &n, ' ', fp) != EOF) { line[strcspn(line, "\n")] = '\0'; if (*line == '@') { if (strstr(line, "@/com/obsproject") != NULL) { ++obsCnt; } } } already_running = obsCnt == 1 ? 0 : 1; free(line); fclose(fp); } #endif #if defined(__FreeBSD__) || defined(__DragonFly__) struct RunOnce { std::thread thr; static const char *thr_name; std::condition_variable cv; std::condition_variable wait_cv; std::mutex mtx; bool exiting = false; bool name_changed = false; void thr_proc() { std::unique_lock lk(mtx); pthread_set_name_np(pthread_self(), thr_name); name_changed = true; wait_cv.notify_all(); cv.wait(lk, [this]() { return exiting; }); } ~RunOnce() { if (thr.joinable()) { std::unique_lock lk(mtx); exiting = true; cv.notify_one(); lk.unlock(); thr.join(); } } } RO; const char *RunOnce::thr_name = "OBS runonce"; void PIDFileCheck(bool &already_running) { std::string tmpfile_name = "/tmp/obs-studio.lock." + to_string(geteuid()); int fd = open(tmpfile_name.c_str(), O_RDWR | O_CREAT | O_EXLOCK, 0600); if (fd == -1) { already_running = true; return; } already_running = false; procstat *ps = procstat_open_sysctl(); unsigned int count; auto procs = procstat_getprocs(ps, KERN_PROC_UID | KERN_PROC_INC_THREAD, geteuid(), &count); for (unsigned int i = 0; i < count; i++) { if (!strncmp(procs[i].ki_tdname, RunOnce::thr_name, sizeof(procs[i].ki_tdname))) { already_running = true; break; } } procstat_freeprocs(ps, procs); procstat_close(ps); RO.thr = std::thread(std::mem_fn(&RunOnce::thr_proc), &RO); { std::unique_lock lk(RO.mtx); RO.wait_cv.wait(lk, []() { return RO.name_changed; }); } unlink(tmpfile_name.c_str()); close(fd); } #endif static inline bool check_path(const char *data, const char *path, string &output) { ostringstream str; str << path << data; output = str.str(); printf("Attempted path: %s\n", output.c_str()); return (access(output.c_str(), R_OK) == 0); } #define INSTALL_DATA_PATH OBS_INSTALL_PREFIX OBS_DATA_PATH "/obs-studio/" bool GetDataFilePath(const char *data, string &output) { char *data_path = getenv("OBS_DATA_PATH"); if (data_path != NULL) { if (check_path(data, data_path, output)) return true; } if (check_path(data, OBS_DATA_PATH "/obs-studio/", output)) return true; if (check_path(data, INSTALL_DATA_PATH, output)) return true; return false; } bool InitApplicationBundle() { return true; } string GetDefaultVideoSavePath() { return string(getenv("HOME")); } vector GetPreferredLocales() { setlocale(LC_ALL, ""); vector matched; string messages = setlocale(LC_MESSAGES, NULL); if (!messages.size() || messages == "C" || messages == "POSIX") return {}; if (messages.size() > 2) messages[2] = '-'; for (auto &locale_pair : GetLocaleNames()) { auto &locale = locale_pair.first; if (locale == messages.substr(0, locale.size())) return {locale}; if (locale.substr(0, 2) == messages.substr(0, 2)) matched.push_back(locale); } return matched; } bool IsAlwaysOnTop(QWidget *window) { return (window->windowFlags() & Qt::WindowStaysOnTopHint) != 0; } void SetAlwaysOnTop(QWidget *window, bool enable) { Qt::WindowFlags flags = window->windowFlags(); if (enable) flags |= Qt::WindowStaysOnTopHint; else flags &= ~Qt::WindowStaysOnTopHint; window->setWindowFlags(flags); window->show(); }