/** @file * @author Edouard DUPIN * @copyright 2015, Edouard DUPIN, all right reserved * @license APACHE v2.0 (see license file) */ #include "debug.h" #include "Interface.h" #include "io/Node.h" #include #include #include #include #undef __class__ #define __class__ "Interface" river::Interface::Interface(void) : m_node(nullptr), m_name(""), m_volume(0.0f) { } bool river::Interface::init(const std::string& _name, float _freq, const std::vector& _map, audio::format _format, const std::shared_ptr& _node, const std::shared_ptr& _config) { m_name = _name; m_node = _node; m_volume = 0.0f; m_config = _config; m_mode = river::modeInterface_unknow; std::string type = m_config->getStringValue("io", "error"); if (type == "output") { m_mode = river::modeInterface_output; } else if (type == "input") { m_mode = river::modeInterface_input; } else if (type == "feedback") { m_mode = river::modeInterface_feedback; } // register interface to be notify from the volume change. m_node->registerAsRemote(shared_from_this()); // Create convertion interface if ( m_node->isInput() == true && m_mode == river::modeInterface_input) { m_process.setInputConfig(m_node->getInterfaceFormat()); // add all time the volume stage : std::shared_ptr algo = drain::Volume::create(); //algo->setInputFormat(m_node->getInterfaceFormat()); algo->setName("volume"); m_process.pushBack(algo); RIVER_INFO("add basic volume stage (1)"); std::shared_ptr tmpVolume = m_node->getVolume(); if (tmpVolume != nullptr) { RIVER_INFO(" add volume for node"); algo->addVolumeStage(tmpVolume); } m_process.setOutputConfig(drain::IOFormatInterface(_map, _format, _freq)); } else if ( m_node->isOutput() == true && m_mode == river::modeInterface_output) { m_process.setInputConfig(drain::IOFormatInterface(_map, _format, _freq)); // add all time the volume stage : std::shared_ptr algo = drain::Volume::create(); //algo->setOutputFormat(m_node->getInterfaceFormat()); algo->setName("volume"); m_process.pushBack(algo); RIVER_INFO("add basic volume stage (2)"); std::shared_ptr tmpVolume = m_node->getVolume(); if (tmpVolume != nullptr) { RIVER_INFO(" add volume for node"); algo->addVolumeStage(tmpVolume); } m_process.setOutputConfig(m_node->getInterfaceFormat()); } else if ( m_node->isOutput() == true && m_mode == river::modeInterface_feedback) { m_process.setInputConfig(m_node->getHarwareFormat()); // add all time the volume stage : std::shared_ptr algo = drain::Volume::create(); //algo->setInputFormat(m_node->getInterfaceFormat()); algo->setName("volume"); m_process.pushBack(algo); // note : feedback has no volume stage ... m_process.setOutputConfig(drain::IOFormatInterface(_map, _format, _freq)); } else { RIVER_ERROR("Can not link virtual interface with type : " << m_mode << " to a hardware interface " << (m_node->isInput()==true?"input":"output")); return false; } return true; } std::shared_ptr river::Interface::create(const std::string& _name, float _freq, const std::vector& _map, audio::format _format, const std::shared_ptr& _node, const std::shared_ptr& _config) { std::shared_ptr out = std::shared_ptr(new river::Interface()); out->init(_name, _freq, _map, _format, _node, _config); return out; } river::Interface::~Interface() { //stop(true, true); std::unique_lock lock(m_mutex); //m_node->interfaceRemove(shared_from_this()); } /* bool river::Interface::hasEndPoint() { } */ void river::Interface::setReadwrite() { std::unique_lock lock(m_mutex); m_process.removeAlgoDynamic(); if (m_process.hasType() ) { RIVER_ERROR("Endpoint is already present ==> can not change"); return; } if (m_node->isInput() == true) { m_process.removeIfLast(); std::shared_ptr algo = drain::EndPointRead::create(); m_process.pushBack(algo); } else { m_process.removeIfFirst(); std::shared_ptr algo = drain::EndPointWrite::create(); m_process.pushFront(algo); } } void river::Interface::setOutputCallback(drain::playbackFunction _function) { std::unique_lock lock(m_mutex); m_process.removeAlgoDynamic(); m_process.removeIfFirst(); std::shared_ptr algo = drain::EndPointCallback::create(_function); m_process.pushFront(algo); } void river::Interface::setInputCallback(drain::recordFunction _function) { std::unique_lock lock(m_mutex); m_process.removeAlgoDynamic(); m_process.removeIfLast(); std::shared_ptr algo = drain::EndPointCallback::create(_function); m_process.pushBack(algo); } void river::Interface::setWriteCallback(drain::playbackFunctionWrite _function) { std::unique_lock lock(m_mutex); m_process.removeAlgoDynamic(); std::shared_ptr algo = m_process.get(0); if (algo == nullptr) { return; } algo->setCallback(_function); } void river::Interface::start(const std::chrono::system_clock::time_point& _time) { std::unique_lock lock(m_mutex); RIVER_DEBUG("start [BEGIN]"); m_process.updateInterAlgo(); m_node->interfaceAdd(shared_from_this()); RIVER_DEBUG("start [ END ]"); } void river::Interface::stop(bool _fast, bool _abort) { std::unique_lock lock(m_mutex); RIVER_DEBUG("stop [BEGIN]"); m_node->interfaceRemove(shared_from_this()); RIVER_DEBUG("stop [ END]"); } void river::Interface::abort() { std::unique_lock lock(m_mutex); RIVER_DEBUG("abort [BEGIN]"); // TODO :... RIVER_DEBUG("abort [ END ]"); } bool river::Interface::setParameter(const std::string& _filter, const std::string& _parameter, const std::string& _value) { RIVER_DEBUG("setParameter [BEGIN] : '" << _filter << "':'" << _parameter << "':'" << _value << "'"); bool out = false; if ( _filter == "volume" && _parameter != "FLOW") { RIVER_ERROR("Interface is not allowed to modify '" << _parameter << "' Volume just allowed to modify 'FLOW' volume"); return false; } std::shared_ptr algo = m_process.get(_filter); if (algo == nullptr) { RIVER_ERROR("setParameter(" << _filter << ") ==> no filter named like this ..."); return false; } out = algo->setParameter(_parameter, _value); RIVER_DEBUG("setParameter [ END ] : '" << out << "'"); return out; } std::string river::Interface::getParameter(const std::string& _filter, const std::string& _parameter) const { RIVER_DEBUG("getParameter [BEGIN] : '" << _filter << "':'" << _parameter << "'"); std::string out; std::shared_ptr algo = m_process.get(_filter); if (algo == nullptr) { RIVER_ERROR("setParameter(" << _filter << ") ==> no filter named like this ..."); return "[ERROR]"; } out = algo->getParameter(_parameter); RIVER_DEBUG("getParameter [ END ] : '" << out << "'"); return out; } std::string river::Interface::getParameterProperty(const std::string& _filter, const std::string& _parameter) const { RIVER_DEBUG("getParameterProperty [BEGIN] : '" << _filter << "':'" << _parameter << "'"); std::string out; std::shared_ptr algo = m_process.get(_filter); if (algo == nullptr) { RIVER_ERROR("setParameter(" << _filter << ") ==> no filter named like this ..."); return "[ERROR]"; } out = algo->getParameterProperty(_parameter); RIVER_DEBUG("getParameterProperty [ END ] : '" << out << "'"); return out; } void river::Interface::write(const void* _value, size_t _nbChunk) { std::unique_lock lock(m_mutex); m_process.updateInterAlgo(); std::shared_ptr algo = m_process.get(0); if (algo == nullptr) { return; } algo->write(_value, _nbChunk); } #if 0 // TODO : add API aCCess mutex for Read and write... std::vector river::Interface::read(size_t _nbChunk) { // TODO :... std::vector data; /* data.resize(_nbChunk*m_map.size(), 0); m_mutex.lock(); int32_t nbChunkBuffer = m_circularBuffer.size() / m_map.size(); m_mutex.unlock(); while (nbChunkBuffer < _nbChunk) { usleep(1000); nbChunkBuffer = m_circularBuffer.size() / m_map.size(); } m_mutex.lock(); for (size_t iii = 0; iii lock(m_mutex); m_process.updateInterAlgo(); // TODO :... } size_t river::Interface::size() const { std::unique_lock lock(m_mutex); // TODO :... return 0; } void river::Interface::setBufferSize(size_t _nbChunk) { std::unique_lock lock(m_mutex); m_process.updateInterAlgo(); // TODO :... } void river::Interface::setBufferSize(const std::chrono::duration& _time) { std::unique_lock lock(m_mutex); m_process.updateInterAlgo(); // TODO :... } void river::Interface::clearInternalBuffer() { std::unique_lock lock(m_mutex); m_process.updateInterAlgo(); // TODO :... } std::chrono::system_clock::time_point river::Interface::getCurrentTime() const { std::unique_lock lock(m_mutex); // TODO :... return std::chrono::system_clock::time_point(); return std::chrono::system_clock::now(); } void river::Interface::addVolumeGroup(const std::string& _name) { std::unique_lock lock(m_mutex); RIVER_DEBUG("addVolumeGroup(" << _name << ")"); std::shared_ptr algo = m_process.get("volume"); if (algo == nullptr) { RIVER_ERROR("addVolumeGroup(" << _name << ") ==> no volume stage ... can not add it ..."); return; } if (_name == "FLOW") { // Local volume name algo->addVolumeStage(std::make_shared(_name)); } else { // get manager unique instance: std::shared_ptr mng = river::io::Manager::getInstance(); algo->addVolumeStage(mng->getVolumeGroup(_name)); } } void river::Interface::systemNewInputData(std::chrono::system_clock::time_point _time, const void* _data, size_t _nbChunk) { std::unique_lock lockProcess(m_mutex); void * tmpData = const_cast(_data); m_process.push(_time, tmpData, _nbChunk); } void river::Interface::systemNeedOutputData(std::chrono::system_clock::time_point _time, void* _data, size_t _nbChunk, size_t _chunkSize) { std::unique_lock lockProcess(m_mutex); m_process.pull(_time, _data, _nbChunk, _chunkSize); } void river::Interface::systemVolumeChange() { std::unique_lock lockProcess(m_mutex); std::shared_ptr algo = m_process.get("volume"); if (algo == nullptr) { return; } algo->volumeChange(); }