分享
 
 
 

Windows CE下的串口通讯类

王朝vc·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

Windows CE下的串口通讯类

串行通讯是目前计算机、通信和控制领域最基本的通信方式。在CSDN的“嵌入式开发/WINCE”社区中,经常有人提问该到哪找串口通讯类,其实这个问题我自己也问过。:)而一般的回答是给你提供一个Pocket PC 2002的SDK例子程序。但到底SDK的程序和MFC的结构有很大的不同,对于想用MFC编写通信程序的人来说也不是很便利。

另一方面,由于Windows CE是一个基于Unicode的操作系统,并且Windows CE不支持Windows下常用的串行通信重叠I/O方式(OVERLAPPED),因此编写Windows CE下的串口通讯类有一些与桌面Windows不同的地方。

以下是我从SDK程序改写而来的MFC类,希望能和致力于WINCE开发的朋友多多交流,由于本人才疏学浅,程序中有许多不完善的地方,请大家指正。我的程序是基于“主动发送请求,被动接收响应”的假设,因此我只设置了一个接收数据的线程。如果有朋友能提供有独立发送数据和接收数据线程的类,我将十分感激。我的E_mail:zhenxizhou@elong.com

感谢“嵌入式开发/WINCE”社区为我提供SDK例子的朋友,感谢CSDN上所有热心的朋友,祝愿中国的软硬件水平能早日挤身世界一流。

头文件Serial.h

// Serial.h: interface for the CSerial class.

//

//////////////////////////////////////////////////////////////////////

#if !defined(AFX_SERIAL_H__59575586_AAA9_4FEF_B2A7_E089553698EF__INCLUDED_)

#define AFX_SERIAL_H__59575586_AAA9_4FEF_B2A7_E089553698EF__INCLUDED_

#if _MSC_VER > 1000

#pragma once

#endif // _MSC_VER > 1000

DWORD WINAPI ReadPortThread(LPVOID lpvoid); //读数据线程

class CSerial

{

public:

BOOL InitCommTimeouts(); //设置超时参数

BOOL InitDCB(); //配置串口

BOOL m_bConnected;

BOOL ClosePort(HANDLE hCommPort); //关闭串口

DWORD WritePort(TCHAR *buf,DWORD dwBytesToWrite); //写数据

BOOL OpenPort(LPTSTR lpszPortName); //打开串口

CSerial();

HANDLE hReadThread;

virtual ~CSerial();

};

#endif // !defined(AFX_SERIAL_H__59575586_AAA9_4FEF_B2A7_E089553698EF__INCLUDED_)

源文件:Serial.cpp

// Serial.cpp: implementation of the CSerial class.

//

//////////////////////////////////////////////////////////////////////

#include "stdafx.h"

#include "Serial.h"

#ifdef _DEBUG

#undef THIS_FILE

static char THIS_FILE[]=__FILE__;

#define new DEBUG_NEW

#endif

HANDLE hPort;

CString strInChar;

//////////////////////////////////////////////////////////////////////

// Construction/Destruction

//////////////////////////////////////////////////////////////////////

CSerial::CSerial()

{

}

CSerial::~CSerial()

{

if(hPort != INVALID_HANDLE_VALUE)

ClosePort(hPort);

}

BOOL CSerial::OpenPort(LPTSTR lpszPortName)

{

DWORD dwError,

dwThreadID;

if(hPort)

{

return FALSE;

}

//打开串口

hPort = CreateFile (lpszPortName, GENERIC_READ | GENERIC_WRITE,

0, NULL, OPEN_EXISTING,0, NULL);

//如果打开端口出错, 返回FALSE

if ( hPort == INVALID_HANDLE_VALUE )

{

//不能打开端口

CString strError;

strError.Format(_T("Unable to open %s, Error No.=%d"),

lpszPortName, GetLastError());

MessageBox (NULL, strError, TEXT("Error"), MB_OK);

return FALSE;

}

//指定端口监测的事件集

SetCommMask (hPort, EV_RXCHAR);

//分配设备缓冲区

SetupComm(hPort,512,512);

//初始化缓冲区中的信息

PurgeComm(hPort,PURGE_TXCLEAR|PURGE_RXCLEAR);

//配置串行端口

if(!InitDCB())

return FALSE;

//设置端口超时值

if(!InitCommTimeouts())

return FALSE;

//设置端口上指定信号的状态

// SETDTR: 发送DTR (data-terminal-ready)信号

// SETRTS: 发送RTS (request-to-send)信号

EscapeCommFunction (hPort, SETDTR);

EscapeCommFunction (hPort, SETRTS);

//创建一个从串口读取数据的线程

if (hReadThread = CreateThread (NULL, 0, ReadPortThread, 0, 0,

&dwThreadID))

{

}

else

{

//不能创建线程

MessageBox (NULL, TEXT("Unable to create the read thread"),

TEXT("Error"), MB_OK);

dwError = GetLastError ();

return FALSE;

}

m_bConnected=TRUE;

return TRUE;

}

DWORD CSerial::WritePort(TCHAR *buf,DWORD dwCharToWrite)

{

BOOL fWriteState;

DWORD dwBytesWritten;

//写入数据

fWriteState=WriteFile(hPort,buf,dwCharToWrite*sizeof(TCHAR),&dwBytesWritten,NULL);

if(!fWriteState)

{

//不能写数据

MessageBox(NULL,TEXT("Can't Write String to Comm"),TEXT("Error"),MB_OK);

dwBytesWritten=0;

}

return dwBytesWritten;

}

DWORD WINAPI ReadPortThread(LPVOID lpvoid)

{

BOOL fReadState;

DWORD dwCommModemStatus;

DWORD dwLength;

COMSTAT ComStat;

DWORD dwErrorFlags;

while (hPort != INVALID_HANDLE_VALUE)

{

//等待串口的事件发生

WaitCommEvent (hPort, &dwCommModemStatus, 0);

if (dwCommModemStatus & EV_RXCHAR)

{

ClearCommError(hPort,&dwErrorFlags,&ComStat);

//cbInQue返回在串行驱动程序输入队列中的字符数

dwLength=ComStat.cbInQue;

if(dwLength>0)

{

//从串口读取数据

TCHAR* buf=new TCHAR[256];

fReadState=ReadFile(hPort,buf,dwLength,&dwLength,NULL);

if(!fReadState)

{

//不能从串口读取数据

MessageBox(NULL,TEXT("Error in read from serial port"),TEXT("Read Error"),MB_OK);

}

else

{

//把数据赋值给全局变量

strInChar=buf;

}

delete[] buf;

}

}

GetCommModemStatus (hPort, &dwCommModemStatus);

}

return 0;

}

BOOL CSerial::ClosePort(HANDLE hCommPort)

{

if (hCommPort != INVALID_HANDLE_VALUE)

{

//设置连接属性为FALSE

m_bConnected=FALSE;

//结束线程中WaitCommEvent的等待

SetCommMask(hPort,0);

//阻塞至线程停止

if(hReadThread)

{

TerminateThread(hReadThread,0);

CloseHandle(hReadThread);

}

//清除端口上指定信号的状态

EscapeCommFunction(hPort,CLRDTR);

EscapeCommFunction(hPort,CLRRTS);

//清除驱动程序内部的发送和接收队列

PurgeComm(hPort,PURGE_TXCLEAR|PURGE_RXCLEAR);

//关闭串口

CloseHandle (hCommPort);

hCommPort = INVALID_HANDLE_VALUE;

return TRUE;

}

else

{

return TRUE;

}

}

BOOL CSerial::InitDCB()

{

DCB PortDCB;

DWORD dwError;

PortDCB.DCBlength = sizeof (DCB);

//得到端口的默认设置信息

GetCommState (hPort, &PortDCB);

//改变DCB结构设置

PortDCB.BaudRate = 19200; //波特率

PortDCB.fBinary = TRUE; //Win32不支持非二进制串行传输模式,必须为TRUE

PortDCB.fParity = TRUE; //启用奇偶校验

PortDCB.fOutxCtsFlow = TRUE; //串行端口的输出由CTS线控制

PortDCB.fOutxDsrFlow = FALSE; //关闭串行端口的DSR流控制

PortDCB.fDtrControl = DTR_CONTROL_ENABLE; //启用DTR线

PortDCB.fDsrSensitivity = FALSE; //如果设为TRUE将忽略任何输入的字节,除非DSR线被启用

//PortDCB.fTXContinueOnXoff = TRUE; //当为TRUE时,如果接收缓冲区已满且驱动程序已传送XOFF字符,将使驱动程序停止传输字符

PortDCB.fTXContinueOnXoff = FALSE;

PortDCB.fOutX = FALSE; //设为TRUE指定XON/XOFF控制被用于控制串行输出

PortDCB.fInX = FALSE; //设为TRUE指定XON/XOFF控制被用于控制串行输入

PortDCB.fErrorChar = FALSE; //WINCE串行驱动程序的默认执行将忽略这个字段

PortDCB.fNull = FALSE; //设为TRUE将使串行驱动程序忽略收到的空字节

PortDCB.fRtsControl = RTS_CONTROL_ENABLE; //启用RTS线

PortDCB.fAbortOnError = FALSE; //WINCE串行驱动程序的默认执行将忽略这个字段

PortDCB.ByteSize = 8; //每字节的位数

PortDCB.Parity = NOPARITY; //无奇偶校验

PortDCB.StopBits = ONESTOPBIT; //每字节一位停止位

//根据DCB结构配置端口

if (!SetCommState (hPort, &PortDCB))

{

//不能配置串行端口

MessageBox (NULL, TEXT("Unable to configure the serial port"),

TEXT("Error"), MB_OK);

dwError = GetLastError ();

return FALSE;

}

return TRUE;

}

BOOL CSerial::InitCommTimeouts()

{

COMMTIMEOUTS CommTimeouts;

DWORD dwError;

//得到超时参数

GetCommTimeouts (hPort, &CommTimeouts);

//改变COMMTIMEOUTS结构设置

CommTimeouts.ReadIntervalTimeout = MAXDWORD;

CommTimeouts.ReadTotalTimeoutMultiplier = 0;

CommTimeouts.ReadTotalTimeoutConstant = 0;

CommTimeouts.WriteTotalTimeoutMultiplier = 10;

CommTimeouts.WriteTotalTimeoutConstant = 1000;

//设置端口超时值

if (!SetCommTimeouts (hPort, &CommTimeouts))

{

//不能设置超时值

MessageBox (NULL, TEXT("Unable to set the time-out parameters"),

TEXT("Error"), MB_OK);

dwError = GetLastError ();

return FALSE;

}

return TRUE;

}

以上类代码在eMbedded Visual C++4.0和基于ARM9的三星S3C2410开发板(运行Windows CE.NET 4.1)上测试通过。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有