2015-01-25 22:10:49 +01:00
|
|
|
/** @file
|
|
|
|
* @author Edouard DUPIN
|
|
|
|
* @copyright 2011, Edouard DUPIN, all right reserved
|
2017-01-05 21:28:23 +01:00
|
|
|
* @license MPL v2.0 (see license file)
|
2015-01-25 22:10:49 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2016-10-02 23:41:27 +02:00
|
|
|
#include <cstdint>
|
2016-10-02 21:41:55 +02:00
|
|
|
#include <audio/format.hpp>
|
|
|
|
#include <audio/channel.hpp>
|
|
|
|
#include <audio/drain/Process.hpp>
|
|
|
|
#include <audio/drain/ChannelReorder.hpp>
|
|
|
|
#include <audio/drain/FormatUpdate.hpp>
|
|
|
|
#include <audio/drain/Resampler.hpp>
|
|
|
|
#include <audio/drain/debug.hpp>
|
2015-01-25 22:10:49 +01:00
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
audio::drain::Process::Process() :
|
2015-02-02 21:48:57 +01:00
|
|
|
m_isConfigured(false) {
|
2015-02-09 21:44:32 +01:00
|
|
|
m_data.clear();
|
2015-01-25 22:10:49 +01:00
|
|
|
}
|
2015-04-10 23:00:13 +02:00
|
|
|
audio::drain::Process::~Process() {
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
|
|
|
m_listAlgo[iii].reset();
|
2015-01-28 22:07:11 +01:00
|
|
|
}
|
|
|
|
}
|
2015-01-25 22:10:49 +01:00
|
|
|
|
2015-04-13 21:49:48 +02:00
|
|
|
bool audio::drain::Process::push(audio::Time& _time,
|
2015-04-10 23:00:13 +02:00
|
|
|
void* _data,
|
|
|
|
size_t _nbChunk) {
|
2015-01-25 22:10:49 +01:00
|
|
|
void* out = nullptr;
|
|
|
|
size_t nbChunkOut;
|
2015-02-05 21:33:12 +01:00
|
|
|
DRAIN_VERBOSE(" Process push");
|
2015-01-25 22:10:49 +01:00
|
|
|
process(_time, _data, _nbChunk, out, nbChunkOut);
|
2015-01-26 21:47:51 +01:00
|
|
|
return true;
|
2015-01-25 22:10:49 +01:00
|
|
|
}
|
|
|
|
|
2015-04-13 21:49:48 +02:00
|
|
|
bool audio::drain::Process::pull(audio::Time& _time,
|
2015-04-10 23:00:13 +02:00
|
|
|
void* _data,
|
|
|
|
size_t _nbChunk,
|
|
|
|
size_t _chunkSize) {
|
2015-01-30 21:36:11 +01:00
|
|
|
while(m_data.size()<_nbChunk*_chunkSize) {
|
|
|
|
void* in = NULL;
|
|
|
|
size_t nbChunkIn = _nbChunk - m_data.size()/_chunkSize;
|
|
|
|
void* out = NULL;
|
|
|
|
size_t nbChunkOut;
|
|
|
|
if (nbChunkIn < 128) {
|
|
|
|
nbChunkIn = 128;
|
|
|
|
}
|
|
|
|
// TODO : maybe remove this for input data ...
|
|
|
|
for (int32_t iii=m_listAlgo.size()-1; iii >=0; --iii) {
|
|
|
|
if (m_listAlgo[iii] != NULL) {
|
|
|
|
nbChunkIn = m_listAlgo[iii]->needInputData(nbChunkIn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nbChunkIn < 32) {
|
|
|
|
nbChunkIn = 32;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get data from the upstream
|
|
|
|
process(_time, in, nbChunkIn, out, nbChunkOut);
|
|
|
|
if (nbChunkOut > 0) {
|
|
|
|
size_t position = m_data.size();
|
|
|
|
m_data.resize(m_data.size() + nbChunkOut*_chunkSize);
|
|
|
|
memcpy(&m_data[position], out, nbChunkOut*_chunkSize);
|
|
|
|
} else {
|
2015-03-18 21:47:27 +01:00
|
|
|
// No more data in the process stream (0 input data might have flush data)
|
2015-01-30 21:36:11 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2015-03-18 21:47:27 +01:00
|
|
|
// copy only data availlable :
|
|
|
|
int32_t minByTeSize = std::min(m_data.size(), _nbChunk*_chunkSize);
|
|
|
|
if (minByTeSize >= 0) {
|
|
|
|
memcpy(_data, &m_data[0], minByTeSize);
|
|
|
|
m_data.erase(m_data.begin(), m_data.begin()+minByTeSize);
|
2015-01-30 21:36:11 +01:00
|
|
|
}
|
2015-01-26 21:47:51 +01:00
|
|
|
return true;
|
2015-01-25 22:10:49 +01:00
|
|
|
}
|
|
|
|
|
2015-01-30 21:36:11 +01:00
|
|
|
|
2015-04-13 21:49:48 +02:00
|
|
|
bool audio::drain::Process::process(audio::Time& _time,
|
2015-04-10 23:00:13 +02:00
|
|
|
void* _inData,
|
|
|
|
size_t _inNbChunk,
|
|
|
|
void*& _outData,
|
|
|
|
size_t& _outNbChunk) {
|
2015-02-08 15:11:53 +01:00
|
|
|
updateInterAlgo();
|
2015-01-25 22:10:49 +01:00
|
|
|
if (m_listAlgo.size() == 0) {
|
|
|
|
_outData = _inData;
|
|
|
|
_outNbChunk = _inNbChunk;
|
|
|
|
return true;
|
|
|
|
}
|
2015-02-08 15:11:53 +01:00
|
|
|
DRAIN_VERBOSE(" process : " << m_listAlgo.size() << " algos nbChunk=" << _inNbChunk);
|
2015-01-25 22:10:49 +01:00
|
|
|
for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
|
|
|
//std::cout << " Algo " << iii+1 << "/" << m_listAlgo.size() << std::endl;
|
|
|
|
if (m_listAlgo[iii] != nullptr) {
|
|
|
|
m_listAlgo[iii]->process(_time, _inData, _inNbChunk, _outData, _outNbChunk);
|
|
|
|
_inData = _outData;
|
|
|
|
_inNbChunk = _outNbChunk;
|
|
|
|
}
|
|
|
|
}
|
2015-01-26 21:47:51 +01:00
|
|
|
return true;
|
2015-01-25 22:10:49 +01:00
|
|
|
}
|
|
|
|
|
2016-07-19 21:43:58 +02:00
|
|
|
void audio::drain::Process::pushBack(ememory::SharedPtr<audio::drain::Algo> _algo) {
|
2015-02-02 21:48:57 +01:00
|
|
|
removeAlgoDynamic();
|
2016-03-08 21:29:34 +01:00
|
|
|
_algo->setStatusFunction(std::bind(&audio::drain::Process::generateStatus, this, std::placeholders::_1, std::placeholders::_2));
|
2015-01-25 22:10:49 +01:00
|
|
|
m_listAlgo.push_back(_algo);
|
|
|
|
}
|
|
|
|
|
2016-07-19 21:43:58 +02:00
|
|
|
void audio::drain::Process::pushFront(ememory::SharedPtr<audio::drain::Algo> _algo) {
|
2015-02-02 21:48:57 +01:00
|
|
|
removeAlgoDynamic();
|
2016-03-08 21:29:34 +01:00
|
|
|
_algo->setStatusFunction(std::bind(&audio::drain::Process::generateStatus, this, std::placeholders::_1, std::placeholders::_2));
|
2015-01-25 22:10:49 +01:00
|
|
|
m_listAlgo.insert(m_listAlgo.begin(), _algo);
|
|
|
|
}
|
|
|
|
|
2015-02-01 22:22:42 +01:00
|
|
|
template<typename T> std::vector<T> getUnion(const std::vector<T>& _out, const std::vector<T>& _in) {
|
|
|
|
std::vector<T> out;
|
|
|
|
if (_out.size() == 0) {
|
|
|
|
// Last is ok for all format
|
|
|
|
// ==> set the limit with the next element
|
|
|
|
out = _in;
|
|
|
|
} else if (_in.size() == 0) {
|
|
|
|
// next is ok for all format
|
|
|
|
} else {
|
|
|
|
// must check all values
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t ooo=0; ooo<_out.size(); ++ooo) {
|
|
|
|
for (size_t iii=0; iii<_in.size(); ++iii) {
|
|
|
|
if (_out[ooo] == _in[iii]) {
|
|
|
|
out.push_back(_out[ooo]);
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2015-02-17 21:08:15 +01:00
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::displayAlgo() {
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" Input : " << m_inputConfig);
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" [" << m_listAlgo[iii]->getType() << "] '" << m_listAlgo[iii]->getName() << "'");
|
2015-02-24 22:20:11 +01:00
|
|
|
if (m_listAlgo[iii]->getInputFormat().getConfigured() == true) {
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" Input : " << m_listAlgo[iii]->getInputFormat());
|
2015-02-01 22:22:42 +01:00
|
|
|
} else {
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" Input : Not configured");
|
|
|
|
DRAIN_DEBUG(" format : " << m_listAlgo[iii]->getFormatSupportedInput());
|
|
|
|
DRAIN_DEBUG(" frequency : " << m_listAlgo[iii]->getFrequencySupportedInput());
|
|
|
|
DRAIN_DEBUG(" map : " << m_listAlgo[iii]->getMapSupportedInput());
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
if (m_listAlgo[iii]->getOutputFormat().getConfigured() == true) {
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" Output: " << m_listAlgo[iii]->getOutputFormat());
|
2015-02-01 22:22:42 +01:00
|
|
|
} else {
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" Output : Not configured");
|
|
|
|
DRAIN_DEBUG(" format : " << m_listAlgo[iii]->getFormatSupportedOutput());
|
|
|
|
DRAIN_DEBUG(" frequency : " << m_listAlgo[iii]->getFrequencySupportedOutput());
|
|
|
|
DRAIN_DEBUG(" map : " << m_listAlgo[iii]->getMapSupportedOutput());
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
|
|
|
}
|
2015-03-04 21:50:51 +01:00
|
|
|
DRAIN_DEBUG(" Output : " << m_outputConfig);
|
2015-02-17 21:08:15 +01:00
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::updateAlgo(size_t _position) {
|
2015-02-17 21:08:15 +01:00
|
|
|
DRAIN_VERBOSE(" id = " << _position);
|
|
|
|
if ( ( _position == 0
|
|
|
|
|| ( _position > 0
|
|
|
|
&& m_listAlgo[_position-1]->getOutputFormat().getConfigured() == false
|
|
|
|
)
|
|
|
|
)
|
|
|
|
&& ( _position == m_listAlgo.size()
|
|
|
|
|| ( _position < m_listAlgo.size()
|
|
|
|
&& m_listAlgo[_position]->getInputFormat().getConfigured() == false
|
|
|
|
)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
// step 1 : check frequency:
|
|
|
|
std::vector<float> freqOut;
|
|
|
|
std::vector<float> freqIn;
|
|
|
|
if (_position == 0) {
|
|
|
|
freqOut.push_back(m_inputConfig.getFrequency());
|
|
|
|
} else {
|
|
|
|
freqOut = m_listAlgo[_position-1]->getFrequencySupportedOutput();
|
|
|
|
}
|
|
|
|
if (_position == m_listAlgo.size()) {
|
|
|
|
freqIn.push_back(m_outputConfig.getFrequency());
|
|
|
|
} else {
|
|
|
|
freqIn = m_listAlgo[_position]->getFrequencySupportedInput();
|
|
|
|
}
|
|
|
|
std::vector<float> freq = getUnion<float>(freqOut, freqIn);
|
|
|
|
DRAIN_VERBOSE(" freq out :" << freqOut);
|
|
|
|
DRAIN_VERBOSE(" freq in :" << freqIn);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE(" freq union :" << freq);
|
2015-02-17 21:08:15 +01:00
|
|
|
|
|
|
|
// step 2 : Check map:
|
2015-02-24 22:20:11 +01:00
|
|
|
std::vector<std::vector<audio::channel> > mapOut;
|
|
|
|
std::vector<std::vector<audio::channel> > mapIn;
|
2015-02-17 21:08:15 +01:00
|
|
|
if (_position == 0) {
|
|
|
|
mapOut.push_back(m_inputConfig.getMap());
|
|
|
|
} else {
|
|
|
|
mapOut = m_listAlgo[_position-1]->getMapSupportedOutput();
|
|
|
|
}
|
|
|
|
if (_position == m_listAlgo.size()) {
|
|
|
|
mapIn.push_back(m_outputConfig.getMap());
|
|
|
|
} else {
|
|
|
|
mapIn = m_listAlgo[_position]->getMapSupportedInput();
|
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
std::vector<std::vector<audio::channel> > map = getUnion<std::vector<audio::channel> >(mapOut, mapIn);
|
2015-02-17 21:08:15 +01:00
|
|
|
DRAIN_VERBOSE(" map out :" << mapOut);
|
|
|
|
DRAIN_VERBOSE(" map in :" << mapIn);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE(" map union :" << map);
|
2015-02-17 21:08:15 +01:00
|
|
|
// step 3 : Check Format:
|
|
|
|
std::vector<audio::format> formatOut;
|
|
|
|
std::vector<audio::format> formatIn;
|
|
|
|
if (_position == 0) {
|
|
|
|
formatOut.push_back(m_inputConfig.getFormat());
|
|
|
|
} else {
|
|
|
|
formatOut = m_listAlgo[_position-1]->getFormatSupportedOutput();
|
|
|
|
}
|
|
|
|
if (_position == m_listAlgo.size()) {
|
|
|
|
formatIn.push_back(m_outputConfig.getFormat());
|
|
|
|
} else {
|
|
|
|
formatIn = m_listAlgo[_position]->getFormatSupportedInput();
|
|
|
|
}
|
|
|
|
std::vector<audio::format> format = getUnion<audio::format>(formatOut, formatIn);
|
|
|
|
DRAIN_VERBOSE(" format out :" << formatOut);
|
|
|
|
DRAIN_VERBOSE(" format in :" << formatIn);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE(" format union :" << format);
|
2015-02-17 21:08:15 +01:00
|
|
|
|
|
|
|
if ( freq.size() >= 1
|
|
|
|
&& map.size() >= 1
|
|
|
|
&& format.size() >= 1) {
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE(" find 1 compatibility :{format=" << format << ",frequency=" << freq << ",map=" << map << "}");
|
2015-02-17 21:08:15 +01:00
|
|
|
drain::IOFormatInterface tmp(map[0], format[0], freq[0]);
|
|
|
|
if (_position > 0) {
|
|
|
|
m_listAlgo[_position-1]->setOutputFormat(tmp);
|
2015-02-08 15:11:53 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
if (_position <m_listAlgo.size()) {
|
|
|
|
m_listAlgo[_position]->setInputFormat(tmp);
|
2015-02-08 15:11:53 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// create mapping to transform:
|
2015-04-10 23:00:13 +02:00
|
|
|
audio::drain::IOFormatInterface out;
|
|
|
|
audio::drain::IOFormatInterface in;
|
2015-02-17 21:08:15 +01:00
|
|
|
if (freq.size() > 0) {
|
|
|
|
out.setFrequency(freq[0]);
|
|
|
|
in.setFrequency(freq[0]);
|
|
|
|
} else {
|
|
|
|
if (freqOut.size() == 0) {
|
|
|
|
if (freqIn.size() == 0) {
|
|
|
|
if (_position == 0) {
|
|
|
|
DRAIN_ERROR("IMPOSSIBLE CASE");
|
|
|
|
} else {
|
|
|
|
out.setFrequency(m_listAlgo[_position-1]->getInputFormat().getFrequency());
|
|
|
|
in.setFrequency(m_listAlgo[_position-1]->getInputFormat().getFrequency());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
out.setFrequency(freqIn[0]);
|
|
|
|
in.setFrequency(freqIn[0]);
|
2015-02-08 15:11:53 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
} else {
|
|
|
|
if (freqIn.size() == 0) {
|
|
|
|
out.setFrequency(freqOut[0]);
|
|
|
|
in.setFrequency(freqOut[0]);
|
|
|
|
} else {
|
|
|
|
out.setFrequency(freqOut[0]);
|
|
|
|
in.setFrequency(freqIn[0]);
|
2015-02-08 15:11:53 +01:00
|
|
|
}
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
}
|
|
|
|
if (map.size() > 0) {
|
|
|
|
out.setMap(map[0]);
|
|
|
|
in.setMap(map[0]);
|
|
|
|
} else {
|
|
|
|
if (mapOut.size() == 0) {
|
|
|
|
if (mapIn.size() == 0) {
|
|
|
|
if (_position == 0) {
|
|
|
|
DRAIN_ERROR("IMPOSSIBLE CASE");
|
2015-02-02 21:48:57 +01:00
|
|
|
} else {
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setMap(m_listAlgo[_position-1]->getInputFormat().getMap());
|
|
|
|
in.setMap(m_listAlgo[_position-1]->getInputFormat().getMap());
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
|
|
|
} else {
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setMap(mapIn[0]);
|
|
|
|
in.setMap(mapIn[0]);
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
|
|
|
} else {
|
2015-02-17 21:08:15 +01:00
|
|
|
if (mapIn.size() == 0) {
|
|
|
|
out.setMap(mapOut[0]);
|
|
|
|
in.setMap(mapOut[0]);
|
2015-02-02 21:48:57 +01:00
|
|
|
} else {
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setMap(mapOut[0]);
|
|
|
|
in.setMap(mapIn[0]);
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
}
|
|
|
|
if (format.size() > 0) {
|
|
|
|
out.setFormat(format[0]);
|
|
|
|
in.setFormat(format[0]);
|
|
|
|
} else {
|
|
|
|
if (formatOut.size() == 0) {
|
|
|
|
if (formatIn.size() == 0) {
|
|
|
|
if (_position == 0) {
|
|
|
|
DRAIN_ERROR("IMPOSSIBLE CASE");
|
2015-02-02 21:48:57 +01:00
|
|
|
} else {
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setFormat(m_listAlgo[_position-1]->getInputFormat().getFormat());
|
|
|
|
in.setFormat(m_listAlgo[_position-1]->getInputFormat().getFormat());
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
|
|
|
} else {
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setFormat(formatIn[0]);
|
|
|
|
in.setFormat(formatIn[0]);
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
} else {
|
|
|
|
if (formatIn.size() == 0) {
|
|
|
|
out.setFormat(formatOut[0]);
|
|
|
|
in.setFormat(formatOut[0]);
|
|
|
|
} else {
|
|
|
|
out.setFormat(formatOut[0]);
|
|
|
|
in.setFormat(formatIn[0]);
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
}
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE(" update: out=" << out);
|
|
|
|
DRAIN_VERBOSE(" in=" << in);
|
2015-02-17 21:08:15 +01:00
|
|
|
if (_position > 0) {
|
|
|
|
m_listAlgo[_position-1]->setOutputFormat(out);
|
|
|
|
}
|
|
|
|
if (_position < m_listAlgo.size()) {
|
|
|
|
m_listAlgo[_position]->setInputFormat(in);
|
|
|
|
}
|
|
|
|
// TODO : Add updater with an optimisation of CPU
|
|
|
|
if (out.getFrequency() != in.getFrequency()) {
|
|
|
|
|
|
|
|
// TODO : Do it better: special check for resampler : only support int16_t
|
|
|
|
if ( out.getFormat() != audio::format_int16
|
|
|
|
/* && out.getFormat() != format_float */) {
|
2015-02-02 21:48:57 +01:00
|
|
|
// need add a format Updater
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::drain::FormatUpdate> algo = audio::drain::FormatUpdate::create();
|
2015-02-02 21:48:57 +01:00
|
|
|
algo->setTemporary();
|
|
|
|
algo->setInputFormat(out);
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setFormat(audio::format_int16);
|
2015-02-02 21:48:57 +01:00
|
|
|
algo->setOutputFormat(out);
|
2016-03-08 21:29:34 +01:00
|
|
|
algo->setStatusFunction(std::bind(&audio::drain::Process::generateStatus, this, std::placeholders::_1, std::placeholders::_2));
|
2015-02-17 21:08:15 +01:00
|
|
|
m_listAlgo.insert(m_listAlgo.begin()+_position, algo);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("convert " << out.getFormat() << " -> " << in.getFormat());
|
2015-02-17 21:08:15 +01:00
|
|
|
_position++;
|
2015-02-02 21:48:57 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
// need add a resampler
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::drain::Resampler> algo = audio::drain::Resampler::create();
|
2015-02-17 21:08:15 +01:00
|
|
|
algo->setTemporary();
|
|
|
|
algo->setInputFormat(out);
|
|
|
|
out.setFrequency(in.getFrequency());
|
|
|
|
algo->setOutputFormat(out);
|
2016-03-08 21:29:34 +01:00
|
|
|
algo->setStatusFunction(std::bind(&audio::drain::Process::generateStatus, this, std::placeholders::_1, std::placeholders::_2));
|
2015-02-17 21:08:15 +01:00
|
|
|
m_listAlgo.insert(m_listAlgo.begin()+_position, algo);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("convert " << out.getFrequency() << " -> " << in.getFrequency());
|
2015-02-17 21:08:15 +01:00
|
|
|
out.setFrequency(in.getFrequency());
|
|
|
|
_position++;
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
if (out.getMap() != in.getMap()) {
|
|
|
|
// need add a channel Reorder
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::drain::ChannelReorder> algo = audio::drain::ChannelReorder::create();
|
2015-02-17 21:08:15 +01:00
|
|
|
algo->setTemporary();
|
|
|
|
algo->setInputFormat(out);
|
|
|
|
out.setMap(in.getMap());
|
|
|
|
algo->setOutputFormat(out);
|
2016-03-08 21:29:34 +01:00
|
|
|
algo->setStatusFunction(std::bind(&audio::drain::Process::generateStatus, this, std::placeholders::_1, std::placeholders::_2));
|
2015-02-17 21:08:15 +01:00
|
|
|
m_listAlgo.insert(m_listAlgo.begin()+_position, algo);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("convert " << out.getMap() << " -> " << in.getMap());
|
2015-02-17 21:08:15 +01:00
|
|
|
_position++;
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
if (out.getFormat() != in.getFormat()) {
|
|
|
|
// need add a format Updater
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::drain::FormatUpdate> algo = audio::drain::FormatUpdate::create();
|
2015-02-17 21:08:15 +01:00
|
|
|
algo->setTemporary();
|
|
|
|
algo->setInputFormat(out);
|
|
|
|
out.setFormat(in.getFormat());
|
|
|
|
algo->setOutputFormat(out);
|
2016-03-08 21:29:34 +01:00
|
|
|
algo->setStatusFunction(std::bind(&audio::drain::Process::generateStatus, this, std::placeholders::_1, std::placeholders::_2));
|
2015-02-17 21:08:15 +01:00
|
|
|
m_listAlgo.insert(m_listAlgo.begin()+_position, algo);
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("convert " << out.getFormat() << " -> " << in.getFormat());
|
2015-02-17 21:08:15 +01:00
|
|
|
_position++;
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
|
|
|
|
} else if ( ( _position > 0
|
|
|
|
&& m_listAlgo[_position-1]->getOutputFormat().getConfigured() == false
|
|
|
|
)
|
|
|
|
|| ( _position < m_listAlgo.size()
|
|
|
|
&& m_listAlgo[_position]->getInputFormat().getConfigured() == false
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
DRAIN_ERROR(" configuration error mode in " << _position-1 << " && " << _position );
|
2015-02-01 22:22:42 +01:00
|
|
|
}
|
2015-02-17 21:08:15 +01:00
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::updateInterAlgo() {
|
2015-02-17 21:08:15 +01:00
|
|
|
if (m_isConfigured == true) {
|
|
|
|
// cahin is already configured
|
|
|
|
return ;
|
|
|
|
}
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("Display properties : nbAlgo : " << m_listAlgo.size());
|
2015-02-17 21:08:15 +01:00
|
|
|
displayAlgo();
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("********* configuration START *************");
|
2015-02-17 21:08:15 +01:00
|
|
|
// configure first the endpoint ...
|
2015-03-23 22:13:24 +01:00
|
|
|
if (m_listAlgo.size() >= 1) {
|
2015-02-17 21:08:15 +01:00
|
|
|
updateAlgo(m_listAlgo.size());
|
|
|
|
}
|
|
|
|
for (size_t iii=0; iii<=m_listAlgo.size(); ++iii) {
|
|
|
|
updateAlgo(iii);
|
|
|
|
}
|
2015-03-03 21:28:07 +01:00
|
|
|
DRAIN_VERBOSE("********* configuration will be done *************");
|
2015-02-17 21:08:15 +01:00
|
|
|
displayAlgo();
|
2015-02-02 21:48:57 +01:00
|
|
|
m_isConfigured = true;
|
2015-02-01 22:22:42 +01:00
|
|
|
//exit(-1);
|
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::removeAlgoDynamic() {
|
2015-02-02 21:48:57 +01:00
|
|
|
if (m_isConfigured == true) {
|
|
|
|
// chain is already unconfigured.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_isConfigured = false;
|
2015-01-25 22:10:49 +01:00
|
|
|
}
|
2015-02-08 15:11:53 +01:00
|
|
|
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
bool audio::drain::Process::processIn(void* _inData,
|
|
|
|
size_t _inNbChunk,
|
|
|
|
void* _outData,
|
|
|
|
size_t _outNbChunk){
|
2015-02-08 15:11:53 +01:00
|
|
|
void* outData = nullptr;
|
|
|
|
size_t outSize = 0;
|
|
|
|
bool error = process(_inData, _inNbChunk, outData, outSize);
|
|
|
|
if (outSize != _outNbChunk) {
|
|
|
|
DRAIN_ERROR("can not copy data to output (not the same chunk number : out=" << outSize << " chunks != request=" << _outNbChunk << " chunks");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// TODO : Do it better ...
|
2015-02-09 21:44:32 +01:00
|
|
|
DRAIN_VERBOSE("Copy " << _outNbChunk << " chunks byte size=" << audio::getFormatBytes(m_outputConfig.getFormat()) << " nbChan=" << m_outputConfig.getMap().size() << " format=" << m_outputConfig.getFormat());
|
2015-02-08 15:11:53 +01:00
|
|
|
memcpy(_outData, outData, _outNbChunk*audio::getFormatBytes(m_outputConfig.getFormat()) * m_outputConfig.getMap().size());
|
|
|
|
return false;
|
2015-03-05 22:14:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void link(etk::FSNode& _node, const std::string& _first, const std::string& _op, const std::string& _second, bool _isLink=true) {
|
|
|
|
if (_op == "->") {
|
|
|
|
if (_isLink) {
|
|
|
|
_node << " " << _first << " -> " << _second << ";\n";
|
|
|
|
} else {
|
|
|
|
_node << " " << _first << " -> " << _second << " [style=dashed];\n";
|
|
|
|
}
|
|
|
|
} else if (_op == "<-") {
|
|
|
|
_node << " " << _first << " -> " <<_second<< " [color=transparent];\n";
|
|
|
|
if (_isLink) {
|
|
|
|
_node << " " << _second << " -> " << _first << " [constraint=false];\n";
|
|
|
|
} else {
|
|
|
|
_node << " " << _second << " -> " << _first << " [constraint=false, style=dashed];\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-17 00:18:49 +01:00
|
|
|
void audio::drain::Process::generateDot(etk::FSNode& _node,
|
|
|
|
int32_t _offset,
|
|
|
|
int32_t _basicID,
|
|
|
|
std::string& _nameIn,
|
|
|
|
std::string& _nameOut,
|
|
|
|
bool _reserseGraph) {
|
2015-03-05 22:14:07 +01:00
|
|
|
_node << " subgraph clusterNode_" << _basicID << "_process {\n";
|
2015-03-18 21:47:27 +01:00
|
|
|
_node << " label=\"Drain::Process" << (_reserseGraph?"_R":"_N") << "\";\n";
|
2015-03-05 22:14:07 +01:00
|
|
|
_node << " node [shape=ellipse];\n";
|
|
|
|
|
|
|
|
if (_reserseGraph == false) {
|
2016-11-17 00:18:49 +01:00
|
|
|
// ----------------------
|
|
|
|
// -- STEP 1 --
|
|
|
|
// ----------------------
|
2015-03-05 22:14:07 +01:00
|
|
|
_nameIn = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_in";
|
|
|
|
_node << " " << _nameIn << " [ label=\"format=" << etk::to_string(getInputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getInputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getInputConfig().getMap()) << "\\n in\" ];\n";
|
2016-11-17 00:18:49 +01:00
|
|
|
// ----------------------
|
|
|
|
// -- STEP 2 --
|
|
|
|
// ----------------------
|
|
|
|
std::string connectString = _nameIn;
|
|
|
|
_node << " node [shape=box];\n";
|
|
|
|
// ----------------------
|
|
|
|
// -- STEP 3 --
|
|
|
|
// ----------------------
|
2015-03-05 22:14:07 +01:00
|
|
|
for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
|
|
|
if (m_listAlgo[iii] == nullptr) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::string connectStringSecond = "ALGO_" + etk::to_string(_basicID) + "__" + etk::to_string(iii);
|
2016-11-17 00:18:49 +01:00
|
|
|
_node << " " << connectStringSecond << " [label=\"ALGO\\ntype='" << m_listAlgo[iii]->getType() << "'";
|
|
|
|
if (m_listAlgo[iii]->getName() != "") {
|
|
|
|
_node << "\\nname='" << m_listAlgo[iii]->getName() << "'";
|
|
|
|
}
|
|
|
|
std::string tmpDesc = m_listAlgo[iii]->getDotDesc();
|
|
|
|
if (tmpDesc.size() != 0) {
|
|
|
|
_node << tmpDesc;
|
|
|
|
}
|
|
|
|
_node << "\" ];\n";
|
2015-03-05 22:14:07 +01:00
|
|
|
link(_node, connectString, "->", connectStringSecond);
|
|
|
|
connectString = connectStringSecond;
|
|
|
|
}
|
2016-11-17 00:18:49 +01:00
|
|
|
// ----------------------
|
|
|
|
// -- STEP 4 --
|
|
|
|
// ----------------------
|
|
|
|
_node << " node [shape=ellipse];\n";
|
|
|
|
// ----------------------
|
|
|
|
// -- STEP 5 --
|
|
|
|
// ----------------------
|
|
|
|
_nameOut = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_out";
|
|
|
|
_node << " " << _nameOut << " [ label=\"format=" << etk::to_string(getOutputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getOutputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getOutputConfig().getMap()) << "\\n out\" ];\n";
|
|
|
|
// ----------------------
|
|
|
|
// -- STEP 6 --
|
|
|
|
// ----------------------
|
|
|
|
link(_node, connectString, "->", _nameOut);
|
2015-03-05 22:14:07 +01:00
|
|
|
} else {
|
2016-11-17 00:18:49 +01:00
|
|
|
// ----------------------
|
|
|
|
// -- STEP 1 --
|
|
|
|
// ----------------------
|
|
|
|
_nameIn = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_out";
|
|
|
|
_node << " " << _nameIn << " [ label=\"format=" << etk::to_string(getOutputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getOutputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getOutputConfig().getMap()) << "\\n out\" ];\n";
|
|
|
|
// ----------------------
|
|
|
|
// -- STEP 2 --
|
|
|
|
// ----------------------
|
|
|
|
std::string connectString = _nameIn;
|
|
|
|
_node << " node [shape=box];\n";
|
|
|
|
// ----------------------
|
|
|
|
// -- STEP 3 --
|
|
|
|
// ----------------------
|
|
|
|
for (int32_t iii=m_listAlgo.size()-1; iii>=0; --iii) {
|
|
|
|
//for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
2015-03-05 22:14:07 +01:00
|
|
|
if (m_listAlgo[iii] == nullptr) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::string connectStringSecond = "ALGO_" + etk::to_string(_basicID) + "__" + etk::to_string(iii);
|
2016-11-17 00:18:49 +01:00
|
|
|
_node << " " << connectStringSecond << " [label=\"ALGO\\ntype='" << m_listAlgo[iii]->getType() << "'";
|
|
|
|
if (m_listAlgo[iii]->getName() != "") {
|
|
|
|
_node << "\\nname='" << m_listAlgo[iii]->getName() << "'";
|
|
|
|
}
|
|
|
|
std::string tmpDesc = m_listAlgo[iii]->getDotDesc();
|
|
|
|
if (tmpDesc.size() != 0) {
|
|
|
|
_node << tmpDesc;
|
|
|
|
}
|
|
|
|
_node << "\" ];\n";
|
|
|
|
//link(_node, connectStringSecond, "<-", connectString);
|
|
|
|
link(_node, connectString, "<-", connectStringSecond);
|
|
|
|
//link(_node, connectStringSecond, "->", connectString);
|
2015-03-05 22:14:07 +01:00
|
|
|
connectString = connectStringSecond;
|
|
|
|
}
|
2016-11-17 00:18:49 +01:00
|
|
|
// ----------------------
|
|
|
|
// -- STEP 4 --
|
|
|
|
// ----------------------
|
|
|
|
_node << " node [shape=ellipse];\n";
|
|
|
|
// ----------------------
|
|
|
|
// -- STEP 5 --
|
|
|
|
// ----------------------
|
2015-03-05 22:14:07 +01:00
|
|
|
_nameOut = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_in";
|
|
|
|
_node << " " << _nameOut << " [ label=\"format=" << etk::to_string(getInputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getInputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getInputConfig().getMap()) << "\\n in\" ];\n";
|
2016-11-17 00:18:49 +01:00
|
|
|
// ----------------------
|
|
|
|
// -- STEP 6 --
|
|
|
|
// ----------------------
|
|
|
|
link(_node, connectString, "<-", _nameOut);
|
2015-03-05 22:14:07 +01:00
|
|
|
}
|
|
|
|
_node << " }\n";
|
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::generateDotProcess(etk::FSNode& _node, int32_t _offset, int32_t _basicID, std::string& _nameIn, std::string& _nameOut, bool _reserseGraph) {
|
2015-03-18 21:47:27 +01:00
|
|
|
_node << " subgraph clusterNode_" << _basicID << "_process {\n";
|
|
|
|
_node << " label=\"Drain::Process" << (_reserseGraph?"_R":"_N") << "\";\n";
|
|
|
|
_node << " node [shape=ellipse];\n";
|
|
|
|
|
|
|
|
if (_reserseGraph == true) {
|
|
|
|
_nameIn = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_in";
|
|
|
|
_node << " " << _nameIn << " [ label=\"format=" << etk::to_string(getInputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getInputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getInputConfig().getMap()) << "\\n in\" ];\n";
|
|
|
|
} else {
|
|
|
|
_nameIn = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_out";
|
|
|
|
_node << " " << _nameIn << " [ label=\"format=" << etk::to_string(getOutputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getOutputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getOutputConfig().getMap()) << "\\n out\" ];\n";
|
|
|
|
}
|
|
|
|
std::string connectString = _nameIn;
|
|
|
|
_node << " node [shape=box];\n";
|
|
|
|
if (_reserseGraph == false) {
|
|
|
|
for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
|
|
|
if (m_listAlgo[iii] == nullptr) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::string connectStringSecond = "ALGO_" + etk::to_string(_basicID) + "__" + etk::to_string(iii);
|
2016-11-17 00:18:49 +01:00
|
|
|
_node << " " << connectStringSecond << " [label=\"ALGO\\ntype='" << m_listAlgo[iii]->getType() << "'";
|
|
|
|
if (m_listAlgo[iii]->getName() != "") {
|
|
|
|
_node << "\\nname='" << m_listAlgo[iii]->getName() << "'";
|
|
|
|
}
|
|
|
|
std::string tmpDesc = m_listAlgo[iii]->getDotDesc();
|
|
|
|
if (tmpDesc.size() != 0) {
|
|
|
|
_node << tmpDesc;
|
|
|
|
}
|
|
|
|
_node << "\" ];\n";
|
2015-03-18 21:47:27 +01:00
|
|
|
link(_node, connectString, "->", connectStringSecond);
|
|
|
|
connectString = connectStringSecond;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
//for (int32_t iii=m_listAlgo.size()-1; iii>=0; --iii) {
|
|
|
|
for (size_t iii=0; iii<m_listAlgo.size(); ++iii) {
|
|
|
|
if (m_listAlgo[iii] == nullptr) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::string connectStringSecond = "ALGO_" + etk::to_string(_basicID) + "__" + etk::to_string(iii);
|
2016-11-17 00:18:49 +01:00
|
|
|
_node << " " << connectStringSecond << " [label=\"ALGO\\ntype='" << m_listAlgo[iii]->getType() << "'";
|
|
|
|
if (m_listAlgo[iii]->getName() != "") {
|
|
|
|
_node << "\\nname='" << m_listAlgo[iii]->getName() << "'";
|
|
|
|
}
|
|
|
|
std::string tmpDesc = m_listAlgo[iii]->getDotDesc();
|
|
|
|
if (tmpDesc.size() != 0) {
|
|
|
|
_node << tmpDesc;
|
|
|
|
}
|
|
|
|
_node << "\" ];\n";
|
2015-03-18 21:47:27 +01:00
|
|
|
link(_node, connectStringSecond, "<-", connectString);
|
|
|
|
connectString = connectStringSecond;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_node << " node [shape=ellipse];\n";
|
|
|
|
if (_reserseGraph == true) {
|
|
|
|
_nameOut = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_out";
|
|
|
|
_node << " " << _nameOut << " [ label=\"format=" << etk::to_string(getOutputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getOutputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getOutputConfig().getMap()) << "\\n out\" ];\n";
|
|
|
|
} else {
|
|
|
|
_nameOut = "INTERFACE_ALGO_" + etk::to_string(_basicID) + "_in";
|
|
|
|
_node << " " << _nameOut << " [ label=\"format=" << etk::to_string(getInputConfig().getFormat())
|
|
|
|
<< "\\n freq=" << getInputConfig().getFrequency()
|
|
|
|
<< "\\n channelMap=" << etk::to_string(getInputConfig().getMap()) << "\\n in\" ];\n";
|
|
|
|
}
|
|
|
|
if (_reserseGraph == false) {
|
|
|
|
link(_node, connectString, "->", _nameOut);
|
|
|
|
} else {
|
|
|
|
link(_node, _nameOut, "<-", connectString);
|
|
|
|
}
|
|
|
|
_node << " }\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::generateStatus(const std::string& _origin, const std::string& _status) {
|
2015-03-18 21:47:27 +01:00
|
|
|
if (m_statusFunction != nullptr) {
|
|
|
|
m_statusFunction(_origin, _status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-10 23:00:13 +02:00
|
|
|
void audio::drain::Process::setStatusFunction(statusFunction _newFunction) {
|
2015-03-18 21:47:27 +01:00
|
|
|
m_statusFunction = _newFunction;
|
|
|
|
}
|