基于stm32f103c8t6、HAL库定时器控制灯亮灭&&PWM实现呼吸灯

一.定时器基本知识

定时器分类

1.基本定时器 基本定时器是一个简单的定时器,通常用于生成精确的时间延迟。它具有单个16位计数器,可以通过设置预分频系数实现不同的计数速度。基本定时器通常用于简单的定时任务。 2.通用定时器 通用定时器是一种更为复杂和功能强大的定时器,通常用于更复杂的定时和计数任务。STM32系列微控制器通常配备了多个通用定时器,如TIM1、TIM2等。通用定时器具有多个16位或32位计数器,可以通过设置预分频系数和计数周期来实现不同的计数速度和定时精度。 定时器应用 STM32定时器广泛应用于各种领域,包括工业控制、通信、嵌入式系统等。以下是一些常见的定时器应用场景:

1.定时中断:可以使用定时器生成周期性的中断信号,用于定时任务的触发和处理。 2.计时测量:可以使用定时器进行时间的测量和计算,如测量脉冲宽度、测量信号的频率等。 3.PWM输出:可以使用定时器生成精确的脉宽调制信号,用于控制电机速度、LED亮度等。 4.输入捕获:可以使用定时器进行外部信号的捕获,用于测量外部事件的时间戳。

定时器配置 为了配置和控制STM32定时器,我们可以使用相应的库函数或直接操作寄存器。以下是一些常用的配置参数:

1.预分频系数:用于设置定时器的计数速度,可以根据需要进行调整。 2.计数周期:用于设置定时器的计数周期,决定定时器溢出的时间间隔。 3.工作模式:定时器可以工作在不同的模式下,如定时器模式、计数器模式、PWM模式等。 4.中断使能:可以选择是否使能定时器的中断功能,用于触发中断任务。

二.定时器的应用1

1. 使用STM32F103的 Tim2~Tim5其一定时器的某一个通道pin(与GPIOx管脚复用,见下图),连接一个LED,用定时器计数方式,控制LED以2s的频率周期性地亮-灭 代码:

#include "led.h"
​
void LED_Config(void)
{
    GPIO_InitTypeDef GPIO_InitStruct;
    
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    
    GPIO_InitStruct.GPIO_Mode=GPIO_Mode_Out_PP;
    GPIO_InitStruct.GPIO_Pin=GPIO_Pin_0;
    GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
    GPIO_Init(GPIOB, &GPIO_InitStruct);
}
​
void LED_Toggle(void)
{
    GPIOB->ODR ^= GPIO_Pin_0;
}
​
​
led.h
​
#ifndef __LED_H
#define __LED_H
​
#include "stm32f10x.h"
​
void LED_Config(void);
void LED_Toggle(void);
​
#endif
​
tim_timebase.c
​
#include "tim_timebase.h"
#include "led.h"
​
int a = 0;
​
static void NVIC_Config(void)
{
    NVIC_InitTypeDef NVIC_InitStruct;
    
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    
    NVIC_InitStruct.NVIC_IRQChannel=TIM3_IRQn;
    NVIC_InitStruct.NVIC_IRQChannelCmd=ENABLE;
    NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority=1;
    NVIC_InitStruct.NVIC_IRQChannelSubPriority=1;
    NVIC_Init(&NVIC_InitStruct);
}
​
void TIM3_Config(void)
{
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
    
    NVIC_Config();
    
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
    
    TIM_TimeBaseInitStruct.TIM_Period=71;
    TIM_TimeBaseInitStruct.TIM_Prescaler=1000;//1ms
    TIM_TimeBaseInit(TIM3, &TIM_TimeBaseInitStruct);
    
    TIM_ClearFlag(TIM3, TIM_FLAG_Update);
    
    TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);
    
    TIM_Cmd(TIM3, ENABLE);
}
​
void TIM3_IRQHandler(void)
{
    if(TIM_GetITStatus(TIM3, TIM_IT_Update)) a++;
    if(a == 1000)
    {
        LED_Toggle();
        a = 0;
    }
    TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
}

tim_timebase.h

#ifndef __TIM_TIMEBASE_H
#define __TIM_TIMEBASE_H
​
#include "stm32f10x.h"
​
void TIM3_Config(void);
​
#endif
 
main.c

#include "stm32f10x.h"
#include "led.h"
#include "tim_timebase.h"
​
int main(void)
{
    LED_Config();
    TIM3_Config();
    
    while(1)
    {
        
    }
}

三.PWM的应用

2. 接上,采用定时器PWM模式,让 LED 以呼吸灯方式渐亮渐灭,周期为1~2秒,自己调整到一个满意效果。使用Keil虚拟示波器,观察 PWM输出波形。

代码 PWM.c

#include "stm32f10x.h"                  // Device header
​
void PWM_Init(void)
{
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
//  RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
//  GPIO_PinRemapConfig(GPIO_PartialRemap1_TIM2, ENABLE);
//  GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;       //GPIO_Pin_15;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    
    TIM_InternalClockConfig(TIM2);
    
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStructure.TIM_Period = 100 - 1;     //ARR
    TIM_TimeBaseInitStructure.TIM_Prescaler = 720 - 1;      //PSC
    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM2, &TIM_TimeBaseInitStructure);
    
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCStructInit(&TIM_OCInitStructure);
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse = 0;      //CCR
    TIM_OC1Init(TIM2, &TIM_OCInitStructure);
    
    TIM_Cmd(TIM2, ENABLE);
}
​
void PWM_SetCompare1(uint16_t Compare)
{
    TIM_SetCompare1(TIM2, Compare);
}
main.c
​
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "PWM.h"
​
uint8_t i;
​
int main(void)
{
    OLED_Init();
    PWM_Init();
    
    while (1)
    {
        for (i = 0; i <= 100; i++)
        {
            PWM_SetCompare1(i);
            Delay_ms(5);
        }
        for (i = 0; i <= 100; i++)
        {
            PWM_SetCompare1(100 - i);
            Delay_ms(5);
        }
    }
}

使用Keil虚拟示波器,观察 PWM输出波形

标为bit显示,用绿色的线表示,得到的波形结果如下图所示:

四.总结

通过本次实验实现了LED以2秒的频率周期性地亮-灭,采用定时器的PWM模式,通过配置定时器的参数和相关寄存器,实现LED了以呼吸灯方式渐亮渐灭 通过Keil虚拟示波器,我们可以观察PWM输出波形,以验证PWM的正确性;在实验前做了充分的准备,仔细阅读了相关的STM32定时器和PWM模式的文档,理解其寄存器的配置方法和中断处理函数的编写并且熟悉了Keil虚拟示波器的使用方法,以便观察波形输出。 本次实验帮助我们更深入地了解STM32的定时器原理和PWM,通过实际操作和观察波形,我们会更好地掌握和理解定时器和PWM的工作原理。在进行呼吸灯效果测试时,发现当PWM周期很大时,呼吸灯效果十分不明显,可以从PWM波原理上看,修改的只是高电平的占比,那么如果周期很大,即使占空比怎么修改,人眼中还是会看出来亮灭的过程,当PWM频率提高时,人眼就无法识别出低电平的间隙,从而看出来是从亮到灭的过程。

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是使用HAL库实现LED闪烁步骤和代码示例[^1][^2]: 1. 配置GPIO引脚作为输出模式,用于连接LED。 2. 配置定时器PWM通道,用于控制LED的亮度。 3. 初始化定时器PWM通道。 4. 设置PWM周期和占空比,以控制LED的闪烁频率和亮度。 5. 启动定时器PWM通道。 下面是一个使用HAL库实现LED闪烁的示例代码: ```c #include "stm32f4xx_hal.h" TIM_HandleTypeDef htim; TIM_OC_InitTypeDef sConfig; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_TIM_Init(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_TIM_Init(); while (1) { HAL_TIM_PWM_Start(&htim, TIM_CHANNEL_1); HAL_Delay(500); HAL_TIM_PWM_Stop(&htim, TIM_CHANNEL_1); HAL_Delay(500); } } void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct; RCC_ClkInitTypeDef RCC_ClkInitStruct; __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = 16; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } 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(); } } static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct; __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitStruct.Pin = GPIO_PIN_5; 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); } static void MX_TIM_Init(void) { TIM_MasterConfigTypeDef sMasterConfig; __HAL_RCC_TIM2_CLK_ENABLE(); htim.Instance = TIM2; htim.Init.Prescaler = 83; htim.Init.CounterMode = TIM_COUNTERMODE_UP; htim.Init.Period = 999; htim.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; if (HAL_TIM_PWM_Init(&htim) != HAL_OK) { Error_Handler(); } sConfig.OCMode = TIM_OCMODE_PWM1; sConfig.Pulse = 500; sConfig.OCPolarity = TIM_OCPOLARITY_HIGH; sConfig.OCFastMode = TIM_OCFAST_DISABLE; if (HAL_TIM_PWM_ConfigChannel(&htim, &sConfig, TIM_CHANNEL_1) != HAL_OK) { Error_Handler(); } sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET; sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE; if (HAL_TIMEx_MasterConfigSynchronization(&htim, &sMasterConfig) != HAL_OK) { Error_Handler(); } } void Error_Handler(void) { while (1) { } } ``` 请注意,以上示例代码是基于STM32F4系列的HAL库,使用了定时器2和GPIOA5控制LED的闪烁。具体的配置和代码可能会因为使用的STM32系列和开发环境而有所不同,请根据实际情况进行调整。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值