⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 serialpro.cpp

📁 51isp下载软件,使用非常方便。非常适应初学者制作使用
💻 CPP
字号:
//////////////////////////////////////////////////////////////////////
// SerialPro.cpp: implementation of the CSerialPro class.
//负责与串行编程器通讯,对编程器发令
//////////////////////////////////////////////////////////////////////

#include "stdafx.h"
#include "EPro.h"
#include "SerialPro.h"
#include "Pro.h"

#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[]=__FILE__;
#define new DEBUG_NEW
#endif

//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////

CSerialPro::CSerialPro()
{
	m_bCanUse=FALSE;
	m_bPort=FALSE;
	m_bThread=FALSE;
	m_hReadRsp=CreateEvent(NULL,FALSE,FALSE,NULL);
	m_hWriteRsp=CreateEvent(NULL,FALSE,FALSE,NULL);
}

CSerialPro::~CSerialPro()
{

}

//发送18个字节,第1个字节表示操作类型,第2个字节表示器件的FID,
//后16个字节是与该操作相关的数据
BOOL CSerialPro::SendData()
{
	if(!m_bPort)
	{
		m_pPro->Notify(PRO_INIT_SERIAL_NOT);
		return FALSE;
	}
	if(m_bCanUse)
	{
		TRACE("[0]%2x [1]%2x [2]%2x [3]%2x [4]%2x [5]%2x [6]%2x [7]%2x\n",TxdBuf[0],TxdBuf[1],TxdBuf[2],TxdBuf[3],TxdBuf[4],TxdBuf[5],TxdBuf[6],TxdBuf[7]);
		SendBlock(TxdBuf,18);
		m_bCanUse=FALSE;
		return TRUE;
	}
	else
	{
		m_pPro->Notify(PRO_INVALID);
		return FALSE;
	}
}

//接收完指定长度数据块后的处理
void CSerialPro::OnReceived(BYTE *pBuf, UINT nLength)
{
	static int n=0;
	n++;
	TRACE("%d-OnReceived %d BYTES\n",n,nLength);
	TRACE("[0]%2x [1]%2x [2]%2x [3]%2x [4]%2x [5]%2x [6]%2x [7]%2x\n",pBuf[0],pBuf[1],pBuf[2],pBuf[3],pBuf[4],pBuf[5],pBuf[6],pBuf[7]);
	if(nLength==18)
	{
		switch(pBuf[0])//是什么操作的回应
		{
		case 0://编程器就绪
			m_bCanUse=TRUE;
			m_pPro->Notify(PRO_AVAILABLE);
			return;
		case 1://编程器读入特征字
			m_bCanUse=TRUE;
			m_pPro->SignByte[0]=pBuf[2];
			m_pPro->SignByte[1]=pBuf[3];
			m_pPro->SignByte[2]=pBuf[4];
			m_pPro->CheckChip();//检查是否可以找到与特征字匹配的器件
			return;
		case 2://编程器插除器件
			m_bCanUse=TRUE;
			m_pPro->Notify(PRO_ERASE);
			return;
		case 3:
			//编程器等待写器件的回应,建立写ROM的线程
			TRACE("m_bThread=%d\n",m_bThread);
			if(m_bThread==FALSE?(AfxBeginThread(WriteRomThread,this)!=NULL):FALSE)
				return;
		case 4:
			//编程器等待读器件的回应,建立读ROM的线程
			TRACE("m_bThread=%d\n",m_bThread);
			if(m_bThread==FALSE?(AfxBeginThread(ReadRomThread,this)!=NULL):FALSE)
				return;
		case 5://编程器写锁定位完成
			m_bCanUse=TRUE;
			m_pPro->Notify(PRO_LOCK_BIT);
			return;
		case 6://读出器件的数据
		 if(m_bThread)
		 {
			SetEvent(m_hReadRsp);//用事件通知读线程
		 }
			return;
		case 0xff://编程器收到无效的命令
			m_bCanUse=TRUE;
			m_pPro->Notify(PRO_WORK_INVALID);
			return;
		default:
			m_bCanUse=TRUE;
			break;
		}
	}
	else if(nLength==1 && m_bThread)//如果只接收到1个字节,是否给写线程的 
	{
		//为提高通讯效率,编程器写完一块数据后回应一个字节
		//表示是否成功写入
		SetEvent(m_hWriteRsp);//用事件通知写线程
	}
	else//通讯出错了
	{
		m_pPro->Notify(PRO_RCV_ER);
		m_bCanUse=FALSE;
	}
}


BOOL CSerialPro::InitPort(CPro* pPro, UINT nPort, UINT nBaud)
{
	m_pPro=pPro;
	if(m_bPort)
	{
		StopMonitoring();//如果以前占用了串口,先关闭监听串口
	}
	//初始化串口通讯
	m_bPort=CSerialPort::InitPort((CWnd*)pPro->m_pDlg,nPort,nBaud,'N');
	if(m_bPort)
	{
		SetExpRcv(18);//设置期望的数据块长度
		StartMonitoring();//监听串口
		m_pPro->Notify(PRO_INIT_SERIAL_SEC);
		TRACE("CSerialPort::InitPort-PASS\n");
	}
	else
	{
		m_pPro->Notify(PRO_INIT_SERIAL_ER);
		TRACE("CSerialPort::InitPort-ERROR\n");
	}
	return m_bPort;
}

void CSerialPro::DetectPro(BYTE FID)
{
	TRACE("DetectPro(%d)\n",FID);
	TxdBuf[0]=0;
	TxdBuf[1]=FID;
	m_bCanUse=TRUE;
	SendData();
}

void CSerialPro::ReadSign(BYTE FID)
{
	TRACE("ReadSign(%d)\n",FID);
	TxdBuf[0]=1;
	TxdBuf[1]=FID;
	SendData();
}

void CSerialPro::Erase(BYTE FID)
{
	TRACE("Erase(%d)\n",FID);
	TxdBuf[0]=2;
	TxdBuf[1]=FID;
	SendData();	
}

//发送开始写器件的命令,让编程器做好准备
void CSerialPro::BeginWrite(BYTE FID)
{
	TRACE("BeginWrite(%d)\n",FID);
	TxdBuf[0]=3;
	TxdBuf[1]=FID;
	SendData();
}

//发送开始读器件的命令,让编程器做好准备
void CSerialPro::BeginRead(BYTE FID)
{
	TRACE("BeginWrite(%d)\n",FID);
	TxdBuf[0]=4;
	TxdBuf[1]=FID;
	SendData();
}

void CSerialPro::LockBit(BYTE FID,int nBit)
{//写锁定位,nBit为锁定位数
	TRACE("LockBit(%d,%d)\n",FID,nBit);
	TxdBuf[0]=5;
	TxdBuf[1]=FID;
	TxdBuf[2]=(BYTE)nBit;
	SendData();
}

//写器件的线程
UINT CSerialPro::WriteRomThread(LPVOID pParam)
{
	TRACE("WriteRomThread(LPVOID pParam)\n");
	CSerialPro* pSerialPro=(CSerialPro*)pParam;
	pSerialPro->m_bThread=TRUE;
	CPro* pPro=pSerialPro->m_pPro;
	pPro->Notify(PRO_WRITE_BEGIN);
	pPro->m_nAddr=0;
	pPro->m_nDataLength=pPro->GetMinLength(pPro->m_pWriteBuf,0x10000);
	//得到缓冲中最小数据块长度,跳过写0xff
	if(pPro->m_nDataLength==0)//如果数据块长度为0,直接退出
	{
		TRACE("WriteRomThread(...) Over\n");
		pSerialPro->m_bThread=FALSE;
		pSerialPro->TxdBuf[0]=0;//通知编程器写结束
		pSerialPro->SendBlock(pSerialPro->TxdBuf,18);
		pPro->Notify(PRO_WRITE_OVER);
		return 0;
	}
	pSerialPro->SetExpRcv(1);//设置只要接收一个字节
	Sleep(5);
	ResetEvent(pSerialPro->m_hWriteRsp);//事件复位
	while(pPro->m_nAddr<pPro->m_nDataLength)
	{//只要ROM地址指针小于数据块长度,就继续写
		TRACE("pPro->m_nAddr:%04x pPro->m_nDataLength:%04x\n",pPro->m_nAddr,pPro->m_nDataLength);
		pSerialPro->TxdBuf[0]=0x07;//表示继续写
		memcpy(&pSerialPro->TxdBuf[2],&pPro->m_pWriteBuf[pPro->m_nAddr],0x10);
		//给发送缓冲填充要写的数据
		pSerialPro->SendBlock(pSerialPro->TxdBuf,18);//发送
		TRACE("WaitForSingleObject(...)\n");
		if(WaitForSingleObject(pSerialPro->m_hWriteRsp,1000)==WAIT_TIMEOUT)
		{//等待编程器回应超时
			TRACE("WaitForSingleObject(...) TimeOut\n");
			pSerialPro->SetExpRcv(18);
			pPro->Notify(PRO_RCV_ER);
			pSerialPro->m_bThread=FALSE;
			pSerialPro->m_bCanUse=TRUE;
			return -1;
			
		}
		TRACE("WaitForSingleObject(...) OVER\n");
		if(pSerialPro->RxdBuf[0]==0x01)
		{//写数据成功
			pPro->m_nAddr+=0x10;
			pPro->Notify(PRO_WRITE_BLOCK);
		}
		else if(pSerialPro->RxdBuf[0]==0xff)
		{//写数据失败
			TRACE("WriteRomThread(...) ERROR\n");
			pSerialPro->SetExpRcv(18);
			pSerialPro->TxdBuf[0]=0;
			pSerialPro->SendBlock(pSerialPro->TxdBuf,18);
			pPro->Notify(PRO_WRITE_ER);
			pSerialPro->m_bThread=FALSE;
			pSerialPro->m_bCanUse=TRUE;
			return -1;
		}
		else//通讯出错
		{
			TRACE("WriteRomThread(...) COM ER\n");
			pSerialPro->SetExpRcv(18);
			pPro->Notify(PRO_RCV_ER);
			pSerialPro->m_bThread=FALSE;
			pSerialPro->m_bCanUse=TRUE;
			return -1;
		}
	}
	pSerialPro->SetExpRcv(18);//还原
	pSerialPro->TxdBuf[0]=0;//通知编程器写结束
	pSerialPro->SendBlock(pSerialPro->TxdBuf,18);
	pPro->Notify(PRO_WRITE_OVER);
	TRACE("WriteRomThread(...) SEC\n");
	pSerialPro->m_bThread=FALSE;
	pSerialPro->m_bCanUse=TRUE;
	return 0;
}

//读器件的线程
UINT CSerialPro::ReadRomThread(LPVOID pParam)
{
	TRACE("ReadRomThread(LPVOID pParam)\n");
	CSerialPro* pSerialPro=(CSerialPro*)pParam;
	pSerialPro->m_bThread=TRUE;
	CPro* pPro=pSerialPro->m_pPro;
	pPro->Notify(PRO_READ_BEGIN);
	BYTE RespByte=1;
	BOOL bReWait=FALSE;
	pPro->m_nAddr=0;
	if(!pPro->m_bCheck)//如果不是效验,则读整个ROM
		pPro->m_nDataLength=pPro->GetCurChip()->RomSize*1024;
	if(pPro->m_nDataLength==0)
	{
		pPro->Notify(PRO_READ_OVER);
		TRACE("ReadRomThread(...) Over\n");
		pSerialPro->m_bThread=FALSE;
		return 0;
	}
	ResetEvent(pSerialPro->m_hReadRsp);//事件复位
	while(pPro->m_nAddr<pPro->m_nDataLength)
	{
		TRACE("pPro->m_nAddr:%04x pPro->m_nDataLength:%04x\n",pPro->m_nAddr,pPro->m_nDataLength);
		pSerialPro->SendBlock(&RespByte,1);//发一个字节给编程器
		TRACE("WaitForSingleObject(...)\n");
		if(WaitForSingleObject(pSerialPro->m_hReadRsp,500)==WAIT_TIMEOUT)
		{//等待回应超时
			if(bReWait)//如果是重发都失败
			{
				RespByte=0;
				pSerialPro->SendBlock(&RespByte,1);
				pPro->Notify(PRO_RCV_ER);
				pSerialPro->m_bThread=FALSE;
				pSerialPro->m_bCanUse=TRUE;
				return -1;
			}
			else//尝试要求重发
			{
				RespByte=0xff;
				bReWait=TRUE;
				continue;
			}
		}
		bReWait=FALSE;
		TRACE("WaitForSingleObject(...) OVER\n");
		memcpy(&pPro->m_pReadBuf[pPro->m_nAddr],&pSerialPro->RxdBuf[2],0x10);
		//收到数据后,填充到读缓冲
		pPro->m_nAddr+=0x10;
		RespByte=0x01;//表示继续读
		pPro->Notify(PRO_READ_BLOCK);
	}
	RespByte=0;//表示读结束
	pSerialPro->SendBlock(&RespByte,1);
	pPro->Notify(PRO_READ_OVER);
	pSerialPro->m_bThread=FALSE;
	pSerialPro->m_bCanUse=TRUE;
	TRACE("ReadRomThread(...) Over\n");
	return 0;
}


⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -