串口发送与接收数据

2016-12-12  本文已影响829人  c与php

使用RS-485串口进行通讯。
1.定义串口接收数据的缓冲区,最大可以保存64个字节u8 RS485_RX_BUF1[64];
2.定义接收发送数据的长度 u8 RS485_RX_CNT;
3.发送数据的函数一般有两个printf和 USART_SendData,这里主要说USART_SendData的使用。printf实现的是格式化字符串,字符串比较有优势。
USART_SendData传递单个字符和指令。
4.给发送方定义一个标记位u32 flags_send1; flags_send1

5.要发送的事件很多定义一个枚举类型的数据,将所有要发送的事件放入枚举类型数据中。

typedef enum
{
    E_ERROR                 = 0,
    E_HKEYDN_RESET      ,    //1
    
    E_HKEYUP_RESET1     , //2
    E_HKEYUP_EMER1      , //3
    E_HKEYUP_PA1            , //4
    E_HKEYDN_PTT1           , //5
    E_HKEYUP_PTT1           , //6
    E_HKEYUP_ATTD1      , //7
    E_HKEYUP_PILOT1     , //8
    E_CKEYDN_INUSE1     ,//9
};

枚举类型数据将第一个定为1,其他的数据依次加1.
6.定义一个发送事件函数SendCmd(u8 cmd)

flags_send1 |= EBIT(cmd);
UARTSend1();//用来发送数据的函数

定义

#define EBIT(a) (1u<<a)//将a位置为1,EBIT(2)就是 11

7.假设要传递E_CKEYDN_INUSE1事件,要实现事件的传递,就应该将事件做参数传入SendCmd(u8 cmd)函数
此时flags_send1 |= EBIT(9)=;由第四点知道flags_send1是一个32位的无符号整型数据。所以flags_send1 = 0000 0000 0000 0000 0000 0001 1111 1111
8.(1)UARTSend1()函数

void UARTSend1()
{
  int I;
  u8 sb[8] = {0x55,0xaa,0x00,0x00,0x00,0x00,0x00,0x00};
  u8 rxlen=RS485_RX_CNT1;
    DelayMS(10);
    if(rxlen==RS485_RX_CNT1)
    {
        if(flags_send1)
        {
            u16 sum = checksum(flags_send1);
            
            sb[0] = 0x55;
            sb[1] = 0xaa;
            sb[2] = (flags_send1 >> 0) & 0xff;
            sb[3] = (flags_send1 >> 8) & 0xff;
            sb[4] = (flags_send1 >> 16) & 0xff;
            sb[5] = (flags_send1 >> 24) & 0xff;
            sb[6] = (sum >> 0) &0xff;
            sb[7] = (sum >> 8) &0xff;
            
            flags_send1 = 0;
            
            for(i=0; i<8; i++)//8位数据
            {
                while(USART_GetFlagStatus(USART1,USART_FLAG_TC)        ==RESET);//检查是否发送完成,完成时,TC中断标志置位
                USART_SendData(USART1, sb[i]);//发送数据函数
            }
            while(USART_GetFlagStatus(USART1,USART_FLAG_TC)==RESET);
        }
    }
}

此时的flags_send1 = 0000 0000 0000 0000 0000 0001 1111 1111;
sb[2] = (flags_send1 >> 0) & 0xff = 1111 1111;
sb[3] = (flags_send1 >> 8) & 0xff = 0000 0001;
sb[4] = (flags_send1 >> 16) & 0xff = 0000 0000;
sb[5] = (flags_send1 >> 24) & 0xff = 0000 0000;
(2)checksum()函数

static u16 checksum(u32 senddata)
{
    u16 sum;
    u8 d0,d1,d2,d3;
    d0 = (senddata >> 0) & 0xff;
    d1 = (senddata >> 8) & 0xff;
    d2 = (senddata >> 16) & 0xff;
    d3 = (senddata >> 24) & 0xff;
    sum = d0 + d1 + d2 + d3; //8位数字
    return ~sum; //取反
}

d0=(senddata >> 0) & 0xff=0000 0001 1111 1111;
d1=(senddata >> 8) & 0xff=0000 0001 & 0xff=0000 0000 0000 0001;
d2=(senddata >> 16) & 0xff=0000 0000 & 0xff = 0000 0000 0000 0000;
d3=(senddata >> 24) & 0xff=0000 0000 & 0xff = 0000 0000 0000 0000;
sum=0000 0010 0000 0000;
~sum=1111 1101 1111 1111;
sb[6] = (sum >> 0) &0xff = 1111 1111 & 0xff = 1111 1111 ;
sb[7] = (sum >> 8) &0xff = 1111 1101;
(3)
为什么定义sb[8]的前两位是0x55,0xaa?

0xaa是1010 1010,0x55是0101 0101在通讯编码原理中,应该避免过多的重复0或者1,因为当传输变成一个长0/1时,一个脉冲干扰就会将数据截断,增加误码的概率。若通讯机不能接受10101010或者01010101,那么就是线路出现问题。这是一个判断线路状态的手段。

9.串口1接收数据
(1)通过UARTRead1()函数实现

void UARTRead1()//串口数据读取函数,数据接收指针RS485_RX_BUF1,
                //读取数据长度,RS485_RX_CNT1,返回值为实际读取到的数据长度
{
    u16 sum;
    u32 receive = 0;
    u8 rxlen=RS485_RX_CNT1;
    DelayMS(10);
    
    if(rxlen==RS485_RX_CNT1)
    {
        if(rxlen >= 8)//读取长度大于接收到的接收数据长度
        {
            DISABLE_INT();    //禁止全局中断

            if(RS485_RX_BUF1[0] == 0x55 && RS485_RX_BUF1[1] == 0xaa)//0x55=85 0xaa=170 包头是0x55 0xAA
            {
                receive += (RS485_RX_BUF1[2] << 0); 
                receive += (RS485_RX_BUF1[3] << 8); 
                receive += (RS485_RX_BUF1[4] << 16); 
                receive += (RS485_RX_BUF1[5] << 24); 
                
                sum = checksum(receive);
                
                if((RS485_RX_BUF1[6] == ((sum >> 0) & 0xff)) && 
                     (RS485_RX_BUF1[7] == ((sum >> 8) & 0xff)) )
                {
                    flags_receive1 = receive;
                }
                else
                {
                    flags_receive1 |= EBIT(E_ERROR);   //32
                }
            }
            else
            {
                flags_receive1 |= EBIT(E_ERROR);
            }
            
            RS485_RX_CNT1 = 0;
            
            ENABLE_INT();  //使能全局中断
        }
    }
}

receive += (RS485_RX_BUF1[2] << 0) = 0000 0000 0000 0000 0000 0000 1111 1111; ;
receive += (RS485_RX_BUF1[3] << 8) = 0000 0000 0000 0000 0000 0001 1111 1111
receive += (RS485_RX_BUF1[4] << 16) = 0000 0000 0000 0000 0000 0001 1111 1111
receive += (RS485_RX_BUF1[5] << 24) = 0000 0000 0000 0000 0000 0001 1111 1111
(2)将receive传入checksum(u32 senddata)函数中
d0= 0000 0000 1111 1111;
d1=0000 0000 0000 0001
d2= 0000 0000 0000 0000
d3=0000 0000 0000 0000
sum=0000 0001 0000 0000
~sum = 1111 1110 1111 1111
(3)定义接收数据标记位是u32 flags_receive1;
通过判断RS485_RX_BUF1[6] 、RS485_RX_BUF1是否等于 ((sum >> 0) & 0xff)、((sum >> 8) & 0xff))来判断读入的数据是不是正确的。
RS485_RX_BUF2[6] ==1111 1111
RS485_RX_BUF2[7] == 1111 1110
正确时
接收到的数据是flags_receive1 = receive= 0000 0000 0000 0000 0000 0001 1111 1111;
不正确时flags_receive1 |= EBIT(E_ERROR);接收数据标记位置为错误位。

上一篇 下一篇

猜你喜欢

热点阅读