#pragma once #include #include #include #include namespace dsp { template class FrequencyXlator : public generic_block> { public: FrequencyXlator() {} FrequencyXlator(stream* in, float sampleRate, float freq) { init(in, sampleRate, freq); } ~FrequencyXlator() { generic_block>::stop(); } void init(stream* in, float sampleRate, float freq) { _in = in; _sampleRate = sampleRate; _freq = freq; phase = lv_cmake(1.0f, 0.0f); phaseDelta = lv_cmake(std::cos((_freq / _sampleRate) * 2.0f * FL_M_PI), std::sin((_freq / _sampleRate) * 2.0f * FL_M_PI)); generic_block>::registerInput(_in); generic_block>::registerOutput(&out); } void setInputSize(stream* in) { std::lock_guard lck(generic_block>::ctrlMtx); generic_block>::tempStop(); generic_block>::unregisterInput(_in); _in = in; generic_block>::registerInput(_in); generic_block>::tempStart(); } void setSampleRate(float sampleRate) { // No need to restart _sampleRate = sampleRate; phaseDelta = lv_cmake(std::cos((_freq / _sampleRate) * 2.0f * FL_M_PI), std::sin((_freq / _sampleRate) * 2.0f * FL_M_PI)); } float getSampleRate() { return _sampleRate; } void setFrequency(float freq) { // No need to restart _freq = freq; phaseDelta = lv_cmake(std::cos((_freq / _sampleRate) * 2.0f * FL_M_PI), std::sin((_freq / _sampleRate) * 2.0f * FL_M_PI)); } float getFrequency() { return _freq; } int run() { count = _in->read(); if (count < 0) { return -1; } if (out.aquire() < 0) { return -1; } // TODO: Do float xlation if constexpr (std::is_same_v) { spdlog::error("XLATOR NOT IMPLEMENTED FOR FLOAT"); } if constexpr (std::is_same_v) { volk_32fc_s32fc_x2_rotator_32fc((lv_32fc_t*)out.data, (lv_32fc_t*)_in->data, phaseDelta, &phase, count); } _in->flush(); out.write(count); return count; } stream out; private: int count; float _sampleRate; float _freq; lv_32fc_t phaseDelta; lv_32fc_t phase; stream* _in; }; class AGC : public generic_block { public: AGC() {} AGC(stream* in, float ratio) { init(in, ratio); } ~AGC() { generic_block::stop(); } void init(stream* in, float ratio) { _in = in; _ratio = ratio; generic_block::registerInput(_in); generic_block::registerOutput(&out); } void setInput(stream* in) { std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); generic_block::unregisterInput(_in); _in = in; generic_block::registerInput(_in); generic_block::tempStart(); } int run() { count = _in->read(); if (count < 0) { return -1; } if (out.aquire() < 0) { return -1; } for (int i = 0; i < count; i++) { level = (fabsf(_in->data[i]) * _ratio) + (level * (1.0f - _ratio)); out.data[i] = _in->data[i] / level; } _in->flush(); out.write(count); return count; } stream out; private: int count; float level = 1.0f; float _ratio; stream* _in; }; template class Volume : public generic_block> { public: Volume() {} Volume(stream* in, float volume) { init(in, volume); } ~Volume() { generic_block>::stop(); } void init(stream* in, float volume) { _in = in; _volume = volume; generic_block>::registerInput(_in); generic_block>::registerOutput(&out); } void setInputSize(stream* in) { std::lock_guard lck(generic_block>::ctrlMtx); generic_block>::tempStop(); generic_block>::unregisterInput(_in); _in = in; generic_block>::registerInput(_in); generic_block>::tempStart(); } void setVolume(float volume) { _volume = volume; level = powf(_volume, 2); } float getVolume() { return _volume; } void setMuted(bool muted) { _muted = muted; } bool getMuted() { return _muted; } int run() { count = _in->read(); if (count < 0) { return -1; } if (out.aquire() < 0) { return -1; } if (_muted) { if constexpr (std::is_same_v) { memset(out.data, 0, sizeof(stereo_t) * count); } else { memset(out.data, 0, sizeof(float) * count); } } else { if constexpr (std::is_same_v) { volk_32f_s32f_multiply_32f((float*)out.data, (float*)_in->data, level, count * 2); } else { volk_32f_s32f_multiply_32f((float*)out.data, (float*)_in->data, level, count); } } _in->flush(); out.write(count); return count; } stream out; private: int count; float level = 1.0f; float _volume = 1.0f; bool _muted = false; stream* _in; }; }