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

📄 terminaldoc.cpp

📁 一个操纵RS232端口的程序
💻 CPP
字号:
// TerminalDoc.cpp : implementation of the CTerminalDoc class
//

#include "stdafx.h"
#include "Terminal.h"
#include "TerminalDoc.h"
#include "DlgComSettings.h"

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

/////////////////////////////////////////////////////////////////////////////
// CTerminalDoc

IMPLEMENT_DYNCREATE(CTerminalDoc, CDocument)

BEGIN_MESSAGE_MAP(CTerminalDoc, CDocument)
	//{{AFX_MSG_MAP(CTerminalDoc)
	ON_COMMAND(IDM_FILE_CONNECT, OnFileConnect)
	ON_COMMAND(IDM_FILE_DISCONNECT, OnFileDisconnect)
	ON_COMMAND(IDM_FILE_SETTINGS, OnFileSettings)
	//}}AFX_MSG_MAP
END_MESSAGE_MAP()

/////////////////////////////////////////////////////////////////////////////
// CTerminalDoc construction/destruction

CTerminalDoc::CTerminalDoc()
{
	m_bConnected = FALSE;	
	m_pThread = NULL;
	m_nBaud = 9600;	
	m_nDataBits = 8;		
	m_nFlowCtrl = 0;
	m_bEcho = FALSE;
	m_bNewLine = FALSE;	
	m_nParity = 0;	
	m_sPort = "COM1";	
	m_nStopBits = 1;
}

CTerminalDoc::~CTerminalDoc()
{
	if(m_bConnected)		
		CloseConnection();
	
	// 删除事件句柄	
	if(m_hPostMsgEvent)		
		CloseHandle(m_hPostMsgEvent);
	
	if(m_osRead.hEvent)		
		CloseHandle(m_osRead.hEvent);
	
	if(m_osWrite.hEvent)		
		CloseHandle(m_osWrite.hEvent);
}

BOOL CTerminalDoc::OnNewDocument()
{
	if (!CDocument::OnNewDocument())
		return FALSE;

	((CEditView*)m_viewList.GetHead())->SetWindowText(NULL);

	// TODO: add reinitialization code here
	// (SDI documents will reuse this document)
	
	// 为WM_COMMNOTIFY消息创建事件对象,手工重置,初始化为有信号的	
	if((m_hPostMsgEvent = CreateEvent(NULL, TRUE, TRUE, NULL)) == NULL)		
		return FALSE;
	
	memset(&m_osRead, 0, sizeof(OVERLAPPED));	
	memset(&m_osWrite, 0, sizeof(OVERLAPPED));
	
	// 为重叠读创建事件对象,手工重置,初始化为无信号的	
	if((m_osRead.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL)) == NULL)		
		return FALSE;
	
	// 为重叠写创建事件对象,手工重置,初始化为无信号的	
	if((m_osWrite.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL))==NULL)		
		return FALSE;

	return TRUE;
}



/////////////////////////////////////////////////////////////////////////////
// CTerminalDoc serialization

void CTerminalDoc::Serialize(CArchive& ar)
{
	// CEditView contains an edit control which handles all serialization
	((CEditView*)m_viewList.GetHead())->SerializeRaw(ar);
}

/////////////////////////////////////////////////////////////////////////////
// CTerminalDoc diagnostics

#ifdef _DEBUG
void CTerminalDoc::AssertValid() const
{
	CDocument::AssertValid();
}

void CTerminalDoc::Dump(CDumpContext& dc) const
{
	CDocument::Dump(dc);
}
#endif //_DEBUG

/////////////////////////////////////////////////////////////////////////////
// CTerminalDoc commands

void CTerminalDoc::OnFileConnect() 
{
	// TODO: Add your command handler code here
	if(!OpenConnection())
		AfxMessageBox("Can't open connection");
}

void CTerminalDoc::OnFileDisconnect() 
{
	// TODO: Add your command handler code here
	CloseConnection(); 	
}

void CTerminalDoc::OnFileSettings() 
{
	// TODO: Add your command handler code here
 	CDlgComSettings dlg;
	CString str;
	dlg.m_bConnected = m_bConnected;
	dlg.m_strPort = m_sPort;
	str.Format("%d",m_nBaud);
	dlg.m_strBaud = str;
 	str.Format("%d",m_nDataBits);
	dlg.m_strDataBits = str;
	str.Format("%d", m_nParity);
	if(m_nParity == 0)
		str = "None";
	else if(m_nParity == 1)
		str = "Even";
	else
		str = "Odd";
	dlg.m_strParity = str;
	str.Format("%d", m_nStopBits);
	dlg.m_strStopBits = str;
	dlg.m_iFlowCtrl = m_nFlowCtrl;
 	dlg.m_bEcho = m_bEcho;
 	dlg.m_bNewLine = m_bNewLine;
	
 	if(dlg.DoModal()==IDOK)
 	{
 		m_sPort = dlg.m_strPort;
 		m_nBaud = atoi(dlg.m_strBaud);
 		m_nDataBits = atoi(dlg.m_strDataBits);
		if(dlg.m_strParity == "None")
			m_nParity = 0;
		else if(dlg.m_strParity == "Even")
			m_nParity = 1;
		else
			m_nParity = 2;
//		m_nStopBits = atoi(dlg.m_strStopBits);
		if(dlg.m_strStopBits == "1")
			m_nStopBits = 0;
		else if(dlg.m_strStopBits == "1.5")
			m_nStopBits = 1;
		else
			m_nStopBits = 2;
 		m_nFlowCtrl = dlg.m_iFlowCtrl;
 		m_bEcho = dlg.m_bEcho;
 		m_bNewLine = dlg.m_bNewLine;
 		if(m_bConnected)
 			if(!ConfigConnection())
 				AfxMessageBox("Can't realize the settings!");
 	}	
}

// 打开并配置串行口,建立工作者线程
BOOL CTerminalDoc::OpenConnection()
{	
	COMMTIMEOUTS TimeOuts;	
	POSITION firstViewPos;	
	CView *pView;			
	firstViewPos = GetFirstViewPosition();	
	pView = GetNextView(firstViewPos);	
	m_hTermWnd = pView->GetSafeHwnd();
	
	if(m_bConnected)
		return FALSE;
	
	m_hCom = CreateFile(m_sPort, GENERIC_READ | GENERIC_WRITE, 0, NULL,		
		OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, 	
		NULL); // 重叠方式
	if(m_hCom == INVALID_HANDLE_VALUE)	
		return FALSE;
	
	SetupComm(m_hCom, MAXBLOCK, MAXBLOCK);	
	SetCommMask(m_hCom, EV_RXCHAR);

	// 把间隔超时设为最大,把总超时设为0将导致ReadFile立即返回并完成操作
	TimeOuts.ReadIntervalTimeout = MAXDWORD; 	
	TimeOuts.ReadTotalTimeoutMultiplier = 0; 
	TimeOuts.ReadTotalTimeoutConstant = 0; 
	
	/* 设置写超时以指定WriteComm成员函数中的
	GetOverlappedResult函数的等待时间*/	
	TimeOuts.WriteTotalTimeoutMultiplier = 50; 	
	TimeOuts.WriteTotalTimeoutConstant = 2000;	
	SetCommTimeouts(m_hCom, &TimeOuts);
	if(ConfigConnection())	
	{	
		m_pThread = AfxBeginThread(CommProc, this, THREAD_PRIORITY_NORMAL, 0, CREATE_SUSPENDED, NULL); // 创建并挂起线程			
		if(m_pThread == NULL)				
		{				
			CloseHandle(m_hCom);				
			return FALSE;				
		}			
		else				
		{				
			m_bConnected = TRUE;				
			m_pThread->ResumeThread(); // 恢复线程运行				
		}			
	}		
	else			
	{			
		CloseHandle(m_hCom);			
		return FALSE;			
	}		
	return TRUE;		
}

// 结束工作者线程,关闭串行口
void CTerminalDoc::CloseConnection()
{
	if(!m_bConnected)
		return;

	m_bConnected = FALSE;
	
	//结束CommProc线程中WaitSingleObject函数的等待
	SetEvent(m_hPostMsgEvent);

	//结束CommProc线程中WaitCommEvent的等待
	SetCommMask(m_hCom, 0);

	//等待辅助线程终止
	WaitForSingleObject(m_pThread->m_hThread, INFINITE);
	
	m_pThread = NULL;
	CloseHandle(m_hCom);
}

// 配置串行口
BOOL CTerminalDoc::ConfigConnection()
{
	DCB dcb;
	if(!GetCommState(m_hCom, &dcb))
		return FALSE;

	dcb.fBinary = TRUE;
	dcb.BaudRate = m_nBaud; // 波特率
	dcb.ByteSize = m_nDataBits; // 每字节位数
	dcb.fParity = TRUE;
	switch(m_nParity) // 校验设置
	{
	case 0:
		dcb.Parity = NOPARITY;
		break;
	case 1:
		dcb.Parity = EVENPARITY;
		break;
	case 2:
		dcb.Parity = ODDPARITY;
		break;
	default:
		;
	}
	
	switch(m_nStopBits) // 停止位
	{
	case 0:
		dcb.StopBits = ONESTOPBIT;
		break;
	case 1:
		dcb.StopBits = ONE5STOPBITS;
		break;
	case 2:
		dcb.StopBits = TWOSTOPBITS;
		break;
	default:
		;
	}
	
	// 硬件流控制设置
	dcb.fOutxCtsFlow = m_nFlowCtrl == 1;
	dcb.fRtsControl = m_nFlowCtrl == 1 ? RTS_CONTROL_HANDSHAKE : RTS_CONTROL_ENABLE;
	
	// XON/XOFF流控制设置
	dcb.fInX = dcb.fOutX = m_nFlowCtrl == 2;
	dcb.XonChar = XON;
	dcb.XoffChar = XOFF;
	dcb.XonLim = 50;
	dcb.XoffLim = 50;
	return SetCommState(m_hCom, &dcb);
}

// 从串行口输入缓冲区中读入指定数量的字符
DWORD CTerminalDoc::ReadComm(char *buf, DWORD dwLength)
{
	DWORD length = 0;
	COMSTAT ComStat;
	DWORD dwErrorFlags;
	ClearCommError(m_hCom, &dwErrorFlags, &ComStat);
	length = min(dwLength, ComStat.cbInQue);
	ReadFile(m_hCom, buf, length, &length, &m_osRead);
	return length;
}

// 将指定数量的字符从串行口输出
DWORD CTerminalDoc::WriteComm(char *buf, DWORD dwLength)
{
	BOOL fState;
	DWORD length = dwLength;
	COMSTAT ComStat;
	DWORD dwErrorFlags;
	ClearCommError(m_hCom, &dwErrorFlags, &ComStat);
	fState=WriteFile(m_hCom, buf, length, &length, &m_osWrite);
	if(!fState)
	{
		if(GetLastError() == ERROR_IO_PENDING)
		{
			GetOverlappedResult(m_hCom, &m_osWrite, &length, TRUE);// 等待
		}
		else
			length = 0;
	}
	return length;
}

// 工作者线程,负责监视串行口
UINT CommProc(LPVOID pParam)
{
	OVERLAPPED os;
	DWORD dwMask, dwTrans;
	COMSTAT ComStat;
	DWORD dwErrorFlags;
	CTerminalDoc *pDoc = (CTerminalDoc*)pParam;
	memset(&os, 0, sizeof(OVERLAPPED));
	os.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
	if(os.hEvent == NULL)
	{
		AfxMessageBox("Can't create event object!");
		return (UINT)-1;
	}
	
	while(pDoc->m_bConnected)
	{
		ClearCommError(pDoc->m_hCom, &dwErrorFlags, &ComStat);
		if(ComStat.cbInQue)
		{
			// 无限等待WM_COMMNOTIFY消息被处理完
			WaitForSingleObject(pDoc->m_hPostMsgEvent, INFINITE);
			ResetEvent(pDoc->m_hPostMsgEvent);
			// 通知视图
			PostMessage(pDoc->m_hTermWnd, WM_COMMNOTIFY, EV_RXCHAR, 0);
			continue;
		}
		dwMask = 0;
		if(!WaitCommEvent(pDoc->m_hCom, &dwMask, &os)) // 重叠操作
		{
			if(GetLastError() == ERROR_IO_PENDING)
				// 无限等待重叠操作结果
				GetOverlappedResult(pDoc->m_hCom, &os, &dwTrans, TRUE);
			else
			{
				CloseHandle(os.hEvent);
				return (UINT)-1;
			}
		}
	}
	
	CloseHandle(os.hEvent);
	return 0;
}





⌨️ 快捷键说明

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