Skip to content

orazdow/PortAudio-Wrapper

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

28 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

PortAudio Wrapper

C++ wrapper for PortAudio

Provides a simple interface to PortAudio's callback API for reading and writing to audio devices.


Basic Example:

#include "pa.h"
#include "math.h"
#define TWO_PI 6.2831853

double phase = 0, step = TWO_PI*440/44100.0;

void paFunc(const float* in, float* out, long frames, void* data){    
    // play a 440Hz sine tone
    for(int i = 0; i < frames; i++ ){
         *out++ = sin(phase)*0.5;
         phase += step;
     }
}

int main() {
    // use default parameters, not passing any data
    Pa a(paFunc, NULL);
    // start stream
    a.start(Pa::waitForKey);
    return 0;
}

Usage:

Requirements:

PortAudio can be obtained here.

Initializing:

Define a callback where audio processing will occur and pass it as the constructor's first argument. The callback can be a simplified type:

void paCallback(const float* inputBuffer, float* ouputBuffer, long framesPerBuffer, void* userData){...}

or have the full signature specified by PortAudio

int paCallback(const void* inputBuffer, void* outputBuffer, unsigned long framesPerBuffer, 
        const PaStreamCallbackTimeInfo* timeInfo, PaStreamCallbackFlags statusFlags, void* userData){...}

The wrapper's constructor can either take 2 arguments:  the callback and a pointer for passing user data (see passing data):

Pa a(callback, &data);

or 6 arguments:  callback, input channels, output channels, sampling rate, frames per buffer (0 will autoselect), userData:

Pa a(callback, 2, 2, 44100, 256, &data);

The 2-argument constructor is equivalent to:

Pa a(callback, 0, 1, 44100, 0, &data);


Starting / stopping the audio stream:

Calling start() will start the stream.

For convenience, there are four enum values that can be passed to start() to control the lifetime of the stream:

  • Pa::dontTerminate  (default) Starts the stream only.  Will stop if stop() is called, or the program exits.

  • Pa::wait  Keeps the stream alive in a while loop, can be stopped from another thread.

  • Pa::sleep  Keeps the stream alive for a specified number of milliseconds and then stops the stream. setSleepTime(millis) can be called to set the time, or a time in milliseconds can be passed directly i.e start(5000)

  • Pa::waitForKey  Keeps the stream alive until a key is entered and then stops the stream.

stop()  stops the stream.  stop(true)  also closes the stream, which frees some additional resources.

PortAudio will be terminated when the wrapper object is destroyed, but can be explicitly terminated by calling terminate()


Passing data:

PortAudio allows for user data to be passed to the audio callback. This can be omitted in favor of global variables. NULL can be passed for the user data parameter in this case.

To pass data to the callback, pass a data pointer to the constructor, and recast the data argument in your callback to your data type. For example:

void paFunc(const float* in, float* out, long frames, void* data){    
    Osc* o = (Osc*)data;
    for(long i = 0; i < frames; i++ ){
         *out++ = o->out();
     }
}
    
Osc osc(440);
Pa a(paFunc, &osc);

PortAudio provides the option to register a callback that will be called when a stream has stopped playing which takes the user data as an argument. This can be used to delete dynamically allocated data, and can be set with setFinishedCallBack().
An example might look like:

void streamFinished(void* data){
    Osc* o = (Osc*)data;
    delete o;
}

int main() {
    Osc* osc = new Osc(440);
    Pa a(paFunc, osc);
    a.setFinishedCallBack(streamFinished);
    a.start(Pa::waitForKey);
    return 0;
}

Alternatively, you can pass the data as a shared_ptr, which will delete resources automatically:

Osc* osc = new Osc(440);
Pa a(paFunc, std::shared_ptr<Osc>(osc));

Querying / setting audio devices:

listDevices() prints a list of available input and output devices, sorted by device ID.

To set an input or output device, enter a device ID with setInputDevice(ID) or setOutputDevice(ID) respectively.

getDeviceInfo(ID) prints additional information about a device.


Reference:

Constructors

Pa(callBack, userData)  

Pa(callBack, inChannels, outChannels, SampleRate, FramesPerBuffer, userData)

Functions

void start()

void start(RunMode mode)

void start(unsigned long sleepTime)

void stop()

void stop(bool close)

void terminate()

void setSleepTime(unsigned long time)

void setSampleFormat(PaSampleFormat format)

void setFinishedCallBack(void(*callback)(void* data))

void listDevices()

void getDeviceInfo(unsigned int index)

void setInputDevice(unsigned int index)

void setOutputDevice(unsigned int index)

Typedefs

void miniCallBack(const float*, float*, long, void*)

int mainCallBack(const void*, void*, unsigned long, const PaStreamCallbackTimeInfo*, PaStreamCallbackFlags, void*)

Enums

RunMode {dontTerminate, wait, sleep, waitForKey}

Releases

No releases published

Packages

No packages published