You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
176 lines
4.4 KiB
176 lines
4.4 KiB
/* |
|
sys_linux.c - Linux system utils |
|
Copyright (C) 2018 a1batross |
|
|
|
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 3 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. |
|
*/ |
|
|
|
#ifndef _GNU_SOURCE |
|
#define _GNU_SOURCE |
|
#endif |
|
|
|
#include <time.h> |
|
#include <stdlib.h> |
|
#include <fcntl.h> |
|
#include <dlfcn.h> |
|
#include <signal.h> |
|
#include <ucontext.h> |
|
#include <time.h> |
|
#include <unistd.h> |
|
#include <math.h> |
|
#include "platform/platform.h" |
|
#include <sys/types.h> |
|
|
|
#if defined( __GLIBC__ ) |
|
#if ( __GLIBC__ <= 2 ) && ( __GLIBC_MINOR__ <= 30 ) |
|
// Library support was added in glibc 2.30. |
|
// Earlier glibc versions did not provide a wrapper for this system call, |
|
// necessitating the use of syscall(2). |
|
#include <sys/syscall.h> |
|
|
|
static pid_t gettid( void ) |
|
{ |
|
return syscall( SYS_gettid ); |
|
} |
|
#endif // ( __GLIBC__ <= 2 ) && ( __GLIBC_MINOR__ <= 30 ) |
|
|
|
// Glibc misses this macro in POSIX headers (bits/types/sigevent_t.h) |
|
// but it does present in Linux headers (asm-generic/siginfo.h) and musl |
|
#if !defined( sigev_notify_thread_id ) |
|
#define sigev_notify_thread_id _sigev_un._tid |
|
#endif // !defined( sigev_notify_thread_id ) |
|
#endif // defined(__GLIBC__) |
|
|
|
static void *g_hsystemd; |
|
static int (*g_pfn_sd_notify)( int unset_environment, const char *state ); |
|
|
|
qboolean Sys_DebuggerPresent( void ) |
|
{ |
|
char buf[4096]; |
|
ssize_t num_read; |
|
int status_fd; |
|
|
|
status_fd = open( "/proc/self/status", O_RDONLY ); |
|
if ( status_fd == -1 ) |
|
return 0; |
|
|
|
num_read = read( status_fd, buf, sizeof( buf ) ); |
|
close( status_fd ); |
|
|
|
if ( num_read > 0 ) |
|
{ |
|
static const char TracerPid[] = "TracerPid:"; |
|
const byte *tracer_pid; |
|
|
|
buf[num_read] = 0; |
|
tracer_pid = (const byte*)Q_strstr( buf, TracerPid ); |
|
if( !tracer_pid ) |
|
return false; |
|
//printf( "%s\n", tracer_pid ); |
|
while( *tracer_pid < '0' || *tracer_pid > '9' ) |
|
if( *tracer_pid++ == '\n' ) |
|
return false; |
|
//printf( "%s\n", tracer_pid ); |
|
return !!Q_atoi( (const char*)tracer_pid ); |
|
} |
|
|
|
return false; |
|
} |
|
|
|
void Platform_SetStatus( const char *status ) |
|
{ |
|
string notify_str; |
|
|
|
if( !g_pfn_sd_notify ) |
|
return; |
|
|
|
// TODO: report STOPPING=1 |
|
Q_snprintf( notify_str, sizeof( notify_str ), |
|
"READY=1\n" |
|
"WATCHDOG=1\n" |
|
"STATUS=%s\n", status ); |
|
|
|
// Quote: In order to support both service managers that implement this |
|
// scheme and those which do not, it is generally recommended to ignore |
|
// the return value of this call |
|
// a1ba: ok, you asked for no error handling :) |
|
g_pfn_sd_notify( false, notify_str ); |
|
} |
|
|
|
void Linux_Init( void ) |
|
{ |
|
// sd_notify only for dedicated targets, don't waste time on full client |
|
if( !Host_IsDedicated( )) |
|
return; |
|
|
|
if(( g_hsystemd = dlopen( "libsystemd.so.0", RTLD_LAZY )) == NULL ) |
|
return; |
|
|
|
if(( g_pfn_sd_notify = dlsym( g_hsystemd, "sd_notify" )) == NULL ) |
|
{ |
|
dlclose( g_hsystemd ); |
|
g_hsystemd = NULL; |
|
} |
|
|
|
Con_Reportf( "%s: sd_notify found, will report status to systemd\n", __func__ ); |
|
} |
|
|
|
void Linux_Shutdown( void ) |
|
{ |
|
g_pfn_sd_notify = NULL; |
|
|
|
if( g_hsystemd ) |
|
{ |
|
dlclose( g_hsystemd ); |
|
g_hsystemd = NULL; |
|
} |
|
} |
|
|
|
static void Linux_TimerHandler( int sig, siginfo_t *si, void *uc ) |
|
{ |
|
timer_t *tidp = si->si_value.sival_ptr; |
|
int overrun = timer_getoverrun( *tidp ); |
|
Con_Printf( "Frame too long (overrun %d)!\n", overrun ); |
|
} |
|
|
|
#define DEBUG_TIMER_SIGNAL SIGRTMIN |
|
|
|
void Linux_SetTimer( float tm ) |
|
{ |
|
static timer_t timerid; |
|
|
|
if( !timerid && tm ) |
|
{ |
|
struct sigevent sev = { 0 }; |
|
struct sigaction sa = { 0 }; |
|
|
|
sa.sa_flags = SA_SIGINFO; |
|
sa.sa_sigaction = Linux_TimerHandler; |
|
sigaction( DEBUG_TIMER_SIGNAL, &sa, NULL ); |
|
// this path availiable in POSIX, but may signal wrong thread... |
|
// sev.sigev_notify = SIGEV_SIGNAL; |
|
sev.sigev_notify = SIGEV_THREAD_ID; |
|
sev.sigev_notify_thread_id = gettid(); |
|
sev.sigev_signo = DEBUG_TIMER_SIGNAL; |
|
sev.sigev_value.sival_ptr = &timerid; |
|
timer_create( CLOCK_REALTIME, &sev, &timerid ); |
|
} |
|
|
|
if( timerid ) |
|
{ |
|
struct itimerspec its = {0}; |
|
its.it_value.tv_sec = tm; |
|
its.it_value.tv_nsec = 1000000000ULL * fmod( tm, 1.0f ); |
|
its.it_interval.tv_sec = 0; |
|
its.it_interval.tv_nsec = 0; |
|
timer_settime( timerid, 0, &its, NULL ); |
|
} |
|
}
|
|
|