打印

gd32f103rct spi0从机 DMA发送异常

[复制链接]
2365|11
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
wllhhu|  楼主 | 2021-4-7 17:13 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式
gd32f103rct  spi0从机 DMA发送一包数据的时候,有个别字节的最高位本来是1的,发送出来是0。 MISO没加上拉电阻。

使用特权

评论回复
沙发
wllhhu|  楼主 | 2021-4-7 17:19 | 只看该作者
而且从示波器里出来的MISO引脚波形比较奇怪。

微信图片_20210407171710.jpg (340.33 KB )

微信图片_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。

使用特权

评论回复
5
probedog| | 2025-5-30 15:23 | 只看该作者
特别是在长距离传输或噪声环境下,信号完整性更容易受到影响。

使用特权

评论回复
6
flycamelaaa| | 2025-5-30 15:24 | 只看该作者
时钟信号不稳定或存在抖动,可能导致数据采样错误,进而出现数据位错误。

使用特权

评论回复
7
stormwind123| | 2025-5-30 15:25 | 只看该作者
硬件连接不良或线路阻抗不匹配可能导致信号衰减或反射,影响数据传输的准确性。

使用特权

评论回复
8
stormwind123| | 2025-5-30 15:25 | 只看该作者
SPI和DMA的配置参数(如数据宽度、传输方向、优先级等)设置不正确,可能导致数据传输错误。

使用特权

评论回复
9
stormwind123| | 2025-5-30 15:25 | 只看该作者
DMA传输过程中,如果缓冲区管理不当(如缓冲区溢出、地址错误等),也可能导致数据错误。

使用特权

评论回复
10
classroom| | 2025-5-30 15:56 | 只看该作者
检查硬件连接是否良好,线路阻抗是否匹配。

使用特权

评论回复
11
probedog| | 2025-5-30 16:00 | 只看该作者
在MISO线上添加上拉电阻。

使用特权

评论回复
12
powerantone| | 2025-5-30 16:30 | 只看该作者
DMA传输过程中,如果时钟与DMA传输的同步处理不当,可能导致数据错误。

使用特权

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

本版积分规则

3

主题

6

帖子

0

粉丝