OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 8239|回复: 23

串口通讯原子自带的协议求帮解析

[复制链接]

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
发表于 2016-5-11 08:47:41 | 显示全部楼层 |阅读模式
1金钱
本帖最后由 加菲老爷 于 2016-5-11 08:53 编辑

以前学串口的时候,其中中断服务函数例程自带的协议没有太多的探索,现在有个项目需要类似的方法处理RS232接收的数据,发现看不太懂,又重新看了一遍例程和PDF电子书 也没有过多的叙述,大神们如果有空就教教我呀,库函数版本,因为要处理发过来的多个数据,需要把接收的数据放入数组中,我先用的例程的485程序 ,按理来说去掉控制引脚应该可以直接用于232上的,结果485收发多字节存入数组中等一切正常,232就接收异常,总是丢数据。。所以想用这个协议,但是发现有些看不懂
[mw_shl_code=applescript,true]void USART1_IRQHandler(void)                        //串口1中断服务程序
        {
        u8 Res;
#ifdef OS_TICKS_PER_SEC                 //如果时钟节拍数定义了,说明要使用ucosII了.
        OSIntEnter();   
#endif
        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)
                {
                Res =USART_ReceiveData(USART1);//(USART1->DR);        //读取接收到的数据
               
                if((USART_RX_STA&0x8000)==0)//接收未完成
                        {
                        if(USART_RX_STA&0x4000)//接收到了0x0d
                                {
                                if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
                                else USART_RX_STA|=0x8000;        //接收完成了
                                }
                        else //还没收到0X0D
                                {       
                                if(Res==0x0d)USART_RX_STA|=0x4000;
                                else
                                        {
                                        USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
                                        USART_RX_STA++;
                                        if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收          
                                        }                 
                                }
                        }                    
     }
#ifdef OS_TICKS_PER_SEC                 //如果时钟节拍数定义了,说明要使用ucosII了.
        OSIntExit();                                                                                           
#endif
}
#endif        [/mw_shl_code]

代码截图

代码截图
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

29

主题

311

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1530
金钱
1530
注册时间
2012-9-4
在线时间
262 小时
发表于 2016-5-11 08:51:41 | 显示全部楼层
直接把if(........)里面的都删了就行了,自己写呗。。。。。。
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-11 08:54:23 | 显示全部楼层
aben 发表于 2016-5-11 08:51
直接把if(........)里面的都删了就行了,自己写呗。。。。。。

有点不知道怎么写,水平太低了
回复

使用道具 举报

29

主题

311

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1530
金钱
1530
注册时间
2012-9-4
在线时间
262 小时
发表于 2016-5-11 08:59:33 | 显示全部楼层
Res =USART_ReceiveData(USART1);//(USART1->DR);  你把Res改成你需要的缓存数组,比如ReceiveBuff[i++]=USART_ReceiveData(USART1);,这句后面的程序都是为了判断接受完成用的,所以接受完成这个判断你可以自己做,比如说你的232数据是按照一定的协议发的,就比较好判断结束
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-11 09:09:48 | 显示全部楼层
aben 发表于 2016-5-11 08:59
Res =USART_ReceiveData(USART1);//(USART1->DR);  你把Res改成你需要的缓存数组,比如ReceiveBuff=USART_Re ...

好的谢谢,那比如我的帧尾固定为0xaa 应该如何做呢, C语言实在是有点糟糕,没有系统学过
回复

使用道具 举报

29

主题

311

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1530
金钱
1530
注册时间
2012-9-4
在线时间
262 小时
发表于 2016-5-11 09:17:50 | 显示全部楼层
加菲老爷 发表于 2016-5-11 09:09
好的谢谢,那比如我的帧尾固定为0xaa 应该如何做呢, C语言实在是有点糟糕,没有系统学过

if(ReceiveBuff==0xaa)
{
   i=0;//重新接受下一组
   sign=1;//接受完成标志位置1 在main()中判断这个标志位 然后处理接受的数据
               //再将sign=0
}
STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32
回复

使用道具 举报

29

主题

311

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1530
金钱
1530
注册时间
2012-9-4
在线时间
262 小时
发表于 2016-5-11 09:19:02 | 显示全部楼层
if(ReceiveBuff[i]==0xaa)
STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32
回复

使用道具 举报

29

主题

311

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1530
金钱
1530
注册时间
2012-9-4
在线时间
262 小时
发表于 2016-5-11 09:20:46 | 显示全部楼层
细节你还需要在调试中修改
STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-11 13:16:39 | 显示全部楼层
aben 发表于 2016-5-11 09:20
细节你还需要在调试中修改

发现232总是接收数据丢失,有什么办法可以解决呢
回复

使用道具 举报

29

主题

311

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1530
金钱
1530
注册时间
2012-9-4
在线时间
262 小时
发表于 2016-5-11 14:14:09 | 显示全部楼层
你先检查一下电路没有什么问题,232芯片外围电路什么的。。。

我经常用的方法是(比较繁琐):用一个好的usb转ttl串口(开发板上有,供地别忘了),去接收232芯片转化成ttl信号,打印到串口助手上,相当于开两个不同com口的串口助手(或是你其它设备发232数据),一个发232转ttl,另一个接收ttl转 usb(或232 确保这个是好的),就能看出你发送和接收的数据是否一致,这样就可以找到你数据丢失的原因是在232硬件还是软件。。。我的方法不需要软件参与。。。
STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32---STM32
回复

使用道具 举报

69

主题

978

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3774
金钱
3774
注册时间
2015-4-26
在线时间
766 小时
发表于 2016-5-11 15:40:50 | 显示全部楼层
加菲老爷 发表于 2016-5-11 13:16
发现232总是接收数据丢失,有什么办法可以解决呢

换条好一点的232线
我有故事,你有酒吗
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-11 17:49:18 | 显示全部楼层

线是公司买的  四五十块一根有应该已经很好了吧
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-11 17:50:13 | 显示全部楼层
aben 发表于 2016-5-11 14:14
你先检查一下电路没有什么问题,232芯片外围电路什么的。。。

我经常用的方法是(比较繁琐):用一个好 ...

用公司开发的板子调 调试口只有232 没有TTL。。
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2016-5-11 20:01:09 | 显示全部楼层
这个我们教程都有说啊。看我们开发板的教程吧。
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-12 08:41:55 | 显示全部楼层
正点原子 发表于 2016-5-11 20:01
这个我们教程都有说啊。看我们开发板的教程吧。

原子哥 这个我都看了的呀,都是一带而过,这一块木有讲的呀
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2016-5-13 21:34:56 | 显示全部楼层
加菲老爷 发表于 2016-5-12 08:41
原子哥 这个我都看了的呀,都是一带而过,这一块木有讲的呀

你现在用485接收?
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

3

主题

10

帖子

0

精华

初级会员

Rank: 2

积分
60
金钱
60
注册时间
2016-4-28
在线时间
8 小时
发表于 2016-5-13 21:57:17 | 显示全部楼层
    uint8_t i,res,check_temp;
    uint8_t length_temp[2];

     if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)
                 {
                                           res=USART_ReceiveData(USART2);                                        //读取接收到的数据
                                     if(xbee_rx_enableflag==1)                                                        //到接受数据标志置位时,接受数据
                                          {                                                                       
                                                        res=USART_ReceiveData(USART2);                                //读取串口标志
                                                        if(xbee_rd_lentemp<2)                                                //包头后第一和第二个数据为 数据的长度
                                                                {
                                                                    if(xbee_rd_lentemp==0)
                                                                    length_temp[0]=res;
                                                                        if(xbee_rd_lentemp==1)
                                                                        {
                                                                    length_temp[1]=res;
                                                                        xbee_rd_len=(length_temp[0]*256)+length_temp[1];
                                                                        }
                                                                        if(xbee_rd_len>=40)
                                                                        {

                                                                                xbee_rx_successflag=0;                        //数据接收成功标志清零
                                                                                xbee_rx_enableflag=0;                        //数据接收完成,数据接收启动标志清零
                                                                                xbee_rd_len=0;                  //数据长度清零
                                                                                xbee_rd_lentemp=0;                                //数据长度暂存器清零                                                                                                                                                               
                                                                        }
                                                                }
               
                                                        else if(xbee_rd_lentemp==xbee_rd_len+2)        //当读取到第usart_rd_lentemp+2个数据时,校验和
                                                                {
                                                                  check_temp=0;
                                                                  for(i=0;i<xbee_rd_len;i++)
                                                                  {
                                                                   check_temp =check_temp+xbee_rx_temporary[i];
                                                                  }
                                                                        check_temp=(0xff-(check_temp&0xff));        //0xff-(数据和)
                                                                        if(res==check_temp)                                        //当数据长度校验正确时
                                                                                xbee_rx_successflag=1;                //数据接收成功 校验标志置一
                                                                        else                                                                                                                                                                        
                                                                        {                                                                      //当数据长度校验错误时

                                                                                xbee_rx_successflag=0;                        //数据接收成功标志清零
                                                                                xbee_rx_enableflag=0;                        //数据接收完成,数据接收启动标志清零
                                                                                xbee_rd_len=0;                  //数据长度清零
                                                                                xbee_rd_lentemp=0;                                //数据长度暂存器清零
                                                                        }
                                                                }
                                                       
                                                        else xbee_rx_temporary[xbee_rd_lentemp-2]=res; //当usart_rd_lentemp为数据段时,将数据存到串口数据接收寄存器中       
                                                               
                                                         xbee_rd_lentemp++;                                                                //每次记录数据,数据长度暂存器自加       
                                                if(xbee_rx_successflag==1)                //如果成功接收到信息数据,将缓存usart_rx_temporary[]内的数据传递给usart_rx_buf[][]
                                                        {
                                                                        //接收处理程序
                                                                        XBee_Processing();
                                                                                xbee_rx_successflag=0;                        //数据接收成功标志清零
                                                                                xbee_rx_enableflag=0;                        //数据接收完成,数据接收启动标志清零
                                                                                xbee_rd_len=0;                  //数据长度清零
                                                                                xbee_rd_lentemp=0;                                //数据长度暂存器清零
                                                        }                       
                                        }
                                                if((res==0x7e)&&(xbee_rx_enableflag==0))                //当接受到包头(0x7e)数据并且还没有成功接收完数据信息
                                                xbee_rx_enableflag=1;
                 }
          USART_ClearITPendingBit(USART2, USART_IT_RXNE);//中断清除

//多看一点协议就好了。 - -
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-16 08:55:34 | 显示全部楼层

用232收发的
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2016-5-18 21:46:38 | 显示全部楼层

那不影响啊,一样可以用
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-19 09:36:06 | 显示全部楼层
正点原子 发表于 2016-5-18 21:46
那不影响啊,一样可以用

现有个小疑问 为什么232接收数据的时候 总丢失 在115200下  9600无异常
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2016-5-20 21:10:27 | 显示全部楼层
加菲老爷 发表于 2016-5-19 09:36
现有个小疑问 为什么232接收数据的时候 总丢失 在115200下  9600无异常

说明高波特率的时候,有误码了。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

4

主题

24

帖子

0

精华

初级会员

Rank: 2

积分
135
金钱
135
注册时间
2014-9-17
在线时间
20 小时
 楼主| 发表于 2016-5-24 17:36:38 | 显示全部楼层
正点原子 发表于 2016-5-20 21:10
说明高波特率的时候,有误码了。

这个问题要怎么解决呢,
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2016-5-25 20:55:41 | 显示全部楼层
加菲老爷 发表于 2016-5-24 17:36
这个问题要怎么解决呢,

检查下你的USB转串口线是否有问题
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

8

主题

50

帖子

0

精华

初级会员

Rank: 2

积分
119
金钱
119
注册时间
2017-3-12
在线时间
27 小时
发表于 2017-6-10 22:15:08 | 显示全部楼层
aben 发表于 2016-5-11 08:59
Res =USART_ReceiveData(USART1);//(USART1->DR);  你把Res改成你需要的缓存数组,比如ReceiveBuff=USART_Re ...

大神 请问 在usart.c里面是不是要先定义 u16 ReceiveBuff[100];  u8 i; 这样呢?
回复

使用道具 举报

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

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-8-18 11:48

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表