mirror of https://github.com/EspoTek/Labrador.git
429 lines
13 KiB
C++
429 lines
13 KiB
C++
#include "genericusbdriver.h"
|
|
|
|
#ifndef LIBRADOR_LIBRARY
|
|
#include "platformspecific.h"
|
|
#endif
|
|
|
|
genericUsbDriver::genericUsbDriver(QWidget *parent) : QLabel(parent)
|
|
{
|
|
connectedStatus(false);
|
|
qDebug() << "Making USB Driver invisible!!";
|
|
this->hide();
|
|
|
|
//Double buffers are used to send the transfers to isoDriver. outBuffers and bufferLengths store the actual data from each transfer as well as length. They are read by isoDriver when it calls isoRead().
|
|
outBuffers[0] = (unsigned char *) calloc(ISO_PACKET_SIZE*ISO_PACKETS_PER_CTX*NUM_ISO_ENDPOINTS + 8, 1);
|
|
outBuffers[1] = (unsigned char *) calloc(ISO_PACKET_SIZE*ISO_PACKETS_PER_CTX*NUM_ISO_ENDPOINTS + 8, 1);
|
|
bufferLengths[0] = 0;
|
|
bufferLengths[1] = 0;
|
|
|
|
for(int k=0; k<NUM_ISO_ENDPOINTS; k++){
|
|
//qDebug() << k;
|
|
//qDebug() << NUM_ISO_ENDPOINTS;
|
|
pipeID[k] = 0x81+k;
|
|
qDebug() << "pipeID" << k << "=" << pipeID[k];
|
|
}
|
|
|
|
connectTimer = new QTimer();
|
|
connectTimer->setTimerType(Qt::PreciseTimer);
|
|
connectTimer->start(USB_RECONNECT_PERIOD);
|
|
connect(connectTimer, SIGNAL(timeout()), this, SLOT(checkConnection()));
|
|
qDebug()<< "Generic Usb Driver setup complete";
|
|
}
|
|
|
|
genericUsbDriver::~genericUsbDriver(void){
|
|
qDebug() << "genericUsbDriver dectructor entering";
|
|
if(connected){
|
|
psuTimer->stop();
|
|
recoveryTimer->stop();
|
|
isoTimer->stop();
|
|
delete(psuTimer);
|
|
delete(recoveryTimer);
|
|
delete(isoTimer);
|
|
}
|
|
qDebug() << "genericUsbDriver dectructor completed";
|
|
}
|
|
|
|
|
|
void genericUsbDriver::setPsu(double voltage){
|
|
|
|
qDebug() << "New voltage =" << voltage;
|
|
currentPsuVoltage = voltage;
|
|
|
|
//if(deviceMode > 6) qFatal("setPsu is not configured for mode 7!!!");
|
|
double vinp = (voltage - psu_offset)/11;
|
|
double vinn = 0;
|
|
//double vref = 1.65;
|
|
double gainPsu = 1;
|
|
|
|
dutyPsu = (int) ((vinp - vinn)/vref * gainPsu * PSU_ADC_TOP);
|
|
|
|
qDebug() << "Going to send value " << dutyPsu;
|
|
}
|
|
|
|
void genericUsbDriver::setFunctionGen(int channel, functionGenControl *fGenControl){
|
|
////////////////////////////
|
|
////NO RESIZING (YET)!!!////
|
|
////////////////////////////
|
|
|
|
//////////////////////////////////////
|
|
//// CH1 is AUX!! CH2 is "MAIN"!!////
|
|
//////////////////////////////////////
|
|
|
|
int length, maxLength, numDivides, maxDivides;
|
|
double freq, amplitude, offset;
|
|
unsigned char *samples;
|
|
|
|
//For recalling on crash.
|
|
if (channel == 0) fGenPtr_CH1 = fGenControl;
|
|
else fGenPtr_CH2 = fGenControl;
|
|
|
|
//Reading in data
|
|
if (channel == 0){
|
|
length = fGenControl->length_CH1;
|
|
freq = fGenControl->freq_CH1;
|
|
amplitude = fGenControl->amplitude_CH1;
|
|
offset = fGenControl->offset_CH1;
|
|
samples = (unsigned char *) malloc(length);
|
|
memcpy(samples, fGenControl->samples_CH1, (unsigned int) length);
|
|
numDivides = fGenControl->divisibility_CH1;
|
|
}
|
|
else if(channel == 1){
|
|
length = fGenControl->length_CH2;
|
|
freq = fGenControl->freq_CH2;
|
|
amplitude = fGenControl->amplitude_CH2;
|
|
offset = fGenControl->offset_CH2;
|
|
samples = (unsigned char *) malloc(length);
|
|
memcpy(samples, fGenControl->samples_CH2, (unsigned int) length);
|
|
numDivides = fGenControl->divisibility_CH2;
|
|
}
|
|
|
|
//Triple mode
|
|
if ((amplitude+offset) > FGEN_LIMIT){
|
|
amplitude = amplitude / 3;
|
|
offset = offset / 3;
|
|
fGenTriple |= ((unsigned char) !channel + 1);
|
|
}
|
|
else fGenTriple &= ((unsigned char) (254 - !channel));
|
|
|
|
//qDebug() << "fGenTriple = " << fGenTriple << "fGenControl = " << fGenControl << "length = " << length << "freq = " << freq << "amplitude = " << amplitude << "offset = " << offset << "samples = " << samples;
|
|
|
|
//Waveform scaling in V
|
|
double tempDouble;
|
|
amplitude = (amplitude * 255) / FGEN_LIMIT;
|
|
offset = (offset * 255) / FGEN_LIMIT;
|
|
if (offset<FGEN_OFFSET){
|
|
if (amplitude>5)
|
|
amplitude -= FGEN_OFFSET;
|
|
else
|
|
amplitude = 0;
|
|
offset = FGEN_OFFSET;
|
|
}
|
|
|
|
#ifdef INVERT_TRIPLE
|
|
unsigned char fGenTemp = 0;
|
|
fGenTemp |= (fGenTriple & 0x01)<<1;
|
|
fGenTemp |= (fGenTriple & 0x02)>>1;
|
|
usbSendControl(0x40, 0xa4, fGenTemp, 0, 0, NULL);
|
|
#else
|
|
usbSendControl(0x40, 0xa4, fGenTriple, 0, 0, NULL);
|
|
#endif
|
|
|
|
//Applying amplitude and offset to all samples.
|
|
for (int i=0;i<length;i++){
|
|
tempDouble = (double) samples[i];
|
|
tempDouble *= amplitude;
|
|
tempDouble /= 255;
|
|
tempDouble += offset;
|
|
samples[i] = (unsigned char) tempDouble;
|
|
}
|
|
|
|
|
|
//Need to increase size of wave if its freq too high, or too low!
|
|
maxDivides = numDivides;
|
|
bool loop_entered = false;
|
|
unsigned char *tempSamples = (unsigned char *) malloc(0);
|
|
|
|
while(length * freq > DAC_SPS){
|
|
loop_entered = true;
|
|
numDivides--;
|
|
if (numDivides==0){
|
|
qDebug("numDivides = 0 - in T-stretching of genericUsbDriver:: setFunctionGen");
|
|
}
|
|
|
|
int shiftTemp = (maxDivides - numDivides);
|
|
length = length >> 1;
|
|
|
|
free(tempSamples);
|
|
tempSamples = (unsigned char *) malloc(length);
|
|
for (int i=0; i<length;i++){
|
|
tempSamples[i] = samples[i<<shiftTemp];
|
|
}
|
|
}
|
|
if(loop_entered){
|
|
samples = tempSamples;
|
|
}
|
|
|
|
//Scaling in t here:
|
|
// Something something maxLength something
|
|
|
|
//Timer Setup
|
|
int validClockDivs[7] = {1, 2, 4, 8, 64, 256, 1024};
|
|
|
|
int clkSetting;
|
|
for(clkSetting = 0; clkSetting<7; clkSetting++){
|
|
if ( (CLOCK_FREQ / (length * validClockDivs[clkSetting] * freq)) < 65535 )
|
|
break;
|
|
}
|
|
int timerPeriod = CLOCK_FREQ / (length * freq * validClockDivs[clkSetting]);
|
|
|
|
clkSetting++; // Change from [0:n] to [1:n]
|
|
|
|
if(deviceMode == 5){
|
|
qDebug("DEVICE IS IN MODE 5");
|
|
}
|
|
|
|
//qDebug() << "First three samples:" << samples[0] << samples[1] << samples[2];
|
|
|
|
if(channel){
|
|
usbSendControl(0x40, 0xa1, timerPeriod, clkSetting, length, samples);
|
|
}
|
|
else usbSendControl(0x40, 0xa2, timerPeriod, clkSetting, length, samples);
|
|
free(tempSamples);
|
|
return;
|
|
}
|
|
|
|
void genericUsbDriver::newDig(int digState){
|
|
qDebug() << "newDig";
|
|
digitalPinState = digState;
|
|
usbSendControl(0x40, 0xa6, digState, 0, 0, NULL);
|
|
}
|
|
|
|
/*
|
|
void genericUsbDriver::setBufferPtr(bufferControl *newPtr){
|
|
bufferPtr = newPtr;
|
|
}
|
|
*/
|
|
|
|
void genericUsbDriver::setDeviceMode(int mode){
|
|
int oldMode = deviceMode;
|
|
deviceMode = mode;
|
|
usbSendControl(0x40, 0xa5, (mode == 5 ? 0 : mode), gainMask, 0, NULL);
|
|
|
|
if (fGenPtr_CH1!=NULL) setFunctionGen(0, fGenPtr_CH1);
|
|
if (fGenPtr_CH2!=NULL) setFunctionGen(1, fGenPtr_CH2);
|
|
|
|
//switch on new deviceMode!!
|
|
switch(deviceMode){
|
|
case 0:
|
|
if(oldMode > 2) sendClearBuffer(1,0,0);
|
|
setVisible_CH2(0);
|
|
checkXY(0);
|
|
break;
|
|
case 1:
|
|
if(oldMode > 2) sendClearBuffer(1,0,0);
|
|
sendClearBuffer(0,1,0);
|
|
setVisible_CH2(1);
|
|
checkXY(0);
|
|
break;
|
|
case 2:
|
|
if(oldMode > 2) sendClearBuffer(1,0,0);
|
|
sendClearBuffer(0,1,0);
|
|
setVisible_CH2(1);
|
|
break;
|
|
case 3:
|
|
if(oldMode != 4) sendClearBuffer(1,0,0);
|
|
sendClearBuffer(0,1,0);
|
|
setVisible_CH2(0);
|
|
checkXY(0);
|
|
break;
|
|
case 4:
|
|
if(oldMode != 3) sendClearBuffer(1,0,0);
|
|
sendClearBuffer(0,1,0);
|
|
setVisible_CH2(1);
|
|
checkXY(0);
|
|
break;
|
|
case 5:
|
|
setVisible_CH2(0);
|
|
checkXY(0);
|
|
break;
|
|
case 6:
|
|
sendClearBuffer(0,0,1);
|
|
setVisible_CH2(0);
|
|
checkXY(0);
|
|
break;
|
|
case 7:
|
|
sendClearBuffer(1,0,0);
|
|
enableMMTimer();
|
|
checkXY(0);
|
|
break;
|
|
default:
|
|
qFatal("Error in genericUsbDriver::setDeviceMode. Invalid device mode.");
|
|
}
|
|
|
|
}
|
|
|
|
void genericUsbDriver::psuTick(){
|
|
if(dutyTemp == dutyPsu) return;
|
|
|
|
if (dutyTemp > dutyPsu){
|
|
if((dutyTemp - dutyPsu)> PSU_STEP){
|
|
dutyTemp -= PSU_STEP;
|
|
} else dutyTemp = dutyPsu;
|
|
}
|
|
|
|
if (dutyTemp < dutyPsu){
|
|
if((dutyPsu - dutyTemp)> PSU_STEP){
|
|
dutyTemp += PSU_STEP;
|
|
} else dutyTemp = dutyPsu;
|
|
}
|
|
|
|
if ((dutyTemp>106) || (dutyTemp<21)){
|
|
qDebug("PSU DUTY CYCLE of dutyTemp = %d OUT OF RANGE (could underflow on SOF)!!! ABORTING!!!", dutyTemp);
|
|
}
|
|
usbSendControl(0x40, 0xa3, dutyTemp, 0, 0, NULL);
|
|
}
|
|
|
|
void genericUsbDriver::setGain(double newGain){
|
|
if (newGain == scopeGain) return; //No update!
|
|
gainBuffers(scopeGain/newGain);
|
|
scopeGain = newGain;
|
|
//See XMEGA_AU Manual, page 359. ADC.CTRL.GAIN.
|
|
if(newGain==0.5) gainMask = 0x07;
|
|
else if (newGain == 1) gainMask = 0x00;
|
|
else if (newGain == 2) gainMask = 0x01;
|
|
else if (newGain == 4) gainMask = 0x02;
|
|
else if (newGain == 8) gainMask = 0x03;
|
|
else if (newGain == 16) gainMask = 0x04;
|
|
else if (newGain == 32) gainMask = 0x05;
|
|
else if (newGain == 64) gainMask = 0x06;
|
|
else qFatal("genericUsbDriver::setGain attempted to set invalid gain value");
|
|
gainMask = gainMask << 2;
|
|
gainMask |= (gainMask << 8);
|
|
/*
|
|
* This bit had to be removed because Android doesn't like log2()
|
|
* if (newGain == 0.5){
|
|
gainMask = 7<<2 | 7<<10;
|
|
}
|
|
else gainMask = (unsigned short)(log2(newGain))<<2 | (unsigned short)(log2(newGain))<<10;
|
|
*/
|
|
qDebug("newGain = %f", newGain);
|
|
qDebug("gainMask = %x", gainMask);
|
|
usbSendControl(0x40, 0xa5, deviceMode, gainMask, 0, NULL);
|
|
}
|
|
|
|
void genericUsbDriver::avrDebug(void){
|
|
usbSendControl(0xc0, 0xa0, 0, 0, sizeof(unified_debug), NULL);
|
|
|
|
qDebug() << "unified debug is of size" << sizeof(unified_debug);
|
|
/*
|
|
#ifndef PLATFORM_ANDROID
|
|
unified_debug *udsPtr = (unified_debug *) inBuffer;
|
|
uint16_t trfcnt0 = (udsPtr->trfcntH0 << 8) + udsPtr->trfcntL0;
|
|
uint16_t trfcnt1 = (udsPtr->trfcntH1 << 8) + udsPtr->trfcntL1;
|
|
uint16_t medianTrfcnt = (udsPtr->medianTrfcntH << 8) + udsPtr->medianTrfcntL;
|
|
uint16_t outOfRange = (udsPtr->outOfRangeH << 8) + udsPtr->outOfRangeL;
|
|
uint16_t counter = (udsPtr->counterH << 8) + udsPtr->counterL;
|
|
uint16_t dma_ch0_cnt = (udsPtr->dma_ch0_cntH << 8) + udsPtr->dma_ch0_cntL;
|
|
uint16_t dma_ch1_cnt = (udsPtr->dma_ch1_cntH << 8) + udsPtr->dma_ch1_cntL;
|
|
|
|
qDebug("%s", udsPtr->header);
|
|
qDebug() << "trfcnt0 =" << trfcnt0;
|
|
qDebug() << "trfcnt1 =" << trfcnt1;
|
|
qDebug() << "medianTrfcnt =" << medianTrfcnt;
|
|
qDebug() << "outOfRange = " << outOfRange;
|
|
qDebug() << "counter = " << counter;
|
|
qDebug() << "calValNeg = " << udsPtr->calValNeg;
|
|
qDebug() << "calValPos = " << udsPtr->calValPos; qDebug() << "CALA = " << udsPtr->CALA;
|
|
qDebug() << "CALB = " << udsPtr->CALB;
|
|
qDebug() << "dma_ch0_cnt = " << dma_ch0_cnt;
|
|
qDebug() << "dma_ch1_cnt = " << dma_ch1_cnt;
|
|
#endif
|
|
*/
|
|
}
|
|
|
|
void genericUsbDriver::kickstartIso(void){
|
|
qDebug() << "Attempting to kickstart iso...";
|
|
usbSendControl(0x40, 0xaa, 0, 0, 0, NULL);
|
|
}
|
|
|
|
void genericUsbDriver::requestFirmwareVersion(void){
|
|
usbSendControl(0xc0, 0xa8, 0, 0, 2, NULL);
|
|
firmver = *((unsigned short *) inBuffer);
|
|
}
|
|
|
|
void genericUsbDriver::requestFirmwareVariant(void){
|
|
usbSendControl(0xc0, 0xa9, 0, 0, 1, NULL);
|
|
variant = *((unsigned char *) inBuffer);
|
|
}
|
|
|
|
|
|
void genericUsbDriver::saveState(int *_out_deviceMode, double *_out_scopeGain, double *_out_currentPsuVoltage, int *_out_digitalPinState){
|
|
*(_out_deviceMode) = deviceMode;
|
|
*(_out_scopeGain) = scopeGain;
|
|
*(_out_currentPsuVoltage) = currentPsuVoltage;
|
|
*(_out_digitalPinState) = digitalPinState;
|
|
return;
|
|
}
|
|
|
|
void genericUsbDriver::checkConnection(){
|
|
//This will connect to the board, then wait one more period before actually starting the stack.
|
|
if(!connected){
|
|
connectedStatus(false);
|
|
qDebug() << "CHECKING CONNECTION!";
|
|
connected = !(usbInit(BOARD_VID, BOARD_PID));
|
|
qDebug() << "Connected";
|
|
return;
|
|
}
|
|
connectTimer->stop();
|
|
|
|
requestFirmwareVersion();
|
|
qDebug("BOARD IS RUNNING FIRMWARE VERSION 0x%04hx", firmver);
|
|
qDebug("EXPECTING FIRMWARE VERSION 0x%04hx", EXPECTED_FIRMWARE_VERSION);
|
|
requestFirmwareVariant();
|
|
qDebug("FIRMWARE VARIANT = 0x%02hx", variant);
|
|
qDebug("EXPECTED VARIANT = 0x%02hx", DEFINED_EXPECTED_VARIANT);
|
|
|
|
if((firmver != EXPECTED_FIRMWARE_VERSION) || (variant != DEFINED_EXPECTED_VARIANT)){
|
|
qDebug() << "Unexpected Firmware!!";
|
|
int flashRet = flashFirmware();
|
|
connected = false;
|
|
connectTimer->start();
|
|
return;
|
|
}
|
|
|
|
qDebug() << "Connecting now!";
|
|
|
|
//This is the actual setup code.
|
|
connectTimer->stop();
|
|
delete(connectTimer);
|
|
|
|
connectedStatus(true);
|
|
|
|
|
|
setDeviceMode(deviceMode);
|
|
newDig(digitalPinState);
|
|
usbIsoInit();
|
|
|
|
psuTimer = new QTimer();
|
|
psuTimer->setTimerType(Qt::PreciseTimer);
|
|
psuTimer->start(PSU_PERIOD);
|
|
connect(psuTimer, SIGNAL(timeout()), this, SLOT(psuTick()));
|
|
|
|
if(killOnConnect) usbSendControl(0x40, 0xa7, 0, 0, 0, NULL);
|
|
|
|
recoveryTimer = new QTimer();
|
|
recoveryTimer->setTimerType(Qt::PreciseTimer);
|
|
recoveryTimer->start(RECOVERY_PERIOD);
|
|
connect(recoveryTimer, SIGNAL(timeout()), this, SLOT(recoveryTick()));
|
|
initialConnectComplete();
|
|
|
|
if(!killOnConnect && calibrateOnConnect){
|
|
calibrateMe();
|
|
}
|
|
}
|
|
|
|
void genericUsbDriver::bootloaderJump(){
|
|
usbSendControl(0x40, 0xa7, 1, 0, 0, NULL);
|
|
}
|
|
|