added trigonometric approximations

This commit is contained in:
Robert 2021-04-30 13:25:09 +02:00
parent 0609fe6c33
commit bb947b4a53
3 changed files with 62 additions and 7 deletions

View file

@ -8,11 +8,24 @@
#define POW_OF_TWO(x) (x && !(x & (x - 1))) #define POW_OF_TWO(x) (x && !(x & (x - 1)))
constexpr double REC_2_FAC = (double)1.0f / (double)2.0f;
constexpr double REC_3_FAC = (double)1.0f / (double)6.0f;
constexpr double REC_4_FAC = (double)1.0f / (double)24.0f;
constexpr double REC_5_FAC = (double)1.0f / (double)120.0f;
constexpr double REC_6_FAC = (double)1.0f / (double)720.0f;
constexpr double REC_7_FAC = (double)1.0f / (double)5040.0f;
constexpr double REC_8_FAC = (double)1.0f / (double)40320.0f;
constexpr double REC_9_FAC = (double)1.0f / (double)362880.0f;
using namespace std::complex_literals; using namespace std::complex_literals;
typedef std::function<double(unsigned int)> WindowFunction; typedef std::function<double(unsigned int)> WindowFunction;
typedef std::function<double(double)> TrigFunction;
typedef std::function<std::complex<double>(double)> ExpFunction;
static WindowFunction window; WindowFunction window;
TrigFunction Sin = std::bind((double(*)(double))& std::sin, std::placeholders::_1);
TrigFunction Cos = std::bind((double(*)(double))& std::cos, std::placeholders::_1);
inline double WindowRectangle(unsigned int k, unsigned int offset, unsigned int width); inline double WindowRectangle(unsigned int k, unsigned int offset, unsigned int width);
inline double WindowVonHann(unsigned int k, unsigned int offset, unsigned int width); inline double WindowVonHann(unsigned int k, unsigned int offset, unsigned int width);
@ -20,6 +33,10 @@ inline double WindowGauss(unsigned int k, unsigned int offset, unsigned int widt
inline double WindowTriangle(unsigned int k, unsigned int offset, unsigned int width); inline double WindowTriangle(unsigned int k, unsigned int offset, unsigned int width);
inline double WindowBlackman(unsigned int k, unsigned int offset, unsigned int width); inline double WindowBlackman(unsigned int k, unsigned int offset, unsigned int width);
double FastCos(double x);
double FastSin(double x);
std::complex<double> ComplexExp(double x);
std::vector<std::complex<double>> std::vector<std::complex<double>>
radix2dit( radix2dit(
const std::vector<double>& list, const std::vector<double>& list,
@ -38,12 +55,12 @@ radix2dit(
std::vector<std::complex<double>> first = radix2dit(list, offset, halfN, s << 1); std::vector<std::complex<double>> first = radix2dit(list, offset, halfN, s << 1);
std::vector<std::complex<double>> second = radix2dit(list, offset + s, halfN, s << 1); std::vector<std::complex<double>> second = radix2dit(list, offset + s, halfN, s << 1);
std::complex<double> coeff = -M_PI * 1.0i / (double)halfN; double coeff = -M_PI / (double)halfN;
for (int k = 0; k < halfN; k++) for (int k = 0; k < halfN; k++)
{ {
std::complex<double> p = first[k]; std::complex<double> p = first[k];
std::complex<double> q = std::exp(coeff * (double)k) * second[k]; std::complex<double> q = ComplexExp(coeff * (double)k) * second[k];
output[k] = p + q; output[k] = p + q;
output[halfN + k] = p - q; output[halfN + k] = p - q;
@ -76,7 +93,7 @@ FFT(const std::vector<double>::const_iterator& begin,
WindowFunction f; WindowFunction f;
std::vector<std::complex<double>> spectrum = radix2dit(signal, 0, N, 1); std::vector<std::complex<double>> spectrum = radix2dit(signal, 0, N, 1);
double freqRes = (double)sampleRate / (double)N; double freqRes = (double)sampleRate / (double)N;
@ -109,6 +126,12 @@ void SetWindowFunction(WindowFunctions func, unsigned int width)
} }
} }
void UseFastFunctions()
{
Sin = std::bind(FastSin, std::placeholders::_1);
Cos = std::bind(FastCos, std::placeholders::_1);
}
inline double WindowRectangle(unsigned int k, unsigned int offset, unsigned int width) inline double WindowRectangle(unsigned int k, unsigned int offset, unsigned int width)
{ {
return ((offset < k) && (k < width)); return ((offset < k) && (k < width));
@ -116,7 +139,7 @@ inline double WindowRectangle(unsigned int k, unsigned int offset, unsigned int
inline double WindowVonHann(unsigned int k, unsigned int offset, unsigned int width) inline double WindowVonHann(unsigned int k, unsigned int offset, unsigned int width)
{ {
return ((offset < k) && (k < width)) ? (0.5f * (1.0f - cos(2.0f * M_PI * k / (width - 1)))) : 0; return ((offset < k) && (k < width)) ? (0.5f * (1.0f - Cos(2.0f * M_PI * k / (width - 1)))) : 0;
} }
inline double WindowGauss(unsigned int k, unsigned int offset, unsigned int width) inline double WindowGauss(unsigned int k, unsigned int offset, unsigned int width)
@ -132,5 +155,30 @@ inline double WindowTriangle(unsigned int k, unsigned int offset, unsigned int w
inline double WindowBlackman(unsigned int k, unsigned int offset, unsigned int width) inline double WindowBlackman(unsigned int k, unsigned int offset, unsigned int width)
{ {
return 0.5f * (1.0f - 0.16f) - 0.5f * cos(2.0f * M_PI * k / (width - 1)) + 0.5f * 0.16f * cos(4.0f * M_PI * k / (width - 1)); return (double)0.5f * ((double)1.0f - (double)0.16f) - 0.5f * Cos(2.0f * M_PI * k / (width - 1)) + (double)0.5f * (double)0.16f * Cos(4.0f * M_PI * k / (width - 1));
}
double FastCos(double x)
{
x -= (x > M_PI) * (double)2.0f * M_PI;
x += (x < -M_PI) * (double)2.0f * M_PI;
double xpow2 = x * x;
double xpow4 = xpow2 * x * x;
double xpow6 = xpow4 * x * x;
return (double)1.0f - xpow2 * REC_2_FAC + xpow4 * REC_4_FAC - xpow6 * REC_6_FAC + xpow6 * x * x * REC_8_FAC;
}
double FastSin(double x)
{
x -= (x > M_PI) * (double)2.0f * M_PI;
x += (x < -M_PI) * (double)2.0f * M_PI;
double xpow3 = x * x * x;
double xpow5 = xpow3 * x * x;
double xpow7 = xpow5 * x * x;
return (double)x - xpow3 * REC_3_FAC + xpow5 * REC_5_FAC - xpow7 * REC_7_FAC + xpow7 * x * x * REC_9_FAC;
}
std::complex<double> ComplexExp(double x)
{
return std::complex<double>(Cos(x), Sin(x));
} }

View file

@ -16,4 +16,5 @@ extern std::vector<std::pair<double, double>> FFT(const std::vector<double>::con
double minFreq, double maxFreq, double minFreq, double maxFreq,
unsigned int zeropadding); unsigned int zeropadding);
extern void SetWindowFunction(WindowFunctions func, unsigned int width); extern void SetWindowFunction(WindowFunctions func, unsigned int width);
extern void UseFastFunctions();

View file

@ -26,6 +26,7 @@ struct Settings {
double minFreq, maxFreq; double minFreq, maxFreq;
unsigned int analyzeChannel; unsigned int analyzeChannel;
unsigned int zeropadding; unsigned int zeropadding;
bool approx;
WindowFunctions window; WindowFunctions window;
}; };
@ -36,6 +37,9 @@ int main(int argc, char** argv)
Settings setts; Settings setts;
setts = Parse(argc, argv); setts = Parse(argc, argv);
if (setts.approx)
UseFastFunctions();
int numFiles = setts.files.size(); int numFiles = setts.files.size();
for (auto& file : setts.files) { for (auto& file : setts.files) {
AudioFile<double> audioFile; AudioFile<double> audioFile;
@ -141,6 +145,7 @@ Settings Parse(int argc, char** argv)
("p,pad", "Add extra zero-padding. By default, the program will pad the signals with 0s until the number of samples is a power of 2 (this would be equivalent to -p 1). With this option you can tell the program to instead pad until the power of 2 after the next one (-p 2) etc. This increases frequency resolution", cxxopts::value<unsigned int>()) ("p,pad", "Add extra zero-padding. By default, the program will pad the signals with 0s until the number of samples is a power of 2 (this would be equivalent to -p 1). With this option you can tell the program to instead pad until the power of 2 after the next one (-p 2) etc. This increases frequency resolution", cxxopts::value<unsigned int>())
("w,window", "Specify the window function used (rectangle (default), von-hann, gauss, triangle, blackman (3-term))", cxxopts::value<std::string>()->default_value("rectangle")) ("w,window", "Specify the window function used (rectangle (default), von-hann, gauss, triangle, blackman (3-term))", cxxopts::value<std::string>()->default_value("rectangle"))
("m,mono", "Analyze only the given channel", cxxopts::value<unsigned int>()->default_value("0")) ("m,mono", "Analyze only the given channel", cxxopts::value<unsigned int>()->default_value("0"))
("approx", "Use faster, but more inaccurate trigonometric functions instead of the std-functions (EXPERIMENTAL)")
("files", "Files to fourier transform", cxxopts::value<std::vector<std::filesystem::path>>()) ("files", "Files to fourier transform", cxxopts::value<std::vector<std::filesystem::path>>())
("h,help", "Print usage") ("h,help", "Print usage")
; ;
@ -176,6 +181,7 @@ Settings Parse(int argc, char** argv)
setts.splitInterval = (result.count("interval") ? result["interval"].as<float>() : 0.0f); setts.splitInterval = (result.count("interval") ? result["interval"].as<float>() : 0.0f);
setts.analyzeChannel = (result.count("mono") ? result["mono"].as<unsigned int>() : 0); setts.analyzeChannel = (result.count("mono") ? result["mono"].as<unsigned int>() : 0);
setts.zeropadding = (result.count("pad") ? result["pad"].as<unsigned int>() : 1); setts.zeropadding = (result.count("pad") ? result["pad"].as<unsigned int>() : 1);
setts.approx = (result.count("approx") ? true : false);
if (!result.count("window")) if (!result.count("window"))
{ {