2b3cb54771
Introduce the OBS_NIX_PLATFORM_WAYLAND enum value, and try to detect it when OBS Studio runs by looking into the platform name.
416 lines
9.1 KiB
C
416 lines
9.1 KiB
C
/******************************************************************************
|
|
Copyright (C) 2013 by Hugh Bailey <obs.jim@gmail.com>
|
|
Copyright (C) 2014 by Zachary Lund <admin@computerquip.com>
|
|
|
|
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 <http://www.gnu.org/licenses/>.
|
|
******************************************************************************/
|
|
|
|
#include "obs-internal.h"
|
|
#include "obs-nix.h"
|
|
#include "obs-nix-platform.h"
|
|
#include "obs-nix-x11.h"
|
|
|
|
#ifdef ENABLE_WAYLAND
|
|
#include "obs-nix-wayland.h"
|
|
#endif
|
|
|
|
#if defined(__FreeBSD__)
|
|
#define _GNU_SOURCE
|
|
#endif
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#if defined(__FreeBSD__) || defined(__OpenBSD__)
|
|
#include <sys/sysctl.h>
|
|
#endif
|
|
#if !defined(__OpenBSD__)
|
|
#include <sys/sysinfo.h>
|
|
#endif
|
|
#include <sys/utsname.h>
|
|
#include <inttypes.h>
|
|
|
|
const char *get_module_extension(void)
|
|
{
|
|
return ".so";
|
|
}
|
|
|
|
#ifdef __LP64__
|
|
#define BIT_STRING "64bit"
|
|
#else
|
|
#define BIT_STRING "32bit"
|
|
#endif
|
|
|
|
static const char *module_bin[] = {"../../obs-plugins/" BIT_STRING,
|
|
OBS_INSTALL_PREFIX
|
|
"/" OBS_PLUGIN_DESTINATION};
|
|
|
|
static const char *module_data[] = {
|
|
OBS_DATA_PATH "/obs-plugins/%module%",
|
|
OBS_INSTALL_DATA_PATH "/obs-plugins/%module%",
|
|
};
|
|
|
|
static const int module_patterns_size =
|
|
sizeof(module_bin) / sizeof(module_bin[0]);
|
|
|
|
static const struct obs_nix_hotkeys_vtable *hotkeys_vtable = NULL;
|
|
|
|
void add_default_module_paths(void)
|
|
{
|
|
for (int i = 0; i < module_patterns_size; i++)
|
|
obs_add_module_path(module_bin[i], module_data[i]);
|
|
}
|
|
|
|
/*
|
|
* /usr/local/share/libobs
|
|
* /usr/share/libobs
|
|
*/
|
|
char *find_libobs_data_file(const char *file)
|
|
{
|
|
struct dstr output;
|
|
dstr_init(&output);
|
|
|
|
if (check_path(file, OBS_DATA_PATH "/libobs/", &output))
|
|
return output.array;
|
|
|
|
if (OBS_INSTALL_PREFIX[0] != 0) {
|
|
if (check_path(file, OBS_INSTALL_DATA_PATH "/libobs/", &output))
|
|
return output.array;
|
|
}
|
|
|
|
dstr_free(&output);
|
|
return NULL;
|
|
}
|
|
|
|
static void log_processor_cores(void)
|
|
{
|
|
blog(LOG_INFO, "Physical Cores: %d, Logical Cores: %d",
|
|
os_get_physical_cores(), os_get_logical_cores());
|
|
}
|
|
|
|
#if defined(__linux__)
|
|
static void log_processor_info(void)
|
|
{
|
|
int physical_id = -1;
|
|
int last_physical_id = -1;
|
|
char *line = NULL;
|
|
size_t linecap = 0;
|
|
|
|
FILE *fp;
|
|
struct dstr proc_name;
|
|
struct dstr proc_speed;
|
|
|
|
fp = fopen("/proc/cpuinfo", "r");
|
|
if (!fp)
|
|
return;
|
|
|
|
dstr_init(&proc_name);
|
|
dstr_init(&proc_speed);
|
|
|
|
while (getline(&line, &linecap, fp) != -1) {
|
|
if (!strncmp(line, "model name", 10)) {
|
|
char *start = strchr(line, ':');
|
|
if (!start || *(++start) == '\0')
|
|
continue;
|
|
|
|
dstr_copy(&proc_name, start);
|
|
dstr_resize(&proc_name, proc_name.len - 1);
|
|
dstr_depad(&proc_name);
|
|
}
|
|
|
|
if (!strncmp(line, "physical id", 11)) {
|
|
char *start = strchr(line, ':');
|
|
if (!start || *(++start) == '\0')
|
|
continue;
|
|
|
|
physical_id = atoi(start);
|
|
}
|
|
|
|
if (!strncmp(line, "cpu MHz", 7)) {
|
|
char *start = strchr(line, ':');
|
|
if (!start || *(++start) == '\0')
|
|
continue;
|
|
|
|
dstr_copy(&proc_speed, start);
|
|
dstr_resize(&proc_speed, proc_speed.len - 1);
|
|
dstr_depad(&proc_speed);
|
|
}
|
|
|
|
if (*line == '\n' && physical_id != last_physical_id) {
|
|
last_physical_id = physical_id;
|
|
blog(LOG_INFO, "CPU Name: %s", proc_name.array);
|
|
blog(LOG_INFO, "CPU Speed: %sMHz", proc_speed.array);
|
|
}
|
|
}
|
|
|
|
fclose(fp);
|
|
dstr_free(&proc_name);
|
|
dstr_free(&proc_speed);
|
|
free(line);
|
|
}
|
|
#elif defined(__FreeBSD__) || defined(__OpenBSD__)
|
|
static void log_processor_speed(void)
|
|
{
|
|
#ifndef __OpenBSD__
|
|
char *line = NULL;
|
|
size_t linecap = 0;
|
|
FILE *fp;
|
|
struct dstr proc_speed;
|
|
|
|
fp = fopen("/var/run/dmesg.boot", "r");
|
|
if (!fp) {
|
|
blog(LOG_INFO, "CPU: Missing /var/run/dmesg.boot !");
|
|
return;
|
|
}
|
|
|
|
dstr_init(&proc_speed);
|
|
|
|
while (getline(&line, &linecap, fp) != -1) {
|
|
if (!strncmp(line, "CPU: ", 5)) {
|
|
char *start = strrchr(line, '(');
|
|
if (!start || *(++start) == '\0')
|
|
continue;
|
|
|
|
size_t len = strcspn(start, "-");
|
|
dstr_ncopy(&proc_speed, start, len);
|
|
}
|
|
}
|
|
|
|
blog(LOG_INFO, "CPU Speed: %sMHz", proc_speed.array);
|
|
|
|
fclose(fp);
|
|
dstr_free(&proc_speed);
|
|
free(line);
|
|
#endif
|
|
}
|
|
|
|
static void log_processor_name(void)
|
|
{
|
|
int mib[2];
|
|
size_t len;
|
|
char *proc;
|
|
|
|
mib[0] = CTL_HW;
|
|
mib[1] = HW_MODEL;
|
|
|
|
sysctl(mib, 2, NULL, &len, NULL, 0);
|
|
proc = bmalloc(len);
|
|
if (!proc)
|
|
return;
|
|
|
|
sysctl(mib, 2, proc, &len, NULL, 0);
|
|
blog(LOG_INFO, "CPU Name: %s", proc);
|
|
|
|
bfree(proc);
|
|
}
|
|
|
|
static void log_processor_info(void)
|
|
{
|
|
log_processor_name();
|
|
log_processor_speed();
|
|
}
|
|
#endif
|
|
|
|
static void log_memory_info(void)
|
|
{
|
|
#if defined(__OpenBSD__)
|
|
int mib[2];
|
|
size_t len;
|
|
int64_t mem;
|
|
|
|
mib[0] = CTL_HW;
|
|
mib[1] = HW_PHYSMEM64;
|
|
len = sizeof(mem);
|
|
|
|
if (sysctl(mib, 2, &mem, &len, NULL, 0) >= 0)
|
|
blog(LOG_INFO, "Physical Memory: %" PRIi64 "MB Total",
|
|
mem / 1024 / 1024);
|
|
#else
|
|
struct sysinfo info;
|
|
if (sysinfo(&info) < 0)
|
|
return;
|
|
|
|
blog(LOG_INFO,
|
|
"Physical Memory: %" PRIu64 "MB Total, %" PRIu64 "MB Free",
|
|
(uint64_t)info.totalram * info.mem_unit / 1024 / 1024,
|
|
((uint64_t)info.freeram + (uint64_t)info.bufferram) *
|
|
info.mem_unit / 1024 / 1024);
|
|
#endif
|
|
}
|
|
|
|
static void log_kernel_version(void)
|
|
{
|
|
struct utsname info;
|
|
if (uname(&info) < 0)
|
|
return;
|
|
|
|
blog(LOG_INFO, "Kernel Version: %s %s", info.sysname, info.release);
|
|
}
|
|
|
|
#if defined(__linux__)
|
|
static void log_distribution_info(void)
|
|
{
|
|
FILE *fp;
|
|
char *line = NULL;
|
|
size_t linecap = 0;
|
|
struct dstr distro;
|
|
struct dstr version;
|
|
|
|
fp = fopen("/etc/os-release", "r");
|
|
if (!fp) {
|
|
blog(LOG_INFO, "Distribution: Missing /etc/os-release !");
|
|
return;
|
|
}
|
|
|
|
dstr_init_copy(&distro, "Unknown");
|
|
dstr_init_copy(&version, "Unknown");
|
|
|
|
while (getline(&line, &linecap, fp) != -1) {
|
|
if (!strncmp(line, "NAME", 4)) {
|
|
char *start = strchr(line, '=');
|
|
if (!start || *(++start) == '\0')
|
|
continue;
|
|
dstr_copy(&distro, start);
|
|
dstr_resize(&distro, distro.len - 1);
|
|
}
|
|
|
|
if (!strncmp(line, "VERSION_ID", 10)) {
|
|
char *start = strchr(line, '=');
|
|
if (!start || *(++start) == '\0')
|
|
continue;
|
|
dstr_copy(&version, start);
|
|
dstr_resize(&version, version.len - 1);
|
|
}
|
|
}
|
|
|
|
blog(LOG_INFO, "Distribution: %s %s", distro.array, version.array);
|
|
|
|
fclose(fp);
|
|
dstr_free(&version);
|
|
dstr_free(&distro);
|
|
free(line);
|
|
}
|
|
|
|
static void log_desktop_session_info(void)
|
|
{
|
|
char *session_ptr = getenv("XDG_SESSION_TYPE");
|
|
if (session_ptr) {
|
|
blog(LOG_INFO, "Session Type: %s", session_ptr);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void log_system_info(void)
|
|
{
|
|
#if defined(__linux__) || defined(__FreeBSD__)
|
|
log_processor_info();
|
|
#endif
|
|
log_processor_cores();
|
|
log_memory_info();
|
|
log_kernel_version();
|
|
#if defined(__linux__)
|
|
log_distribution_info();
|
|
log_desktop_session_info();
|
|
#endif
|
|
switch (obs_get_nix_platform()) {
|
|
case OBS_NIX_PLATFORM_X11_GLX:
|
|
case OBS_NIX_PLATFORM_X11_EGL:
|
|
obs_nix_x11_log_info();
|
|
break;
|
|
#ifdef ENABLE_WAYLAND
|
|
case OBS_NIX_PLATFORM_WAYLAND:
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
bool obs_hotkeys_platform_init(struct obs_core_hotkeys *hotkeys)
|
|
{
|
|
switch (obs_get_nix_platform()) {
|
|
case OBS_NIX_PLATFORM_X11_GLX:
|
|
case OBS_NIX_PLATFORM_X11_EGL:
|
|
hotkeys_vtable = obs_nix_x11_get_hotkeys_vtable();
|
|
break;
|
|
#ifdef ENABLE_WAYLAND
|
|
case OBS_NIX_PLATFORM_WAYLAND:
|
|
hotkeys_vtable = obs_nix_wayland_get_hotkeys_vtable();
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
return hotkeys_vtable->init(hotkeys);
|
|
}
|
|
|
|
void obs_hotkeys_platform_free(struct obs_core_hotkeys *hotkeys)
|
|
{
|
|
hotkeys_vtable->free(hotkeys);
|
|
hotkeys_vtable = NULL;
|
|
}
|
|
|
|
bool obs_hotkeys_platform_is_pressed(obs_hotkeys_platform_t *context,
|
|
obs_key_t key)
|
|
{
|
|
return hotkeys_vtable->is_pressed(context, key);
|
|
}
|
|
|
|
void obs_key_to_str(obs_key_t key, struct dstr *dstr)
|
|
{
|
|
return hotkeys_vtable->key_to_str(key, dstr);
|
|
}
|
|
|
|
obs_key_t obs_key_from_virtual_key(int sym)
|
|
{
|
|
return hotkeys_vtable->key_from_virtual_key(sym);
|
|
}
|
|
|
|
int obs_key_to_virtual_key(obs_key_t key)
|
|
{
|
|
return hotkeys_vtable->key_to_virtual_key(key);
|
|
}
|
|
|
|
static inline void add_combo_key(obs_key_t key, struct dstr *str)
|
|
{
|
|
struct dstr key_str = {0};
|
|
|
|
obs_key_to_str(key, &key_str);
|
|
|
|
if (!dstr_is_empty(&key_str)) {
|
|
if (!dstr_is_empty(str)) {
|
|
dstr_cat(str, " + ");
|
|
}
|
|
dstr_cat_dstr(str, &key_str);
|
|
}
|
|
|
|
dstr_free(&key_str);
|
|
}
|
|
|
|
void obs_key_combination_to_str(obs_key_combination_t combination,
|
|
struct dstr *str)
|
|
{
|
|
if ((combination.modifiers & INTERACT_CONTROL_KEY) != 0) {
|
|
add_combo_key(OBS_KEY_CONTROL, str);
|
|
}
|
|
if ((combination.modifiers & INTERACT_COMMAND_KEY) != 0) {
|
|
add_combo_key(OBS_KEY_META, str);
|
|
}
|
|
if ((combination.modifiers & INTERACT_ALT_KEY) != 0) {
|
|
add_combo_key(OBS_KEY_ALT, str);
|
|
}
|
|
if ((combination.modifiers & INTERACT_SHIFT_KEY) != 0) {
|
|
add_combo_key(OBS_KEY_SHIFT, str);
|
|
}
|
|
if (combination.key != OBS_KEY_NONE) {
|
|
add_combo_key(combination.key, str);
|
|
}
|
|
}
|