异步写串口会丢失指令的问题,谢谢

yifuzhiming 2013-12-04 08:58:42
写串口采用异步的方式,我用一个线程去写串口,发了100多条指令,只有20多条有发送出去,其它的都丢失了,我采用同步的方式就不会,我怀疑是不是发送太快导致的,但我给线程每次都sleep了500毫秒还是不行,请教下各位,谢谢了


bool OperatorComPort::WriteCom(unsigned char *str, int len)
{
DWORD dwNumWrite = 0;
if (NULL == m_hComm)
{
return FALSE;
}

if (WriteFile(m_hComm, str, (DWORD)len, &dwNumWrite, &m_osWrite))
{
return TRUE;
}

if (GetLastError() != ERROR_IO_PENDING)
{
return FALSE;
}
GetOverlappedResult(m_hComm, &m_osWrite, &dwNumWrite, TRUE);
if ((DWORD)len != dwNumWrite)
{
return FALSE;
}
return ((DWORD)len == dwNumWrite);
}


bool OperatorComPort::InitComport(int pPort)
{
DCB dcb; // 串口控制块

COMMTIMEOUTS timeouts;
//设置读超时
timeouts.ReadIntervalTimeout = 100;
timeouts.ReadTotalTimeoutMultiplier = 500; //500;
timeouts.ReadTotalTimeoutConstant = 2000;//2000;
//设置写超时
timeouts.WriteTotalTimeoutMultiplier = 500;
timeouts.WriteTotalTimeoutConstant = 2000;

wchar_t *szPort = new wchar_t[50];
swprintf(szPort, L"COM%d", pPort);

m_hComm = CreateFile(szPort, // 串口名称或设备路径
GENERIC_READ | GENERIC_WRITE, // 读写方式
0, // 共享方式:独占
NULL, // 默认的安全描述符
OPEN_EXISTING, // 创建方式
FILE_FLAG_OVERLAPPED,//异步操作
NULL); // 不需参照模板文件

if (m_hComm == INVALID_HANDLE_VALUE) // 打开串口失败
{
delete [] szPort;
//_snprintf(m_szErrorInfo, MAX_ERR_BUFF_LEN, "open %s error, not exist or already in use", pPort);
return FALSE;
}

if (0 == GetCommState(m_hComm, &dcb)) // 取DCB
{
delete [] szPort;
//_snprintf(m_szErrorInfo, MAX_ERR_BUFF_LEN, "open %s error, GetCommState failed!", pPort);
CloseCom();
return FALSE;
}

dcb.BaudRate = CBR_38400;
dcb.ByteSize = 8;
dcb.Parity = NOPARITY;
dcb.StopBits = ONESTOPBIT;

if (0 == SetCommState(m_hComm, &dcb)) // 设置DCB
{
delete [] szPort;
//_snprintf(m_szErrorInfo, MAX_ERR_BUFF_LEN, "open %s error, SetCommState failed! ErrorNo: %d", pPort, GetLastError());
CloseCom();
return FALSE;
}

if (0 == SetupComm(m_hComm, 1024, 1024)) // 设置输入输出缓冲区大小
{
delete [] szPort;
//_snprintf(m_szErrorInfo, MAX_ERR_BUFF_LEN, "open %s error, SetupComm failed! ErrorNo: %d", pPort, GetLastError());
CloseCom();
return FALSE;
}
//PurgeComm(m_hCom, PURGE_TXCLEAR | PURGE_RXCLEAR);

if (0 == SetCommTimeouts(m_hComm, &timeouts)) // 设置超时
{
delete [] szPort;
//_snprintf(m_szErrorInfo, MAX_ERR_BUFF_LEN, "open %s error, SetCommTimeouts failed! ErrorNo: %d", pPort, GetLastError());
CloseCom();
return FALSE;
}
PurgeComm(m_hComm, PURGE_RXCLEAR | PURGE_TXCLEAR | PURGE_RXABORT | PURGE_TXABORT);
delete [] szPort;
m_connectype = COM;
return TRUE;
}

...全文
88 3 打赏 收藏 转发到动态 举报
写回复
用AI写文章
3 条回复
切换为时间正序
请发表友善的回复…
发表回复
liquanhai 2013-12-04
  • 打赏
  • 举报
回复
给你个参数设置进去看看,应该是可以的:

	COMMTIMEOUTS CommTimeOuts;// 定义超时参数

	//GetCommTimeouts(m_hComm, &CommTimeOuts);		
	CommTimeOuts.ReadIntervalTimeout = MAXDWORD;	// 接收字符间最大时间间隔
	CommTimeOuts.ReadTotalTimeoutMultiplier = 0;		
	CommTimeOuts.ReadTotalTimeoutConstant = 0;	// 读数据总超时常量

	CommTimeOuts.WriteTotalTimeoutMultiplier = 50;
	CommTimeOuts.WriteTotalTimeoutConstant = 100;
向立天 2013-12-04
  • 打赏
  • 举报
回复
应该还是上一个还没法出去下一个就给覆盖了 你看看同步发这些指令需要多少时间 可能异步也不应该小于这个时间太多 否则处理不过来 异步并不能加快处理速度 只是可以有比较友好的界面相应
躺着睡的蜗牛 2013-12-04
  • 打赏
  • 举报
回复
有可能是串口命令用的同一个Buff导致的, 把串口命令放到一个队列中发试试。
单片机控制GSM模块实现短信收发的软件设计 摘要:借助系统模型,阐明GSM模块收发短信的基本概念以及串口控制SMS的基本原理。 详细介绍单片机控制GSM模块工作的软件实现过程,对怎样用单片机控制GSM模块收发短 信进行探讨,也对程序设计的主体思想作了较为细致的分析。 关键词:单片机 短信收发 软件设计 GSM(Global System for Mobile communication)系统是目前基于时分多址技术的移动通信体制中,比较成熟完善,且应 用最广泛的一种系统。目前已建成的覆盖全国的GSM数字蜂窝移动通信网,是我国公众移 动通信网的主要方式。基于GSM的短信信息服务,是一种在移动网络上传送简短信息的无 线应用,是一种信息在移动网络上存储和转寄的过程。由于公众GSM网络在全球范围内实 现了联网和漫游,建议上述系统不需再组建专用通信网络,所以具有实时传输数据功能 的短信应用将得到迅速普及。笔者开发设计的基于GSM网络的温度数据采集与无线传输系 统正是借助该网络平台,利用短信息业务实现数据的自动双向传递。系统模型图如图1所 示。 本系统由数据采集部分、数据接收和发送部分、终端处理部分三个模块组成。数据采 集模块将采集到的温度数据存入存储器中。数据收发模块采用双单片机共用E2RPOM的方 式,单片机2控制数据从存储器转存入E2PROM中;单片机1负责将数据从E2PROM中读出, 并经GSM模块2借助GSM网络将数据发送出去。单片机1不仅控制数据的发送,也控制数据 的接收。在这里,E2PROM是温度数据临时存储和上传的中转站。终端处理模块负责将接 收到的数据交给计算机处理,并将处理后的结果存放到数据库中,以供查询。当终端处 理模块需要向GSM模块2发送控制命令时,GSM模块2接收过程正好与上述过程相反,从而 实现数据的自动双向传递。 系统中,三个模块相互独立,彼此又相互依赖,共同完成数据的传输。数据收发模块在 系统中起着承上启下的作用,是系统的核心模块。该模块以双单片机为核心,以RS232通 信接口,在物理层上实现与GSM模块的连接。由于篇幅的限制,本文主要介绍单片机控制 这一模块工作的软件实现过程,旨在对怎样用单片机控制GSM模块收发短信息进行探讨。 1 GSM模块MZ28 MZ28是中兴通讯推出的GSM无线双频调制解调器,主要为语音传输、短信发送和数据 业务提供无线接口。MZ28集成了完整的射频电路和GSM的基带处理器,特别适合于迅速开 发基于GSM无线网络的无线应用产品。带有人机接口(MMI)界面的应用产品内部与MZ28 的通信可通过标准的串行接口(RS232)进行。MZ28使用简单的20-PIN ZIP插座与用户自己的应用系统相连,此ZIP连接方式提供开发所需的数据通信、音频和 电源等接口信号。MZ28可以作为无线引擎,嵌入到用户自己的产品当中,用户可以用单 片机或其它CPU的UART口,使用相应的AT命令,对模块进行控制,达到使其产品可以轻松 进入GSM网络的目的。 2 串口控制SMS的工作原理 单片机与GSM模块一般采用串行异步通信接口,通信速度可设定,通常为19200bps。 采用这种RSM232电缆方式进行连接时,数据传输的可靠性较好。RS232接口方式连接,通 过串行接口集成电路和电平转换电路与GSM模块连接,电路比较简单,所涉及的芯片包括 单片机89C52和电平转换芯片MAX232,是非常常见的接口电路。需要说明的是,该接口通 过I2C总线扩展了一个E2PROM存储器芯片AT24C64,它的主要作用是存储数据,而且断电 信息也不丢失,这些特性正是存储数据所必须的。 GSM的短信息业务SMS利用信令信道传输,这是GSM通信网所特有的。它不用拨号建立 连接,把要发的信息加上目的数据发送到短信息服务中心,经短信服务中心完成存储后 再发送给最终的信宿。所以当目的GSM终端没开机时信息不丢失。每个短信的信息量限 制为160字节。 现在市场上大多数手机均支持GSM07.05规定的AT指令集。该指令集是ETSI(欧洲通信 技术委员)发布的,其中包含了对SMS的控制。利用GSM手机的串行接口,单片机向手 机收发一系列的AT命令,就能达到控制GSM模块收发SMS的目的。必须注意的是,用单片 机实现时,编程必须注意它发送指令与接收到的响应都是字符的ASCII码。用单片机控制 GSM模块收发短信息所涉及以的AT指令如表1所列。 表1 AT指令 AT指令 功 能 描 述 AT+OFF 关机并重新启动 AT+CSDH=0 在TEXT模式下在返回值中不显示详细的头信息 ATE0 关闭回显 AT+CMGF=1 选择短信格式为TEXT模式 AT+CMGS 发送短信息 AT+CMGR 读取短信息 AT+CMGD=0 删除全部短信息 3

16,471

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Web++
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧