【STM32+HAL】PWM呼吸灯实现

一、准备工作:

有关CUBEMX的初始化配置,参见我的另一篇blog:【STM32+HAL】CUBEMX初始化配置

二、所用工具:

1、芯片: STM32F407ZGT6

2、STM32CubeMx软件

3、IDE: MDK-Keil软件

4、STM32F4xxHAL库

三、实现功能:

1、控制LED一秒闪烁一次

2、实现串口通信:电脑控制pwm频率,串口输出pwm频率

3、按键控制呼吸灯暂停与继续

4、非线性呼吸灯实现

四、HAL库配置 (初始配置完成):

1、配置引脚

根据数据手册配置相应引脚的参数,此芯片PF10为LED灯,PB8,PB9,PA0为按键

2、设置优先级

为避免在按键中断中延时函数卡死,需改变Time base优先级 

3、开启定时器1

设置定时器频率为1HZ

定时器频率=168MHZ / (PSC+1) / (ARR+1)

4、开启PWM引脚

占空比先不填,到代码中实现

5、生成代码  

至此,CUBEMX配置完成 

五、KEIL填写代码:

1、串口重定向

#include "stdio.h"
int fputc(int ch, FILE *f)
{
    HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, 0xffff);
    return ch;
}
int fgetc(FILE *f)
{
    uint8_t ch = 0;
    HAL_UART_Receive(&huart1, &ch, 1, 0xffff);
    return ch;
}

2、初始化

	HAL_TIM_Base_Start_IT(&htim1);//定时器1中断初始化
	HAL_TIM_PWM_Start (&htim14, TIM_CHANNEL_1);//Pwm初始化
	printf("Hello World\r\n");//串口输出测试
	HAL_UART_Receive_IT(&huart1, &aRxBuffer, 1);	//串口接收初始化

3、pwm函数,在while(1)中运行

void pwm(void)
{
	if(fxx==0){   //若为线性呼吸灯
		if(k&&flag){
			if(bri>=TIM14->ARR){
				dir=-1;
			 }
			  else if(bri<=0){
				dir=1;
			  }
			  bri+=dir;
		}
	}
	else if(fxx==1){   //若为非线性呼吸灯
		if(bri>=TIM14->ARR){
			dir=-1;
			i=0;
			flag_fxx=1;
		  }
		  else if(bri<=0){
			dir=1;
			flag_fxx=0;
		  }
		  i=(dir+i>=100||flag_fxx)?0:i+1;
		  bri+=dir+i;
	}
	TIM14->CCR1=bri;
	HAL_Delay(10);
}

4、接收数据函数

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    UNUSED(huart);
if(huart==&huart2){
	if(Uart1_Rx_Cnt >= 255) {  
		Uart1_Rx_Cnt = 0;
		memset(RxBuffer,0x00,sizeof(RxBuffer));
		HAL_UART_Transmit(&huart1, (uint8_t *)"数据溢出", 10,0xFFFF); 	
	}
	else{
		RxBuffer[Uart1_Rx_Cnt++] = aRxBuffer; 
		if((RxBuffer[Uart1_Rx_Cnt-1] == 0x0A)&&(RxBuffer[Uart1_Rx_Cnt-2] == 0x0D)) {
			strcpy(rx_buf,RxBuffer); 
			pl=0;
			pl=atoi(rx_buf);
			htim14.Instance->PSC = (uint16_t )((double)84000000/(double)(htim14.Instance->ARR+1)/pl)+1;

            while(HAL_UART_GetState(&huart1) == HAL_UART_STATE_BUSY_TX); 
			Uart1_Rx_Cnt = 0;
			memset(RxBuffer,0x00,sizeof(RxBuffer)); 
		}
		flag=0;
	}
	HAL_UART_Receive_IT(&huart1, (uint8_t *)&aRxBuffer, 1); 
}
}

5、定时器中断

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim){
	if ( htim -> Instance == TIM1 ){
			HAL_GPIO_TogglePin(GPIOF,GPIO_PIN_10);
			duty=(double)(htim14.Instance->CCR1)/(double)(htim14.Instance->ARR+1);
			HZ=84000000/(htim14.Instance->ARR +1)/(htim14.Instance->PSC);
			printf ("HZ==%.0fhz      DuTy==%.1f%%\r\n \r\n",HZ,duty*100);
	}
}

6、按键中断

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)  
{
	if(HAL_GPIO_ReadPin(KEY_GPIO_Port,KEY_Pin) == GPIO_PIN_RESET){
		HAL_Delay(20); //延时消抖
		if(GPIO_Pin == KEY0_Pin){
			k=1-k; 
			flag=1-flag;
			printf("NO.%d\r\n",no++);
		}
	}
	else if(HAL_GPIO_ReadPin(KEY_GPIO_Port,KEY_Pin) == GPIO_PIN_RESET){
		HAL_Delay(20);
		if(GPIO_Pin == KEY1_Pin){
			fxx=1-fxx; 
			printf("NO.%d\r\n",no++);
		}
	}
}

7、主函数

while(1){	
    if(k&&flag==0){ 
        HAL_Delay(5);
		k=1;
		flag=1;
	}
	pwm();
}

完工

六、原理讲解:

参见

STM32CubeMX学习笔记(13)——PWM输出(呼吸灯)使用

【STM32】HAL库 STM32CubeMX教程七---PWM输出(呼吸灯)

PWM原理 PWM频率与占空比详解

七、源码提供:

【免费】【STM32+HAL】PWM呼吸灯实现

  • 2
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是一个简单的示例代码,实现了按键控制 PWM 波输出: ```c #include "stm32f10x.h" #define PWM_FREQ 1000 // PWM 波频率为 1kHz #define PWM_PERIOD 3600 // PWM 周期为 3600,即 72MHz / 3600 = 20kHz void GPIO_Config(void); void TIM_Config(void); int main(void) { GPIO_Config(); TIM_Config(); while (1) { if (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == Bit_RESET) // 检测按键是否按下 { TIM_SetCompare1(TIM2, PWM_PERIOD / 2); // 设置 PWM 占空比为 50% while (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == Bit_RESET); // 等待按键释放 } else { TIM_SetCompare1(TIM2, 0); // 关闭 PWM 输出 } } } void GPIO_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; // 上拉输入 GPIO_Init(GPIOA, &GPIO_InitStructure); } void TIM_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; // 复用推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); TIM_TimeBaseStructure.TIM_Period = PWM_PERIOD - 1; // 自动重装载值 TIM_TimeBaseStructure.TIM_Prescaler = 71; // 分频器,即 72MHz / (71 + 1) = 1MHz TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse = PWM_PERIOD / 2; // 初始占空比为 50% TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC1Init(TIM2, &TIM_OCInitStructure); TIM_Cmd(TIM2, ENABLE); } ``` 该代码使用了 STM32 的 TIM2 定时器生成 PWM 波,并使用 GPIOA 的第 0 个引脚作为按键输入。在主循环中,通过检测按键状态来控制 PWM 波的输出。当按键按下时,将 PWM 波的占空比设置为 50%,即输出一个矩形波;当按键释放时,关闭 PWM 输出。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值