///////////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2012 maintech GmbH, Otto-Hahn-Str. 15, 97204 Hoechberg, Germany //
// written by Christian Daniel                                                   //
//                                                                               //
// This program is free software; you can redistribute it and/or modify          //
// it under the terms of the GNU General Public License as published by          //
// the Free Software Foundation as version 3 of the License, or                  //
//                                                                               //
// This program is distributed in the hope that it will be useful,               //
// but WITHOUT ANY WARRANTY; without even the implied warranty of                //
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
// GNU General Public License V3 for more details.                               //
//                                                                               //
// You should have received a copy of the GNU General Public License             //
// along with this program. If not, see <http://www.gnu.org/licenses/>.          //
///////////////////////////////////////////////////////////////////////////////////

#include <stdio.h>
#include <errno.h>
#include "v4lthread.h"
#include "dsp/samplefifo.h"

#define BLOCKSIZE 16384

V4LThread::V4LThread(SampleFifo* sampleFifo, QObject* parent) :
	QThread(parent),
	m_running(false),
	m_dev(1),
	m_convertBuffer(BLOCKSIZE),
	m_sampleFifo(sampleFifo)
{
	m_samplerate = 2500000;
}

V4LThread::~V4LThread()
{
	stopWork();
}

void V4LThread::startWork()
{
	m_startWaitMutex.lock();
	start();
	while(!m_running)
		m_startWaiter.wait(&m_startWaitMutex, 100);
	m_startWaitMutex.unlock();
}

void V4LThread::stopWork()
{
	m_running = false;
	wait();
}

void V4LThread::setSamplerate(int samplerate)
{
	m_samplerate = samplerate;
}

void V4LThread::run()
{
	int res;

	m_running = true;
	m_startWaiter.wakeAll();
/*
	while(m_running) {
		if((res = rtlsdr_read_async(m_dev, &V4LThread::callbackHelper, this, 32, BLOCKSIZE)) < 0) {
			qCritical("V4LThread: async error: %s", strerror(errno));
			break;
		}
	}
*/
	m_running = false;
}

void V4LThread::decimate2(SampleVector::iterator* it, const quint8* buf, qint32 len)
{
	qint16 xreal, yimag;
	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)++;
		xreal = buf[pos+7] - buf[pos+4];
		yimag = 255 - buf[pos+5] - buf[pos+6];
		Sample t( xreal << 3, yimag << 3 );
		**it = t;
		(*it)++;
	}
}
void V4LThread::decimate4(SampleVector::iterator* it, const quint8* buf, qint32 len)
{
	qint16 xreal, yimag;
		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];
		Sample s( xreal << 3, yimag << 3 );
		**it = s;
		(*it)++;
	}
}

void V4LThread::decimate8(SampleVector::iterator* it, const quint8* 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];
		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 s( xreal << 3, yimag << 3 );
		**it = s;
		(*it)++;
	}
}

void V4LThread::decimate16(SampleVector::iterator* it, const quint8* buf, qint32 len)
{
	// Offset tuning: 4x downsample and rotate, then
	// downsample 4x more. [ rotate:  0, 1, -3, 2, -4, -5, 7, -6]
	qint16 xreal, yimag;
	for (int step = 0; step < len - 31; step +=32) {
		xreal = yimag = 0;
		for (int pos = step; pos < step + 32; 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 s( xreal << 3, yimag << 3 );
		**it = s;
		(*it)++;
	}
}

void V4LThread::callback(const quint8* buf, qint32 len)
{
	qint16 xreal, yimag, phase;
	SampleVector::iterator it = m_convertBuffer.begin();

	switch(4) {
		case 0: // 1:1 = no decimation
			// just rotation
			phase = -(1<<2);
			for (int pos = 0; pos < len + 3; pos += 4) {
				phase *= -1;
				xreal = phase * (2 * buf[pos+0] - 255);
				yimag = phase * (2 * buf[pos+1] - 255);
				*it++ = Sample(xreal, yimag);
				xreal = phase * (255 - 2 * buf[pos+3]);
				yimag = phase * (2 * buf[pos+2] - 255);
				*it++ = Sample(xreal, yimag);
			}
			break;
		case 1: // 1:2
			decimate2(&it, buf, len);
			break;

		case 2: // 1:4
			decimate4(&it, buf, len);
			break;

		case 3: // 1:8
			decimate8(&it, buf, len);
			break;

		default:
		case 4: // 1:16
			decimate16(&it, buf, len);
			break;
	}

	m_sampleFifo->write(m_convertBuffer.begin(), it);
/*
	if(!m_running)
		rtlsdr_cancel_async(m_dev);
*/
}

void V4LThread::callbackHelper(unsigned char* buf, quint32 len, void* ctx)
{
	V4LThread* thread = (V4LThread*)ctx;
	thread->callback(buf, len);
}