目录
效果展示
前言
一、设计背景
1.1、知识储备
二、系统设计方案
2.1、实现功能
2.2、硬件部分
2.3、软件部分
三、软件设计
3.1、设计流程图
3.2、音乐频率的设置
3.3、编程主要思路
四、程序设计
效果展示
基于STM32用PWM控制蜂鸣器
前言
音乐播放器项目,我使用PWM输出捕获的方式控制无源蜂鸣器响,这样可以解放主函数的空间,用来控制其他外设。 (PWM的介绍在STM32学习笔记中我做了详细的介绍和应用实例)
一、设计背景
1.1、知识储备
用到了蜂鸣器、独立按键、LED、PWM输出捕获、引脚重映射、定时器中断等方面的知识。同时还要了解STM32内部时钟树,我用到的开发工具有:STM32 CubeMX和MDK kile5,并且我使用的是HAL库进行的编程。为什么不用标准库主要是因为,HAL库是ST公司一直维护和支持更新的库,它的支持性更好,标准库在2011年就已经停止了维护,但是HAL库的调用更抽象,所以我在学习时用标准库,更加形象具体,能快速的打好基础,而做项目则用的HAL库,可以用cubeMX进行图形化编程,效率更高。两种库各有利弊,不存在谁好谁坏。 而且HAL库只是支持ST公司的单片机,这变相限制了开发移植性。
二、系统设计方案
2.1、实现功能
1.开发板通电,蜂鸣器能正常播放音乐;
2.通过按键可以向前\向后切换歌曲;
3.LED指示灯随着按键的按下进行翻转指示;
2.2、硬件部分
使用的是正点原子精英开发板,芯片是STM32F103ZET6,内部时钟为72M。蜂鸣器是无源蜂鸣器,无内部振荡电路,可以改变其频率,使其播放音乐。独立按键控制音乐的播放并且可以切歌。LED做为指示灯,用来判断独立按键是否按下。
2.3、软件部分
PE5、PB5为LED引脚,PB7为PWM输出捕获引脚(TIM4、CH2),PE3、PE4为独立按键控制引脚(开启中断),开启定时器TIM2;
三、软件设计
3.1、设计流程图
3.2、音乐频率的设置
这个项目的难点就是设置音乐频率,按照我之前使用51单片机制作音乐播放器的经验,音乐频率,主要是音乐的音符,音调和节拍之间的关系。
音符决定发音单元,音调决定音量的高低,节拍决定音符发音的时间长短。
3.3、编程主要思路
因为我采用PWM和定时器控制蜂鸣器,主要关系是这样的:定时器把握音乐的节拍,PWM把握音调和音符。
还是得说一下PWM。PWM 周期由定时器分频系数和重装载值共同决定,STM32F103ZET6的主频为72MHZ,设置分频为9,所以最大是8MHZ。而我们所需要的最小频率是262HZ,所以我们只需要计算得到的重装载值不大于定时器的重装载最大值(65535)即可。
所以每次转换音符时,动态修改重装载值即可实现 PWM 输出频率的变化,进而控制无源蜂鸣器发声。定时器将进行 ms 中断,判断每拍时间是否到达,到达则播放下一拍。
四、程序设计
篇幅有限,只能展示部分代码,需要全部代码,请私信博主,谢谢。
music.c程序如下:
#include "music.h"
uint16_t C_FREQ[]={0,262,294,330,349,392,440,
494,523,587,659,698,784,880,988,
1046,1175,1318,1397,1568,1760,1976};
/*-------------------两只老虎--------------------*/
/*频率*/
uint8_t music_two_tiger_f[]={
1,2,3,1,
1,2,3,1,
3,4,5,
3,4,5,
5,6,5,6,3,1,
5,6,5,4,3,1,
1,5,1,
1,5,1,
};
/*1/4拍长*/
u16 music_two_tiger_t_echo=150;
/*长度*/
u16 music_two_tiger_len=(sizeof(music_two_tiger_f)/sizeof(uint8_t));
/*拍长*/
uint8_t music_two_tiger_t[]={
P_1,P_1,P_1,P_1,
P_1,P_1,P_1,P_1,
P_1,P_1,P_2,
P_1,P_1,P_2,
P_4_3,P_4_1,P_4_3,P_4_1,P_1,P_1,
P_4_3,P_4_1,P_4_3,P_4_1,P_1,P_1,
P_1,P_1,P_2,
P_1,P_1,P_2,
};
/*-------------------春节序曲--------------------*/
/*频率*/
uint8_t music_happy_newyear_f[]={
3,2,3,5,5,6,
7,6,7,7,
3,7,6,5,3,5,
6,5,6,5,
5,6,1,6,1,2,
1,6,5,3,
5,6,1,2,1,7,
6,5,6,5,
3,2,3,5,5,6,
1,6,3,1,
};
/*1/4拍长*/
u16 music_happy_newyear_t_echo=200;
/*长度*/
u16 music_happy_newyear_len=(sizeof(music_happy_newyear_f)/sizeof(uint8_t));
/*拍长*/
uint8_t music_happy_newyear_t[]={
P_2_1,P_1,P_2_1,P_2_1,P_1,P_2_1,
P_1,P_2_1,P_2_1,P_2,
P_2_1,P_1,P_2_1,P_2_1,P_1,P_2_1,
P_4_3,P_4_1,P_1,P_2,
P_4_3,P_4_1,P_1,P_2_1,P_1,P_2_1,
P_1,P_2_1,P_2_1,P_2,
P_4_3,P_4_1,P_1,P_2_1,P_1,P_2_1,
P_4_3,P_4_1,P_1,P_2,
P_2_1,P_1,P_2_1,P_2_1,P_1,P_2_1,
P_1,P_2_1,P_2_1,P_2,
};
/*-------------------难忘今宵--------------------*/
/*频率*/
uint8_t music_bitter_unforget_f[]={
2,3,2,1,2,3,2,5,
5,5,5,2,4,3,2,
7,1,2,3,1,7,1,2,3,5,
5,4,3,4,3,2,1,
5,6,5,4,1,3,4,6,5,5,
2,6,5,6,4,6,5,
3,4,3,2,5,3,4,3,2,1,
5,4,3,4,3,2,1,
6,5,4,1,4,6,5,5,
6,5,4,1,4,6,5,5,
3,2,1,5,1,3,2,2,
3,2,1,5,1,3,5,5,
3,2,1,5,1,3,1,1,
};
/*1/4拍长*/
u16 music_bitter_unforget_t_echo=300;
/*长度*/
u16 music_bitter_unforget_len=(sizeof(music_bitter_unforget_f)/sizeof(uint8_t));
/*拍长*/
uint8_t music_bitter_unforget_t[]={
P_2_1,P_4_1,P_4_1,P_1,P_2_1,P_4_1,P_4_1,P_1,
P_2_1,P_2_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_1,P_4_1,P_4_1,P_4_1,P_4_1,P_1,
P_2_1,P_2_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2,
P_2_1,P_4_1,P_4_1,P_2_1,P_2_1,P_4_1,P_4_1,P_4_1,P_4_1,P_1,
P_2_1,P_2_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_1,P_4_1,P_4_1,P_4_1,P_4_1,P_1,
P_2_1,P_2_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2_1,P_2,
P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_4_1,P_2_1,P_2,
};
struct MUSIC_T music_t;
u8 B0[]="stop";
u8 B1[]="two tigers";
u8 B2[]="happy newyear";
u8 B3[]="bitter_unforget";
void music_tea_seeds(u8 n){
switch(n){
case 0:{
music_t.f=NULL;
music_t.t=NULL;
music_t.len=0;
music_t.t_each=0;
music_t.name=B0;
};break;
case 1:{
music_t.f=music_two_tiger_f;
music_t.t=music_two_tiger_t;
music_t.len=music_two_tiger_len;
music_t.t_each=music_two_tiger_t_echo;
music_t.name=B1;
}; break;
case 2:{
music_t.f=music_happy_newyear_f;
music_t.t=music_happy_newyear_t;
music_t.len=music_happy_newyear_len;
music_t.t_each=music_happy_newyear_t_echo;
music_t.name=B2;
};break;
case 3:{
music_t.f=music_bitter_unforget_f;
music_t.t=music_bitter_unforget_t;
music_t.len=music_bitter_unforget_len;
music_t.t_each=music_bitter_unforget_t_echo;
music_t.name=B3;
};break;
case 4:{
}; break;
case 5:{
};break;
}
}
beep.c
#include "beep.h"
#include "music.h"
#include "tim.h"
#define TEA_VOL 99
/*
蜂鸣器根据频率发声音
note_f:频率,取值来自于数组
vol:声音,取值范围区间:[0,100]
*/
void my_passive_buzzer_set(uint16_t note_f)
{
/*计算自动重装载值,计算新的频率*/
uint16_t Autoreload=(80000.0/(float)C_FREQ[note_f])-1;
/*计算音量*/
uint16_t volume=(((float)Autoreload)/100.0)*TEA_VOL;
/*设置自动重装载值*/
__HAL_TIM_SET_AUTORELOAD(&htim4,Autoreload);
/*设置音量*/
__HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_2,volume);
/*情况计算值*/
__HAL_TIM_SET_COUNTER(&htim4,0);
}
u32 paly_delay_ms=0;
void my_buzzer_play()
{
static u8 last_flg=255;//记录上一首歌
/*歌曲切换*/
if(last_flg!=music_t.now_flg)
{
music_t.now_len=0;
music_code_get(music_t.now_flg);
last_flg=mymusic_t.now_flg;
}
if(music_t.now_flg==0)
{
passive_buzzer_set(0);
return;
}
if(paly_delay_ms==0)
{
paly_delay_ms=music_t.t_each*music_t.t[music_t.now_len];
passive_buzzer_set(music_t.f[music_t.now_len]);
if(music_t.now_len>=(music_t.len-1))
{
music_t.now_len=0;
}else
{
music_t.now_len++;
}
}else
{
paly_delay_ms--;
}
}
/*定时器1ms中断*/
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
if(htim->Instance == TIM2)
{
buzzer_play();
}
}
tim.c文章来源:https://www.toymoban.com/news/detail-779054.html
/* USER CODE BEGIN Header */
/**
******************************************************************************
* @file tim.c
* @brief This file provides code for the configuration
* of the TIM instances.
******************************************************************************
* @attention
*
* Copyright (c) 2022 STMicroelectronics.
* All rights reserved.
*
* This software is licensed under terms that can be found in the LICENSE file
* in the root directory of this software component.
* If no LICENSE file comes with this software, it is provided AS-IS.
*
******************************************************************************
*/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "tim.h"
/* USER CODE BEGIN 0 */
/* USER CODE END 0 */
TIM_HandleTypeDef htim2;
TIM_HandleTypeDef htim4;
/* TIM2 init function */
void MX_TIM2_Init(void)
{
/* USER CODE BEGIN TIM2_Init 0 */
/* USER CODE END TIM2_Init 0 */
TIM_ClockConfigTypeDef sClockSourceConfig = {0};
TIM_MasterConfigTypeDef sMasterConfig = {0};
/* USER CODE BEGIN TIM2_Init 1 */
/* USER CODE END TIM2_Init 1 */
htim2.Instance = TIM2;
htim2.Init.Prescaler = 7200-1;
htim2.Init.CounterMode = TIM_COUNTERMODE_UP;
htim2.Init.Period = 9;
htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_ENABLE;
if (HAL_TIM_Base_Init(&htim2) != HAL_OK)
{
Error_Handler();
}
sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
if (HAL_TIM_ConfigClockSource(&htim2, &sClockSourceConfig) != HAL_OK)
{
Error_Handler();
}
sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK)
{
Error_Handler();
}
/* USER CODE BEGIN TIM2_Init 2 */
/* USER CODE END TIM2_Init 2 */
}
/* TIM4 init function */
void MX_TIM4_Init(void)
{
/* USER CODE BEGIN TIM4_Init 0 */
/* USER CODE END TIM4_Init 0 */
TIM_ClockConfigTypeDef sClockSourceConfig = {0};
TIM_MasterConfigTypeDef sMasterConfig = {0};
TIM_OC_InitTypeDef sConfigOC = {0};
/* USER CODE BEGIN TIM4_Init 1 */
/* USER CODE END TIM4_Init 1 */
htim4.Instance = TIM4;
htim4.Init.Prescaler = 9-1;
htim4.Init.CounterMode = TIM_COUNTERMODE_UP;
htim4.Init.Period = 65535-1;
htim4.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
htim4.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_ENABLE;
if (HAL_TIM_Base_Init(&htim4) != HAL_OK)
{
Error_Handler();
}
sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
if (HAL_TIM_ConfigClockSource(&htim4, &sClockSourceConfig) != HAL_OK)
{
Error_Handler();
}
if (HAL_TIM_PWM_Init(&htim4) != HAL_OK)
{
Error_Handler();
}
sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
if (HAL_TIMEx_MasterConfigSynchronization(&htim4, &sMasterConfig) != HAL_OK)
{
Error_Handler();
}
sConfigOC.OCMode = TIM_OCMODE_PWM1;
sConfigOC.Pulse = 0;
sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
if (HAL_TIM_PWM_ConfigChannel(&htim4, &sConfigOC, TIM_CHANNEL_2) != HAL_OK)
{
Error_Handler();
}
/* USER CODE BEGIN TIM4_Init 2 */
/* USER CODE END TIM4_Init 2 */
HAL_TIM_MspPostInit(&htim4);
}
void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* tim_baseHandle)
{
if(tim_baseHandle->Instance==TIM2)
{
/* USER CODE BEGIN TIM2_MspInit 0 */
/* USER CODE END TIM2_MspInit 0 */
/* TIM2 clock enable */
__HAL_RCC_TIM2_CLK_ENABLE();
/* TIM2 interrupt Init */
HAL_NVIC_SetPriority(TIM2_IRQn, 0, 0);
HAL_NVIC_EnableIRQ(TIM2_IRQn);
/* USER CODE BEGIN TIM2_MspInit 1 */
/* USER CODE END TIM2_MspInit 1 */
}
else if(tim_baseHandle->Instance==TIM4)
{
/* USER CODE BEGIN TIM4_MspInit 0 */
/* USER CODE END TIM4_MspInit 0 */
/* TIM4 clock enable */
__HAL_RCC_TIM4_CLK_ENABLE();
/* TIM4 interrupt Init */
HAL_NVIC_SetPriority(TIM4_IRQn, 0, 0);
HAL_NVIC_EnableIRQ(TIM4_IRQn);
/* USER CODE BEGIN TIM4_MspInit 1 */
/* USER CODE END TIM4_MspInit 1 */
}
}
void HAL_TIM_MspPostInit(TIM_HandleTypeDef* timHandle)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
if(timHandle->Instance==TIM4)
{
/* USER CODE BEGIN TIM4_MspPostInit 0 */
/* USER CODE END TIM4_MspPostInit 0 */
__HAL_RCC_GPIOB_CLK_ENABLE();
/**TIM4 GPIO Configuration
PB7 ------> TIM4_CH2
*/
GPIO_InitStruct.Pin = GPIO_PIN_7;
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
/* USER CODE BEGIN TIM4_MspPostInit 1 */
/* USER CODE END TIM4_MspPostInit 1 */
}
}
void HAL_TIM_Base_MspDeInit(TIM_HandleTypeDef* tim_baseHandle)
{
if(tim_baseHandle->Instance==TIM2)
{
/* USER CODE BEGIN TIM2_MspDeInit 0 */
/* USER CODE END TIM2_MspDeInit 0 */
/* Peripheral clock disable */
__HAL_RCC_TIM2_CLK_DISABLE();
/* TIM2 interrupt Deinit */
HAL_NVIC_DisableIRQ(TIM2_IRQn);
/* USER CODE BEGIN TIM2_MspDeInit 1 */
/* USER CODE END TIM2_MspDeInit 1 */
}
else if(tim_baseHandle->Instance==TIM4)
{
/* USER CODE BEGIN TIM4_MspDeInit 0 */
/* USER CODE END TIM4_MspDeInit 0 */
/* Peripheral clock disable */
__HAL_RCC_TIM4_CLK_DISABLE();
/* TIM4 interrupt Deinit */
HAL_NVIC_DisableIRQ(TIM4_IRQn);
/* USER CODE BEGIN TIM4_MspDeInit 1 */
/* USER CODE END TIM4_MspDeInit 1 */
}
}
/* USER CODE BEGIN 1 */
/* USER CODE END 1 */
谢谢支持~文章来源地址https://www.toymoban.com/news/detail-779054.html
到了这里,关于基于STM32制作的音乐播放器,用PWM控制蜂鸣器的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!