打印
[研电赛技术支持]

【GD32F303红枫派使用手册】第十八讲 USART-485通信实验

[复制链接]
1637|1
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主

18.1 实验内容
通过本实验主要学习以下内容:
• 485工作原理
串口单线工作原理
18.2 实验原理
18.2.1 485工作原理
485一般指RS485RS485TIA-485-A, ANSI/TIA/EIA-485TIA/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协议输出的是差分信号,经过TTL485芯片的转换后其有效信息为两条信号线的电压差,即可大大消除通信时的共模干扰,同时由于其传递的信息随时可以在硬件层面上被测量,而且整个转换过程完全为硬件操作,无需软件编写,因此是种硬件协议。
TTL-485转换器的真值表
实际操作时,芯片的接收器输出端RO与单片机的Rx相连,驱动器输端DI则与单片机的Td相连
驱动器的输出逻辑
485芯片既有全双通信,也有半双工通信,如果485为半双工通信模式,其在发送信息时便无法读取信息,因此当DE被拉高时完全处于发送信息的状态,此时DI接受单片机写入的数字信号,当输信号DI1时输出正的差分信号,即A-B>0.2V。当输信号DI0时输出负差分信号,即B-A>-0.2V(有些芯片是0.3V,如SP3485
DE被拉低时,依据/RE(低电平有效)的电平判断作状态,当/RE为高时,整个器件不工作,输出高阻态,当其在低电平下使能时,则由输的AB差分信号向RO输出01
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个字节以下长度的数据。在while1)循环中循环查询uart_receive_complete_flag标志位,当该标志位为“SET”时,表示IDLE中断被触发,一帧数据接受完,最后将接收到的帧数据通过DMA发送方式原封不动发送到485上。
18.5 实验结果
使用USB485转接线,将AB线接好,使用串口调试助手发送一帧数据到MCUMCU会将这帧数据回发到串口调试助手中。

本教程由GD32 MCU方案商聚沃科技原创发布,了解更多GD32 MCU教程,关注聚沃科技官网,GD32MCU技术交流群:859440462

使用特权

评论回复
沙发
yangxiaor520| | 2024-6-19 08:13 | 只看该作者
红枫派,紫藤派,同一个品牌还有不同命名。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

170

主题

190

帖子

9

粉丝