mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-26 15:23:48 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			404 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			404 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file paqa_errs.c
 | |
| 	@ingroup qa_src
 | |
| 	@brief Self Testing Quality Assurance app for PortAudio
 | |
| 	Do lots of bad things to test error reporting.
 | |
| 	@author Phil Burk  http://www.softsynth.com
 | |
|     Pieter Suurmond adapted to V19 API.
 | |
| */
 | |
| /*
 | |
|  * $Id: paqa_errs.c 1756 2011-09-08 06:09:29Z philburk $
 | |
|  *
 | |
|  * This program uses the PortAudio Portable Audio Library.
 | |
|  * For more information see: http://www.portaudio.com
 | |
|  * Copyright (c) 1999-2000 Ross Bencina and Phil Burk
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining
 | |
|  * a copy of this software and associated documentation files
 | |
|  * (the "Software"), to deal in the Software without restriction,
 | |
|  * including without limitation the rights to use, copy, modify, merge,
 | |
|  * publish, distribute, sublicense, and/or sell copies of the Software,
 | |
|  * and to permit persons to whom the Software is furnished to do so,
 | |
|  * subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be
 | |
|  * included in all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 | |
|  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | |
|  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 | |
|  * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
 | |
|  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
 | |
|  * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 | |
|  * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * The text above constitutes the entire PortAudio license; however, 
 | |
|  * the PortAudio community also makes the following non-binding requests:
 | |
|  *
 | |
|  * Any person wishing to distribute modifications to the Software is
 | |
|  * requested to send the modifications to the original developer so that
 | |
|  * they can be incorporated into the canonical version. It is also 
 | |
|  * requested that these non-binding requests be included along with the 
 | |
|  * license above.
 | |
|  */
 | |
|  
 | |
| #include <stdio.h>
 | |
| #include <math.h>
 | |
| 
 | |
| #include "portaudio.h"
 | |
| 
 | |
| /*--------- Definitions ---------*/
 | |
| #define MODE_INPUT        (0)
 | |
| #define MODE_OUTPUT       (1)
 | |
| #define FRAMES_PER_BUFFER (64)
 | |
| #define SAMPLE_RATE       (44100.0)
 | |
| 
 | |
| typedef struct PaQaData
 | |
| {
 | |
|     unsigned long  framesLeft;
 | |
|     int            numChannels;
 | |
|     int            bytesPerSample;
 | |
|     int            mode;
 | |
| }
 | |
| PaQaData;
 | |
| 
 | |
| static int gNumPassed = 0; /* Two globals */
 | |
| static int gNumFailed = 0;
 | |
| 
 | |
| /*------------------- Macros ------------------------------*/
 | |
| /* Print ERROR if it fails. Tally success or failure. Odd  */
 | |
| /* do-while wrapper seems to be needed for some compilers. */
 | |
| 
 | |
| #define EXPECT(_exp) \
 | |
|     do \
 | |
|     { \
 | |
|         if ((_exp)) {\
 | |
|             gNumPassed++; \
 | |
|         } \
 | |
|         else { \
 | |
|             printf("\nERROR - 0x%x - %s for %s\n", result, Pa_GetErrorText(result), #_exp ); \
 | |
|             gNumFailed++; \
 | |
|             goto error; \
 | |
|         } \
 | |
|     } while(0)
 | |
| 
 | |
| #define HOPEFOR(_exp) \
 | |
|     do \
 | |
|     { \
 | |
|         if ((_exp)) {\
 | |
|             gNumPassed++; \
 | |
|         } \
 | |
|         else { \
 | |
|             printf("\nERROR - 0x%x - %s for %s\n", result, Pa_GetErrorText(result), #_exp ); \
 | |
|             gNumFailed++; \
 | |
|         } \
 | |
|     } while(0)
 | |
| 
 | |
| /*-------------------------------------------------------------------------*/
 | |
| /* This routine will be called by the PortAudio engine when audio is needed.
 | |
|    It may be called at interrupt level on some machines so don't do anything
 | |
|    that could mess up the system like calling malloc() or free().
 | |
| */
 | |
| static int QaCallback( const void*                      inputBuffer,
 | |
|                        void*                            outputBuffer,
 | |
|                        unsigned long                    framesPerBuffer,
 | |
| 			           const PaStreamCallbackTimeInfo*  timeInfo,
 | |
| 			           PaStreamCallbackFlags            statusFlags,
 | |
|                        void*                            userData )
 | |
| {
 | |
|     unsigned long   i;
 | |
|     unsigned char*  out = (unsigned char *) outputBuffer;
 | |
|     PaQaData*       data = (PaQaData *) userData;
 | |
|     
 | |
|     (void)inputBuffer; /* Prevent "unused variable" warnings. */
 | |
| 
 | |
|     /* Zero out buffer so we don't hear terrible noise. */
 | |
|     if( data->mode == MODE_OUTPUT )
 | |
|     {
 | |
|         unsigned long numBytes = framesPerBuffer * data->numChannels * data->bytesPerSample;
 | |
|         for( i=0; i<numBytes; i++ )
 | |
|         {
 | |
|             *out++ = 0;
 | |
|         }
 | |
|     }
 | |
|     /* Are we through yet? */
 | |
|     if( data->framesLeft > framesPerBuffer )
 | |
|     {
 | |
|         data->framesLeft -= framesPerBuffer;
 | |
|         return 0;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         data->framesLeft = 0;
 | |
|         return 1;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PaDeviceIndex FindInputOnlyDevice(void)
 | |
| {
 | |
|     PaDeviceIndex result = Pa_GetDefaultInputDevice();
 | |
|     if( result != paNoDevice && Pa_GetDeviceInfo(result)->maxOutputChannels == 0 )
 | |
|         return result;
 | |
| 
 | |
|     for( result = 0; result < Pa_GetDeviceCount(); ++result )
 | |
|     {
 | |
|         if( Pa_GetDeviceInfo(result)->maxOutputChannels == 0 )
 | |
|             return result;
 | |
|     }
 | |
| 
 | |
|     return paNoDevice;
 | |
| }
 | |
| 
 | |
| static PaDeviceIndex FindOutputOnlyDevice(void)
 | |
| {
 | |
|     PaDeviceIndex result = Pa_GetDefaultOutputDevice();
 | |
|     if( result != paNoDevice && Pa_GetDeviceInfo(result)->maxInputChannels == 0 )
 | |
|         return result;
 | |
| 
 | |
|     for( result = 0; result < Pa_GetDeviceCount(); ++result )
 | |
|     {
 | |
|         if( Pa_GetDeviceInfo(result)->maxInputChannels == 0 )
 | |
|             return result;
 | |
|     }
 | |
| 
 | |
|     return paNoDevice;
 | |
| }
 | |
| 
 | |
| /*-------------------------------------------------------------------------------------------------*/
 | |
| static int TestBadOpens( void )
 | |
| {
 | |
|     PaStream*           stream = NULL;
 | |
|     PaError             result;
 | |
|     PaQaData            myData;
 | |
|     PaStreamParameters  ipp, opp;
 | |
|     const PaDeviceInfo* info = NULL;
 | |
| 
 | |
|     
 | |
|     /* Setup data for synthesis thread. */
 | |
|     myData.framesLeft = (unsigned long) (SAMPLE_RATE * 100); /* 100 seconds */
 | |
|     myData.numChannels = 1;
 | |
|     myData.mode = MODE_OUTPUT;
 | |
| 
 | |
|     /*----------------------------- No devices specified: */
 | |
|     ipp.device                    = opp.device                    = paNoDevice;
 | |
|     ipp.channelCount              = opp.channelCount              = 0; /* Also no channels. */
 | |
|     ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|     ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|     /* Take the low latency of the default device for all subsequent tests. */
 | |
|     info = Pa_GetDeviceInfo(Pa_GetDefaultInputDevice());
 | |
|     ipp.suggestedLatency          = info ? info->defaultLowInputLatency : 0.100;
 | |
|     info = Pa_GetDeviceInfo(Pa_GetDefaultOutputDevice());
 | |
|     opp.suggestedLatency          = info ? info->defaultLowOutputLatency : 0.100;
 | |
|     HOPEFOR(((result = Pa_OpenStream(&stream, &ipp, &opp,
 | |
|                                      SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                      paClipOff, QaCallback, &myData )) == paInvalidDevice));
 | |
| 
 | |
|     /*----------------------------- No devices specified #2: */
 | |
|     HOPEFOR(((result = Pa_OpenStream(&stream, NULL, NULL,
 | |
|                                      SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                      paClipOff, QaCallback, &myData )) == paInvalidDevice));
 | |
| 
 | |
|     /*----------------------------- Out of range input device specified: */
 | |
|     ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|     ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|     ipp.channelCount = 0;           ipp.device = Pa_GetDeviceCount(); /* And no output device, and no channels. */
 | |
|     opp.channelCount = 0;           opp.device = paNoDevice;
 | |
|     HOPEFOR(((result = Pa_OpenStream(&stream, &ipp, NULL,
 | |
|                                      SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                      paClipOff, QaCallback, &myData )) == paInvalidDevice));
 | |
| 
 | |
|     /*----------------------------- Out of range output device specified: */
 | |
|     ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|     ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|     ipp.channelCount = 0;           ipp.device = paNoDevice; /* And no input device, and no channels. */
 | |
|     opp.channelCount = 0;           opp.device = Pa_GetDeviceCount();
 | |
|     HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                      SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                      paClipOff, QaCallback, &myData )) == paInvalidDevice));
 | |
| 
 | |
|     if (Pa_GetDefaultInputDevice() != paNoDevice) {
 | |
|         /*----------------------------- Zero input channels: */
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = Pa_GetDefaultInputDevice();
 | |
|         opp.channelCount = 0;           opp.device = paNoDevice;    /* And no output device, and no output channels. */   
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, &ipp, NULL,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff, QaCallback, &myData )) == paInvalidChannelCount));
 | |
|     }
 | |
| 
 | |
|     if (Pa_GetDefaultOutputDevice() != paNoDevice) {
 | |
|         /*----------------------------- Zero output channels: */
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice; /* And no input device, and no input channels. */
 | |
|         opp.channelCount = 0;           opp.device = Pa_GetDefaultOutputDevice();
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff, QaCallback, &myData )) == paInvalidChannelCount));
 | |
|     }
 | |
|     /*----------------------------- Nonzero input and output channels but no output device: */
 | |
|     ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|     ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|     ipp.channelCount = 2;           ipp.device = Pa_GetDefaultInputDevice();        /* Both stereo. */
 | |
|     opp.channelCount = 2;           opp.device = paNoDevice;
 | |
|     HOPEFOR(((result = Pa_OpenStream(&stream, &ipp, &opp,
 | |
|                                      SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                      paClipOff, QaCallback, &myData )) == paInvalidDevice));
 | |
| 
 | |
|     /*----------------------------- Nonzero input and output channels but no input device: */
 | |
|     ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|     ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|     ipp.channelCount = 2;           ipp.device = paNoDevice;
 | |
|     opp.channelCount = 2;           opp.device = Pa_GetDefaultOutputDevice();
 | |
|     HOPEFOR(((result = Pa_OpenStream(&stream, &ipp, &opp,
 | |
|                                      SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                      paClipOff, QaCallback, &myData )) == paInvalidDevice));
 | |
| 
 | |
|     if (Pa_GetDefaultOutputDevice() != paNoDevice) {
 | |
|         /*----------------------------- NULL stream pointer: */
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice;           /* Output is more likely than input. */
 | |
|         opp.channelCount = 2;           opp.device = Pa_GetDefaultOutputDevice();    /* Only 2 output channels. */
 | |
|         HOPEFOR(((result = Pa_OpenStream(NULL, &ipp, &opp,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff, QaCallback, &myData )) == paBadStreamPtr));
 | |
| 
 | |
|         /*----------------------------- Low sample rate: */
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice;
 | |
|         opp.channelCount = 2;           opp.device = Pa_GetDefaultOutputDevice();
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                          1.0, FRAMES_PER_BUFFER, /* 1 cycle per second (1 Hz) is too low. */
 | |
|                                          paClipOff, QaCallback, &myData )) == paInvalidSampleRate));
 | |
|         
 | |
|         /*----------------------------- High sample rate: */
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice;
 | |
|         opp.channelCount = 2;           opp.device = Pa_GetDefaultOutputDevice();
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                          10000000.0, FRAMES_PER_BUFFER, /* 10^6 cycles per second (10 MHz) is too high. */
 | |
|                                          paClipOff, QaCallback, &myData )) == paInvalidSampleRate));
 | |
| 
 | |
|         /*----------------------------- NULL callback: */
 | |
|         /* NULL callback is valid in V19 -- it means use blocking read/write stream
 | |
|     
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice;
 | |
|         opp.channelCount = 2;           opp.device = Pa_GetDefaultOutputDevice();
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff,
 | |
|                                          NULL,
 | |
|                                          &myData )) == paNullCallback));
 | |
|         */
 | |
| 
 | |
|         /*----------------------------- Bad flag: */
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice;
 | |
|         opp.channelCount = 2;           opp.device = Pa_GetDefaultOutputDevice();
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          255,                      /* Is 8 maybe legal V19 API? */
 | |
|                                          QaCallback, &myData )) == paInvalidFlag));
 | |
|     }
 | |
| 
 | |
|     /*----------------------------- using input device as output device: */
 | |
|     if( FindInputOnlyDevice() != paNoDevice )
 | |
|     {
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 0;           ipp.device = paNoDevice; /* And no input device, and no channels. */
 | |
|         opp.channelCount = 2;           opp.device = FindInputOnlyDevice();
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, &opp,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff, QaCallback, &myData )) == paInvalidChannelCount));
 | |
|     }
 | |
| 
 | |
|     /*----------------------------- using output device as input device: */
 | |
|     if( FindOutputOnlyDevice() != paNoDevice )
 | |
|     {
 | |
|         ipp.hostApiSpecificStreamInfo = opp.hostApiSpecificStreamInfo = NULL;
 | |
|         ipp.sampleFormat              = opp.sampleFormat              = paFloat32;
 | |
|         ipp.channelCount = 2;           ipp.device = FindOutputOnlyDevice();
 | |
|         opp.channelCount = 0;           opp.device = paNoDevice;  /* And no output device, and no channels. */
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, &ipp, NULL,
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff, QaCallback, &myData )) == paInvalidChannelCount));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     if( stream != NULL ) Pa_CloseStream( stream );
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| /*-----------------------------------------------------------------------------------------*/
 | |
| static int TestBadActions( void )
 | |
| {
 | |
|     PaStream*           stream = NULL;
 | |
|     const PaDeviceInfo* deviceInfo = NULL;
 | |
|     PaError             result = 0;
 | |
|     PaQaData            myData;
 | |
|     PaStreamParameters  opp;
 | |
|     const PaDeviceInfo* info = NULL;
 | |
| 
 | |
|     /* Setup data for synthesis thread. */
 | |
|     myData.framesLeft = (unsigned long)(SAMPLE_RATE * 100); /* 100 seconds */
 | |
|     myData.numChannels = 1;
 | |
|     myData.mode = MODE_OUTPUT;
 | |
| 
 | |
|     opp.device                    = Pa_GetDefaultOutputDevice(); /* Default output. */
 | |
|     opp.channelCount              = 2;                           /* Stereo output.  */
 | |
|     opp.hostApiSpecificStreamInfo = NULL;
 | |
|     opp.sampleFormat              = paFloat32;
 | |
|     info = Pa_GetDeviceInfo(opp.device);
 | |
|     opp.suggestedLatency          = info ? info->defaultLowOutputLatency : 0.100;
 | |
| 
 | |
|     if (opp.device != paNoDevice) {
 | |
|         HOPEFOR(((result = Pa_OpenStream(&stream, NULL, /* Take NULL as input parame-     */
 | |
|                                          &opp,          /* ters, meaning try only output. */
 | |
|                                          SAMPLE_RATE, FRAMES_PER_BUFFER,
 | |
|                                          paClipOff, QaCallback, &myData )) == paNoError));
 | |
|     }
 | |
|  
 | |
|     HOPEFOR(((deviceInfo = Pa_GetDeviceInfo(paNoDevice))    == NULL));
 | |
|     HOPEFOR(((deviceInfo = Pa_GetDeviceInfo(87654))    == NULL));
 | |
|     HOPEFOR(((result = Pa_StartStream(NULL))    == paBadStreamPtr));
 | |
|     HOPEFOR(((result = Pa_StopStream(NULL))     == paBadStreamPtr));
 | |
|     HOPEFOR(((result = Pa_IsStreamStopped(NULL)) == paBadStreamPtr));
 | |
|     HOPEFOR(((result = Pa_IsStreamActive(NULL)) == paBadStreamPtr));
 | |
|     HOPEFOR(((result = Pa_CloseStream(NULL))    == paBadStreamPtr));
 | |
|     HOPEFOR(((result = Pa_SetStreamFinishedCallback(NULL, NULL)) == paBadStreamPtr));
 | |
|     HOPEFOR(((result = !Pa_GetStreamInfo(NULL))));
 | |
|     HOPEFOR(((result = Pa_GetStreamTime(NULL))  == 0.0));
 | |
|     HOPEFOR(((result = Pa_GetStreamCpuLoad(NULL))  == 0.0));
 | |
|     HOPEFOR(((result = Pa_ReadStream(NULL, NULL, 0))  == paBadStreamPtr));
 | |
|     HOPEFOR(((result = Pa_WriteStream(NULL, NULL, 0))  == paBadStreamPtr));
 | |
| 
 | |
|     /** @todo test Pa_GetStreamReadAvailable and Pa_GetStreamWriteAvailable */
 | |
| 
 | |
|     if (stream != NULL) Pa_CloseStream(stream);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| /*---------------------------------------------------------------------*/
 | |
| int main(void);
 | |
| int main(void)
 | |
| {
 | |
|     PaError result;
 | |
|     
 | |
|     EXPECT(((result = Pa_Initialize()) == paNoError));
 | |
|     TestBadOpens();
 | |
|     TestBadActions();
 | |
| error:
 | |
|     Pa_Terminate();
 | |
|     printf("QA Report: %d passed, %d failed.\n", gNumPassed, gNumFailed);
 | |
|     return 0;
 | |
| }
 |