mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-25 18:10:22 -04: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;
 | |
| }
 |