gd32f103rct spi0从机 DMA发送异常

[复制链接]
 楼主| wllhhu 发表于 2021-4-7 17:13 | 显示全部楼层 |阅读模式
gd32f103rct  spi0从机 DMA发送一包数据的时候,有个别字节的最高位本来是1的,发送出来是0。 MISO没加上拉电阻。
 楼主| wllhhu 发表于 2021-4-7 17:19 | 显示全部楼层
而且从示波器里出来的MISO引脚波形比较奇怪。
微信图片_20210407171710.jpg
 楼主| wllhhu 发表于 2021-4-7 17:20 | 显示全部楼层
初始化代码如下

void  init_spi_slave(void)
{
    spi_parameter_struct spi_init_struct;
    //打开spi口GPIO的时钟
rcu_periph_clock_enable(RCU_GPIOA);
   
    //打开spi口外设的时钟
    rcu_periph_clock_enable(clk_spi_slave);
   
    //Configure SPI_SLAVE pins: NSS, SCK and MOSI
    gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_10MHZ, GPIO_Pin_5);
    gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_10MHZ, GPIO_Pin_7);
    //cs
    gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_10MHZ, GPIO_Pin_4);
    //MISO
    gpio_init(GPIOA, GPIO_MODE_AF_PP,       GPIO_OSPEED_10MHZ, GPIO_Pin_6);
    spi_i2s_deinit(spi_slave);
    spi_struct_para_init(&spi_init_struct);
    spi_init_struct.trans_mode           = SPI_TRANSMODE_FULLDUPLEX;
    spi_init_struct.device_mode          = SPI_SLAVE;
    spi_init_struct.frame_size           = SPI_FRAMESIZE_8BIT;
    spi_init_struct.clock_polarity_phase = SPI_CK_PL_LOW_PH_2EDGE;
    spi_init_struct.nss                  = SPI_NSS_SOFT;
    spi_init_struct.prescale             = SPI_PSC_8;
    spi_init_struct.endian               = SPI_ENDIAN_MSB;
    spi_init(spi_slave, &spi_init_struct);
    spi_crc_off(spi_slave);
    spi_nss_internal_low(spi_slave);
    spi_enable(spi_slave);
}
void  spi_slave_rev_start(void)
{
    memset(link_rx, 0, 48);
    dma_memory_address_config(DMA0, DMA_CH1, (uint32_t)link_rx);
    dma_transfer_number_config(DMA0, DMA_CH1, link_rxbuf_len);
    dma_channel_enable(DMA0, DMA_CH1);
}
void  spi_slave_rev_stop(void)
{
    dma_transfer_number_config(DMA0, DMA_CH1, 0);
    dma_channel_disable(DMA0, DMA_CH1);
}
void  spi_slave_send_start(uint16_t  count)
{
    dma_memory_address_config(DMA0, DMA_CH2, (uint32_t)link_tx);
    dma_transfer_number_config(DMA0, DMA_CH2, count);
    dma_channel_enable(DMA0, DMA_CH2);
}
void  spi_slave_send_stop(void)
{
    dma_transfer_number_config(DMA0, DMA_CH2, 0);
    dma_channel_disable(DMA0, DMA_CH2);
}



void  init_spi_slave_dma(void)
{
    dma_parameter_struct dma_init_struct;
    rcu_periph_clock_enable(RCU_DMA0);
   
    dma_deinit(DMA0, DMA_CH1);
    dma_struct_para_init(&dma_init_struct);
   
    dma_init_struct.direction = DMA_PERIPHERAL_TO_MEMORY;
    dma_init_struct.memory_addr = (uint32_t)link_rx;
    dma_init_struct.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT;
    dma_init_struct.number = link_rxbuf_len;
    dma_init_struct.periph_addr = (uint32_t)&SPI_DATA(spi_slave);
    dma_init_struct.periph_inc = DMA_PERIPH_INCREASE_DISABLE;
    dma_init_struct.periph_width = DMA_PERIPHERAL_WIDTH_8BIT;
    dma_init_struct.priority = DMA_PRIORITY_ULTRA_HIGH;
    dma_init(DMA0, DMA_CH1, &dma_init_struct);
    dma_circulation_disable(DMA0, DMA_CH1);
    dma_memory_to_memory_disable(DMA0, DMA_CH1);
    nvic_irq_enable(DMA0_Channel1_IRQn, 6, 0);
   
    dma_interrupt_enable(DMA0, DMA_CH1, DMA_INT_FTF);
    dma_interrupt_enable(DMA0, DMA_CH1, DMA_INT_ERR);
   
    dma_deinit(DMA0, DMA_CH2);
    dma_struct_para_init(&dma_init_struct);
   
    dma_init_struct.direction = DMA_MEMORY_TO_PERIPHERAL;
    dma_init_struct.memory_addr = (uint32_t)link_tx;
    dma_init_struct.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT;
    dma_init_struct.number = link_txbuf_len;
    dma_init_struct.periph_addr = (uint32_t)&SPI_DATA(spi_slave);
    dma_init_struct.periph_inc = DMA_PERIPH_INCREASE_DISABLE;
    dma_init_struct.periph_width = DMA_PERIPHERAL_WIDTH_8BIT;
    dma_init_struct.priority = DMA_PRIORITY_ULTRA_HIGH;
    dma_init(DMA0, DMA_CH2, &dma_init_struct);
    dma_circulation_disable(DMA0, DMA_CH2);
    dma_memory_to_memory_disable(DMA0, DMA_CH2);
    nvic_irq_enable(DMA0_Channel2_IRQn, 6, 0);
   
    dma_interrupt_enable(DMA0, DMA_CH2, DMA_INT_FTF);
    dma_interrupt_enable(DMA0, DMA_CH2, DMA_INT_ERR);
    spi_dma_enable(spi_slave, SPI_DMA_TRANSMIT);
    spi_dma_enable(spi_slave, SPI_DMA_RECEIVE);
    slave_dma_send_stop();
}

probedog 发表于 2025-5-30 15:23 | 显示全部楼层
MISO线未加上拉电阻,可能导致信号在空闲时浮空,受到噪声干扰后,信号电平可能发生异常变化,如本应为1的位被干扰为0。
probedog 发表于 2025-5-30 15:23 | 显示全部楼层
特别是在长距离传输或噪声环境下,信号完整性更容易受到影响。
flycamelaaa 发表于 2025-5-30 15:24 | 显示全部楼层
时钟信号不稳定或存在抖动,可能导致数据采样错误,进而出现数据位错误。
stormwind123 发表于 2025-5-30 15:25 | 显示全部楼层
硬件连接不良或线路阻抗不匹配可能导致信号衰减或反射,影响数据传输的准确性。
stormwind123 发表于 2025-5-30 15:25 | 显示全部楼层
SPI和DMA的配置参数(如数据宽度、传输方向、优先级等)设置不正确,可能导致数据传输错误。
stormwind123 发表于 2025-5-30 15:25 | 显示全部楼层
DMA传输过程中,如果缓冲区管理不当(如缓冲区溢出、地址错误等),也可能导致数据错误。
classroom 发表于 2025-5-30 15:56 | 显示全部楼层
检查硬件连接是否良好,线路阻抗是否匹配。
probedog 发表于 2025-5-30 16:00 | 显示全部楼层
在MISO线上添加上拉电阻。
powerantone 发表于 2025-5-30 16:30 | 显示全部楼层
DMA传输过程中,如果时钟与DMA传输的同步处理不当,可能导致数据错误。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

4

主题

9

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部