2014-03-11 21:46:00 +01:00
|
|
|
/**
|
|
|
|
* @author Gary P. SCAVONE
|
|
|
|
*
|
|
|
|
* @copyright 2001-2013 Gary P. Scavone, all right reserved
|
|
|
|
*
|
|
|
|
* @license like MIT (see license file)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __AIRTAUDIO_RTAUDIO_H__
|
|
|
|
#define __AIRTAUDIO_RTAUDIO_H__
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
#include <airtaudio/base.h>
|
|
|
|
#include <airtaudio/int24_t.h>
|
|
|
|
#include <airtaudio/CallbackInfo.h>
|
|
|
|
#include <airtaudio/Api.h>
|
|
|
|
#include <airtaudio/api/Alsa.h>
|
2014-03-13 21:16:30 +01:00
|
|
|
#include <airtaudio/api/Android.h>
|
2014-03-11 21:46:00 +01:00
|
|
|
#include <airtaudio/api/Asio.h>
|
|
|
|
#include <airtaudio/api/Core.h>
|
|
|
|
#include <airtaudio/api/Ds.h>
|
|
|
|
#include <airtaudio/api/Dummy.h>
|
|
|
|
#include <airtaudio/api/Jack.h>
|
|
|
|
#include <airtaudio/api/Oss.h>
|
|
|
|
#include <airtaudio/api/Pulse.h>
|
|
|
|
namespace airtaudio {
|
|
|
|
/**
|
|
|
|
* @brief airtaudio::Interface class declaration.
|
|
|
|
*
|
|
|
|
* airtaudio::Interface is a "controller" used to select an available audio i/o
|
|
|
|
* interface. It presents a common API for the user to call but all
|
|
|
|
* functionality is implemented by the class RtApi and its
|
|
|
|
* subclasses. RtAudio creates an instance of an RtApi subclass
|
|
|
|
* based on the user's API choice. If no choice is made, RtAudio
|
|
|
|
* attempts to make a "logical" API selection.
|
|
|
|
*/
|
|
|
|
class Interface {
|
2014-03-11 22:37:22 +01:00
|
|
|
protected:
|
|
|
|
std::vector<std::pair<airtaudio::api::type, Api* (*)(void)>> m_apiAvaillable;
|
2014-03-11 21:46:00 +01:00
|
|
|
protected:
|
|
|
|
airtaudio::Api *m_rtapi;
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* @brief A static function to determine the current airtaudio version.
|
|
|
|
*/
|
|
|
|
static std::string getVersion(void) {
|
|
|
|
return airtaudio::VERSION;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief A static function to determine the available compiled audio APIs.
|
|
|
|
*
|
|
|
|
* The values returned in the std::vector can be compared against
|
|
|
|
* the enumerated list values. Note that there can be more than one
|
|
|
|
* API compiled for certain operating systems.
|
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
std::vector<airtaudio::api::type> getCompiledApi(void);
|
2014-03-11 21:46:00 +01:00
|
|
|
/**
|
|
|
|
* @brief The class constructor.
|
2014-03-11 22:37:22 +01:00
|
|
|
* @note the creating of the basic instance is done by Instanciate
|
2014-03-11 21:46:00 +01:00
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
Interface(void);
|
2014-03-11 21:46:00 +01:00
|
|
|
/**
|
|
|
|
* @brief The destructor.
|
|
|
|
*
|
|
|
|
* If a stream is running or open, it will be stopped and closed
|
|
|
|
* automatically.
|
|
|
|
*/
|
|
|
|
~Interface(void);
|
2014-03-11 22:37:22 +01:00
|
|
|
/**
|
|
|
|
* @brief Add an interface of the Possible List.
|
|
|
|
* @param[in] _api Type of the interface.
|
|
|
|
* @param[in] _callbackCreate API creation callback.
|
|
|
|
*/
|
|
|
|
void addInterface(airtaudio::api::type _api, Api* (*_callbackCreate)(void));
|
|
|
|
/**
|
|
|
|
* @brief Create an interface instance
|
|
|
|
*/
|
|
|
|
enum airtaudio::errorType instanciate(airtaudio::api::type _api = airtaudio::api::UNSPECIFIED);
|
2014-03-11 21:46:00 +01:00
|
|
|
/**
|
|
|
|
* @return the audio API specifier for the current instance of airtaudio.
|
|
|
|
*/
|
|
|
|
airtaudio::api::type getCurrentApi(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return airtaudio::api::UNSPECIFIED;
|
|
|
|
}
|
|
|
|
return m_rtapi->getCurrentApi();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief A public function that queries for the number of audio devices available.
|
|
|
|
*
|
|
|
|
* This function performs a system query of available devices each time it
|
|
|
|
* is called, thus supporting devices connected \e after instantiation. If
|
|
|
|
* a system error occurs during processing, a warning will be issued.
|
|
|
|
*/
|
|
|
|
uint32_t getDeviceCount(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return m_rtapi->getDeviceCount();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Any device integer between 0 and getDeviceCount() - 1 is valid.
|
|
|
|
* If an invalid argument is provided, an RtError (type = INVALID_USE)
|
|
|
|
* will be thrown. If a device is busy or otherwise unavailable, the
|
|
|
|
* structure member "probed" will have a value of "false" and all
|
|
|
|
* other members are undefined. If the specified device is the
|
|
|
|
* current default input or output device, the corresponding
|
|
|
|
* "isDefault" member will have a value of "true".
|
|
|
|
*
|
|
|
|
* @return An airtaudio::DeviceInfo structure for a specified device number.
|
|
|
|
*/
|
|
|
|
airtaudio::DeviceInfo getDeviceInfo(uint32_t _device) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return airtaudio::DeviceInfo();
|
|
|
|
}
|
|
|
|
return m_rtapi->getDeviceInfo(_device);
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief A function that returns the index of the default output device.
|
|
|
|
*
|
|
|
|
* If the underlying audio API does not provide a "default
|
|
|
|
* device", or if no devices are available, the return value will be
|
|
|
|
* 0. Note that this is a valid device identifier and it is the
|
|
|
|
* client's responsibility to verify that a device is available
|
|
|
|
* before attempting to open a stream.
|
|
|
|
*/
|
|
|
|
uint32_t getDefaultOutputDevice(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return m_rtapi->getDefaultOutputDevice();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief A function that returns the index of the default input device.
|
|
|
|
*
|
|
|
|
* If the underlying audio API does not provide a "default
|
|
|
|
* device", or if no devices are available, the return value will be
|
|
|
|
* 0. Note that this is a valid device identifier and it is the
|
|
|
|
* client's responsibility to verify that a device is available
|
|
|
|
* before attempting to open a stream.
|
|
|
|
*/
|
|
|
|
uint32_t getDefaultInputDevice(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return m_rtapi->getDefaultInputDevice();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief A public function for opening a stream with the specified parameters.
|
|
|
|
*
|
|
|
|
* An RtError (type = SYSTEM_ERROR) is thrown if a stream cannot be
|
|
|
|
* opened with the specified parameters or an error occurs during
|
|
|
|
* processing. An RtError (type = INVALID_USE) is thrown if any
|
|
|
|
* invalid device ID or channel number parameters are specified.
|
|
|
|
* @param _outputParameters Specifies output stream parameters to use
|
|
|
|
* when opening a stream, including a device ID, number of channels,
|
|
|
|
* and starting channel number. For input-only streams, this
|
|
|
|
* argument should be NULL. The device ID is an index value between
|
|
|
|
* 0 and getDeviceCount() - 1.
|
|
|
|
* @param _inputParameters Specifies input stream parameters to use
|
|
|
|
* when opening a stream, including a device ID, number of channels,
|
|
|
|
* and starting channel number. For output-only streams, this
|
|
|
|
* argument should be NULL. The device ID is an index value between
|
|
|
|
* 0 and getDeviceCount() - 1.
|
|
|
|
* @param _format An airtaudio::format specifying the desired sample data format.
|
|
|
|
* @param _sampleRate The desired sample rate (sample frames per second).
|
|
|
|
* @param *_bufferFrames A pointer to a value indicating the desired
|
|
|
|
* internal buffer size in sample frames. The actual value
|
|
|
|
* used by the device is returned via the same pointer. A
|
|
|
|
* value of zero can be specified, in which case the lowest
|
|
|
|
* allowable value is determined.
|
|
|
|
* @param _callback A client-defined function that will be invoked
|
|
|
|
* when input data is available and/or output data is needed.
|
|
|
|
* @param _userData An optional pointer to data that can be accessed
|
|
|
|
* from within the callback function.
|
|
|
|
* @param _options An optional pointer to a structure containing various
|
|
|
|
* global stream options, including a list of OR'ed airtaudio::streamFlags
|
|
|
|
* and a suggested number of stream buffers that can be used to
|
|
|
|
* control stream latency. More buffers typically result in more
|
|
|
|
* robust performance, though at a cost of greater latency. If a
|
|
|
|
* value of zero is specified, a system-specific median value is
|
|
|
|
* chosen. If the airtaudio_MINIMIZE_LATENCY flag bit is set, the
|
|
|
|
* lowest allowable value is used. The actual value used is
|
|
|
|
* returned via the structure argument. The parameter is API dependent.
|
|
|
|
* @param _errorCallback A client-defined function that will be invoked
|
|
|
|
* when an error has occured.
|
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
enum airtaudio::errorType openStream(airtaudio::StreamParameters *_outputParameters,
|
|
|
|
airtaudio::StreamParameters *_inputParameters,
|
|
|
|
airtaudio::format _format,
|
|
|
|
uint32_t _sampleRate,
|
|
|
|
uint32_t *_bufferFrames,
|
|
|
|
airtaudio::AirTAudioCallback _callback,
|
|
|
|
void *_userData = NULL,
|
2014-03-12 23:55:49 +01:00
|
|
|
airtaudio::StreamOptions *_options = NULL);
|
2014-03-11 21:46:00 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief A function that closes a stream and frees any associated stream memory.
|
|
|
|
*
|
|
|
|
* If a stream is not open, this function issues a warning and
|
|
|
|
* returns (no exception is thrown).
|
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
enum airtaudio::errorType closeStream(void) {
|
2014-03-11 21:46:00 +01:00
|
|
|
if (m_rtapi == NULL) {
|
2014-03-12 23:55:49 +01:00
|
|
|
return airtaudio::errorInputNull;
|
2014-03-11 21:46:00 +01:00
|
|
|
}
|
|
|
|
return m_rtapi->closeStream();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief A function that starts a stream.
|
|
|
|
*
|
|
|
|
* An RtError (type = SYSTEM_ERROR) is thrown if an error occurs
|
|
|
|
* during processing. An RtError (type = INVALID_USE) is thrown if a
|
|
|
|
* stream is not open. A warning is issued if the stream is already
|
|
|
|
* running.
|
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
enum airtaudio::errorType startStream(void) {
|
2014-03-11 21:46:00 +01:00
|
|
|
if (m_rtapi == NULL) {
|
2014-03-12 23:55:49 +01:00
|
|
|
return airtaudio::errorInputNull;
|
2014-03-11 21:46:00 +01:00
|
|
|
}
|
|
|
|
return m_rtapi->startStream();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Stop a stream, allowing any samples remaining in the output queue to be played.
|
|
|
|
*
|
|
|
|
* An RtError (type = SYSTEM_ERROR) is thrown if an error occurs
|
|
|
|
* during processing. An RtError (type = INVALID_USE) is thrown if a
|
|
|
|
* stream is not open. A warning is issued if the stream is already
|
|
|
|
* stopped.
|
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
enum airtaudio::errorType stopStream(void) {
|
2014-03-11 21:46:00 +01:00
|
|
|
if (m_rtapi == NULL) {
|
2014-03-12 23:55:49 +01:00
|
|
|
return airtaudio::errorInputNull;
|
2014-03-11 21:46:00 +01:00
|
|
|
}
|
|
|
|
return m_rtapi->stopStream();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Stop a stream, discarding any samples remaining in the input/output queue.
|
|
|
|
* An RtError (type = SYSTEM_ERROR) is thrown if an error occurs
|
|
|
|
* during processing. An RtError (type = INVALID_USE) is thrown if a
|
|
|
|
* stream is not open. A warning is issued if the stream is already
|
|
|
|
* stopped.
|
|
|
|
*/
|
2014-03-11 22:37:22 +01:00
|
|
|
enum airtaudio::errorType abortStream(void) {
|
2014-03-11 21:46:00 +01:00
|
|
|
if (m_rtapi == NULL) {
|
2014-03-12 23:55:49 +01:00
|
|
|
return airtaudio::errorInputNull;
|
2014-03-11 21:46:00 +01:00
|
|
|
}
|
|
|
|
return m_rtapi->abortStream();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @return true if a stream is open and false if not.
|
|
|
|
*/
|
|
|
|
bool isStreamOpen(void) const {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return m_rtapi->isStreamOpen();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @return true if the stream is running and false if it is stopped or not open.
|
|
|
|
*/
|
|
|
|
bool isStreamRunning(void) const {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return m_rtapi->isStreamRunning();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief If a stream is not open, an RtError (type = INVALID_USE) will be thrown.
|
|
|
|
* @return the number of elapsed seconds since the stream was started.
|
|
|
|
*/
|
|
|
|
double getStreamTime(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
return m_rtapi->getStreamTime();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief The stream latency refers to delay in audio input and/or output
|
|
|
|
* caused by internal buffering by the audio system and/or hardware.
|
|
|
|
* For duplex streams, the returned value will represent the sum of
|
|
|
|
* the input and output latencies. If a stream is not open, an
|
|
|
|
* RtError (type = INVALID_USE) will be thrown. If the API does not
|
|
|
|
* report latency, the return value will be zero.
|
|
|
|
* @return The internal stream latency in sample frames.
|
|
|
|
*/
|
|
|
|
long getStreamLatency(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return m_rtapi->getStreamLatency();
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief On some systems, the sample rate used may be slightly different
|
|
|
|
* than that specified in the stream parameters. If a stream is not
|
|
|
|
* open, an RtError (type = INVALID_USE) will be thrown.
|
|
|
|
* @return Returns actual sample rate in use by the stream.
|
|
|
|
*/
|
|
|
|
uint32_t getStreamSampleRate(void) {
|
|
|
|
if (m_rtapi == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return m_rtapi->getStreamSampleRate();
|
|
|
|
}
|
|
|
|
protected:
|
|
|
|
void openRtApi(airtaudio::api::type _api);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|