之前一直使用ST的STM32F031單片機,但是由于疫情還是啥啥原因,ST的芯片價格漲得沒法看,因為我們是做產品,而且量比較大,ST的芯片就無法再用了,這個成本真的扛不起。
于是在很多國產MCU里面做了甄選,最終GD的因為新能優越,價格便宜獲選。GD32E230對標的STM32F031,實現了PIN TO PIN兼容,寄存器不是完全兼容,但是GD的主頻可以實現72M,這就很恐怖,STM32F031才48M,之前還得超頻到56M使用。不得不說,GD強!
僅僅對比固件庫,GD的庫函數封裝的比ST的庫要好很多,當然,ST現在主推HAL庫,這個HAL庫確實也很好。
在使用串口之前同樣要配置引腳,時鐘。
GPIO引腳配置
void com_gpio_init(void){rcu_periph_clock_enable(RCU_GPIOA);gpio_af_set(GPIOA, GPIO_AF_1, GPIO_PIN_9);gpio_af_set(GPIOA, GPIO_AF_1, GPIO_PIN_10);
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);
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);
gpio_mode_set(GPIOA, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO_PIN_8);gpio_output_options_set(GPIOA, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8);GPIO_BOP(GPIOA) = (uint32_t)GPIO_PIN_8;GPIO_BC(GPIOA) = (uint32_t)GPIO_PIN_8;}
串口使用USART0,對應PA9和PA10,相當于STM32的USART1;
配置USART時需要先復用PA9和PA10,使用gpio_mode_set()配置IO口的工作模式、輸入輸出類型。gpio_output_options_set()配置速度等參數。PA8為RS485的使能引腳。串口配置
void com_usart_init(void){ /* 使能USART時鐘*/ rcu_periph_clock_enable(RCU_USART0);
/* USART 配置*/ usart_deinit(USART0); usart_baudrate_set(USART0,2500000U); usart_receive_config(USART0, USART_RECEIVE_ENABLE); usart_transmit_config(USART0, USART_TRANSMIT_ENABLE);
usart_enable(USART0);}
使能USART的時鐘,設置波特率,使能接收和發送。串口中斷配置串口中斷的配置只需要一個API函數,相當簡單。
nvic_irq_enable(USART0_IRQn, 0);
下面是中斷服務函數,被屏蔽的代碼為直接操作寄存器,加快代碼的速度,因為我的項目對串口的速度要求較高。
串口接收數據進入中斷后首先獲取USART中斷標志位狀態,通過usart_interrupt_flag_get(EVAL_COM, USART_INT_FLAG_RBNE)判斷。然后用數組接收串口數據,判斷數據是否是0x1A,符合條件進入if函數。判斷串口數據接收標志位是否為RESET,然后通過函數發送數據,發送完失能串口中斷,以便下一次進入中斷,這里和st的庫函數處理方法有所不同。我在測試的時候想用usart_flag_clear()函數清除掉中斷標志位,但是在手冊里沒有清除中斷標志位的選項,GD是采用失能串口中斷的方式退出中斷。
void USART0_IRQHandler(void){// if(RESET != (USART_STAT(USART0)&0x00000010))// {// GPIO_BOP(GPIOA) = (uint32_t)GPIO_PIN_8;// receiver_buffer[0] = (uint16_t)(GET_BITS(USART_RDATA(USART0), 0U, 8U));// USART_REG_VAL(USART0, USART_INT_TBE) |= BIT(USART_BIT_POS(USART_INT_TBE));// // }// // if(RESET != (USART_STAT(USART0)&0x00000040))// {// USART_TDATA(USART0) = (USART_TDATA_TDATA & transmitter_buffer[txcount++]); // if(txcount == transfersize){// USART_REG_VAL(USART0, USART_INT_TBE) &= ~BIT(USART_BIT_POS(USART_INT_TBE));
// GPIO_BC(GPIOA) = (uint32_t)GPIO_PIN_8;// txcount = 0 ;// }// } if(RESET != usart_interrupt_flag_get(EVAL_COM, USART_INT_FLAG_RBNE)){ /* receive data */ receiver_buffer[0] = usart_data_receive(EVAL_COM); usart_interrupt_enable(EVAL_COM, USART_INT_TBE); if(receiver_buffer[0] == 0x1A) { /* transmit data */ gpio_bit_set(GPIOA,GPIO_PIN_8); while(usart_flag_get(USART0,USART_FLAG_TC)==RESET); usart_data_transmit(EVAL_COM, transmitter_buffer[4]); while(usart_flag_get(USART0,USART_FLAG_TC)==RESET); gpio_bit_reset(GPIOA,GPIO_PIN_8);// usart_flag_clear(USART0,USART_FLAG_TC); usart_interrupt_disable(EVAL_COM, USART_INT_TBE); } }}
運行結果
到此已經實現了USART接收中斷,接收判斷之后回復數據。
DMA配置
void USART_DMA_Init(void){ dma_parameter_struct dma_init_struct; /* enable DMA clock */ rcu_periph_clock_enable(RCU_DMA); rcu_periph_clock_enable(RCU_CFGCMP); syscfg_dma_remap_enable(SYSCFG_DMA_REMAP_USART0TX); /* deinitialize DMA channel1 */ dma_deinit(DMA_CH3); dma_init_struct.direction = DMA_MEMORY_TO_PERIPHERAL; dma_init_struct.memory_addr = (uint32_t)RS485_TX_BUF; dma_init_struct.memory_inc = DMA_MEMORY_INCREASE_ENABLE; dma_init_struct.memory_width = DMA_MEMORY_WIDTH_8BIT; dma_init_struct.number = 11; 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_MEDIUM; dma_init(DMA_CH3,&dma_init_struct); /* configure DMA mode */ dma_circulation_disable(DMA_CH3); dma_memory_to_memory_disable(DMA_CH3);}
DMA的配置過程和STM32差不多,同樣是配置DMA的時鐘,配置數據方向,基地址,外設地址,數據寬度,數據量等等。DMA發送數據因為我只用到DMA的發送,這里只介紹DMA的發送。
void MYDMA_Send(uint8_t *buffer,uint16_t size){ DMA_CHCTL(DMA_CH3) &= ~DMA_CHXCTL_CHEN;//失能DMA DMA_CHMADDR(DMA_CH3) = (uint32_t)buffer; //設置要發送的數據地址 DMA_CHCNT(DMA_CH3) = size ; //設置要發送的字節數目 DMA_CHCTL(DMA_CH3) |= DMA_CHXCTL_CHEN;//使能DMA}
void RS_485_SEND(uint8_t *psrc_data,int num){ GPIO_BOP(GPIOA) = (uint32_t)GPIO_PIN_8; MYDMA_Send(psrc_data,num); while(RESET == usart_flag_get(USART0, USART_FLAG_TC)); GPIO_BC(GPIOA) = (uint32_t)GPIO_PIN_8;}
當需要發送數據時直接調用RS_485_SEND即可。psrc_data為發送數組,num為發送數量。
原文標題:不會用國產單片機?看這里:GD32E230串口通信
文章出處:【微信公眾號:嵌入式ARM】歡迎添加關注!文章轉載請注明出處。
責任編輯:haq
-
單片機
+關注
關注
6040文章
44594瀏覽量
636958 -
通信
+關注
關注
18文章
6049瀏覽量
136227 -
STM32
+關注
關注
2270文章
10915瀏覽量
356778
原文標題:不會用國產單片機?看這里:GD32E230串口通信
文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論