Ftdi芯片可以工作后,开始了串口测试工作。我们使用Ft2232d芯片的板子,从一个USB口扩展出两个标准RS232串口,测试目标就是让两个串口可以同时收发数据。
对于测试工作当然采用软件编程来实现了,PC端的串口测试程序多不胜数,所以主要做一个Wince端的测试程序,目标就是要让串口可以接收和发送文件。(CE端的串口测试程序也有不少,但是很少发现带有传输文件功能的程序,所以只好自己动手了)
1. 打开串口:
HADNLE hComm = CreateFile(_T("com1:"), GENERIC_READ|GENERIC_WRITE, 0, NULL,
OPEN_EXISTING,0,0);
CloseHandle(hComm) // 关闭串口
2. 配置串口:
DCB dcb;
GetCommState(hComm,&dcb)
SetCommState(hComm,&dcb)
我只修改了部分参数,其他为默认配置。
dcb.BaudRate = CBR_115200;
dcb.ByteSize = 8;
dcb.fBinary = TRUE;
dcb.fParity = TRUE;
dcb.Parity = NOPARITY;
dcb.StopBits = ONESTOPBIT;
dcb.fDtrControl = DTR_CONTROL_ENABLE; // 与硬件相关,尚不清楚有什么作用?
dcb.fRtsControl = RTS_CONTROL_ENABLE; // 同上
3. 配置超时:
COMMTIMEOUTS Timeouts;
SetCommTimeouts(hComm,&Timeouts)
对于超时没有什么深刻体会,只是按照理解配置了相应的时间。
Timeouts.ReadIntervalTimeout = 100;
Timeouts.ReadTotalTimeoutConstant = 100;
Timeouts.ReadTotalTimeoutMultiplier = 10;
Timeouts.WriteTotalTimeoutConstant = 0;
Timeouts.WriteTotalTimeoutMultiplier = 0;
读超时有两种计算方法,一是ReadIntervalTimeout指定在接收字符间最大的时间间隔,二是基于要接
收的字符数量,ReadTotalTimeoutConstant+ReadTotalTimeoutMultiplier*要读的字节数。当两种
方式都设置了时间时,同时有效,影响ReadFile(...)返回。
因为要做的是读取文件,所有按照理解,设置字符间最大时间间隔为100ms,这个用来接收文件名和文
件大小。在发送文件的线程中控制发送,在读取文件名后,获取文件大小,并将文件名和文件大小组合,使
用WriteFile(...)发送到串口,成功后使用Sleep(200),“挂起”发送线程一段时间,这样当接收线程收到
串口事件时,就可以接收到文件名和文件大小,并且在超时100ms后,ReadFile(...)返回,防止接收到文
件数据。
对于ReadTotalTimeoutConstant 和 ReadTotalTimeoutMultiplier 就是凭感觉设置的,没什么太
多想法了。
4. 收发数据:
WriteFile(...)
ReadFile(...)
SetCommMask(hComm,EV_RXCHAR);
ClearCommError(hComm,&dwReadError,&cmState);
WaitCommEvent(hComm,&evtMask,NULL)
感觉上发送线程比较好处理,使用CFileDialog获取选择的文件,同样使用CreateFile打开文件,通过
GetFileSize获取文件大小,使用CString::Format方法很容易将数据转换为字符串,通过重载的‘+’方
法,组合文件名和文件大小。
m_strFileName = dlg.GetFileName();
m_strPathName = dlg.GetPathName();
hFile = CreateFile(m_strPathName,GENERIC_READ,0,NULL,OPEN_EXISTING,0,NULL);
strFile.Format(_T("%lu"),GetFileSize(hFile,NULL));
strFile = m_strFileName + _T('//') + strFile;
if( WriteFile(m_hComm,strFile.GetBuffer(),strFile.GetLength()*sizeof(TCHAR),
&dwWriteLen,NULL))
{
Sleep(200);
}
while。。。 // 发送文件数据
接收线程就比较麻烦了,通过SetCommMask()设置串口事件,然后通过WaitCommEvent()等待事件
发生。(并不清楚设置怎样的事件组合比较适当,所有只等待接收字符事件)由于要接收文件名和文件数
据,所以设定在while循环外接收文件名,并且创建文件,然后就是while循环接收数据。
if(WaitCommEvent(hComm,&evtMask,NULL))
{
SetCommMask(hComm,EV_RXCHAR);
if(evtMask & EV_RXCHAR)
{
ClearCommError(hComm,&dwReadError,&cmState);
dwWillReadLen = cmState.cbInQue;
CString str(_T('/0'),MAX_PATH);
if(ReadFile(hComm,str.GetBuffer(),dwWillReadLen,&dwReadLen,0))
{
str.ReleaseBuffer();
... // 处理字符串
hFile = CreateFile(...);
}
While。。。 // 接收文件数据
}
}
这样完成了串口的相关编程工作,做的还是比较简单了,经过测试,从PC上可以向手持设备发送文件了,并且手持设备可以完整的接收文件。同样两个串口可以一同工作,收发文件大小为1MB的,都没有问题。当时就有同事感叹呐,FTDI如何做到的同时由两个串口向一个USB口发数据的。
虽然测试程序还存在诸多问题,手持设备上的程序只能接收一次文件,想要接收第二次必须重新连接串口,原因就是接收线程在接收到文件后就会退出了,这个问题应该可以通过多线程同步对象解决。还有就是存在内存泄漏问题......问题原因不详(没去找)。另外在某些情况下,程序会罢工,手持设备收不到文件,而PC端发送正常。。。
首次开发串口程序,还有些疑问:
1,配置串口,还有许多参数不知道如何进行配置,当然也不知道如何使用。
2,配置超时,很迷茫,不知怎样配合程序进行配置。
3,串口事件,还有很多不知道如何使用,或者说不知道何时会用到。
4,关于一些API的使用还不熟练,
ClearCommError, EscapeCommFunction, GetCommModemStatus,
GetCommProperties, PurgeComm, SetupComm, TransmitCommChar
以上所写全部为个人理解,限于水平有限,难免存在缺漏和不足,也请看到文章的人海涵。另外说明本博客内凡是技术文章全部为原创,书写风格类似于工作笔记,本着尊重个人劳动成果的原则,如需转载或引用,请添加原始链接或注明出处。
本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/embest_mhq/archive/2009/03/11/3980751.aspx