STM32实战-无源蜂鸣器

前言:

主要通过无源蜂鸣器实现功能有:

1、上电后,无源蜂鸣器发出警报声;

2、通过触摸按键1打开或关闭蜂鸣器;


目录

 1、硬件电路部分

2、技术讲解 

2.1通用定时器(TIMx) 

2.2主要特性 

 2.3框图

3.软件编程 

3.1参数配置 

3.2程序框架 

3.3蜂鸣器函数 

3.4回调函数

 



 1、硬件电路部分

  通过改变输出pwm波的频率和占空比,改变无源蜂鸣器的声音,硬件电路如下所示:

2、技术讲解 

2.1通用定时器(TIMx) 

通用定时器是一个通过可编程预分频器驱动的 16 位自动装载计数器构成。它适用于多种场合,包括测量输入信号的脉冲长度(输入采集)或者产生输出波形(输出比较和 PWM)。
使用定时器预分频器和 RCC 时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。
定时器是完全独立的,而且没有互相共享任何资源。它们可以一起同步操作。

2.2主要特性
 

● 16 位向上,向下,向上/向下自动装载计数器
● 16 位可编程预分频器,计数器时钟频率的分频系数为 1~65535 之间的任意数值
4 个独立通道:
- 输入捕获
- 输出比较
- PWM 生成(边缘或中间对齐模式)
- 单脉冲模式输出
● 使用外部信号控制定时器和定时器互连的同步电路
● 如下事件发生时产生中断/DMA:
- 更新:计数器向上溢出/向下溢出,计数器初始化(通过软件或者内部/外部触发)
- 触发事件(计数器启动,停止,初始化或者由内部/外部触发计数)
- 输入捕获
- 输出比较

 2.3框图

 

 内部时钟可以为其提供72M的时钟频率,我们可以通过预分频器PSC将频率分频成1M,通过CNT的数值,得到我们想要的频率,比较寄存器CCR可以改变占空比,图解如下:

 

3.软件编程 

3.1参数配置 

因为我原理图上的PA8只能是TIM1高级定时器,但是也可以将其当作普通低俗定时器使用。并将通道打开。 

 配置预分频参数,使能自动重装载,并将CCR设置为ARR参数值的一般使其占空比为50%,配置如下:

3.2程序框架 

整体框架不变,新增一个蜂鸣器函数

 

3.3蜂鸣器函数 

定义一个蜂鸣器开关状态的枚举常量以及,结构体封装一个蜂鸣器开关函数,代码如下:

//定义枚举类型
typedef enum
{
	ON_Status  = (uint8_t)0x01,
	OFF_Status = (uint8_t)0x02,
}Buzzer_Status_t;

//¶¨Òå½á¹¹ÌåÀàÐÍ
typedef struct
{
	uint8_t Status;       //状态

	void (*ON)(void);     //打开
	void (*OFF)(void);    //关闭
} Buzzer_t;

/* extern variables-----------------------------------------------------------*/
extern Buzzer_t  Buzzer;

定义完成后,需要对其实现,实现函数代码如下:

static void Buzzer_ON(void); 
static void Buzzer_OFF(void); 

/* Public variables-----------------------------------------------------------*/
Buzzer_t Buzzer = 
{
	OFF_Status,
	
	Buzzer_ON,
	Buzzer_OFF
};

/* Private function prototypes------------------------------------------------*/      

/*
	* @name   Buzzer_ON
	* @brief  打开蜂鸣器
	* @param  None
	* @retval None      
*/
static void Buzzer_ON(void)
{
	Buzzer.Status = ON_Status;
	HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_1);//使用了库函数
}

从上面代码种我们可以看到,打开蜂鸣器函数体内部调用了 HAL库,我们使用这个库函数,需要对它基本应用了解,传参,以及调用,它的具体实现代码如下:

HAL_StatusTypeDef HAL_TIM_PWM_Start(TIM_HandleTypeDef *htim, uint32_t Channel)
{
  uint32_t tmpsmcr;

  /* Check the parameters */
  assert_param(IS_TIM_CCX_INSTANCE(htim->Instance, Channel));

  /* Enable the Capture compare channel */
  TIM_CCxChannelCmd(htim->Instance, Channel, TIM_CCx_ENABLE);

  if (IS_TIM_BREAK_INSTANCE(htim->Instance) != RESET)
  {
    /* Enable the main output */
    __HAL_TIM_MOE_ENABLE(htim);
  }

  /* Enable the Peripheral, except in trigger mode where enable is automatically done with trigger */
  tmpsmcr = htim->Instance->SMCR & TIM_SMCR_SMS;
  if (!IS_TIM_SLAVEMODE_TRIGGER_ENABLED(tmpsmcr))
  {
    __HAL_TIM_ENABLE(htim);
  }

  /* Return function status */
  return HAL_OK;
}

3.4回调函数

我们在初始化的时候,就已经启动了中断函数,当我们响应中断之后就会进入回调函数执行相应功能,具体代码如下:

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	if(GPIO_Pin == KEY1_Pin)
	{
		LED.LED_Flip(LED2);
		
		//控制蜂鸣器开关
		if(Buzzer.Status == ON_Status)
		{
			Buzzer.OFF();
		}
		else
		{
			Buzzer.ON();
		}
	}
}

/*
	* @name   HAL_TIM_PeriodElapsedCallback
	* @brief  定时器中断回调函数
	* @param  *htim -> ´处理定时器结构体指针
	* @retval None      
*/
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
	static uint8_t Fre_Cnt = 0;
	
	if(htim->Instance == htim6.Instance)
	{
		//指示灯间隔一秒闪烁
		if(++Timer6.usMCU_Run_Timer >= TIMER0_1S)
		{
			Timer6.usMCU_Run_Timer = 0;
			
			LED.LED_Flip(LED1);
		}
		
		//控制PWM波的频率的长度与大小
		if(Fre_Cnt++ >= 2)
		{
			Fre_Cnt = 0;
			
			//¶¨Ê±Æ÷ʱÖÓ = 1MHz
			PWM = 1/((1/1000000)*ARR) = 1000000/ARR  
			//ARR = 250, PWM4KHz
			//ARR = 500, PWM2KHz
			//ARR = 1000,PWM1KHz
			//ARR = 2000,PWM0.5KHZ
			TIM1->ARR -= 10;
			if(TIM1->ARR <= 500)
				TIM1->ARR = 2000;
			
			//设置占空比为50%
			TIM1->CCR1 = TIM1->ARR / 2;
		}
	}
}
/**************************************************

 

 

  • 13
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要点亮STM32F429的无蜂鸣器,你需要使用一个GPIO引脚来控制它。首先,确保你已经正确连接了无蜂鸣器STM32F429开发板上。 在编程方面,你需要进行以下几个步骤: 1. 配置GPIO引脚:选择一个GPIO引脚,并将其配置为输出模式。你可以使用STM32的寄存器来完成这个设置。 2. 控制GPIO引脚:通过将GPIO引脚的输出状态设置为高电平或低电平来控制无蜂鸣器的开关。你可以使用寄存器来设置引脚的状态。 下面是一个示例代码,用于点亮STM32F429的无蜂鸣器: ```c #include "stm32f4xx.h" #define BEEP_PIN GPIO_Pin_0 #define BEEP_PORT GPIOA void Delay(uint32_t nCount) { for(; nCount != 0; nCount--); } int main(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = BEEP_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(BEEP_PORT, &GPIO_InitStructure); while(1) { GPIO_SetBits(BEEP_PORT, BEEP_PIN); // 设置引脚为高电平 Delay(1000000); // 延时 GPIO_ResetBits(BEEP_PORT, BEEP_PIN); // 设置引脚为低电平 Delay(1000000); // 延时 } } ``` 这段代码将无蜂鸣器连接到GPIOA的引脚0(PA0)。它通过循环不断地将引脚的输出状态切换为高电平和低电平,以产生蜂鸣声。你可以根据需要修改延时的时间来调整蜂鸣器的频率。 请注意,这只是一个示例代码,实际应用中你可能需要根据你的具体需求进行适当的修改。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值