mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-23 16:38:37 -05:00
1131 lines
31 KiB
C++
1131 lines
31 KiB
C++
/**
|
|
@file MCU_BD.cpp
|
|
@author Lime Microsystems
|
|
@brief Implementation of interacting with on board MCU
|
|
*/
|
|
|
|
#include "MCU_BD.h"
|
|
using namespace std;
|
|
#include <string>
|
|
#include "MCU_File.h"
|
|
#include <sstream>
|
|
#include <fstream>
|
|
#include "LMS64CProtocol.h"
|
|
#include <assert.h>
|
|
#include <thread>
|
|
#include <list>
|
|
#include "ErrorReporting.h"
|
|
#include "LMS7002M.h"
|
|
|
|
using namespace lime;
|
|
|
|
MCU_BD::MCU_BD()
|
|
{
|
|
mLoadedProgramFilename = "";
|
|
m_bLoadedDebug = 0;
|
|
m_bLoadedProd = 0;
|
|
stepsTotal = 0;
|
|
stepsDone = 0;
|
|
aborted = false;
|
|
callback = nullptr;
|
|
mChipID =0;
|
|
//ctor
|
|
int i=0;
|
|
m_serPort=NULL;
|
|
//default value,
|
|
//must be verified during program exploatation
|
|
m_iLoopTries=20;
|
|
byte_array_size = cMaxFWSize;
|
|
// array initiallization
|
|
for (i=0; i<=255; i++){
|
|
m_SFR[i]=0x00;
|
|
m_IRAM[i]=0x00;
|
|
}
|
|
for (i=0; i<byte_array_size; i++){
|
|
byte_array[i]=0x00;
|
|
};
|
|
}
|
|
|
|
MCU_BD::~MCU_BD()
|
|
{
|
|
//dtor
|
|
}
|
|
|
|
void MCU_BD::Initialize(IConnection* pSerPort, unsigned chipID, unsigned size)
|
|
{
|
|
m_serPort = pSerPort;
|
|
mChipID = chipID;
|
|
if (size > 0)
|
|
byte_array_size = size;
|
|
}
|
|
|
|
/** @brief Read program code from file into memory
|
|
@param inFileName source file path
|
|
@param bin binary or hex file
|
|
@return 0:success, -1:file not found
|
|
*/
|
|
int MCU_BD:: GetProgramCode(const char* inFileName, bool bin)
|
|
{
|
|
unsigned char ch=0x00;
|
|
bool find_byte=false;
|
|
int i=0;
|
|
|
|
if(!bin)
|
|
{
|
|
MCU_File inFile(inFileName, "rb");
|
|
if (inFile.FileOpened() == false)
|
|
return -1;
|
|
|
|
mLoadedProgramFilename = inFileName;
|
|
try
|
|
{
|
|
inFile.ReadHex(byte_array_size-1);
|
|
}
|
|
catch (...)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
for (i=0; i<byte_array_size; i++)
|
|
{
|
|
find_byte=inFile.GetByte(i, ch);
|
|
if (find_byte==true)
|
|
byte_array[i]=ch;
|
|
else
|
|
byte_array[i]=0x00;
|
|
};
|
|
}
|
|
else
|
|
{
|
|
char inByte = 0;
|
|
fstream fin;
|
|
fin.open(inFileName, ios::in | ios::binary);
|
|
if (fin.good() == false)
|
|
return -1;
|
|
mLoadedProgramFilename = inFileName;
|
|
memset(byte_array, 0, byte_array_size);
|
|
for(int i=0; i<byte_array_size && !fin.eof(); ++i)
|
|
{
|
|
inByte = 0;
|
|
fin.read(&inByte, 1);
|
|
byte_array[i]=inByte;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void MCU_BD:: mSPI_write(
|
|
unsigned short addr_reg, // takes 16 bit address
|
|
unsigned short data_reg) // takes 16 bit value
|
|
{
|
|
if(m_serPort == nullptr)
|
|
return;
|
|
uint32_t wrdata = addr_reg << 16 | data_reg;
|
|
m_serPort->WriteLMS7002MSPI(&wrdata, 1, mChipID);
|
|
}
|
|
|
|
|
|
unsigned short MCU_BD:: mSPI_read(
|
|
unsigned short addr_reg) // takes 16 bit address
|
|
{// returns 16 bit value
|
|
if(m_serPort == nullptr)
|
|
return 0;
|
|
uint32_t wrdata = addr_reg << 16;
|
|
uint32_t rddata = 0;
|
|
if(m_serPort->ReadLMS7002MSPI(&wrdata, &rddata, 1, mChipID)!=0)
|
|
return 0;
|
|
|
|
return rddata & 0xFFFF;
|
|
}
|
|
|
|
int MCU_BD::WaitUntilWritten(){
|
|
|
|
// waits if WRITE_REQ (REG3[2]) flag is equal to '1'
|
|
// this means that write operation is in progress
|
|
unsigned short tempi=0x0000;
|
|
int countDown=m_iLoopTries; // Time out value
|
|
tempi=mSPI_read(0x0003); // REG3 read
|
|
|
|
while (( (tempi&0x0004) == 0x0004) && (countDown>0))
|
|
{
|
|
tempi=mSPI_read(0x0003); // REG3 read
|
|
countDown--;
|
|
}
|
|
if (countDown==0)
|
|
return -1; // an error occured, timer elapsed
|
|
else
|
|
return 0; // Finished regularly
|
|
// pass if WRITE_REQ is '0'
|
|
}
|
|
|
|
int MCU_BD::ReadOneByte(unsigned char * data)
|
|
{
|
|
unsigned short tempi=0x0000;
|
|
int countDown=m_iLoopTries;
|
|
|
|
// waits when READ_REQ (REG3[3]) flag is equal to '0'
|
|
// this means that there is nothing to read
|
|
tempi=mSPI_read(0x0003); // REG3 read
|
|
|
|
while (((tempi&0x0008)==0x0000) && (countDown>0))
|
|
{
|
|
// wait if READ_REQ is '0'
|
|
tempi=mSPI_read(0x0003); // REG3 read
|
|
countDown--;
|
|
}
|
|
|
|
if (countDown>0)
|
|
{ // Time out has not occured
|
|
tempi=mSPI_read(0x0005); // REG5 read
|
|
// return the read byte
|
|
(* data) = (unsigned char) (tempi);
|
|
}
|
|
else
|
|
(* data) =0;
|
|
// return the zero, default value
|
|
|
|
if (countDown==0)
|
|
return -1; // an error occured
|
|
else
|
|
return 0; // finished regularly
|
|
}
|
|
|
|
int MCU_BD::One_byte_command(unsigned short data1, unsigned char * rdata1)
|
|
{
|
|
unsigned char tempc=0x00;
|
|
int retval=0;
|
|
*rdata1=0x00; //default return value
|
|
|
|
// sends the one byte command
|
|
mSPI_write(0x8004, data1); //REG4 write
|
|
retval=WaitUntilWritten();
|
|
if (retval==-1) return -1;
|
|
// error if operation executes too long
|
|
|
|
// gets the one byte answer
|
|
retval=ReadOneByte(&tempc);
|
|
if (retval==-1) return -1;
|
|
// error if operation takes too long
|
|
|
|
*rdata1=tempc;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MCU_BD::Three_byte_command(
|
|
unsigned char data1,unsigned char data2,unsigned char data3,
|
|
unsigned char * rdata1,unsigned char * rdata2,unsigned char * rdata3){
|
|
|
|
|
|
int retval=0;
|
|
*rdata1=0x00;
|
|
*rdata2=0x00;
|
|
*rdata3=0x00;
|
|
|
|
mSPI_write(0x8004, (unsigned short)(data1)); //REG4 write
|
|
retval=WaitUntilWritten();
|
|
if (retval==-1) return -1;
|
|
|
|
mSPI_write(0x8004, (unsigned short)(data2)); //REG4 write
|
|
retval=WaitUntilWritten();
|
|
if (retval==-1) return -1;
|
|
|
|
mSPI_write(0x8004, (unsigned short)(data3)); //REG4 write
|
|
retval=WaitUntilWritten();
|
|
if (retval==-1) return -1;
|
|
|
|
retval= ReadOneByte(rdata1);
|
|
if (retval==-1) return -1;
|
|
|
|
retval= ReadOneByte(rdata2);
|
|
if (retval==-1) return -1;
|
|
|
|
retval= ReadOneByte(rdata3);
|
|
if (retval==-1) return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int MCU_BD::Change_MCUFrequency(unsigned char data) {
|
|
|
|
unsigned char tempc1, tempc2, tempc3=0x00;
|
|
int retval=0;
|
|
// code 0x7E is for writing the SFR registers
|
|
retval=Three_byte_command(0x7E, 0x8E, data, &tempc1, &tempc2, &tempc3);
|
|
// PMSR register, address 0x8E
|
|
return retval;
|
|
}
|
|
|
|
int MCU_BD::Read_IRAM()
|
|
{
|
|
unsigned char tempc1, tempc2, tempc3=0x00;
|
|
int i=0;
|
|
int retval=0;
|
|
|
|
//default
|
|
//IRAM array initialization
|
|
for (i=0; i<=255; i++)
|
|
m_IRAM[i]=0x00;
|
|
|
|
stepsTotal.store(256);
|
|
stepsDone.store(0);
|
|
aborted.store(false);
|
|
for (i=0; i<=255; i++)
|
|
{
|
|
// code 0x78 is for reading the IRAM locations
|
|
retval=Three_byte_command(0x78, ((unsigned char)(i)), 0x00,&tempc1, &tempc2, &tempc3);
|
|
if (retval==0)
|
|
m_IRAM[i]=tempc3;
|
|
else
|
|
{
|
|
i=256; // error, stop
|
|
aborted.store(true);
|
|
}
|
|
++stepsDone;
|
|
#ifndef NDEBUG
|
|
printf("MCU reading IRAM: %2i/256\r", stepsDone.load());
|
|
#endif
|
|
Wait_CLK_Cycles(64);
|
|
}
|
|
#ifndef NDEBUG
|
|
printf("\nMCU reading IRAM finished\n");
|
|
#endif
|
|
return retval;
|
|
}
|
|
|
|
int MCU_BD::Erase_IRAM()
|
|
{
|
|
unsigned char tempc1, tempc2, tempc3=0x00;
|
|
int retval=0;
|
|
int i=0;
|
|
|
|
//default ini.
|
|
for (i=0; i<=255; i++)
|
|
m_IRAM[i]=0x00;
|
|
|
|
stepsTotal.store(256);
|
|
stepsDone.store(0);
|
|
aborted.store(false);
|
|
for (i=0; i<=255; i++)
|
|
{
|
|
m_IRAM[i]=0x00;
|
|
// code 0x7C is for writing the IRAM locations
|
|
retval=Three_byte_command(0x7C, ((unsigned char)(i)), 0x00,&tempc1, &tempc2, &tempc3);
|
|
if (retval == -1)
|
|
{
|
|
i = 256;
|
|
aborted.store(true);
|
|
}
|
|
++stepsDone;
|
|
#ifndef NDEBUG
|
|
printf("MCU erasing IRAM: %2i/256\r", stepsDone.load());
|
|
#endif
|
|
}
|
|
#ifndef NDEBUG
|
|
printf("\nMCU erasing IRAM finished\n");
|
|
#endif
|
|
return retval;
|
|
}
|
|
|
|
int MCU_BD::Read_SFR()
|
|
{
|
|
int i=0;
|
|
unsigned char tempc1, tempc2, tempc3=0x00;
|
|
int retval=0;
|
|
|
|
stepsTotal.store(48);
|
|
stepsDone.store(0);
|
|
aborted.store(false);
|
|
|
|
//default m_SFR array initialization
|
|
for (i=0; i<=255; i++)
|
|
m_SFR[i]=0x00;
|
|
|
|
// code 0x7A is for reading the SFR registers
|
|
retval=Three_byte_command(0x7A, 0x80, 0x00, &tempc1, &tempc2, &tempc3); // P0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x80]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x81, 0x00, &tempc1, &tempc2, &tempc3); // SP
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x81]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x82, 0x00, &tempc1, &tempc2, &tempc3); // DPL0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x82]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x83, 0x00, &tempc1, &tempc2, &tempc3); // DPH0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x83]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x84, 0x00, &tempc1, &tempc2, &tempc3); // DPL1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x84]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x85, 0x00, &tempc1, &tempc2, &tempc3); // DPH1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x85]=tempc3;
|
|
|
|
stepsDone.store(6);
|
|
|
|
retval=Three_byte_command(0x7A, 0x86, 0x00, &tempc1, &tempc2, &tempc3); // DPS
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x86]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x87, 0x00, &tempc1, &tempc2, &tempc3); // PCON
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x87]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x88, 0x00, &tempc1, &tempc2, &tempc3); // TCON
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x88]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x89, 0x00, &tempc1, &tempc2, &tempc3); // TMOD
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x89]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x8A, 0x00, &tempc1, &tempc2, &tempc3); // TL0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x8A]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x8B, 0x00, &tempc1, &tempc2, &tempc3); // TL1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x8B]=tempc3;
|
|
|
|
stepsDone.store(12);
|
|
|
|
retval=Three_byte_command(0x7A, 0x8C, 0x00, &tempc1, &tempc2, &tempc3); // TH0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x8C]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x8D, 0x00, &tempc1, &tempc2, &tempc3); // TH1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x8D]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x8E, 0x00, &tempc1, &tempc2, &tempc3); // PMSR
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x8E]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x90, 0x00, &tempc1, &tempc2, &tempc3); // P1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x90]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x91, 0x00, &tempc1, &tempc2, &tempc3); // DIR1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x91]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0x98, 0x00, &tempc1, &tempc2, &tempc3); // SCON
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x98]=tempc3;
|
|
|
|
stepsDone.store(18);
|
|
|
|
retval=Three_byte_command(0x7A, 0x99, 0x00, &tempc1, &tempc2, &tempc3); // SBUF
|
|
if (retval==-1) return -1;
|
|
m_SFR[0x99]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xA0, 0x00, &tempc1, &tempc2, &tempc3); // P2
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xA0]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xA1, 0x00, &tempc1, &tempc2, &tempc3); // DIR2
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xA1]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xA2, 0x00, &tempc1, &tempc2, &tempc3); // DIR0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xA2]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xA8, 0x00, &tempc1, &tempc2, &tempc3); // IEN0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xA8]=tempc3;
|
|
|
|
stepsDone.store(24);
|
|
|
|
retval=Three_byte_command(0x7A, 0xA9, 0x00, &tempc1, &tempc2, &tempc3); // IEN1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xA9]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xB0, 0x00, &tempc1, &tempc2, &tempc3); // EECTRL
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xB0]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xB1, 0x00, &tempc1, &tempc2, &tempc3); // EEDATA
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xB1]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xB8, 0x00, &tempc1, &tempc2, &tempc3); // IP0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xB8]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xB9, 0x00, &tempc1, &tempc2, &tempc3); // IP1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xB9]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xBF, 0x00, &tempc1, &tempc2, &tempc3); // USR2
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xBF]=tempc3;
|
|
|
|
stepsDone.store(30);
|
|
|
|
retval=Three_byte_command(0x7A, 0xC0, 0x00, &tempc1, &tempc2, &tempc3); // IRCON
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xC0]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xC8, 0x00, &tempc1, &tempc2, &tempc3); // T2CON
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xC8]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xCA, 0x00, &tempc1, &tempc2, &tempc3); // RCAP2L
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xCA]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xCB, 0x00, &tempc1, &tempc2, &tempc3); // RCAP2H
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xCB]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xCC, 0x00, &tempc1, &tempc2, &tempc3); // TL2
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xCC]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xCD, 0x00, &tempc1, &tempc2, &tempc3); // TH2
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xCD]=tempc3;
|
|
|
|
stepsDone.store(36);
|
|
|
|
retval=Three_byte_command(0x7A, 0xD0, 0x00, &tempc1, &tempc2, &tempc3); // PSW
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xD0]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xE0, 0x00, &tempc1, &tempc2, &tempc3); // ACC
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xE0]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xF0, 0x00, &tempc1, &tempc2, &tempc3); // B
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xF0]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xEC, 0x00, &tempc1, &tempc2, &tempc3); // REG0
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xEC]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xED, 0x00, &tempc1, &tempc2, &tempc3); // REG1
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xED]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xEE, 0x00, &tempc1, &tempc2, &tempc3); // REG2
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xEE]=tempc3;
|
|
|
|
stepsDone.store(42);
|
|
|
|
retval=Three_byte_command(0x7A, 0xEF, 0x00, &tempc1, &tempc2, &tempc3); // REG3
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xEF]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xF4, 0x00, &tempc1, &tempc2, &tempc3); // REG4
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xF4]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xF5, 0x00, &tempc1, &tempc2, &tempc3); // REG5
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xF5]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xF6, 0x00, &tempc1, &tempc2, &tempc3); // REG6
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xF6]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xF7, 0x00, &tempc1, &tempc2, &tempc3); // REG7
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xF7]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xFC, 0x00, &tempc1, &tempc2, &tempc3); // REG8
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xFC]=tempc3;
|
|
|
|
retval=Three_byte_command(0x7A, 0xFD, 0x00, &tempc1, &tempc2, &tempc3); // REG9
|
|
if (retval==-1) return -1;
|
|
m_SFR[0xFD]=tempc3;
|
|
|
|
stepsDone.store(48);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void MCU_BD::Wait_CLK_Cycles(int delay)
|
|
{
|
|
//// some delay
|
|
int i=0;
|
|
for (i=0;i<(delay/64);i++)
|
|
mSPI_read(0x0003);
|
|
}
|
|
|
|
/** @brief Upload program code from memory into MCU
|
|
@return 0:success, -1:failed
|
|
*/
|
|
int MCU_BD::Program_MCU(int m_iMode1, int m_iMode0)
|
|
{
|
|
IConnection::MCU_PROG_MODE mode;
|
|
switch(m_iMode1 << 1 | m_iMode0)
|
|
{
|
|
case 0: mode = IConnection::MCU_PROG_MODE::RESET; break;
|
|
case 1: mode = IConnection::MCU_PROG_MODE::EEPROM_AND_SRAM; break;
|
|
case 2: mode = IConnection::MCU_PROG_MODE::SRAM; break;
|
|
case 3: mode = IConnection::MCU_PROG_MODE::BOOT_SRAM_FROM_EEPROM; break;
|
|
default: mode = IConnection::MCU_PROG_MODE::RESET; break;
|
|
}
|
|
return Program_MCU(byte_array,mode);
|
|
}
|
|
|
|
int MCU_BD::Program_MCU(const uint8_t* buffer, const IConnection::MCU_PROG_MODE mode)
|
|
{
|
|
if(!m_serPort)
|
|
return ReportError(ENOLINK, "Device not connected");
|
|
|
|
if (byte_array_size <= 8192)
|
|
return m_serPort->ProgramMCU(buffer, byte_array_size, mode, callback);
|
|
#ifndef NDEBUG
|
|
auto timeStart = std::chrono::high_resolution_clock::now();
|
|
#endif
|
|
const auto timeout = std::chrono::milliseconds(100);
|
|
const uint32_t controlAddr = 0x0002 << 16;
|
|
const uint32_t statusReg = 0x0003 << 16;
|
|
const uint32_t addrDTM = 0x0004 << 16; //data to MCU
|
|
const uint16_t EMTPY_WRITE_BUFF = 1 << 0;
|
|
const uint16_t PROGRAMMED = 1 << 6;
|
|
const uint8_t fifoLen = 64;
|
|
uint32_t wrdata[fifoLen];
|
|
uint32_t rddata = 0;
|
|
int status;
|
|
bool abort = false;
|
|
//reset MCU, set mode
|
|
wrdata[0] = controlAddr | 0;
|
|
wrdata[1] = controlAddr | (mode & 0x3);
|
|
if((status = m_serPort->WriteLMS7002MSPI(wrdata, 2, mChipID))!=0)
|
|
return status;
|
|
|
|
if(callback)
|
|
abort = callback(0, byte_array_size, "");
|
|
|
|
for(uint16_t i=0; i<byte_array_size && !abort; i+=fifoLen)
|
|
{
|
|
//wait till EMPTY_WRITE_BUFF = 1
|
|
bool fifoEmpty = false;
|
|
wrdata[0] = statusReg;
|
|
auto t1 = std::chrono::high_resolution_clock::now();
|
|
auto t2 = t1;
|
|
do{
|
|
if((status = m_serPort->ReadLMS7002MSPI(wrdata, &rddata, 1, mChipID))!=0)
|
|
return status;
|
|
fifoEmpty = rddata & EMTPY_WRITE_BUFF;
|
|
t2 = std::chrono::high_resolution_clock::now();
|
|
}while( (!fifoEmpty) && (t2-t1)<timeout);
|
|
|
|
if(!fifoEmpty)
|
|
return ReportError(ETIMEDOUT, "MCU FIFO full");
|
|
|
|
//write 32 bytes into FIFO
|
|
for(uint8_t j=0; j<fifoLen; ++j)
|
|
wrdata[j] = addrDTM | buffer[i+j];
|
|
if((status = m_serPort->WriteLMS7002MSPI(wrdata,fifoLen, mChipID))!=0)
|
|
return status;
|
|
if(callback)
|
|
abort = callback(i+fifoLen, byte_array_size, "");
|
|
#ifndef NDEBUG
|
|
printf("MCU programming : %4i/%4li\r", i+fifoLen, long(byte_array_size));
|
|
#endif
|
|
};
|
|
if(abort)
|
|
return ReportError(-1, "operation aborted by user");
|
|
|
|
//wait until programmed flag
|
|
wrdata[0] = statusReg;
|
|
bool programmed = false;
|
|
auto t1 = std::chrono::high_resolution_clock::now();
|
|
auto t2 = t1;
|
|
do{
|
|
if((status = m_serPort->ReadLMS7002MSPI(wrdata, &rddata, 1, mChipID))!=0)
|
|
return status;
|
|
programmed = rddata & PROGRAMMED;
|
|
t2 = std::chrono::high_resolution_clock::now();
|
|
}while( (!programmed) && (t2-t1)<timeout);
|
|
#ifndef NDEBUG
|
|
auto timeEnd = std::chrono::high_resolution_clock::now();
|
|
printf("\nMCU Programming finished, %li ms\n",
|
|
std::chrono::duration_cast<std::chrono::milliseconds>
|
|
(timeEnd-timeStart).count());
|
|
#endif
|
|
if(!programmed)
|
|
return ReportError(ETIMEDOUT, "MCU not programmed");
|
|
return 0;
|
|
}
|
|
|
|
void MCU_BD::Reset_MCU()
|
|
{
|
|
unsigned short tempi=0x0000; // was 0x0000
|
|
mSPI_write(0x8002, tempi);
|
|
tempi=0x0000;
|
|
mSPI_write(0x8000, tempi);
|
|
}
|
|
|
|
int MCU_BD::RunProductionTest_MCU()
|
|
{
|
|
string temps;
|
|
unsigned short tempi = 0x0080; // was 0x0000
|
|
int m_iMode1_ = 0;
|
|
int m_iMode0_ = 0;
|
|
|
|
if (m_bLoadedProd == 0)
|
|
{
|
|
if (GetProgramCode("lms7suite_mcu/ptest.hex", false) != 0)
|
|
return -1;
|
|
}
|
|
//MCU gets control over SPI switch
|
|
mSPI_write(0x0006, 0x0001); //REG6 write
|
|
|
|
// reset MCU
|
|
tempi = 0x0080;
|
|
mSPI_write(0x8002, tempi); // REG2
|
|
tempi = 0x0000;
|
|
mSPI_write(0x8000, tempi); // REG0
|
|
|
|
if (m_bLoadedProd == 0)
|
|
{
|
|
//select programming mode "01" for SRAM and EEPROM
|
|
m_iMode1_ = 0; m_iMode0_ = 1;
|
|
}
|
|
else
|
|
{
|
|
//boot from EEPROM
|
|
m_iMode1_ = 1; m_iMode0_ = 1;
|
|
}
|
|
|
|
//upload hex file
|
|
if (Program_MCU(m_iMode1_, m_iMode0_) != 0)
|
|
return -1; //failed to program
|
|
|
|
if (m_bLoadedProd == 0)
|
|
{
|
|
Wait_CLK_Cycles(256 * 100); // for programming mode, prog.code has been already loaded into MCU
|
|
m_iMode1_ = 0; m_iMode0_ = 1;
|
|
}
|
|
else
|
|
{
|
|
Wait_CLK_Cycles(256 * 400);
|
|
// for booting from EEPROM mode, must wait for some longer delay, at least 8kB/(80kb/s)=0.1s
|
|
m_iMode1_ = 1; m_iMode0_ = 1;
|
|
}
|
|
|
|
// global variable
|
|
m_bLoadedProd = 1; // the ptest.hex has been loaded
|
|
m_bLoadedDebug = 0;
|
|
|
|
//tempi = 0x0000;
|
|
// EXT_INT2=1, external interrupt 2 is raised
|
|
mSPI_write(0x8002, formREG2command(0, 0, 0, 1, m_iMode1_, m_iMode0_)); // EXT_INT2=1
|
|
// here you can put any Delay function
|
|
Wait_CLK_Cycles(256);
|
|
// EXT_INT2=0, external interrupt 2 is pulled down
|
|
mSPI_write(0x8002, formREG2command(0, 0, 0, 0, m_iMode1_, m_iMode0_)); // EXT_INT2=0
|
|
|
|
// wait for some time MCU to execute the tests
|
|
// the time is approximately 20ms
|
|
// here you can put any Delay function
|
|
Wait_CLK_Cycles(256 * 100);
|
|
|
|
unsigned short retval = 0;
|
|
retval = mSPI_read(1); //REG1 read
|
|
|
|
// show the return value at the MCU Control Panel
|
|
//int temps = wxString::Format("Result is: 0x%02X", retval);
|
|
//ReadResult->SetLabel(temps);
|
|
|
|
if (retval == 0x10)
|
|
{
|
|
tempi = 0x0055;
|
|
mSPI_write(0x8000, tempi); // P0=0x55;
|
|
// EXT_INT3=1, external interrupt 3 is raised
|
|
mSPI_write(0x8002, formREG2command(0, 0, 1, 0, m_iMode1_, m_iMode0_)); // EXT_INT3=1
|
|
// here you can put any Delay function
|
|
Wait_CLK_Cycles(256);
|
|
// EXT_INT3=0, external interrupt 3 is pulled down
|
|
mSPI_write(0x8002, formREG2command(0, 0, 0, 0, m_iMode1_, m_iMode0_)); // EXT_INT3=0
|
|
Wait_CLK_Cycles(256 * 5);
|
|
retval = mSPI_read(1); //REG1 read
|
|
if (retval != 0x55)
|
|
temps = "Ext. interrupt 3 test failed.";
|
|
else
|
|
{
|
|
tempi = 0x00AA;
|
|
mSPI_write(0x8000, tempi); // P0=0xAA;
|
|
// EXT_INT4=1, external interrupt 4 is raised
|
|
mSPI_write(0x8002, formREG2command(0, 1, 0, 0, m_iMode1_, m_iMode0_)); // EXT_INT4=1
|
|
// here you can put any Delay function
|
|
Wait_CLK_Cycles(256);
|
|
// EXT_INT4=0, external interrupt 4 is pulled down
|
|
mSPI_write(0x8002, formREG2command(0, 0, 0, 0, m_iMode1_, m_iMode0_)); // EXT_INT4=0
|
|
Wait_CLK_Cycles(256 * 5);
|
|
retval = mSPI_read(1); //REG1 read
|
|
if (retval != 0xAA) temps = "Ext. interrupt 4 test failed.";
|
|
else {
|
|
tempi = 0x0055;
|
|
mSPI_write(0x8000, tempi); // P0=0x55;
|
|
// EXT_INT5=1, external interrupt 5 is raised
|
|
mSPI_write(0x8002, formREG2command(1, 0, 0, 0, m_iMode1_, m_iMode0_)); // EXT_INT5=1
|
|
// here you can put any Delay function
|
|
Wait_CLK_Cycles(256);
|
|
// EXT_INT5=0, external interrupt 5 is pulled down
|
|
mSPI_write(0x8002, formREG2command(0, 0, 0, 0, m_iMode1_, m_iMode0_)); // EXT_INT5=0
|
|
Wait_CLK_Cycles(256 * 5);
|
|
retval = mSPI_read(1); //REG1 read
|
|
if (retval != 0x55)
|
|
{
|
|
temps = "Ext. interrupt 5 test failed.";
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
temps = "Production test finished. MCU is OK.";
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((retval & 0xF0) == 0x30)
|
|
{ // detected error code
|
|
if ((retval & 0x0F) > 0)
|
|
{
|
|
char ctemp[64];
|
|
sprintf(ctemp, "Test %i failed", (retval & 0x0F));
|
|
temps = ctemp;
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
temps = "Test failed";
|
|
return -1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// test too long. Failure.
|
|
temps = "Test failed.";
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
//Baseband gets back the control over SPI switch
|
|
mSPI_write(0x0006, 0x0000); //REG6 write
|
|
|
|
return 0;
|
|
}
|
|
|
|
void MCU_BD::RunTest_MCU(int m_iMode1, int m_iMode0, unsigned short test_code, int m_iDebug) {
|
|
|
|
int i=0;
|
|
int limit=0;
|
|
unsigned short tempi=0x0000;
|
|
unsigned short basei=0x0000;
|
|
|
|
if (test_code<=15) basei=(test_code<<4);
|
|
else basei=0x0000;
|
|
|
|
basei=basei&0xFFF0; // not necessery
|
|
// 4 LSBs are zeros
|
|
|
|
// variable basei contains test no. value at bit positions 7-4
|
|
// used for driving the P0 input
|
|
// P0 defines the test no.
|
|
|
|
if ((test_code>7)||(test_code==0))
|
|
limit=1;
|
|
else
|
|
limit=50;
|
|
|
|
// tests 8 to 14 have short duration
|
|
|
|
if (m_iDebug==1) return; // normal MCU operating mode required
|
|
|
|
// EXT_INT2=1, external interrupt 2 is raised
|
|
tempi=0x0000; // changed
|
|
int m_iExt2=1;
|
|
|
|
if (m_iExt2==1) tempi=tempi|0x0004;
|
|
if (m_iMode1==1) tempi=tempi|0x0002;
|
|
if (m_iMode0==1) tempi=tempi|0x0001;
|
|
|
|
// tempi variable is driving the mspi_REG2
|
|
|
|
mSPI_write(0x8002, tempi); // REG2 write
|
|
|
|
// generating waveform
|
|
for (i=0; i<=limit; i++)
|
|
{
|
|
tempi=basei|0x000C;
|
|
mSPI_write(0x8000, tempi);
|
|
// REG0 write
|
|
Wait_CLK_Cycles(256);
|
|
tempi=basei|0x000D;
|
|
mSPI_write(0x8000, tempi);
|
|
// REG0 write - P0(0) set
|
|
Wait_CLK_Cycles(256);
|
|
tempi=basei|0x000C;
|
|
mSPI_write(0x8000, tempi);
|
|
// REG0 write
|
|
Wait_CLK_Cycles(256);
|
|
tempi=basei|0x000E;
|
|
mSPI_write(0x8000, tempi);
|
|
// REG0 write - PO(1) set
|
|
Wait_CLK_Cycles(256);
|
|
|
|
if (i==0) {
|
|
// EXT_INT2=0
|
|
// external interrupt 2 is pulled down
|
|
tempi=0x0000; // changed
|
|
m_iExt2=0;
|
|
if (m_iExt2==1) tempi=tempi|0x0004;
|
|
if (m_iMode1==1) tempi=tempi|0x0002;
|
|
if (m_iMode0==1) tempi=tempi|0x0001;
|
|
mSPI_write(0x8002, tempi);
|
|
// REG2 write
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void MCU_BD::RunFabTest_MCU(int m_iMode1, int m_iMode0, int m_iDebug) {
|
|
|
|
unsigned short tempi=0x0000;
|
|
|
|
if (m_iDebug==1) return; // normal MCU operating mode required
|
|
|
|
// EXT_INT2=1, external interrupt 2 is raised
|
|
tempi=0x0000; // changed
|
|
int m_iExt2=1;
|
|
if (m_iExt2==1) tempi=tempi|0x0004;
|
|
if (m_iMode1==1) tempi=tempi|0x0002;
|
|
if (m_iMode0==1) tempi=tempi|0x0001;
|
|
mSPI_write(0x8002, tempi); // REG2 write
|
|
|
|
Wait_CLK_Cycles(256);
|
|
|
|
// EXT_INT2=0, external interrupt 2 is pulled down
|
|
tempi=0x0000; // changed
|
|
m_iExt2=0;
|
|
if (m_iExt2==1) tempi=tempi|0x0004;
|
|
if (m_iMode1==1) tempi=tempi|0x0002;
|
|
if (m_iMode0==1) tempi=tempi|0x0001;
|
|
mSPI_write(0x8002, tempi);
|
|
|
|
Wait_CLK_Cycles(256);
|
|
|
|
}
|
|
|
|
void MCU_BD::DebugModeSet_MCU(int m_iMode1, int m_iMode0)
|
|
{
|
|
unsigned short tempi=0x00C0;
|
|
// bit DEBUG is set
|
|
int m_iExt2=0;
|
|
if (m_iExt2==1) tempi=tempi|0x0004;
|
|
if (m_iMode1==1) tempi=tempi|0x0002;
|
|
if (m_iMode0==1) tempi=tempi|0x0001;
|
|
|
|
// Select debug mode
|
|
mSPI_write(0x8002, tempi);
|
|
// REG2 write
|
|
}
|
|
|
|
void MCU_BD::DebugModeExit_MCU(int m_iMode1, int m_iMode0)
|
|
{
|
|
|
|
unsigned short tempi=0x0000; // bit DEBUG is zero
|
|
int m_iExt2=0;
|
|
|
|
if (m_iExt2==1) tempi=tempi|0x0004;
|
|
if (m_iMode1==1) tempi=tempi|0x0002;
|
|
if (m_iMode0==1) tempi=tempi|0x0001;
|
|
// To run mode
|
|
mSPI_write(0x8002, tempi); // REG2 write
|
|
}
|
|
|
|
int MCU_BD::ResetPC_MCU()
|
|
{
|
|
unsigned char tempc1=0x00;
|
|
int retval=0;
|
|
retval=One_byte_command(0x70, &tempc1);
|
|
return retval;
|
|
}
|
|
|
|
int MCU_BD::RunInstr_MCU(unsigned short * pPCVAL)
|
|
{
|
|
unsigned char tempc1, tempc2, tempc3=0x00;
|
|
int retval=0;
|
|
retval=Three_byte_command(0x74, 0x00, 0x00, &tempc1, &tempc2, &tempc3);
|
|
if (retval==-1) (*pPCVAL)=0;
|
|
else (*pPCVAL)=tempc2*256+tempc3;
|
|
return retval;
|
|
}
|
|
|
|
void MCU_BD::Log(const char* msg)
|
|
{
|
|
printf("%s", msg);
|
|
}
|
|
|
|
/** @brief Returns information about programming or reading data progress
|
|
*/
|
|
MCU_BD::ProgressInfo MCU_BD::GetProgressInfo() const
|
|
{
|
|
ProgressInfo info;
|
|
info.stepsDone = stepsDone.load();
|
|
info.stepsTotal = stepsTotal.load();
|
|
info.aborted = aborted.load();
|
|
return info;
|
|
}
|
|
|
|
unsigned int MCU_BD::formREG2command(int m_iExt5, int m_iExt4, int m_iExt3, int m_iExt2, int m_iMode1, int m_iMode0) {
|
|
unsigned int tempi = 0x0000;
|
|
if (m_iExt5 == 1) tempi = tempi | 0x0020;
|
|
if (m_iExt4 == 1) tempi = tempi | 0x0010;
|
|
if (m_iExt3 == 1) tempi = tempi | 0x0008;
|
|
if (m_iExt2 == 1) tempi = tempi | 0x0004;
|
|
if (m_iMode1 == 1) tempi = tempi | 0x0002;
|
|
if (m_iMode0 == 1) tempi = tempi | 0x0001;
|
|
return(tempi);
|
|
}
|
|
|
|
std::string MCU_BD::GetProgramFilename() const
|
|
{
|
|
return mLoadedProgramFilename;
|
|
}
|
|
|
|
/** @brief Starts algorithm in MCU
|
|
*/
|
|
void MCU_BD::RunProcedure(uint8_t id)
|
|
{
|
|
mSPI_write(0x0006, id != 0);
|
|
mSPI_write(0x0000, id);
|
|
uint8_t x0002reg = mSPI_read(0x0002);
|
|
const uint8_t interupt6 = 0x08;
|
|
mSPI_write(0x0002, x0002reg | interupt6);
|
|
mSPI_write(0x0002, x0002reg & ~interupt6);
|
|
std::this_thread::sleep_for(std::chrono::microseconds(10));
|
|
}
|
|
|
|
|
|
/** @brief Waits for MCU to finish executing program
|
|
@return 0 success, 255 idle, 244 running, else algorithm status
|
|
*/
|
|
int MCU_BD::WaitForMCU(uint32_t timeout_ms)
|
|
{
|
|
auto t1 = std::chrono::high_resolution_clock::now();
|
|
auto t2 = t1;
|
|
unsigned short value = 0;
|
|
std::this_thread::sleep_for(std::chrono::microseconds(50));
|
|
do {
|
|
value = mSPI_read(0x0001) & 0xFF;
|
|
if (value != 0xFF) //working
|
|
break;
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
|
t2 = std::chrono::high_resolution_clock::now();
|
|
}while (std::chrono::duration_cast<std::chrono::milliseconds>(t2 - t1).count() < timeout_ms);
|
|
mSPI_write(0x0006, 0); //return SPI control to PC
|
|
//if((value & 0x7f) != 0)
|
|
std::printf("MCU algorithm time: %li ms\n", std::chrono::duration_cast<std::chrono::milliseconds>(t2 - t1).count());
|
|
return value & 0x7F;
|
|
}
|
|
|
|
void MCU_BD::SetParameter(MCU_Parameter param, float value)
|
|
{
|
|
uint8_t inputRegs[3];
|
|
value /= 1e6;
|
|
inputRegs[0] = (uint8_t)value; //frequency integer part
|
|
|
|
uint16_t fracPart = value * 1000.0 - inputRegs[0]*1000.0;
|
|
inputRegs[1] = (fracPart >> 8) & 0xFF;
|
|
inputRegs[2] = fracPart & 0xFF;
|
|
const uint8_t x0002reg = mSPI_read(0x0002);
|
|
const uint8_t interupt7 = 0x04;
|
|
for(uint8_t i = 0; i < 3; ++i)
|
|
{
|
|
mSPI_write(0, inputRegs[2-i]);
|
|
mSPI_write(0x0002, x0002reg | interupt7);
|
|
mSPI_write(0x0002, x0002reg & ~interupt7);
|
|
this_thread::sleep_for(chrono::microseconds(5));
|
|
}
|
|
if(param==MCU_REF_CLK)
|
|
RunProcedure(4);
|
|
if(param == MCU_BW)
|
|
RunProcedure(3);
|
|
}
|
|
|
|
/** @brief Switches MCU into debug mode, MCU program execution is halted
|
|
@param mode MCU memory initialization mode
|
|
@return Operation status
|
|
*/
|
|
MCU_BD::OperationStatus MCU_BD::SetDebugMode(bool enabled, IConnection::MCU_PROG_MODE mode)
|
|
{
|
|
uint8_t regValue = 0;
|
|
switch (mode)
|
|
{
|
|
case IConnection::MCU_PROG_MODE::RESET:
|
|
break;
|
|
case IConnection::MCU_PROG_MODE::EEPROM_AND_SRAM:
|
|
regValue |= 0x01; break;
|
|
case IConnection::MCU_PROG_MODE::SRAM:
|
|
regValue |= 0x02; break;
|
|
case IConnection::MCU_PROG_MODE::BOOT_SRAM_FROM_EEPROM:
|
|
regValue |= 0x03; break;
|
|
}
|
|
if (enabled)
|
|
regValue |= 0xC0;
|
|
mSPI_write(0x8002, regValue);
|
|
return SUCCESS;
|
|
}
|
|
|
|
MCU_BD::OperationStatus MCU_BD::readIRAM(const uint8_t *addr, uint8_t* values, const uint8_t count)
|
|
{
|
|
uint8_t cmd = 0x78; //
|
|
int retval;
|
|
for (int i = 0; i < count; ++i)
|
|
{
|
|
mSPI_write(0x8004, cmd); //REG4 write cmd
|
|
retval = WaitUntilWritten();
|
|
if (retval == -1) return FAILURE;
|
|
|
|
mSPI_write(0x8004, addr[i]); //REG4 write IRAM address
|
|
retval = WaitUntilWritten();
|
|
if (retval == -1) return FAILURE;
|
|
|
|
mSPI_write(0x8004, 0); //REG4 nop
|
|
retval = WaitUntilWritten();
|
|
if (retval == -1) return FAILURE;
|
|
|
|
uint8_t result = 0;
|
|
retval = ReadOneByte(&result);
|
|
if (retval == -1) return FAILURE;
|
|
|
|
retval = ReadOneByte(&result);
|
|
if (retval == -1) return FAILURE;
|
|
|
|
retval = ReadOneByte(&result);
|
|
if (retval == -1) return FAILURE;
|
|
values[i] = result;
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
MCU_BD::OperationStatus MCU_BD::writeIRAM(const uint8_t *addr, const uint8_t* values, const uint8_t count)
|
|
{
|
|
return FAILURE;
|
|
}
|
|
|
|
uint8_t MCU_BD::ReadMCUProgramID()
|
|
{
|
|
RunProcedure(255);
|
|
auto statusMcu = WaitForMCU(10);
|
|
return statusMcu & 0x7F;
|
|
}
|