C — одновременный ввод и вывод звука с использованием portaudio и функции обратного вызова

#c #audio #callback #real-time #portaudio

#c #Аудио #обратный вызов #в режиме реального времени #portaudio

Вопрос:

Я пытаюсь создать программу для полнодуплексного аудио, которая обнаруживает аудиовход с микрофона и направляет его на динамик в режиме реального времени, используя библиотеку portaudio и функцию обратного вызова, но я продолжаю получать ошибки. Я попытался открыть полный поток ввода-вывода, затем начать чтение с микрофона и отправить захваченные данные в функцию обратного вызова, которая запустит потоковую передачу звука на динамик, Может кто-нибудь помочь мне с этим?? Вот ошибки:

 ALSA lib pcm.c:2495:(snd_pcm_open_noupdate) Unknown PCM cards.pcm.rear
ALSA lib pcm.c:2495:(snd_pcm_open_noupdate) Unknown PCM cards.pcm.center_lfe
ALSA lib pcm.c:2495:(snd_pcm_open_noupdate) Unknown PCM cards.pcm.side
ALSA lib pcm_route.c:867:(find_matching_chmap) Found no matching channel map
ALSA lib pcm_route.c:867:(find_matching_chmap) Found no matching channel map
ALSA lib pcm_route.c:867:(find_matching_chmap) Found no matching channel map
ALSA lib pcm_route.c:867:(find_matching_chmap) Found no matching channel map
Output device # 14.
Output LL: 0.00870748 s
Output HL: 0.0348299 s
Now recording!!
An error occured while using the portaudio stream
Error number: -9988
Error message: Invalid stream pointer
An error occured while using the portaudio stream
Error number: -9988
Error message: Invalid stream pointer
An error occured while using the portaudio stream
Error number: -9977
Error message: Can't read from a callback stream
  

и это мой код:

 
#include "diffusion.c"
#include "confusion.c"
#include "portaudio.h"
#include "wave_file.c"

/* #define SAMPLE_RATE  (17932) // Test failure to open with this value. */
#define SAMPLE_RATE  (44100)
#define FRAMES_PER_BUFFER (0)
#define NUM_SECONDS     (2)
#define NUM_CHANNELS    (2)
/* #define DITHER_FLAG     (paDitherOff)  */
#define DITHER_FLAG     (0) /**/

#define WRITE_TO_WAVE_FILE  (1)

/* define sample format. */

#define PA_SAMPLE_TYPE  paInt16
typedef short SAMPLE;
#define SAMPLE_SILENCE  (0)
#define PRINTF_S_FORMAT "%d"


PaStreamParameters inputParameters, outputParameters;
    PaStream *stream;
    PaError err;
    SAMPLE *recordedSamples, *recordedSamples2;
    int i;
    int totalFrames;
    int numSamples;
    int numBytes;

static int playCallback (const void *input,
                                      void *output,
                                      unsigned long frameCount,
                                      const PaStreamCallbackTimeInfo* timeInfo,
                                      PaStreamCallbackFlags statusFlags,
                                      void *userData
 ) 
{


    err = Pa_StartStream( stream );
    if( err != paNoError ) goto error;
    printf("Now recording!!n"); fflush(stdout);

    err = Pa_WriteStream( stream, userData, totalFrames );
    if( err != paNoError ) goto error;
    
    err = Pa_CloseStream( stream );
    if( err != paNoError ) goto error;
    free( userData );
    return paContinue;

error:
    fprintf( stderr, "An error occured while using the portaudio streamn" );
    fprintf( stderr, "Error number: %dn", err );
    fprintf( stderr, "Error message: %sn", Pa_GetErrorText( err ) );
    return paContinue;
}





int main(void);
int main(void)
{
    PaStreamParameters inputParameters, outputParameters;
    PaStream *stream;
    PaError err;
    SAMPLE *recordedSamples, *recordedSamples2;
    int i;
    int totalFrames;
    int numSamples;
    int numBytes;
    
    printf("patest_read_record.cn"); fflush(stdout);
    totalFrames = NUM_SECONDS * SAMPLE_RATE; /* Record for a few seconds. */
    numSamples = totalFrames * NUM_CHANNELS;

    numBytes = numSamples * sizeof(SAMPLE);
    recordedSamples = (SAMPLE *) malloc( numBytes );
    if( recordedSamples == NULL )
    {
        printf("Could not allocate record array.n");
        exit(1);
    }
    for( i=0; i<numSamples; i   ) recordedSamples[i] = 0;

    err = Pa_Initialize();
    if( err != paNoError ) goto error;

    inputParameters.device = Pa_GetDefaultInputDevice(); /* default input device */
    if (inputParameters.device == paNoDevice) {
      fprintf(stderr,"Error: No default input device.n");
      goto error;
    }
    inputParameters.channelCount = NUM_CHANNELS;
    inputParameters.sampleFormat = PA_SAMPLE_TYPE;
    inputParameters.suggestedLatency = Pa_GetDeviceInfo( inputParameters.device )->defaultLowInputLatency;
    inputParameters.hostApiSpecificStreamInfo = NULL;

    outputParameters.device = Pa_GetDefaultOutputDevice(); /* default output device */
    printf( "Output device # %d.n", outputParameters.device );
    printf( "Output LL: %g sn", Pa_GetDeviceInfo( outputParameters.device )->defaultLowOutputLatency );
    printf( "Output HL: %g sn", Pa_GetDeviceInfo( outputParameters.device )->defaultHighOutputLatency );
    outputParameters.channelCount = NUM_CHANNELS;
    outputParameters.sampleFormat = PA_SAMPLE_TYPE;
    outputParameters.suggestedLatency = Pa_GetDeviceInfo( outputParameters.device )->defaultHighOutputLatency;
    outputParameters.hostApiSpecificStreamInfo = NULL;

    /* Record some audio. -------------------------------------------- */
    err = Pa_OpenStream(
              amp;stream,
              amp;inputParameters,
              amp;outputParameters,                  /* amp;outputParameters, */
              SAMPLE_RATE,
              paFramesPerBufferUnspecified,
              paClipOff,      /* we won't output out of range samples so don't bother clipping them */
              playCallback, /* no callback, use blocking API */
              recordedSamples ); /* no callback, so no callback userData */
    if( err != paNoError ) goto error;

    err = Pa_StartStream( stream );
    if( err != paNoError ) goto error;
    printf("Now recording!!n"); fflush(stdout);

    err = Pa_ReadStream( stream, recordedSamples, totalFrames );
    if( err != paNoError ) goto error;
    
    err = Pa_CloseStream( stream );
    if( err != paNoError ) goto error;
     /* save the recorded data in a wave file " recording.wav ".  -------------------------- */
    saveInWaveFile(recordedSamples, totalFrames, "recording.wav");
    for (int i =0; i< numSamples ; i  ){
        recordedSamples2[i] = recordedSamples[i];
    }

    free( recordedSamples );
    Pa_Terminate();
    return 0;

error:
    Pa_Terminate();
    fprintf( stderr, "An error occured while using the portaudio streamn" );
    fprintf( stderr, "Error number: %dn", err );
    fprintf( stderr, "Error message: %sn", Pa_GetErrorText( err ) );
    return -1;
}

  

Ответ №1:

Сначала убедитесь, что устройства ввода и вывода по умолчанию установлены правильно. Вы можете сделать это, сначала запустив пример pa_devs.c в папке examples, чтобы распечатать список разрешенных аудиоустройств, подключенных к компьютеру. Как только вы определили правильные устройства, замените строки номером устройства, например

 inputParameters.device = Pa_GetDefaultInputDevice();
  

для

 inputParameters.device = <device number (e.g. 11)>;
  

и

 outputParameters.device = Pa_GetDefaultOutputDevice();
  

для

 outputParameters.device = <device number (e.g. 11)>;
  

Другой способ, который вы можете попробовать, — это пример с именем paex_read_write_wire.c. Это должно обеспечить ту же функциональность, о которой вы говорите, однако использует блокирующий подход, а не функцию обратного вызова, которую вы использовали в своем коде.

Ответ №2:

Я не могу протестировать это сейчас, но он должен воспроизводить (контролировать) входной буфер (микрофон):

 static int patestCallback( const void *inputBuffer, void *outputBuffer,
                       unsigned long framesPerBuffer,
                       const PaStreamCallbackTimeInfo* timeInfo,
                       PaStreamCallbackFlags statusFlags,
                       void *userData )
{
    paTestData *data = (paTestData*)userData; 
    float *out = (float*)outputBuffer;
    unsigned int i;
    float *in = (float*)inputBuffer;

    for( i=0; i<framesPerBuffer; i   )
    {
         out   = in  ;  /* left */
         out   = in  ;  /* right*/
    }
    return 0;
}