赞
踩
原理:当捕获/比较寄存器与计数器的值相等时发生翻转(高电平变低电平,低电平变高电平)
看CubeMX的配置:
代码里捕获/比较寄存器的值设置为100,当计数值计数到100就会翻转。开启更新中断然后重新去设置捕获比较寄存器的值。比如第一次溢出设置600。那么下次CNT到600又会翻转。持续不断每次进中断比上一次多加500。就会产生连续的方波。但这样CNT的肯定会溢出(16位:65535 )。捕获比较寄存器的值是16位的设置的值不能超过65535 如果超过那么会自动减65535(溢出)。又重复从0~65535。所以一直有方波产生。
main.c
- int main(void)
- {
- /* 复位所有外设,初始化Flash接口和系统滴答定时器 */
- HAL_Init();
- /* 配置系统时钟 */
- SystemClock_Config();
-
- /* 高级定时器初始化*/
- ADVANCED_TIMx_Init();
-
- /* 启动CH1比较输出并开启中断 */
- HAL_TIM_OC_Start_IT(&htimx,TIM_CHANNEL_1);
-
- /* 无限循环 */
- while (1){}
- }
- /* 定时器的更新中断 */
- void HAL_TIM_OC_DelayElapsedCallback(TIM_HandleTypeDef *htim) // CNT 500+500=1000 f = 2Mh(分频后的频率)/1000 = 2khz 可以用示波器去捕获
- {
- __IO uint16_t count = 0; // 因为捕获比较寄存器的值是16位的设置的值不能超过65535 如果超过那么会自动减65535(溢出)
- count = __HAL_TIM_GetCounter(htim); // 获取捕获比较寄存器的值
- __HAL_TIM_SET_COMPARE(htim,TIM_CHANNEL_1,count+499); // 重新设置捕获比较寄存器的值
- }
定时器的.h 文件 这里方便移植写成了板级支持包
- #define ADVANCED_TIMx TIM8
- #define ADVANCED_TIMx_GPIO_AF GPIO_AF3_TIM8
- #define ADVANCED_TIM_RCC_CLK_ENABLE() __HAL_RCC_TIM8_CLK_ENABLE()
- #define ADVANCED_TIM_RCC_CLK_DISABLE() __HAL_RCC_TIM8_CLK_DISABLE()
- #define ADVANCED_TIM_GPIO_RCC_CLK_ENABLE() __HAL_RCC_GPIOI_CLK_ENABLE()
-
- #define ADVANCED_TIM_OC_IRQn TIM8_CC_IRQn
- #define ADVANCED_TIM_OC_IRQHANDLER TIM8_CC_IRQHandler
-
- #define ADVANCED_TIM_CH1_PORT GPIOI
- #define ADVANCED_TIM_CH1_PIN GPIO_PIN_5
- #define ADVANCED_TIM_CH2_PORT GPIOI
- #define ADVANCED_TIM_CH2_PIN GPIO_PIN_6
- #define ADVANCED_TIM_CH3_PORT GPIOI
- #define ADVANCED_TIM_CH3_PIN GPIO_PIN_7
- #define ADVANCED_TIM_CH4_PORT GPIOI
- #define ADVANCED_TIM_CH4_PIN GPIO_PIN_2
-
- #endif
-
- // 定义定时器预分频,定时器实际时钟频率为:168MHz/(ADVANCED_TIMx_PRESCALER+1)
- #define ADVANCED_TIM_PRESCALER 83 // 实际时钟频率为:2MHz
-
- // 定义定时器周期,当定时器开始计数到ADVANCED_TIMx_PERIOD值并且重复计数寄存器为0时更新定时器并生成对应事件和中断
- #define ADVANCED_TIM_PERIOD 0xFFFF // 定时器产生中断频率为:1MHz/(35536)=1KHz,即1ms定时周期
- // 定义高级定时器重复计数寄存器值,
- #define ADVANCED_TIM_REPETITIONCOUNTER 0
-
- // 最终定时器频率计算为: 168MHz/(ADVANCED_TIMx_PRESCALER+1)/(ADVANCED_TIM_REPETITIONCOUNTER+1)/(ADVANCED_TIMx_PERIOD+1)
- // 比如需要产生1ms周期定时,可以设置为: 168MHz/(167+1)/(0+1)/(999+1)=1KHz,即1ms周期
- // 这里设置 ADVANCED_TIMx_PRESCALER=167;ADVANCED_TIM_REPETITIONCOUNTER=0;ADVANCED_TIMx_PERIOD=999;
-
- /* 扩展变量 ------------------------------------------------------------------*/
- extern TIM_HandleTypeDef htimx;
- /* 函数声明 ------------------------------------------------------------------*/
- void ADVANCED_TIMx_Init(void);
要开启定时器的四个通道,并且用的高级定时器。
定时器的.c文件
- #include "AdvancedTIM/bsp_AdvancedTIM.h"
-
- /* 私有类型定义 --------------------------------------------------------------*/
- /* 私有宏定义 ----------------------------------------------------------------*/
- /* 私有变量 ------------------------------------------------------------------*/
- TIM_HandleTypeDef htimx; /* 句柄结构体 */
-
- void HAL_TIM_OC_MspInit(TIM_HandleTypeDef* htim)
- {
- GPIO_InitTypeDef GPIO_InitStruct;
- if(htim->Instance==ADVANCED_TIMx)
- {
- /* 定时器通道功能引脚端口时钟使能 */
- ADVANCED_TIM_GPIO_RCC_CLK_ENABLE();
-
- /* 定时器通道1功能引脚IO初始化 */
- GPIO_InitStruct.Pin = ADVANCED_TIM_CH1_PIN;
- GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
- GPIO_InitStruct.Pull = GPIO_NOPULL;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
- GPIO_InitStruct.Alternate = ADVANCED_TIMx_GPIO_AF;
- HAL_GPIO_Init(ADVANCED_TIM_CH1_PORT, &GPIO_InitStruct);
-
- /* 定时器通道2功能引脚IO初始化 */
- GPIO_InitStruct.Pin = ADVANCED_TIM_CH2_PIN;
- HAL_GPIO_Init(ADVANCED_TIM_CH2_PORT, &GPIO_InitStruct);
-
- /* 定时器通道3功能引脚IO初始化 */
- GPIO_InitStruct.Pin = ADVANCED_TIM_CH3_PIN;
- HAL_GPIO_Init(ADVANCED_TIM_CH3_PORT, &GPIO_InitStruct);
-
- /* 定时器通道4功能引脚IO初始化 */
- GPIO_InitStruct.Pin = ADVANCED_TIM_CH4_PIN;
- HAL_GPIO_Init(ADVANCED_TIM_CH4_PORT, &GPIO_InitStruct);
-
- HAL_NVIC_SetPriority(ADVANCED_TIM_OC_IRQn,1,0); /* 输出比较中断 */
- HAL_NVIC_EnableIRQ(ADVANCED_TIM_OC_IRQn); /* 输出比较中断使能 */
- }
- }
-
-
- void ADVANCED_TIMx_Init(void)
- {
- TIM_ClockConfigTypeDef sClockSourceConfig; /* 定时器时钟结构体 */
- TIM_MasterConfigTypeDef sMasterConfig; /* 主模式配置结构体 */
- TIM_OC_InitTypeDef sConfigOC; /* 输出比较结构体 */
-
- /* 定时器外设时钟使能 */
- ADVANCED_TIM_RCC_CLK_ENABLE();
-
- htimx.Instance = ADVANCED_TIMx;
- htimx.Init.Prescaler = ADVANCED_TIM_PRESCALER;
- htimx.Init.CounterMode = TIM_COUNTERMODE_UP;
- htimx.Init.Period = ADVANCED_TIM_PERIOD;
- htimx.Init.ClockDivision=TIM_CLOCKDIVISION_DIV1;
- htimx.Init.RepetitionCounter = ADVANCED_TIM_REPETITIONCOUNTER;//重复计数为0+1
- HAL_TIM_OC_Init(&htimx);
-
- sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;//选择内部时钟
- HAL_TIM_ConfigClockSource(&htimx, &sClockSourceConfig);
-
- sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
- sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
- HAL_TIMEx_MasterConfigSynchronization(&htimx, &sMasterConfig);
-
-
- sConfigOC.OCMode = TIM_OCMODE_TOGGLE; // 比较输出翻转模式
- sConfigOC.Pulse = 100; // CH1当CNT计数到100时 电平翻转
- sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; /* 输出极性的选择 */
- sConfigOC.OCNPolarity = TIM_OCNPOLARITY_HIGH; /* 互补输出极性的选择 */
- sConfigOC.OCFastMode = TIM_OCFAST_DISABLE; // 快速模式
- sConfigOC.OCIdleState = TIM_OCIDLESTATE_RESET; // 空闲状态
- sConfigOC.OCNIdleState = TIM_OCNIDLESTATE_RESET; // 互补的空闲状态
-
- /* 配置CH1为比较输出模式 */
- HAL_TIM_OC_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_1);
- /* 配置CH2为比较输出模式 */
- sConfigOC.Pulse = 200; // CH2当CNT计数到200时 电平翻转
- HAL_TIM_OC_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_2);
- /* 配置CH3为比较输出模式 */
- sConfigOC.Pulse = 300; // CH2当CNT计数到300时 电平翻转
- HAL_TIM_OC_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_3);
- /* 配置CH4为比较输出模式 */
- sConfigOC.Pulse = 400; // CH2当CNT计数到400时 电平翻转
- HAL_TIM_OC_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_4 );
- }
在这里最核心的就是定时器输出比较结构体的配置:
- sConfigOC.OCMode = TIM_OCMODE_TOGGLE; // 比较输出翻转模式
- sConfigOC.Pulse = 100; // CH1当CNT计数到100时 电平翻转
- sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; /* 输出极性的选择 */
- sConfigOC.OCNPolarity = TIM_OCNPOLARITY_HIGH; /* 互补输出极性的选择 */
- sConfigOC.OCFastMode = TIM_OCFAST_DISABLE; // 快速模式
- sConfigOC.OCIdleState = TIM_OCIDLESTATE_RESET; // 空闲状态
- sConfigOC.OCNIdleState = TIM_OCNIDLESTATE_RESET; // 互补的空闲状态
代码中定时器的CH1为100,CH2为200,CH3为300,CH4为400.当用示波器去捕捉。
PWM就是用数字信号达到模拟信号的效果。要理解占空比的概念:在一个周期内高电平的持续时间。
下面是CubeMX的配置:
在这里配置PWM1的模式(与PWM2是相反的会一个可以了)
TM32 定时器的 PWM 存在两种模式,即 PWM1 模式与 PWM2 模式,两种模式相似却又恰恰相反。
PWM1 模式:向上计数,当 TIMx_CNT < TIMx_CCRn 时,定时器 TIMx 的通道 n 为有效电平,否则为无效电平;向下计数,当 TIMx_CNT > TIMx_CCRn 时,定时器 TIMx 的通道 n 为无效电平,否则为有效电平。
PWM2 模式:向上计数,当 TIMx_CNT < TIMx_CCRn 时,定时器 TIMx 的通道 n 为无效电平,否则为有效电平;向下计数,当 TIMx_CNT > TIMx_CCRn 时,定时器 TIMx 的通道 n 为有效电平,否则为无效电平。
其实在定时器的输出比较就是CNT不停跟捕获/比较寄存器的值做比较。
main.c
- int main(void)
- {
- /* 复位所有外设,初始化Flash接口和系统滴答定时器 */
- HAL_Init();
- /* 配置系统时钟 */
- SystemClock_Config();
- /* 初始化串口并配置串口中断优先级 */
- MX_DEBUG_USART_Init();
- /* 通用定时器初始化并配置PWM输出功能 */
- GENERAL_TIMx_Init();
-
- /* 启动通道PWM输出并且使能定时器 */
- HAL_TIM_PWM_Start(&htimx,TIM_CHANNEL_1); // 20%的高电平 80%的低电平
- HAL_TIM_PWM_Start(&htimx,TIM_CHANNEL_2); // 40%的高电平 60%的低电平
- HAL_TIM_PWM_Start(&htimx,TIM_CHANNEL_3); // 60%的高电平 40%的低电平
- HAL_TIM_PWM_Start(&htimx,TIM_CHANNEL_4); // 80%的高电平 20%的低电平
- /* 无限循环 */
- while (1)
- {
- }
- }
定时器的.h
- #define GENERAL_TIMx TIM2
- #define GENERAL_TIM_GPIO_AF GPIO_AF1_TIM2
- #define GENERAL_TIM_RCC_CLK_ENABLE() __HAL_RCC_TIM2_CLK_ENABLE()
- #define GENERAL_TIM_RCC_CLK_DISABLE() __HAL_RCC_TIM2_CLK_DISABLE()
- #define GENERAL_TIM_GPIO_RCC_CLK_ENABLE() {__HAL_RCC_GPIOA_CLK_ENABLE();__HAL_RCC_GPIOB_CLK_ENABLE();}
- #define GENERAL_TIM_CH1_PORT GPIOA
- #define GENERAL_TIM_CH1_PIN GPIO_PIN_15
- #define GENERAL_TIM_CH2_PORT GPIOB
- #define GENERAL_TIM_CH2_PIN GPIO_PIN_3
- #define GENERAL_TIM_CH3_PORT GPIOB
- #define GENERAL_TIM_CH3_PIN GPIO_PIN_10
- #define GENERAL_TIM_CH4_PORT GPIOB
- #define GENERAL_TIM_CH4_PIN GPIO_PIN_11
-
- // 定义定时器预分频,定时器实际时钟频率为:84MHz/(GENERAL_TIMx_PRESCALER+1)
- #define GENERAL_TIM_PRESCALER 41 // 实际时钟频率为:1MHz
-
- // 定义定时器周期,当定时器开始计数到GENERAL_TIMx_PERIOD值是更新定时器并生成对应事件和中断
- #define GENERAL_TIM_PERIOD 999 // 定时器产生中断频率为:1MHz/(999+1)=1KHz,即1ms定时周期(改变捕获比较寄存器的值周期不会改变只会改变占空比)
-
- #define GENERAL_TIM_CH1_PULSE 200 // 定时器通道1占空比为:GENERAL_TIM_CH1_PULSE/GENERAL_TIM_PERIOD*100%=200/1000*100%=20%
- #define GENERAL_TIM_CH2_PULSE 400 // 定时器通道2占空比为:GENERAL_TIM_CH2_PULSE/GENERAL_TIM_PERIOD*100%=400/1000*100%=40%
- #define GENERAL_TIM_CH3_PULSE 600 // 定时器通道3占空比为:GENERAL_TIM_CH3_PULSE/GENERAL_TIM_PERIOD*100%=600/1000*100%=60%
- #define GENERAL_TIM_CH4_PULSE 800 // 定时器通道4占空比为:GENERAL_TIM_CH4_PULSE/GENERAL_TIM_PERIOD*100%=800/1000*100%=80%
-
-
- /* 扩展变量 ------------------------------------------------------------------*/
- extern TIM_HandleTypeDef htimx;
-
- /* 函数声明 ------------------------------------------------------------------*/
- void GENERAL_TIMx_Init(void);
定时器的.c
- TIM_HandleTypeDef htimx;
-
- /* 扩展变量 ------------------------------------------------------------------*/
- /* 私有函数原形 --------------------------------------------------------------*/
- /* 函数体 --------------------------------------------------------------------*/
- /**
- * 函数功能: 定时器硬件初始化配置
- * 输入参数: htim:定时器句柄类型指针
- * 返 回 值: 无
- * 说 明: 该函数被GENERAL_TIMx_Init函数调用
- */
- void HAL_GeneralTIM_MspPostInit(void)
- {
- GPIO_InitTypeDef GPIO_InitStruct;
-
- /* 基本定时器外设时钟使能 */
- GENERAL_TIM_RCC_CLK_ENABLE();
- /* 定时器通道功能引脚端口时钟使能 */
- GENERAL_TIM_GPIO_RCC_CLK_ENABLE();
-
- /* 定时器通道1功能引脚IO初始化 */
- GPIO_InitStruct.Pin = GENERAL_TIM_CH1_PIN;
- GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
- GPIO_InitStruct.Alternate = GENERAL_TIM_GPIO_AF;
- HAL_GPIO_Init(GENERAL_TIM_CH1_PORT, &GPIO_InitStruct);
-
- /* 定时器通道2功能引脚IO初始化 */
- GPIO_InitStruct.Pin = GENERAL_TIM_CH2_PIN;
- GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
- GPIO_InitStruct.Alternate = GENERAL_TIM_GPIO_AF;
- HAL_GPIO_Init(GENERAL_TIM_CH2_PORT, &GPIO_InitStruct);
-
- /* 定时器通道3功能引脚IO初始化 */
- GPIO_InitStruct.Pin = GENERAL_TIM_CH3_PIN;
- GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
- GPIO_InitStruct.Alternate = GENERAL_TIM_GPIO_AF;
- HAL_GPIO_Init(GENERAL_TIM_CH3_PORT, &GPIO_InitStruct);
-
- /* 定时器通道4功能引脚IO初始化 */
- GPIO_InitStruct.Pin = GENERAL_TIM_CH4_PIN;
- GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
- GPIO_InitStruct.Alternate = GENERAL_TIM_GPIO_AF;
- HAL_GPIO_Init(GENERAL_TIM_CH4_PORT, &GPIO_InitStruct);
- }
-
- /**
- * 函数功能: 通用定时器初始化并配置通道PWM输出
- * 输入参数: 无
- * 返 回 值: 无
- * 说 明: 无
- */
- void GENERAL_TIMx_Init(void)
- {
- TIM_ClockConfigTypeDef sClockSourceConfig;
- TIM_MasterConfigTypeDef sMasterConfig;
- TIM_OC_InitTypeDef sConfigOC;
-
- HAL_GeneralTIM_MspPostInit();
-
- htimx.Instance = GENERAL_TIMx;
- htimx.Init.Prescaler = GENERAL_TIM_PRESCALER;
- htimx.Init.CounterMode = TIM_COUNTERMODE_UP;
- htimx.Init.Period = GENERAL_TIM_PERIOD;
- htimx.Init.ClockDivision=TIM_CLOCKDIVISION_DIV1;
- HAL_TIM_Base_Init(&htimx);
-
- sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;//选择内部时钟
- HAL_TIM_ConfigClockSource(&htimx, &sClockSourceConfig);
-
- sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
- sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
- HAL_TIMEx_MasterConfigSynchronization(&htimx, &sMasterConfig);
-
- sConfigOC.OCMode = TIM_OCMODE_PWM1;//选择PWM模式1
- sConfigOC.Pulse = GENERAL_TIM_CH1_PULSE;
- sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; // 有效电平为高电平
- sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
- HAL_TIM_PWM_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_1);
-
- sConfigOC.Pulse = GENERAL_TIM_CH2_PULSE;
- HAL_TIM_PWM_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_2);
-
- sConfigOC.Pulse = GENERAL_TIM_CH3_PULSE;
- HAL_TIM_PWM_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_3);
-
- sConfigOC.Pulse = GENERAL_TIM_CH4_PULSE;
- HAL_TIM_PWM_ConfigChannel(&htimx, &sConfigOC, TIM_CHANNEL_4);
-
- }
四个通道的占空比不是一样的,当周期是一样(没有改变ARR值所以不会改变周期)
如果要做呼吸灯那么可以通过不停改占空比比如:ARR = 1000, 可以在0~1000内设置一组指数曲线上升的值与指数曲线下降的值。(设置不同的捕获比较寄存器的值)在比较低内的时间内不停改变循环捕获比较寄存器的值就可以了。重点还是:定时器的输出比较就是CNT不停跟捕获/比较寄存器的值做比较。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。