这里使用的是stm32f103芯片, 使用串口向单片机发送数据,然后串口原样发送给PC。
1.变量:
uart.h中声明以下变量,方便在主函数中调用。其中RxMax为选定合适的最大接收值,用于计算接收个数,同时防止数组越界。
extern UART_HandleTypeDef UART1;
extern uint8_t RX_STATE;
extern uint8_t TX_DATA[256],RX_DATA[256];
extern uint8_t RxCounter;
#define RxMax 256
uart.c中变量定义如下:串口的初始化以及很多函数都需要用到串口句柄,不妨使用一个全局变量,同时定义一个合适大小的收发缓冲区。
UART_HandleTypeDef UART1;
uint8_t TX_DATA[256],RX_DATA[256];
uint8_t RX_STATE; //TX_DATA写入标志
uint8_t RxCounter; //用于计算收到数据的长度
2.串口初始化:
void UART1_Init(uint32_t bandrate){ //串口初始化,返回值:无,参数:波特率
UART1.Instance = USART1;
UART1.Init.BaudRate =bandrate;
UART1.Init.HwFlowCtl =UART_HWCONTROL_NONE ;
UART1.Init.Mode = UART_MODE_TX_RX;
UART1.Init.Parity =UART_PARITY_NONE ;
UART1.Init.StopBits =USART_STOPBITS_1;
UART1.Init.WordLength =UART_WORDLENGTH_8B;
HAL_UART_Init(&UART1);
__HAL_UART_ENABLE_IT(&UART1,UART_IT_IDLE); //使能空闲中断
HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax); //使能接收
}
void HAL_UART_MspInit(UART_HandleTypeDef *huart){ //串口相关硬件的初始化
if(huart->Instance == USART1){
__HAL_RCC_GPIOA_CLK_ENABLE();//打开gpioa的时钟
__HAL_RCC_USART1_CLK_ENABLE();//打开串口1的时钟
GPIO_InitTypeDef GPIO_Init;
GPIO_Init.Mode = GPIO_MODE_AF_PP ;
GPIO_Init.Pin = GPIO_PIN_9;//串口1的TX
GPIO_Init.Speed =GPIO_SPEED_FREQ_MEDIUM;
HAL_GPIO_Init(GPIOA,&GPIO_Init);
GPIO_Init.Mode = GPIO_MODE_AF_INPUT ;
GPIO_Init.Pin = GPIO_PIN_10;//串口1的RX
GPIO_Init.Pull = GPIO_NOPULL; //上下拉只有输入模式有效,输出模式上下拉电阻被禁止
HAL_GPIO_Init(GPIOA,&GPIO_Init);
HAL_NVIC_SetPriority(USART1_IRQn ,3,0);//配置 中断优先级
HAL_NVIC_EnableIRQ(USART1_IRQn);//使能中断
}
}
3.中断处理:
如果使用简单的串口中断接收方式,预定接收10个字节,但是如果只收到5个字节,那么串口的接收计数器RxXfercounter不能见到0,也就无法调用中断接收完成的回调函数。但是,在收到五个字节后并且没有继续收到数据,串口会出现空闲帧,我们在串口的总中断处理函数中,判断是否发生了空闲中断,然后清除中断标志位,并进行想要的操作。
void USART1_IRQHandler(void){
HAL_UART_IRQHandler(&UART1); //这个函数处理UART中断请求。
if(__HAL_UART_GET_FLAG(&UART1,UART_FLAG_IDLE)){ //如果产生了空闲中断
__HAL_UART_CLEAR_IDLEFLAG(&UART1);
RxCounter = 0;
RxCounter += (RxMax - UART1.RxXferCount);
HAL_UART_AbortReceive_IT(&UART1);
}
}
以上程序在发生空闲中断的条件下,计算收到的字节,并调用中断停止接收(HAL_UART_AbortReceive_IT),该函数会调用中断停止接收回调函数(HAL_UART_AbortReceiveCpltCallback)。 它是一个弱声明函数,我们需要对它进行强声明。
void HAL_UART_AbortReceiveCpltCallback(UART_HandleTypeDef *huart){
if(huart->Instance == USART1){ //停止接收中断回调函数
memcpy(TX_DATA,RX_DATA,RxCounter);
RX_STATE=1;
}
HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax);
}
在这个函数中,我们把收到的数拷贝到TX_DATA中,并进行置位。 主函数如下,实现把数据发回电脑。
int main(){
HAL_Init();
MyRcc_Init(); //时钟初始化,读者需要更改为自己写的时钟初始化函数
UART1_Init(9600);
LED_Init();
while(1){
if(RX_STATE==1){
RX_STATE=0;
HAL_UART_Transmit_IT(&UART1,TX_DATA,RxCounter);
}
}
}
4.完整代码
uart.c
#include "uart.h"
UART_HandleTypeDef UART1;
uint8_t TX_DATA[256],RX_DATA[256];
uint8_t RX_STATE; //TX_DATA写入标志
uint8_t RxCounter; //用于计算收到数据的长度
void UART1_Init(uint32_t bandrate){ //串口初始化
UART1.Instance = USART1;
UART1.Init.BaudRate =bandrate;
UART1.Init.HwFlowCtl =UART_HWCONTROL_NONE ;
UART1.Init.Mode = UART_MODE_TX_RX;
UART1.Init.Parity =UART_PARITY_NONE ;
UART1.Init.StopBits =USART_STOPBITS_1;
UART1.Init.WordLength =UART_WORDLENGTH_8B;
HAL_UART_Init(&UART1);
__HAL_UART_ENABLE_IT(&UART1,UART_IT_IDLE); //使能空闲中断
HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax); //使能接收
}
void HAL_UART_MspInit(UART_HandleTypeDef *huart){
if(huart->Instance == USART1){
__HAL_RCC_GPIOA_CLK_ENABLE();//打开gpioa的时钟
__HAL_RCC_USART1_CLK_ENABLE();//打开串口1的时钟
GPIO_InitTypeDef GPIO_Init;
GPIO_Init.Mode = GPIO_MODE_AF_PP ;
GPIO_Init.Pin = GPIO_PIN_9;//串口1的TX
GPIO_Init.Speed =GPIO_SPEED_FREQ_MEDIUM;
HAL_GPIO_Init(GPIOA,&GPIO_Init);
GPIO_Init.Mode = GPIO_MODE_AF_INPUT ;
GPIO_Init.Pin = GPIO_PIN_10;//串口1的RX
GPIO_Init.Pull = GPIO_NOPULL; //上下拉只有输入模式有效,输出模式上下拉电阻被禁止
HAL_GPIO_Init(GPIOA,&GPIO_Init);
HAL_NVIC_SetPriority(USART1_IRQn ,3,0);//配置 中断优先级
HAL_NVIC_EnableIRQ(USART1_IRQn);//使能中断
}
}
void USART1_IRQHandler(void){
HAL_UART_IRQHandler(&UART1); //这个函数处理UART中断请求。
if(__HAL_UART_GET_FLAG(&UART1,UART_FLAG_IDLE)){ //如果产生了空闲中断
__HAL_UART_CLEAR_IDLEFLAG(&UART1);
RxCounter = 0;
RxCounter += (RxMax - UART1.RxXferCount);
HAL_UART_AbortReceive_IT(&UART1);
}
}
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart){
if(huart->Instance == USART1){
}
}
void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart){
if(huart->Instance == USART1){ //发送完成中断回调函数
}
}
void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart){
if(huart->Instance == USART1){ //错误中断回调函数
}
}
void HAL_UART_AbortReceiveCpltCallback(UART_HandleTypeDef *huart){
if(huart->Instance == USART1){ //停止接收中断回调函数
memcpy(TX_DATA,RX_DATA,RxCounter);
RX_STATE=1;
// HAL_UART_Receive_IT(&UART1,RX_DATA,RxCounter);
}
HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax);
}
uart.h
#ifndef __UART_H
#define __UART_H
#include "stdint.h"
#include "string.h"
#include "stm32f1xx_hal.h"
extern UART_HandleTypeDef UART1;
extern uint8_t RX_STATE;
extern uint8_t TX_DATA[256],RX_DATA[256];
extern uint8_t RxCounter;
#define RxMax 256
void UART1_Init(uint32_t bandrate);
#endif
main.c
#include "stm32f1xx_hal.h"
#include "myrcc.h" //读者需要更换为自己的时钟文件,或者拷备以下代码
#include "uart.h"
int main(){
HAL_Init();
MyRcc_Init();
UART1_Init(9600);
LED_Init();
while(1){
if(RX_STATE==1){
RX_STATE=0;
HAL_UART_Transmit_IT(&UART1,TX_DATA,RxCounter);
}
}
}
myrcc.c,
#include "stm32f1xx_hal.h"
#include "myrcc.h"
void MyRcc_Init(void){ //实现使用外部时钟源通过pll倍频
/*时钟初始化
1.初始化RCC振荡器
2.初始化时钟总线*/
RCC_OscInitTypeDef RCC_OscInitType;
RCC_OscInitType.OscillatorType = RCC_OSCILLATORTYPE_HSE ;//选择振荡电路;外部高速时钟
RCC_OscInitType.HSEState = RCC_HSE_ON; //打开hse(外部高速时钟
RCC_OscInitType.HSEPredivValue = RCC_HSE_PREDIV_DIV1; //预分频
RCC_OscInitType.PLL.PLLState = RCC_PLL_ON; //pll倍频开启
RCC_OscInitType.PLL.PLLSource = RCC_PLLSOURCE_HSE; //pll时钟源:HSE
RCC_OscInitType.PLL.PLLMUL = RCC_PLL_MUL9;
HAL_RCC_OscConfig(&RCC_OscInitType); //参数中指定的参数初始化RCC振荡器
RCC_ClkInitTypeDef RCC_ClkInitStruct;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; //AHB分频系数
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; //APB1分频系数
RCC_ClkInitStruct.APB2CLKDivider =RCC_HCLK_DIV1; //APB2分频系数
RCC_ClkInitStruct.ClockType =RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; //需要配置的时钟
RCC_ClkInitStruct.SYSCLKSource =RCC_SYSCLKSOURCE_PLLCLK; //系统时钟输入源
HAL_RCC_ClockConfig(&RCC_ClkInitStruct,FLASH_LATENCY_2);//根据指定初始化CPU、AHB和APB总线时钟,RCC_ClkInitStruct中的*参数。
}
myrcc.h
#ifndef __MYRCC_H
#define __MYRCC_H
void MyRcc_Init(void);
#endif