2024-03-11 16:04:24 +01:00
|
|
|
#include <pybind11/pybind11.h>
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
2022-10-11 09:43:36 +02:00
|
|
|
#include "arma_npy.h"
|
2022-09-22 10:18:38 +02:00
|
|
|
#include "lasp_avpowerspectra.h"
|
2022-08-16 21:22:35 +02:00
|
|
|
#include "lasp_biquadbank.h"
|
2022-09-22 10:18:38 +02:00
|
|
|
#include "lasp_fft.h"
|
2022-10-06 21:13:21 +02:00
|
|
|
#include "lasp_filter.h"
|
2024-03-12 09:21:07 +01:00
|
|
|
#include "lasp_freqsmooth.h"
|
2022-08-16 21:22:35 +02:00
|
|
|
#include "lasp_slm.h"
|
2022-10-11 09:43:36 +02:00
|
|
|
#include "lasp_streammgr.h"
|
2022-08-16 21:22:35 +02:00
|
|
|
#include "lasp_window.h"
|
2022-06-29 12:25:32 +02:00
|
|
|
|
|
|
|
using std::cerr;
|
2022-10-11 09:43:36 +02:00
|
|
|
using std::endl;
|
2022-06-29 12:25:32 +02:00
|
|
|
namespace py = pybind11;
|
2022-10-11 14:50:44 +02:00
|
|
|
using rte = std::runtime_error;
|
2022-06-29 12:25:32 +02:00
|
|
|
|
2022-09-22 10:18:38 +02:00
|
|
|
/**
|
|
|
|
* \ingroup pybind
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize DSP code
|
|
|
|
*
|
|
|
|
* @param m The Python module to add classes and methods to
|
|
|
|
*/
|
2022-10-11 09:43:36 +02:00
|
|
|
|
2022-08-16 21:22:35 +02:00
|
|
|
void init_dsp(py::module &m) {
|
2022-09-03 20:59:14 +02:00
|
|
|
py::class_<Fft> fft(m, "Fft");
|
|
|
|
fft.def(py::init<us>());
|
2022-10-11 14:50:44 +02:00
|
|
|
fft.def("fft", [](Fft &f, dpyarray dat) {
|
|
|
|
if (dat.ndim() == 1) {
|
|
|
|
return ColToNpy<c>(f.fft(NpyToCol<d, false>(dat)));
|
|
|
|
} else if (dat.ndim() == 2) {
|
|
|
|
return MatToNpy<c>(f.fft(NpyToMat<d, false>(dat)));
|
|
|
|
} else {
|
|
|
|
throw rte("Invalid dimensions of array");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
fft.def("ifft", [](Fft &f, cpyarray dat) {
|
|
|
|
if (dat.ndim() == 1) {
|
|
|
|
return ColToNpy<d>(f.ifft(NpyToCol<c, false>(dat)));
|
|
|
|
} else if (dat.ndim() == 2) {
|
|
|
|
return MatToNpy<d>(f.ifft(NpyToMat<c, false>(dat)));
|
|
|
|
} else {
|
|
|
|
throw rte("Invalid dimensions of array");
|
|
|
|
}
|
|
|
|
});
|
2022-09-03 20:59:14 +02:00
|
|
|
|
2022-09-22 10:18:38 +02:00
|
|
|
fft.def_static("load_fft_wisdom", &Fft::load_fft_wisdom);
|
|
|
|
fft.def_static("store_fft_wisdom", &Fft::store_fft_wisdom);
|
2022-09-03 20:59:14 +02:00
|
|
|
|
2022-10-05 21:51:47 +02:00
|
|
|
/// Window
|
2022-06-29 12:25:32 +02:00
|
|
|
py::class_<Window> w(m, "Window");
|
|
|
|
|
|
|
|
py::enum_<Window::WindowType>(w, "WindowType")
|
|
|
|
.value("Hann", Window::WindowType::Hann)
|
|
|
|
.value("Hamming", Window::WindowType::Hamming)
|
|
|
|
.value("Bartlett", Window::WindowType::Bartlett)
|
|
|
|
.value("Blackman", Window::WindowType::Bartlett)
|
2022-09-22 10:18:38 +02:00
|
|
|
.value("Rectangular", Window::WindowType::Rectangular);
|
|
|
|
|
|
|
|
w.def_static("toTxt", &Window::toText);
|
2022-06-29 12:25:32 +02:00
|
|
|
|
2022-10-05 21:51:47 +02:00
|
|
|
py::class_<Filter, std::shared_ptr<Filter>> filter(m, "Filter");
|
2022-07-20 14:58:48 +02:00
|
|
|
|
2022-10-05 21:51:47 +02:00
|
|
|
/// SeriesBiquad
|
|
|
|
py::class_<SeriesBiquad, std::shared_ptr<SeriesBiquad>> sbq(m, "SeriesBiquad",
|
|
|
|
filter);
|
2022-10-11 10:43:31 +02:00
|
|
|
sbq.def(py::init([](dpyarray filter) {
|
2022-10-11 14:50:44 +02:00
|
|
|
return std::make_shared<SeriesBiquad>(NpyToCol<d, false>(filter));
|
2022-10-11 10:43:31 +02:00
|
|
|
}));
|
2022-10-11 09:43:36 +02:00
|
|
|
sbq.def("filter", [](SeriesBiquad &s, dpyarray input) {
|
|
|
|
vd res = NpyToCol<d, true>(input);
|
2022-08-16 21:22:35 +02:00
|
|
|
s.filter(res);
|
2022-10-11 09:43:36 +02:00
|
|
|
return ColToNpy<d>(res);
|
2022-08-16 21:22:35 +02:00
|
|
|
});
|
2022-10-05 21:51:47 +02:00
|
|
|
|
2022-09-22 10:18:38 +02:00
|
|
|
/// BiquadBank
|
2022-10-12 21:32:24 +02:00
|
|
|
py::class_<BiquadBank, Filter, std::shared_ptr<BiquadBank>> bqb(m,
|
|
|
|
"BiquadBank");
|
|
|
|
bqb.def(py::init([](dpyarray coefs) {
|
|
|
|
return std::make_shared<BiquadBank>(NpyToMat<d, false>(coefs));
|
|
|
|
}));
|
|
|
|
bqb.def(py::init([](dpyarray coefs, dpyarray gains) {
|
|
|
|
vd gains_arma = NpyToMat<d, false>(gains);
|
|
|
|
return std::make_shared<BiquadBank>(NpyToMat<d, false>(coefs), &gains_arma);
|
|
|
|
}));
|
|
|
|
|
2022-10-11 10:43:31 +02:00
|
|
|
bqb.def("setGains",
|
|
|
|
[](BiquadBank &b, dpyarray gains) { b.setGains(NpyToCol(gains)); });
|
2022-10-11 09:43:36 +02:00
|
|
|
bqb.def("filter", [](BiquadBank &b, dpyarray input) {
|
2022-10-12 21:32:24 +02:00
|
|
|
// Yes, a copy here
|
2022-10-11 09:43:36 +02:00
|
|
|
vd inout = NpyToCol<d, true>(input);
|
|
|
|
b.filter(inout);
|
|
|
|
return ColToNpy(inout);
|
|
|
|
});
|
2022-08-16 21:22:35 +02:00
|
|
|
|
2022-10-05 21:51:47 +02:00
|
|
|
/// PowerSpectra
|
2022-09-03 20:59:14 +02:00
|
|
|
py::class_<PowerSpectra> ps(m, "PowerSpectra");
|
2022-09-22 10:18:38 +02:00
|
|
|
ps.def(py::init<const us, const Window::WindowType>());
|
2022-10-11 09:43:36 +02:00
|
|
|
ps.def("compute", [](PowerSpectra &ps, dpyarray input) {
|
|
|
|
return CubeToNpy<c>(ps.compute(NpyToMat<d, false>(input)));
|
|
|
|
});
|
2022-09-03 20:59:14 +02:00
|
|
|
|
2022-10-05 21:51:47 +02:00
|
|
|
/// AvPowerSpectra
|
2022-09-03 20:59:14 +02:00
|
|
|
py::class_<AvPowerSpectra> aps(m, "AvPowerSpectra");
|
2022-10-06 21:13:21 +02:00
|
|
|
aps.def(py::init<const us, const Window::WindowType, const d, const d>(),
|
2022-10-05 21:51:47 +02:00
|
|
|
py::arg("nfft") = 2048,
|
|
|
|
py::arg("windowType") = Window::WindowType::Hann,
|
|
|
|
py::arg("overlap_percentage") = 50.0, py::arg("time_constant") = -1);
|
2022-10-01 19:59:35 +02:00
|
|
|
|
2022-10-11 09:43:36 +02:00
|
|
|
aps.def("compute", [](AvPowerSpectra &aps, dpyarray timedata) {
|
2022-10-11 14:50:44 +02:00
|
|
|
std::optional<ccube> res;
|
2024-03-06 22:12:42 +01:00
|
|
|
dmat timedata_mat = NpyToMat<d, false>(timedata);
|
2022-10-11 14:50:44 +02:00
|
|
|
{
|
|
|
|
py::gil_scoped_release release;
|
2024-03-06 22:12:42 +01:00
|
|
|
res = aps.compute(timedata_mat);
|
2022-10-11 14:50:44 +02:00
|
|
|
}
|
2022-10-11 09:43:36 +02:00
|
|
|
|
2022-10-11 14:50:44 +02:00
|
|
|
return CubeToNpy<c>(res.value_or(ccube(0, 0, 0)));
|
|
|
|
});
|
|
|
|
aps.def("get_est", [](const AvPowerSpectra &ps) {
|
2022-10-16 21:26:06 +02:00
|
|
|
ccube est = ps.get_est();
|
|
|
|
return CubeToNpy<c>(est);
|
2022-10-05 21:51:47 +02:00
|
|
|
});
|
2022-09-03 20:59:14 +02:00
|
|
|
|
2022-10-01 19:59:35 +02:00
|
|
|
py::class_<SLM> slm(m, "cppSLM");
|
2022-09-03 20:59:14 +02:00
|
|
|
|
2022-10-11 09:43:36 +02:00
|
|
|
slm.def_static("fromBiquads", [](const d fs, const d Lref, const us ds,
|
|
|
|
const d tau, dpyarray bandpass) {
|
|
|
|
return SLM::fromBiquads(fs, Lref, ds, tau, NpyToMat<d, false>(bandpass));
|
|
|
|
});
|
|
|
|
|
|
|
|
slm.def_static("fromBiquads", [](const d fs, const d Lref, const us ds,
|
|
|
|
const d tau, dpyarray prefilter,
|
|
|
|
py::array_t<d> bandpass) {
|
|
|
|
return SLM::fromBiquads(fs, Lref, ds, tau, NpyToCol<d, false>(prefilter),
|
|
|
|
NpyToMat<d, false>(bandpass));
|
|
|
|
});
|
2022-10-06 21:13:21 +02:00
|
|
|
|
2022-10-11 09:43:36 +02:00
|
|
|
slm.def("run", [](SLM &slm, dpyarray in) {
|
|
|
|
return MatToNpy<d>(slm.run(NpyToCol<d, false>(in)));
|
|
|
|
});
|
|
|
|
slm.def("Pm", [](const SLM &slm) { return ColToNpy<d>(slm.Pm); });
|
|
|
|
slm.def("Pmax", [](const SLM &slm) { return ColToNpy<d>(slm.Pmax); });
|
|
|
|
slm.def("Ppeak", [](const SLM &slm) { return ColToNpy<d>(slm.Ppeak); });
|
|
|
|
|
|
|
|
slm.def("Leq", [](const SLM &slm) { return ColToNpy<d>(slm.Leq()); });
|
|
|
|
slm.def("Lmax", [](const SLM &slm) { return ColToNpy<d>(slm.Lmax()); });
|
|
|
|
slm.def("Lpeak", [](const SLM &slm) { return ColToNpy<d>(slm.Lpeak()); });
|
|
|
|
slm.def_static("suggestedDownSamplingFac", &SLM::suggestedDownSamplingFac);
|
2024-03-11 16:04:24 +01:00
|
|
|
|
|
|
|
// Frequency smoother
|
|
|
|
m.def("freqSmooth", [](dpyarray freq, dpyarray X, unsigned w) {
|
|
|
|
vd freqa = NpyToCol<d, false>(freq);
|
|
|
|
vd Xa = NpyToCol<d, false>(X);
|
|
|
|
return ColToNpy(freqSmooth(freqa, Xa, w));
|
|
|
|
});
|
2022-06-29 12:25:32 +02:00
|
|
|
}
|
2022-09-22 10:18:38 +02:00
|
|
|
/** @} */
|