fix filter generation
This commit is contained in:
parent
507c5f96df
commit
281b18b6ca
@ -2,6 +2,7 @@
|
|||||||
#define FILTER_IFILTER_H
|
#define FILTER_IFILTER_H
|
||||||
|
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace Filter {
|
namespace Filter {
|
||||||
@ -14,6 +15,8 @@ class IFilter {
|
|||||||
IFilter(std::size_t size);
|
IFilter(std::size_t size);
|
||||||
virtual ~IFilter(){};
|
virtual ~IFilter(){};
|
||||||
|
|
||||||
|
void Dump(const std::string& path) const;
|
||||||
|
|
||||||
void ConvolveIQ(const std::vector<float>& input, std::vector<float>& output);
|
void ConvolveIQ(const std::vector<float>& input, std::vector<float>& output);
|
||||||
};
|
};
|
||||||
} // namespace Filter
|
} // namespace Filter
|
||||||
|
@ -10,7 +10,7 @@ namespace Filter {
|
|||||||
|
|
||||||
class LowPassFilter : public IFilter {
|
class LowPassFilter : public IFilter {
|
||||||
public:
|
public:
|
||||||
LowPassFilter(std::size_t size, std::size_t cutoffHz, std::size_t sampleRateHz, const std::vector<float>& window);
|
LowPassFilter(std::int32_t size, std::size_t cutoffHz, std::size_t sampleRateHz, const std::vector<float>& window);
|
||||||
~LowPassFilter() override{};
|
~LowPassFilter() override{};
|
||||||
};
|
};
|
||||||
} // namespace Filter
|
} // namespace Filter
|
||||||
|
@ -14,7 +14,9 @@ class HammingWindow : public IWindow {
|
|||||||
HammingWindow(std::size_t width);
|
HammingWindow(std::size_t width);
|
||||||
|
|
||||||
~HammingWindow() override {}
|
~HammingWindow() override {}
|
||||||
const std::vector<float>& getWindow() override { return window; }
|
|
||||||
|
void Dump(const std::string& path) const override;
|
||||||
|
const std::vector<float>& getWindow() const override { return window; }
|
||||||
};
|
};
|
||||||
} // namespace Filter
|
} // namespace Filter
|
||||||
|
|
||||||
|
@ -1,13 +1,16 @@
|
|||||||
#ifndef FILTER_IWINDOW_H
|
#ifndef FILTER_IWINDOW_H
|
||||||
#define FILTER_IWINDOW_H
|
#define FILTER_IWINDOW_H
|
||||||
|
|
||||||
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace Filter {
|
namespace Filter {
|
||||||
class IWindow {
|
class IWindow {
|
||||||
public:
|
public:
|
||||||
virtual ~IWindow() {}
|
virtual ~IWindow() {}
|
||||||
virtual const std::vector<float>& getWindow() = 0;
|
|
||||||
|
virtual void Dump(const std::string& path) const = 0;
|
||||||
|
virtual const std::vector<float>& getWindow() const = 0;
|
||||||
};
|
};
|
||||||
} // namespace Filter
|
} // namespace Filter
|
||||||
|
|
||||||
|
@ -1,31 +1,43 @@
|
|||||||
#include <cstddef>
|
#include <fstream>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include <filter/filter/filter.hpp>
|
#include <filter/filter/filter.hpp>
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
namespace Filter {
|
namespace Filter {
|
||||||
IFilter::IFilter(std::size_t size) {
|
IFilter::IFilter(std::size_t size) : impulseResponse(size) {}
|
||||||
impulseResponse.reserve(size);
|
|
||||||
|
void IFilter::Dump(const std::string& path) const {
|
||||||
|
std::ofstream output;
|
||||||
|
output.open(path, std::ios::out | std::ios::binary);
|
||||||
|
output.write(reinterpret_cast<const char*>(impulseResponse.data()), sizeof(float) * impulseResponse.size());
|
||||||
|
output.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
void IFilter::ConvolveIQ(const std::vector<float>& input, std::vector<float>& output) {
|
void IFilter::ConvolveIQ(const std::vector<float>& input, std::vector<float>& output) {
|
||||||
// TODO: SIMD
|
// TODO: SIMD
|
||||||
|
|
||||||
|
output.clear();
|
||||||
|
|
||||||
auto itIn = input.begin();
|
auto itIn = input.begin();
|
||||||
auto itOut = output.begin();
|
auto itOut = output.begin();
|
||||||
|
|
||||||
while (itIn != input.end()) {
|
while (itIn != input.end()) {
|
||||||
|
float I = 0, Q = 0;
|
||||||
for (float h : impulseResponse) {
|
for (float h : impulseResponse) {
|
||||||
// I
|
// I
|
||||||
*itOut += *itIn * h;
|
I += *itIn * h;
|
||||||
itIn++;
|
|
||||||
itOut++;
|
|
||||||
|
|
||||||
// Q
|
// Q
|
||||||
*itOut += *itIn * h;
|
Q += *itIn * h;
|
||||||
itIn++;
|
|
||||||
itOut++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
*(itOut + 0) = I;
|
||||||
|
*(itOut + 1) = Q;
|
||||||
|
|
||||||
|
itIn += 2;
|
||||||
|
itOut += 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} // namespace Filter
|
} // namespace Filter
|
@ -1,21 +1,26 @@
|
|||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
#include <cstddef>
|
||||||
|
#include <iostream>
|
||||||
#include <numbers>
|
#include <numbers>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include <filter/filter/low_pass_filter.hpp>
|
#include <filter/filter/low_pass_filter.hpp>
|
||||||
|
|
||||||
|
using std::numbers::pi;
|
||||||
|
|
||||||
namespace Filter {
|
namespace Filter {
|
||||||
LowPassFilter::LowPassFilter(std::size_t size, std::size_t cutoffHz, std::size_t sampleRateHz, const std::vector<float>& window) : IFilter(size) {
|
LowPassFilter::LowPassFilter(std::int32_t size, std::size_t cutoffHz, std::size_t sampleRateHz, const std::vector<float>& window) : IFilter(size) {
|
||||||
for (std::size_t i = 0; i < size; i++) {
|
for (std::int32_t i = 0; i < size; i++) {
|
||||||
std::size_t shift = std::round(size / 2);
|
int shift = size / 2.0f;
|
||||||
float sampleTime = 1.0f / sampleRateHz;
|
float sampleTimeS = 1.0f / sampleRateHz;
|
||||||
|
|
||||||
if (shift != i) {
|
if (shift != i) {
|
||||||
IFilter::impulseResponse[i] = std::sin(2.0f * std::numbers::pi * sampleTime * (i - shift)) / (std::numbers::pi * sampleTime * (i - shift));
|
IFilter::impulseResponse[i] = std::sin(2.0f * pi * cutoffHz * sampleTimeS * (i - shift)) / (pi * sampleTimeS * (i - shift));
|
||||||
} else {
|
} else {
|
||||||
IFilter::impulseResponse[i] = 2.0f * cutoffHz;
|
IFilter::impulseResponse[i] = 2.0f * cutoffHz;
|
||||||
}
|
}
|
||||||
|
|
||||||
IFilter::impulseResponse[i] *= sampleTime * window[i];
|
IFilter::impulseResponse[i] *= sampleTimeS * window[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
#include <cxxopts.hpp>
|
#include <cxxopts.hpp>
|
||||||
|
|
||||||
const std::size_t BUFFER_SIZE = 256;
|
const std::size_t BUFFER_SIZE = 256;
|
||||||
|
const std::size_t FILTER_SIZE = 64;
|
||||||
|
|
||||||
using namespace Filter;
|
using namespace Filter;
|
||||||
|
|
||||||
@ -45,14 +46,15 @@ int main(int argc, char** argv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// create low pass filter
|
// create low pass filter
|
||||||
LowPassFilter lpf(BUFFER_SIZE, result["cutoff"].as<std::size_t>(), result["sample_rate"].as<std::size_t>(), HammingWindow(BUFFER_SIZE).getWindow());
|
LowPassFilter lpf(FILTER_SIZE, result["cutoff"].as<std::size_t>(), result["sample_rate"].as<std::size_t>(), HammingWindow(FILTER_SIZE).getWindow());
|
||||||
|
|
||||||
|
lpf.Dump("filter.dmp");
|
||||||
|
HammingWindow(FILTER_SIZE).Dump("window.dmp");
|
||||||
|
|
||||||
// read data
|
// read data
|
||||||
std::vector<float> inputBuffer;
|
std::vector<float> inputBuffer(BUFFER_SIZE);
|
||||||
inputBuffer.reserve(BUFFER_SIZE);
|
|
||||||
|
|
||||||
std::vector<float> outputBuffer;
|
std::vector<float> outputBuffer(BUFFER_SIZE);
|
||||||
outputBuffer.reserve(BUFFER_SIZE);
|
|
||||||
|
|
||||||
std::size_t len;
|
std::size_t len;
|
||||||
|
|
||||||
@ -63,6 +65,8 @@ int main(int argc, char** argv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
lpf.ConvolveIQ(inputBuffer, outputBuffer);
|
lpf.ConvolveIQ(inputBuffer, outputBuffer);
|
||||||
|
|
||||||
|
std::fwrite(outputBuffer.data(), sizeof(float), BUFFER_SIZE, stdout);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1,13 +1,21 @@
|
|||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <fstream>
|
||||||
#include <numbers>
|
#include <numbers>
|
||||||
|
|
||||||
#include <filter/window/hamming_window.hpp>
|
#include <filter/window/hamming_window.hpp>
|
||||||
|
|
||||||
namespace Filter {
|
namespace Filter {
|
||||||
HammingWindow::HammingWindow(std::size_t width) {
|
HammingWindow::HammingWindow(std::size_t width) : window(width) {
|
||||||
window.reserve(width);
|
|
||||||
for (std::size_t i = 0; i < width; i++) {
|
for (std::size_t i = 0; i < width; i++) {
|
||||||
window[i] = (25 / 46.0) - (21 / 46.0) * std::cos(2 * std::numbers::pi * i / width);
|
window[i] = (25 / 46.0f) - (21 / 46.0f) * std::cos(2.0f * std::numbers::pi * i / (float)width);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HammingWindow::Dump(const std::string& path) const {
|
||||||
|
std::ofstream output;
|
||||||
|
output.open(path, std::ios::out | std::ios::binary);
|
||||||
|
output.write(reinterpret_cast<const char*>(getWindow().data()), sizeof(float) * getWindow().size());
|
||||||
|
output.close();
|
||||||
|
}
|
||||||
} // namespace Filter
|
} // namespace Filter
|
Loading…
Reference in New Issue
Block a user