同步串口通信代码

最近做了一个串口通信相关的项目。主要涉及到同步串口通信。把代码保存下来以便以后使用。同时也能和大家一起共享,希望对大家有所帮助.

头文件:SerialPort.h

 1 #pragma once
 2 
 3 #include <Windows.h>
 4 #include "DataTypes.h"
 5 
 6 
 7 #define COM_BUFFER_LENGTH 409600
 8 
 9 class CSerialPort
10 {
11 public:
12     CSerialPort(void);
13     ~CSerialPort(void);
14 public:
15     bool InitPort(UINT nPortNo=1,UINT baud=CBR_115200,char parity='N',UINT databits=8,UINT stopbits=1,DWORD dwCommEvents=EV_RXCHAR);
16     bool WriteData(char* pData,int length);//向串口中写入数据
17     UINT GetBytesInCom();//获取串口缓冲区中的字节数
18     bool ReadChar(char &cRecved);//从串口缓冲区读取一个字节的数据
19 public:
20     bool OpenPort(UINT portNo);//打开串口
21     void ClosePort();//关闭串口
22     bool ReadData(char* pData);
23     char* m_data;
24 public:
25     HANDLE m_hCom;//串口句柄
26     CRITICAL_SECTION m_cs;
27 };

实现文件:SerialPort.cpp

  1 #include "StdAfx.h"
  2 #include "SerialPort.h"
  3 #include<iostream>
  4 #include <process.h>
  5 
  6 UINT const SLEEP_TIMEINTERVAL=5;//串口无数据时,休眠,单位是s
  7 
  8 CSerialPort::CSerialPort(void)
  9 {
 10     m_hCom=INVALID_HANDLE_VALUE;
 11     m_data=new char[COM_BUFFER_LENGTH];
 12     memset(m_data,0,COM_BUFFER_LENGTH);
 13     InitializeCriticalSection(&m_cs);
 14 }
 15 
 16 CSerialPort::~CSerialPort(void)
 17 {
 18     if (m_hCom!=INVALID_HANDLE_VALUE)
 19     {
 20         CloseHandle(m_hCom);
 21     }
 22     if (m_data!=NULL)
 23     {
 24         delete[] m_data;
 25     }
 26     DeleteCriticalSection(&m_cs);
 27 }
 28 
 29 bool CSerialPort::InitPort(UINT nPortNo,UINT baud,char parity,UINT databits,UINT stopbits,DWORD dwCommEvents)
 30 {
 31     char szDCBparam[50];
 32     sprintf_s(szDCBparam,"baud=%d parity=%c data=%d stop=%d",CBR_115200,parity,databits,stopbits);
 33 
 34     if (OpenPort(nPortNo)==false)
 35     {
 36         return false;
 37     }
 38 
 39     EnterCriticalSection(&m_cs);
 40 
 41     BOOL bIsSuccess=TRUE;
 42     
 43     if (bIsSuccess)
 44     {
 45         bIsSuccess=SetupComm(m_hCom,COM_BUFFER_LENGTH,COM_BUFFER_LENGTH);//输入输出缓冲区设置为4k
 46     }
 47 
 48     //设置串口超时时间。全为0,则不使用串口超时设置,超时
 49     COMMTIMEOUTS comTms;
 50     comTms.ReadIntervalTimeout=MAXWORD;
 51     comTms.ReadTotalTimeoutConstant=1000;
 52     comTms.ReadTotalTimeoutMultiplier=1000;
 53     comTms.WriteTotalTimeoutConstant=1000;
 54     comTms.WriteTotalTimeoutMultiplier=1000;
 55     if (bIsSuccess)
 56     {
 57         bIsSuccess=SetCommTimeouts(m_hCom,&comTms);
 58     }
 59 
 60     //设置DCB结构
 61     DCB dcb;
 62     if (bIsSuccess)
 63     {
 64         bIsSuccess=GetCommState(m_hCom,&dcb);
 65         dcb.BaudRate=128000;
 66         dcb.ByteSize=8;
 67         dcb.Parity=NOPARITY;
 68         dcb.StopBits=stopbits;
 69     }
 70 
 71     if (bIsSuccess)
 72     {
 73         //使用dcb参数配置串口状态
 74         bIsSuccess=SetCommState(m_hCom,&dcb);
 75     }
 76 
 77     //清空串口缓冲区
 78     PurgeComm(m_hCom,PURGE_RXABORT|PURGE_RXCLEAR|PURGE_TXABORT|PURGE_TXCLEAR);
 79     LeaveCriticalSection(&m_cs);
 80 
 81     return (bIsSuccess==TRUE);
 82 }
 83 
 84 
 85 bool CSerialPort::OpenPort(UINT portNo)
 86 {
 87     EnterCriticalSection(&m_cs);
 88     char szPort[50];
 89     sprintf_s(szPort,"COM%d",portNo);
 90     
 91     m_hCom=CreateFile(szPort,GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,0,0);
 92     if (m_hCom==INVALID_HANDLE_VALUE)
 93     {
 94         LeaveCriticalSection(&m_cs);
 95         return false;
 96     }
 97     LeaveCriticalSection(&m_cs);
 98     return true;
 99 }
100 
101 void CSerialPort::ClosePort()
102 {
103     if (m_hCom!=INVALID_HANDLE_VALUE)
104     {
105         CloseHandle(m_hCom);
106         m_hCom=INVALID_HANDLE_VALUE;
107     }
108 }
109 
110 bool CSerialPort::WriteData(char* pData,int length)
111 {
112     BOOL bResult=TRUE;
113     DWORD bytesToSend=0;
114     if (m_hCom==INVALID_HANDLE_VALUE)
115     {
116         return false;
117     }
118     EnterCriticalSection(&m_cs);
119 
120     bResult=WriteFile(m_hCom,pData,(unsigned int)length,&bytesToSend,NULL);
121     if (bResult==FALSE)
122     {
123         DWORD dwErr=GetLastError();
124         PurgeComm(m_hCom,PURGE_TXABORT|PURGE_TXCLEAR);
125         LeaveCriticalSection(&m_cs);
126         return false;
127     }
128     LeaveCriticalSection(&m_cs);
129     return true;
130 }
131 
132 
133 bool CSerialPort::ReadData(char* pData)
134 {
135     DWORD bytes=0;
136     DWORD err=0;
137     COMSTAT cmt={0};
138     DWORD readSize=0;
139     bool bResult=false;
140     int bufferLen=COM_BUFFER_LENGTH;
141     int offset=0;
142 
143     //轮询读取串口数据
144     while (true)
145     {
146         //设置读取1个字节数据,当缓存中有数据到达时则会立即返回,否则直到超时 
147         bResult=ReadFile(m_hCom,m_data+offset,1,&readSize,NULL);
148         if (bResult==false||readSize==0)
149         {
150             break;
151         }
152         offset+=readSize;
153 
154         //若有数据到达,则bResult为true,同时readsize等于1
155         if (bResult==true&&readSize==1)
156         {
157             int readLen=0;
158             ClearCommError(m_hCom,&err,&cmt);
159             readLen=((int)cmt.cbInQue>bufferLen) ? bufferLen : cmt.cbInQue;
160             if (readLen>0)
161             {
162                 bResult=ReadFile(m_hCom,m_data+offset,readLen,&readSize,NULL);
163                 if (bResult==false||readSize<COM_BUFFER_LENGTH)
164                 {
165                     break;
166                 }
167                 offset+=readSize;
168                 PurgeComm(m_hCom,PURGE_RXABORT|PURGE_RXCLEAR);
169             }
170         }
171     }
172     memcpy(pData,m_data,COM_BUFFER_LENGTH);
173     return true;
174 }

 

posted @ 2018-01-23 20:20  夜雨翛然  阅读(2566)  评论(0编辑  收藏  举报