美文网首页
串口发送与接收数据

串口发送与接收数据

作者: c与php | 来源:发表于2016-12-12 16:46 被阅读829次

    使用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);接收数据标记位置为错误位。

    相关文章

      网友评论

          本文标题:串口发送与接收数据

          本文链接:https://www.haomeiwen.com/subject/fiimmttx.html