GD32 DMA串口通讯DEMO

GD32 USART0 DMA发送与接收不定长数据

GD32 USART0发送与接收不定长数据例程

GD32E230简介

GD32E230G是兆易创新于2018年11月1日正式推出的主频高达72MHz的超值型微控制器,是GD32 MCU家族基于Cortex-M23内核的首个产品系列。
ARM Cortex-M23是Cortex-M0和Cortex-M0+的继任者,基于最新的Armv8-M架构的嵌入式微处理器内核。
GD32E230系列超值型新品主频高达72MHz,配备了16KB到64KB的嵌入式闪存及4KB到8KB的SRAM,配合内置的硬件乘法器、除法器和加速单元,在最高主频下的工作性能可达55DMIPS。

GD32E230G-START是兆易创新官方提供的开发板,使用GD32E230G8U6作为主控器,评估板使用MINI USB接口提供5V电源,提供包括扩展引脚在内的即RESET,BOOT,USER KEY,LED,GD-LINK等外设资源。支持第三方集成仿真环境并可通过仿真工具在次班上调试软件,支持2线SWD接口。

GD32E230采用DMA进行串口通讯例程

1,配置滴答时钟
2,配置系统中断
3,配置串口usart0
4,配置串口usart0 tx的DMA通道ch1
5,配置串口usart0 rx的DMA通道ch2
初始化

发送不定长数据

1,失能DMA_CH1。
2,DMA_CH1配置内存地址以及发送长度。
3,使能DMA_CH1。
4,使能usart0的DMA发送
5,等待DMA发送完成
发送不定长数据

接收不定长数据

该功能需要与串口的空闲中断配合使用,当串口空闲中断发生时,检测接收的DMA通道中是否有接收到数据。
1,串口空闲中断
2,关闭DMA_CH2
3,清空DMA_CH2相关的flag
4,计算DMA_CH2中接收到的数据量
5,重新配置DMA_CH2
串口空闲中断
串口DMA接收

完整代码

usart.c

#include "gd32e23x.h"
#include "usart.h"
#include <stdio.h>
#include <string.h>

uint8_t rxbuffer[USART0_RX_SIZE];
uint8_t txbuffer[]="\n\rUSART DMA receive and transmit example, please input 10 bytes:\r\n";
uart_buf usart0;

void usart0_init(uint32_t baudrate)
{
    /* enable COM GPIO clock */
    rcu_periph_clock_enable(RCU_GPIOA);
    
     /* enable USART clock */
    rcu_periph_clock_enable(RCU_USART0);


    /* connect port to USARTx_Tx */
    gpio_af_set(GPIOA, GPIO_AF_1, GPIO_PIN_9);

    /* connect port to USARTx_Rx */
    gpio_af_set(GPIOA, GPIO_AF_1, GPIO_PIN_10);

    /* configure USART Tx as alternate function push-pull */
    gpio_mode_set(GPIOA, GPIO_MODE_AF, GPIO_PUPD_PULLUP, GPIO_PIN_9);
    gpio_output_options_set(GPIOA, GPIO_OTYPE_PP, GPIO_OSPEED_10MHZ, GPIO_PIN_9);

    /* configure USART Rx as alternate function push-pull */
    gpio_mode_set(GPIOA, GPIO_MODE_AF, GPIO_PUPD_PULLUP, GPIO_PIN_10);
    gpio_output_options_set(GPIOA, GPIO_OTYPE_PP, GPIO_OSPEED_10MHZ, GPIO_PIN_10);
    

    /* USART configure */
    usart_deinit(USART0);
    usart_baudrate_set(USART0, baudrate);
    usart_receive_config(USART0, USART_RECEIVE_ENABLE);
    usart_transmit_config(USART0, USART_TRANSMIT_ENABLE);
    
    //开启空闲中断
    usart_interrupt_enable(USART0,USART_INT_IDLE);
    
    usart_enable(USART0);
}

void usart0_dma_send(uint8_t* buffer, uint16_t len)
{
    dma_channel_disable(DMA_CH1);
    
    dma_memory_address_config(DMA_CH1, (uint32_t)buffer);
    dma_transfer_number_config(DMA_CH1, len);
    
    /* enable DMA channel1 */
    dma_channel_enable(DMA_CH1);
    /* USART DMA enable for transmission and reception */
    usart_dma_transmit_config(USART0, USART_DENT_ENABLE);
    /* wait DMA Channel transfer complete */
    while(RESET == dma_flag_get(DMA_CH1, DMA_FLAG_FTF));
}

void usart0_dma_txinit()
{
    dma_parameter_struct dma_init_struct;
    /* enable DMA clock */
    rcu_periph_clock_enable(RCU_DMA);
    
    /* deinitialize DMA channel1 */
    dma_deinit(DMA_CH1);
    dma_init_struct.direction = DMA_MEMORY_TO_PERIPHERAL;
    dma_init_struct.memory_addr = NULL;
    dma_init_struct.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT;
    dma_init_struct.number = 0;
    dma_init_struct.periph_addr = USART0_TDATA_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(DMA_CH1,&dma_init_struct);
    
    /* configure DMA mode */
    dma_circulation_disable(DMA_CH1);
    dma_memory_to_memory_disable(DMA_CH1);
}

void usart0_dma_rxinit(void)
{
    dma_parameter_struct dma_init_struct;
    /* deinitialize DMA channel2 */
    dma_deinit(DMA_CH2);
    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 = USART0_RX_SIZE;
    dma_init_struct.periph_addr = USART0_RDATA_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(DMA_CH2, &dma_init_struct);
    
    dma_circulation_disable(DMA_CH2);
    dma_memory_to_memory_disable(DMA_CH2);

    /* enable DMA channel2 transfer complete interrupt */
    dma_interrupt_enable(DMA_CH2, DMA_INT_FTF);
    usart_dma_receive_config(USART0, USART_DENR_ENABLE);
    dma_channel_enable(DMA_CH2);
}

uint8_t usart0_dma_rxdata(void)
{
    dma_channel_disable(DMA_CH2);
    dma_flag_clear(DMA_CH2, DMA_FLAG_G);
    usart0.rx_index = USART0_RX_SIZE - dma_transfer_number_get(DMA_CH2);
    
    dma_transfer_number_config(DMA_CH2,USART0_RX_SIZE);
    dma_channel_enable(DMA_CH2);
    
    if( usart0.rx_index == 0 )
        return 0;
    else
        return 1;
}

gd23e23x_it.c

#include "gd32e23x_it.h"
#include "main.h"
#include "systick.h"
#include "usart.h"
extern uint8_t flag;
extern FlagStatus g_recv_complete;

/*!
    \brief      this function handles NMI exception
    \param[in]  none
    \param[out] none
    \retval     none
*/
void NMI_Handler(void)
{
}

/*!
    \brief      this function handles HardFault exception
    \param[in]  none
    \param[out] none
    \retval     none
*/
void HardFault_Handler(void)
{
    /* if Hard Fault exception occurs, go to infinite loop */
    while (1){
    }
}

/*!
    \brief      this function handles SVC exception
    \param[in]  none
    \param[out] none
    \retval     none
*/
void SVC_Handler(void)
{
}

/*!
    \brief      this function handles PendSV exception
    \param[in]  none
    \param[out] none
    \retval     none
*/
void PendSV_Handler(void)
{
}

/*!
    \brief      this function handles SysTick exception
    \param[in]  none
    \param[out] none
    \retval     none
*/
void SysTick_Handler(void)
{
//    led_spark();
    delay_decrement();
}

//1ms定时器
void TIMER0_BRK_UP_TRG_COM_IRQHandler(void)
{
    if( timer_flag_get(TIMER0, TIMER_INT_FLAG_UP) == SET )
    {
        timer_flag_clear(TIMER0,TIMER_INT_FLAG_UP);
    }
}

/*!
    \brief      this function handles DMA_Channel1_2_IRQHandler interrupt
    \param[in]  none
    \param[out] none
    \retval     none
*/
void DMA_Channel1_2_IRQHandler(void)
{
    //DMA接收缓存满,产生DMA中断
    if(RESET != dma_interrupt_flag_get(DMA_CH2, DMA_INT_FLAG_FTF)){
        dma_interrupt_flag_clear(DMA_CH2, DMA_INT_FLAG_G);
        g_recv_complete = SET;
    }
}


void USART0_IRQHandler()
{
    //利用串口的空闲中断,以及DMA的接收数据,判断串口是否接收完成
    if( SET == usart_interrupt_flag_get(USART0, USART_INT_FLAG_IDLE) )
    {
        usart_interrupt_flag_clear(USART0,USART_INT_FLAG_IDLE);
        if ( usart0_dma_rxdata() )
            g_recv_complete = SET;
    }
}

  • 15
    点赞
  • 58
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值