当前位置:网站首页 > 编程语言 > 正文

485协议代码(485功能代码说明)



18.1 实验内容

通过本实验主要学习以下内容:

• 485工作原理

• 串口单线工作原理

18.2 实验原理

18.2.1 485工作原理

485一般指RS485。RS485名TIA-485-A, ANSI/TIA/EIA-485或TIA/EIA-485,是由电信业协会和电业联盟定义。RS485就是个硬件通信协议,它规定当两线间电压差为+2V ~ +6V时为逻辑“1”,电压差为-2V ~ -6V时为逻辑“0”

RS-485的特点:由于485信号是利用差模传输的,即由485+与485-的电压差来作为信号传输。如果外部有个扰源对其进行干扰,使双绞线进行485信号传输的时候,由于其双绞,干扰对于485+,485-的干扰效果都是样的,那电压差依然是不变的,对于485信号的干扰缩到了最小。同样的道理,如果有屏蔽线起到屏蔽作的话,外部扰源对于其的扰影响也可以尽可能的缩小。

485布线规范是必须要牵的布线,旦没有借助485集线器和485中继器直接布设成星型连接和树形连接,很容易造成信号反射导致总线不稳定。

485总线必须要单点可靠接地。单点就是整个485总线上只能是有个点接地,不能多点接地,因为将其接地是因为要将地线(般都是屏蔽线作地线)上的电压保持致,防止共模扰,如果多点接地适得其反。

RS-485 与 MCU

MCU的输出和读取都是TTL电平,一般情况下由地线和信号线组成,在远距离传输的情况下,信号线上的干扰信号会随着有效信号被传递到接收端,使得通信容易被干扰。 与之相对的,485协议输出的是差分信号,经过TTL转485芯片的转换后其有效信息为两条信号线的电压差,即可大大消除通信时的共模干扰,同时由于其传递的信息随时可以在硬件层面上被测量,而且整个转换过程完全为硬件操作,无需软件编写,因此是种硬件协议。

TTL-485转换器的真值表

实际操作时,芯片的接收器输出端RO与单片机的Rx相连,驱动器输端DI则与单片机的Td相连

驱动器的输出逻辑

485芯片既有全双通信,也有半双工通信,如果485为半双工通信模式,其在发送信息时便无法读取信息,因此当DE被拉高时完全处于发送信息的状态,此时DI接受单片机写入的数字信号,当输信号DI为1时输出正的差分信号,即A-B>0.2V。当输信号DI为0时输出负差分信号,即B-A>-0.2V(有些芯片是0.3V,如SP3485)

当DE被拉低时,依据/RE(低电平有效)的电平判断作状态,当/RE为高时,整个器件不工作,输出高阻态,当其在低电平下使能时,则由输的AB差分信号向RO输出0或1。

18.2.2 串口单线工作原理

本实验中,我们使用串口的TX线同时实现发送和接受的功能。

通过设置USART_CTL2寄存器的HDEN位,可以使能单线模式。在单线模式下, TX引脚和RX引脚将从内部连接到一起, RX引脚不再使用。 TX引脚应该被配置为开漏输出模式。通信冲突由软件处理。当发送时,需要关闭串口接受功能,打开发送功能;当接受时,需要关闭串口发送功能,打开接受功能。

18.3 硬件设计

红枫派开发板485硬件设计如下:

即使用PB6实现发送和接收,使用PG15用来控制485传输方向。

18.4 代码解析

18.4.1 485发送函数

在bsp_uart.c中,定义了485发送函数:

C

void bsp_rs485_uart_transmit(uint8_t *pbuff,uint16_t length)

{

    uint32_t timeout = driver_tick;    

    while(BOARD_UART.uart_control.Com_Flag.Bits.RecState==1 && BOARD_UART.uart_control.RecCount!=0){

        if((timeout+UART_TIMEOUT_MS) <= driver_tick) {              

            BOARD_UART.uart_control.Com_Flag.Bits.RecState=0;

        }

    }    

    driver_gpio_pin_set(&RS485_DIR);  

    usart_receive_config(BOARD_UART.uart_x, USART_RECEIVE_DISABLE);

    usart_transmit_config(BOARD_UART.uart_x, USART_TRANSMIT_ENABLE);            

    if(BOARD_UART.uart_mode_tx==MODE_DMA)      

    {

        driver_uart_dma_transmit(&BOARD_UART,pbuff,length);

    }

    else if(BOARD_UART.uart_mode_tx==MODE_INT)

    {

        driver_uart_int_transmit(&BOARD_UART,pbuff,length);        

    }

    else if(BOARD_UART.uart_mode_tx==MODE_POLL)

    {

        driver_uart_poll_transmit(&BOARD_UART,pbuff,length);

        

        usart_receive_config(BOARD_UART.uart_x, USART_RECEIVE_ENABLE);

        usart_transmit_config(BOARD_UART.uart_x, USART_TRANSMIT_DISABLE);          

        driver_gpio_pin_reset(&RS485_DIR);      

    }

}

18.4.2 485接受函数

在bsp_uart.c中定义了485接受函数:

C

void bsp_rs485_uart_receive(uint8_t *pbuff,uint16_t length)

{

    uint32_t timeout = driver_tick;    

    while(BOARD_UART.uart_control.Com_Flag.Bits.SendState==1){

        if((timeout+UART_TIMEOUT_MS) <= driver_tick) {              

            BOARD_UART.uart_control.Com_Flag.Bits.SendState=0;

        }

    }    

    usart_receive_config(BOARD_UART.uart_x, USART_RECEIVE_ENABLE);

    usart_transmit_config(BOARD_UART.uart_x, USART_TRANSMIT_DISABLE);     

    driver_gpio_pin_reset(&RS485_DIR);   

    if(BOARD_UART.uart_mode_rx==MODE_DMA)

    {

        driver_uart_dma_receive(&BOARD_UART,pbuff,length);

    }

    else if(BOARD_UART.uart_mode_rx==MODE_INT)

    {

        driver_uart_int_receive(&BOARD_UART,pbuff,length);        

    }

    else if(BOARD_UART.uart_mode_rx==MODE_POLL)

    {

        driver_uart_poll_receive(&BOARD_UART,pbuff,length);

    }

}

18.4.3 main函数实现

以下为main函数代码:

C

int main(void)

{

    delay_init();

    //初始化UART为中断模式,注册接受完成(IDLE)回调函数

    BOARD_UART.uart_mode_tx=MODE_DMA;

    BOARD_UART.uart_mode_rx=MODE_DMA;

    BOARD_UART.uart_idle_callback=user_receive_complete_callback;       

    bsp_rs485_uart_init();

    nvic_irq_enable(USART0_IRQn,2,0);

    delay_ms(1000);    

    //配置UART接受,最长100byte

    bsp_rs485_uart_receive(uart_rec_buff,100);    

        while (1)

        {          

        //查询到接受完成回调函数标志

        if(uart_receive_complete_flag==SET)

        {

            uart_receive_complete_flag=RESET;            

            //发送刚接受到的数据

            bsp_rs485_uart_transmit(uart_rec_buff,uart_receive_count);             

        }

        }

}

本例程main函数首先进行了延时函数初始化,再初始化485为中断模式,接着配置串口BOARD_UART,开启串口中断NVIC,这里使用到了IDLE中断,然后配置485接受(DMA方式),最长100个字节,所以我们可以给485发送100个字节以下长度的数据。在while(1)循环中循环查询uart_receive_complete_flag标志位,当该标志位为“SET”时,表示IDLE中断被触发,一帧数据接受完,最后将接收到的帧数据通过DMA发送方式原封不动发送到485上。

18.5 实验结果

使用USB转485转接线,将A、B线接好,使用串口调试助手发送一帧数据到MCU,MCU会将这帧数据回发到串口调试助手中。

到此这篇485协议代码(485功能代码说明)的文章就介绍到这了,更多相关内容请继续浏览下面的相关推荐文章,希望大家都能在编程的领域有一番成就!

版权声明


相关文章:

  • 开源代码网站github(开源代码网站github下载代码)2025-02-05 17:54:09
  • w25Q128图纸(5q12656图纸)2025-02-05 17:54:09
  • pass平台和sass平台(sass平台什么意思)2025-02-05 17:54:09
  • 阻塞队列和普通队列(什么叫阻塞队列)2025-02-05 17:54:09
  • 认证码是啥意思怎么查(认证码什么意思)2025-02-05 17:54:09
  • dv实验(dv试验和pv试验的区别)2025-02-05 17:54:09
  • 快速促排卵药(促排卵怎么用药)2025-02-05 17:54:09
  • autonomy翻译成中文(automobile翻译成中文)2025-02-05 17:54:09
  • pek文件怎么打开(pek文件是什么意思)2025-02-05 17:54:09
  • 完整的网页代码模板(完整的网页代码模板有哪些)2025-02-05 17:54:09
  • 全屏图片