audio-river/audio/river/io/Manager.cpp

420 lines
13 KiB
C++
Raw Normal View History

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-04-11 09:38:30 +02:00
#include <audio/river/io/Manager.h>
#include <audio/river/debug.h>
#include <audio/river/river.h>
#include <audio/river/io/Node.h>
#include <audio/river/io/NodeAEC.h>
#include <audio/river/io/NodeMuxer.h>
#include <audio/river/io/NodeOrchestra.h>
#include <audio/river/io/NodePortAudio.h>
2015-02-18 15:22:48 +01:00
#include <etk/os/FSNode.h>
2015-09-14 21:11:04 +02:00
#include <memory>
#include <etk/types.h>
#include <utility>
2015-01-25 22:17:06 +01:00
2015-01-27 21:01:52 +01:00
#undef __class__
#define __class__ "io::Manager"
2015-04-11 17:43:09 +02:00
#ifdef AUDIO_RIVER_BUILD_PORTAUDIO
2015-10-14 21:21:03 +02:00
#include <portaudio/portaudio.h>
2015-02-19 22:00:21 +01:00
#endif
2015-02-13 21:06:55 +01:00
static std::string basicAutoConfig =
"{\n"
" microphone:{\n"
" io:'input',\n"
" map-on:{\n"
" interface:'auto',\n"
" name:'default',\n"
" },\n"
" frequency:0,\n"
" channel-map:[\n"
" 'front-left', 'front-right'\n"
" ],\n"
" type:'auto',\n"
" nb-chunk:1024\n"
" },\n"
" speaker:{\n"
" io:'output',\n"
" map-on:{\n"
" interface:'auto',\n"
" name:'default',\n"
" },\n"
" frequency:0,\n"
" channel-map:[\n"
" 'front-left', 'front-right',\n"
" ],\n"
" type:'auto',\n"
" nb-chunk:1024,\n"
" volume-name:'MASTER'\n"
" }\n"
"}\n";
2015-04-11 09:38:30 +02:00
audio::river::io::Manager::Manager() {
2015-04-11 17:43:09 +02:00
#ifdef AUDIO_RIVER_BUILD_PORTAUDIO
2015-02-19 22:00:21 +01:00
PaError err = Pa_Initialize();
if(err != paNoError) {
RIVER_WARNING("Can not initialize portaudio : " << Pa_GetErrorText(err));
}
#endif
2015-03-12 22:28:15 +01:00
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Manager::init(const std::string& _filename) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 22:28:15 +01:00
if (_filename == "") {
RIVER_INFO("Load default config");
m_config.parse(basicAutoConfig);
} else if (m_config.load(_filename) == false) {
RIVER_ERROR("you must set a basic configuration file for harware configuration: '" << _filename << "'");
}
}
2015-03-12 23:17:14 +01:00
2015-04-11 09:38:30 +02:00
void audio::river::io::Manager::initString(const std::string& _data) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 23:17:14 +01:00
m_config.parse(_data);
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Manager::unInit() {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 23:17:14 +01:00
// TODO : ...
2015-03-12 22:28:15 +01:00
}
2015-02-19 22:00:21 +01:00
2015-04-11 09:38:30 +02:00
audio::river::io::Manager::~Manager() {
2015-04-11 17:43:09 +02:00
#ifdef AUDIO_RIVER_BUILD_PORTAUDIO
2015-02-19 22:00:21 +01:00
PaError err = Pa_Terminate();
if(err != paNoError) {
RIVER_WARNING("Can not initialize portaudio : " << Pa_GetErrorText(err));
}
#endif
};
std::shared_ptr<audio::river::io::Manager> audio::river::io::Manager::getInstance() {
2015-04-11 09:38:30 +02:00
if (audio::river::isInit() == false) {
return std::shared_ptr<audio::river::io::Manager>();
2015-03-12 22:28:15 +01:00
}
static std::shared_ptr<audio::river::io::Manager> manager(new Manager());
2015-01-25 22:17:06 +01:00
return manager;
}
2015-03-12 22:28:15 +01:00
2015-04-11 09:38:30 +02:00
std::vector<std::string> audio::river::io::Manager::getListStreamInput() {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 22:28:15 +01:00
std::vector<std::string> output;
std::vector<std::string> keys = m_config.getKeys();
2016-04-20 21:19:11 +02:00
for (auto &it : keys) {
const ejson::Object tmppp = m_config[it].toObject();
if (tmppp.exist() == true) {
std::string type = tmppp.getStringValue("io", "error");
2015-03-12 22:28:15 +01:00
if ( type == "input"
|| type == "PAinput") {
2016-04-20 21:19:11 +02:00
output.push_back(it);
2015-03-12 22:28:15 +01:00
}
}
}
return output;
}
2015-04-11 09:38:30 +02:00
std::vector<std::string> audio::river::io::Manager::getListStreamOutput() {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 22:28:15 +01:00
std::vector<std::string> output;
std::vector<std::string> keys = m_config.getKeys();
2016-04-20 21:19:11 +02:00
for (auto &it : keys) {
const ejson::Object tmppp = m_config[it].toObject();
if (tmppp.exist() == true) {
std::string type = tmppp.getStringValue("io", "error");
2015-03-12 22:28:15 +01:00
if ( type == "output"
|| type == "PAoutput") {
2016-04-20 21:19:11 +02:00
output.push_back(it);
2015-03-12 22:28:15 +01:00
}
}
}
return output;
}
2015-04-11 09:38:30 +02:00
std::vector<std::string> audio::river::io::Manager::getListStreamVirtual() {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 22:28:15 +01:00
std::vector<std::string> output;
std::vector<std::string> keys = m_config.getKeys();
2016-04-20 21:19:11 +02:00
for (auto &it : keys) {
const ejson::Object tmppp = m_config[it].toObject();
if (tmppp.exist() == true) {
std::string type = tmppp.getStringValue("io", "error");
2015-03-12 22:28:15 +01:00
if ( type != "input"
&& type != "PAinput"
&& type != "output"
&& type != "PAoutput"
&& type != "error") {
2016-04-20 21:19:11 +02:00
output.push_back(it);
2015-03-12 22:28:15 +01:00
}
}
}
return output;
}
2015-04-11 09:38:30 +02:00
std::vector<std::string> audio::river::io::Manager::getListStream() {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-03-12 22:28:15 +01:00
std::vector<std::string> output;
std::vector<std::string> keys = m_config.getKeys();
2016-04-20 21:19:11 +02:00
for (auto &it : keys) {
const ejson::Object tmppp = m_config[it].toObject();
if (tmppp.exist() == true) {
std::string type = tmppp.getStringValue("io", "error");
2015-03-12 22:28:15 +01:00
if (type != "error") {
2016-04-20 21:19:11 +02:00
output.push_back(it);
2015-03-12 22:28:15 +01:00
}
}
}
return output;
}
std::shared_ptr<audio::river::io::Node> audio::river::io::Manager::getNode(const std::string& _name) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-02-19 22:00:21 +01:00
RIVER_WARNING("Get node : " << _name);
// search in the standalone list :
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_list.size(); ++iii) {
std::shared_ptr<audio::river::io::Node> tmppp = m_list[iii].lock();
if ( tmppp != nullptr
&& _name == tmppp->getName()) {
RIVER_WARNING(" find it ... in standalone");
2015-01-25 22:17:06 +01:00
return tmppp;
}
}
// search in the group list:
{
for (std::map<std::string, std::shared_ptr<audio::river::io::Group> >::iterator it(m_listGroup.begin());
it != m_listGroup.end();
++it) {
if (it->second != nullptr) {
std::shared_ptr<audio::river::io::Node> node = it->second->getNode(_name);
if (node != nullptr) {
RIVER_WARNING(" find it ... in group: " << it->first);
return node;
}
}
}
}
2015-04-11 17:43:09 +02:00
RIVER_WARNING("Try create a new one : " << _name);
// check if the node can be open :
2016-04-20 21:19:11 +02:00
const ejson::Object tmpObject = m_config[_name].toObject();
if (tmpObject.exist() == true) {
//Check if it is in a group:
2016-04-20 21:19:11 +02:00
std::string groupName = tmpObject.getStringValue("group", "");
// get type : io
2016-04-20 21:19:11 +02:00
std::string ioType = tmpObject.getStringValue("io", "error");
if ( groupName != ""
&& ( ioType == "input"
|| ioType == "output"
|| ioType == "PAinput"
|| ioType == "PAoutput") ) {
std::shared_ptr<audio::river::io::Group> tmpGroup = getGroup(groupName);
if (tmpGroup == nullptr) {
RIVER_WARNING("Can not get group ... '" << groupName << "'");
return std::shared_ptr<audio::river::io::Node>();
}
return tmpGroup->getNode(_name);
} else {
if (groupName != "") {
RIVER_WARNING("Group is only availlable for Hardware interface ... '" << _name << "'");
}
// TODO : Create a standalone group for every single element ==> simplify understanding ... but not for virtual interface ...
2015-04-11 17:43:09 +02:00
if ( ioType == "input"
|| ioType == "output") {
2015-04-11 17:43:09 +02:00
#ifdef AUDIO_RIVER_BUILD_ORCHESTRA
std::shared_ptr<audio::river::io::Node> tmp = audio::river::io::NodeOrchestra::create(_name, tmpObject);
2015-04-11 17:43:09 +02:00
m_list.push_back(tmp);
return tmp;
#else
RIVER_WARNING("not present interface");
#endif
}
if ( ioType == "PAinput"
|| ioType == "PAoutput") {
2015-04-11 17:43:09 +02:00
#ifdef AUDIO_RIVER_BUILD_PORTAUDIO
std::shared_ptr<audio::river::io::Node> tmp = audio::river::io::NodePortAudio::create(_name, tmpObject);
2015-04-11 17:43:09 +02:00
m_list.push_back(tmp);
return tmp;
#else
RIVER_WARNING("not present interface");
#endif
}
if (ioType == "aec") {
std::shared_ptr<audio::river::io::Node> tmp = audio::river::io::NodeAEC::create(_name, tmpObject);
m_list.push_back(tmp);
return tmp;
}
2015-03-04 22:48:04 +01:00
if (ioType == "muxer") {
std::shared_ptr<audio::river::io::Node> tmp = audio::river::io::NodeMuxer::create(_name, tmpObject);
2015-03-04 22:48:04 +01:00
m_list.push_back(tmp);
return tmp;
}
}
}
RIVER_ERROR("Can not create the interface : '" << _name << "' the node is not DEFINED in the configuration file availlable : " << m_config.getKeys());
return std::shared_ptr<audio::river::io::Node>();
}
std::shared_ptr<audio::drain::VolumeElement> audio::river::io::Manager::getVolumeGroup(const std::string& _name) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
if (_name == "") {
RIVER_ERROR("Try to create an audio group with no name ...");
return std::shared_ptr<audio::drain::VolumeElement>();
}
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_volumeGroup.size(); ++iii) {
if (m_volumeGroup[iii] == nullptr) {
continue;
}
2015-02-24 22:20:11 +01:00
if (m_volumeGroup[iii]->getName() == _name) {
return m_volumeGroup[iii];
}
}
RIVER_DEBUG("Add a new volume group : '" << _name << "'");
std::shared_ptr<audio::drain::VolumeElement> tmpVolume = std::make_shared<audio::drain::VolumeElement>(_name);
m_volumeGroup.push_back(tmpVolume);
return tmpVolume;
2015-02-03 23:29:34 +01:00
}
2015-04-11 09:38:30 +02:00
bool audio::river::io::Manager::setVolume(const std::string& _volumeName, float _valuedB) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
std::shared_ptr<audio::drain::VolumeElement> volume = getVolumeGroup(_volumeName);
2015-02-04 21:08:06 +01:00
if (volume == nullptr) {
RIVER_ERROR("Can not set volume ... : '" << _volumeName << "'");
return false;
}
if ( _valuedB < -300
|| _valuedB > 300) {
RIVER_ERROR("Can not set volume ... : '" << _volumeName << "' out of range : [-300..300]");
2015-02-04 21:08:06 +01:00
return false;
}
volume->setVolume(_valuedB);
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_list.size(); ++iii) {
std::shared_ptr<audio::river::io::Node> val = m_list[iii].lock();
2015-02-04 21:08:06 +01:00
if (val != nullptr) {
val->volumeChange();
2015-02-03 23:29:34 +01:00
}
}
2015-02-04 21:08:06 +01:00
return true;
2015-02-03 23:29:34 +01:00
}
2015-04-11 09:38:30 +02:00
float audio::river::io::Manager::getVolume(const std::string& _volumeName) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
std::shared_ptr<audio::drain::VolumeElement> volume = getVolumeGroup(_volumeName);
2015-02-04 21:08:06 +01:00
if (volume == nullptr) {
RIVER_ERROR("Can not get volume ... : '" << _volumeName << "'");
2015-02-04 21:08:06 +01:00
return 0.0f;
2015-02-03 23:29:34 +01:00
}
2015-02-04 21:08:06 +01:00
return volume->getVolume();
2015-02-03 23:29:34 +01:00
}
2015-04-11 09:38:30 +02:00
std::pair<float,float> audio::river::io::Manager::getVolumeRange(const std::string& _volumeName) const {
2015-02-04 21:08:06 +01:00
return std::make_pair<float,float>(-300, 300);
2015-02-03 23:29:34 +01:00
}
2015-02-18 15:22:48 +01:00
2015-07-02 21:17:24 +02:00
void audio::river::io::Manager::setMute(const std::string& _volumeName, bool _mute) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
std::shared_ptr<audio::drain::VolumeElement> volume = getVolumeGroup(_volumeName);
2015-07-02 21:17:24 +02:00
if (volume == nullptr) {
RIVER_ERROR("Can not set volume ... : '" << _volumeName << "'");
return;
}
volume->setMute(_mute);
for (size_t iii=0; iii<m_list.size(); ++iii) {
std::shared_ptr<audio::river::io::Node> val = m_list[iii].lock();
2015-07-02 21:17:24 +02:00
if (val != nullptr) {
val->volumeChange();
}
}
}
bool audio::river::io::Manager::getMute(const std::string& _volumeName) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
std::shared_ptr<audio::drain::VolumeElement> volume = getVolumeGroup(_volumeName);
2015-07-02 21:17:24 +02:00
if (volume == nullptr) {
RIVER_ERROR("Can not get volume ... : '" << _volumeName << "'");
return false;
}
return volume->getMute();
}
2015-04-11 09:38:30 +02:00
void audio::river::io::Manager::generateDot(const std::string& _filename) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
2015-02-18 15:22:48 +01:00
etk::FSNode node(_filename);
RIVER_INFO("Generate the DOT files: " << node);
if (node.fileOpenWrite() == false) {
RIVER_ERROR("Can not Write the dot file (fail to open) : " << node);
return;
}
node << "digraph G {" << "\n";
2015-02-19 22:00:21 +01:00
node << " rankdir=\"LR\";\n";
2015-03-05 21:28:39 +01:00
// First Step : Create all HW interface:
{
// standalone
for (size_t iii=0; iii<m_list.size(); ++iii) {
std::shared_ptr<audio::river::io::Node> val = m_list[iii].lock();
2015-03-05 21:28:39 +01:00
if (val != nullptr) {
if (val->isHarwareNode() == true) {
val->generateDot(node);
}
}
}
for (std::map<std::string, std::shared_ptr<audio::river::io::Group> >::iterator it(m_listGroup.begin());
2015-03-05 21:28:39 +01:00
it != m_listGroup.end();
++it) {
if (it->second != nullptr) {
it->second->generateDot(node, true);
}
2015-02-18 15:22:48 +01:00
}
}
2015-03-05 21:28:39 +01:00
// All other ...
{
// standalone
for (size_t iii=0; iii<m_list.size(); ++iii) {
std::shared_ptr<audio::river::io::Node> val = m_list[iii].lock();
2015-03-05 21:28:39 +01:00
if (val != nullptr) {
if (val->isHarwareNode() == false) {
val->generateDot(node);
}
}
}
for (std::map<std::string, std::shared_ptr<audio::river::io::Group> >::iterator it(m_listGroup.begin());
2015-03-05 21:28:39 +01:00
it != m_listGroup.end();
++it) {
if (it->second != nullptr) {
it->second->generateDot(node, false);
}
2015-03-04 22:15:35 +01:00
}
}
2015-03-05 21:28:39 +01:00
2015-02-18 15:22:48 +01:00
node << "}" << "\n";
node.fileClose();
RIVER_INFO("Generate the DOT files: " << node << " (DONE)");
}
std::shared_ptr<audio::river::io::Group> audio::river::io::Manager::getGroup(const std::string& _name) {
std::unique_lock<std::recursive_mutex> lock(m_mutex);
std::shared_ptr<audio::river::io::Group> out;
std::map<std::string, std::shared_ptr<audio::river::io::Group> >::iterator it = m_listGroup.find(_name);
if (it == m_listGroup.end()) {
RIVER_INFO("Create a new group: " << _name << " (START)");
out = std::make_shared<audio::river::io::Group>();
if (out != nullptr) {
out->createFrom(m_config, _name);
std::pair<std::string, std::shared_ptr<audio::river::io::Group> > plop(std::string(_name), out);
m_listGroup.insert(plop);
RIVER_INFO("Create a new group: " << _name << " ( END )");
} else {
RIVER_ERROR("Can not create new group: " << _name << " ( END )");
}
} else {
out = it->second;
}
return out;
}