RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE); //使能USART1,GPIOA時鐘
GPIO_InitTypeDef GPIO_InitStructure;
//USART1_TX GPIOA.9
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //複用推輓輸出
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9
//USART1_RX GPIOA.10初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空輸入
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10
NVIC_InitTypeDef NVIC_InitStructure;
//Usart1 NVIC 配置
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ; //搶佔優先級3
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子優先級3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根據指定的參數初始化NVIC寄存器
//如果需要接收串口數據,則開啓串口接收中斷
USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//開啓串口接受中斷
typedef struct {
uint32_t USART_BaudRate; // 波特率
uint16_t USART_WordLength; // 字長
uint16_t USART_StopBits; // 停止位
uint16_t USART_Parity; // 校驗位
uint16_t USART_Mode; // USART 模式
uint16_t USART_HardwareFlowControl; // 硬件流控制
} USART_InitTypeDef;
USART_InitTypeDef USART_InitStructure;
USART_InitStructure.USART_BaudRate = bound;//串口波特率
USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字長爲8位數據格式
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; //收發模式
USART_Init(USART1, &USART_InitStructure); //初始化串口1
USART_Cmd(USART1, ENABLE); //使能串口1
//發送一個字節
void USART1_Send_Byte(u8 Data)
{
USART_GetFlagStatus(USART1, USART_FLAG_TC);
USART_SendData(USART1,Data);
while( USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET );
}
//發送字符串,遇到字符串結尾標誌'\0'結束
void USART1_Send_String(u8 *Data)
{
while(*Data)
USART1_Send_Byte(*Data++);
}
//按長度發送字符串,這種方法可以發送含0x00的字符串
void USART1_Send_String_By_Lens(u8 *Data, int Len)
{
int i;
for(i=0; i<Len; i++)
{
USART_SendData(USART1, Data[i]);
while(USART_GetFlagStatus(USART1, USART_FLAG_TC)==RESET); //串口1發送數據
}
}
//重定向printf函數發送字符串,一般使用此函數直接輸出打印調試信息,使用方法跟C語言中的使用方法一致。
int fputc(int ch, FILE *f)
{
USART_SendData( DEBUG_USARTx, (uint8_t) ch);
/* 等待發送完畢 */
while (USART_GetFlagStatus(DEBUG_USARTx, USART_FLAG_TXE) == RESET);
return ch;
}
//串口1中斷服務程序,此接收的數據是以0x0D、0x0A結尾爲標誌的數據幀。
void USART1_IRQHandler(void)
{
u8 Res;
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中斷(接收到的數據必須是0x0d 0x0a結尾)
{
USART_ClearFlag(USART1, USART_IT_RXNE); //清除標誌位
Res =USART_ReceiveData(USART1); //讀取接收到的數據
if((USART_RX_STA&0x8000)==0)//接收未完成
{
if(USART_RX_STA&0x4000)//接收到了0x0d
{
if(Res==0x0D)
USART_RX_STA|=0x4000;
else if(Res!=0x0a)
USART_RX_STA=0;//接收錯誤,重新開始
else
USART_RX_STA|=0x8000; //接收完成了
}
else //還沒收到0X0D
{
if(Res==0x0d)
USART_RX_STA|=0x4000;
else
{
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))
USART_RX_STA=0;//接收數據錯誤,重新開始接收
}
}
}
}
}
//接收完數據之後,在main函數中對接收到的數據進行處理。
if(USART_RX_STA&0x8000)
{
//得到此次接收到的數據長度,即USART_RX_BUF數組中的有效數據長度
uart1Len=USART_RX_STA&0x3f;
//對接收到的數據進行數據處理,接收的數據暫存在USART_RX_BUF數組中
//... ...
USART_RX_STA=0;
memset(USART_RX_BUF, 0, sizeof(USART_RX_BUF)); //清空數組
}
喜歡請關注微信公衆號:程序員小哈
公衆號內容面向在校大學生、電子愛好者、嵌入式工程師;
涉及電子製作、模塊使用、單片機技術、物聯網相關知識分享;
軟硬件全棧工程師,玩模塊,學硬件,帶你從0走到1
若覺得本次分享的文章對您有幫助,隨手關注並轉發分享,也是對我的支持。