2014-03-13 21:16:30 +01:00
|
|
|
/**
|
|
|
|
* @author Edouard DUPIN
|
|
|
|
*
|
|
|
|
* @license like MIT (see license file)
|
|
|
|
*/
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
#ifdef __ANDROID_JAVA__
|
2014-03-13 21:16:30 +01:00
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
#include <ewol/context/Context.h>
|
2014-03-13 21:16:30 +01:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <airtaudio/Interface.h>
|
|
|
|
#include <airtaudio/debug.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
airtaudio::Api* airtaudio::api::Android::Create(void) {
|
|
|
|
ATA_INFO("Create Android device ... ");
|
|
|
|
return new airtaudio::api::Android();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
airtaudio::api::Android::Android(void) {
|
|
|
|
ATA_INFO("new Android");
|
2014-03-13 21:16:30 +01:00
|
|
|
// On android, we set a static device ...
|
2014-03-20 10:25:05 +01:00
|
|
|
ATA_INFO("get context");
|
|
|
|
ewol::Context& tmpContext = ewol::getContext();
|
|
|
|
ATA_INFO("done p=" << (int64_t)&tmpContext);
|
|
|
|
int32_t deviceCount = tmpContext.audioGetDeviceCount();
|
|
|
|
ATA_ERROR("Get count devices : " << deviceCount);
|
|
|
|
for (int32_t iii=0; iii<deviceCount; ++iii) {
|
|
|
|
std::string property = tmpContext.audioGetDeviceProperty(iii);
|
|
|
|
ATA_ERROR("Get devices property : " << property);
|
|
|
|
std::vector<std::string> listProperty = std::split(property, ':');
|
|
|
|
airtaudio::DeviceInfo tmp;
|
|
|
|
tmp.name = listProperty[0];
|
|
|
|
std::vector<std::string> listFreq = std::split(listProperty[2], ',');
|
|
|
|
for(size_t fff=0; fff<listFreq.size(); ++fff) {
|
|
|
|
tmp.sampleRates.push_back(std::stoi(listFreq[fff]));
|
|
|
|
}
|
|
|
|
tmp.outputChannels = 0;
|
|
|
|
tmp.inputChannels = 0;
|
|
|
|
tmp.duplexChannels = 0;
|
|
|
|
if (listProperty[1] == "out") {
|
|
|
|
tmp.isDefaultOutput = true;
|
|
|
|
tmp.isDefaultInput = false;
|
|
|
|
tmp.outputChannels = std::stoi(listProperty[3]);
|
|
|
|
} else if (listProperty[1] == "in") {
|
|
|
|
tmp.isDefaultOutput = false;
|
|
|
|
tmp.isDefaultInput = true;
|
|
|
|
tmp.inputChannels = std::stoi(listProperty[3]);
|
|
|
|
} else {
|
|
|
|
/* duplex */
|
|
|
|
tmp.isDefaultOutput = true;
|
|
|
|
tmp.isDefaultInput = true;
|
|
|
|
tmp.duplexChannels = std::stoi(listProperty[3]);
|
|
|
|
}
|
|
|
|
std::vector<std::string> listFormat = std::split(listProperty[4], ',');
|
|
|
|
tmp.nativeFormats = 0;
|
|
|
|
for(size_t fff=0; fff<listFormat.size(); ++fff) {
|
|
|
|
if (listFormat[fff] == "float") {
|
|
|
|
tmp.nativeFormats |= FLOAT32;
|
|
|
|
} else if (listFormat[fff] == "double") {
|
|
|
|
tmp.nativeFormats |= FLOAT64;
|
|
|
|
} else if (listFormat[fff] == "s32") {
|
|
|
|
tmp.nativeFormats |= SINT32;
|
|
|
|
} else if (listFormat[fff] == "s24") {
|
|
|
|
tmp.nativeFormats |= SINT24;
|
|
|
|
} else if (listFormat[fff] == "s16") {
|
|
|
|
tmp.nativeFormats |= SINT16;
|
|
|
|
} else if (listFormat[fff] == "s8") {
|
|
|
|
tmp.nativeFormats |= SINT8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_devices.push_back(tmp);
|
|
|
|
}
|
|
|
|
ATA_INFO("Create Android interface (end)");
|
2014-03-13 21:16:30 +01:00
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
airtaudio::api::Android::~Android(void) {
|
2014-03-13 21:16:30 +01:00
|
|
|
ATA_INFO("Destroy Android interface");
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
uint32_t airtaudio::api::Android::getDeviceCount(void) {
|
|
|
|
//ATA_INFO("Get device count:"<< m_devices.size());
|
2014-03-13 21:16:30 +01:00
|
|
|
return m_devices.size();
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
airtaudio::DeviceInfo airtaudio::api::Android::getDeviceInfo(uint32_t _device) {
|
|
|
|
//ATA_INFO("Get device info ...");
|
2014-03-13 21:16:30 +01:00
|
|
|
return m_devices[_device];
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
enum airtaudio::errorType airtaudio::api::Android::closeStream(void) {
|
2014-03-13 21:16:30 +01:00
|
|
|
ATA_INFO("Clese Stream");
|
|
|
|
// Can not close the stream now...
|
|
|
|
return airtaudio::errorNone;
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
enum airtaudio::errorType airtaudio::api::Android::startStream(void) {
|
2014-03-13 21:16:30 +01:00
|
|
|
ATA_INFO("Start Stream");
|
|
|
|
// Can not close the stream now...
|
|
|
|
return airtaudio::errorNone;
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
enum airtaudio::errorType airtaudio::api::Android::stopStream(void) {
|
2014-03-13 21:16:30 +01:00
|
|
|
ATA_INFO("Stop stream");
|
2014-03-20 10:25:05 +01:00
|
|
|
ewol::Context& tmpContext = ewol::getContext();
|
|
|
|
tmpContext.audioCloseDevice(0);
|
2014-03-13 21:16:30 +01:00
|
|
|
// Can not close the stream now...
|
|
|
|
return airtaudio::errorNone;
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
enum airtaudio::errorType airtaudio::api::Android::abortStream(void) {
|
2014-03-13 21:16:30 +01:00
|
|
|
ATA_INFO("Abort Stream");
|
2014-03-20 10:25:05 +01:00
|
|
|
ewol::Context& tmpContext = ewol::getContext();
|
|
|
|
tmpContext.audioCloseDevice(0);
|
2014-03-13 21:16:30 +01:00
|
|
|
// Can not close the stream now...
|
|
|
|
return airtaudio::errorNone;
|
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
void airtaudio::api::Android::callBackEvent(void* _data,
|
|
|
|
int32_t _frameRate) {
|
|
|
|
int32_t doStopStream = 0;
|
|
|
|
airtaudio::AirTAudioCallback callback = (airtaudio::AirTAudioCallback) m_stream.callbackInfo.callback;
|
|
|
|
double streamTime = getStreamTime();
|
|
|
|
airtaudio::streamStatus status = 0;
|
|
|
|
if (m_stream.doConvertBuffer[OUTPUT] == true) {
|
|
|
|
doStopStream = callback(m_stream.userBuffer[OUTPUT],
|
|
|
|
NULL,
|
|
|
|
_frameRate,
|
|
|
|
streamTime,
|
|
|
|
status,
|
|
|
|
m_stream.callbackInfo.userData);
|
|
|
|
convertBuffer((char*)_data, (char*)m_stream.userBuffer[OUTPUT], m_stream.convertInfo[OUTPUT]);
|
|
|
|
} else {
|
|
|
|
doStopStream = callback(_data,
|
|
|
|
NULL,
|
|
|
|
_frameRate,
|
|
|
|
streamTime,
|
|
|
|
status,
|
|
|
|
m_stream.callbackInfo.userData);
|
|
|
|
}
|
|
|
|
if (doStopStream == 2) {
|
|
|
|
abortStream();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
airtaudio::Api::tickStreamTime();
|
|
|
|
}
|
|
|
|
|
|
|
|
void airtaudio::api::Android::androidCallBackEvent(void* _data,
|
|
|
|
int32_t _frameRate,
|
|
|
|
void* _userData) {
|
|
|
|
if (_userData == NULL) {
|
|
|
|
ATA_INFO("callback event ... NULL pointer");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
airtaudio::api::Android* myClass = static_cast<airtaudio::api::Android*>(_userData);
|
|
|
|
myClass->callBackEvent(_data, _frameRate/2);
|
2014-03-13 21:16:30 +01:00
|
|
|
}
|
|
|
|
|
2014-03-20 10:25:05 +01:00
|
|
|
bool airtaudio::api::Android::probeDeviceOpen(uint32_t _device,
|
|
|
|
airtaudio::api::StreamMode _mode,
|
|
|
|
uint32_t _channels,
|
|
|
|
uint32_t _firstChannel,
|
|
|
|
uint32_t _sampleRate,
|
|
|
|
airtaudio::format _format,
|
|
|
|
uint32_t *_bufferSize,
|
|
|
|
airtaudio::StreamOptions *_options) {
|
2014-03-13 21:16:30 +01:00
|
|
|
ATA_INFO("Probe : device=" << _device << " channels=" << _channels << " firstChannel=" << _firstChannel << " sampleRate=" << _sampleRate);
|
2014-03-20 10:25:05 +01:00
|
|
|
if (_mode != OUTPUT) {
|
|
|
|
ATA_ERROR("Can not start a device input or duplex for Android ...");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
m_stream.userFormat = _format;
|
|
|
|
m_stream.nUserChannels[_mode] = _channels;
|
|
|
|
ewol::Context& tmpContext = ewol::getContext();
|
|
|
|
bool ret = false;
|
|
|
|
if (_format == SINT8) {
|
|
|
|
ret = tmpContext.audioOpenDevice(_device, _sampleRate, _channels, 0, androidCallBackEvent, this);
|
|
|
|
} else {
|
|
|
|
ret = tmpContext.audioOpenDevice(_device, _sampleRate, _channels, 1, androidCallBackEvent, this);
|
|
|
|
}
|
|
|
|
m_stream.bufferSize = 256;
|
|
|
|
m_stream.sampleRate = _sampleRate;
|
|
|
|
m_stream.doByteSwap[_mode] = false; // for endienness ...
|
|
|
|
|
|
|
|
// TODO : For now, we write it in hard ==> to bu update later ...
|
|
|
|
m_stream.deviceFormat[_mode] = SINT16;
|
|
|
|
m_stream.nDeviceChannels[_mode] = 2;
|
|
|
|
m_stream.deviceInterleaved[_mode] = true;
|
|
|
|
|
|
|
|
m_stream.doConvertBuffer[_mode] = false;
|
|
|
|
if (m_stream.userFormat != m_stream.deviceFormat[_mode]) {
|
|
|
|
m_stream.doConvertBuffer[_mode] = true;
|
|
|
|
}
|
|
|
|
if (m_stream.nUserChannels[_mode] < m_stream.nDeviceChannels[_mode]) {
|
|
|
|
m_stream.doConvertBuffer[_mode] = true;
|
|
|
|
}
|
|
|
|
if ( m_stream.userInterleaved != m_stream.deviceInterleaved[_mode]
|
|
|
|
&& m_stream.nUserChannels[_mode] > 1) {
|
|
|
|
m_stream.doConvertBuffer[_mode] = true;
|
|
|
|
}
|
|
|
|
if (m_stream.doConvertBuffer[_mode] == true) {
|
|
|
|
// Allocate necessary internal buffers.
|
|
|
|
uint64_t bufferBytes = m_stream.nUserChannels[_mode] * m_stream.bufferSize * formatBytes(m_stream.userFormat);
|
|
|
|
m_stream.userBuffer[_mode] = (char *) calloc(bufferBytes, 1);
|
|
|
|
if (m_stream.userBuffer[_mode] == NULL) {
|
|
|
|
ATA_ERROR("airtaudio::api::Android::probeDeviceOpen: error allocating user buffer memory.");
|
|
|
|
}
|
|
|
|
setConvertInfo(_mode, _firstChannel);
|
|
|
|
}
|
|
|
|
ATA_INFO("device format : " << m_stream.deviceFormat[_mode] << " user format : " << m_stream.userFormat);
|
|
|
|
ATA_INFO("device channels : " << m_stream.nDeviceChannels[_mode] << " user channels : " << m_stream.nUserChannels[_mode]);
|
|
|
|
ATA_INFO("do convert buffer : " << m_stream.doConvertBuffer[_mode]);
|
|
|
|
if (ret == false) {
|
|
|
|
ATA_ERROR("Can not open device.");
|
|
|
|
}
|
|
|
|
return ret;
|
2014-03-13 21:16:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|