summaryrefslogtreecommitdiff
path: root/test/ui_pipe_waterfall.c
diff options
context:
space:
mode:
authorFreeArtMan <dos21h@gmail.com>2016-01-16 21:05:25 +0000
committerFreeArtMan <dos21h@gmail.com>2016-01-16 21:05:25 +0000
commit9b95088bddcf1f83e3a8f73f08f49b38ecb0f500 (patch)
tree44aaacaf005768d783b63d13f5ffd640c47eb0c4 /test/ui_pipe_waterfall.c
parent891ba007a455590ac544bbebd270ebe8d3417f2c (diff)
downloadradiola-9b95088bddcf1f83e3a8f73f08f49b38ecb0f500.tar.gz
radiola-9b95088bddcf1f83e3a8f73f08f49b38ecb0f500.zip
Added ui_pipe_waterfall outputing in comma seperated way waterfall. To easy use in shell pipe
Diffstat (limited to 'test/ui_pipe_waterfall.c')
-rw-r--r--test/ui_pipe_waterfall.c253
1 files changed, 253 insertions, 0 deletions
diff --git a/test/ui_pipe_waterfall.c b/test/ui_pipe_waterfall.c
new file mode 100644
index 0000000..a5f1769
--- /dev/null
+++ b/test/ui_pipe_waterfall.c
@@ -0,0 +1,253 @@
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <math.h>
+
+//radiola
+#include <hw/hw.h>
+#include <hw/sdr.h>
+
+#define SAMPLE_RATE 2048000
+
+#define CENTER_FREQ 446018750
+#define FFT_LEVEL 10
+#define FFT_SIZE (1 << FFT_LEVEL)
+#define SAMPLE_LENGHT (2 * FFT_SIZE)
+#define PRESCALE 8
+#define POSTSCALE 2
+#define WIDTH_POINTS 32
+
+int16_t* Sinewave;
+int N_WAVE, LOG2_N_WAVE;
+double* power_table;
+
+//better to have size size mod olen == 0
+int normalise( uint8_t *ibuf, int ilen, uint8_t *obuf, int olen )
+{
+ uint32_t i,j,m;
+ uint32_t ppi;
+ uint32_t sum;
+ div_t d;
+
+ if ( ilen >= olen )
+ {
+ d = div(ilen,olen);
+ ppi = d.quot;
+ if (d.rem > 0)
+ ppi += 1;
+ } else {
+ printf("Input buffer smaller ther output buffer\n");
+ return -1;
+ }
+
+ m = 0;
+ i = 0;
+ for (i=0,m=0; i < ilen+ppi; i += ppi, m++)
+ {
+ sum = 0;
+ for (j=0;j<ppi;j++)
+ sum += ibuf[i+j];
+ obuf[m] = sum/ppi;
+ }
+
+ return 0;
+}
+
+void sine_table(int size)
+{
+
+ int i;
+ double d;
+ LOG2_N_WAVE = size;
+ N_WAVE = 1 << LOG2_N_WAVE;
+ Sinewave = malloc(sizeof(int16_t) * N_WAVE*3/4);
+ power_table = malloc(sizeof(double) * N_WAVE);
+ for (i=0; i<N_WAVE*3/4; i++)
+ {
+ d = (double)i * 2.0 * M_PI / N_WAVE;
+ Sinewave[i] = (int)round(32767*sin(d));
+ }
+}
+
+int16_t FIX_MPY(int16_t a, int16_t b)
+/* fixed point multiply and scale */
+{
+ int c = ((int)a * (int)b) >> 14;
+ b = c & 0x01;
+ return (c >> 1) + b;
+}
+
+int32_t real_conj(int16_t real, int16_t imag)
+/* real(n * conj(n)) */
+{
+ return ((int32_t)real*(int32_t)real + (int32_t)imag*(int32_t)imag);
+}
+
+int fix_fft(int16_t iq[], int m)
+/* interleaved iq[], 0 <= n < 2**m, changes in place */
+{
+ int mr, nn, i, j, l, k, istep, n, shift;
+ int16_t qr, qi, tr, ti, wr, wi;
+ n = 1 << m;
+ if (n > N_WAVE)
+ {return -1;}
+ mr = 0;
+ nn = n - 1;
+ /* decimation in time - re-order data */
+ for (m=1; m<=nn; ++m) {
+ l = n;
+ do
+ {l >>= 1;}
+ while (mr+l > nn);
+ mr = (mr & (l-1)) + l;
+ if (mr <= m)
+ {continue;}
+ // real = 2*m, imag = 2*m+1
+ tr = iq[2*m];
+ iq[2*m] = iq[2*mr];
+ iq[2*mr] = tr;
+ ti = iq[2*m+1];
+ iq[2*m+1] = iq[2*mr+1];
+ iq[2*mr+1] = ti;
+ }
+ l = 1;
+ k = LOG2_N_WAVE-1;
+ while (l < n) {
+ shift = 1;
+ istep = l << 1;
+ for (m=0; m<l; ++m) {
+ j = m << k;
+ wr = Sinewave[j+N_WAVE/4];
+ wi = -Sinewave[j];
+ if (shift) {
+ wr >>= 1; wi >>= 1;}
+ for (i=m; i<n; i+=istep) {
+ j = i + l;
+ tr = FIX_MPY(wr,iq[2*j]) - FIX_MPY(wi,iq[2*j+1]);
+ ti = FIX_MPY(wr,iq[2*j+1]) + FIX_MPY(wi,iq[2*j]);
+ qr = iq[2*i];
+ qi = iq[2*i+1];
+ if (shift) {
+ qr >>= 1; qi >>= 1;}
+ iq[2*j] = qr - tr;
+ iq[2*j+1] = qi - ti;
+ iq[2*i] = qr + tr;
+ iq[2*i+1] = qi + ti;
+ }
+ }
+ --k;
+ l = istep;
+ }
+ return 0;
+}
+
+//fftize
+int simple_fft( uint8_t *buf, int len )
+{
+ int i,j;
+ uint16_t p;
+ uint16_t buf1[SAMPLE_LENGHT];
+ uint16_t buf2[SAMPLE_LENGHT];
+ int fft_len;
+
+ for (i=0; i<len; i++)
+ {
+ buf1[i] = buf[i] * PRESCALE;
+ }
+
+ fix_fft( (uint16_t *)buf1, FFT_LEVEL );
+
+ for (i=0; i<FFT_SIZE; i+=1)
+ {
+ //buf1[i] = rtl_out.buf[i];
+ //p = buf1[i] * buf1[i];
+ j = i*2;
+ p = (int16_t)real_conj(buf1[j], buf1[j + 1]);
+ buf2[i] = p;
+ }
+
+ fft_len = FFT_SIZE / 2;
+ for (i=0; i<fft_len; i++)
+ {
+ buf[i] = (int)log10(POSTSCALE * (float)buf2[i+fft_len]);
+ buf[i+fft_len] = (int)log10(POSTSCALE * (float)buf2[i]);
+ }
+
+ return 0;
+}
+
+int pipe_watefall( int size, uint8_t *buf )
+{
+ int i;
+ for (i=0;i<size;i++)
+ {
+ printf("%02d",(unsigned char)buf[i]);
+ if (i!=size-1) printf(",");
+ }
+ printf("\n");
+ return 0;
+}
+
+int main()
+{
+
+ int i,j;
+ uint8_t *buf, *sample_buf;
+ int buf_len, sample_len;
+ uint32_t dev_num;
+
+ sdr_t *sdr = NULL;
+ dongle_t *dongle = NULL;
+
+
+ if ( (sdr = sdr_init()) == NULL )
+ {
+ printf("Cannot init sdr manager\n");
+ goto main_exit;
+ }
+
+ if ( sdr_open_device( sdr, 0 ) != 0 )
+ {
+ printf("Cannot open dongle 0\n");
+ goto main_exit;
+ }
+ dongle = sdr_get_device_id( sdr, 0 );
+ dongle_set_freq( dongle, CENTER_FREQ );
+ dongle_set_sample_rate( dongle, SAMPLE_RATE );
+
+ sine_table( FFT_LEVEL );
+
+
+
+
+ buf_len = WIDTH_POINTS;
+ buf = malloc( buf_len );
+
+ sample_len = SAMPLE_LENGHT;
+ sample_buf = malloc( sample_len );
+
+ srand(0); //fake seed
+ for ( i=0; i<40;i++ )
+ {
+
+ //read some samples
+ dongle_read_samples( dongle, sample_buf, sample_len );
+
+ //do fft
+ simple_fft( sample_buf, sample_len );
+
+ //prepare to show on the screen
+ if (normalise( sample_buf, sample_len/2, buf, buf_len ) == -1)
+ {
+ printf("Cannot normalise\n");
+ }
+ pipe_watefall( buf_len, buf );
+ usleep(1000000);
+ }
+
+main_exit:
+ //close gui, restore terminal mode
+ sdr_close( sdr );
+
+ return 0;
+}