【GD32】GD32F303串口设置DMA发生中断无法进入中断函数

在这里插入图片描述在GD32F303官方提供的串口例程中,有一个DMA发生和接收中断例程,在模仿着写的过程中,能够正常发送数据,但是无法进入中断函数。在这里插入图片描述DMA0_Channel3_IRQHandler函数时官方定义的弱函数,需要自己重新实现。如果开启了DMA0通道3相关的中断,在发生中断的时候就会进入该函数。DMA的中断主要有以下3个,每一个通道都有自己的3个中断。
在这里插入图片描述
例程的代码主要:

int main(void)
{
    dma_parameter_struct dma_init_struct;
    /* enable DMA0 */
    rcu_periph_clock_enable(RCU_DMA0);
    /* initialize USART */
    gd_eval_com_init(EVAL_COM0);
    /*configure DMA0 interrupt*/
    nvic_config();
    
    /* deinitialize DMA channel3(USART0 tx) */
    dma_deinit(DMA0, DMA_CH3);
    dma_struct_para_init(&dma_init_struct);

    dma_init_struct.direction = DMA_MEMORY_TO_PERIPHERAL;
    dma_init_struct.memory_addr = (uint32_t)txbuffer;
    dma_init_struct.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT;
    dma_init_struct.number = ARRAYNUM(txbuffer);
    dma_init_struct.periph_addr = USART0_DATA_ADDRESS;
    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_CH3, &dma_init_struct);

    /* deinitialize DMA channel4 (USART0 rx) */
    dma_deinit(DMA0, DMA_CH4);
    dma_struct_para_init(&dma_init_struct);

    dma_init_struct.direction = DMA_PERIPHERAL_TO_MEMORY;
    dma_init_struct.memory_addr = (uint32_t)rxbuffer;
    dma_init_struct.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT;
    dma_init_struct.number = 10;
    dma_init_struct.periph_addr = USART0_DATA_ADDRESS;
    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_CH4, &dma_init_struct);

    /* configure DMA mode */
    dma_circulation_disable(DMA0, DMA_CH3);
    dma_memory_to_memory_disable(DMA0, DMA_CH3);
    dma_circulation_disable(DMA0, DMA_CH4);
    dma_memory_to_memory_disable(DMA0, DMA_CH4);
    
    /* enable USART DMA for reception */
    usart_dma_receive_config(USART0, USART_RECEIVE_DMA_ENABLE);
    /* enable DMA0 channel4 transfer complete interrupt */
    dma_interrupt_enable(DMA0, DMA_CH4, DMA_INT_FTF);
    /* enable DMA0 channel4 */
    dma_channel_enable(DMA0, DMA_CH4);
    /* enable USART DMA for transmission */
    usart_dma_transmit_config(USART0, USART_TRANSMIT_DMA_ENABLE);
    /* enable DMA0 channel3 transfer complete interrupt */
    dma_interrupt_enable(DMA0, DMA_CH3, DMA_INT_FTF);
    /* enable DMA0 channel3 */
    dma_channel_enable(DMA0, DMA_CH3);
    
    /* waiting for the transfer to complete*/
    while(RESET == g_transfer_complete){
    }
    
    g_transfer_complete = RESET;

    /* waiting for the transfer to complete*/
    while(RESET == g_transfer_complete){
    }

    printf("\n\r%s\n\r", rxbuffer);

    while(1){
    }
}

/*!
    \brief      configure DMA interrupt
    \param[in]  none
    \param[out] none
    \retval     none
*/
void nvic_config(void)
{
    nvic_irq_enable(DMA0_Channel3_IRQn, 0, 0);
    nvic_irq_enable(DMA0_Channel4_IRQn, 0, 1);
}

这里要特别注意是dma_interrupt_enable(DMA0, DMA_CH3, DMA_INT_FTF);,必须要在dma_deinit(DMA0, DMA_CH3);这个函数的后面。如果他们两个顺序反了,就会无法成功开启中断。

dma_deinit(DMA0, DMA_CH3);该函数主要是将DMA0的通道3去初始化,后面会接着dma_init函数。该函数主要将各种寄存器数值重置为0,有点类似于计算器的归0。
在这里插入图片描述
在这里插入图片描述
在函数内部会将0赋值给CHCTL寄存器,这个寄存器很重要。看数据手册,CHCTL是通道控制寄存器,bit0是CHEN通道使能位,bit1时FTFIE通道传输完成中断使能为,bit2是HTFIE通道半传输完成中断使能,bit3是ERRIE通道错误中断使能位。所以中断使能函数操作的就是该寄存器的某个位。
在这里插入图片描述
在这里插入图片描述
dma_interrupt_enable(DMA0, DMA_CH3, DMA_INT_FTF);该函数主要是开启DMA通道的中断,DMA_INT_FTF开启的是通道传输完成中断。如下所示,主要是CHCTL寄存器赋值,也就是将CHCTL的bit1置1。
在这里插入图片描述
到这里,我就看出来了。我在配置DMA通道中断的过程中,将dma_interrupt_enable(DMA0, DMA_CH3, DMA_INT_FTF);提前了,想着和nvic_irq_enable函数凑在一起,都是使能中断,看起来顺眼,没想到不能提前开启中断。
在这里插入图片描述
与上述函数类似不能放在前面的还有dma_channel_enable函数。
在这里插入图片描述
仅此记录。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个GD32F303通过DMA发送串口中断接收的RS485串口DMA函数的示例代码: ```c #include "gd32f30x.h" #define USART USART1 #define USART_DMA_CHANNEL DMA_CH5 #define USART_DMA_STREAM DMA0_STREAM5 #define BUFFER_SIZE 128 uint8_t tx_buffer[BUFFER_SIZE]; uint8_t rx_buffer[BUFFER_SIZE]; uint16_t rx_len = 0; void rs485_init(void) { rcu_periph_clock_enable(RCU_USART1); rcu_periph_clock_enable(RCU_AF); gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_9); gpio_init(GPIOA, GPIO_MODE_AF_OD, GPIO_OSPEED_50MHZ, GPIO_PIN_10); usart_deinit(USART); usart_baudrate_set(USART, 9600); usart_parity_config(USART, USART_PM_NONE); usart_word_length_set(USART, USART_WL_8BIT); usart_stop_bit_set(USART, USART_STB_1BIT); usart_hardware_flow_rts_config(USART, USART_RTS_DISABLE); usart_hardware_flow_cts_config(USART, USART_CTS_DISABLE); usart_transmit_config(USART, USART_TRANSMIT_ENABLE); usart_receive_config(USART, USART_RECEIVE_ENABLE); usart_enable(USART); nvic_irq_enable(USART1_IRQn, 0, 0); } void rs485_dma_init(void) { rcu_periph_clock_enable(RCU_DMA0); dma_deinit(USART_DMA_STREAM); dma_parameter_struct dma_init_struct; dma_init_struct.direction = DMA_MEMORY_TO_PERIPHERAL; dma_init_struct.memory_addr = (uint32_t)tx_buffer; dma_init_struct.memory_inc = DMA_MEMORY_INC_ENABLE; dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT; dma_init_struct.periph_addr = (uint32_t)&USART_DATA(USART); dma_init_struct.periph_inc = DMA_PERIPH_INC_DISABLE; dma_init_struct.periph_width = DMA_PERIPHERAL_WIDTH_8BIT; dma_init_struct.priority = DMA_PRIORITY_HIGH; dma_init(USART_DMA_STREAM, dma_init_struct); dma_circulation_disable(USART_DMA_STREAM); dma_memory_to_memory_disable(USART_DMA_STREAM); dma_channel_subperipheral_select(USART_DMA_STREAM, DMA_SUBPERI0_USART1_TX); usart_dma_transmit_config(USART, USART_DMA_CHANNEL, USART_DENT_ENABLE); dma_interrupt_enable(USART_DMA_STREAM, DMA_INT_FTF); nvic_irq_enable(DMA0_Channel5_IRQn, 0, 0); } void rs485_send_dma(uint8_t *buffer, uint16_t len) { memcpy(tx_buffer, buffer, len); dma_transfer_number_config(USART_DMA_STREAM, len); dma_memory_address_config(USART_DMA_STREAM, (uint32_t)tx_buffer); dma_channel_enable(USART_DMA_STREAM); } void rs485_receive_dma(void) { rx_len = 0; dma_transfer_number_config(USART_DMA_STREAM, BUFFER_SIZE); dma_memory_address_config(USART_DMA_STREAM, (uint32_t)rx_buffer); dma_channel_enable(USART_DMA_STREAM); } void USART1_IRQHandler(void) { if (usart_interrupt_flag_get(USART, USART_INT_FLAG_RBNE)) { uint8_t data = usart_data_receive(USART); // 处理接收到的数据 if (rx_len < BUFFER_SIZE) { rx_buffer[rx_len++] = data; } } } void DMA0_Channel5_IRQHandler(void) { if (dma_interrupt_flag_get(DMA0, DMA_CH5, DMA_INT_FLAG_FTF)) { dma_interrupt_flag_clear(DMA0, DMA_CH5, DMA_INT_FLAG_FTF); dma_channel_disable(USART_DMA_STREAM); } } ``` 这个示例代码中,我们使用了DMA来发送和接收RS485串口数据。在初始化函数 `rs485_dma_init()` 中,我们配置了USARTDMA传输通道,并设置DMA的参数。在发送函数 `rs485_send_dma()` 中,我们将要发送的数据拷贝到发送缓冲区,并启动DMA传输。在接收函数 `rs485_receive_dma()` 中,我们启动DMA接收。 同时,我们也实现了USARTDMA中断处理函数 `USART1_IRQHandler()` 和 `DMA0_Channel5_IRQHandler()`,用于处理接收和发送完成的中断事件。 希望这个示例对你有帮助!如果有任何问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值