annotate src/portaudio_20161030_catalina_patch/test/patest_dither.c @ 83:ae30d91d2ffe

Replace these with versions built using an older toolset (so as to avoid ABI compatibilities when linking on Ubuntu 14.04 for packaging purposes)
author Chris Cannam
date Fri, 07 Feb 2020 11:51:13 +0000
parents 4edcd14160a5
children
rev   line source
Chris@4 1 /** @file patest_dither.c
Chris@4 2 @ingroup test_src
Chris@4 3 @brief Attempt to hear difference between dithered and non-dithered signal.
Chris@4 4
Chris@4 5 This only has an effect if the native format is 16 bit.
Chris@4 6
Chris@4 7 @author Phil Burk http://www.softsynth.com
Chris@4 8 */
Chris@4 9 /*
Chris@55 10 * $Id$
Chris@4 11 *
Chris@4 12 * This program uses the PortAudio Portable Audio Library.
Chris@4 13 * For more information see: http://www.portaudio.com
Chris@4 14 * Copyright (c) 1999-2000 Ross Bencina and Phil Burk
Chris@4 15 *
Chris@4 16 * Permission is hereby granted, free of charge, to any person obtaining
Chris@4 17 * a copy of this software and associated documentation files
Chris@4 18 * (the "Software"), to deal in the Software without restriction,
Chris@4 19 * including without limitation the rights to use, copy, modify, merge,
Chris@4 20 * publish, distribute, sublicense, and/or sell copies of the Software,
Chris@4 21 * and to permit persons to whom the Software is furnished to do so,
Chris@4 22 * subject to the following conditions:
Chris@4 23 *
Chris@4 24 * The above copyright notice and this permission notice shall be
Chris@4 25 * included in all copies or substantial portions of the Software.
Chris@4 26 *
Chris@4 27 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
Chris@4 28 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
Chris@4 29 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
Chris@4 30 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
Chris@4 31 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
Chris@4 32 * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
Chris@4 33 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Chris@4 34 */
Chris@4 35
Chris@4 36 /*
Chris@4 37 * The text above constitutes the entire PortAudio license; however,
Chris@4 38 * the PortAudio community also makes the following non-binding requests:
Chris@4 39 *
Chris@4 40 * Any person wishing to distribute modifications to the Software is
Chris@4 41 * requested to send the modifications to the original developer so that
Chris@4 42 * they can be incorporated into the canonical version. It is also
Chris@4 43 * requested that these non-binding requests be included along with the
Chris@4 44 * license above.
Chris@4 45 */
Chris@4 46
Chris@4 47 #include <stdio.h>
Chris@4 48 #include <math.h>
Chris@4 49
Chris@4 50 #include "portaudio.h"
Chris@4 51
Chris@4 52 #define NUM_SECONDS (5)
Chris@4 53 #define SAMPLE_RATE (44100)
Chris@4 54 #ifndef M_PI
Chris@4 55 #define M_PI (3.14159265)
Chris@4 56 #endif
Chris@4 57 #define TABLE_SIZE (200)
Chris@4 58
Chris@4 59 typedef struct paTestData
Chris@4 60 {
Chris@4 61 float sine[TABLE_SIZE];
Chris@4 62 float amplitude;
Chris@4 63 int left_phase;
Chris@4 64 int right_phase;
Chris@4 65 }
Chris@4 66 paTestData;
Chris@4 67
Chris@4 68 /* This routine will be called by the PortAudio engine when audio is needed.
Chris@4 69 ** It may called at interrupt level on some machines so don't do anything
Chris@4 70 ** that could mess up the system like calling malloc() or free().
Chris@4 71 */
Chris@4 72 static int sineCallback( const void *inputBuffer, void *outputBuffer,
Chris@4 73 unsigned long framesPerBuffer,
Chris@4 74 const PaStreamCallbackTimeInfo *timeInfo,
Chris@4 75 PaStreamCallbackFlags statusFlags, void *userData )
Chris@4 76 {
Chris@4 77 paTestData *data = (paTestData*)userData;
Chris@4 78 float *out = (float*)outputBuffer;
Chris@4 79 float amplitude = data->amplitude;
Chris@4 80 unsigned int i;
Chris@4 81 (void) inputBuffer;
Chris@4 82
Chris@4 83 for( i=0; i<framesPerBuffer; i++ )
Chris@4 84 {
Chris@4 85 *out++ = amplitude * data->sine[data->left_phase]; /* left */
Chris@4 86 *out++ = amplitude * data->sine[data->right_phase]; /* right */
Chris@4 87 data->left_phase += 1;
Chris@4 88 if( data->left_phase >= TABLE_SIZE ) data->left_phase -= TABLE_SIZE;
Chris@4 89 data->right_phase += 3; /* higher pitch so we can distinguish left and right. */
Chris@4 90 if( data->right_phase >= TABLE_SIZE ) data->right_phase -= TABLE_SIZE;
Chris@4 91 }
Chris@4 92 return 0;
Chris@4 93 }
Chris@4 94
Chris@4 95 /*****************************************************************************/
Chris@4 96 /*
Chris@4 97 V18 version did not call Pa_Terminate() if Pa_Initialize() failed.
Chris@4 98 This V19 version ALWAYS calls Pa_Terminate(). PS.
Chris@4 99 */
Chris@4 100 PaError PlaySine( paTestData *data, PaStreamFlags flags, float amplitude );
Chris@4 101 PaError PlaySine( paTestData *data, PaStreamFlags flags, float amplitude )
Chris@4 102 {
Chris@4 103 PaStream* stream;
Chris@4 104 PaStreamParameters outputParameters;
Chris@4 105 PaError err;
Chris@4 106
Chris@4 107 data->left_phase = data->right_phase = 0;
Chris@4 108 data->amplitude = amplitude;
Chris@4 109
Chris@4 110 err = Pa_Initialize();
Chris@4 111 if (err != paNoError)
Chris@4 112 goto done;
Chris@4 113
Chris@4 114 outputParameters.device = Pa_GetDefaultOutputDevice(); /* default output device */
Chris@4 115 if (outputParameters.device == paNoDevice) {
Chris@4 116 fprintf(stderr,"Error: No default output device.\n");
Chris@4 117 goto done;
Chris@4 118 }
Chris@4 119 outputParameters.channelCount = 2; /* stereo output */
Chris@4 120 outputParameters.hostApiSpecificStreamInfo = NULL;
Chris@4 121 outputParameters.sampleFormat = paFloat32; /* 32 bit floating point output. */
Chris@4 122 /* When you change this, also */
Chris@4 123 /* adapt the callback routine! */
Chris@4 124 outputParameters.suggestedLatency = Pa_GetDeviceInfo( outputParameters.device )
Chris@4 125 ->defaultLowOutputLatency; /* Low latency. */
Chris@4 126 err = Pa_OpenStream( &stream,
Chris@4 127 NULL, /* No input. */
Chris@4 128 &outputParameters,
Chris@4 129 SAMPLE_RATE,
Chris@4 130 1024, /* frames per buffer */
Chris@4 131 flags,
Chris@4 132 sineCallback,
Chris@4 133 (void*)data );
Chris@4 134 if (err != paNoError)
Chris@4 135 goto done;
Chris@4 136
Chris@4 137 err = Pa_StartStream( stream );
Chris@4 138 if (err != paNoError)
Chris@4 139 goto done;
Chris@4 140
Chris@4 141 Pa_Sleep( NUM_SECONDS * 1000 );
Chris@4 142 printf("CPULoad = %8.6f\n", Pa_GetStreamCpuLoad(stream));
Chris@4 143
Chris@4 144 err = Pa_CloseStream( stream );
Chris@4 145 done:
Chris@4 146 Pa_Sleep( 250 ); /* Just a small silence. */
Chris@4 147 Pa_Terminate();
Chris@4 148 return err;
Chris@4 149 }
Chris@4 150
Chris@4 151
Chris@4 152 /*******************************************************************/
Chris@4 153 int main(void);
Chris@4 154 int main(void)
Chris@4 155 {
Chris@4 156 PaError err;
Chris@4 157 paTestData DATA;
Chris@4 158 int i;
Chris@4 159 float amplitude = 4.0 / (1<<15);
Chris@4 160
Chris@4 161 printf("PortAudio Test: output EXTREMELY QUIET sine wave with and without dithering.\n");
Chris@4 162 /* initialise sinusoidal wavetable */
Chris@4 163 for( i=0; i<TABLE_SIZE; i++ )
Chris@4 164 {
Chris@4 165 DATA.sine[i] = (float) sin( ((double)i/(double)TABLE_SIZE) * M_PI * 2. );
Chris@4 166 }
Chris@4 167 printf("\nNo treatment..\n"); fflush(stdout);
Chris@4 168 err = PlaySine( &DATA, paClipOff | paDitherOff, amplitude );
Chris@4 169 if( err < 0 ) goto done;
Chris@4 170
Chris@4 171 printf("\nClip..\n");
Chris@4 172 fflush(stdout);
Chris@4 173 err = PlaySine( &DATA, paDitherOff, amplitude );
Chris@4 174 if( err < 0 ) goto done;
Chris@4 175
Chris@4 176 printf("\nClip and Dither..\n");
Chris@4 177 fflush(stdout);
Chris@4 178 err = PlaySine( &DATA, paNoFlag, amplitude );
Chris@4 179 done:
Chris@4 180 if (err)
Chris@4 181 {
Chris@4 182 fprintf( stderr, "An error occured while using the portaudio stream\n" );
Chris@4 183 fprintf( stderr, "Error number: %d\n", err );
Chris@4 184 fprintf( stderr, "Error message: %s\n", Pa_GetErrorText( err ) );
Chris@4 185 err = 1; /* Though PlaySine() already called Pa_Terminate(), */
Chris@4 186 } /* we may still call Pa_GetErrorText(). */
Chris@4 187 else
Chris@4 188 printf("\n(Don't forget to turn the VOLUME DOWN after listening so carefully.)\n");
Chris@4 189 return err; /* 0 or 1. */
Chris@4 190 }