498 lines
19 KiB
C++
Raw Permalink Normal View History

2015-01-25 22:17:06 +01:00
/** @file
* @author Edouard DUPIN
* @copyright 2015, Edouard DUPIN, all right reserved
* @license MPL v2.0 (see license file)
2015-01-25 22:17:06 +01:00
*/
2016-10-02 22:19:26 +02:00
#include "Node.hpp"
#include <audio/river/debug.hpp>
2015-01-25 22:17:06 +01:00
2017-08-28 00:08:50 +02:00
audio::river::io::Node::Node(const etk::String& _name, const ejson::Object& _config) :
m_config(_config),
m_name(_name),
m_isInput(false) {
2015-02-18 23:37:09 +01:00
static uint32_t uid=0;
m_uid = uid++;
RIVER_INFO("-----------------------------------------------------------------");
RIVER_INFO("-- CREATE NODE --");
RIVER_INFO("-----------------------------------------------------------------");
2015-04-11 09:38:30 +02:00
audio::drain::IOFormatInterface interfaceFormat;
audio::drain::IOFormatInterface hardwareFormat;
/**
io:"input", # input, output or aec
frequency:48000, # frequency to open device
channel-map:[ # mapping of the harware device (to change map if needed)
"front-left", "front-right",
"read-left", "rear-right",
],
# format to open device (int8, int16, int16-on-ont32, int24, int32, float)
type:"int16",
# muxer/demuxer format type (int8-on-int16, int16-on-int32, int24-on-int32, int32-on-int64, float)
mux-demux-type:"int16_on_int32",
*/
2017-08-28 00:08:50 +02:00
etk::String interfaceType = m_config["io"].toString().get();
2015-03-13 23:22:17 +01:00
RIVER_INFO("interfaceType=" << interfaceType);
if ( interfaceType == "input"
2015-02-19 22:00:21 +01:00
|| interfaceType == "PAinput"
2015-03-04 22:48:04 +01:00
|| interfaceType == "aec"
|| interfaceType == "muxer") {
m_isInput = true;
} else {
m_isInput = false;
}
int32_t frequency = m_config["frequency"].toNumber().get(1);
// Get audio format type:
2017-08-28 00:08:50 +02:00
etk::String type = m_config["type"].toString().get("int16");
enum audio::format formatType = audio::getFormatFromString(type);
// Get volume stage :
2017-08-28 00:08:50 +02:00
etk::String volumeName = m_config["volume-name"].toString().get();
if (volumeName != "") {
RIVER_INFO("add node volume stage : '" << volumeName << "'");
2015-02-03 23:29:34 +01:00
// use global manager for volume ...
2015-04-11 09:38:30 +02:00
m_volume = audio::river::io::Manager::getInstance()->getVolumeGroup(volumeName);
}
// Get map type :
2017-08-28 00:08:50 +02:00
etk::Vector<audio::channel> map;
2016-04-20 21:19:11 +02:00
const ejson::Array listChannelMap = m_config["channel-map"].toArray();
if ( listChannelMap.exist() == false
|| listChannelMap.size() == 0) {
// set default channel property:
2017-08-28 00:08:50 +02:00
map.pushBack(audio::channel_frontLeft);
map.pushBack(audio::channel_frontRight);
2015-02-02 22:04:11 +01:00
} else {
2016-04-20 21:19:11 +02:00
for (auto it : listChannelMap) {
2017-08-28 00:08:50 +02:00
etk::String value = it.toString().get();
map.pushBack(audio::getChannelFromString(value));
}
2015-01-25 22:17:06 +01:00
}
hardwareFormat.set(map, formatType, frequency);
2015-01-25 22:17:06 +01:00
2017-08-28 00:08:50 +02:00
etk::String muxerDemuxerConfig;
2015-03-13 23:22:17 +01:00
if (m_isInput == true) {
muxerDemuxerConfig = m_config["mux-demux-type"].toString().get("int16");
2015-03-13 23:22:17 +01:00
} else {
muxerDemuxerConfig = m_config["mux-demux-type"].toString().get("int16-on-int32");
2015-03-13 23:22:17 +01:00
}
enum audio::format muxerFormatType = audio::getFormatFromString(muxerDemuxerConfig);
2015-01-26 22:04:29 +01:00
if (m_isInput == true) {
// Support all ...
2015-01-26 22:04:29 +01:00
} else {
if ( muxerFormatType != audio::format_int8_on_int16
&& muxerFormatType != audio::format_int16_on_int32
&& muxerFormatType != audio::format_int24_on_int32
&& muxerFormatType != audio::format_int32_on_int64
&& muxerFormatType != audio::format_float
&& muxerFormatType != audio::format_double) {
RIVER_CRITICAL("not supported demuxer type ... " << muxerFormatType << " for OUTPUT set in file:" << muxerDemuxerConfig);
}
2015-01-25 22:17:06 +01:00
}
// no map change and no frequency change ...
interfaceFormat.set(map, muxerFormatType, frequency);
// configure process interface
if (m_isInput == true) {
m_process.setInputConfig(hardwareFormat);
m_process.setOutputConfig(interfaceFormat);
} else {
m_process.setOutputConfig(hardwareFormat);
m_process.setInputConfig(interfaceFormat);
}
//m_process.updateInterAlgo();
2015-01-25 22:17:06 +01:00
}
2015-04-11 09:38:30 +02:00
audio::river::io::Node::~Node() {
RIVER_INFO("-----------------------------------------------------------------");
RIVER_INFO("-- DESTROY NODE --");
RIVER_INFO("-----------------------------------------------------------------");
2015-01-25 22:17:06 +01:00
};
2015-04-11 09:38:30 +02:00
size_t audio::river::io::Node::getNumberOfInterface(enum audio::river::modeInterface _interfaceType) {
2015-02-19 22:00:21 +01:00
size_t out = 0;
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
2015-02-19 22:00:21 +01:00
continue;
}
2015-02-24 22:20:11 +01:00
if (m_list[iii]->getMode() == _interfaceType) {
2015-02-19 22:00:21 +01:00
out++;
}
}
return out;
}
2015-04-11 09:38:30 +02:00
size_t audio::river::io::Node::getNumberOfInterfaceAvaillable(enum audio::river::modeInterface _interfaceType) {
2015-03-04 22:15:35 +01:00
size_t out = 0;
for (size_t iii=0; iii<m_listAvaillable.size(); ++iii) {
2017-09-26 15:57:44 +02:00
auto element = m_listAvaillable[iii].lock();
2018-06-19 22:13:48 +02:00
if (element == null) {
2015-03-04 22:15:35 +01:00
continue;
}
if (element->getMode() == _interfaceType) {
out++;
}
}
return out;
}
2015-02-19 22:00:21 +01:00
2016-07-19 21:43:58 +02:00
void audio::river::io::Node::registerAsRemote(const ememory::SharedPtr<audio::river::Interface>& _interface) {
2017-09-26 15:57:44 +02:00
auto it = m_listAvaillable.begin();
2015-02-04 21:08:06 +01:00
while (it != m_listAvaillable.end()) {
if (it->expired() == true) {
it = m_listAvaillable.erase(it);
continue;
2015-02-04 21:08:06 +01:00
}
++it;
}
2017-08-28 00:08:50 +02:00
m_listAvaillable.pushBack(_interface);
2015-02-04 21:08:06 +01:00
}
2016-07-19 21:43:58 +02:00
void audio::river::io::Node::interfaceAdd(const ememory::SharedPtr<audio::river::Interface>& _interface) {
2015-01-28 23:01:12 +01:00
{
2017-09-14 00:59:21 +02:00
ethread::UniqueLock lock(m_mutex);
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_list.size(); ++iii) {
if (_interface == m_list[iii]) {
2015-01-28 23:01:12 +01:00
return;
}
2015-01-25 22:17:06 +01:00
}
RIVER_INFO("ADD interface for stream : '" << m_name << "' mode=" << (m_isInput?"input":"output") );
2017-08-28 00:08:50 +02:00
m_list.pushBack(_interface);
2015-01-25 22:17:06 +01:00
}
2015-01-26 22:04:29 +01:00
if (m_list.size() == 1) {
startInGroup();
2015-01-26 22:04:29 +01:00
}
2015-01-25 22:17:06 +01:00
}
2016-07-19 21:43:58 +02:00
void audio::river::io::Node::interfaceRemove(const ememory::SharedPtr<audio::river::Interface>& _interface) {
2015-01-28 23:01:12 +01:00
{
2017-09-14 00:59:21 +02:00
ethread::UniqueLock lock(m_mutex);
2015-01-28 23:01:12 +01:00
for (size_t iii=0; iii< m_list.size(); ++iii) {
if (_interface == m_list[iii]) {
m_list.erase(m_list.begin()+iii);
RIVER_INFO("RM interface for stream : '" << m_name << "' mode=" << (m_isInput?"input":"output") );
2015-01-28 23:01:12 +01:00
break;
}
2015-01-25 22:17:06 +01:00
}
}
if (m_list.size() == 0) {
stopInGroup();
2015-01-25 22:17:06 +01:00
}
}
2015-02-04 21:08:06 +01:00
2015-04-11 09:38:30 +02:00
void audio::river::io::Node::volumeChange() {
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii< m_listAvaillable.size(); ++iii) {
2017-09-26 15:57:44 +02:00
auto node = m_listAvaillable[iii].lock();
2018-06-19 22:13:48 +02:00
if (node != null) {
2015-02-04 21:08:06 +01:00
node->systemVolumeChange();
}
}
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Node::newInput(const void* _inputBuffer,
uint32_t _nbChunk,
const audio::Time& _time) {
2018-06-19 22:13:48 +02:00
if (_inputBuffer == null) {
return;
}
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
continue;
}
2015-04-11 09:38:30 +02:00
if (m_list[iii]->getMode() != audio::river::modeInterface_input) {
continue;
}
2015-02-24 22:20:11 +01:00
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName());
m_list[iii]->systemNewInputData(_time, _inputBuffer, _nbChunk);
}
RIVER_VERBOSE("data Input size request :" << _nbChunk << " [ END ]");
return;
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Node::newOutput(void* _outputBuffer,
2015-06-26 22:20:04 +02:00
uint32_t _nbChunk,
const audio::Time& _time) {
2018-06-19 22:13:48 +02:00
if (_outputBuffer == null) {
return;
}
enum audio::format muxerFormatType = m_process.getInputConfig().getFormat();
2017-08-28 00:08:50 +02:00
etk::Vector<uint8_t> outputTmp2;
uint32_t nbByteTmpBuffer = audio::getFormatBytes(muxerFormatType)*m_process.getInputConfig().getMap().size()*_nbChunk;
RIVER_VERBOSE("resize=" << nbByteTmpBuffer);
outputTmp2.resize(nbByteTmpBuffer);
if (muxerFormatType == audio::format_int8_on_int16) {
//////////////////////////////////////////////////////////////////////////////////////////////////
// process 16 bits
//////////////////////////////////////////////////////////////////////////////////////////////////
// $$$$ change the int16
2017-08-28 00:08:50 +02:00
etk::Vector<int16_t> output;
RIVER_VERBOSE("resize=" << _nbChunk*m_process.getInputConfig().getMap().size());
output.resize(_nbChunk*m_process.getInputConfig().getMap().size(), 0);
// $$$$ change the int16
2018-06-19 22:13:48 +02:00
const int16_t* outputTmp = null;
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
continue;
}
if (m_list[iii]->getMode() != audio::river::modeInterface_output) {
continue;
}
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName() << " " << iii);
// clear datas ...
memset(&outputTmp2[0], 0, nbByteTmpBuffer);
RIVER_VERBOSE(" request Data="<< _nbChunk << " time=" << _time);
m_list[iii]->systemNeedOutputData(_time, &outputTmp2[0], _nbChunk, audio::getFormatBytes(muxerFormatType)*m_process.getInputConfig().getMap().size());
// $$$$ change the int16
outputTmp = reinterpret_cast<const int16_t*>(&outputTmp2[0]);
RIVER_VERBOSE(" Mix it ...");
// Add data to the output tmp buffer:
for (size_t kkk=0; kkk<output.size(); ++kkk) {
output[kkk] += outputTmp[kkk];
}
// TODO : if a signal is upper than 256* the maximum of 1 it can create a real problem ...
}
RIVER_VERBOSE(" End stack process data ...");
m_process.processIn(&output[0], _nbChunk, _outputBuffer, _nbChunk);
} else if ( muxerFormatType == audio::format_int16_on_int32
|| muxerFormatType == audio::format_int24_on_int32) {
//////////////////////////////////////////////////////////////////////////////////////////////////
// process 32 bits
//////////////////////////////////////////////////////////////////////////////////////////////////
2017-08-28 00:08:50 +02:00
etk::Vector<int32_t> output;
RIVER_VERBOSE("resize=" << _nbChunk*m_process.getInputConfig().getMap().size());
output.resize(_nbChunk*m_process.getInputConfig().getMap().size(), 0);
2018-06-19 22:13:48 +02:00
const int32_t* outputTmp = null;
2015-03-18 21:48:57 +01:00
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
2015-03-18 21:48:57 +01:00
continue;
}
2015-04-11 09:38:30 +02:00
if (m_list[iii]->getMode() != audio::river::modeInterface_output) {
2015-03-18 21:48:57 +01:00
continue;
}
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName() << " " << iii);
// clear datas ...
memset(&outputTmp2[0], 0, nbByteTmpBuffer);
2015-03-18 21:48:57 +01:00
RIVER_VERBOSE(" request Data="<< _nbChunk << " time=" << _time);
m_list[iii]->systemNeedOutputData(_time, &outputTmp2[0], _nbChunk, audio::getFormatBytes(muxerFormatType)*m_process.getInputConfig().getMap().size());
2015-03-18 21:48:57 +01:00
outputTmp = reinterpret_cast<const int32_t*>(&outputTmp2[0]);
RIVER_VERBOSE(" Mix it ...");
// Add data to the output tmp buffer:
2015-03-18 21:48:57 +01:00
for (size_t kkk=0; kkk<output.size(); ++kkk) {
output[kkk] += outputTmp[kkk];
}
// TODO : if a signal is upper than 256* (for 24 bits) or 65335* (for 16 bits) the maximum of 1 it can create a real problem ...
}
RIVER_VERBOSE(" End stack process data ...");
m_process.processIn(&output[0], _nbChunk, _outputBuffer, _nbChunk);
} else if (muxerFormatType == audio::format_int32_on_int64) {
//////////////////////////////////////////////////////////////////////////////////////////////////
// process 64 bits
//////////////////////////////////////////////////////////////////////////////////////////////////
2017-08-28 00:08:50 +02:00
etk::Vector<int64_t> output;
RIVER_VERBOSE("resize=" << _nbChunk*m_process.getInputConfig().getMap().size());
output.resize(_nbChunk*m_process.getInputConfig().getMap().size(), 0);
2018-06-19 22:13:48 +02:00
const int64_t* outputTmp = null;
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
continue;
}
if (m_list[iii]->getMode() != audio::river::modeInterface_output) {
continue;
}
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName() << " " << iii);
// clear datas ...
memset(&outputTmp2[0], 0, nbByteTmpBuffer);
RIVER_VERBOSE(" request Data="<< _nbChunk << " time=" << _time);
m_list[iii]->systemNeedOutputData(_time, &outputTmp2[0], _nbChunk, audio::getFormatBytes(muxerFormatType)*m_process.getInputConfig().getMap().size());
outputTmp = reinterpret_cast<const int64_t*>(&outputTmp2[0]);
RIVER_VERBOSE(" Mix it ...");
// Add data to the output tmp buffer:
for (size_t kkk=0; kkk<output.size(); ++kkk) {
output[kkk] += outputTmp[kkk];
}
// TODO : if a signal is upper than 2000000000* the maximum of 1 it can create a real problem ...
}
RIVER_VERBOSE(" End stack process data ...");
m_process.processIn(&output[0], _nbChunk, _outputBuffer, _nbChunk);
} else if (muxerFormatType == audio::format_float) {
//////////////////////////////////////////////////////////////////////////////////////////////////
// process 32 bits FLOAT
//////////////////////////////////////////////////////////////////////////////////////////////////
2017-08-28 00:08:50 +02:00
etk::Vector<float> output;
RIVER_VERBOSE("resize=" << _nbChunk*m_process.getInputConfig().getMap().size());
output.resize(_nbChunk*m_process.getInputConfig().getMap().size(), 0);
2018-06-19 22:13:48 +02:00
const float* outputTmp = null;
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
continue;
}
if (m_list[iii]->getMode() != audio::river::modeInterface_output) {
continue;
}
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName() << " " << iii);
// clear datas ...
memset(&outputTmp2[0], 0, nbByteTmpBuffer);
RIVER_VERBOSE(" request Data="<< _nbChunk << " time=" << _time);
m_list[iii]->systemNeedOutputData(_time, &outputTmp2[0], _nbChunk, audio::getFormatBytes(muxerFormatType)*m_process.getInputConfig().getMap().size());
outputTmp = reinterpret_cast<const float*>(&outputTmp2[0]);
RIVER_VERBOSE(" Mix it ...");
// Add data to the output tmp buffer:
for (size_t kkk=0; kkk<output.size(); ++kkk) {
output[kkk] += outputTmp[kkk];
}
}
RIVER_VERBOSE(" End stack process data ...");
m_process.processIn(&output[0], _nbChunk, _outputBuffer, _nbChunk);
} else if (muxerFormatType == audio::format_double) {
//////////////////////////////////////////////////////////////////////////////////////////////////
// process 64 bits FLOAT
//////////////////////////////////////////////////////////////////////////////////////////////////
2017-08-28 00:08:50 +02:00
etk::Vector<double> output;
RIVER_VERBOSE("resize=" << _nbChunk*m_process.getInputConfig().getMap().size());
output.resize(_nbChunk*m_process.getInputConfig().getMap().size(), 0);
2018-06-19 22:13:48 +02:00
const double* outputTmp = null;
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
continue;
}
if (m_list[iii]->getMode() != audio::river::modeInterface_output) {
continue;
}
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName() << " " << iii);
// clear datas ...
memset(&outputTmp2[0], 0, nbByteTmpBuffer);
RIVER_VERBOSE(" request Data="<< _nbChunk << " time=" << _time);
m_list[iii]->systemNeedOutputData(_time, &outputTmp2[0], _nbChunk, audio::getFormatBytes(muxerFormatType)*m_process.getInputConfig().getMap().size());
outputTmp = reinterpret_cast<const double*>(&outputTmp2[0]);
RIVER_VERBOSE(" Mix it ...");
// Add data to the output tmp buffer:
for (size_t kkk=0; kkk<output.size(); ++kkk) {
output[kkk] += outputTmp[kkk];
}
}
RIVER_VERBOSE(" End stack process data ...");
m_process.processIn(&output[0], _nbChunk, _outputBuffer, _nbChunk);
} else {
RIVER_ERROR("Wrong demuxer type: " << muxerFormatType);
return;
}
// The feedback get the real output data (after processing ...==> then no nneed to specify for each channels
RIVER_VERBOSE(" Feedback :");
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii< m_list.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_list[iii] == null) {
continue;
}
2015-04-11 09:38:30 +02:00
if (m_list[iii]->getMode() != audio::river::modeInterface_feedback) {
continue;
}
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName() << " (feedback) time=" << _time);
2015-02-24 22:20:11 +01:00
m_list[iii]->systemNewInputData(_time, _outputBuffer, _nbChunk);
}
RIVER_VERBOSE("data Output size request :" << _nbChunk << " [ END ]");
return;
}
2015-02-18 15:22:48 +01:00
2018-10-23 22:19:32 +02:00
static void link(ememory::SharedPtr<etk::io::Interface>& _io, const etk::String& _first, const etk::String& _op, const etk::String& _second) {
2015-02-19 22:00:21 +01:00
if (_op == "->") {
2018-10-23 22:19:32 +02:00
*_io << " " << _first << " -> " << _second << ";\n";
2015-02-19 22:00:21 +01:00
} else if (_op == "<-") {
2018-10-23 22:19:32 +02:00
*_io << " " << _first << " -> " <<_second<< " [color=transparent];\n";
*_io << " " << _second << " -> " << _first << " [constraint=false];\n";
2015-02-19 22:00:21 +01:00
}
}
2018-10-23 22:19:32 +02:00
void audio::river::io::Node::generateDot(ememory::SharedPtr<etk::io::Interface>& _io) {
*_io << " subgraph clusterNode_" << m_uid << " {\n";
*_io << " color=blue;\n";
*_io << " label=\"[" << m_uid << "] IO::Node : " << m_name << "\";\n";
2015-02-18 23:37:09 +01:00
if (m_isInput == true) {
2018-10-23 22:19:32 +02:00
*_io << " node [shape=rarrow];\n";
*_io << " NODE_" << m_uid << "_HW_interface [ label=\"HW interface\\n interface=ALSA\\n stream=" << m_name << "\\n type=input\" ];\n";
2017-08-28 00:08:50 +02:00
etk::String nameIn;
etk::String nameOut;
2018-10-23 22:19:32 +02:00
m_process.generateDotProcess(_io, 3, m_uid, nameIn, nameOut, false);
*_io << " node [shape=square];\n";
*_io << " NODE_" << m_uid << "_demuxer [ label=\"DEMUXER\\n format=" << etk::toString(m_process.getOutputConfig().getFormat()) << "\" ];\n";
2015-02-18 23:37:09 +01:00
// Link all nodes :
2018-10-23 22:19:32 +02:00
*_io << " NODE_" << m_uid << "_HW_interface -> " << nameIn << " [arrowhead=\"open\"];\n";
*_io << " " << nameOut << " -> NODE_" << m_uid << "_demuxer [arrowhead=\"open\"];\n";
2015-02-18 23:37:09 +01:00
} else {
2015-04-11 09:38:30 +02:00
size_t nbOutput = getNumberOfInterfaceAvaillable(audio::river::modeInterface_output);
size_t nbfeedback = getNumberOfInterfaceAvaillable(audio::river::modeInterface_feedback);
2018-10-23 22:19:32 +02:00
*_io << " node [shape=larrow];\n";
*_io << " NODE_" << m_uid << "_HW_interface [ label=\"HW interface\\n interface=ALSA\\n stream=" << m_name << "\\n type=output\" ];\n";
2017-08-28 00:08:50 +02:00
etk::String nameIn;
etk::String nameOut;
2015-02-19 22:00:21 +01:00
if (nbOutput>0) {
2018-10-23 22:19:32 +02:00
m_process.generateDotProcess(_io, 3, m_uid, nameIn, nameOut, true);
2015-02-19 22:00:21 +01:00
}
2018-10-23 22:19:32 +02:00
*_io << " node [shape=square];\n";
2015-02-19 22:00:21 +01:00
if (nbOutput>0) {
2018-10-23 22:19:32 +02:00
*_io << " NODE_" << m_uid << "_muxer [ label=\"MUXER\\n format=" << etk::toString(m_process.getInputConfig().getFormat()) << "\" ];\n";
2015-02-19 22:00:21 +01:00
}
if (nbfeedback>0) {
2018-10-23 22:19:32 +02:00
*_io << " NODE_" << m_uid << "_demuxer [ label=\"DEMUXER\\n format=" << etk::toString(m_process.getOutputConfig().getFormat()) << "\" ];\n";
2015-02-19 22:00:21 +01:00
}
2015-02-18 23:37:09 +01:00
// Link all nodes :
2015-02-19 22:00:21 +01:00
if (nbOutput>0) {
2018-10-23 22:19:32 +02:00
link(_io, "NODE_" + etk::toString(m_uid) + "_HW_interface", "<-", nameOut);
link(_io, nameIn, "<-", "NODE_" + etk::toString(m_uid) + "_muxer");
2015-02-19 22:00:21 +01:00
}
if (nbfeedback>0) {
2018-10-23 22:19:32 +02:00
*_io << " NODE_" << m_uid << "_HW_interface -> NODE_" << m_uid << "_demuxer [arrowhead=\"open\"];\n";
2015-02-19 22:00:21 +01:00
}
if ( nbOutput>0
&& nbfeedback>0) {
2018-10-23 22:19:32 +02:00
*_io << " { rank=same; NODE_" << m_uid << "_demuxer; NODE_" << m_uid << "_muxer }\n";
2015-02-19 22:00:21 +01:00
}
2015-02-18 23:37:09 +01:00
}
2018-10-23 22:19:32 +02:00
*_io << " }\n \n";
2015-02-18 23:37:09 +01:00
2015-03-04 22:15:35 +01:00
for (size_t iii=0; iii< m_listAvaillable.size(); ++iii) {
if (m_listAvaillable[iii].expired() == true) {
continue;
}
2016-07-19 21:43:58 +02:00
ememory::SharedPtr<audio::river::Interface> element = m_listAvaillable[iii].lock();
2018-06-19 22:13:48 +02:00
if (element == null) {
2015-03-04 22:15:35 +01:00
continue;
}
bool isLink = false;
for (size_t jjj=0; jjj<m_list.size(); ++jjj) {
if (element == m_list[jjj]) {
isLink = true;
}
}
2018-06-19 22:13:48 +02:00
if (element != null) {
2015-03-04 22:15:35 +01:00
if (element->getMode() == modeInterface_input) {
2018-10-23 22:19:32 +02:00
element->generateDot(_io, "NODE_" + etk::toString(m_uid) + "_demuxer", isLink);
2015-03-04 22:15:35 +01:00
} else if (element->getMode() == modeInterface_output) {
2018-10-23 22:19:32 +02:00
element->generateDot(_io, "NODE_" + etk::toString(m_uid) + "_muxer", isLink);
2015-03-04 22:15:35 +01:00
} else if (element->getMode() == modeInterface_feedback) {
2018-10-23 22:19:32 +02:00
element->generateDot(_io, "NODE_" + etk::toString(m_uid) + "_demuxer", isLink);
2015-02-18 23:37:09 +01:00
} else {
}
}
}
2015-02-18 15:22:48 +01:00
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Node::startInGroup() {
2016-07-19 21:43:58 +02:00
ememory::SharedPtr<audio::river::io::Group> group = m_group.lock();
2018-06-19 22:13:48 +02:00
if (group != null) {
group->start();
} else {
start();
}
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Node::stopInGroup() {
2016-07-19 21:43:58 +02:00
ememory::SharedPtr<audio::river::io::Group> group = m_group.lock();
2018-06-19 22:13:48 +02:00
if (group != null) {
group->stop();
} else {
stop();
}
2016-05-02 22:01:55 +02:00
}