基于HAL库的LED流水灯

基于HAL库的LED流水灯&STM32外部中断模式控制流水灯亮灭

一、STM32CubeMX 下载及安装

如要从零开始,此工具可谓理想之选。

STM32CubeMX工具可简化硬件和外设配置过程,而且用于开发的Nucleo板也拥有较高的性价比。同时也提供了丰富的部件供用户选择。

STM32CubeMX官网

STM32CubeMX下载安装教程

二、使用HAL库完成3只LED红绿灯的周期闪烁

1、新建STM32CubeMx项目

  • 新建项目
    image-20231020192519762
  • 选择芯片型号
    image-20231020194630015

2、配置系统调试接口sys

image-20231020195126088

3、配置时钟RCC

image-20231020195750398

4、配置输出端口

  • 选择PC15、PA5、PB15为输出端口
    image-20231020202357909

5、配置时钟树

image-20231020202915227

6、配置项目

image-20231020204149897 image-20231020204714967

7、创建完成并用keil打开

image-20231020205113618 image-20231020205131644

8、修改main.c实现流水灯

while (1)
  {
	HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET);  // PA5亮
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_SET);  // PA灭
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_15, GPIO_PIN_RESET);  // PB15亮
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_15, GPIO_PIN_SET);  // PB15灭
	HAL_GPIO_WritePin(GPIOC, GPIO_PIN_15, GPIO_PIN_RESET);  // PC15亮
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(GPIOC, GPIO_PIN_15, GPIO_PIN_SET);  // PC15灭
  }

9、编译下载

在这里插入图片描述

二、STM32外部中断模式控制流水灯亮灭

用stm32F103核心板的GPIOA端某一管脚接一个开关(用杜邦线模拟代替)。采用中断模式编程,当开关接高电平时,LED流水灯工作;接低电平时,LED流水灯停止工作。

1、配置开关

  • 使用PB5作为开关引脚使用,中断配置为上升沿和下降沿都触发
  • User Label可以为当前引脚设置一个别名,方便后续自己调用。这里命名为SWITCH
    image-20231021165743969

2、配置GPIO

  • PA5设置为GPIO_Output,起别名LED_A5
  • PB15设置为GPIO_Output,起别名LED_B15
  • PC15设置为GPIO_Output,起别名LED_C15
  • PC13设置为GPIO_Output,做测试用
image-20231021142625568

3、配置EXIT

image-20231021143253635

4、创建完成并用Keil打开

  • 找到stm32f1xx_it.c
222
  • 找到EXTI9_5_IRQHandler这个函数,选中HAL_GPIO_EXTI_IRQHandler这个语句按F12跳到该函数
image-20231021143516416
  • 找到HAL_GPIO_EXTI_Callback这个函数
image-20231020233218521

该函数是给用户自己重写的,可以在这里根据不同的中断来执行不同的处理。在这里我们需要根据PB5的不同中断来实现流水灯的亮灭。

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	//获取B5的电位
        GPIO_PinState pinState = HAL_GPIO_ReadPin(SWITCH_GPIO_Port,SWITCH_Pin);
	//低电位
        if(pinState==GPIO_PIN_RESET)
	{
		 HAL_GPIO_WritePin(LED_A5_GPIO_Port,LED_A5_Pin,GPIO_PIN_RESET);//把A5变为低电位
		 HAL_GPIO_WritePin(LED_B15_GPIO_Port,LED_B15_Pin,GPIO_PIN_RESET);//把A5变为低电位
		 HAL_GPIO_WritePin(LED_C15_GPIO_Port,LED_C15_Pin,GPIO_PIN_RESET);//把A5变为低电位
	}    
}
  • main函数
 while (1)
  {
	HAL_GPIO_WritePin(LED_A5_GPIO_Port,LED_A5_Pin,GPIO_PIN_SET);
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(LED_A5_GPIO_Port,LED_A5_Pin,GPIO_PIN_RESET);
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(LED_B15_GPIO_Port,LED_B15_Pin,GPIO_PIN_SET);
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(LED_B15_GPIO_Port,LED_B15_Pin,GPIO_PIN_RESET);
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(LED_C15_GPIO_Port,LED_C15_Pin,GPIO_PIN_SET);
	HAL_Delay(500); // 延时0.5s
	HAL_GPIO_WritePin(LED_C15_GPIO_Port,LED_C15_Pin,GPIO_PIN_RESET);
	HAL_Delay(500); // 延时0.5s
  }

假如工程中存在了一个__weak 修饰符的函数,用户还可以在工程中重新定义一个与之同名的函数,最终编译器编译的时候,会选择没有__weak 修饰符的函数,如果用户没有重新定义新的函数,那么编译器就会执行 __weak 声明的函数,并且编译器不会报错。

  • 注意:这时会报错有些管脚未定义,要将头文件main.h中的宏定义复制到stm32f1xx_hal_gpio.c中

image-20231021143741265

image-20231021143819374

image-20231021183626627
  • main执行流水灯程序,当进入中断时流水灯熄灭,退出中断时,还原到主程序即流水灯继续运行

5、编译下载

在这里插入图片描述

三、观察GPIO端口的输出波形

1、设置options for target

①Target设置

image-20231014140317340

②Debug设置

image-20231014140817544

Dialog.DLL:DARMSTM.DLL

Parameter:要和自己的单片机型号一致

设置后点击OK,完成设置。

2、点击Debug,进入调试界面

image-20231014141158151

3、选择逻辑分析仪

image-20231014141303122

4、选择要观察的引脚

image-20231021184449502

5、添加引脚

  • 我选择PA5、PB15、PC15

    image-20231021195835128

6、运行

image-20231014143353259

7、观察波形

image-20231021200030515
  • 由波形图得知周期为3s,符合一个LED灯亮灭的周期

213213

8、观察波形

image-20231021200030515
  • 由波形图得知周期为3s,符合一个LED灯亮灭的周期
    image-20231021200030515
好的,以下是基于STM32CubeIDE和HAL库的定时器中断实现LED流水灯的代码: ```c #include "main.h" #include "stm32f4xx_hal.h" /* 定义GPIO引脚 */ #define LED1_Pin GPIO_PIN_0 #define LED1_GPIO_Port GPIOA #define LED2_Pin GPIO_PIN_1 #define LED2_GPIO_Port GPIOA #define LED3_Pin GPIO_PIN_2 #define LED3_GPIO_Port GPIOA #define LED4_Pin GPIO_PIN_3 #define LED4_GPIO_Port GPIOA /* 定义全局变量和函数 */ TIM_HandleTypeDef htim2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_TIM2_Init(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_TIM2_Init(); HAL_TIM_Base_Start_IT(&htim2); /* 启动定时器并开启中断 */ while (1) { /* 主函数不做任何事情 */ } } /* 定时器中断处理函数 */ void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { static uint8_t led_status = 0x01; /* 初始状态下第一个LED亮 */ /* 根据led_status位控制LED的亮灭 */ if (led_status & 0x01) HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, GPIO_PIN_RESET); if (led_status & 0x02) HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_RESET); if (led_status & 0x04) HAL_GPIO_WritePin(LED3_GPIO_Port, LED3_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED3_GPIO_Port, LED3_Pin, GPIO_PIN_RESET); if (led_status & 0x08) HAL_GPIO_WritePin(LED4_GPIO_Port, LED4_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED4_GPIO_Port, LED4_Pin, GPIO_PIN_RESET); /* 更新led_status的值 */ if (led_status == 0x08) led_status = 0x01; else led_status <<= 1; } /* System Clock Configuration */ void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Configure the main internal regulator output voltage */ __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } /** Initializes the CPU, AHB and APB buses clocks */ RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK) { Error_Handler(); } } /* GPIO初始化 */ void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* 使能GPIOA时钟 */ __HAL_RCC_GPIOA_CLK_ENABLE(); /* 配置GPIO引脚为输出模式 */ GPIO_InitStruct.Pin = LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } /* 定时器初始化 */ void MX_TIM2_Init(void) { /* 使能TIM2时钟 */ __HAL_RCC_TIM2_CLK_ENABLE(); /* 初始化htim2的各项参数 */ htim2.Instance = TIM2; htim2.Init.Prescaler = 8399; /* 预分频值 */ htim2.Init.CounterMode = TIM_COUNTERMODE_UP; /* 向上计数模式 */ htim2.Init.Period = 999; /* 自动重装值 */ htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; /* 时钟分频 */ if (HAL_TIM_Base_Init(&htim2) != HAL_OK) { Error_Handler(); } } ``` 注释已经非常详细了,简单来说就是通过定时器中断不断地改变LED的状态,从而实现LED流水灯效果。每当定时器计数器达到自动重装值时,就会产生一次定时器中断,然后在中断处理函数中改变LED的状态。 需要注意的是,这里使用的定时器是TIM2,预分频值为8399,自动重装值为999,因此定时器的计数频率为84MHz / (8399 + 1) = 10kHz,即每隔100ms产生一次定时器中断。如果需要改变LED流水灯的速度,可以调整预分频值和自动重装值。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值