2015-01-25 22:17:06 +01:00
|
|
|
/** @file
|
|
|
|
* @author Edouard DUPIN
|
|
|
|
* @copyright 2015, Edouard DUPIN, all right reserved
|
|
|
|
* @license APACHE v2.0 (see license file)
|
|
|
|
*/
|
|
|
|
|
2015-01-26 22:04:29 +01:00
|
|
|
#include "Node.h"
|
2015-02-05 19:10:53 +01:00
|
|
|
#include <river/debug.h>
|
2015-01-25 22:17:06 +01:00
|
|
|
|
2015-01-27 21:01:52 +01:00
|
|
|
#undef __class__
|
|
|
|
#define __class__ "io::Node"
|
|
|
|
|
2015-01-25 22:17:06 +01:00
|
|
|
|
2015-02-24 22:20:11 +01:00
|
|
|
river::io::Node::Node(const std::string& _name, const std11::shared_ptr<const ejson::Object>& _config) :
|
2015-02-03 21:29:23 +01:00
|
|
|
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++;
|
2015-02-05 21:33:12 +01:00
|
|
|
RIVER_INFO("-----------------------------------------------------------------");
|
|
|
|
RIVER_INFO("-- CREATE NODE --");
|
|
|
|
RIVER_INFO("-----------------------------------------------------------------");
|
2015-02-08 15:11:02 +01:00
|
|
|
drain::IOFormatInterface interfaceFormat;
|
|
|
|
drain::IOFormatInterface hardwareFormat;
|
2015-02-03 21:29:23 +01:00
|
|
|
/**
|
2015-02-16 21:04:18 +01:00
|
|
|
io:"input", # input, output or aec
|
2015-02-03 21:29:23 +01:00
|
|
|
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",
|
|
|
|
],
|
2015-02-16 21:04:18 +01:00
|
|
|
# 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",
|
2015-02-03 21:29:23 +01:00
|
|
|
*/
|
2015-02-16 21:04:18 +01:00
|
|
|
std::string interfaceType = m_config->getStringValue("io");
|
|
|
|
if ( interfaceType == "input"
|
2015-02-19 22:00:21 +01:00
|
|
|
|| interfaceType == "PAinput"
|
2015-02-16 21:04:18 +01:00
|
|
|
|| interfaceType == "aec") {
|
|
|
|
m_isInput = true;
|
2015-02-03 21:29:23 +01:00
|
|
|
} else {
|
2015-02-16 21:04:18 +01:00
|
|
|
m_isInput = false;
|
2015-02-03 21:29:23 +01:00
|
|
|
}
|
2015-02-16 21:04:18 +01:00
|
|
|
|
2015-02-13 21:06:55 +01:00
|
|
|
int32_t frequency = m_config->getNumberValue("frequency", 1);
|
2015-02-16 21:04:18 +01:00
|
|
|
// Get audio format type:
|
2015-02-03 21:29:23 +01:00
|
|
|
std::string type = m_config->getStringValue("type", "int16");
|
2015-02-16 21:04:18 +01:00
|
|
|
enum audio::format formatType = audio::getFormatFromString(type);
|
|
|
|
// Get volume stage :
|
2015-02-03 21:29:23 +01:00
|
|
|
std::string volumeName = m_config->getStringValue("volume-name", "");
|
|
|
|
if (volumeName != "") {
|
2015-02-05 21:33:12 +01:00
|
|
|
RIVER_INFO("add node volume stage : '" << volumeName << "'");
|
2015-02-03 23:29:34 +01:00
|
|
|
// use global manager for volume ...
|
2015-02-05 19:10:53 +01:00
|
|
|
m_volume = river::io::Manager::getInstance()->getVolumeGroup(volumeName);
|
2015-02-03 21:29:23 +01:00
|
|
|
}
|
2015-02-16 21:04:18 +01:00
|
|
|
// Get map type :
|
2015-02-04 22:39:05 +01:00
|
|
|
std::vector<audio::channel> map;
|
2015-02-24 22:20:11 +01:00
|
|
|
const std11::shared_ptr<const ejson::Array> listChannelMap = m_config->getArray("channel-map");
|
2015-02-16 21:04:18 +01:00
|
|
|
if ( listChannelMap == nullptr
|
|
|
|
|| listChannelMap->size() == 0) {
|
|
|
|
// set default channel property:
|
|
|
|
map.push_back(audio::channel_frontLeft);
|
|
|
|
map.push_back(audio::channel_frontRight);
|
2015-02-02 22:04:11 +01:00
|
|
|
} else {
|
2015-02-16 21:04:18 +01:00
|
|
|
for (size_t iii=0; iii<listChannelMap->size(); ++iii) {
|
|
|
|
std::string value = listChannelMap->getStringValue(iii);
|
|
|
|
map.push_back(audio::getChannelFromString(value));
|
2015-02-08 15:11:02 +01:00
|
|
|
}
|
2015-01-25 22:17:06 +01:00
|
|
|
}
|
2015-02-16 21:04:18 +01:00
|
|
|
hardwareFormat.set(map, formatType, frequency);
|
2015-01-25 22:17:06 +01:00
|
|
|
|
|
|
|
|
2015-02-16 21:04:18 +01:00
|
|
|
std::string muxerDemuxerConfig = m_config->getStringValue("mux-demux-type", "int16-on-int32");
|
|
|
|
enum audio::format muxerFormatType = audio::getFormatFromString(muxerDemuxerConfig);
|
2015-01-26 22:04:29 +01:00
|
|
|
if (m_isInput == true) {
|
2015-02-16 21:04:18 +01:00
|
|
|
if (muxerFormatType != audio::format_int16) {
|
|
|
|
RIVER_CRITICAL("not supported demuxer type ... " << muxerFormatType << " for INPUT set in file:" << muxerDemuxerConfig);
|
|
|
|
}
|
2015-01-26 22:04:29 +01:00
|
|
|
} else {
|
2015-02-16 21:04:18 +01:00
|
|
|
if (muxerFormatType != audio::format_int16_on_int32) {
|
|
|
|
RIVER_CRITICAL("not supported demuxer type ... " << muxerFormatType << " for OUTPUT set in file:" << muxerDemuxerConfig);
|
|
|
|
}
|
2015-01-25 22:17:06 +01:00
|
|
|
}
|
2015-02-16 21:04:18 +01:00
|
|
|
// no map change and no frequency change ...
|
|
|
|
interfaceFormat.set(map, muxerFormatType, frequency);
|
|
|
|
// configure process interface
|
2015-02-08 15:11:02 +01:00
|
|
|
if (m_isInput == true) {
|
|
|
|
m_process.setInputConfig(hardwareFormat);
|
|
|
|
m_process.setOutputConfig(interfaceFormat);
|
|
|
|
} else {
|
|
|
|
m_process.setOutputConfig(hardwareFormat);
|
|
|
|
m_process.setInputConfig(interfaceFormat);
|
|
|
|
}
|
2015-02-16 21:04:18 +01:00
|
|
|
//m_process.updateInterAlgo();
|
2015-01-25 22:17:06 +01:00
|
|
|
}
|
|
|
|
|
2015-02-05 19:10:53 +01:00
|
|
|
river::io::Node::~Node() {
|
2015-02-05 21:33:12 +01:00
|
|
|
RIVER_INFO("-----------------------------------------------------------------");
|
|
|
|
RIVER_INFO("-- DESTROY NODE --");
|
|
|
|
RIVER_INFO("-----------------------------------------------------------------");
|
2015-01-25 22:17:06 +01:00
|
|
|
};
|
|
|
|
|
2015-02-19 22:00:21 +01:00
|
|
|
size_t river::io::Node::getNumberOfInterface(enum river::modeInterface _interfaceType) {
|
|
|
|
size_t out = 0;
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii<m_list.size(); ++iii) {
|
|
|
|
if (m_list[iii] == nullptr) {
|
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-02-24 22:20:11 +01:00
|
|
|
void river::io::Node::registerAsRemote(const std11::shared_ptr<river::Interface>& _interface) {
|
|
|
|
std::vector<std11::weak_ptr<river::Interface> >::iterator 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);
|
|
|
|
}
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
m_listAvaillable.push_back(_interface);
|
|
|
|
}
|
|
|
|
|
2015-02-24 22:20:11 +01:00
|
|
|
void river::io::Node::interfaceAdd(const std11::shared_ptr<river::Interface>& _interface) {
|
2015-01-28 23:01:12 +01:00
|
|
|
{
|
2015-02-24 22:20:11 +01:00
|
|
|
std11::unique_lock<std11::mutex> lock(m_mutex);
|
|
|
|
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
|
|
|
}
|
2015-02-05 21:33:12 +01:00
|
|
|
RIVER_INFO("ADD interface for stream : '" << m_name << "' mode=" << (m_isInput?"input":"output") );
|
2015-01-28 23:01:12 +01:00
|
|
|
m_list.push_back(_interface);
|
2015-01-25 22:17:06 +01:00
|
|
|
}
|
2015-01-26 22:04:29 +01:00
|
|
|
if (m_list.size() == 1) {
|
2015-03-03 21:28:07 +01:00
|
|
|
startInGroup();
|
2015-01-26 22:04:29 +01:00
|
|
|
}
|
2015-01-25 22:17:06 +01:00
|
|
|
}
|
|
|
|
|
2015-02-24 22:20:11 +01:00
|
|
|
void river::io::Node::interfaceRemove(const std11::shared_ptr<river::Interface>& _interface) {
|
2015-01-28 23:01:12 +01:00
|
|
|
{
|
2015-02-24 22:20:11 +01:00
|
|
|
std11::unique_lock<std11::mutex> 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);
|
2015-02-05 21:33:12 +01:00
|
|
|
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) {
|
2015-03-03 21:28:07 +01:00
|
|
|
stopInGroup();
|
2015-01-25 22:17:06 +01:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-02-04 21:08:06 +01:00
|
|
|
|
2015-02-05 19:10:53 +01:00
|
|
|
void river::io::Node::volumeChange() {
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii< m_listAvaillable.size(); ++iii) {
|
|
|
|
std11::shared_ptr<river::Interface> node = m_listAvaillable[iii].lock();
|
2015-02-04 21:08:06 +01:00
|
|
|
if (node != nullptr) {
|
|
|
|
node->systemVolumeChange();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-16 21:04:18 +01:00
|
|
|
|
|
|
|
int32_t river::io::Node::newInput(const void* _inputBuffer,
|
|
|
|
uint32_t _nbChunk,
|
2015-02-24 22:20:11 +01:00
|
|
|
const std11::chrono::system_clock::time_point& _time) {
|
2015-02-16 21:04:18 +01:00
|
|
|
if (_inputBuffer == nullptr) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
const int16_t* inputBuffer = static_cast<const int16_t *>(_inputBuffer);
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii< m_list.size(); ++iii) {
|
|
|
|
if (m_list[iii] == nullptr) {
|
2015-02-16 21:04:18 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
if (m_list[iii]->getMode() != river::modeInterface_input) {
|
2015-02-16 21:04:18 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName());
|
|
|
|
m_list[iii]->systemNewInputData(_time, inputBuffer, _nbChunk);
|
2015-02-16 21:04:18 +01:00
|
|
|
}
|
|
|
|
RIVER_VERBOSE("data Input size request :" << _nbChunk << " [ END ]");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t river::io::Node::newOutput(void* _outputBuffer,
|
|
|
|
uint32_t _nbChunk,
|
2015-02-24 22:20:11 +01:00
|
|
|
const std11::chrono::system_clock::time_point& _time) {
|
2015-02-16 21:04:18 +01:00
|
|
|
if (_outputBuffer == nullptr) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
std::vector<int32_t> output;
|
|
|
|
RIVER_VERBOSE("resize=" << _nbChunk*m_process.getInputConfig().getMap().size());
|
|
|
|
output.resize(_nbChunk*m_process.getInputConfig().getMap().size(), 0);
|
|
|
|
const int32_t* outputTmp = nullptr;
|
|
|
|
std::vector<uint8_t> outputTmp2;
|
|
|
|
RIVER_VERBOSE("resize=" << sizeof(int32_t)*m_process.getInputConfig().getMap().size()*_nbChunk);
|
|
|
|
outputTmp2.resize(sizeof(int32_t)*m_process.getInputConfig().getMap().size()*_nbChunk, 0);
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii< m_list.size(); ++iii) {
|
|
|
|
if (m_list[iii] == nullptr) {
|
2015-02-16 21:04:18 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
if (m_list[iii]->getMode() != river::modeInterface_output) {
|
2015-02-16 21:04:18 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
RIVER_VERBOSE(" IO name="<< m_list[iii]->getName());
|
2015-02-16 21:04:18 +01:00
|
|
|
// clear datas ...
|
|
|
|
memset(&outputTmp2[0], 0, sizeof(int32_t)*m_process.getInputConfig().getMap().size()*_nbChunk);
|
2015-03-03 21:28:07 +01:00
|
|
|
RIVER_VERBOSE(" request Data="<< _nbChunk << " time=" << _time);
|
2015-02-24 22:20:11 +01:00
|
|
|
m_list[iii]->systemNeedOutputData(_time, &outputTmp2[0], _nbChunk, sizeof(int32_t)*m_process.getInputConfig().getMap().size());
|
2015-02-16 21:04:18 +01:00
|
|
|
RIVER_VERBOSE(" Mix it ...");
|
|
|
|
outputTmp = reinterpret_cast<const int32_t*>(&outputTmp2[0]);
|
|
|
|
// 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(&outputTmp2[0], _nbChunk, _outputBuffer, _nbChunk);
|
|
|
|
RIVER_VERBOSE(" Feedback :");
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii< m_list.size(); ++iii) {
|
|
|
|
if (m_list[iii] == nullptr) {
|
2015-02-16 21:04:18 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-02-24 22:20:11 +01:00
|
|
|
if (m_list[iii]->getMode() != river::modeInterface_feedback) {
|
2015-02-16 21:04:18 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-03-03 21:28:07 +01:00
|
|
|
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);
|
2015-02-16 21:04:18 +01:00
|
|
|
}
|
|
|
|
RIVER_VERBOSE("data Output size request :" << _nbChunk << " [ END ]");
|
|
|
|
return 0;
|
|
|
|
}
|
2015-02-18 15:22:48 +01:00
|
|
|
|
2015-02-19 22:00:21 +01:00
|
|
|
static void link(etk::FSNode& _node, const std::string& _first, const std::string& _op, const std::string& _second) {
|
|
|
|
if (_op == "->") {
|
|
|
|
_node << " " << _first << " -> " << _second << ";\n";
|
|
|
|
} else if (_op == "<-") {
|
|
|
|
_node << " " << _first << " -> " <<_second<< " [color=transparent];\n";
|
|
|
|
_node << " " << _second << " -> " << _first << " [constraint=false];\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-02-18 15:22:48 +01:00
|
|
|
void river::io::Node::generateDot(etk::FSNode& _node) {
|
2015-02-18 23:37:09 +01:00
|
|
|
_node << "subgraph clusterNode_" << m_uid << " {\n";
|
|
|
|
_node << " color=blue;\n";
|
2015-02-19 22:00:21 +01:00
|
|
|
_node << " label=\"[" << m_uid << "] IO::Node : " << m_name << "\";\n";
|
2015-02-18 23:37:09 +01:00
|
|
|
if (m_isInput == true) {
|
2015-02-19 22:00:21 +01:00
|
|
|
_node << " node [shape=rarrow];\n";
|
2015-02-18 23:37:09 +01:00
|
|
|
_node << " NODE_" << m_uid << "_HW_interface [ label=\"HW interface\n interface=ALSA\n stream=MICROPHONE\n type=input\" ];\n";
|
|
|
|
_node << " subgraph clusterNode_" << m_uid << "_process {\n";
|
|
|
|
_node << " label=\"Drain::Process\";\n";
|
|
|
|
_node << " node [shape=ellipse];\n";
|
2015-02-19 22:00:21 +01:00
|
|
|
_node << " node_ALGO_" << m_uid << "_in [ label=\"format=xxx\n freq=yyy\n channelMap={left,right}\" ];\n";
|
|
|
|
_node << " node_ALGO_" << m_uid << "_out [ label=\"format=xxx\n freq=yyy\n channelMap={left,right}\" ];\n";
|
2015-02-18 23:37:09 +01:00
|
|
|
|
|
|
|
_node << " }\n";
|
|
|
|
_node << " node [shape=square];\n";
|
|
|
|
_node << " NODE_" << m_uid << "_demuxer [ label=\"DEMUXER\n format=xxx\" ];\n";
|
|
|
|
// Link all nodes :
|
2015-02-19 22:00:21 +01:00
|
|
|
_node << " NODE_" << m_uid << "_HW_interface -> node_ALGO_" << m_uid << "_in [arrowhead=\"open\"];\n";
|
|
|
|
_node << " node_ALGO_" << m_uid << "_in -> node_ALGO_" << m_uid << "_out [arrowhead=\"open\"];\n";
|
|
|
|
_node << " node_ALGO_" << m_uid << "_out -> NODE_" << m_uid << "_demuxer [arrowhead=\"open\"];\n";
|
2015-02-18 23:37:09 +01:00
|
|
|
} else {
|
2015-02-19 22:00:21 +01:00
|
|
|
size_t nbOutput = getNumberOfInterface(river::modeInterface_output);
|
|
|
|
size_t nbfeedback = getNumberOfInterface(river::modeInterface_feedback);
|
|
|
|
_node << " node [shape=larrow];\n";
|
2015-02-18 23:37:09 +01:00
|
|
|
_node << " NODE_" << m_uid << "_HW_interface [ label=\"HW interface\n interface=ALSA\n stream=SPEAKER\n type=output\" ];\n";
|
2015-02-19 22:00:21 +01:00
|
|
|
if (nbOutput>0) {
|
|
|
|
_node << " subgraph clusterNode_" << m_uid << "_process {\n";
|
|
|
|
_node << " label=\"Drain::Process\";\n";
|
|
|
|
_node << " node [shape=ellipse];\n";
|
|
|
|
_node << " node_ALGO_" << m_uid << "_out [ label=\"format=xxx\n freq=yyy\n channelMap={left,right}\" ];\n";
|
|
|
|
_node << " node_ALGO_" << m_uid << "_in [ label=\"format=xxx\n freq=yyy\n channelMap={left,right}\" ];\n";
|
|
|
|
_node << " }\n";
|
|
|
|
}
|
2015-02-18 23:37:09 +01:00
|
|
|
_node << " node [shape=square];\n";
|
2015-02-19 22:00:21 +01:00
|
|
|
if (nbOutput>0) {
|
|
|
|
_node << " NODE_" << m_uid << "_muxer [ label=\"MUXER\n format=xxx\" ];\n";
|
|
|
|
}
|
|
|
|
if (nbfeedback>0) {
|
|
|
|
_node << " NODE_" << m_uid << "_demuxer [ label=\"DEMUXER\n format=xxx\" ];\n";
|
|
|
|
}
|
2015-02-18 23:37:09 +01:00
|
|
|
// Link all nodes :
|
2015-02-19 22:00:21 +01:00
|
|
|
if (nbOutput>0) {
|
|
|
|
link(_node, "NODE_" + etk::to_string(m_uid) + "_HW_interface", "<-", "node_ALGO_" + etk::to_string(m_uid) + "_out");
|
|
|
|
link(_node, "node_ALGO_" + etk::to_string(m_uid) + "_out", "<-", "node_ALGO_" + etk::to_string(m_uid) + "_in");
|
|
|
|
link(_node, "node_ALGO_" + etk::to_string(m_uid) + "_in", "<-", "NODE_" + etk::to_string(m_uid) + "_muxer");
|
|
|
|
}
|
|
|
|
if (nbfeedback>0) {
|
|
|
|
_node << " NODE_" << m_uid << "_HW_interface -> NODE_" << m_uid << "_demuxer [arrowhead=\"open\"];\n";
|
|
|
|
}
|
|
|
|
if ( nbOutput>0
|
|
|
|
&& nbfeedback>0) {
|
|
|
|
_node << " { rank=same; NODE_" << m_uid << "_demuxer; NODE_" << m_uid << "_muxer }\n";
|
|
|
|
}
|
2015-02-18 23:37:09 +01:00
|
|
|
|
|
|
|
}
|
2015-02-19 22:00:21 +01:00
|
|
|
_node << "}\n";
|
2015-02-18 23:37:09 +01:00
|
|
|
|
2015-02-24 22:20:11 +01:00
|
|
|
for (size_t iii=0; iii< m_list.size(); ++iii) {
|
|
|
|
if (m_list[iii] != nullptr) {
|
|
|
|
if (m_list[iii]->getMode() == modeInterface_input) {
|
|
|
|
m_list[iii]->generateDot(_node, "NODE_" + etk::to_string(m_uid) + "_demuxer");
|
|
|
|
} else if (m_list[iii]->getMode() == modeInterface_output) {
|
|
|
|
m_list[iii]->generateDot(_node, "NODE_" + etk::to_string(m_uid) + "_muxer");
|
|
|
|
} else if (m_list[iii]->getMode() == modeInterface_feedback) {
|
|
|
|
m_list[iii]->generateDot(_node, "NODE_" + etk::to_string(m_uid) + "_demuxer");
|
2015-02-18 23:37:09 +01:00
|
|
|
} else {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-18 15:22:48 +01:00
|
|
|
}
|
|
|
|
|
2015-03-03 21:28:07 +01:00
|
|
|
|
|
|
|
void river::io::Node::startInGroup() {
|
|
|
|
std11::shared_ptr<river::io::Group> group = m_group.lock();
|
|
|
|
if (group != nullptr) {
|
|
|
|
group->start();
|
|
|
|
} else {
|
|
|
|
start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void river::io::Node::stopInGroup() {
|
|
|
|
std11::shared_ptr<river::io::Group> group = m_group.lock();
|
|
|
|
if (group != nullptr) {
|
|
|
|
group->stop();
|
|
|
|
} else {
|
|
|
|
stop();
|
|
|
|
}
|
|
|
|
}
|