📄 ttydoc.cpp
字号:
/./ TTYDoc.cpp : implementation of the CTTYDoc class
//
#include "stdafx.h"
#include "TTY.h"
#include "ComsetDlg.h"
#include "TTYDoc.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/////////////////////////////////////////////////////////////////////////////
// CTTYDoc
IMPLEMENT_DYNCREATE(CTTYDoc, CDocument)
BEGIN_MESSAGE_MAP(CTTYDoc, CDocument)
//{{AFX_MSG_MAP(CTTYDoc)
ON_COMMAND(ID_API_SETTINGS, OnApiSettings)
ON_COMMAND(ID_API_CONNECT, OnApiConnect)
ON_COMMAND(ID_API_DISCONNECT, OnApiDisconnect)
ON_UPDATE_COMMAND_UI(ID_API_CONNECT, OnUpdateApiConnect)
ON_UPDATE_COMMAND_UI(ID_API_DISCONNECT, OnUpdateApiDisconnect)
ON_UPDATE_COMMAND_UI(ID_API_SETTINGS, OnUpdateApiSettings)
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CTTYDoc construction/destruction
CTTYDoc::CTTYDoc()
{
// TODO: add one-time construction code here
//Initial operation
m_bConnected = FALSE; //Serial port disconnected menu invalid
m_pThread = NULL;
m_nBaud = 9600;
m_nDataBits = 8;
m_bEcho = FALSE;
m_bNewLine = FALSE;
m_nParity = 0;
m_sPort = "COM1";
m_nStopBits = 0;
}
CTTYDoc::~CTTYDoc()
{
//Delete thread, close serial port at end of the program
if (m_bConnected)
{
CloseConnection();
}
//delete event handler
if (m_hPostMsgEvent)
{
CloseHandle(m_hPostMsgEvent);
}
if (m_osRead.hEvent)
{
CloseHandle(m_osRead.hEvent);
}
if (m_osWrite.hEvent)
{
CloseHandle(m_osWrite.hEvent);
}
}
BOOL CTTYDoc::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消息创建事件对象,手工重置,初始化为有信号的
//在Windows version 3.1中,当串口事件产生时,通讯设备驱动程序发送消息WM_COMMNOTIFY给Windows,
//该消息指示windows输入输出队列的状态。在基于Win32的应用程序中,已经不支持消息WM_COMMNOTIFY了。
//如果需要在Windows NT平台上运行基于Win16的应用程序,可以使用API EnableCommNotification()。
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;
}
/////////////////////////////////////////////////////////////////////////////
// CTTYDoc serialization
void CTTYDoc::Serialize(CArchive& ar)
{
// CEditView contains an edit control which handles all serialization
((CEditView*)m_viewList.GetHead())->SerializeRaw(ar);
}
/////////////////////////////////////////////////////////////////////////////
// CTTYDoc diagnostics
#ifdef _DEBUG
void CTTYDoc::AssertValid() const
{
CDocument::AssertValid();
}
void CTTYDoc::Dump(CDumpContext& dc) const
{
CDocument::Dump(dc);
}
#endif //_DEBUG
/////////////////////////////////////////////////////////////////////////////
// CTTYDoc commands
UINT Comproc(LPVOID pParam)
{
OVERLAPPED os;
DWORD dwMask, dwTrans;
COMSTAT ComStat;
DWORD dwErrorFlags;
CTTYDoc* pDoc = (CTTYDoc*)pParam;
memset(&os, 0, sizeof(OVERLAPPED));
os.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (os.hEvent == NULL)
{
AfxMessageBox("Can not 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;
}
//将指定数量的字符从串口输出
DWORD CTTYDoc::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;
}
//从串口输入缓冲区中读入指定数量的字符
DWORD CTTYDoc::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;
}
void CTTYDoc::OnApiConnect()
{
// TODO: Add your command handler code here
if (!OpenConnection())
{
AfxMessageBox("Can not open connection!");
}
}
void CTTYDoc::OnUpdateApiConnect(CCmdUI* pCmdUI)
{
// TODO: Add your command update UI handler code here
pCmdUI->Enable(!m_bConnected);
}
void CTTYDoc::OnApiDisconnect()
{
// TODO: Add your command handler code here
CloseConnection();
}
void CTTYDoc::OnUpdateApiDisconnect(CCmdUI* pCmdUI)
{
// TODO: Add your command update UI handler code here
pCmdUI->Enable(m_bConnected);
}
void CTTYDoc::OnApiSettings()
{
// TODO: Add your command handler code here
CComsetDlg dlg;
CString str;
dlg.m_bConnected = m_bConnected;
dlg.m_sPort = m_sPort;
str.Format("%d", m_nBaud);
dlg.m_sBaud = str;
str.Format("%d", m_nDataBits);
dlg.m_sDataBits = str;
dlg.m_nParity = m_nParity;
dlg.m_nStopBits = m_nStopBits;
dlg.m_bEcho = m_bEcho;
dlg.m_bNewLine = m_bNewLine;
if (dlg.DoModal() == IDOK)
{
m_sPort = dlg.m_sPort;
m_nBaud = atoi(dlg.m_sBaud);
m_nDataBits = atoi(dlg.m_sDataBits);
m_nParity = dlg.m_nParity;
m_nStopBits = dlg.m_nStopBits;
m_bEcho = dlg.m_bEcho;
m_bNewLine = dlg.m_bNewLine;
if (m_bConnected)
{
if (!ConfigConnection())
{
AfxMessageBox("Can not realize the settings!");
}
}
}
}
void CTTYDoc::OnUpdateApiSettings(CCmdUI* pCmdUI)
{
// TODO: Add your command update UI handler code here
}
BOOL CTTYDoc::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) //stop bit
{
case 0:
dcb.StopBits = ONESTOPBIT;
break;
case 1:
dcb.StopBits = ONE5STOPBITS;
break;
case 2:
dcb.StopBits = TWOSTOPBITS;
break;
default:
;
}
//硬件流控制设置
dcb.fOutxCtsFlow = TRUE;
dcb.fRtsControl = TRUE;
//XON/XOFF设置
dcb.fInX = dcb.fOutX = TRUE;
dcb.XonChar = XON;
dcb.XoffChar = XOFF;
dcb.XonLim = 50;
dcb.XoffLim = 50;
return SetCommState(m_hCom, &dcb);
}
//open and configure the serial port
BOOL CTTYDoc::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.ReadTotalTimeoutConstant = 0;
TimeOuts.ReadTotalTimeoutMultiplier = 0;
//设置超时以指定WriteComm成员函数中的GetOverlappedResult函数的等待时间
TimeOuts.WriteTotalTimeoutConstant = 2000;
TimeOuts.WriteTotalTimeoutMultiplier = 50;
SetCommTimeouts(m_hCom, &TimeOuts);
if (ConfigConnection())
{
m_pThread = AfxBeginThread(Comproc, 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 CTTYDoc::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);
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -