mirror of
https://github.com/f4exb/sdrangel.git
synced 2025-04-04 10:38:45 -04:00
Deep redesign: more optimizations
This commit is contained in:
parent
f5021f5b9e
commit
b3fd6bad15
37
Readme.md
37
Readme.md
@ -22,6 +22,24 @@ These plugins come from the parent code base and are still present in the source
|
||||
- osmosdr
|
||||
- v4l-msi
|
||||
- v4l-rtl
|
||||
|
||||
<h3>Gnuradio</h3>
|
||||
|
||||
The Gnuradio plugin source needs extra packages, including `liblog4cpp-dev libboost-system-dev gnuradio-dev libosmosdr-dev`
|
||||
|
||||
If you use your own location for Gnuradio install directory you need to specify library and include locations. Example with `/opt/install/gnuradio-3.7.5.1` with the following defines on `cmake` command line:
|
||||
|
||||
`-DGNURADIO_RUNTIME_LIBRARIES=/opt/install/gnuradio-3.7.5.1/lib/libgnuradio-runtime.so -DGNURADIO_RUNTIME_INCLUDE_DIRS=/opt/install/gnuradio-3.7.5.1/include`
|
||||
|
||||
<h3>osmosdr</h3>
|
||||
|
||||
If you use your own location for gr.osmocom install directory you need to specify library and include locations. Example with `/opt/install/gr-osmosdr` with the following defines on `cmake` command line:
|
||||
|
||||
`-DGNURADIO_OSMOSDR_LIBRARIES=/opt/install/gr-osmosdr/lib/libgnuradio-osmosdr.so -DGNURADIO_OSMOSDR_INCLUDE_DIRS=/opt/install/gr-osmosdr/include`
|
||||
|
||||
<h3>v4l-*</h3>
|
||||
|
||||
Use `cmake ../ -DV4L-RTL=ON` to build the Linux kernel driver for RTL-SDR (Experimental). Needs a recent kernel and libv4l2. Will need extra work to support SDRPlay. Needs `cp KERNEL_SOURCE/include/linux/compiler.h /usr/include/linux/` and `cp KERNEL_SOURCE/include/uapi/linux/videodev2.h /usr/include/uapi/linux/` and package `libv4l-dev`.
|
||||
|
||||
<h1>Supported hardware</h1>
|
||||
|
||||
@ -57,22 +75,6 @@ If you use your own location for librtlsdr install directory you need to specify
|
||||
|
||||
`librtlsdr-dev` is in the `universe` repo. (utopic 14.10 amd64.)
|
||||
|
||||
Use `cmake ../ -DV4L-RTL=ON` to build the Linux kernel driver for RTL-SDR (Experimental). Needs a recent kernel and libv4l2. Will need extra work to support SDRPlay. Needs `cp KERNEL_SOURCE/include/linux/compiler.h /usr/include/linux/` and `cp KERNEL_SOURCE/include/uapi/linux/videodev2.h /usr/include/uapi/linux/` and package `libv4l-dev`.
|
||||
|
||||
The Gnuradio plugin source needs extra packages, including `liblog4cpp-dev libboost-system-dev gnuradio-dev libosmosdr-dev`
|
||||
|
||||
For non standard installations of RTL-SDR library, the GNU Radio runtime and gr.osmocom drivers use the following variables in the cmake command line. The paths specified are just examples:
|
||||
|
||||
- For GNU Radio runtime:
|
||||
- Includes: `-DGNURADIO_RUNTIME_INCLUDE_DIRS=/opt/install/gnuradio-3.7.5.1/include`
|
||||
- Library: `-DGNURADIO_RUNTIME_LIBRARIES=/opt/install/gnuradio-3.7.5.1/lib/libgnuradio-runtime.so`
|
||||
- For gr.osmocom:
|
||||
- Includes: `-DGNURADIO_OSMOSDR_INCLUDE_DIRS=/opt/install/gr-osmosdr/include`
|
||||
- Library: `-DGNURADIO_OSMOSDR_LIBRARIES=/opt/install/gr-osmosdr/lib/libgnuradio-osmosdr.so`
|
||||
- For RTL-SDR library:
|
||||
- Includes: `-DLIBRTLSDR_INCLUDE_DIR=/opt/install/rtlsdr/include`
|
||||
- Library: `-DLIBRTLSDR_LIBRARIES=/opt/install/rtlsdr/lib/librtlsdr.so`
|
||||
|
||||
There is no installation procedure the executable is at the root of the build directory
|
||||
|
||||
<h2>For Debian</h2>
|
||||
@ -87,8 +89,6 @@ Assuming Debian Jessie is used:
|
||||
|
||||
`mkdir build && cd build && cmake ../ && make`
|
||||
|
||||
Then the same remarks as for Ubuntu apply...
|
||||
|
||||
<h1>Known Issues</h1>
|
||||
|
||||
- You will need to stop input before changing preset then start again
|
||||
@ -147,6 +147,7 @@ Then the same remarks as for Ubuntu apply...
|
||||
- Message queuing and handling redesign. Still not completely satisfactory
|
||||
- Objects have their own input and output message queues
|
||||
- Dedicated message queue to communicate to the GUI for objects coupled with a GUI
|
||||
- Optimizations with Valgrind cachegrind
|
||||
- Many other little things...
|
||||
|
||||
<h1>To Do</h1>
|
||||
|
@ -76,32 +76,33 @@ public:
|
||||
Type acc = 0;
|
||||
int a = m_ptr;
|
||||
int b = a - 1;
|
||||
int i, size;
|
||||
int i, n_taps, size;
|
||||
|
||||
m_samples[m_ptr] = sample;
|
||||
size = m_samples.size(); // Valgrind optim (2)
|
||||
|
||||
while(b < 0)
|
||||
{
|
||||
b += m_samples.size();
|
||||
b += size;
|
||||
}
|
||||
|
||||
size = (int) m_taps.size() - 1; // Valgrind optim
|
||||
n_taps = m_taps.size() - 1; // Valgrind optim
|
||||
|
||||
for(i = 0; i < size; i++)
|
||||
for(i = 0; i < n_taps; i++)
|
||||
{
|
||||
acc += (m_samples[a] + m_samples[b]) * m_taps[i];
|
||||
a++;
|
||||
|
||||
while(a >= (int)m_samples.size())
|
||||
while (a >= size)
|
||||
{
|
||||
a -= m_samples.size();
|
||||
a -= size;
|
||||
}
|
||||
|
||||
b--;
|
||||
|
||||
while(b < 0)
|
||||
{
|
||||
b += m_samples.size();
|
||||
b += size;
|
||||
}
|
||||
}
|
||||
|
||||
@ -109,9 +110,9 @@ public:
|
||||
|
||||
m_ptr++;
|
||||
|
||||
while(m_ptr >= (int)m_samples.size())
|
||||
while (m_ptr >= size)
|
||||
{
|
||||
m_ptr -= m_samples.size();
|
||||
m_ptr -= size;
|
||||
}
|
||||
|
||||
return acc;
|
||||
|
@ -58,9 +58,11 @@ void Decimators<T>::decimate1(SampleVector::iterator* it, const T* buf, qint32 l
|
||||
for (int pos = 0; pos < len; pos += 2) {
|
||||
xreal = buf[pos+0];
|
||||
yimag = buf[pos+1];
|
||||
Sample s( xreal * 16, yimag * 16 ); // shift by 4 bit positions (signed)
|
||||
**it = s;
|
||||
(*it)++;
|
||||
(**it).setReal(xreal * 16); // Valgrind optim (2 - comment not repeated)
|
||||
(**it).setImag(yimag * 16);
|
||||
//Sample s( xreal * 16, yimag * 16 ); // shift by 4 bit positions (signed)
|
||||
//**it = s;
|
||||
++(*it); // Valgrind optim (comment not repeated)
|
||||
}
|
||||
}
|
||||
|
||||
@ -68,17 +70,23 @@ template<typename T>
|
||||
void Decimators<T>::decimate2_u(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 7; pos += 8) {
|
||||
|
||||
for (int pos = 0; pos < len - 7; pos += 8)
|
||||
{
|
||||
xreal = buf[pos+0] - buf[pos+3];
|
||||
yimag = buf[pos+1] + buf[pos+2] - 255;
|
||||
Sample s( xreal << 3, yimag << 3 );
|
||||
**it = s;
|
||||
(*it)++;
|
||||
//Sample s( xreal << 3, yimag << 3 );
|
||||
//**it = s;
|
||||
(**it).setReal(xreal << 3);
|
||||
(**it).setImag(yimag << 3);
|
||||
++(*it);
|
||||
xreal = buf[pos+7] - buf[pos+4];
|
||||
yimag = 255 - buf[pos+5] - buf[pos+6];
|
||||
Sample t( xreal << 3, yimag << 3 );
|
||||
**it = t;
|
||||
(*it)++;
|
||||
//Sample t( xreal << 3, yimag << 3 );
|
||||
//**it = t;
|
||||
(**it).setReal(xreal << 3);
|
||||
(**it).setImag(yimag << 3);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -86,17 +94,23 @@ template<typename T>
|
||||
void Decimators<T>::decimate2(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 7; pos += 8) {
|
||||
|
||||
for (int pos = 0; pos < len - 7; pos += 8)
|
||||
{
|
||||
xreal = buf[pos+0] - buf[pos+3];
|
||||
yimag = buf[pos+1] + buf[pos+2];
|
||||
Sample s( xreal << 3, yimag << 3 );
|
||||
**it = s;
|
||||
(*it)++;
|
||||
//Sample s( xreal << 3, yimag << 3 );
|
||||
//**it = s;
|
||||
(**it).setReal(xreal << 3);
|
||||
(**it).setImag(yimag << 3);
|
||||
++(*it);
|
||||
xreal = buf[pos+7] - buf[pos+4];
|
||||
yimag = - buf[pos+5] - buf[pos+6];
|
||||
Sample t( xreal << 3, yimag << 3 );
|
||||
**it = t;
|
||||
(*it)++;
|
||||
//Sample t( xreal << 3, yimag << 3 );
|
||||
//**it = t;
|
||||
(**it).setReal(xreal << 3);
|
||||
(**it).setImag(yimag << 3);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -104,192 +118,255 @@ template<typename T>
|
||||
void Decimators<T>::decimate2_sup(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 7; pos += 8) {
|
||||
|
||||
for (int pos = 0; pos < len - 7; pos += 8)
|
||||
{
|
||||
xreal = buf[pos+1] - buf[pos+2];
|
||||
yimag = - buf[pos+0] - buf[pos+3];
|
||||
Sample s( xreal << 3, yimag << 3 );
|
||||
**it = s;
|
||||
(*it)++;
|
||||
//Sample s( xreal << 3, yimag << 3 );
|
||||
//**it = s;
|
||||
(**it).setReal(xreal << 3);
|
||||
(**it).setImag(yimag << 3);
|
||||
++(*it);
|
||||
xreal = buf[pos+6] - buf[pos+5];
|
||||
yimag = buf[pos+4] + buf[pos+7];
|
||||
Sample t( xreal << 3, yimag << 3 );
|
||||
**it = t;
|
||||
(*it)++;
|
||||
//Sample t( xreal << 3, yimag << 3 );
|
||||
//**it = t;
|
||||
(**it).setReal(xreal << 3);
|
||||
(**it).setImag(yimag << 3);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate2_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
int pos = 0;
|
||||
|
||||
while (pos < len - 3)
|
||||
{
|
||||
Sample s0(buf[pos+0] << 3, buf[pos+1] << 3);
|
||||
pos += 2;
|
||||
|
||||
if (m_decimator2.workDecimateCenter(&s0))
|
||||
{
|
||||
**it = s0;
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}*/
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate2_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
for (int pos = 0; pos < len - 3; pos += 4) {
|
||||
Sample s1(buf[pos+0] << 3, buf[pos+1] << 3);
|
||||
Sample s2(buf[pos+2] << 3, buf[pos+3] << 3);
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
**it = s2;
|
||||
(*it)++;
|
||||
int pos = 0;
|
||||
|
||||
while (pos < len - 3)
|
||||
{
|
||||
qint16 x0 = buf[pos+0] << 3;
|
||||
qint16 y0 = buf[pos+1] << 3;
|
||||
pos += 2;
|
||||
|
||||
if (m_decimator2.workDecimateCenter(&x0, &y0))
|
||||
{
|
||||
(**it).setReal(x0);
|
||||
(**it).setImag(y0);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate4_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
int pos = 0;
|
||||
|
||||
while (pos < len)
|
||||
{
|
||||
Sample s0(buf[pos+0] << 4, buf[pos+1] << 4);
|
||||
pos += 2;
|
||||
|
||||
if (m_decimator2.workDecimateCenter(&s0))
|
||||
{
|
||||
Sample s1 = s0;
|
||||
|
||||
if (m_decimator4.workDecimateCenter(&s1))
|
||||
{
|
||||
(**it) = s1;
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
}*/
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate4_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
for (int pos = 0; pos < len - 7; pos += 8) {
|
||||
Sample s1(buf[pos+0] << 4, buf[pos+1] << 4);
|
||||
Sample s2(buf[pos+2] << 4, buf[pos+3] << 4);
|
||||
Sample s3(buf[pos+4] << 4, buf[pos+5] << 4);
|
||||
Sample s4(buf[pos+6] << 4, buf[pos+7] << 4);
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
m_decimator2.myDecimate(&s3, &s4);
|
||||
m_decimator4.myDecimate(&s2, &s4);
|
||||
**it = s4;
|
||||
(*it)++;
|
||||
int pos = 0;
|
||||
|
||||
while (pos < len)
|
||||
{
|
||||
qint16 x0 = buf[pos+0] << 3;
|
||||
qint16 y0 = buf[pos+1] << 3;
|
||||
pos += 2;
|
||||
|
||||
if (m_decimator2.workDecimateCenter(&x0, &y0))
|
||||
{
|
||||
qint16 x1 = x0;
|
||||
qint16 y1 = y0;
|
||||
|
||||
if (m_decimator4.workDecimateCenter(&x1, &y1))
|
||||
{
|
||||
(**it).setReal(x0);
|
||||
(**it).setImag(y0);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate8_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
for (int pos = 0; pos < len - 15; pos += 16) {
|
||||
Sample s1(buf[pos+0] << 4, buf[pos+1] << 4);
|
||||
Sample s2(buf[pos+2] << 4, buf[pos+3] << 4);
|
||||
Sample s3(buf[pos+4] << 4, buf[pos+5] << 4);
|
||||
Sample s4(buf[pos+6] << 4, buf[pos+7] << 4);
|
||||
Sample s5(buf[pos+8] << 4, buf[pos+9] << 4);
|
||||
Sample s6(buf[pos+10] << 4, buf[pos+11] << 4);
|
||||
Sample s7(buf[pos+12] << 4, buf[pos+13] << 4);
|
||||
Sample s8(buf[pos+14] << 4, buf[pos+15] << 4);
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
m_decimator2.myDecimate(&s3, &s4);
|
||||
m_decimator2.myDecimate(&s5, &s6);
|
||||
m_decimator2.myDecimate(&s7, &s8);
|
||||
m_decimator4.myDecimate(&s2, &s4);
|
||||
m_decimator4.myDecimate(&s6, &s8);
|
||||
m_decimator8.myDecimate(&s4, &s8);
|
||||
**it = s8;
|
||||
(*it)++;
|
||||
int pos = 0;
|
||||
|
||||
while (pos < len)
|
||||
{
|
||||
qint16 x0 = buf[pos+0] << 4;
|
||||
qint16 y0 = buf[pos+1] << 4;
|
||||
pos += 2;
|
||||
|
||||
if (m_decimator2.workDecimateCenter(&x0, &y0))
|
||||
{
|
||||
qint16 x1 = x0;
|
||||
qint16 y1 = y0;
|
||||
|
||||
if (m_decimator4.workDecimateCenter(&x1, &y1))
|
||||
{
|
||||
qint16 x2 = x1;
|
||||
qint16 y2 = y1;
|
||||
|
||||
if (m_decimator8.workDecimateCenter(&x2, &y2))
|
||||
{
|
||||
(**it).setReal(x2);
|
||||
(**it).setImag(y2);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate8_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
int pos = 0;
|
||||
|
||||
while (pos < len)
|
||||
{
|
||||
Sample s0(buf[pos+0] << 4, buf[pos+1] << 4);
|
||||
pos += 2;
|
||||
|
||||
if (m_decimator2.workDecimateCenter(&s0))
|
||||
{
|
||||
Sample s1 = s0;
|
||||
|
||||
if (m_decimator4.workDecimateCenter(&s1))
|
||||
{
|
||||
Sample s2 = s1;
|
||||
|
||||
if (m_decimator8.workDecimateCenter(&s2))
|
||||
{
|
||||
(**it) = s2;
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}*/
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate16_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
for (int pos = 0; pos < len - 31; pos += 32) {
|
||||
Sample s1(buf[pos+0] << 4, buf[pos+1] << 4);
|
||||
Sample s2(buf[pos+2] << 4, buf[pos+3] << 4);
|
||||
Sample s3(buf[pos+4] << 4, buf[pos+5] << 4);
|
||||
Sample s4(buf[pos+6] << 4, buf[pos+7] << 4);
|
||||
Sample s5(buf[pos+8] << 4, buf[pos+9] << 4);
|
||||
Sample s6(buf[pos+10] << 4, buf[pos+11] << 4);
|
||||
Sample s7(buf[pos+12] << 4, buf[pos+13] << 4);
|
||||
Sample s8(buf[pos+14] << 4, buf[pos+15] << 4);
|
||||
Sample s9(buf[pos+16] << 4, buf[pos+17] << 4);
|
||||
Sample s10(buf[pos+18] << 4, buf[pos+19] << 4);
|
||||
Sample s11(buf[pos+20] << 4, buf[pos+21] << 4);
|
||||
Sample s12(buf[pos+22] << 4, buf[pos+23] << 4);
|
||||
Sample s13(buf[pos+24] << 4, buf[pos+25] << 4);
|
||||
Sample s14(buf[pos+26] << 4, buf[pos+27] << 4);
|
||||
Sample s15(buf[pos+28] << 4, buf[pos+29] << 4);
|
||||
Sample s16(buf[pos+30] << 4, buf[pos+31] << 4);
|
||||
int pos = 0;
|
||||
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
m_decimator2.myDecimate(&s3, &s4);
|
||||
m_decimator2.myDecimate(&s5, &s6);
|
||||
m_decimator2.myDecimate(&s7, &s8);
|
||||
m_decimator2.myDecimate(&s9, &s10);
|
||||
m_decimator2.myDecimate(&s11, &s12);
|
||||
m_decimator2.myDecimate(&s13, &s14);
|
||||
m_decimator2.myDecimate(&s15, &s16);
|
||||
while (pos < len)
|
||||
{
|
||||
qint16 x0 = buf[pos+0] << 4;
|
||||
qint16 y0 = buf[pos+1] << 4;
|
||||
pos += 2;
|
||||
|
||||
m_decimator4.myDecimate(&s2, &s4);
|
||||
m_decimator4.myDecimate(&s6, &s8);
|
||||
m_decimator4.myDecimate(&s10, &s12);
|
||||
m_decimator4.myDecimate(&s14, &s16);
|
||||
if (m_decimator2.workDecimateCenter(&x0, &y0))
|
||||
{
|
||||
qint16 x1 = x0;
|
||||
qint16 y1 = y0;
|
||||
|
||||
m_decimator8.myDecimate(&s4, &s8);
|
||||
m_decimator8.myDecimate(&s12, &s16);
|
||||
if (m_decimator4.workDecimateCenter(&x1, &y1))
|
||||
{
|
||||
qint16 x2 = x1;
|
||||
qint16 y2 = y1;
|
||||
|
||||
m_decimator16.myDecimate(&s8, &s16);
|
||||
if (m_decimator8.workDecimateCenter(&x2, &y2))
|
||||
{
|
||||
qint16 x3 = x2;
|
||||
qint16 y3 = y2;
|
||||
|
||||
**it = s16;
|
||||
(*it)++;
|
||||
if (m_decimator16.workDecimateCenter(&x3, &y3))
|
||||
{
|
||||
(**it).setReal(x3);
|
||||
(**it).setImag(y3);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate32_cen(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
for (int pos = 0; pos < len - 63; pos += 64) {
|
||||
Sample s1(buf[pos+0] << 4, buf[pos+1] << 4);
|
||||
Sample s2(buf[pos+2] << 4, buf[pos+3] << 4);
|
||||
Sample s3(buf[pos+4] << 4, buf[pos+5] << 4);
|
||||
Sample s4(buf[pos+6] << 4, buf[pos+7] << 4);
|
||||
Sample s5(buf[pos+8] << 4, buf[pos+9] << 4);
|
||||
Sample s6(buf[pos+10] << 4, buf[pos+11] << 4);
|
||||
Sample s7(buf[pos+12] << 4, buf[pos+13] << 4);
|
||||
Sample s8(buf[pos+14] << 4, buf[pos+15] << 4);
|
||||
Sample s9(buf[pos+16] << 4, buf[pos+17] << 4);
|
||||
Sample s10(buf[pos+18] << 4, buf[pos+19] << 4);
|
||||
Sample s11(buf[pos+20] << 4, buf[pos+21] << 4);
|
||||
Sample s12(buf[pos+22] << 4, buf[pos+23] << 4);
|
||||
Sample s13(buf[pos+24] << 4, buf[pos+25] << 4);
|
||||
Sample s14(buf[pos+26] << 4, buf[pos+27] << 4);
|
||||
Sample s15(buf[pos+28] << 4, buf[pos+29] << 4);
|
||||
Sample s16(buf[pos+30] << 4, buf[pos+31] << 4);
|
||||
Sample s17(buf[pos+32] << 4, buf[pos+33] << 4);
|
||||
Sample s18(buf[pos+34] << 4, buf[pos+35] << 4);
|
||||
Sample s19(buf[pos+36] << 4, buf[pos+37] << 4);
|
||||
Sample s20(buf[pos+38] << 4, buf[pos+39] << 4);
|
||||
Sample s21(buf[pos+40] << 4, buf[pos+41] << 4);
|
||||
Sample s22(buf[pos+42] << 4, buf[pos+43] << 4);
|
||||
Sample s23(buf[pos+44] << 4, buf[pos+45] << 4);
|
||||
Sample s24(buf[pos+46] << 4, buf[pos+47] << 4);
|
||||
Sample s25(buf[pos+48] << 4, buf[pos+49] << 4);
|
||||
Sample s26(buf[pos+50] << 4, buf[pos+51] << 4);
|
||||
Sample s27(buf[pos+52] << 4, buf[pos+53] << 4);
|
||||
Sample s28(buf[pos+54] << 4, buf[pos+55] << 4);
|
||||
Sample s29(buf[pos+56] << 4, buf[pos+57] << 4);
|
||||
Sample s30(buf[pos+58] << 4, buf[pos+59] << 4);
|
||||
Sample s31(buf[pos+60] << 4, buf[pos+61] << 4);
|
||||
Sample s32(buf[pos+62] << 4, buf[pos+63] << 4);
|
||||
int pos = 0;
|
||||
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
m_decimator2.myDecimate(&s3, &s4);
|
||||
m_decimator2.myDecimate(&s5, &s6);
|
||||
m_decimator2.myDecimate(&s7, &s8);
|
||||
m_decimator2.myDecimate(&s9, &s10);
|
||||
m_decimator2.myDecimate(&s11, &s12);
|
||||
m_decimator2.myDecimate(&s13, &s14);
|
||||
m_decimator2.myDecimate(&s15, &s16);
|
||||
m_decimator2.myDecimate(&s17, &s18);
|
||||
m_decimator2.myDecimate(&s19, &s20);
|
||||
m_decimator2.myDecimate(&s21, &s22);
|
||||
m_decimator2.myDecimate(&s23, &s24);
|
||||
m_decimator2.myDecimate(&s25, &s26);
|
||||
m_decimator2.myDecimate(&s27, &s28);
|
||||
m_decimator2.myDecimate(&s29, &s30);
|
||||
m_decimator2.myDecimate(&s31, &s32);
|
||||
while (pos < len)
|
||||
{
|
||||
qint16 x0 = buf[pos+0] << 4;
|
||||
qint16 y0 = buf[pos+1] << 4;
|
||||
pos += 2;
|
||||
|
||||
m_decimator4.myDecimate(&s2, &s4);
|
||||
m_decimator4.myDecimate(&s6, &s8);
|
||||
m_decimator4.myDecimate(&s10, &s12);
|
||||
m_decimator4.myDecimate(&s14, &s16);
|
||||
m_decimator4.myDecimate(&s18, &s20);
|
||||
m_decimator4.myDecimate(&s22, &s24);
|
||||
m_decimator4.myDecimate(&s26, &s28);
|
||||
m_decimator4.myDecimate(&s30, &s32);
|
||||
if (m_decimator2.workDecimateCenter(&x0, &y0))
|
||||
{
|
||||
qint16 x1 = x0;
|
||||
qint16 y1 = y0;
|
||||
|
||||
m_decimator8.myDecimate(&s4, &s8);
|
||||
m_decimator8.myDecimate(&s12, &s16);
|
||||
m_decimator8.myDecimate(&s20, &s24);
|
||||
m_decimator8.myDecimate(&s28, &s32);
|
||||
if (m_decimator4.workDecimateCenter(&x1, &y1))
|
||||
{
|
||||
qint16 x2 = x1;
|
||||
qint16 y2 = y1;
|
||||
|
||||
m_decimator16.myDecimate(&s8, &s16);
|
||||
m_decimator16.myDecimate(&s24, &s32);
|
||||
if (m_decimator8.workDecimateCenter(&x2, &y2))
|
||||
{
|
||||
qint16 x3 = x2;
|
||||
qint16 y3 = y2;
|
||||
|
||||
m_decimator32.myDecimate(&s16, &s32);
|
||||
if (m_decimator16.workDecimateCenter(&x3, &y3))
|
||||
{
|
||||
qint16 x4 = x3;
|
||||
qint16 y4 = y3;
|
||||
|
||||
**it = s32;
|
||||
(*it)++;
|
||||
if (m_decimator32.workDecimateCenter(&x4, &y4))
|
||||
{
|
||||
(**it).setReal(x4);
|
||||
(**it).setImag(y4);
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -297,12 +374,18 @@ template<typename T>
|
||||
void Decimators<T>::decimate4(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 7; pos += 8) {
|
||||
|
||||
for (int pos = 0; pos < len - 7; pos += 8)
|
||||
{
|
||||
xreal = buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4];
|
||||
yimag = buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6];
|
||||
|
||||
(**it).setReal(xreal << 2);
|
||||
(**it).setImag(yimag << 2);
|
||||
/*
|
||||
Sample s( xreal << 2, yimag << 2 ); // was shift 3
|
||||
**it = s;
|
||||
(*it)++;
|
||||
**it = s;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -316,52 +399,72 @@ void Decimators<T>::decimate4_sup(SampleVector::iterator* it, const T* buf, qint
|
||||
// x y x y x y x y / x -> 0,-3,-4,7 / y -> 1,2,-5,-6
|
||||
// [ rotate: 0, 1, -3, 2, -4, -5, 7, -6]
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 7; pos += 8) {
|
||||
|
||||
for (int pos = 0; pos < len - 7; pos += 8)
|
||||
{
|
||||
xreal = buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6];
|
||||
yimag = - buf[pos+0] - buf[pos+3] + buf[pos+4] + buf[pos+7];
|
||||
//xreal = buf[pos+0] - buf[pos+3] - buf[pos+4] + buf[pos+7];
|
||||
//yimag = buf[pos+1] + buf[pos+2] - buf[pos+5] - buf[pos+6];
|
||||
|
||||
(**it).setReal(xreal << 2);
|
||||
(**it).setImag(yimag << 2);
|
||||
/*
|
||||
Sample s( xreal << 2, yimag << 2 ); // was shift 3
|
||||
**it = s;
|
||||
(*it)++;
|
||||
**it = s;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate8(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 15; pos += 8) {
|
||||
xreal = buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4];
|
||||
yimag = buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6];
|
||||
Sample s1( xreal << 2, yimag << 2 ); // was shift 3
|
||||
pos += 8;
|
||||
xreal = buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4];
|
||||
yimag = buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6];
|
||||
Sample s2( xreal << 2, yimag << 2 ); // was shift 3
|
||||
qint16 xreal[2], yimag[2];
|
||||
|
||||
for (int pos = 0; pos < len - 15; pos += 8)
|
||||
{
|
||||
xreal[0] = (buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4]) << 2;
|
||||
yimag[0] = (buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6]) << 2;
|
||||
//Sample s1( xreal << 2, yimag << 2 ); // was shift 3
|
||||
pos += 8;
|
||||
xreal[1] = (buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4]) << 2;
|
||||
yimag[1] = (buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6]) << 2;
|
||||
//Sample s2( xreal << 2, yimag << 2 ); // was shift 3
|
||||
|
||||
m_decimator2.myDecimate(xreal[0], yimag[0], &xreal[1], &yimag[1]);
|
||||
|
||||
(**it).setReal(xreal[1]);
|
||||
(**it).setImag(yimag[1]);
|
||||
/*
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
**it = s2;
|
||||
(*it)++;
|
||||
**it = s2;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void Decimators<T>::decimate8_sup(SampleVector::iterator* it, const T* buf, qint32 len)
|
||||
{
|
||||
qint16 xreal, yimag;
|
||||
for (int pos = 0; pos < len - 15; pos += 8) {
|
||||
xreal = buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6];
|
||||
yimag = - buf[pos+0] - buf[pos+3] + buf[pos+4] + buf[pos+7];
|
||||
Sample s1( xreal << 2, yimag << 2 ); // was shift 3
|
||||
pos += 8;
|
||||
xreal = buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6];
|
||||
yimag = - buf[pos+0] - buf[pos+3] + buf[pos+4] + buf[pos+7];
|
||||
Sample s2( xreal << 2, yimag << 2 ); // was shift 3
|
||||
qint16 xreal[2], yimag[2];
|
||||
|
||||
for (int pos = 0; pos < len - 15; pos += 8)
|
||||
{
|
||||
xreal[0] = (buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6]) << 2;
|
||||
yimag[0] = (- buf[pos+0] - buf[pos+3] + buf[pos+4] + buf[pos+7]) << 2;
|
||||
//Sample s1( xreal << 2, yimag << 2 ); // was shift 3
|
||||
pos += 8;
|
||||
xreal[1] = (buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6]) << 2;
|
||||
yimag[1] = (- buf[pos+0] - buf[pos+3] + buf[pos+4] + buf[pos+7]) << 2;
|
||||
//Sample s2( xreal << 2, yimag << 2 ); // was shift 3
|
||||
|
||||
m_decimator2.myDecimate(xreal[0], yimag[0], &xreal[1], &yimag[1]);
|
||||
|
||||
(**it).setReal(xreal[1]);
|
||||
(**it).setImag(yimag[1]);
|
||||
/*
|
||||
m_decimator2.myDecimate(&s1, &s2);
|
||||
**it = s2;
|
||||
(*it)++;
|
||||
**it = s2;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -372,13 +475,23 @@ void Decimators<T>::decimate16(SampleVector::iterator* it, const T* buf, qint32
|
||||
// downsample 4x more. [ rotate: 0, 1, -3, 2, -4, -5, 7, -6]
|
||||
qint16 xreal[4], yimag[4];
|
||||
|
||||
for (int pos = 0; pos < len - 31; ) {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
for (int pos = 0; pos < len - 31; )
|
||||
{
|
||||
for (int i = 0; i < 4; i++)
|
||||
{
|
||||
xreal[i] = (buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4]) << 2; // was shift 4
|
||||
yimag[i] = (buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6]) << 2; // was shift 4
|
||||
pos += 8;
|
||||
}
|
||||
|
||||
m_decimator2.myDecimate(xreal[0], yimag[0], &xreal[1], &yimag[1]);
|
||||
m_decimator2.myDecimate(xreal[2], yimag[2], &xreal[3], &yimag[3]);
|
||||
m_decimator4.myDecimate(xreal[1], yimag[1], &xreal[3], &yimag[3]);
|
||||
|
||||
(**it).setReal(xreal[3]);
|
||||
(**it).setImag(yimag[3]);
|
||||
|
||||
/* Valgrind optim
|
||||
Sample s1( xreal[0], yimag[0] );
|
||||
Sample s2( xreal[1], yimag[1] );
|
||||
Sample s3( xreal[2], yimag[2] );
|
||||
@ -389,8 +502,8 @@ void Decimators<T>::decimate16(SampleVector::iterator* it, const T* buf, qint32
|
||||
|
||||
m_decimator4.myDecimate(&s2, &s4);
|
||||
|
||||
**it = s4;
|
||||
(*it)++;
|
||||
**it = s4;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -401,13 +514,23 @@ void Decimators<T>::decimate16_sup(SampleVector::iterator* it, const T* buf, qin
|
||||
// downsample 4x more. [ rotate: 1, 0, -2, 3, -5, -4, 6, -7]
|
||||
qint16 xreal[4], yimag[4];
|
||||
|
||||
for (int pos = 0; pos < len - 31; ) {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
for (int pos = 0; pos < len - 31; )
|
||||
{
|
||||
for (int i = 0; i < 4; i++)
|
||||
{
|
||||
xreal[i] = (buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6]) << 2; // was shift 4
|
||||
yimag[i] = (buf[pos+4] + buf[pos+7] - buf[pos+0] - buf[pos+3]) << 2; // was shift 4
|
||||
pos += 8;
|
||||
}
|
||||
|
||||
m_decimator2.myDecimate(xreal[0], yimag[0], &xreal[1], &yimag[1]);
|
||||
m_decimator2.myDecimate(xreal[2], yimag[2], &xreal[3], &yimag[3]);
|
||||
|
||||
m_decimator4.myDecimate(xreal[1], yimag[1], &xreal[3], &yimag[3]);
|
||||
|
||||
(**it).setReal(xreal[3]);
|
||||
(**it).setImag(yimag[3]);
|
||||
/*
|
||||
Sample s1( xreal[0], yimag[0] );
|
||||
Sample s2( xreal[1], yimag[1] );
|
||||
Sample s3( xreal[2], yimag[2] );
|
||||
@ -418,8 +541,8 @@ void Decimators<T>::decimate16_sup(SampleVector::iterator* it, const T* buf, qin
|
||||
|
||||
m_decimator4.myDecimate(&s2, &s4);
|
||||
|
||||
**it = s4;
|
||||
(*it)++;
|
||||
**it = s4;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -428,13 +551,28 @@ void Decimators<T>::decimate32(SampleVector::iterator* it, const T* buf, qint32
|
||||
{
|
||||
qint16 xreal[8], yimag[8];
|
||||
|
||||
for (int pos = 0; pos < len - 63; ) {
|
||||
for (int i = 0; i < 8; i++) {
|
||||
for (int pos = 0; pos < len - 63; )
|
||||
{
|
||||
for (int i = 0; i < 8; i++)
|
||||
{
|
||||
xreal[i] = (buf[pos+0] - buf[pos+3] + buf[pos+7] - buf[pos+4]) << 2;
|
||||
yimag[i] = (buf[pos+1] - buf[pos+5] + buf[pos+2] - buf[pos+6]) << 2;
|
||||
pos += 8;
|
||||
}
|
||||
|
||||
m_decimator2.myDecimate(xreal[0], yimag[0], &xreal[1], &yimag[1]);
|
||||
m_decimator2.myDecimate(xreal[2], yimag[2], &xreal[3], &yimag[3]);
|
||||
m_decimator2.myDecimate(xreal[4], yimag[4], &xreal[5], &yimag[3]);
|
||||
m_decimator2.myDecimate(xreal[6], yimag[6], &xreal[7], &yimag[7]);
|
||||
|
||||
m_decimator4.myDecimate(xreal[1], yimag[1], &xreal[3], &yimag[3]);
|
||||
m_decimator4.myDecimate(xreal[5], yimag[5], &xreal[7], &yimag[7]);
|
||||
|
||||
m_decimator8.myDecimate(xreal[3], yimag[3], &xreal[7], &yimag[7]);
|
||||
|
||||
(**it).setReal(xreal[7]);
|
||||
(**it).setImag(yimag[7]);
|
||||
/*
|
||||
Sample s1( xreal[0], yimag[0] );
|
||||
Sample s2( xreal[1], yimag[1] );
|
||||
Sample s3( xreal[2], yimag[2] );
|
||||
@ -454,8 +592,8 @@ void Decimators<T>::decimate32(SampleVector::iterator* it, const T* buf, qint32
|
||||
|
||||
m_decimator8.myDecimate(&s4, &s8);
|
||||
|
||||
**it = s8;
|
||||
(*it)++;
|
||||
**it = s8;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
@ -464,13 +602,28 @@ void Decimators<T>::decimate32_sup(SampleVector::iterator* it, const T* buf, qin
|
||||
{
|
||||
qint16 xreal[8], yimag[8];
|
||||
|
||||
for (int pos = 0; pos < len - 63; ) {
|
||||
for (int i = 0; i < 8; i++) {
|
||||
for (int pos = 0; pos < len - 63; )
|
||||
{
|
||||
for (int i = 0; i < 8; i++)
|
||||
{
|
||||
xreal[i] = (buf[pos+1] - buf[pos+2] - buf[pos+5] + buf[pos+6]) << 2;
|
||||
yimag[i] = (buf[pos+4] + buf[pos+7] - buf[pos+0] - buf[pos+3]) << 2;
|
||||
pos += 8;
|
||||
}
|
||||
|
||||
m_decimator2.myDecimate(xreal[0], yimag[0], &xreal[1], &yimag[1]);
|
||||
m_decimator2.myDecimate(xreal[2], yimag[2], &xreal[3], &yimag[3]);
|
||||
m_decimator2.myDecimate(xreal[4], yimag[4], &xreal[5], &yimag[3]);
|
||||
m_decimator2.myDecimate(xreal[6], yimag[6], &xreal[7], &yimag[7]);
|
||||
|
||||
m_decimator4.myDecimate(xreal[1], yimag[1], &xreal[3], &yimag[3]);
|
||||
m_decimator4.myDecimate(xreal[5], yimag[5], &xreal[7], &yimag[7]);
|
||||
|
||||
m_decimator8.myDecimate(xreal[3], yimag[3], &xreal[7], &yimag[7]);
|
||||
|
||||
(**it).setReal(xreal[7]);
|
||||
(**it).setImag(yimag[7]);
|
||||
/*
|
||||
Sample s1( xreal[0], yimag[0] );
|
||||
Sample s2( xreal[1], yimag[1] );
|
||||
Sample s3( xreal[2], yimag[2] );
|
||||
@ -490,8 +643,8 @@ void Decimators<T>::decimate32_sup(SampleVector::iterator* it, const T* buf, qin
|
||||
|
||||
m_decimator8.myDecimate(&s4, &s8);
|
||||
|
||||
**it = s8;
|
||||
(*it)++;
|
||||
**it = s8;*/
|
||||
++(*it);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -56,27 +56,43 @@ public:
|
||||
Type acc = 0;
|
||||
int a = m_ptr;
|
||||
int b = a - 1;
|
||||
int i;
|
||||
int i, n_taps, size;
|
||||
|
||||
m_samples[m_ptr] = sample;
|
||||
size = m_samples.size(); // Valgrind optim (2)
|
||||
|
||||
while(b < 0)
|
||||
b += m_samples.size();
|
||||
{
|
||||
b += size;
|
||||
}
|
||||
|
||||
for(i = 0; i < (int)m_taps.size() - 1; i++) {
|
||||
n_taps = m_taps.size() - 1; // Valgrind optim
|
||||
|
||||
for (i = 0; i < n_taps; i++)
|
||||
{
|
||||
acc += (m_samples[a] + m_samples[b]) * m_taps[i];
|
||||
a++;
|
||||
while(a >= (int)m_samples.size())
|
||||
a -= m_samples.size();
|
||||
b--;
|
||||
while(b < 0)
|
||||
b += m_samples.size();
|
||||
}
|
||||
acc += m_samples[a] * m_taps[i];
|
||||
|
||||
while (a >= size)
|
||||
{
|
||||
a -= size;
|
||||
}
|
||||
|
||||
b--;
|
||||
|
||||
while (b < 0)
|
||||
{
|
||||
b += size;
|
||||
}
|
||||
}
|
||||
|
||||
acc += m_samples[a] * m_taps[i];
|
||||
m_ptr++;
|
||||
while(m_ptr >= (int)m_samples.size())
|
||||
m_ptr -= m_samples.size();
|
||||
|
||||
while (m_ptr >= size)
|
||||
{
|
||||
m_ptr -= size;
|
||||
}
|
||||
|
||||
return acc;
|
||||
}
|
||||
|
@ -24,7 +24,8 @@ public:
|
||||
m_samples[m_ptr][0] = sample->real();
|
||||
m_samples[m_ptr][1] = sample->imag();
|
||||
|
||||
switch(m_state) {
|
||||
switch(m_state)
|
||||
{
|
||||
case 0:
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
@ -50,10 +51,44 @@ public:
|
||||
}
|
||||
}
|
||||
|
||||
bool workDecimateCenter(qint16 *x, qint16 *y)
|
||||
{
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = *x;
|
||||
m_samples[m_ptr][1] = *y;
|
||||
|
||||
switch(m_state)
|
||||
{
|
||||
case 0:
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 1;
|
||||
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
|
||||
default:
|
||||
// save result
|
||||
doFIR(x, y);
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 0;
|
||||
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
// downsample by 2, return edges of spectrum rotated into center
|
||||
bool workDecimateFullRotate(Sample* sample)
|
||||
{
|
||||
switch(m_state) {
|
||||
switch(m_state)
|
||||
{
|
||||
case 0:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->real();
|
||||
@ -90,140 +125,142 @@ public:
|
||||
// downsample by 2, return lower half of original spectrum
|
||||
bool workDecimateLowerHalf(Sample* sample)
|
||||
{
|
||||
switch(m_state) {
|
||||
case 0:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->imag();
|
||||
m_samples[m_ptr][1] = sample->real();
|
||||
switch(m_state)
|
||||
{
|
||||
case 0:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->imag();
|
||||
m_samples[m_ptr][1] = sample->real();
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 1;
|
||||
// next state
|
||||
m_state = 1;
|
||||
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
|
||||
case 1:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->real();
|
||||
m_samples[m_ptr][1] = -sample->imag();
|
||||
case 1:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->real();
|
||||
m_samples[m_ptr][1] = -sample->imag();
|
||||
|
||||
// save result
|
||||
doFIR(sample);
|
||||
// save result
|
||||
doFIR(sample);
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 2;
|
||||
// next state
|
||||
m_state = 2;
|
||||
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
|
||||
case 2:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->imag();
|
||||
m_samples[m_ptr][1] = -sample->real();
|
||||
case 2:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->imag();
|
||||
m_samples[m_ptr][1] = -sample->real();
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 3;
|
||||
// next state
|
||||
m_state = 3;
|
||||
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
|
||||
default:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->real();
|
||||
m_samples[m_ptr][1] = sample->imag();
|
||||
default:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->real();
|
||||
m_samples[m_ptr][1] = sample->imag();
|
||||
|
||||
// save result
|
||||
doFIR(sample);
|
||||
// save result
|
||||
doFIR(sample);
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 0;
|
||||
// next state
|
||||
m_state = 0;
|
||||
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
}
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
// downsample by 2, return upper half of original spectrum
|
||||
bool workDecimateUpperHalf(Sample* sample)
|
||||
{
|
||||
switch(m_state) {
|
||||
case 0:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->imag();
|
||||
m_samples[m_ptr][1] = -sample->real();
|
||||
switch(m_state)
|
||||
{
|
||||
case 0:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->imag();
|
||||
m_samples[m_ptr][1] = -sample->real();
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 1;
|
||||
// next state
|
||||
m_state = 1;
|
||||
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
|
||||
case 1:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->real();
|
||||
m_samples[m_ptr][1] = -sample->imag();
|
||||
case 1:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->real();
|
||||
m_samples[m_ptr][1] = -sample->imag();
|
||||
|
||||
// save result
|
||||
doFIR(sample);
|
||||
// save result
|
||||
doFIR(sample);
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 2;
|
||||
// next state
|
||||
m_state = 2;
|
||||
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
|
||||
case 2:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->imag();
|
||||
m_samples[m_ptr][1] = sample->real();
|
||||
case 2:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = -sample->imag();
|
||||
m_samples[m_ptr][1] = sample->real();
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 3;
|
||||
// next state
|
||||
m_state = 3;
|
||||
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
// tell caller we don't have a new sample
|
||||
return false;
|
||||
|
||||
default:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->real();
|
||||
m_samples[m_ptr][1] = sample->imag();
|
||||
default:
|
||||
// insert sample into ring-buffer
|
||||
m_samples[m_ptr][0] = sample->real();
|
||||
m_samples[m_ptr][1] = sample->imag();
|
||||
|
||||
// save result
|
||||
doFIR(sample);
|
||||
// save result
|
||||
doFIR(sample);
|
||||
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
// advance write-pointer
|
||||
m_ptr = (m_ptr + HB_FILTERORDER) % (HB_FILTERORDER + 1);
|
||||
|
||||
// next state
|
||||
m_state = 0;
|
||||
// next state
|
||||
m_state = 0;
|
||||
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
}
|
||||
// tell caller we have a new sample
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
void myDecimate(Sample* sample1, Sample* sample2)
|
||||
void myDecimate(const Sample* sample1, Sample* sample2)
|
||||
{
|
||||
static const qint16 mod33[38] = { 31,32,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
|
||||
20,21,22,23,24,25,26,27,28,29,30,31,32,0,1,2};
|
||||
@ -240,8 +277,25 @@ public:
|
||||
m_ptr = mod33[m_ptr + 2 - 1];
|
||||
}
|
||||
|
||||
void myDecimate(qint16 x1, qint16 y1, qint16 *x2, qint16 *y2)
|
||||
{
|
||||
static const qint16 mod33[38] = { 31,32,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
|
||||
20,21,22,23,24,25,26,27,28,29,30,31,32,0,1,2};
|
||||
|
||||
m_samples[m_ptr][0] = x1;
|
||||
m_samples[m_ptr][1] = y1;
|
||||
m_ptr = mod33[m_ptr + 2 - 1];
|
||||
|
||||
m_samples[m_ptr][0] = *x2;
|
||||
m_samples[m_ptr][1] = *y2;
|
||||
|
||||
doFIR(x2, y2);
|
||||
|
||||
m_ptr = mod33[m_ptr + 2 - 1];
|
||||
}
|
||||
|
||||
protected:
|
||||
qint16 m_samples[HB_FILTERORDER + 1][2];
|
||||
qint32 m_samples[HB_FILTERORDER + 1][2]; // Valgrind optim (from qint16)
|
||||
qint16 m_ptr;
|
||||
int m_state;
|
||||
|
||||
@ -330,6 +384,93 @@ protected:
|
||||
sample->setReal(iAcc >> HB_SHIFT);
|
||||
sample->setImag(qAcc >> HB_SHIFT);
|
||||
}
|
||||
|
||||
void doFIR(qint16 *x, qint16 *y)
|
||||
{
|
||||
// coefficents
|
||||
|
||||
#if HB_FILTERORDER == 64
|
||||
static const qint32 COEFF[16] = {
|
||||
-0.001114417441601693505720538368564120901 * (1 << HB_SHIFT),
|
||||
0.001268007827185253051302527005361753254 * (1 << HB_SHIFT),
|
||||
-0.001959831378850490895410230152151598304 * (1 << HB_SHIFT),
|
||||
0.002878308307661380308073439948657323839 * (1 << HB_SHIFT),
|
||||
-0.004071361818258721100571850826099762344 * (1 << HB_SHIFT),
|
||||
0.005597288494657440618973431867289036745 * (1 << HB_SHIFT),
|
||||
-0.007532345003308904551886371336877346039 * (1 << HB_SHIFT),
|
||||
0.009980346844667375288961963519795972388 * (1 << HB_SHIFT),
|
||||
-0.013092614174300500062830820979797863401 * (1 << HB_SHIFT),
|
||||
0.01710934914871829748417297878404497169 * (1 << HB_SHIFT),
|
||||
-0.022443558692997273018576720460259821266 * (1 << HB_SHIFT),
|
||||
0.029875811511593811098386197500076377764 * (1 << HB_SHIFT),
|
||||
-0.041086352085710403647667021687084343284 * (1 << HB_SHIFT),
|
||||
0.060465467462665789533104998554335907102 * (1 << HB_SHIFT),
|
||||
-0.104159517495977321788203084906854201108 * (1 << HB_SHIFT),
|
||||
0.317657589850154464805598308885237202048 * (1 << HB_SHIFT),
|
||||
};
|
||||
#elif HB_FILTERORDER == 48
|
||||
static const qint32 COEFF[12] = {
|
||||
-0.004102576237611492253332112767338912818 * (1 << HB_SHIFT),
|
||||
0.003950551047979387886410762575906119309 * (1 << HB_SHIFT),
|
||||
-0.005807875789391703583164350277456833282 * (1 << HB_SHIFT),
|
||||
0.00823497890520805998770814682075069868 * (1 << HB_SHIFT),
|
||||
-0.011372226513199541059195851744334504474 * (1 << HB_SHIFT),
|
||||
0.015471557140973646315984524335362948477 * (1 << HB_SHIFT),
|
||||
-0.020944996398689276484450516591095947661 * (1 << HB_SHIFT),
|
||||
0.028568078132034283034279553703527199104 * (1 << HB_SHIFT),
|
||||
-0.040015143905614086738964374490024056286 * (1 << HB_SHIFT),
|
||||
0.059669519431831075095828964549582451582 * (1 << HB_SHIFT),
|
||||
-0.103669138691865420076609893840213771909 * (1 << HB_SHIFT),
|
||||
0.317491986549921390015072120149852707982 * (1 << HB_SHIFT)
|
||||
};
|
||||
#elif HB_FILTERORDER == 32
|
||||
static const qint16 mod33[38] = { 31,32,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
|
||||
20,21,22,23,24,25,26,27,28,29,30,31,32,0,1,2} ;
|
||||
static const qint32 COEFF[8] = {
|
||||
(qint32)(-0.015956912844043127236437484839370881673 * (1 << HB_SHIFT)),
|
||||
(qint32)( 0.013023031678944928940522274274371739011 * (1 << HB_SHIFT)),
|
||||
(qint32)(-0.01866942273717486777684371190844103694 * (1 << HB_SHIFT)),
|
||||
(qint32)( 0.026550887571157304190005987720724078827 * (1 << HB_SHIFT)),
|
||||
(qint32)(-0.038350314277854319344740474662103224546 * (1 << HB_SHIFT)),
|
||||
(qint32)( 0.058429248652825838128421764849917963147 * (1 << HB_SHIFT)),
|
||||
(qint32)(-0.102889802028955756885153505209018476307 * (1 << HB_SHIFT)),
|
||||
(qint32)( 0.317237706405931241260276465254719369113 * (1 << HB_SHIFT))
|
||||
};
|
||||
#else
|
||||
#error unsupported filter order
|
||||
#endif
|
||||
|
||||
|
||||
// init read-pointer
|
||||
int a = mod33[m_ptr + 2 + 1]; // 0 + 1
|
||||
int b = mod33[m_ptr + 2 - 2]; //-1 - 1
|
||||
|
||||
// go through samples in buffer
|
||||
qint32 iAcc = 0;
|
||||
qint32 qAcc = 0;
|
||||
|
||||
for (int i = 0; i < HB_FILTERORDER / 4; i++)
|
||||
{
|
||||
// do multiply-accumulate
|
||||
//qint32 iTmp = m_samples[a][0] + m_samples[b][0]; // Valgrind optim
|
||||
//qint32 qTmp = m_samples[a][1] + m_samples[b][1]; // Valgrind optim
|
||||
iAcc += (m_samples[a][0] + m_samples[b][0]) * COEFF[i];
|
||||
qAcc += (m_samples[a][1] + m_samples[b][1]) * COEFF[i];
|
||||
|
||||
// update read-pointer
|
||||
a = mod33[a + 2 + 2];
|
||||
b = mod33[b + 2 - 2];
|
||||
}
|
||||
|
||||
a = mod33[a + 2 - 1];
|
||||
|
||||
iAcc += ((qint32)m_samples[a][0] + 1) << (HB_SHIFT - 1);
|
||||
qAcc += ((qint32)m_samples[a][1] + 1) << (HB_SHIFT - 1);
|
||||
|
||||
*x = iAcc >> HB_SHIFT;
|
||||
*y = qAcc >> HB_SHIFT;
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
#endif // INCLUDE_INTHALFBANDFILTER_H
|
||||
|
@ -54,42 +54,42 @@ public:
|
||||
Type acc = 0;
|
||||
int a = m_ptr;
|
||||
int b = a - 1;
|
||||
int i, size;
|
||||
int i, n_taps, size;
|
||||
|
||||
m_samples[m_ptr] = sample;
|
||||
size = m_samples.size(); // Valgrind optim (2)
|
||||
|
||||
while (b < 0)
|
||||
{
|
||||
b += m_samples.size();
|
||||
b += size;
|
||||
}
|
||||
|
||||
size = (int) m_taps.size() - 1; // Valgrind optim
|
||||
n_taps = m_taps.size() - 1; // Valgrind optim
|
||||
|
||||
for (i = 0; i < size; i++)
|
||||
for (i = 0; i < n_taps; i++)
|
||||
{
|
||||
acc += (m_samples[a] + m_samples[b]) * m_taps[i];
|
||||
a++;
|
||||
|
||||
while (a >= (int)m_samples.size())
|
||||
while (a >= size)
|
||||
{
|
||||
a -= m_samples.size();
|
||||
a -= size;
|
||||
}
|
||||
|
||||
b--;
|
||||
|
||||
while(b < 0)
|
||||
{
|
||||
b += m_samples.size();
|
||||
b += size;
|
||||
}
|
||||
}
|
||||
|
||||
acc += m_samples[a] * m_taps[i];
|
||||
|
||||
m_ptr++;
|
||||
|
||||
while(m_ptr >= (int)m_samples.size())
|
||||
while(m_ptr >= size)
|
||||
{
|
||||
m_ptr -= m_samples.size();
|
||||
m_ptr -= size;
|
||||
}
|
||||
|
||||
return acc;
|
||||
|
@ -28,21 +28,22 @@ typedef std::complex<Real> Complex;
|
||||
typedef qint16 FixReal;
|
||||
|
||||
#pragma pack(push, 1)
|
||||
struct Sample {
|
||||
struct Sample
|
||||
{
|
||||
Sample() {}
|
||||
Sample(FixReal real) : m_real(real), m_imag(0) {}
|
||||
Sample(FixReal real, FixReal imag) : m_real(real), m_imag(imag) {}
|
||||
Sample(const Sample& other) : m_real(other.m_real), m_imag(other.m_imag) {}
|
||||
Sample& operator=(const Sample& other) { m_real = other.m_real; m_imag = other.m_imag; return *this; }
|
||||
inline Sample& operator=(const Sample& other) { m_real = other.m_real; m_imag = other.m_imag; return *this; }
|
||||
|
||||
Sample& operator+=(const Sample& other) { m_real += other.m_real; m_imag += other.m_imag; return *this; }
|
||||
Sample& operator-=(const Sample& other) { m_real -= other.m_real; m_imag -= other.m_imag; return *this; }
|
||||
inline Sample& operator+=(const Sample& other) { m_real += other.m_real; m_imag += other.m_imag; return *this; }
|
||||
inline Sample& operator-=(const Sample& other) { m_real -= other.m_real; m_imag -= other.m_imag; return *this; }
|
||||
|
||||
void setReal(FixReal v) { m_real = v; }
|
||||
void setImag(FixReal v) { m_imag = v; }
|
||||
inline void setReal(FixReal v) { m_real = v; }
|
||||
inline void setImag(FixReal v) { m_imag = v; }
|
||||
|
||||
FixReal real() const { return m_real; }
|
||||
FixReal imag() const { return m_imag; }
|
||||
inline FixReal real() const { return m_real; }
|
||||
inline FixReal imag() const { return m_imag; }
|
||||
|
||||
FixReal m_real;
|
||||
FixReal m_imag;
|
||||
|
Loading…
Reference in New Issue
Block a user