/* 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 #include #include #include #include #include #include #include #include #include "platform/platform.h" #include #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 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 Platform_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 ); } }