|
|
|
/*
|
|
|
|
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 );
|
|
|
|
}
|
|
|
|
}
|