mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-11-04 16:14:00 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			396 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			396 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/** @file patest_converters.c
 | 
						|
	@ingroup test_src
 | 
						|
	@brief Tests the converter functions in pa_converters.c
 | 
						|
	@author Ross Bencina <rossb@audiomulch.com>
 | 
						|
 | 
						|
    Link with pa_dither.c and pa_converters.c
 | 
						|
 | 
						|
    see http://www.portaudio.com/trac/wiki/V19ConvertersStatus for a discussion of this.
 | 
						|
*/
 | 
						|
/*
 | 
						|
 * $Id: $
 | 
						|
 *
 | 
						|
 * This program uses the PortAudio Portable Audio Library.
 | 
						|
 * For more information see: http://www.portaudio.com/
 | 
						|
 * Copyright (c) 1999-2008 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 <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <math.h>
 | 
						|
 | 
						|
#include "portaudio.h"
 | 
						|
#include "pa_converters.h"
 | 
						|
#include "pa_dither.h"
 | 
						|
#include "pa_types.h"
 | 
						|
#include "pa_endianness.h"
 | 
						|
 | 
						|
#ifndef M_PI
 | 
						|
#define M_PI  (3.14159265)
 | 
						|
#endif
 | 
						|
 | 
						|
#define MAX_PER_CHANNEL_FRAME_COUNT     (2048)
 | 
						|
#define MAX_CHANNEL_COUNT               (8)
 | 
						|
 | 
						|
 | 
						|
#define SAMPLE_FORMAT_COUNT (6)
 | 
						|
 | 
						|
static PaSampleFormat sampleFormats_[ SAMPLE_FORMAT_COUNT ] = 
 | 
						|
    { paFloat32, paInt32, paInt24, paInt16, paInt8, paUInt8 }; /* all standard PA sample formats */
 | 
						|
 | 
						|
static const char* sampleFormatNames_[SAMPLE_FORMAT_COUNT] = 
 | 
						|
    { "paFloat32", "paInt32", "paInt24", "paInt16", "paInt8", "paUInt8" };
 | 
						|
 | 
						|
 | 
						|
static const char* abbreviatedSampleFormatNames_[SAMPLE_FORMAT_COUNT] = 
 | 
						|
    { "f32", "i32", "i24", "i16", " i8", "ui8" };
 | 
						|
 | 
						|
 | 
						|
PaError My_Pa_GetSampleSize( PaSampleFormat format );
 | 
						|
 | 
						|
/*
 | 
						|
    available flags are paClipOff and paDitherOff
 | 
						|
    clipping is usually applied for float -> int conversions
 | 
						|
    dither is usually applied for all downconversions (ie anything but 8bit->8bit conversions
 | 
						|
*/
 | 
						|
 | 
						|
static int CanClip( PaSampleFormat sourceFormat, PaSampleFormat destinationFormat )
 | 
						|
{
 | 
						|
    if( sourceFormat == paFloat32 && destinationFormat != sourceFormat )
 | 
						|
        return 1;
 | 
						|
    else
 | 
						|
        return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int CanDither( PaSampleFormat sourceFormat, PaSampleFormat destinationFormat )
 | 
						|
{
 | 
						|
    if( sourceFormat < destinationFormat && sourceFormat != paInt8 )
 | 
						|
        return 1;
 | 
						|
    else
 | 
						|
        return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void GenerateOneCycleSineReference( double *out, int frameCount, int strideFrames )
 | 
						|
{
 | 
						|
    int i;
 | 
						|
    for( i=0; i < frameCount; ++i ){
 | 
						|
        *out = sin( ((double)i/(double)frameCount) * 2. * M_PI );
 | 
						|
        out += strideFrames;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void GenerateOneCycleSine( PaSampleFormat format, void *buffer, int frameCount, int strideFrames )
 | 
						|
{
 | 
						|
    switch( format ){
 | 
						|
 | 
						|
        case paFloat32:
 | 
						|
            {
 | 
						|
                int i;
 | 
						|
                float *out = (float*)buffer;
 | 
						|
                for( i=0; i < frameCount; ++i ){
 | 
						|
                    *out = (float).9 * sin( ((double)i/(double)frameCount) * 2. * M_PI );
 | 
						|
                    out += strideFrames;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case paInt32:
 | 
						|
            {
 | 
						|
                int i;
 | 
						|
                PaInt32 *out = (PaInt32*)buffer;
 | 
						|
                for( i=0; i < frameCount; ++i ){
 | 
						|
                    *out = (PaInt32)(.9 * sin( ((double)i/(double)frameCount) * 2. * M_PI ) * 0x7FFFFFFF);
 | 
						|
                    out += strideFrames;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case paInt24:
 | 
						|
            {
 | 
						|
                int i;
 | 
						|
                unsigned char *out = (unsigned char*)buffer;
 | 
						|
                for( i=0; i < frameCount; ++i ){
 | 
						|
                    signed long temp = (PaInt32)(.9 * sin( ((double)i/(double)frameCount) * 2. * M_PI ) * 0x7FFFFFFF);
 | 
						|
                    
 | 
						|
                    #if defined(PA_LITTLE_ENDIAN)
 | 
						|
                            out[0] = (unsigned char)(temp >> 8) & 0xFF;
 | 
						|
                            out[1] = (unsigned char)(temp >> 16) & 0xFF;
 | 
						|
                            out[2] = (unsigned char)(temp >> 24) & 0xFF;
 | 
						|
                    #elif defined(PA_BIG_ENDIAN)
 | 
						|
                            out[0] = (unsigned char)(temp >> 24) & 0xFF;
 | 
						|
                            out[1] = (unsigned char)(temp >> 16) & 0xFF;
 | 
						|
                            out[2] = (unsigned char)(temp >> 8) & 0xFF;
 | 
						|
                    #endif
 | 
						|
                    out += 3;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case paInt16:
 | 
						|
            {
 | 
						|
                int i;
 | 
						|
                PaInt16 *out = (PaInt16*)buffer;
 | 
						|
                for( i=0; i < frameCount; ++i ){
 | 
						|
                    *out = (PaInt16)(.9 * sin( ((double)i/(double)frameCount) * 2. * M_PI ) * 0x7FFF );
 | 
						|
                    out += strideFrames;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case paInt8:
 | 
						|
            {
 | 
						|
                int i;
 | 
						|
                signed char *out = (signed char*)buffer;
 | 
						|
                for( i=0; i < frameCount; ++i ){
 | 
						|
                    *out = (signed char)(.9 * sin( ((double)i/(double)frameCount) * 2. * M_PI ) * 0x7F );
 | 
						|
                    out += strideFrames;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case paUInt8:
 | 
						|
            {
 | 
						|
                int i;
 | 
						|
                unsigned char *out = (unsigned char*)buffer;
 | 
						|
                for( i=0; i < frameCount; ++i ){
 | 
						|
                    *out = (unsigned char)( .5 * (1. + (.9 * sin( ((double)i/(double)frameCount) * 2. * M_PI ))) * 0xFF  );
 | 
						|
                    out += strideFrames;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
int TestNonZeroPresent( void *buffer, int size )
 | 
						|
{
 | 
						|
    char *p = (char*)buffer;
 | 
						|
    int i;
 | 
						|
 | 
						|
    for( i=0; i < size; ++i ){
 | 
						|
    
 | 
						|
        if( *p != 0 )
 | 
						|
            return 1;
 | 
						|
        ++p;
 | 
						|
    }   
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
float MaximumAbsDifference( float* sourceBuffer, float* referenceBuffer, int count )
 | 
						|
{
 | 
						|
    float result = 0;
 | 
						|
    float difference;
 | 
						|
    while( count-- ){
 | 
						|
        difference = fabs( *sourceBuffer++ - *referenceBuffer++ );
 | 
						|
        if( difference > result )
 | 
						|
            result = difference;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}  
 | 
						|
 | 
						|
int main( const char **argv, int argc )
 | 
						|
{
 | 
						|
    PaUtilTriangularDitherGenerator ditherState;
 | 
						|
    PaUtilConverter *converter;
 | 
						|
    void *destinationBuffer, *sourceBuffer;
 | 
						|
    double *referenceBuffer;
 | 
						|
    int sourceFormatIndex, destinationFormatIndex;
 | 
						|
    PaSampleFormat sourceFormat, destinationFormat;
 | 
						|
    PaStreamFlags flags;
 | 
						|
    int passFailMatrix[SAMPLE_FORMAT_COUNT][SAMPLE_FORMAT_COUNT]; // [source][destination]
 | 
						|
    float noiseAmplitudeMatrix[SAMPLE_FORMAT_COUNT][SAMPLE_FORMAT_COUNT]; // [source][destination]
 | 
						|
    float amp;
 | 
						|
 | 
						|
#define FLAG_COMBINATION_COUNT (4)
 | 
						|
    PaStreamFlags flagCombinations[FLAG_COMBINATION_COUNT] = { paNoFlag, paClipOff, paDitherOff, paClipOff | paDitherOff };
 | 
						|
    const char *flagCombinationNames[FLAG_COMBINATION_COUNT] = { "paNoFlag", "paClipOff", "paDitherOff", "paClipOff | paDitherOff" };
 | 
						|
    int flagCombinationIndex;
 | 
						|
 | 
						|
    PaUtil_InitializeTriangularDitherState( &ditherState );
 | 
						|
 | 
						|
    /* allocate more than enough space, we use sizeof(float) but we need to fit any 32 bit datum */
 | 
						|
 | 
						|
    destinationBuffer = (void*)malloc( MAX_PER_CHANNEL_FRAME_COUNT * MAX_CHANNEL_COUNT * sizeof(float) );
 | 
						|
    sourceBuffer = (void*)malloc( MAX_PER_CHANNEL_FRAME_COUNT * MAX_CHANNEL_COUNT * sizeof(float) );
 | 
						|
    referenceBuffer = (void*)malloc( MAX_PER_CHANNEL_FRAME_COUNT * MAX_CHANNEL_COUNT * sizeof(float) );
 | 
						|
 | 
						|
 | 
						|
    /* the first round of tests simply iterates through the buffer combinations testing
 | 
						|
        that putting something in gives something out */
 | 
						|
 | 
						|
    printf( "= Sine wave in, something out =\n" );
 | 
						|
 | 
						|
    printf( "\n" );
 | 
						|
 | 
						|
    GenerateOneCycleSine( paFloat32, referenceBuffer, MAX_PER_CHANNEL_FRAME_COUNT, 1 );
 | 
						|
 | 
						|
    for( flagCombinationIndex = 0; flagCombinationIndex < FLAG_COMBINATION_COUNT; ++flagCombinationIndex ){
 | 
						|
        flags = flagCombinations[flagCombinationIndex];
 | 
						|
 | 
						|
        printf( "\n" );
 | 
						|
        printf( "== flags = %s ==\n", flagCombinationNames[flagCombinationIndex] );
 | 
						|
 | 
						|
        for( sourceFormatIndex = 0; sourceFormatIndex < SAMPLE_FORMAT_COUNT; ++sourceFormatIndex ){
 | 
						|
            for( destinationFormatIndex = 0; destinationFormatIndex < SAMPLE_FORMAT_COUNT; ++destinationFormatIndex ){
 | 
						|
                sourceFormat = sampleFormats_[sourceFormatIndex];
 | 
						|
                destinationFormat = sampleFormats_[destinationFormatIndex];
 | 
						|
                //printf( "%s -> %s ", sampleFormatNames_[ sourceFormatIndex ], sampleFormatNames_[ destinationFormatIndex ] );
 | 
						|
 | 
						|
                converter = PaUtil_SelectConverter( sourceFormat, destinationFormat, flags );
 | 
						|
 | 
						|
                /* source is a sinewave */
 | 
						|
                GenerateOneCycleSine( sourceFormat, sourceBuffer, MAX_PER_CHANNEL_FRAME_COUNT, 1 );
 | 
						|
 | 
						|
                /* zero destination */
 | 
						|
                memset( destinationBuffer, 0, MAX_PER_CHANNEL_FRAME_COUNT * My_Pa_GetSampleSize( destinationFormat ) );
 | 
						|
 | 
						|
                (*converter)( destinationBuffer, 1, sourceBuffer, 1, MAX_PER_CHANNEL_FRAME_COUNT, &ditherState );
 | 
						|
 | 
						|
    /*
 | 
						|
    Other ways we could test this would be:
 | 
						|
        - pass a constant, check for a constant (wouldn't work with dither)
 | 
						|
        - pass alternating +/-, check for the same...
 | 
						|
    */
 | 
						|
                if( TestNonZeroPresent( destinationBuffer, MAX_PER_CHANNEL_FRAME_COUNT * My_Pa_GetSampleSize( destinationFormat ) ) ){
 | 
						|
                    //printf( "PASSED\n" );
 | 
						|
                    passFailMatrix[sourceFormatIndex][destinationFormatIndex] = 1;
 | 
						|
                }else{
 | 
						|
                    //printf( "FAILED\n" );
 | 
						|
                    passFailMatrix[sourceFormatIndex][destinationFormatIndex] = 0;
 | 
						|
                }
 | 
						|
 | 
						|
                
 | 
						|
                /* try to measure the noise floor (comparing output signal to a float32 sine wave) */
 | 
						|
 | 
						|
                if( passFailMatrix[sourceFormatIndex][destinationFormatIndex] ){
 | 
						|
 | 
						|
                    /* convert destination back to paFloat32 into source */
 | 
						|
                    converter = PaUtil_SelectConverter( destinationFormat, paFloat32, paNoFlag );
 | 
						|
 | 
						|
                    memset( sourceBuffer, 0, MAX_PER_CHANNEL_FRAME_COUNT * My_Pa_GetSampleSize( paFloat32 ) );
 | 
						|
                    (*converter)( sourceBuffer, 1, destinationBuffer, 1, MAX_PER_CHANNEL_FRAME_COUNT, &ditherState );
 | 
						|
 | 
						|
                    if( TestNonZeroPresent( sourceBuffer, MAX_PER_CHANNEL_FRAME_COUNT * My_Pa_GetSampleSize( paFloat32 ) ) ){
 | 
						|
    
 | 
						|
                        noiseAmplitudeMatrix[sourceFormatIndex][destinationFormatIndex] = MaximumAbsDifference( (float*)sourceBuffer, (float*)referenceBuffer, MAX_PER_CHANNEL_FRAME_COUNT );
 | 
						|
                        
 | 
						|
                    }else{
 | 
						|
                        /* can't test noise floor because there is no conversion from dest format to float available */
 | 
						|
                        noiseAmplitudeMatrix[sourceFormatIndex][destinationFormatIndex] = -1; // mark as failed
 | 
						|
                    }
 | 
						|
                }else{
 | 
						|
                    noiseAmplitudeMatrix[sourceFormatIndex][destinationFormatIndex] = -1; // mark as failed
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        printf( "\n" );
 | 
						|
        printf( "=== Output contains non-zero data ===\n" );
 | 
						|
        printf( "Key: . - pass, X - fail\n" );
 | 
						|
        printf( "{{{\n" ); // trac preformated text tag
 | 
						|
        printf( "in|  out:    " );
 | 
						|
        for( destinationFormatIndex = 0; destinationFormatIndex < SAMPLE_FORMAT_COUNT; ++destinationFormatIndex ){
 | 
						|
            printf( "  %s   ", abbreviatedSampleFormatNames_[destinationFormatIndex] );
 | 
						|
        }
 | 
						|
        printf( "\n" );
 | 
						|
 | 
						|
        for( sourceFormatIndex = 0; sourceFormatIndex < SAMPLE_FORMAT_COUNT; ++sourceFormatIndex ){
 | 
						|
            printf( "%s         ", abbreviatedSampleFormatNames_[sourceFormatIndex] );
 | 
						|
            for( destinationFormatIndex = 0; destinationFormatIndex < SAMPLE_FORMAT_COUNT; ++destinationFormatIndex ){
 | 
						|
                printf( "   %s   ", (passFailMatrix[sourceFormatIndex][destinationFormatIndex])? " ." : " X" );
 | 
						|
            }
 | 
						|
            printf( "\n" );
 | 
						|
        }
 | 
						|
        printf( "}}}\n" ); // trac preformated text tag
 | 
						|
 | 
						|
        printf( "\n" );
 | 
						|
        printf( "=== Combined dynamic range (src->dest->float32) ===\n" );
 | 
						|
        printf( "Key: Noise amplitude in dBfs, X - fail (either above failed or dest->float32 failed)\n" );
 | 
						|
        printf( "{{{\n" ); // trac preformated text tag
 | 
						|
        printf( "in|  out:    " );
 | 
						|
        for( destinationFormatIndex = 0; destinationFormatIndex < SAMPLE_FORMAT_COUNT; ++destinationFormatIndex ){
 | 
						|
            printf( "  %s   ", abbreviatedSampleFormatNames_[destinationFormatIndex] );
 | 
						|
        }
 | 
						|
        printf( "\n" );
 | 
						|
 | 
						|
        for( sourceFormatIndex = 0; sourceFormatIndex < SAMPLE_FORMAT_COUNT; ++sourceFormatIndex ){
 | 
						|
            printf( " %s       ", abbreviatedSampleFormatNames_[sourceFormatIndex] );
 | 
						|
            for( destinationFormatIndex = 0; destinationFormatIndex < SAMPLE_FORMAT_COUNT; ++destinationFormatIndex ){
 | 
						|
                amp = noiseAmplitudeMatrix[sourceFormatIndex][destinationFormatIndex];
 | 
						|
                if( amp < 0. )
 | 
						|
                    printf( "    X   " );
 | 
						|
                else
 | 
						|
                    printf( " % 6.1f ", 20.*log10(amp) );
 | 
						|
            }
 | 
						|
            printf( "\n" );
 | 
						|
        }
 | 
						|
        printf( "}}}\n" ); // trac preformated text tag
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    free( destinationBuffer );
 | 
						|
    free( sourceBuffer );
 | 
						|
    free( referenceBuffer );
 | 
						|
}
 | 
						|
 | 
						|
// copied here for now otherwise we need to include the world just for this function.
 | 
						|
PaError My_Pa_GetSampleSize( PaSampleFormat format )
 | 
						|
{
 | 
						|
    int result;
 | 
						|
 | 
						|
    switch( format & ~paNonInterleaved )
 | 
						|
    {
 | 
						|
 | 
						|
    case paUInt8:
 | 
						|
    case paInt8:
 | 
						|
        result = 1;
 | 
						|
        break;
 | 
						|
 | 
						|
    case paInt16:
 | 
						|
        result = 2;
 | 
						|
        break;
 | 
						|
 | 
						|
    case paInt24:
 | 
						|
        result = 3;
 | 
						|
        break;
 | 
						|
 | 
						|
    case paFloat32:
 | 
						|
    case paInt32:
 | 
						|
        result = 4;
 | 
						|
        break;
 | 
						|
 | 
						|
    default:
 | 
						|
        result = paSampleFormatNotSupported;
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    return (PaError) result;
 | 
						|
}
 |