stm32的串口接收字符串以十六進(jìn)制數(shù)
掃描二維碼
隨時(shí)隨地手機(jī)看文章
#include "pbdata.h"
uint8_t TxBuffer1[] = "USART Interrupt Example: This isUSART1 DEMO";
uint8_t RxBuffer1[],rec_f,tx_flag;
volatile uint8_t TxCounter1 = 0x00;
volatile uint8_t RxCounter1 = 0x00;
uint32_t Rec_Len;
int main(void)
{
u8 a=0;
RCC_Configuration();
NVIC_Configuration();
GPIO_Configuration();
USART_Config(USART1);
while(1)
{
if(rec_f==1)
{
rec_f=0;
USART_OUT(USART1,&TxBuffer1[0]);
if(a==0){GPIO_SetBits(GPIOA, GPIO_Pin_2); a=1;}
else{GPIO_ResetBits(GPIOA, GPIO_Pin_2);a=0; }
}
}
}
這是主函數(shù)部分,在主函數(shù)中只有幾個(gè)函數(shù)的初始化,還有就是定義的數(shù)組和標(biāo)志位。
在一般的串口歷程中大家會看到的就是定義一個(gè)緩沖區(qū),將接收到的串口數(shù)據(jù)通過串口中斷存放到緩沖區(qū)中然后在發(fā)送到串口中,但是在接收字符串的時(shí)候就要用到逐位發(fā)送,新手自己有些不了程序,所以只能一直處于蒙著的狀態(tài)。其實(shí)個(gè)人感覺整點(diǎn)原子的程序?qū)懙恼娴耐玫?,建議新手開始學(xué)習(xí)的時(shí)候看他的程序,有的人就是不喜歡他寫程序的風(fēng)格,這個(gè)因人而異,在這里只是建議一下。原子的串口就給出了字符串就收的歷程,但是用這個(gè)歷程的時(shí)候結(jié)尾必須要是0d 0a結(jié)尾的也就是空格和回車。
在這了給出一個(gè)自己定義的頭和尾的串口程序。
串口接收字符串的原理和接收單字符的差不多,只是在接收的時(shí)候定義的緩沖區(qū)是一個(gè)數(shù)組,將接受到的數(shù)據(jù)存放到數(shù)組中,在從數(shù)組中讀出想要的十六進(jìn)制數(shù)在主程序中調(diào)用。
以上的主函數(shù)中定義個(gè)幾個(gè)數(shù)組
uint8_t TxBuffer1[] = "USART Interrupt Example: This isUSART1 DEMO";
uint8_t RxBuffer1[],rec_f,tx_flag;
volatile uint8_t TxCounter1 = 0x00;
volatile uint8_t RxCounter1 = 0x00;
第一個(gè)是發(fā)送的緩沖區(qū)數(shù)組,將接受的數(shù)據(jù)放到這里發(fā)送到串口。
第二個(gè)是接收緩沖區(qū)的數(shù)組,將接受回來的數(shù)據(jù)放到這里面。
下面兩個(gè)是定義的變量,因?yàn)榻邮盏臅r(shí)候是逐個(gè)位就收的所以接收以為就要將地址加一位存放到數(shù)組中,否則就會出現(xiàn)發(fā)送的數(shù)據(jù)直接受到一位的現(xiàn)象,應(yīng)為由于地址沒有加1所以導(dǎo)致數(shù)據(jù)被覆蓋掉了。
在主函數(shù)中有幾個(gè)調(diào)用的函數(shù),就是串口初始化的函數(shù),
RCC_Configuration();
NVIC_Configuration();
GPIO_Configuration();
USART_Config(USART1);
首先是 RCC_Configuration();函數(shù)
由于是我自己寫的模板所以喜歡講所有的時(shí)鐘都放到一個(gè)函數(shù)中。這個(gè)根據(jù)個(gè)人喜好而定。
voidRCC_Configuration(void)
{
SystemInit();
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
RCC_APB2PeriphClockCmd( RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB
|RCC_APB2Periph_GPIOD|RCC_APB2Periph_AFIO , ENABLE);
}
NVIC_Configuration();中斷向量配置,如果沒有需要串口搶占的直接默認(rèn)優(yōu)先級就好
voidNVIC_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
/* Configure the NVIC Preemption Priority Bits*/
/* Configure one bit for preemption priority*/ NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);
NVIC_InitStructure.NVIC_IRQChannel= USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority= 0;
NVIC_InitStructure.NVIC_IRQChannelCmd =ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
GPIO_Configuration();這個(gè)是一個(gè)led燈的配置,功能是發(fā)送數(shù)據(jù)時(shí)單片機(jī)成功接收后會出現(xiàn)電平反轉(zhuǎn),也就是亮滅變化。同時(shí)串口的IO口初始化也在這里。
voidGPIO_Configuration(void)
{
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed =GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_AF_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
GPIO_InitStructure.GPIO_Speed =GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(GPIOD, &GPIO_InitStructure);
GPIO_ResetBits(GPIOD, GPIO_Pin_13);
}
USART_Config(USART1); 最后就是串口配置了。
voidUSART_Config(USART_TypeDef* USARTx)
{
USART_InitStructure.USART_BaudRate = 9600;
USART_InitStructure.USART_WordLength =USART_WordLength_8b;
USART_InitStructure.USART_StopBits =USART_StopBits_1;
USART_InitStructure.USART_Parity =USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl= USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode =USART_Mode_Rx | USART_Mode_Tx;
/* Configure USART1 */
USART_Init(USARTx, &USART_InitStructure);
/* Enable USART1 Receive and Transmitinterrupts */
USART_ITConfig(USART1, USART_IT_RXNE,ENABLE);
USART_ITConfig(USART1, USART_IT_TXE, ENABLE);
/* Enable the USART1 */
USART_Cmd(USART1, ENABLE);
}
這些都是固定的東西按照手冊配置就行,不過多廢話了,主要的部分是在中斷函數(shù)中
extern uint8_tTxBuffer1[];
extern uint8_tTxBuffer2[];
extern uint8_tRxBuffer1[];
extern uint8_tRxBuffer2[];
extern volatileuint8_t RxCounter1;
extern volatileuint8_t RxCounter2;
extern uint8_trec_f,tx_flag;
以上用到的是extern的定義,目的是在外部函數(shù)中可以調(diào)用。下面是串口函數(shù)的編寫。
我直接將注釋打到語句的后面,在最后又源程序,可下載調(diào)試。
voidUSART1_IRQHandler(void)
{
unsigned int i;//定義一個(gè)變量,在后面以為用
if(USART_GetITStatus(U