【RS485 - 总线冲突】

最近写了一个小小的串口通信程序,逻辑上感觉没什么问题,但就是数据与设定值不对。最后发现应该是总线冲突的问题。

在RS485通信中,如果在接收完数据后立即发送数据,可能会引发总线冲突问题。这是因为在RS485通信中,设备是共享同一个通信总线的,当多个设备同时尝试在总线上发送数据时,可能会导致数据冲突和干扰,从而造成通信失败。

情景解释:

假设有两个设备 A 和 B,它们通过共享的RS485总线进行通信。设备 A 正在接收来自设备 B 的数据,当接收完成后,设备 A 需要发送一组响应数据给设备 B。然而,如果设备 B还未完全释放总线,比如设备B切换收发方向不及时,而设备 A 在接收完数据后立即发送响应数据,就会发生总线冲突。

问题示例

下面代码的初衷是,如果我接收到 1F ,就发送0x1F, 0x08, 0x01, 0x06, 0x41, 0x43, 0x54, 0x34, 0x33, 0x30, 0xAE, 0x04

int main(void)
{
  	uint8_t send_data[] = {0x1F, 0x08, 0x01, 0x06, 0x41, 0x43, 0x54, 0x34, 0x33, 0x30, 0xAE, 0x04};
	uint8_t read_data = 0x00;

  /* Placeholder for user application code. The while loop below can be replaced with user application code. */
 while(1U)
 {
      UART_Receive(&UART_0, &read_data, 1); //此函数定义由DAVE直接生成
      //delay_100us();
      if(read_data==0x1F)
	  {
    		UART_Transmit(&UART_0, send_data, sizeof(send_data));//此函数定义由DAVE直接生成
    		read_data = 0x00;
      }
 }
}

然而实际收到的数据是这样的,发送的数据完全不是设定值
在这里插入图片描述

解决办法:

为了解决总线冲突问题,通常会在发送数据之前引入一定的延时,以确保其他设备有足够的时间完成它们的操作。这样可以有效避免数据冲突,提高通信的可靠性。以下是一些可能的解决办法:

  1. 固定延时: 在发送数据之前,引入一个固定的短暂延时。这样可以确保发送方有足够的时间切换收发方向,可以减小多个设备在同一时间尝试发送数据的可能性,从而降低冲突的概率。

代码示例

只要在接收数据后加延时delay_5ms;就可以解决这个问题。

void delay_100us(void)
{
    for (volatile uint32_t i = 0; i < 1000; ++i)
    {
        // Empty loop to create delay
        __NOP(); 
        // `__NOP()`  function is being called inside the loop which stands for "no operation"
        // and is typically used to insert a delay or to pad out a loop. 
        // This means that the processor will execute a no-operation instruction  `1000`  times, 
        // which will introduce a delay in the loop.

		// If don't write `__NOP()`, there is nothing inside the loop, 
		// so it will iterate  `1000`  times without executing any instructions. 
		// This will be a faster loop compared to the one with `__NOP()`,
		// because there are no instructions to execute inside the loop body.

		// Therefore, the loop with the  `__NOP()`  function will introduce a delay, 
		// while the loop without the  `__NOP()` will not introduce any delay 
		// and will simply iterate  `1000`  times.
    }
}
void delay_5ms(void)
{
    for (volatile uint32_t i = 0; i < 50000; ++i)
    {
        // Empty loop to create delay
        __NOP(); 
    }
}

int main(void)
{
  	uint8_t send_data[] = {0x1F, 0x08, 0x01, 0x06, 0x41, 0x43, 0x54, 0x34, 0x33, 0x30, 0xAE, 0x04};
	uint8_t read_data = 0x00;

  /* Placeholder for user application code. The while loop below can be replaced with user application code. */
 while(1U)
 {
      UART_Receive(&UART_0, &read_data, 1); //此函数定义由DAVE直接生成
      delay_100us();
      if(read_data==0x1F)
	  {
	  		delay_5ms();
    		UART_Transmit(&UART_0, send_data, sizeof(send_data));//此函数定义由DAVE直接生成
    		read_data = 0x00;
      }
 }
}

如图
在这里插入图片描述

  1. 轮询方式: 设备之间可以通过轮询的方式进行通信,即每个设备在特定的时间间隔内依次发送数据。这样可以确保每个设备都有机会在总线上发送数据,减少冲突。

  2. 及时切换方向: 优化代码逻辑,确保在接收完数据后,发送方能够及时切换收发方向,避免因延迟造成的冲突。

  3. 主从架构: 在系统中设定一个主设备,其他设备作为从设备,只有在主设备允许的情况下才能发送数据。主设备可以控制从设备发送数据的时机,从而避免冲突。

  4. 通信协议: 设计合理的通信协议,定义数据传输的时序和规则,确保每个设备都遵循协议要求,避免冲突发生。

总之,为了避免RS485通信中的总线冲突,需要在设计通信系统时充分考虑设备之间的通信时序,并采取适当的措施,如引入延时、采用轮询方式或设计合理的通信协议。这样可以确保通信的稳定性和可靠性。

### 回答1: 如果 STM32 在使用SPI通信时出现了冲突,可能是因为你同时在使用SPI和RS485两种通信方式造成的。你可以尝试以下方法来解决这个问题: 1. 将两种通信方式的使用分开,避免同时使用。 2. 调整使用顺序,先使用一种通信方式,再使用另一种。 3. 使用中断机制或者 DMA 来避免在通信过程中被打断。 4. 将两种通信方式的使用分别放在不同的时间段,例如使用定时器来控制通信时间。 5. 如果你使用的是串口通信,可以考虑使用串口多路复用技术,将两种通信方式放在不同的串口上使用。 希望以上建议能帮助你解决问题。 ### 回答2: 当STM32的SPI与W25QXX和RS485同时使用时,可能会出现冲突的原因是由于SPI接口的共享问题。SPI是一种用于串行通信的接口,它允许多个设备通过单条总线进行通信。但是,在同时使用SPI接口与W25QXX和RS485时,需要考虑以下问题: 1. 引脚冲突:SPI接口与W25QXX和RS485通常使用相同的引脚,如CLK、MISO、MOSI等。如果这些引脚被配置为SPI模式,而同时又用于W25QXX或RS485通信,那么就会发生冲突。 2. 通信协议冲突:SPI接口和RS485通常使用不同的通信协议。SPI接口使用的是主从模式,而RS485则是多主多从模式。如果同时使用SPI接口与W25QXX和RS485,通信协议的不匹配也会导致冲突和通信错误。 为了解决这个冲突,可以采取以下方法: 1. 分时使用接口:可以通过在不同的时间段分别使用SPI接口与W25QXX和RS485来避免冲突。比如在某个时间段使用SPI接口与W25QXX进行通信,另一个时间段再使用SPI接口与RS485进行通信。 2.引脚切换:可以通过软件或硬件方式切换引脚的功能,将SPI接口和W25QXX或RS485分别连接到不同的引脚上。这样,在使用SPI接口与W25QXX通信时,将SPI接口的引脚配置为SPI模式;而在使用SPI接口与RS485通信时,则将SPI接口的引脚切换为RS485模式。 需要注意的是,在分时使用接口或引脚切换时,需要根据具体的硬件设计和软件编程来进行操作,以确保不会产生冲突和错误。同时,还需要注意在使用同一个SPI接口时,正确配置和操作SPI的相关寄存器,以确保通信的正确性。 ### 回答3: STM32是一种非常常用的微控制器系列,它具有强大的功能和广泛的应用领域。其中包括SPI(串行外设接口)和RS485(串行通信标准)的功能。然而,在某些情况下,使用SPI接口与W25QXX闪存和RS485通信可能引起冲突冲突可能是因为SPI和RS485都使用了相同的引脚。例如,SPI的时钟线(CLK)、数据线(MISO和MOSI)和片选线(SS)与RS485总线A和B相同。因此,在同时使用这两个接口时,必须小心配置STM32的引脚分配。 解决这个冲突的方法包括以下几点: 1. 引脚分配:通过配置STM32的引脚复用功能,将SPI和RS485分配到不同的引脚上。这样可以避免引脚冲突。 2. 使用片选:在使用SPI与W25QXX进行通信时,可以通过片选线(SS)将芯片选中。当需要使用RS485进行通信时,可以通过控制芯片选线,将SPI失能。 3. 设计硬件接口:在设计硬件时,可以使用外部开关或多路复用器(MUX)来切换SPI和RS485的连接。这种方法可以根据需要快速切换。 4. 软件控制:通过软件编程来控制SPI和RS485的使用。在使用SPI与W25QXX进行通信时,将RS485的通信关闭,反之亦然。 总之,解决STM32 SPI与W25QXX和RS485冲突的关键在于正确的引脚分配和合理的控制策略。通过在硬件设计和软件编程中进行适当的设置,可以避免冲突,并确保两个接口的正常工作。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

六月悉茗

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值