2015-06-21 15:33:46 +00:00
/**
* Original code : automated SDL audio test written by Edgar Simo " bobbens "
* New / updated tests : aschiffler at ferzkopp dot net
*/
/* quiet windows compiler warnings */
2022-10-05 23:09:32 +00:00
# if defined(_MSC_VER) && !defined(_CRT_SECURE_NO_WARNINGS)
2022-11-30 20:51:59 +00:00
# define _CRT_SECURE_NO_WARNINGS
2022-10-05 23:09:32 +00:00
# endif
2015-06-21 15:33:46 +00:00
2023-03-09 09:34:51 +00:00
# include <math.h>
2015-06-21 15:33:46 +00:00
# include <stdio.h>
# include <string.h>
# include "SDL.h"
# include "SDL_test.h"
/* ================= Test Case Implementation ================== */
/* Fixture */
2022-11-30 20:51:59 +00:00
void _audioSetUp ( void * arg )
2015-06-21 15:33:46 +00:00
{
/* Start SDL audio subsystem */
2022-11-30 20:51:59 +00:00
int ret = SDL_InitSubSystem ( SDL_INIT_AUDIO ) ;
SDLTest_AssertPass ( " Call to SDL_InitSubSystem(SDL_INIT_AUDIO) " ) ;
SDLTest_AssertCheck ( ret = = 0 , " Check result from SDL_InitSubSystem(SDL_INIT_AUDIO) " ) ;
2015-06-21 15:33:46 +00:00
if ( ret ! = 0 ) {
2022-11-30 20:51:59 +00:00
SDLTest_LogError ( " %s " , SDL_GetError ( ) ) ;
}
2015-06-21 15:33:46 +00:00
}
2022-11-30 20:51:59 +00:00
void _audioTearDown ( void * arg )
2015-06-21 15:33:46 +00:00
{
/* Remove a possibly created file from SDL disk writer audio driver; ignore errors */
2022-12-01 21:07:03 +00:00
( void ) remove ( " sdlaudio.raw " ) ;
2015-06-21 15:33:46 +00:00
SDLTest_AssertPass ( " Cleanup of test files completed " ) ;
}
/* Global counter for callback invocation */
int _audio_testCallbackCounter ;
/* Global accumulator for total callback length */
int _audio_testCallbackLength ;
/* Test callback function */
2017-08-14 04:06:52 +00:00
void SDLCALL _audio_testCallback ( void * userdata , Uint8 * stream , int len )
2015-06-21 15:33:46 +00:00
{
2022-11-30 20:51:59 +00:00
/* track that callback was called */
_audio_testCallbackCounter + + ;
_audio_testCallbackLength + = len ;
2015-06-21 15:33:46 +00:00
}
2023-10-15 16:15:32 +00:00
# if defined(__linux__)
/* Linux builds can include many audio drivers, but some are very
* obscure and typically unsupported on modern systems . They will
* be skipped in tests that run against all included drivers , as
* they are basically guaranteed to fail .
*/
static SDL_bool DriverIsProblematic ( const char * driver )
{
static const char * driverList [ ] = {
/* Omnipresent in Linux builds, but deprecated since 2002,
* very rarely used on Linux nowadays , and is almost certainly
* guaranteed to fail .
*/
" dsp " ,
/* OpenBSD sound API. Can be used on Linux, but very rare. */
" sndio " ,
/* Always fails on initialization and/or opening a device.
* Does anyone or anything actually use this ?
*/
" nas "
} ;
int i ;
for ( i = 0 ; i < SDL_arraysize ( driverList ) ; + + i ) {
if ( SDL_strcmp ( driver , driverList [ i ] ) = = 0 ) {
return SDL_TRUE ;
}
}
return SDL_FALSE ;
}
# endif
2015-06-21 15:33:46 +00:00
/* Test case functions */
/**
* \ brief Stop and restart audio subsystem
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_QuitSubSystem
* \ sa https : //wiki.libsdl.org/SDL_InitSubSystem
2015-06-21 15:33:46 +00:00
*/
int audio_quitInitAudioSubSystem ( )
{
/* Stop SDL audio subsystem */
2022-11-30 20:51:59 +00:00
SDL_QuitSubSystem ( SDL_INIT_AUDIO ) ;
SDLTest_AssertPass ( " Call to SDL_QuitSubSystem(SDL_INIT_AUDIO) " ) ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
/* Restart audio again */
_audioSetUp ( NULL ) ;
2015-06-21 15:33:46 +00:00
return TEST_COMPLETED ;
}
/**
* \ brief Start and stop audio directly
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_InitAudio
* \ sa https : //wiki.libsdl.org/SDL_QuitAudio
2015-06-21 15:33:46 +00:00
*/
int audio_initQuitAudio ( )
{
2022-11-30 20:51:59 +00:00
int result ;
2015-06-21 15:33:46 +00:00
int i , iMax ;
2022-11-30 20:51:59 +00:00
const char * audioDriver ;
2024-01-08 21:38:29 +00:00
const char * hint = SDL_GetHint ( SDL_HINT_AUDIODRIVER ) ;
2015-06-21 15:33:46 +00:00
/* Stop SDL audio subsystem */
2022-11-30 20:51:59 +00:00
SDL_QuitSubSystem ( SDL_INIT_AUDIO ) ;
SDLTest_AssertPass ( " Call to SDL_QuitSubSystem(SDL_INIT_AUDIO) " ) ;
2023-10-15 16:15:32 +00:00
/* Was a specific driver requested? */
audioDriver = SDL_GetHint ( SDL_HINT_AUDIODRIVER ) ;
if ( audioDriver = = NULL ) {
/* Loop over all available audio drivers */
iMax = SDL_GetNumAudioDrivers ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDrivers() " ) ;
SDLTest_AssertCheck ( iMax > 0 , " Validate number of audio drivers; expected: >0 got: %d " , iMax ) ;
} else {
/* A specific driver was requested for testing */
iMax = 1 ;
}
2022-11-30 20:51:59 +00:00
for ( i = 0 ; i < iMax ; i + + ) {
2023-10-15 16:15:32 +00:00
if ( audioDriver = = NULL ) {
audioDriver = SDL_GetAudioDriver ( i ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDriver(%d) " , i ) ;
SDLTest_Assert ( audioDriver ! = NULL , " Audio driver name is not NULL " ) ;
SDLTest_AssertCheck ( audioDriver [ 0 ] ! = ' \0 ' , " Audio driver name is not empty; got: %s " , audioDriver ) ; /* NOLINT(clang-analyzer-core.NullDereference): Checked for NULL above */
# if defined(__linux__)
if ( DriverIsProblematic ( audioDriver ) ) {
SDLTest_Log ( " Audio driver '%s' flagged as problematic: skipping init/quit test (set SDL_AUDIODRIVER=%s to force) " , audioDriver , audioDriver ) ;
audioDriver = NULL ;
continue ;
}
# endif
}
2022-11-30 20:51:59 +00:00
2024-01-08 21:38:29 +00:00
if ( hint & & SDL_strcmp ( audioDriver , hint ) ! = 0 ) {
continue ;
}
2022-11-30 20:51:59 +00:00
/* Call Init */
result = SDL_AudioInit ( audioDriver ) ;
SDLTest_AssertPass ( " Call to SDL_AudioInit('%s') " , audioDriver ) ;
SDLTest_AssertCheck ( result = = 0 , " Validate result value; expected: 0 got: %d " , result ) ;
/* Call Quit */
SDL_AudioQuit ( ) ;
SDLTest_AssertPass ( " Call to SDL_AudioQuit() " ) ;
2023-10-15 16:15:32 +00:00
audioDriver = NULL ;
2015-06-21 15:33:46 +00:00
}
/* NULL driver specification */
audioDriver = NULL ;
/* Call Init */
result = SDL_AudioInit ( audioDriver ) ;
SDLTest_AssertPass ( " Call to SDL_AudioInit(NULL) " ) ;
SDLTest_AssertCheck ( result = = 0 , " Validate result value; expected: 0 got: %d " , result ) ;
/* Call Quit */
SDL_AudioQuit ( ) ;
SDLTest_AssertPass ( " Call to SDL_AudioQuit() " ) ;
2022-11-30 20:51:59 +00:00
/* Restart audio again */
_audioSetUp ( NULL ) ;
2015-06-21 15:33:46 +00:00
return TEST_COMPLETED ;
}
/**
* \ brief Start , open , close and stop audio
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_InitAudio
* \ sa https : //wiki.libsdl.org/SDL_OpenAudio
* \ sa https : //wiki.libsdl.org/SDL_CloseAudio
* \ sa https : //wiki.libsdl.org/SDL_QuitAudio
2015-06-21 15:33:46 +00:00
*/
int audio_initOpenCloseQuitAudio ( )
{
int result , expectedResult ;
int i , iMax , j , k ;
2022-11-30 20:51:59 +00:00
const char * audioDriver ;
2015-06-21 15:33:46 +00:00
SDL_AudioSpec desired ;
2024-01-08 21:38:29 +00:00
const char * hint = SDL_GetHint ( SDL_HINT_AUDIODRIVER ) ;
2015-06-21 15:33:46 +00:00
/* Stop SDL audio subsystem */
2022-11-30 20:51:59 +00:00
SDL_QuitSubSystem ( SDL_INIT_AUDIO ) ;
SDLTest_AssertPass ( " Call to SDL_QuitSubSystem(SDL_INIT_AUDIO) " ) ;
2023-10-15 16:15:32 +00:00
/* Was a specific driver requested? */
audioDriver = SDL_GetHint ( SDL_HINT_AUDIODRIVER ) ;
if ( audioDriver = = NULL ) {
/* Loop over all available audio drivers */
iMax = SDL_GetNumAudioDrivers ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDrivers() " ) ;
SDLTest_AssertCheck ( iMax > 0 , " Validate number of audio drivers; expected: >0 got: %d " , iMax ) ;
} else {
/* A specific driver was requested for testing */
iMax = 1 ;
}
2022-11-30 20:51:59 +00:00
for ( i = 0 ; i < iMax ; i + + ) {
2023-10-15 16:15:32 +00:00
if ( audioDriver = = NULL ) {
audioDriver = SDL_GetAudioDriver ( i ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDriver(%d) " , i ) ;
SDLTest_Assert ( audioDriver ! = NULL , " Audio driver name is not NULL " ) ;
SDLTest_AssertCheck ( audioDriver [ 0 ] ! = ' \0 ' , " Audio driver name is not empty; got: %s " , audioDriver ) ; /* NOLINT(clang-analyzer-core.NullDereference): Checked for NULL above */
# if defined(__linux__)
if ( DriverIsProblematic ( audioDriver ) ) {
SDLTest_Log ( " Audio driver '%s' flagged as problematic: skipping device open/close test (set SDL_AUDIODRIVER=%s to force) " , audioDriver , audioDriver ) ;
audioDriver = NULL ;
continue ;
}
# endif
}
2022-11-30 20:51:59 +00:00
2024-01-08 21:38:29 +00:00
if ( hint & & SDL_strcmp ( audioDriver , hint ) ! = 0 ) {
continue ;
}
2022-11-30 20:51:59 +00:00
/* Change specs */
for ( j = 0 ; j < 2 ; j + + ) {
/* Call Init */
result = SDL_AudioInit ( audioDriver ) ;
SDLTest_AssertPass ( " Call to SDL_AudioInit('%s') " , audioDriver ) ;
SDLTest_AssertCheck ( result = = 0 , " Validate result value; expected: 0 got: %d " , result ) ;
2023-10-15 16:15:32 +00:00
/* Check for output devices */
result = SDL_GetNumAudioDevices ( SDL_FALSE ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(SDL_FALSE) " ) ;
SDLTest_AssertCheck ( result > = 0 , " Validate result value; expected: >=0 got: %d " , result ) ;
if ( result < = 0 ) {
SDLTest_Log ( " No output devices for '%s': skipping device open/close test " , audioDriver ) ;
SDL_AudioQuit ( ) ;
SDLTest_AssertPass ( " Call to SDL_AudioQuit() " ) ;
break ;
}
2022-11-30 20:51:59 +00:00
/* Set spec */
SDL_memset ( & desired , 0 , sizeof ( desired ) ) ;
switch ( j ) {
case 0 :
/* Set standard desired spec */
desired . freq = 22050 ;
desired . format = AUDIO_S16SYS ;
desired . channels = 2 ;
desired . samples = 4096 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
2023-03-28 17:24:33 +00:00
break ;
2022-11-30 20:51:59 +00:00
case 1 :
/* Set custom desired spec */
desired . freq = 48000 ;
desired . format = AUDIO_F32SYS ;
desired . channels = 2 ;
desired . samples = 2048 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
break ;
2015-06-21 15:33:46 +00:00
}
/* Call Open (maybe multiple times) */
2022-11-30 20:51:59 +00:00
for ( k = 0 ; k < = j ; k + + ) {
2015-06-21 15:33:46 +00:00
result = SDL_OpenAudio ( & desired , NULL ) ;
2022-11-30 20:51:59 +00:00
SDLTest_AssertPass ( " Call to SDL_OpenAudio(desired_spec_%d, NULL), call %d " , j , k + 1 ) ;
expectedResult = ( k = = 0 ) ? 0 : - 1 ;
2015-06-21 15:33:46 +00:00
SDLTest_AssertCheck ( result = = expectedResult , " Verify return value; expected: %d, got: %d " , expectedResult , result ) ;
}
/* Call Close (maybe multiple times) */
2022-11-30 20:51:59 +00:00
for ( k = 0 ; k < = j ; k + + ) {
2015-06-21 15:33:46 +00:00
SDL_CloseAudio ( ) ;
2022-11-30 20:51:59 +00:00
SDLTest_AssertPass ( " Call to SDL_CloseAudio(), call %d " , k + 1 ) ;
2015-06-21 15:33:46 +00:00
}
/* Call Quit (maybe multiple times) */
2022-11-30 20:51:59 +00:00
for ( k = 0 ; k < = j ; k + + ) {
2015-06-21 15:33:46 +00:00
SDL_AudioQuit ( ) ;
2022-11-30 20:51:59 +00:00
SDLTest_AssertPass ( " Call to SDL_AudioQuit(), call %d " , k + 1 ) ;
2015-06-21 15:33:46 +00:00
}
} /* spec loop */
2023-10-15 16:15:32 +00:00
audioDriver = NULL ;
} /* driver loop */
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
/* Restart audio again */
_audioSetUp ( NULL ) ;
2015-06-21 15:33:46 +00:00
return TEST_COMPLETED ;
}
/**
* \ brief Pause and unpause audio
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_PauseAudio
2015-06-21 15:33:46 +00:00
*/
int audio_pauseUnpauseAudio ( )
{
int result ;
int i , iMax , j , k , l ;
int totalDelay ;
int pause_on ;
int originalCounter ;
2022-11-30 20:51:59 +00:00
const char * audioDriver ;
2015-06-21 15:33:46 +00:00
SDL_AudioSpec desired ;
2024-01-08 21:38:29 +00:00
const char * hint = SDL_GetHint ( SDL_HINT_AUDIODRIVER ) ;
2015-06-21 15:33:46 +00:00
/* Stop SDL audio subsystem */
2022-11-30 20:51:59 +00:00
SDL_QuitSubSystem ( SDL_INIT_AUDIO ) ;
SDLTest_AssertPass ( " Call to SDL_QuitSubSystem(SDL_INIT_AUDIO) " ) ;
2023-10-15 16:15:32 +00:00
/* Was a specific driver requested? */
audioDriver = SDL_GetHint ( SDL_HINT_AUDIODRIVER ) ;
if ( audioDriver = = NULL ) {
/* Loop over all available audio drivers */
iMax = SDL_GetNumAudioDrivers ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDrivers() " ) ;
SDLTest_AssertCheck ( iMax > 0 , " Validate number of audio drivers; expected: >0 got: %d " , iMax ) ;
} else {
/* A specific driver was requested for testing */
iMax = 1 ;
}
2022-11-30 20:51:59 +00:00
for ( i = 0 ; i < iMax ; i + + ) {
2023-10-15 16:15:32 +00:00
if ( audioDriver = = NULL ) {
audioDriver = SDL_GetAudioDriver ( i ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDriver(%d) " , i ) ;
SDLTest_Assert ( audioDriver ! = NULL , " Audio driver name is not NULL " ) ;
SDLTest_AssertCheck ( audioDriver [ 0 ] ! = ' \0 ' , " Audio driver name is not empty; got: %s " , audioDriver ) ; /* NOLINT(clang-analyzer-core.NullDereference): Checked for NULL above */
# if defined(__linux__)
if ( DriverIsProblematic ( audioDriver ) ) {
SDLTest_Log ( " Audio driver '%s' flagged as problematic: skipping pause/unpause test (set SDL_AUDIODRIVER=%s to force) " , audioDriver , audioDriver ) ;
audioDriver = NULL ;
continue ;
}
# endif
}
2022-11-30 20:51:59 +00:00
2024-01-08 21:38:29 +00:00
if ( hint & & SDL_strcmp ( audioDriver , hint ) ! = 0 ) {
continue ;
}
2022-11-30 20:51:59 +00:00
/* Change specs */
for ( j = 0 ; j < 2 ; j + + ) {
/* Call Init */
result = SDL_AudioInit ( audioDriver ) ;
SDLTest_AssertPass ( " Call to SDL_AudioInit('%s') " , audioDriver ) ;
SDLTest_AssertCheck ( result = = 0 , " Validate result value; expected: 0 got: %d " , result ) ;
2023-10-15 16:15:32 +00:00
result = SDL_GetNumAudioDevices ( SDL_FALSE ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(SDL_FALSE) " ) ;
SDLTest_AssertCheck ( result > = 0 , " Validate result value; expected: >=0 got: %d " , result ) ;
if ( result < = 0 ) {
SDLTest_Log ( " No output devices for '%s': skipping pause/unpause test " , audioDriver ) ;
SDL_AudioQuit ( ) ;
SDLTest_AssertPass ( " Call to SDL_AudioQuit() " ) ;
break ;
}
2022-11-30 20:51:59 +00:00
/* Set spec */
SDL_memset ( & desired , 0 , sizeof ( desired ) ) ;
switch ( j ) {
case 0 :
/* Set standard desired spec */
desired . freq = 22050 ;
desired . format = AUDIO_S16SYS ;
desired . channels = 2 ;
desired . samples = 4096 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
2023-03-18 19:34:56 +00:00
break ;
2022-11-30 20:51:59 +00:00
case 1 :
/* Set custom desired spec */
desired . freq = 48000 ;
desired . format = AUDIO_F32SYS ;
desired . channels = 2 ;
desired . samples = 2048 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
break ;
2015-06-21 15:33:46 +00:00
}
/* Call Open */
result = SDL_OpenAudio ( & desired , NULL ) ;
SDLTest_AssertPass ( " Call to SDL_OpenAudio(desired_spec_%d, NULL) " , j ) ;
SDLTest_AssertCheck ( result = = 0 , " Verify return value; expected: 0 got: %d " , result ) ;
/* Start and stop audio multiple times */
2022-11-30 20:51:59 +00:00
for ( l = 0 ; l < 3 ; l + + ) {
SDLTest_Log ( " Pause/Unpause iteration: %d " , l + 1 ) ;
2015-06-21 15:33:46 +00:00
/* Reset callback counters */
_audio_testCallbackCounter = 0 ;
_audio_testCallbackLength = 0 ;
/* Un-pause audio to start playing (maybe multiple times) */
pause_on = 0 ;
2022-11-30 20:51:59 +00:00
for ( k = 0 ; k < = j ; k + + ) {
2015-06-21 15:33:46 +00:00
SDL_PauseAudio ( pause_on ) ;
2022-11-30 20:51:59 +00:00
SDLTest_AssertPass ( " Call to SDL_PauseAudio(%d), call %d " , pause_on , k + 1 ) ;
2015-06-21 15:33:46 +00:00
}
2022-11-30 20:51:59 +00:00
2015-06-21 15:33:46 +00:00
/* Wait for callback */
totalDelay = 0 ;
do {
SDL_Delay ( 10 ) ;
totalDelay + = 10 ;
2022-11-30 20:51:59 +00:00
} while ( _audio_testCallbackCounter = = 0 & & totalDelay < 1000 ) ;
2015-06-21 15:33:46 +00:00
SDLTest_AssertCheck ( _audio_testCallbackCounter > 0 , " Verify callback counter; expected: >0 got: %d " , _audio_testCallbackCounter ) ;
SDLTest_AssertCheck ( _audio_testCallbackLength > 0 , " Verify callback length; expected: >0 got: %d " , _audio_testCallbackLength ) ;
/* Pause audio to stop playing (maybe multiple times) */
2022-11-30 20:51:59 +00:00
for ( k = 0 ; k < = j ; k + + ) {
pause_on = ( k = = 0 ) ? 1 : SDLTest_RandomIntegerInRange ( 99 , 9999 ) ;
2015-06-21 15:33:46 +00:00
SDL_PauseAudio ( pause_on ) ;
2022-11-30 20:51:59 +00:00
SDLTest_AssertPass ( " Call to SDL_PauseAudio(%d), call %d " , pause_on , k + 1 ) ;
2015-06-21 15:33:46 +00:00
}
2022-11-30 20:51:59 +00:00
2015-06-21 15:33:46 +00:00
/* Ensure callback is not called again */
originalCounter = _audio_testCallbackCounter ;
SDL_Delay ( totalDelay + 10 ) ;
SDLTest_AssertCheck ( originalCounter = = _audio_testCallbackCounter , " Verify callback counter; expected: %d, got: %d " , originalCounter , _audio_testCallbackCounter ) ;
}
/* Call Close */
SDL_CloseAudio ( ) ;
SDLTest_AssertPass ( " Call to SDL_CloseAudio() " ) ;
/* Call Quit */
SDL_AudioQuit ( ) ;
SDLTest_AssertPass ( " Call to SDL_AudioQuit() " ) ;
} /* spec loop */
2023-10-15 16:15:32 +00:00
audioDriver = NULL ;
} /* driver loop */
2015-06-21 15:33:46 +00:00
/* Restart audio again */
_audioSetUp ( NULL ) ;
return TEST_COMPLETED ;
}
/**
* \ brief Enumerate and name available audio devices ( output and capture ) .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_GetNumAudioDevices
* \ sa https : //wiki.libsdl.org/SDL_GetAudioDeviceName
2015-06-21 15:33:46 +00:00
*/
int audio_enumerateAndNameAudioDevices ( )
{
2022-11-30 20:51:59 +00:00
int t , tt ;
int i , n , nn ;
const char * name , * nameAgain ;
/* Iterate over types: t=0 output device, t=1 input/capture device */
for ( t = 0 ; t < 2 ; t + + ) {
/* Get number of devices. */
n = SDL_GetNumAudioDevices ( t ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(%i) " , t ) ;
SDLTest_Log ( " Number of %s devices < 0, reported as %i " , ( t ) ? " capture " : " output " , n ) ;
SDLTest_AssertCheck ( n > = 0 , " Validate result is >= 0, got: %i " , n ) ;
/* Variation of non-zero type */
if ( t = = 1 ) {
tt = t + SDLTest_RandomIntegerInRange ( 1 , 10 ) ;
nn = SDL_GetNumAudioDevices ( tt ) ;
SDLTest_AssertCheck ( n = = nn , " Verify result from SDL_GetNumAudioDevices(%i), expected same number of audio devices %i, got %i " , tt , n , nn ) ;
nn = SDL_GetNumAudioDevices ( - tt ) ;
SDLTest_AssertCheck ( n = = nn , " Verify result from SDL_GetNumAudioDevices(%i), expected same number of audio devices %i, got %i " , - tt , n , nn ) ;
}
/* List devices. */
if ( n > 0 ) {
for ( i = 0 ; i < n ; i + + ) {
name = SDL_GetAudioDeviceName ( i , t ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDeviceName(%i, %i) " , i , t ) ;
SDLTest_AssertCheck ( name ! = NULL , " Verify result from SDL_GetAudioDeviceName(%i, %i) is not NULL " , i , t ) ;
if ( name ! = NULL ) {
SDLTest_AssertCheck ( name [ 0 ] ! = ' \0 ' , " verify result from SDL_GetAudioDeviceName(%i, %i) is not empty, got: '%s' " , i , t , name ) ;
if ( t = = 1 ) {
/* Also try non-zero type */
tt = t + SDLTest_RandomIntegerInRange ( 1 , 10 ) ;
nameAgain = SDL_GetAudioDeviceName ( i , tt ) ;
SDLTest_AssertCheck ( nameAgain ! = NULL , " Verify result from SDL_GetAudioDeviceName(%i, %i) is not NULL " , i , tt ) ;
if ( nameAgain ! = NULL ) {
SDLTest_AssertCheck ( nameAgain [ 0 ] ! = ' \0 ' , " Verify result from SDL_GetAudioDeviceName(%i, %i) is not empty, got: '%s' " , i , tt , nameAgain ) ;
SDLTest_AssertCheck ( SDL_strcmp ( name , nameAgain ) = = 0 ,
" Verify SDL_GetAudioDeviceName(%i, %i) and SDL_GetAudioDeviceName(%i %i) return the same string " ,
i , t , i , tt ) ;
}
}
}
2015-06-21 15:33:46 +00:00
}
2022-11-30 20:51:59 +00:00
}
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/**
* \ brief Negative tests around enumeration and naming of audio devices .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_GetNumAudioDevices
* \ sa https : //wiki.libsdl.org/SDL_GetAudioDeviceName
2015-06-21 15:33:46 +00:00
*/
int audio_enumerateAndNameAudioDevicesNegativeTests ( )
{
2022-11-30 20:51:59 +00:00
int t ;
int i , j , no , nc ;
const char * name ;
/* Get number of devices. */
no = SDL_GetNumAudioDevices ( 0 ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(0) " ) ;
nc = SDL_GetNumAudioDevices ( 1 ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(1) " ) ;
/* Invalid device index when getting name */
for ( t = 0 ; t < 2 ; t + + ) {
/* Negative device index */
i = SDLTest_RandomIntegerInRange ( - 10 , - 1 ) ;
name = SDL_GetAudioDeviceName ( i , t ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDeviceName(%i, %i) " , i , t ) ;
SDLTest_AssertCheck ( name = = NULL , " Check SDL_GetAudioDeviceName(%i, %i) result NULL, expected NULL, got: %s " , i , t , ( name = = NULL ) ? " NULL " : name ) ;
/* Device index past range */
for ( j = 0 ; j < 3 ; j + + ) {
i = ( t ) ? nc + j : no + j ;
name = SDL_GetAudioDeviceName ( i , t ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDeviceName(%i, %i) " , i , t ) ;
SDLTest_AssertCheck ( name = = NULL , " Check SDL_GetAudioDeviceName(%i, %i) result, expected: NULL, got: %s " , i , t , ( name = = NULL ) ? " NULL " : name ) ;
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
/* Capture index past capture range but within output range */
if ( ( no > 0 ) & & ( no > nc ) & & ( t = = 1 ) ) {
i = no - 1 ;
name = SDL_GetAudioDeviceName ( i , t ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDeviceName(%i, %i) " , i , t ) ;
SDLTest_AssertCheck ( name = = NULL , " Check SDL_GetAudioDeviceName(%i, %i) result, expected: NULL, got: %s " , i , t , ( name = = NULL ) ? " NULL " : name ) ;
}
}
return TEST_COMPLETED ;
}
2015-06-21 15:33:46 +00:00
/**
* \ brief Checks available audio driver n ames .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_GetNumAudioDrivers
* \ sa https : //wiki.libsdl.org/SDL_GetAudioDriver
2015-06-21 15:33:46 +00:00
*/
int audio_printAudioDrivers ( )
{
2022-11-30 20:51:59 +00:00
int i , n ;
const char * name ;
/* Get number of drivers */
n = SDL_GetNumAudioDrivers ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDrivers() " ) ;
SDLTest_AssertCheck ( n > = 0 , " Verify number of audio drivers >= 0, got: %i " , n ) ;
/* List drivers. */
if ( n > 0 ) {
for ( i = 0 ; i < n ; i + + ) {
name = SDL_GetAudioDriver ( i ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDriver(%i) " , i ) ;
SDLTest_AssertCheck ( name ! = NULL , " Verify returned name is not NULL " ) ;
if ( name ! = NULL ) {
SDLTest_AssertCheck ( name [ 0 ] ! = ' \0 ' , " Verify returned name is not empty, got: '%s' " , name ) ;
}
}
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
}
2015-06-21 15:33:46 +00:00
/**
* \ brief Checks current audio driver name with initialized audio .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_GetCurrentAudioDriver
2015-06-21 15:33:46 +00:00
*/
int audio_printCurrentAudioDriver ( )
{
2022-11-30 20:51:59 +00:00
/* Check current audio driver */
const char * name = SDL_GetCurrentAudioDriver ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetCurrentAudioDriver() " ) ;
SDLTest_AssertCheck ( name ! = NULL , " Verify returned name is not NULL " ) ;
if ( name ! = NULL ) {
SDLTest_AssertCheck ( name [ 0 ] ! = ' \0 ' , " Verify returned name is not empty, got: '%s' " , name ) ;
}
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/* Definition of all formats, channels, and frequencies used to test audio conversions */
const int _numAudioFormats = 18 ;
SDL_AudioFormat _audioFormats [ ] = { AUDIO_S8 , AUDIO_U8 , AUDIO_S16LSB , AUDIO_S16MSB , AUDIO_S16SYS , AUDIO_S16 , AUDIO_U16LSB ,
2022-11-30 20:51:59 +00:00
AUDIO_U16MSB , AUDIO_U16SYS , AUDIO_U16 , AUDIO_S32LSB , AUDIO_S32MSB , AUDIO_S32SYS , AUDIO_S32 ,
AUDIO_F32LSB , AUDIO_F32MSB , AUDIO_F32SYS , AUDIO_F32 } ;
2021-11-19 22:02:02 +00:00
const char * _audioFormatsVerbose [ ] = { " AUDIO_S8 " , " AUDIO_U8 " , " AUDIO_S16LSB " , " AUDIO_S16MSB " , " AUDIO_S16SYS " , " AUDIO_S16 " , " AUDIO_U16LSB " ,
2022-11-30 20:51:59 +00:00
" AUDIO_U16MSB " , " AUDIO_U16SYS " , " AUDIO_U16 " , " AUDIO_S32LSB " , " AUDIO_S32MSB " , " AUDIO_S32SYS " , " AUDIO_S32 " ,
" AUDIO_F32LSB " , " AUDIO_F32MSB " , " AUDIO_F32SYS " , " AUDIO_F32 " } ;
2015-06-21 15:33:46 +00:00
const int _numAudioChannels = 4 ;
Uint8 _audioChannels [ ] = { 1 , 2 , 4 , 6 } ;
const int _numAudioFrequencies = 4 ;
int _audioFrequencies [ ] = { 11025 , 22050 , 44100 , 48000 } ;
/**
* \ brief Builds various audio conversion structures
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_BuildAudioCVT
2015-06-21 15:33:46 +00:00
*/
int audio_buildAudioCVT ( )
{
2022-11-30 20:51:59 +00:00
int result ;
SDL_AudioCVT cvt ;
SDL_AudioSpec spec1 ;
SDL_AudioSpec spec2 ;
int i , ii , j , jj , k , kk ;
/* No conversion needed */
spec1 . format = AUDIO_S16LSB ;
spec1 . channels = 2 ;
spec1 . freq = 22050 ;
result = SDL_BuildAudioCVT ( & cvt , spec1 . format , spec1 . channels , spec1 . freq ,
spec1 . format , spec1 . channels , spec1 . freq ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(spec1 ==> spec1) " ) ;
SDLTest_AssertCheck ( result = = 0 , " Verify result value; expected: 0, got: %i " , result ) ;
/* Typical conversion */
spec1 . format = AUDIO_S8 ;
spec1 . channels = 1 ;
spec1 . freq = 22050 ;
spec2 . format = AUDIO_S16LSB ;
spec2 . channels = 2 ;
spec2 . freq = 44100 ;
result = SDL_BuildAudioCVT ( & cvt , spec1 . format , spec1 . channels , spec1 . freq ,
spec2 . format , spec2 . channels , spec2 . freq ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(spec1 ==> spec2) " ) ;
SDLTest_AssertCheck ( result = = 1 , " Verify result value; expected: 1, got: %i " , result ) ;
/* All source conversions with random conversion targets, allow 'null' conversions */
for ( i = 0 ; i < _numAudioFormats ; i + + ) {
for ( j = 0 ; j < _numAudioChannels ; j + + ) {
for ( k = 0 ; k < _numAudioFrequencies ; k + + ) {
spec1 . format = _audioFormats [ i ] ;
spec1 . channels = _audioChannels [ j ] ;
spec1 . freq = _audioFrequencies [ k ] ;
ii = SDLTest_RandomIntegerInRange ( 0 , _numAudioFormats - 1 ) ;
jj = SDLTest_RandomIntegerInRange ( 0 , _numAudioChannels - 1 ) ;
kk = SDLTest_RandomIntegerInRange ( 0 , _numAudioFrequencies - 1 ) ;
spec2 . format = _audioFormats [ ii ] ;
spec2 . channels = _audioChannels [ jj ] ;
spec2 . freq = _audioFrequencies [ kk ] ;
result = SDL_BuildAudioCVT ( & cvt , spec1 . format , spec1 . channels , spec1 . freq ,
spec2 . format , spec2 . channels , spec2 . freq ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(format[%i]=%s(%i),channels[%i]=%i,freq[%i]=%i ==> format[%i]=%s(%i),channels[%i]=%i,freq[%i]=%i) " ,
i , _audioFormatsVerbose [ i ] , spec1 . format , j , spec1 . channels , k , spec1 . freq , ii , _audioFormatsVerbose [ ii ] , spec2 . format , jj , spec2 . channels , kk , spec2 . freq ) ;
SDLTest_AssertCheck ( result = = 0 | | result = = 1 , " Verify result value; expected: 0 or 1, got: %i " , result ) ;
if ( result < 0 ) {
SDLTest_LogError ( " %s " , SDL_GetError ( ) ) ;
} else {
SDLTest_AssertCheck ( cvt . len_mult > 0 , " Verify that cvt.len_mult value; expected: >0, got: %i " , cvt . len_mult ) ;
}
}
2015-06-21 15:33:46 +00:00
}
}
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/**
* \ brief Checkes calls with invalid input to SDL_BuildAudioCVT
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_BuildAudioCVT
2015-06-21 15:33:46 +00:00
*/
int audio_buildAudioCVTNegative ( )
{
2022-11-30 20:51:59 +00:00
const char * expectedError = " Parameter 'cvt' is invalid " ;
const char * error ;
int result ;
SDL_AudioCVT cvt ;
SDL_AudioSpec spec1 ;
SDL_AudioSpec spec2 ;
int i ;
char message [ 256 ] ;
/* Valid format */
2015-06-21 15:33:46 +00:00
spec1 . format = AUDIO_S8 ;
spec1 . channels = 1 ;
spec1 . freq = 22050 ;
spec2 . format = AUDIO_S16LSB ;
spec2 . channels = 2 ;
spec2 . freq = 44100 ;
SDL_ClearError ( ) ;
SDLTest_AssertPass ( " Call to SDL_ClearError() " ) ;
2022-11-30 20:51:59 +00:00
/* NULL input for CVT buffer */
result = SDL_BuildAudioCVT ( ( SDL_AudioCVT * ) NULL , spec1 . format , spec1 . channels , spec1 . freq ,
spec2 . format , spec2 . channels , spec2 . freq ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(NULL,...) " ) ;
2015-06-21 15:33:46 +00:00
SDLTest_AssertCheck ( result = = - 1 , " Verify result value; expected: -1, got: %i " , result ) ;
error = SDL_GetError ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetError() " ) ;
2022-11-30 20:51:59 +00:00
SDLTest_AssertCheck ( error ! = NULL , " Validate that error message was not NULL " ) ;
if ( error ! = NULL ) {
SDLTest_AssertCheck ( SDL_strcmp ( error , expectedError ) = = 0 ,
" Validate error message, expected: '%s', got: '%s' " , expectedError , error ) ;
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
/* Invalid conversions */
for ( i = 1 ; i < 64 ; i + + ) {
/* Valid format to start with */
spec1 . format = AUDIO_S8 ;
spec1 . channels = 1 ;
spec1 . freq = 22050 ;
spec2 . format = AUDIO_S16LSB ;
spec2 . channels = 2 ;
spec2 . freq = 44100 ;
SDL_ClearError ( ) ;
SDLTest_AssertPass ( " Call to SDL_ClearError() " ) ;
/* Set various invalid format inputs */
SDL_strlcpy ( message , " Invalid: " , 256 ) ;
if ( i & 1 ) {
SDL_strlcat ( message , " spec1.format " , 256 ) ;
spec1 . format = 0 ;
}
if ( i & 2 ) {
SDL_strlcat ( message , " spec1.channels " , 256 ) ;
spec1 . channels = 0 ;
}
if ( i & 4 ) {
SDL_strlcat ( message , " spec1.freq " , 256 ) ;
spec1 . freq = 0 ;
}
if ( i & 8 ) {
SDL_strlcat ( message , " spec2.format " , 256 ) ;
spec2 . format = 0 ;
}
if ( i & 16 ) {
SDL_strlcat ( message , " spec2.channels " , 256 ) ;
spec2 . channels = 0 ;
}
if ( i & 32 ) {
SDL_strlcat ( message , " spec2.freq " , 256 ) ;
spec2 . freq = 0 ;
}
SDLTest_Log ( " %s " , message ) ;
result = SDL_BuildAudioCVT ( & cvt , spec1 . format , spec1 . channels , spec1 . freq ,
spec2 . format , spec2 . channels , spec2 . freq ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(spec1 ==> spec2) " ) ;
SDLTest_AssertCheck ( result = = - 1 , " Verify result value; expected: -1, got: %i " , result ) ;
error = SDL_GetError ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetError() " ) ;
SDLTest_AssertCheck ( error ! = NULL & & error [ 0 ] ! = ' \0 ' , " Validate that error message was not NULL or empty " ) ;
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
SDL_ClearError ( ) ;
SDLTest_AssertPass ( " Call to SDL_ClearError() " ) ;
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/**
* \ brief Checks current audio status .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_GetAudioStatus
2015-06-21 15:33:46 +00:00
*/
int audio_getAudioStatus ( )
{
2022-11-30 20:51:59 +00:00
SDL_AudioStatus result ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
/* Check current audio status */
result = SDL_GetAudioStatus ( ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioStatus() " ) ;
SDLTest_AssertCheck ( result = = SDL_AUDIO_STOPPED | | result = = SDL_AUDIO_PLAYING | | result = = SDL_AUDIO_PAUSED ,
" Verify returned value; expected: STOPPED (%i) | PLAYING (%i) | PAUSED (%i), got: %i " ,
SDL_AUDIO_STOPPED , SDL_AUDIO_PLAYING , SDL_AUDIO_PAUSED , result ) ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/**
* \ brief Opens , checks current audio status , and closes a device .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_GetAudioStatus
2015-06-21 15:33:46 +00:00
*/
int audio_openCloseAndGetAudioStatus ( )
{
2022-11-30 20:51:59 +00:00
SDL_AudioStatus result ;
int i ;
int count ;
const char * device ;
SDL_AudioDeviceID id ;
SDL_AudioSpec desired , obtained ;
/* Get number of devices. */
count = SDL_GetNumAudioDevices ( 0 ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(0) " ) ;
if ( count > 0 ) {
for ( i = 0 ; i < count ; i + + ) {
/* Get device name */
device = SDL_GetAudioDeviceName ( i , 0 ) ;
SDLTest_AssertPass ( " SDL_GetAudioDeviceName(%i,0) " , i ) ;
SDLTest_AssertCheck ( device ! = NULL , " Validate device name is not NULL; got: %s " , ( device ! = NULL ) ? device : " NULL " ) ;
if ( device = = NULL ) {
return TEST_ABORTED ;
}
/* Set standard desired spec */
desired . freq = 22050 ;
desired . format = AUDIO_S16SYS ;
desired . channels = 2 ;
desired . samples = 4096 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
/* Open device */
id = SDL_OpenAudioDevice ( device , 0 , & desired , & obtained , SDL_AUDIO_ALLOW_ANY_CHANGE ) ;
SDLTest_AssertPass ( " SDL_OpenAudioDevice('%s',...) " , device ) ;
SDLTest_AssertCheck ( id > 1 , " Validate device ID; expected: >=2, got: % " SDL_PRIu32 , id ) ;
if ( id > 1 ) {
/* Check device audio status */
result = SDL_GetAudioDeviceStatus ( id ) ;
SDLTest_AssertPass ( " Call to SDL_GetAudioDeviceStatus() " ) ;
SDLTest_AssertCheck ( result = = SDL_AUDIO_STOPPED | | result = = SDL_AUDIO_PLAYING | | result = = SDL_AUDIO_PAUSED ,
" Verify returned value; expected: STOPPED (%i) | PLAYING (%i) | PAUSED (%i), got: %i " ,
SDL_AUDIO_STOPPED , SDL_AUDIO_PLAYING , SDL_AUDIO_PAUSED , result ) ;
/* Close device again */
SDL_CloseAudioDevice ( id ) ;
SDLTest_AssertPass ( " Call to SDL_CloseAudioDevice() " ) ;
}
}
} else {
SDLTest_Log ( " No devices to test with " ) ;
}
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/**
* \ brief Locks and unlocks open audio device .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_LockAudioDevice
* \ sa https : //wiki.libsdl.org/SDL_UnlockAudioDevice
2015-06-21 15:33:46 +00:00
*/
int audio_lockUnlockOpenAudioDevice ( )
{
2022-11-30 20:51:59 +00:00
int i ;
int count ;
const char * device ;
SDL_AudioDeviceID id ;
SDL_AudioSpec desired , obtained ;
/* Get number of devices. */
count = SDL_GetNumAudioDevices ( 0 ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(0) " ) ;
if ( count > 0 ) {
for ( i = 0 ; i < count ; i + + ) {
/* Get device name */
device = SDL_GetAudioDeviceName ( i , 0 ) ;
SDLTest_AssertPass ( " SDL_GetAudioDeviceName(%i,0) " , i ) ;
SDLTest_AssertCheck ( device ! = NULL , " Validate device name is not NULL; got: %s " , ( device ! = NULL ) ? device : " NULL " ) ;
if ( device = = NULL ) {
return TEST_ABORTED ;
}
/* Set standard desired spec */
desired . freq = 22050 ;
desired . format = AUDIO_S16SYS ;
desired . channels = 2 ;
desired . samples = 4096 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
/* Open device */
id = SDL_OpenAudioDevice ( device , 0 , & desired , & obtained , SDL_AUDIO_ALLOW_ANY_CHANGE ) ;
SDLTest_AssertPass ( " SDL_OpenAudioDevice('%s',...) " , device ) ;
SDLTest_AssertCheck ( id > 1 , " Validate device ID; expected: >=2, got: % " SDL_PRIu32 , id ) ;
if ( id > 1 ) {
/* Lock to protect callback */
SDL_LockAudioDevice ( id ) ;
SDLTest_AssertPass ( " SDL_LockAudioDevice(% " SDL_PRIu32 " ) " , id ) ;
/* Simulate callback processing */
SDL_Delay ( 10 ) ;
SDLTest_Log ( " Simulate callback processing - delay " ) ;
/* Unlock again */
SDL_UnlockAudioDevice ( id ) ;
SDLTest_AssertPass ( " SDL_UnlockAudioDevice(% " SDL_PRIu32 " ) " , id ) ;
/* Close device again */
SDL_CloseAudioDevice ( id ) ;
SDLTest_AssertPass ( " Call to SDL_CloseAudioDevice() " ) ;
}
}
} else {
SDLTest_Log ( " No devices to test with " ) ;
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
}
2015-06-21 15:33:46 +00:00
/**
* \ brief Convert audio using various conversion structures
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_BuildAudioCVT
* \ sa https : //wiki.libsdl.org/SDL_ConvertAudio
2015-06-21 15:33:46 +00:00
*/
int audio_convertAudio ( )
{
2022-11-30 20:51:59 +00:00
int result ;
SDL_AudioCVT cvt ;
SDL_AudioSpec spec1 ;
SDL_AudioSpec spec2 ;
int c ;
char message [ 128 ] ;
int i , ii , j , jj , k , kk , l , ll ;
/* Iterate over bitmask that determines which parameters are modified in the conversion */
for ( c = 1 ; c < 8 ; c + + ) {
SDL_strlcpy ( message , " Changing: " , 128 ) ;
if ( c & 1 ) {
SDL_strlcat ( message , " Format " , 128 ) ;
}
if ( c & 2 ) {
SDL_strlcat ( message , " Channels " , 128 ) ;
}
if ( c & 4 ) {
SDL_strlcat ( message , " Frequencies " , 128 ) ;
}
SDLTest_Log ( " %s " , message ) ;
/* All source conversions with random conversion targets */
for ( i = 0 ; i < _numAudioFormats ; i + + ) {
for ( j = 0 ; j < _numAudioChannels ; j + + ) {
for ( k = 0 ; k < _numAudioFrequencies ; k + + ) {
spec1 . format = _audioFormats [ i ] ;
spec1 . channels = _audioChannels [ j ] ;
spec1 . freq = _audioFrequencies [ k ] ;
/* Ensure we have a different target format */
do {
if ( c & 1 ) {
ii = SDLTest_RandomIntegerInRange ( 0 , _numAudioFormats - 1 ) ;
} else {
ii = 1 ;
}
if ( c & 2 ) {
jj = SDLTest_RandomIntegerInRange ( 0 , _numAudioChannels - 1 ) ;
} else {
jj = j ;
}
if ( c & 4 ) {
kk = SDLTest_RandomIntegerInRange ( 0 , _numAudioFrequencies - 1 ) ;
} else {
kk = k ;
}
} while ( ( i = = ii ) & & ( j = = jj ) & & ( k = = kk ) ) ;
spec2 . format = _audioFormats [ ii ] ;
spec2 . channels = _audioChannels [ jj ] ;
spec2 . freq = _audioFrequencies [ kk ] ;
result = SDL_BuildAudioCVT ( & cvt , spec1 . format , spec1 . channels , spec1 . freq ,
spec2 . format , spec2 . channels , spec2 . freq ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(format[%i]=%s(%i),channels[%i]=%i,freq[%i]=%i ==> format[%i]=%s(%i),channels[%i]=%i,freq[%i]=%i) " ,
i , _audioFormatsVerbose [ i ] , spec1 . format , j , spec1 . channels , k , spec1 . freq , ii , _audioFormatsVerbose [ ii ] , spec2 . format , jj , spec2 . channels , kk , spec2 . freq ) ;
SDLTest_AssertCheck ( result = = 1 , " Verify result value; expected: 1, got: %i " , result ) ;
if ( result ! = 1 ) {
SDLTest_LogError ( " %s " , SDL_GetError ( ) ) ;
} else {
SDLTest_AssertCheck ( cvt . len_mult > 0 , " Verify that cvt.len_mult value; expected: >0, got: %i " , cvt . len_mult ) ;
if ( cvt . len_mult < 1 ) {
return TEST_ABORTED ;
}
/* Create some random data to convert */
l = 64 ;
ll = l * cvt . len_mult ;
SDLTest_Log ( " Creating dummy sample buffer of %i length (%i bytes) " , l , ll ) ;
cvt . len = l ;
cvt . buf = ( Uint8 * ) SDL_malloc ( ll ) ;
SDLTest_AssertCheck ( cvt . buf ! = NULL , " Check data buffer to convert is not NULL " ) ;
if ( cvt . buf = = NULL ) {
return TEST_ABORTED ;
}
/* Convert the data */
result = SDL_ConvertAudio ( & cvt ) ;
SDLTest_AssertPass ( " Call to SDL_ConvertAudio() " ) ;
SDLTest_AssertCheck ( result = = 0 , " Verify result value; expected: 0; got: %i " , result ) ;
SDLTest_AssertCheck ( cvt . buf ! = NULL , " Verify conversion buffer is not NULL " ) ;
SDLTest_AssertCheck ( cvt . len_ratio > 0.0 , " Verify conversion length ratio; expected: >0; got: %f " , cvt . len_ratio ) ;
/* Free converted buffer */
SDL_free ( cvt . buf ) ;
cvt . buf = NULL ;
}
}
2022-11-27 16:38:43 +00:00
}
2015-06-21 15:33:46 +00:00
}
}
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
2015-06-21 15:33:46 +00:00
}
/**
* \ brief Opens , checks current connected status , and closes a device .
*
2021-04-01 16:15:28 +00:00
* \ sa https : //wiki.libsdl.org/SDL_AudioDeviceConnected
2015-06-21 15:33:46 +00:00
*/
int audio_openCloseAudioDeviceConnected ( )
{
2022-11-30 20:51:59 +00:00
int result = - 1 ;
int i ;
int count ;
const char * device ;
SDL_AudioDeviceID id ;
SDL_AudioSpec desired , obtained ;
/* Get number of devices. */
count = SDL_GetNumAudioDevices ( 0 ) ;
SDLTest_AssertPass ( " Call to SDL_GetNumAudioDevices(0) " ) ;
if ( count > 0 ) {
for ( i = 0 ; i < count ; i + + ) {
/* Get device name */
device = SDL_GetAudioDeviceName ( i , 0 ) ;
SDLTest_AssertPass ( " SDL_GetAudioDeviceName(%i,0) " , i ) ;
SDLTest_AssertCheck ( device ! = NULL , " Validate device name is not NULL; got: %s " , ( device ! = NULL ) ? device : " NULL " ) ;
if ( device = = NULL ) {
return TEST_ABORTED ;
}
/* Set standard desired spec */
desired . freq = 22050 ;
desired . format = AUDIO_S16SYS ;
desired . channels = 2 ;
desired . samples = 4096 ;
desired . callback = _audio_testCallback ;
desired . userdata = NULL ;
/* Open device */
id = SDL_OpenAudioDevice ( device , 0 , & desired , & obtained , SDL_AUDIO_ALLOW_ANY_CHANGE ) ;
SDLTest_AssertPass ( " SDL_OpenAudioDevice('%s',...) " , device ) ;
SDLTest_AssertCheck ( id > 1 , " Validate device ID; expected: >1, got: % " SDL_PRIu32 , id ) ;
if ( id > 1 ) {
2015-06-21 15:33:46 +00:00
/* TODO: enable test code when function is available in SDL2 */
# ifdef AUDIODEVICECONNECTED_DEFINED
2022-11-30 20:51:59 +00:00
/* Get connected status */
result = SDL_AudioDeviceConnected ( id ) ;
SDLTest_AssertPass ( " Call to SDL_AudioDeviceConnected() " ) ;
2015-06-21 15:33:46 +00:00
# endif
2022-11-30 20:51:59 +00:00
SDLTest_AssertCheck ( result = = 1 , " Verify returned value; expected: 1; got: %i " , result ) ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
/* Close device again */
SDL_CloseAudioDevice ( id ) ;
SDLTest_AssertPass ( " Call to SDL_CloseAudioDevice() " ) ;
}
}
} else {
SDLTest_Log ( " No devices to test with " ) ;
}
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
return TEST_COMPLETED ;
}
2015-06-21 15:33:46 +00:00
2023-03-09 09:34:51 +00:00
static double sine_wave_sample ( const Sint64 idx , const Sint64 rate , const Sint64 freq , const double phase )
{
/* Using integer modulo to avoid precision loss caused by large floating
* point numbers . Sint64 is needed for the large integer multiplication .
* The integers are assumed to be non - negative so that modulo is always
* non - negative .
* sin ( i / rate * freq * 2 * M_PI + phase )
* = sin ( mod ( i / rate * freq , 1 ) * 2 * M_PI + phase )
* = sin ( mod ( i * freq , rate ) / rate * 2 * M_PI + phase ) */
return SDL_sin ( ( ( double ) ( idx * freq % rate ) ) / ( ( double ) rate ) * ( M_PI * 2 ) + phase ) ;
}
/**
* \ brief Check signal - to - noise ratio and maximum error of audio resampling .
*
* \ sa https : //wiki.libsdl.org/SDL_BuildAudioCVT
* \ sa https : //wiki.libsdl.org/SDL_ConvertAudio
*/
int audio_resampleLoss ( )
{
/* Note: always test long input time (>= 5s from experience) in some test
* cases because an improper implementation may suffer from low resampling
* precision with long input due to e . g . doing subtraction with large floats . */
struct test_spec_t {
int time ;
int freq ;
double phase ;
int rate_in ;
int rate_out ;
double signal_to_noise ;
double max_error ;
} test_specs [ ] = {
{ 50 , 440 , 0 , 44100 , 48000 , 60 , 0.0025 } ,
{ 50 , 5000 , M_PI / 2 , 20000 , 10000 , 65 , 0.0010 } ,
{ 0 }
} ;
int spec_idx = 0 ;
for ( spec_idx = 0 ; test_specs [ spec_idx ] . time > 0 ; + + spec_idx ) {
const struct test_spec_t * spec = & test_specs [ spec_idx ] ;
const int frames_in = spec - > time * spec - > rate_in ;
const int frames_target = spec - > time * spec - > rate_out ;
const int len_in = frames_in * ( int ) sizeof ( float ) ;
const int len_target = frames_target * ( int ) sizeof ( float ) ;
Uint64 tick_beg = 0 ;
Uint64 tick_end = 0 ;
SDL_AudioCVT cvt ;
int i = 0 ;
int ret = 0 ;
double max_error = 0 ;
double sum_squared_error = 0 ;
double sum_squared_value = 0 ;
double signal_to_noise = 0 ;
SDLTest_AssertPass ( " Test resampling of %i s %i Hz %f phase sine wave from sampling rate of %i Hz to %i Hz " ,
spec - > time , spec - > freq , spec - > phase , spec - > rate_in , spec - > rate_out ) ;
2024-01-10 11:05:34 +00:00
ret = SDL_BuildAudioCVT ( & cvt , AUDIO_F32SYS , 1 , spec - > rate_in , AUDIO_F32SYS , 1 , spec - > rate_out ) ;
SDLTest_AssertPass ( " Call to SDL_BuildAudioCVT(&cvt, AUDIO_F32SYS, 1, %i, AUDIO_F32SYS, 1, %i) " , spec - > rate_in , spec - > rate_out ) ;
2023-03-09 09:34:51 +00:00
SDLTest_AssertCheck ( ret = = 1 , " Expected SDL_BuildAudioCVT to succeed and conversion to be needed. " ) ;
if ( ret ! = 1 ) {
return TEST_ABORTED ;
}
cvt . buf = ( Uint8 * ) SDL_malloc ( len_in * cvt . len_mult ) ;
SDLTest_AssertCheck ( cvt . buf ! = NULL , " Expected input buffer to be created. " ) ;
if ( cvt . buf = = NULL ) {
return TEST_ABORTED ;
}
cvt . len = len_in ;
for ( i = 0 ; i < frames_in ; + + i ) {
* ( ( ( float * ) cvt . buf ) + i ) = ( float ) sine_wave_sample ( i , spec - > rate_in , spec - > freq , spec - > phase ) ;
}
tick_beg = SDL_GetPerformanceCounter ( ) ;
ret = SDL_ConvertAudio ( & cvt ) ;
tick_end = SDL_GetPerformanceCounter ( ) ;
SDLTest_AssertPass ( " Call to SDL_ConvertAudio(&cvt) " ) ;
SDLTest_AssertCheck ( ret = = 0 , " Expected SDL_ConvertAudio to succeed. " ) ;
SDLTest_AssertCheck ( cvt . len_cvt = = len_target , " Expected output length %i, got %i. " , len_target , cvt . len_cvt ) ;
if ( ret ! = 0 | | cvt . len_cvt ! = len_target ) {
SDL_free ( cvt . buf ) ;
return TEST_ABORTED ;
}
SDLTest_Log ( " Resampling used %f seconds. " , ( ( double ) ( tick_end - tick_beg ) ) / SDL_GetPerformanceFrequency ( ) ) ;
for ( i = 0 ; i < frames_target ; + + i ) {
const float output = * ( ( ( float * ) cvt . buf ) + i ) ;
const double target = sine_wave_sample ( i , spec - > rate_out , spec - > freq , spec - > phase ) ;
const double error = SDL_fabs ( target - output ) ;
max_error = SDL_max ( max_error , error ) ;
sum_squared_error + = error * error ;
sum_squared_value + = target * target ;
}
SDL_free ( cvt . buf ) ;
signal_to_noise = 10 * SDL_log10 ( sum_squared_value / sum_squared_error ) ; /* decibel */
SDLTest_AssertCheck ( isfinite ( sum_squared_value ) , " Sum of squared target should be finite. " ) ;
SDLTest_AssertCheck ( isfinite ( sum_squared_error ) , " Sum of squared error should be finite. " ) ;
/* Infinity is theoretically possible when there is very little to no noise */
SDLTest_AssertCheck ( ! isnan ( signal_to_noise ) , " Signal-to-noise ratio should not be NaN. " ) ;
SDLTest_AssertCheck ( isfinite ( max_error ) , " Maximum conversion error should be finite. " ) ;
SDLTest_AssertCheck ( signal_to_noise > = spec - > signal_to_noise , " Conversion signal-to-noise ratio %f dB should be no less than %f dB. " ,
signal_to_noise , spec - > signal_to_noise ) ;
SDLTest_AssertCheck ( max_error < = spec - > max_error , " Maximum conversion error %f should be no more than %f. " ,
max_error , spec - > max_error ) ;
}
return TEST_COMPLETED ;
}
2015-06-21 15:33:46 +00:00
/* ================= Test Case References ================== */
/* Audio test cases */
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest1 = {
( SDLTest_TestCaseFp ) audio_enumerateAndNameAudioDevices , " audio_enumerateAndNameAudioDevices " , " Enumerate and name available audio devices (output and capture) " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest2 = {
( SDLTest_TestCaseFp ) audio_enumerateAndNameAudioDevicesNegativeTests , " audio_enumerateAndNameAudioDevicesNegativeTests " , " Negative tests around enumeration and naming of audio devices. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest3 = {
( SDLTest_TestCaseFp ) audio_printAudioDrivers , " audio_printAudioDrivers " , " Checks available audio driver names. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest4 = {
( SDLTest_TestCaseFp ) audio_printCurrentAudioDriver , " audio_printCurrentAudioDriver " , " Checks current audio driver name with initialized audio. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest5 = {
( SDLTest_TestCaseFp ) audio_buildAudioCVT , " audio_buildAudioCVT " , " Builds various audio conversion structures. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest6 = {
( SDLTest_TestCaseFp ) audio_buildAudioCVTNegative , " audio_buildAudioCVTNegative " , " Checks calls with invalid input to SDL_BuildAudioCVT " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest7 = {
( SDLTest_TestCaseFp ) audio_getAudioStatus , " audio_getAudioStatus " , " Checks current audio status. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest8 = {
( SDLTest_TestCaseFp ) audio_openCloseAndGetAudioStatus , " audio_openCloseAndGetAudioStatus " , " Opens and closes audio device and get audio status. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest9 = {
( SDLTest_TestCaseFp ) audio_lockUnlockOpenAudioDevice , " audio_lockUnlockOpenAudioDevice " , " Locks and unlocks an open audio device. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
/* TODO: enable test when SDL_ConvertAudio segfaults on cygwin have been fixed. */
/* For debugging, test case can be run manually using --filter audio_convertAudio */
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest10 = {
( SDLTest_TestCaseFp ) audio_convertAudio , " audio_convertAudio " , " Convert audio using available formats. " , TEST_DISABLED
} ;
2015-06-21 15:33:46 +00:00
/* TODO: enable test when SDL_AudioDeviceConnected has been implemented. */
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest11 = {
( SDLTest_TestCaseFp ) audio_openCloseAudioDeviceConnected , " audio_openCloseAudioDeviceConnected " , " Opens and closes audio device and get connected status. " , TEST_DISABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest12 = {
( SDLTest_TestCaseFp ) audio_quitInitAudioSubSystem , " audio_quitInitAudioSubSystem " , " Quit and re-init audio subsystem. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest13 = {
( SDLTest_TestCaseFp ) audio_initQuitAudio , " audio_initQuitAudio " , " Init and quit audio drivers directly. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest14 = {
( SDLTest_TestCaseFp ) audio_initOpenCloseQuitAudio , " audio_initOpenCloseQuitAudio " , " Cycle through init, open, close and quit with various audio specs. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference audioTest15 = {
( SDLTest_TestCaseFp ) audio_pauseUnpauseAudio , " audio_pauseUnpauseAudio " , " Pause and Unpause audio for various audio specs while testing callback. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
2023-03-09 09:34:51 +00:00
static const SDLTest_TestCaseReference audioTest16 = {
( SDLTest_TestCaseFp ) audio_resampleLoss , " audio_resampleLoss " , " Check signal-to-noise ratio and maximum error of audio resampling. " , TEST_ENABLED
} ;
2015-06-21 15:33:46 +00:00
/* Sequence of Audio test cases */
2022-11-30 20:51:59 +00:00
static const SDLTest_TestCaseReference * audioTests [ ] = {
2015-06-21 15:33:46 +00:00
& audioTest1 , & audioTest2 , & audioTest3 , & audioTest4 , & audioTest5 , & audioTest6 ,
& audioTest7 , & audioTest8 , & audioTest9 , & audioTest10 , & audioTest11 ,
2023-03-09 09:34:51 +00:00
& audioTest12 , & audioTest13 , & audioTest14 , & audioTest15 , & audioTest16 , NULL
2015-06-21 15:33:46 +00:00
} ;
/* Audio test suite (global) */
SDLTest_TestSuiteReference audioTestSuite = {
" Audio " ,
_audioSetUp ,
audioTests ,
_audioTearDown
} ;