4116|7

67

帖子

0

TA的资源

一粒金砂(初级)

楼主
 

EVC下串口程序,急!!!! [复制链接]

按照ZLG的WINCE书编写了一个串口测试程序,做打开串口,简单的接收发送数据

应该跟硬件驱动没有直接关系

但是现在有些问题:
1、编译成功后用EVC自带的模拟器仿真,总是COM1可以打开,COM2-COM6都是“无法按当前参数配置端口,请检查参数!”
   VIA的X86BSP编译的SDK下编译成功后在实际的VIA的板子上运行又只有COM2可以打开,其他的都提示“无法按当前参数配置端口,请检查参数!”
2、一点发送就程序就死在那了

程序我找重要的粘贴如下:
**********************************
函数名称: CSerialPortDlg::OpenPort
描    述: 打开串口
输入参数: LPCTSTR Port: 串口名,如"COM0:","COM1:"
                  int BaudRate: 波特率
                  int DataBits: 数据位, 取值为7或8
                  int StopBits: 停止位
                  int Parity  : 奇偶校验位
***********************************
BOOL CSerialPortDlg::OpenPort(LPCTSTR Port, int BaudRate, int DataBits, int StopBits, int Parity)
{
        COMMTIMEOUTS CommTimeOuts;

        // 打开串口
        m_hComm = CreateFile(Port, GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, 0, 0);
        if(m_hComm == INVALID_HANDLE_VALUE)
        {
                MessageBox(_T("无法打开端口或端口已打开!请检查是否已被占用."));
                return FALSE;
        }

        GetCommState(m_hComm, &dcb);                                                /* 读取串口的DCB */
        dcb.BaudRate = BaudRate;                       
        dcb.ByteSize = DataBits;
        dcb.Parity = Parity;
        dcb.StopBits = StopBits;       
        dcb.fParity = FALSE;                                                                /* 禁止奇偶校验 */
        dcb.fBinary = TRUE;
        dcb.fDtrControl = 0;                                                                /* 禁止流量控制 */
        dcb.fRtsControl = 0;
        dcb.fOutX = 0;
        dcb.fInX = 0;
        dcb.fTXContinueOnXoff = 0;
       
        //设置状态参数
        SetCommMask(m_hComm, EV_RXCHAR);                                        /* 串口事件:接收到一个字符 */       
        SetupComm(m_hComm, 16384, 16384);                                        /* 设置接收与发送的缓冲区大小 */
        if(!SetCommState(m_hComm, &dcb))                                        /* 设置串口的DCB */
        {
                MessageBox(_T("无法按当前参数配置端口,请检查参数!"));
                ClosePort();
                return FALSE;
        }
               
        //设置超时参数
        GetCommTimeouts(m_hComm, &CommTimeOuts);               
        CommTimeOuts.ReadIntervalTimeout = 100;                                /* 接收字符间最大时间间隔 */
        CommTimeOuts.ReadTotalTimeoutMultiplier = 1;               
        CommTimeOuts.ReadTotalTimeoutConstant = 100;                /* 读数据总超时常量 */
        CommTimeOuts.WriteTotalTimeoutMultiplier = 0;
        CommTimeOuts.WriteTotalTimeoutConstant = 0;               
        if(!SetCommTimeouts(m_hComm, &CommTimeOuts))
        {
                MessageBox(_T("无法设置超时参数!"));
                ClosePort();
                return FALSE;
        }
               
        PurgeComm(m_hComm, PURGE_TXCLEAR | PURGE_RXCLEAR);         /* 清除收/发缓冲区 */               
        return TRUE;               
}
/**************************************************
函数名称: CALLBACK CSerialPortDlg::OnCommRecv
描    述: 串口接收数据成功回调函数
****************************************************/
void CALLBACK CSerialPortDlg::OnCommRecv(CWnd* pWnd, char *buf, int buflen)
{
        CString tmp;

        CSerialPortDlg * pDlg = (CSerialPortDlg*)pWnd;
        CEdit *pRecvStrEdit = (CEdit*)pDlg->GetDlgItem(IDC_REC_DISP);
                                                                                                                /* 取得控件指针 */
        for (int i = 0; i < buflen; i++, buf++)
        {
                tmp.Format(_T("%c"), *buf);                                                /* 将字符转换为字符串 */
                pDlg->m_strRecDisp += tmp;
        }

        pRecvStrEdit->SetWindowText(pDlg->m_strRecDisp);        /* 显示在窗口上 */
}
/*************************************************
函数名称: CSerialPortDlg::CommRecvTread
描    述: 串口接收线程
输入参数: LPVOID lparam: 线程参数,创建线程时传入
返    回: 0: 线程退出, 返回值没特殊含义
*************************************************/
DWORD CSerialPortDlg::CommRecvTread(LPVOID lparam)
{
        DWORD dwLength;
        char *recvBuf = new char[1024];
        CSerialPortDlg *pDlg = (CSerialPortDlg*)lparam;

        while(TRUE)
        {                                                                                                                                /* 等待线程退出事件 */
                if (WaitForSingleObject(pDlg->m_ExitThreadEvent, 0) == WAIT_OBJECT_0)
                        break;       

                if (pDlg->m_hComm != INVALID_HANDLE_VALUE)
                {                                                                                                                        /* 从串口读取数据 */
                        BOOL fReadState = ReadFile(pDlg->m_hComm, recvBuf, 1024, &dwLength, NULL);
                        if(!fReadState)
                        {
                                //MessageBox(_T("无法从串口读取数据!"));
                        }
                        else
                        {
                                if(dwLength != 0)
                                        OnCommRecv(pDlg, recvBuf, dwLength);                        /* 接收成功调用回调函数 */
                        }
                }
        }               

        delete[] recvBuf;
        return 0;
}


// 定义串口设置参数表格
const CString PorTbl[6] = {_T("COM1:"),_T("COM2:"),_T("COM3:"),_T("COM4:"),_T("COM5:"), _T("COM6:")};
const DWORD BaudTbl[6] = {4800, 9600, 19200, 38400, 57600,115200};       
const DWORD DataBitTbl[2] = {7, 8};
const BYTE  StopBitTbl[3] = {ONESTOPBIT, ONE5STOPBITS, TWOSTOPBITS};
const BYTE  ParityTbl[4] = {NOPARITY, ODDPARITY, EVENPARITY, MARKPARITY};


/*************************************
函数名称: CSerialPortDlg::OnOpenCom
描    述: "打开端口" 按键单击事件代码
*************************************/
void CSerialPortDlg::OnOpenCom()
{
        DWORD IDThread;
        HANDLE hRecvThread;                                                                                                /* 接收线程句柄 */
        UpdateData(TRUE);

        CString strPort = PorTbl[m_ComboPort.GetCurSel()];                                /* 查表获取参数值 */
        DWORD baud                = BaudTbl[m_ComboBaud.GetCurSel()];
        DWORD databit   = DataBitTbl[m_ComboData.GetCurSel()];
        BYTE stopbit    = StopBitTbl[m_ComboStop.GetCurSel()];
        BYTE parity                = ParityTbl[m_ComboParity.GetCurSel()];

        BOOL ret = OpenPort(strPort, baud, databit, stopbit, parity);        /* 打开串口 */
        if (ret == FALSE)
                return;

        m_ExitThreadEvent = CreateEvent(NULL, TRUE, FALSE, NULL);                /* 创建串口接收线程退出事件*/

        // 创建串口接收线程
        hRecvThread = CreateThread(0, 0, CommRecvTread, this, 0, &IDThread);
        if (hRecvThread == NULL)
        {
                MessageBox(_T("创建接收线程失败!"));
                return;
        }       
        CloseHandle(hRecvThread);

        m_ButOpen.EnableWindow(FALSE);                                                                        /* 打开端口按键禁止 */                               
        m_ButClose.EnableWindow(TRUE);                                                                        /* 关闭端口按键使能 */       
        MessageBox(_T("打开") + strPort + _T("成功!"));
}
/******************************************
函数名称: CSerialPortDlg::OnOpenCom
描    述: "关闭端口" 按键单击事件代码
******************************************/
void CSerialPortDlg::OnCloseCom()
{
        if (m_ExitThreadEvent != NULL)
        {
                SetEvent(m_ExitThreadEvent);                                        /* 通知线程退出 */
                Sleep(1000);
                CloseHandle(m_ExitThreadEvent);
                m_ExitThreadEvent = NULL;
        }

        m_ButOpen.EnableWindow(TRUE);                                                /* 打开端口按键禁止 */                               
        m_ButClose.EnableWindow(FALSE);                                                /* 关闭端口按键使能 */       
        ClosePort();
}
/****************************************
函数名称: CSerialPortDlg::OnOpenCom
描    述: "关闭端口" 按键单击事件代码
******************************************/
void CSerialPortDlg::OnSend()
{
        DWORD dwactlen;

        if (m_hComm == INVALID_HANDLE_VALUE)
        {
                MessageBox(_T("串口未打开!"));
                return;
        }

        UpdateData(TRUE);
        int len = m_strSendEdit.GetLength();                                 /* 取得输入字符串长度 */
        char *psendbuf = new char[len];

        for(int i = 0; i < len;i++)
                psendbuf = (char)m_strSendEdit.GetAt(i);                 /* 转换为单字节字符 */

        WriteFile(m_hComm, psendbuf, len, &dwactlen, NULL);         /* 从串口发送数据 */
       
        delete[] psendbuf;
}
/***************************************
函数名称: CSerialPortDlg::OnClearSend
描    述: "清除发送缓冲区" 按键单击事件代码
************************************/
void CSerialPortDlg::OnClearSend()
{
        m_strSendEdit = _T("");                /* 清除发送区的字符 */
        UpdateData(FALSE);
}
/********************************************
函数名称: CSerialPortDlg::OnClearRec
描    述: "清除接收缓冲区" 按键单击事件代码
*********************************************/
void CSerialPortDlg::OnClearRec()
{
        m_strRecDisp = _T("");                                       
        SetDlgItemText(IDC_REC_DISP,m_strRecDisp);                /* 清除接收区的字符 */
}
请问程序是不是可以动态的选择COM口??
应该不是写死的吧
这个跟驱动有关系吗??
该程序本来是ZLGMagic2410的串口程序
我现在重新用X86编译,在X86的板子上也可以运行该程序,可是不能通信,是本应用程序的问题还是需要修改X86下的串口驱动??(也就是说该应用程序跟驱动有直接关系吗?)
PB制定系统时是不是需要添加一些串口的支持??

最新回复

怎么才能上传附件??  详情 回复 发表于 2008-7-24 21:16
点赞 关注

回复
举报

61

帖子

0

TA的资源

一粒金砂(初级)

沙发
 
shuiyan yashi lenux.......等高手请帮助小弟啊
急!!!!!!!!!
 
 

回复

81

帖子

0

TA的资源

一粒金砂(初级)

板凳
 

是不是要确认一下你的模拟器里是不是有除了com1以外的串口?
 
 
 

回复

75

帖子

0

TA的资源

一粒金砂(初级)

4
 

急能解决问题吗?


看你贴了一大堆,都不知道你要说什么?



1、调试硬件等,尽量不要在模拟器中进行;

2、开发板上的 COM1 打不开,可能是因为已经被占用了,一般用作输出信息的;

3、开发板上有几个串行口那要看说明,一般也就3个。
 
 
 

回复

73

帖子

0

TA的资源

一粒金砂(初级)

5
 
看程序COM口没有写死
串口应用程序和驱动没有什么关系,都是调用系统api,只要驱动没有问题就好
PB定制的时候只要加上串口驱动就可以了
 
 
 

回复

69

帖子

0

TA的资源

一粒金砂(初级)

6
 
哦非常感谢各位
!!!
1、TO:zzhll“ 是不是要确认一下你的模拟器里是不是有除了com1以外的串口?”
        请问怎么确定模拟器上有几个串口??
2、TO:zaodt
   我不急了!
   COM1打不开可能是已经被占用了,X86下的BSP看不到驱动原程序.....
   开发板上有4串口,只有COM2、COM3可以成功打开
   但是打开之后,在发送区发送数据程序就死在那了,但是系统还没死机。

  发送按钮响应函数如下:
void CSerialPortDlg::OnSend()
{
DWORD dwactlen;

if (m_hComm == INVALID_HANDLE_VALUE)
{
MessageBox(_T("串口未打开!"));
return;
}

UpdateData(TRUE);
int len = m_strSendEdit.GetLength(); /* 取得输入字符串长度 */
char *psendbuf = new char[len];

for(int i = 0; i < len;i++)
psendbuf = (char)m_strSendEdit.GetAt(i); /* 转换为单字节字符 */

WriteFile(m_hComm, psendbuf, len, &dwactlen, NULL); /* 从串口发送数据 */

delete[] psendbuf;

3、 TO:Reallyu:“PB定制的时候只要加上串口驱动”
    我加了几乎所有关于关键字“serial”的组件,也加了COM16550

是不是可以打开COM2,COM3就说明CE系统已经有串口驱动了??
为什么打开之后发送数据死机呢???


另外我可以把串口的‘2’、‘3’脚短接实现自发自收吧??
 
 
 

回复

64

帖子

0

TA的资源

一粒金砂(初级)

7
 
uP各位高手帮忙
 
 
 

回复

67

帖子

0

TA的资源

一粒金砂(初级)

8
 
怎么才能上传附件??
 
 
 

回复
您需要登录后才可以回帖 登录 | 注册

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/6 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表