engine: platform: sdl: fix possible buffer overrun in SDL port audio code

This commit is contained in:
Alibek Omarov 2023-05-24 03:16:54 +03:00
parent 275cd73ade
commit 356f78ee81

View File

@ -49,20 +49,26 @@ static SDL_AudioDeviceID in_dev = 0;
static SDL_AudioFormat sdl_format; static SDL_AudioFormat sdl_format;
static char sdl_backend_name[32]; static char sdl_backend_name[32];
//static qboolean snd_firsttime = true;
//static qboolean primary_format_set;
void SDL_SoundCallback( void *userdata, Uint8 *stream, int len ) void SDL_SoundCallback( void *userdata, Uint8 *stream, int len )
{ {
int size = dma.samples << 1; const int size = dma.samples << 1;
int pos = dma.samplepos << 1; int pos;
int wrapped = pos + len - size; int wrapped;
#if ! SDL_VERSION_ATLEAST( 2, 0, 0 ) #if ! SDL_VERSION_ATLEAST( 2, 0, 0 )
if( !dma.buffer ) if( !dma.buffer )
{
memset( stream, 0, len );
return; return;
}
#endif #endif
pos = dma.samplepos << 1;
if( pos >= size )
pos = dma.samplepos = 0;
wrapped = pos + len - size;
if( wrapped < 0 ) if( wrapped < 0 )
{ {
memcpy( stream, dma.buffer + pos, len ); memcpy( stream, dma.buffer + pos, len );
@ -71,10 +77,14 @@ void SDL_SoundCallback( void *userdata, Uint8 *stream, int len )
else else
{ {
int remaining = size - pos; int remaining = size - pos;
memcpy( stream, dma.buffer + pos, remaining ); memcpy( stream, dma.buffer + pos, remaining );
memcpy( stream + remaining, dma.buffer, wrapped ); memcpy( stream + remaining, dma.buffer, wrapped );
dma.samplepos = wrapped >> 1; dma.samplepos = wrapped >> 1;
} }
if( dma.samplepos >= size )
dma.samplepos = 0;
} }
/* /*
@ -90,19 +100,17 @@ qboolean SNDDMA_Init( void )
SDL_AudioSpec desired, obtained; SDL_AudioSpec desired, obtained;
int samplecount; int samplecount;
// even if we don't have PA
// we still can safely set env variables
SDL_setenv( "PULSE_PROP_application.name", GI->title, 1 );
SDL_setenv( "PULSE_PROP_media.role", "game", 1 );
if( SDL_Init( SDL_INIT_AUDIO )) if( SDL_Init( SDL_INIT_AUDIO ))
{ {
Con_Reportf( S_ERROR "Audio: SDL: %s \n", SDL_GetError( ) ); Con_Reportf( S_ERROR "Audio: SDL: %s \n", SDL_GetError( ) );
return false; return false;
} }
// even if we don't have PA
// we still can safely set env variables
#if XASH_POSIX
setenv( "PULSE_PROP_application.name", GI->title, 1 );
setenv( "PULSE_PROP_media.role", "game", 1 );
#endif // XASH_POSIX
memset( &desired, 0, sizeof( desired ) ); memset( &desired, 0, sizeof( desired ) );
desired.freq = SOUND_DMA_SPEED; desired.freq = SOUND_DMA_SPEED;
desired.format = AUDIO_S16LSB; desired.format = AUDIO_S16LSB;
@ -166,7 +174,7 @@ Makes sure dma.buffer is valid
*/ */
void SNDDMA_BeginPainting( void ) void SNDDMA_BeginPainting( void )
{ {
// SDL_LockAudioDevice( sdl_dev ); SDL_LockAudioDevice( sdl_dev );
} }
/* /*
@ -179,7 +187,7 @@ Also unlocks the dsound buffer
*/ */
void SNDDMA_Submit( void ) void SNDDMA_Submit( void )
{ {
// SDL_UnlockAudioDevice( sdl_dev ); SDL_UnlockAudioDevice( sdl_dev );
} }
/* /*