之前一直使用ST的STM32F031單片機(jī),但是由于疫情還是啥啥原因,ST的芯片價(jià)格漲得沒(méi)法看,因?yàn)槲覀兪亲霎a(chǎn)品,而且量比較大,ST的芯片就無(wú)法再用了,這個(gè)成本真的扛不起。
于是在很多國(guó)產(chǎn)MCU里面做了甄選,最終GD的因?yàn)樾履軆?yōu)越,價(jià)格便宜獲選。GD32E230對(duì)標(biāo)的STM32F031,實(shí)現(xiàn)了PIN TO PIN兼容,寄存器不是完全兼容,但是GD的主頻可以實(shí)現(xiàn)72M,這就很恐怖,STM32F031才48M,之前還得超頻到56M使用。不得不說(shuō),GD強(qiáng)!

僅僅對(duì)比固件庫(kù),GD的庫(kù)函數(shù)封裝的比ST的庫(kù)要好很多,當(dāng)然,ST現(xiàn)在主推HAL庫(kù),這個(gè)HAL庫(kù)確實(shí)也很好。
在使用串口之前同樣要配置引腳,時(shí)鐘。
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,對(duì)應(yīng)PA9和PA10,相當(dāng)于STM32的USART1;
配置USART時(shí)需要先復(fù)用PA9和PA10,使用gpio_mode_set()配置IO口的工作模式、輸入輸出類型。gpio_output_options_set()配置速度等參數(shù)。PA8為RS485的使能引腳。串口配置
void com_usart_init(void){ /* 使能USART時(shí)鐘*/ 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的時(shí)鐘,設(shè)置波特率,使能接收和發(fā)送。串口中斷配置串口中斷的配置只需要一個(gè)API函數(shù),相當(dāng)簡(jiǎn)單。
nvic_irq_enable(USART0_IRQn, 0);

下面是中斷服務(wù)函數(shù),被屏蔽的代碼為直接操作寄存器,加快代碼的速度,因?yàn)槲业捻?xiàng)目對(duì)串口的速度要求較高。
串口接收數(shù)據(jù)進(jìn)入中斷后首先獲取USART中斷標(biāo)志位狀態(tài),通過(guò)usart_interrupt_flag_get(EVAL_COM, USART_INT_FLAG_RBNE)判斷。然后用數(shù)組接收串口數(shù)據(jù),判斷數(shù)據(jù)是否是0x1A,符合條件進(jìn)入if函數(shù)。判斷串口數(shù)據(jù)接收標(biāo)志位是否為RESET,然后通過(guò)函數(shù)發(fā)送數(shù)據(jù),發(fā)送完失能串口中斷,以便下一次進(jìn)入中斷,這里和st的庫(kù)函數(shù)處理方法有所不同。我在測(cè)試的時(shí)候想用usart_flag_clear()函數(shù)清除掉中斷標(biāo)志位,但是在手冊(cè)里沒(méi)有清除中斷標(biāo)志位的選項(xiàng),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); } }}
運(yùn)行結(jié)果

到此已經(jīng)實(shí)現(xiàn)了USART接收中斷,接收判斷之后回復(fù)數(shù)據(jù)。
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的配置過(guò)程和STM32差不多,同樣是配置DMA的時(shí)鐘,配置數(shù)據(jù)方向,基地址,外設(shè)地址,數(shù)據(jù)寬度,數(shù)據(jù)量等等。DMA發(fā)送數(shù)據(jù)因?yàn)槲抑挥玫紻MA的發(fā)送,這里只介紹DMA的發(fā)送。
void MYDMA_Send(uint8_t *buffer,uint16_t size){ DMA_CHCTL(DMA_CH3) &= ~DMA_CHXCTL_CHEN;//失能DMA DMA_CHMADDR(DMA_CH3) = (uint32_t)buffer; //設(shè)置要發(fā)送的數(shù)據(jù)地址 DMA_CHCNT(DMA_CH3) = size ; //設(shè)置要發(fā)送的字節(jié)數(shù)目 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;}
當(dāng)需要發(fā)送數(shù)據(jù)時(shí)直接調(diào)用RS_485_SEND即可。psrc_data為發(fā)送數(shù)組,num為發(fā)送數(shù)量。
原文標(biāo)題:不會(huì)用國(guó)產(chǎn)單片機(jī)?看這里:GD32E230串口通信
文章出處:【微信公眾號(hào):嵌入式ARM】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
責(zé)任編輯:haq
-
單片機(jī)
+關(guān)注
關(guān)注
6077文章
45513瀏覽量
671474 -
通信
+關(guān)注
關(guān)注
18文章
6405瀏覽量
140101 -
STM32
+關(guān)注
關(guān)注
2310文章
11171瀏覽量
373775
原文標(biāo)題:不會(huì)用國(guó)產(chǎn)單片機(jī)?看這里:GD32E230串口通信
文章出處:【微信號(hào):gh_c472c2199c88,微信公眾號(hào):嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
GD01C 4G路由模組:串口轉(zhuǎn)4G/WiFi/網(wǎng)口通信
串口調(diào)試步驟(適合免驅(qū)動(dòng)的232串口設(shè)備)
GD32VF103串口讀取PM2.5傳感器
rt thread 按照官方視頻一個(gè)字一個(gè)字敲都不行,連main函數(shù)都無(wú)法進(jìn)入,為什么?
GD32的串口DMA收發(fā)數(shù)據(jù)失敗怎么解決?
關(guān)于stm32f4zgt6和mspm0g3507串口通信問(wèn)題
串口轉(zhuǎn)網(wǎng)口網(wǎng)關(guān)有哪些功能應(yīng)用
ESP32用作經(jīng)典藍(lán)牙串口透?jìng)髂K與手機(jī)進(jìn)行串口通信
請(qǐng)問(wèn)K230串口連接方式是什么?
基于GD32E230冰箱柜變頻驅(qū)動(dòng)解決方案
國(guó)產(chǎn)MCU的GD32E230串口通信
評(píng)論