• 9902阅读
  • 1回复

Windows CE下的串口通讯类[转][sunny12] [复制链接]

上一主题 下一主题
离线XChinux
 

只看楼主 倒序阅读 楼主  发表于: 2005-07-30
串行通讯是目前计算机、通信和控制领域最基本的通信方式。在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)上测试通过。
二笔 openSUSE Vim N9 BB10 XChinux@163.com 网易博客 腾讯微博
承接C++/Qt、Qt UI界面、PHP及预算报销系统开发业务
离线sunxiaoming
只看该作者 1楼 发表于: 2006-03-08
这个好东西,收了
谢谢lz
快速回复
限100 字节
 
上一个 下一个