STM32G431-基于HAL库(第十四届蓝桥杯嵌入式模拟题2)


前言

相关说明:

开发板:CT117E-M4(STM32G431RB 蓝桥杯嵌入式比赛板)
开发环境: CubeMX+Keil5
涉及题目:第十四届蓝桥杯嵌入式模拟题2
题目难点:难度相对较低
题目来源:4T


CubeMX配置、主要函数代码及说明:

一、CubeMX配置(第十四届模拟题2完整版)

1.使能外部高速时钟:
在这里插入图片描述
2.配置时钟树:
在这里插入图片描述
3.GPIO:

在这里插入图片描述

4.TIM2(PA1输出):在这里插入图片描述在这里插入图片描述
5.TIM3(同上,PA7输出):在这里插入图片描述
在这里插入图片描述
6.TIM6(控制LED闪烁):
在这里插入图片描述

7.TIM7(判断串口接收,同上,5ms中断):
在这里插入图片描述

8.NVIC(输入捕获中断配置):在这里插入图片描述

二、代码相关定义、声明

1.函数声明

main.c
void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim);//输入捕获中断函数 计算输入信号频率
void LCD_Init_Show(void);	//LCD初始化显示
void LCD_Refresh(void);		//LCD更新显示 

gpio.h
void KEY_Scan(void);//按键扫描
void LED_AllClose(uint8_t *LED_Close);//LED更新显示
void LED_Change(void);//LED状态改变

adc.h
double ADC_GetValue(void);//获取R37电压值

time.h
void PWM_Out(double R37_V,uint32_t FRQ,uint8_t	R);//PWM输出

2.宏定义

在CubeMX中可以配置User Label选项即可生成对应宏定义
在这里插入图片描述生成宏定义:

main.h
#define LED6_Pin GPIO_PIN_13
#define LED6_GPIO_Port GPIOC
#define LED7_Pin GPIO_PIN_14
#define LED7_GPIO_Port GPIOC
#define LED8_Pin GPIO_PIN_15
#define LED8_GPIO_Port GPIOC
#define KEY4_Pin GPIO_PIN_0
#define KEY4_GPIO_Port GPIOA
#define KEY1_Pin GPIO_PIN_0
#define KEY1_GPIO_Port GPIOB
......此处省略

自定义宏定义(灯的控制,PD2引脚控制):

main.h
#define OFF GPIO_PIN_SET
#define ON  GPIO_PIN_RESET

#define LED1(a) HAL_GPIO_WritePin(LED1_GPIO_Port,LED1_Pin,a)
#define LED2(a) HAL_GPIO_WritePin(LED2_GPIO_Port,LED2_Pin,a)
#define LED3(a) HAL_GPIO_WritePin(LED3_GPIO_Port,LED3_Pin,a)
#define LED4(a) HAL_GPIO_WritePin(LED4_GPIO_Port,LED4_Pin,a)
#define LED5(a) HAL_GPIO_WritePin(LED5_GPIO_Port,LED5_Pin,a)
#define LED6(a) HAL_GPIO_WritePin(LED6_GPIO_Port,LED6_Pin,a)
#define LED7(a) HAL_GPIO_WritePin(LED7_GPIO_Port,LED7_Pin,a)
#define LED8(a) HAL_GPIO_WritePin(LED8_GPIO_Port,LED8_Pin,a)

#define LED_All(a) HAL_GPIO_WritePin(LED8_GPIO_Port,GPIO_PIN_All,a) 

#define LOCK_HIGH() HAL_GPIO_WritePin(LOCK_GPIO_Port,LOCK_Pin,GPIO_PIN_SET) 
#define LOCK_LOW()  HAL_GPIO_WritePin(LOCK_GPIO_Port,LOCK_Pin,GPIO_PIN_RESET) 

3.变量定义

char buf[20];				//字符串拼接数组
uint8_t LED[4] = {
    
    1,0,1,0}; //LED状态数组

uint16_t PA7_HZ = 1000;		//PA7输出信号频率
uint8_t PA7_D = 10;			//PA7输出信号占空比
uint16_t PA1_HZ = 1000;		//PA1输出信号频率
uint8_t PA1_D = 10;			//PA1输出信号占空比
uint8_t page = 1;			//page1:PA1数据页 page2:PA7数据页
uint8_t rec_byte;			//串口数据存储
uint8_t mode = 1;			//当前模式 1:按键控制 0:串口控制
uint8_t rec_flag = 0;		//串口接收标志位
uint16_t rec_num = 0;		//串口一次接收字节数

三、主要函数

1.按键扫描

此处将按键按下后的操作都封装成独立的函数调用:
KEY1 – 改变频率
KEY2 – 改变占空比
KEY3 – 数据页改变
KEY4 – 操作模式改变

void KEY_Process()
{
    
    
	if((HAL_GPIO_ReadPin(KEY1_GPIO_Port,KEY1_Pin) == GPIO_PIN_RESET))
	{
    
    
		HAL_Delay(10);
		if(HAL_GPIO_ReadPin(KEY1_GPIO_Port,KEY1_Pin) == GPIO_PIN_RESET)
		{
    
    
			HZ_Change();//频率改变
			while(HAL_GPIO_ReadPin(KEY1_GPIO_Port,KEY1_Pin) == GPIO_PIN_RESET);
		}
	}
	
	else if(HAL_GPIO_ReadPin(KEY2_GPIO_Port,KEY2_Pin) == GPIO_PIN_RESET)
	{
    
    
		HAL_Delay(10);
		if(HAL_GPIO_ReadPin(KEY2_GPIO_Port,KEY2_Pin) == GPIO_PIN_RESET)
		{
    
    
			D_Change();//占空比改变
			while(HAL_GPIO_ReadPin(KEY2_GPIO_Port,KEY2_Pin) == GPIO_PIN_RESET);
		}
	}
	
	else if(HAL_GPIO_ReadPin(KEY3_GPIO_Port,KEY3_Pin) == GPIO_PIN_RESET && (mode == 1))
	{
    
    
		HAL_Delay(10);
		if(HAL_GPIO_ReadPin(KEY3_GPIO_Port,KEY3_Pin) == GPIO_PIN_RESET)
		{
    
    
			Page_Change();//数据页改变
			while(HAL_GPIO_ReadPin(KEY3_GPIO_Port,KEY3_Pin) == GPIO_PIN_RESET);
		}
	}
	
	else if(HAL_GPIO_ReadPin(KEY4_GPIO_Port,KEY4_Pin) == GPIO_PIN_RESET)
	{
    
    
		HAL_Delay(10);
		if(HAL_GPIO_ReadPin(KEY4_GPIO_Port,KEY4_Pin) == GPIO_PIN_RESET)
		{
    
    
			Mode_Change();//模式改变
			while(HAL_GPIO_ReadPin(KEY4_GPIO_Port,KEY4_Pin) == GPIO_PIN_RESET);
		}
	}
}

2.各参数控制

输出信号频率、占空比、数据页以及操作模式改变

/*相关变量*/
uint16_t PA7_HZ = 1000;			//PA7输出信号频率
uint8_t PA7_D = 10;				//PA7输出信号占空比
uint16_t PA1_HZ = 1000;			//PA1输出信号频率
uint8_t PA1_D = 10;				//PA1输出信号占空比
uint8_t page = 1;				//page1:PA1数据页 page2:PA7数据页
uint8_t mode = 1;				//当前模式 1:按键控制 0:串口控制

const uint16_t HZ_pace = 1000;
const uint16_t HZ_max = 10000;
const uint16_t HZ_min = 1000;
const uint8_t D_pace = 10;
const uint8_t D_max = 90;
const uint8_t D_min = 10;

/*频率改变*/
void HZ_Change()
{
    
    
	if(page == 1)//PA1
	{
    
    
		PA1_HZ+=HZ_pace;
		if(PA1_HZ > HZ_max)
		{
    
    
			PA1_HZ = HZ_min;
		}
	}
	else if(page == 2)//PA7
	{
    
    
		PA7_HZ+=HZ_pace;
		if(PA7_HZ > HZ_max)
		{
    
    
			PA7_HZ = HZ_min;
		}
	}
	
	PWM_change();//频率、占空比改变
	HAL_TIM_Base_Start(&htim6);//开启定时器
}

/*占空比改变*/
void D_Change()
{
    
    
	if(page == 1)//PA1
	{
    
    
		PA1_D+=D_pace;
		if(PA1_D > D_max)
		{
    
    
			PA1_D = D_min;
		}
	}
	else if(page == 2)//PA7
	{
    
    
		PA7_D+=D_pace;
		if(PA7_D > D_max)
		{
    
    
			PA7_D = D_min;
		}
	}
	
	PWM_change();
}

/*模式改变*/
void Mode_Change()
{
    
    
	mode = !mode;//模式改变
	
	if(mode == 1)//按键模式
	{
    
    
		LED[3] = 0;
	}
	else//串口模式
	{
    
    
		LED[3] = 1;
	}
	
	LED_Control(LED);//LED更新
}

/*数据页改变*/
void Page_Change()
{
    
    
	if(page == 1)//PA1数据页->PA7数据页
	{
    
    
		page = 2;
		sprintf(buf,"    PA7    "); 
		LED[1] = 1;
		LED[2] = 0;
	}
	else if(page == 2)//PA7数据页->PA1数据页
	{
    
    
		page = 1;
		sprintf(buf,"    PA1    "); 
		LED[1] = 0;
		LED[2] = 1;
	}
	
	LED_Control(LED);//LED更新
	LCD_DisplayStringLine(Line2,(u8*)buf);//JLCD更新
}

3.LCD显示

共有两个函数:
1.LCD_InitShow(),在上电启动后对LCD进行初始化显示操作。
2.LCD_Refresh(),LCD更新显示,数据更新后需要实时进行更新显示。

/*LCD初始化显示函数*/
void LCD_InitShow()
{
    
    
	LCD_Clear(Black);
	LCD_SetBackColor(Black);
	LCD_SetTextColor(White);
	
	LCD_DisplayStringLine(Line0,(u8*)"                    ");
	LCD_DisplayStringLine(Line1,(u8*)"                    ");
	LCD_DisplayStringLine(Line2,(u8*)"    PA1             ");
	LCD_DisplayStringLine(Line3,(u8*)"                    ");
	
	sprintf(buf,"    F:%d    ",PA1_HZ);
	LCD_DisplayStringLine(Line4,(u8*)buf);
	
	LCD_DisplayStringLine(Line5,(u8*)"                    ");
	
	sprintf(buf,"    D:%d    ",PA1_D);
	LCD_DisplayStringLine(Line6,(u8*)buf);
	LCD_DisplayStringLine(Line7,(u8*)"                    ");
	LCD_DisplayStringLine(Line8,(u8*)"                    ");
	LCD_DisplayStringLine(Line9,(u8*)"                    ");
}

/*LCD更新函数 page1:PA1 page2:PA7*/
void LCD_Refresh()
{
    
    
	if(page == 1)
	{
    
    
		sprintf(buf,"    F:%d    ",PA1_HZ);
		LCD_DisplayStringLine(Line4,(u8*)buf);
		sprintf(buf,"    D:%d    ",PA1_D);
		LCD_DisplayStringLine(Line6,(u8*)buf);
	}
	else if(page == 2)
	{
    
    
		sprintf(buf,"    F:%d    ",PA7_HZ);
		LCD_DisplayStringLine(Line4,(u8*)buf);
		sprintf(buf,"    D:%d    ",PA7_D);
		LCD_DisplayStringLine(Line6,(u8*)buf);
	}
}

4.输出信号改变

直接对TIM2和TIM3的寄存器操作即可
ARR – 寄存器存储的是重装载值
CCRx – 定时器x通道的比较值
要使定时器输出对应频率的信号,重装载值=定时器工作频率/目标频率
控制该信号的占空比,比较值=重装载值*占空比

void PWM_change()
{
    
    
	if(page == 1)//PA1
	{
    
    
		TIM2->ARR = 1000000/PA1_HZ;//改变频率
		TIM2->CCR2 = TIM2->ARR*PA1_D/100;//改变占空比
		TIM2->CNT = 0;//清空计数值
	}
	else if(page == 2)
	{
    
    
		TIM3->ARR = 1000000/PA7_HZ;//改变频率
		TIM3->CCR2 = TIM3->ARR*PA7_D/100;//改变占空比
		TIM3->CNT = 0;//清空计数值
	}
}

5.串口接收

每次串口接收到数据都重新开启定时器,这样能使最后一字节数据接收完5ms后才开始对接收数据进行处理,而不是每接收一个字节数据都去处理一次

/*相关变量*/
uint8_t rec_flag = 0;				//串口接收标志位
uint16_t rec_num = 0;				//串口一次接收字节数

/*定时器回调函数*/
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) 
{
    
    
	...
	if(htim == &htim7)
	{
    
    
		HAL_TIM_Base_Stop_IT(&htim7);//关闭定时器7
		rec_flag = 1;//接收检测标志位置1
	}
	...
}

/*串口接收回调函数*/
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) 
{
    
    
	rec_num++;//接收字节数增加
	HAL_UART_Receive_IT(&huart1,&rec_byte,1);//再次打开串口接收中断
	TIM7->CNT = 0;//定时器7计数值清零 重新计时
	HAL_TIM_Base_Start_IT(&htim7);//开始计时
}

6.Main函数

注意调用PWM的start函数打开PWM、以中断方式打开定时器以及打开串口接收中断等

int main(void)
{
    
    
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_TIM6_Init();
  MX_USART1_UART_Init();
  MX_TIM2_Init();
  MX_TIM3_Init();
  MX_TIM7_Init();
  /* USER CODE BEGIN 2 */
	LCD_Init();//LCD初始化
	LCD_InitShow();//LCD初始化显示
	LED_Control(LED);//LED初始化
	
	HAL_UART_Receive_IT(&huart1,&rec_byte,1);//打开串口接收中断
	
	HAL_TIM_Base_Start_IT(&htim6);//开启定时器6
	HAL_TIM_PWM_Start(&htim2,TIM_CHANNEL_2);//PWM启动
	HAL_TIM_PWM_Start(&htim3,TIM_CHANNEL_2);//PWM启动
	
	printf("uart test...\n");
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    
    
		KEY_Process();//按键处理
		LCD_Refresh();//LCD更新
		
		if(rec_flag == 1)//串口数据接收标志位
		{
    
    
			rec_flag = 0;//标志位清零
			Check_UartRec();//串口数据处理
		}
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

四、实验结果

1.数据页1

上电默认值F为1000 D为10
请添加图片描述

2.数据页2

上电默认值F为1000 D为10
请添加图片描述

3.输出频率

默认D值为10,占空比为10%,F为1000,输出频率为1KHz
请添加图片描述

4.串口数据

1.按键控制模式:
在这里插入图片描述2.串口控制模式,数据长度不符:
在这里插入图片描述3.切换到PA7数据页,操作成功:
在这里插入图片描述

五、查漏补缺!

1.蓝桥杯嵌入式赛前梳理
2.一文看懂如何使用RTC秒中断
3.一文看懂如何解决LED与LCD冲突
4.一文看懂如何玩转显示高亮

六、源码(转载请注明出处)

在这里插入图片描述


总结

以上就是全部内容,如有错误请批评指正。

猜你喜欢

转载自blog.csdn.net/Octopus1633/article/details/129891787