audio-drain/audio/drain/Volume.cpp

452 lines
14 KiB
C++
Raw Normal View History

2015-01-30 21:36:11 +01:00
/** @file
* @author Edouard DUPIN
* @copyright 2011, Edouard DUPIN, all right reserved
* @license MPL v2.0 (see license file)
2015-01-30 21:36:11 +01:00
*/
2016-10-02 21:41:55 +02:00
#include <audio/drain/Volume.hpp>
2017-09-26 15:57:44 +02:00
extern "C" {
#include <math.h>
}
2016-10-02 21:41:55 +02:00
#include <audio/drain/debug.hpp>
2015-01-30 21:36:11 +01:00
2015-04-10 23:00:13 +02:00
audio::drain::Volume::Volume() :
2015-02-02 21:48:57 +01:00
m_volumeAppli(1.0f),
2018-06-19 22:13:48 +02:00
m_functionConvert(null) {
2015-01-30 21:36:11 +01:00
}
2015-04-10 23:00:13 +02:00
void audio::drain::Volume::init() {
audio::drain::Algo::init();
2015-02-01 22:22:42 +01:00
m_type = "Volume";
2017-08-28 00:08:17 +02:00
m_supportedFormat.pushBack(audio::format_int16);
m_supportedFormat.pushBack(audio::format_int16_on_int32);
2015-01-30 21:36:11 +01:00
}
2016-07-19 21:43:58 +02:00
ememory::SharedPtr<audio::drain::Volume> audio::drain::Volume::create() {
ememory::SharedPtr<audio::drain::Volume> tmp(ETK_NEW(audio::drain::Volume));
2015-01-30 21:36:11 +01:00
tmp->init();
return tmp;
}
2015-04-10 23:00:13 +02:00
audio::drain::Volume::~Volume() {
2015-01-30 21:36:11 +01:00
}
2015-02-03 21:29:23 +01:00
static int32_t neareastsss(float _val) {
int32_t out = 0;
while (_val > float(1<<out)) {
out++;
}
2017-08-28 00:08:17 +02:00
return etk::min(16,out);
2015-02-03 21:29:23 +01:00
}
2015-02-02 21:48:57 +01:00
static void convert__int16__to__int16(void* _input, void* _output, size_t _nbSample, int32_t _volumeCoef, int32_t _volumeDecalage, float _volumeAppli) {
int16_t* in = static_cast<int16_t*>(_input);
int16_t* out = static_cast<int16_t*>(_output);
for (size_t iii=0; iii<_nbSample; ++iii) {
out[iii] = int16_t((int32_t(in[iii]) * int32_t(_volumeCoef)) >> _volumeDecalage);
}
}
static void convert__int16__to__float(void* _input, void* _output, size_t _nbSample, int32_t _volumeCoef, int32_t _volumeDecalage, float _volumeAppli) {
int16_t* in = static_cast<int16_t*>(_input);
float* out = static_cast<float*>(_output);
for (size_t iii=0; iii<_nbSample; ++iii) {
out[iii] = double((int32_t(in[iii]) * int32_t(_volumeCoef)) >> _volumeDecalage) * 0.00003051757;
}
}
2015-02-02 21:48:57 +01:00
static void convert__int16__to__int32(void* _input, void* _output, size_t _nbSample, int32_t _volumeCoef, int32_t _volumeDecalage, float _volumeAppli) {
int16_t* in = static_cast<int16_t*>(_input);
int32_t* out = static_cast<int32_t*>(_output);
for (size_t iii=0; iii<_nbSample; ++iii) {
out[iii] = (int32_t(in[iii]) * int32_t(_volumeCoef)) >> _volumeDecalage;
}
//DRAIN_INFO("plop " << in[0] << " >> " << out[0]);
2015-02-02 21:48:57 +01:00
}
static void convert__int32__to__int16(void* _input, void* _output, size_t _nbSample, int32_t _volumeCoef, int32_t _volumeDecalage, float _volumeAppli) {
int32_t* in = static_cast<int32_t*>(_input);
int16_t* out = static_cast<int16_t*>(_output);
for (size_t iii=0; iii<_nbSample; ++iii) {
int32_t value = in[iii] >> 16;
out[iii] = int16_t((int64_t(in[iii]) * int64_t(_volumeCoef)) >> _volumeDecalage);
}
}
static void convert__int32__to__int32(void* _input, void* _output, size_t _nbSample, int32_t _volumeCoef, int32_t _volumeDecalage, float _volumeAppli) {
int32_t* in = static_cast<int32_t*>(_input);
int32_t* out = static_cast<int32_t*>(_output);
for (size_t iii=0; iii<_nbSample; ++iii) {
out[iii] = int32_t((int64_t(in[iii]) * int64_t(_volumeCoef)) >> _volumeDecalage);
}
//DRAIN_INFO("plop " << in[0] << " >> " << out[0]);
2015-02-02 21:48:57 +01:00
}
static void convert__float__to__float(void* _input, void* _output, size_t _nbSample, int32_t _volumeCoef, int32_t _volumeDecalage, float _volumeAppli) {
float* in = static_cast<float*>(_input);
float* out = static_cast<float*>(_output);
for (size_t iii=0; iii<_nbSample; ++iii) {
out[iii] = in[iii] * _volumeAppli;
}
}
2015-04-10 23:00:13 +02:00
void audio::drain::Volume::configurationChange() {
audio::drain::Algo::configurationChange();
2015-02-02 21:48:57 +01:00
switch (m_input.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
switch (m_output.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
m_functionConvert = &convert__int16__to__int16;
DRAIN_DEBUG("Use converter : 'convert__int16__to__int16' for " << m_input.getFormat() << " to " << m_output.getFormat());
2015-02-02 21:48:57 +01:00
break;
case audio::format_int16_on_int32:
case audio::format_int32:
2015-02-02 21:48:57 +01:00
m_functionConvert = &convert__int16__to__int32;
DRAIN_DEBUG("Use converter : 'convert__int16__to__int32' for " << m_input.getFormat() << " to " << m_output.getFormat());
2015-02-02 21:48:57 +01:00
break;
case audio::format_float:
m_functionConvert = &convert__int16__to__float;
2015-02-02 21:48:57 +01:00
break;
}
break;
case audio::format_int16_on_int32:
case audio::format_int32:
2015-02-02 21:48:57 +01:00
switch (m_output.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
m_functionConvert = &convert__int32__to__int16;
DRAIN_DEBUG("Use converter : 'convert__int32__to__int16' for " << m_input.getFormat() << " to " << m_output.getFormat());
2015-02-02 21:48:57 +01:00
break;
case audio::format_int16_on_int32:
case audio::format_int32:
2015-02-02 21:48:57 +01:00
m_functionConvert = &convert__int32__to__int32;
DRAIN_DEBUG("Use converter : 'convert__int32__to__int32' for " << m_input.getFormat() << " to " << m_output.getFormat());
2015-02-02 21:48:57 +01:00
break;
case audio::format_float:
DRAIN_ERROR("Impossible case 2");
2015-02-02 21:48:57 +01:00
break;
}
break;
case audio::format_float:
2015-02-02 21:48:57 +01:00
switch (m_output.getFormat()) {
default:
case audio::format_int16:
case audio::format_int16_on_int32:
case audio::format_int32:
DRAIN_ERROR("Impossible case 4");
2015-02-02 21:48:57 +01:00
break;
case audio::format_float:
2015-02-02 21:48:57 +01:00
m_functionConvert = &convert__float__to__float;
DRAIN_DEBUG("Use converter : 'convert__float__to__float' for " << m_input.getFormat() << " to " << m_output.getFormat());
2015-02-02 21:48:57 +01:00
break;
}
break;
2015-01-30 21:36:11 +01:00
}
if (m_input.getMap() != m_output.getMap()) {
DRAIN_ERROR("Volume map change is not supported");
2015-01-30 21:36:11 +01:00
}
if (m_input.getFrequency() != m_output.getFrequency()) {
DRAIN_ERROR("Volume frequency change is not supported");
2015-01-30 21:36:11 +01:00
}
// nee to process all time (the format not change (just a simple filter))
2015-02-02 21:48:57 +01:00
m_needProcess = true;
2015-02-04 21:08:06 +01:00
volumeChange();
2015-02-02 21:48:57 +01:00
}
2015-04-10 23:00:13 +02:00
void audio::drain::Volume::volumeChange() {
2015-02-02 21:48:57 +01:00
//m_volumeAppli = 20 * log(m_volumedB);
2015-02-03 21:29:23 +01:00
float volumedB = 0.0f;
2015-07-02 21:17:24 +02:00
bool mute = false;
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_volumeList.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_volumeList[iii] == null) {
2015-02-03 21:29:23 +01:00
continue;
}
2015-07-02 21:17:24 +02:00
if (m_volumeList[iii]->getMute() == true) {
mute = true;
}
2015-02-24 22:20:11 +01:00
volumedB += m_volumeList[iii]->getVolume();
DRAIN_VERBOSE("append volume : '" << m_volumeList[iii]->getName() << " vol=" << m_volumeList[iii]->getVolume() << "dB");
2015-02-03 21:29:23 +01:00
}
DRAIN_DEBUG(" Total volume : " << volumedB << "dB nbVolume=" << m_volumeList.size());
2015-07-02 21:17:24 +02:00
if (mute == true) {
m_volumeAppli = 0.0f;
m_volumeCoef = 0;
m_volumeDecalage = 0;
return;
}
#if (defined(__STDCPP_LLVM__) || __cplusplus < 201103L)
2015-02-12 21:24:08 +01:00
m_volumeAppli = pow(10.0f, volumedB/20.0f);
#else
2017-09-26 15:57:44 +02:00
m_volumeAppli = etk::pow(10.0f, volumedB/20.0f);
2015-02-12 21:24:08 +01:00
#endif
2015-02-02 21:48:57 +01:00
switch (m_input.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
switch (m_output.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 16;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 16-neareast;
}
break;
case audio::format_int16_on_int32:
2015-02-02 21:48:57 +01:00
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 16;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 16-neareast;
}
break;
case audio::format_int32:
2015-02-02 21:48:57 +01:00
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 0;
break;
case audio::format_float:
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 16;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 16-neareast;
}
2015-02-02 21:48:57 +01:00
break;
}
break;
case audio::format_int16_on_int32:
2015-02-02 21:48:57 +01:00
switch (m_output.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 16;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 16-neareast;
}
break;
case audio::format_int16_on_int32:
2015-02-02 21:48:57 +01:00
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 16;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 16-neareast;
}
break;
case audio::format_int32:
2015-02-02 21:48:57 +01:00
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 0;
break;
case audio::format_float:
DRAIN_ERROR("Impossible case 2");
2015-02-02 21:48:57 +01:00
break;
}
break;
case audio::format_int32:
2015-02-02 21:48:57 +01:00
switch (m_output.getFormat()) {
default:
case audio::format_int16:
2015-02-02 21:48:57 +01:00
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 32;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 32-neareast;
}
break;
case audio::format_int16_on_int32:
2015-02-02 21:48:57 +01:00
if (m_volumeAppli <= 1.0f) {
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 32;
} else {
int32_t neareast = neareastsss(m_volumeAppli);
m_volumeCoef = m_volumeAppli*float(1<<(16-neareast));
m_volumeDecalage = 32-neareast;
}
break;
case audio::format_int32:
2015-02-02 21:48:57 +01:00
m_volumeCoef = m_volumeAppli*float(1<<16);
m_volumeDecalage = 16;
break;
case audio::format_float:
DRAIN_ERROR("Impossible case 3");
2015-02-02 21:48:57 +01:00
break;
}
break;
case audio::format_float:
2015-02-02 21:48:57 +01:00
// nothing to do (use m_volumeAppli)
break;
}
2015-01-30 21:36:11 +01:00
}
2015-02-02 21:48:57 +01:00
2017-08-28 00:08:17 +02:00
etk::Vector<audio::format> audio::drain::Volume::getFormatSupportedInput() {
etk::Vector<audio::format> tmp;
if (m_output.getFormat() == audio::format_float) {
2017-08-28 00:08:17 +02:00
tmp.pushBack(audio::format_float);
2015-02-02 21:48:57 +01:00
}
if ( m_output.getFormat() == audio::format_int16
|| m_output.getFormat() == audio::format_int16_on_int32
|| m_output.getFormat() == audio::format_int32) {
2017-08-28 00:08:17 +02:00
tmp.pushBack(audio::format_int16);
tmp.pushBack(audio::format_int16_on_int32);
tmp.pushBack(audio::format_int32);
2015-02-02 21:48:57 +01:00
}
return tmp;
2015-02-01 22:22:42 +01:00
};
2017-08-28 00:08:17 +02:00
etk::Vector<audio::format> audio::drain::Volume::getFormatSupportedOutput() {
etk::Vector<audio::format> tmp;
if (m_input.getFormat() == audio::format_float) {
2017-08-28 00:08:17 +02:00
tmp.pushBack(audio::format_float);
2015-02-02 21:48:57 +01:00
}
if ( m_input.getFormat() == audio::format_int16
|| m_input.getFormat() == audio::format_int16_on_int32
|| m_input.getFormat() == audio::format_int32) {
2017-08-28 00:08:17 +02:00
tmp.pushBack(audio::format_int16);
tmp.pushBack(audio::format_int16_on_int32);
tmp.pushBack(audio::format_int32);
2015-02-02 21:48:57 +01:00
}
return tmp;
2015-02-01 22:22:42 +01:00
};
2015-04-13 21:49:48 +02:00
bool audio::drain::Volume::process(audio::Time& _time,
2015-04-10 23:00:13 +02:00
void* _input,
size_t _inputNbChunk,
void*& _output,
size_t& _outputNbChunk) {
audio::drain::AutoLogInOut tmpLog("Volume");
2015-02-02 21:48:57 +01:00
// chack if we need to process:
if (m_needProcess == false) {
2015-01-30 21:36:11 +01:00
_output = _input;
_outputNbChunk = _inputNbChunk;
return true;
}
2018-06-19 22:13:48 +02:00
if (_input == null) {
2015-02-02 21:48:57 +01:00
_output = &(m_outputData[0]);
_outputNbChunk = 0;
DRAIN_ERROR("null pointer input ... ");
2015-02-02 21:48:57 +01:00
return false;
}
_outputNbChunk = _inputNbChunk;
m_outputData.resize(_outputNbChunk*m_input.getMap().size()*m_formatSize);
_output = &(m_outputData[0]);
2018-06-19 22:13:48 +02:00
if (m_functionConvert == null) {
DRAIN_ERROR("null function ptr");
2015-02-02 21:48:57 +01:00
return false;
}
//DRAIN_WARNING("Apply volume : " << m_volumedB << "dB " << m_volumeAppli << " ==> x*" << m_volumeCoef << ">>" << m_volumeDecalage << " ex:50*C>>D=" << (50*m_volumeCoef>>m_volumeDecalage) );
2015-02-02 21:48:57 +01:00
m_functionConvert(_input, _output, _outputNbChunk*m_input.getMap().size(), m_volumeCoef, m_volumeDecalage, m_volumeAppli);
2015-02-01 22:22:42 +01:00
return true;
2015-01-30 21:36:11 +01:00
}
2015-02-03 21:29:23 +01:00
2016-07-19 21:43:58 +02:00
void audio::drain::Volume::addVolumeStage(const ememory::SharedPtr<audio::drain::VolumeElement>& _volume) {
2018-06-19 22:13:48 +02:00
if (_volume == null) {
2015-02-03 21:29:23 +01:00
return;
}
2015-02-24 22:20:11 +01:00
for (size_t iii=0; iii<m_volumeList.size(); ++iii) {
2018-06-19 22:13:48 +02:00
if (m_volumeList[iii] == null) {
2015-02-03 21:29:23 +01:00
continue;
}
2015-02-24 22:20:11 +01:00
if (m_volumeList[iii] == _volume) {
// already done ...
2015-02-03 21:29:23 +01:00
return;
}
2015-02-24 22:20:11 +01:00
if (m_volumeList[iii]->getName() == _volume->getName()) {
2015-02-03 21:29:23 +01:00
return;
}
}
2017-08-28 00:08:17 +02:00
m_volumeList.pushBack(_volume);
2015-02-04 21:08:06 +01:00
volumeChange();
2015-02-03 21:29:23 +01:00
}
2017-08-28 00:08:17 +02:00
bool audio::drain::Volume::setParameter(const etk::String& _parameter, const etk::String& _value) {
2015-02-03 21:29:23 +01:00
if (_parameter == "FLOW") {
// set Volume ...
for (auto &it : m_volumeList) {
2018-06-19 22:13:48 +02:00
if (it == null) {
2015-02-03 21:29:23 +01:00
continue;
}
if (it->getName() == "FLOW") {
2015-02-03 21:29:23 +01:00
float value = 0;
2015-02-03 23:29:53 +01:00
if (sscanf(_value.c_str(), "%fdB", &value) != 1) {
return false;
}
if ( value < -300
|| value > 300) {
DRAIN_ERROR("Can not set volume ... : '" << _parameter << "' out of range : [-300..300]");
return false;
}
it->setVolume(value);
DRAIN_DEBUG("Set volume : FLOW = " << value << " dB (from:" << _value << ")");
2015-02-04 21:08:06 +01:00
volumeChange();
2015-02-03 21:29:23 +01:00
return true;
}
}
}
DRAIN_ERROR("unknow set Parameter : '" << _parameter << "' with Value: '" << _value << "'");
2015-02-03 21:29:23 +01:00
return false;
}
2017-08-28 00:08:17 +02:00
etk::String audio::drain::Volume::getParameter(const etk::String& _parameter) const {
2015-02-03 21:29:23 +01:00
if (_parameter == "FLOW") {
// set Volume ...
for (auto &it : m_volumeList) {
2018-06-19 22:13:48 +02:00
if (it == null) {
2015-02-03 21:29:23 +01:00
continue;
}
if (it->getName() == "FLOW") {
2017-08-28 00:08:17 +02:00
return etk::toString(it->getVolume()) + "dB";
2015-02-03 21:29:23 +01:00
}
}
}
DRAIN_ERROR("unknow get Parameter : '" << _parameter << "'");
2015-02-03 21:29:23 +01:00
return "[ERROR]";
}
2017-08-28 00:08:17 +02:00
etk::String audio::drain::Volume::getParameterProperty(const etk::String& _parameter) const {
2015-02-03 21:29:23 +01:00
if (_parameter == "FLOW") {
// set Volume ...
for (auto &it : m_volumeList) {
2018-06-19 22:13:48 +02:00
if (it == null) {
2015-02-03 21:29:23 +01:00
continue;
}
if (it->getName() == "FLOW") {
2015-02-03 21:29:23 +01:00
return "[-300..300]dB";
}
}
}
DRAIN_ERROR("unknow Parameter property for: '" << _parameter << "'");
2015-02-03 21:29:23 +01:00
return "[ERROR]";
2015-02-04 21:08:06 +01:00
}
2017-08-28 00:08:17 +02:00
etk::String audio::drain::Volume::getDotDesc() {
etk::String out = audio::drain::Algo::getDotDesc();
for (auto &it : m_volumeList) {
2018-06-19 22:13:48 +02:00
if (it == null) {
continue;
}
2017-08-28 00:08:17 +02:00
out += "\\n" + it->getName() + "=" + etk::toString(it->getVolume()) + "dB";
if (it->getMute() == true) {
out += " MUTE";
}
}
return out;
}