STM32实验-高级定时器输出指定个数PWM

STM32F103ZET6中有TIM1,TIM8两个高级定时器,每一定时器都有

1、一个16位向上、向下、向上/下自动装载计数器
2、一个16位预分频器和四个独立从输入输出通道
3、每一个通道都可用于输入捕获、输出比较、PWM和单脉冲模式(除了基本定时器,高级定时器和通用定时器都能产生PWM)

当高级计时器发生溢出的时候都能使重复计数器减一,当重复计数器减到0之后,再发生一次计时器溢出,就会导致一次事件更新。

如下图所示,当RCR寄存器数据为0时候,每次计数器溢出都会导致事件更新;当RCR寄存器数据为2时候,当发生2次计数器溢出的时候,再发生一次溢出就会触发事件触发,即每3次溢出都会发生一次事件更新。如果设置了软件更新重新同步,则会导致软件更新的时候会直接触发重复计数器和基本计时器重置。

PWM的周期和基本定时器一样是Tout=(ARR+1)*(PSC+1)/Ft(Ft=72M)

实验要求是通过定时器8通道1输出指定个数PWM输出,用于控制LED1的输出。

我们已经知道LED1的引脚是PE5,查询引脚图我们会得知TIM8_CH1对应引脚为PC6。

 接下来编写我们的实验代码。

先编写头文件代码atim.h:

#ifndef __ATIM_H
#define __ATIM_H

#include "./SYSTEM/sys/sys.h"

extern TIM_HandleTypeDef g_timx_pwm_chy_handle;

void atim_timx_npwm_chy_init(uint16_t arr, uint16_t psc);
void HAL_TIM_PWM_MspInit(TIM_HandleTypeDef *htim);
void atim_timx_npwm_chy_set(uint8_t npwm);
void TIM8_UP_IRQHandler(void);
void HAL_TIM_PeriodElapseCallback(TIM_HandleTypeDef *htim);

#endif

再编写函数程序文件atim.c:

#include "./BSP/GTIM/gtim.h"
#include "./BSP/LED/led.h"

TIM_HandleTypeDef g_timx_npwm_chy_handle;

static uint8_t g_npwm_remain = 0;
 
void atim_timx_npwm_chy_init(uint16_t arr, uint16_t psc){
 
    TIM_OC_InitTypeDef timx_oc_npwm_chy = {0};
 
    g_timx_npwm_chy_handle.Instance = TIM8;
    g_timx_npwm_chy_handel.Init.Prescaler = psc;
    g_timx_npwm_chy_handle.Init.Period = arr;
    g_timx_npwm_chy_handle.Init.CounterMode = TIM_COUNTERMODE_UP;//向上计数模式
    HAL_TIM_PWM_Init(&g_timx_npwm_chy_handle);
 
    timx_oc_npwm_chy.OCMode = TIM_OCMODE_PWM1;//选择PWM1模式
    timx_oc_npwm_chy.Pulse = arr / 2;//设置占空比的比较值
    timx_oc_npwm_chy.OCPolarity = TIM_OCPOLARITY_HIGH;//输出极性:高极性
    HAL_TIM_PWM_ConfigChannel(&g_timx_npwm_chy_handle, &timx_oc_npwm_chy, TIM_CHANNEL_1);
 
    __HAL_RCC_GPIOE_CLK_ENABLE();
    gpio_init_struct.Pin = GPIO_PIN_5;
    gpio_init_struct.Mode = GPIO_MODE_INPUT;
    gpio_init_struct.Pull = GPIO_PULLUP;
    gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOE, &gpio_init_struct);

    __HAL_TIM_ENABLE_IT(&g_timx_npwm_chy_handle, TIM_IT_UPDATE);
    HAL_TIM_PWM_Start(&g_timx_npwm_chy_handle, TIM_CHANNEL_1);
}
 
void HAL_TIM_PWM_MspInit(TIM_HandleTypeDef *htim){
 
    if(htim->Instance == TIM8){
        GPIO_InitTypeDef gpio_init_struct;
        __HAL_RCC_GPIOC_CLK_ENABLE();
        __HAL_RCC_TIM8_CLK_ENABLE();
 
        
        gpio_init_struct.Pin = GPIO_PIN_6;
        gpio_init_struct.Mode = GPIO_MODE_AF_PP;
        gpio_init_struct.Pull = GPIO_PULLUP;
        gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;
        HAL_GPIO_Init(GPIOC, &gpio_init_struct);
 
        __HAL_RCC_AFIO_CLK_ENABLE();
        __HAL_AFIO_REMAP_TIM8_PARTIAL();
        HAL_NVIC_SetPriority(TIM8_UP_IRQn, 1, 3);
        HAL_NVIC_ENABLEIRQ(TIM8_UP_IRQn);
    }
}

void atim_timx_npwm_chy_set(uint8_t npwm){

    if(npwm == 0)return;

    g_npwm_remain = npwm;
    HAL_TIM_GenerateEvent(&g_timx_npwm_chy_handle, TIM_EVENTSOURCE_UPDATE);
}

void TIM8_UP_IRQHandler(void){

    HAL_TIM_IRQHandler(&g_timx_npwm_chy_handle);
}

void HAL_TIM_PeriodElapseCallback(TIM_HandleTypeDef *htim){

    if(htim->Instance == TIM8){
        if(g_npwm_remain){
            TIM8->RCR = g_npwm_remain - 1;
            HAL_TIM_GenerateEvent(&g_timx_npwm_chy_handle, TIM_EVENTSOURCE_UPDATE);
            __HAL_TIM_ENABLE(&g_timx_npwm_chy_handle);
            g_npwm_remain = 0;
        }
        else{
            TIM8->RC1 &= ~(1 << 0);
        }
    }
}

再编写主函数的代码main.c:

#include "./SYSTEM/delay/delay.h"
#include "./SYSTEM/usart/usart.h"
#include "./BSP/LED/led.h"
#include "./BSP/ATIM/atim.h"
 
int main(void){

    uint8_t key = 0;
    uint8_t t = 0;

    HAL_Init();
    sys_stm32_clock_init(RCC_PLL_MUL9);
    delay_init(72);
    led_init();
    key_init();
    usart_init(115200);
    atim_timx_pwm_chy_init(5000 - 1, 7200 - 1);

    atim_timx_npwm_chy_set(5);
 
    while(1){
        key = key_scan(0);
        if(key == KEY0_PRES){
            atim_timx_npwm_chy_set(10);
        }
    }
}

这样我们的实验代码就写完了。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
要在STM32固件库中使用高级定时器输出指定个数PWM信号,你可以按照以下步骤进行操作: 1. 首先,选择一个可用的高级定时器,例如TIM1、TIM2等,确保该定时器没有被其他模块使用。 2. 初始化定时器和GPIO引脚,以配置PWM输出。你需要设置定时器的工作模式、预分频系数、计数模式等,并将相关的GPIO引脚配置为替代功能以输出PWM信号。 3. 配置定时器的周期和占空比。通过设置定时器的自动重载寄存器(ARR)和占空比寄存器(CCR)来实现。ARR决定了PWM信号的周期,CCR则确定了占空比。 4. 启动定时器。通过设置定时器控制寄存器(CR)中的相关位来启动定时器。 以下是一个简单的示例代码,展示了如何使用高级定时器输出指定个数PWM信号: ```c #include "stm32f4xx.h" void TIM_PWM_Init(void) { // 选择高级定时器 TIM1 TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; GPIO_InitTypeDef GPIO_InitStructure; TIM_OCInitTypeDef TIM_OCInitStructure; // 使能定时器时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE); // 使能GPIO时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); // 配置GPIO引脚为替代功能 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_PinAFConfig(GPIOA, GPIO_PinSource8, GPIO_AF_TIM1); // 将引脚与定时器的通道关联起来 // 初始化定时器 TIM_TimeBaseStructure.TIM_Prescaler = 83; // 预分频系数,使定时器时钟为84MHz TIM_TimeBaseStructure.TIM_Period = 999; // PWM信号周期为1000个计数单位 TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_RepetitionCounter = 0; TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure); // 配置定时器通道1 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OCInitStructure.TIM_Pulse = 500; // 占空比为50% TIM_OC1Init(TIM1, &TIM_OCInitStructure); TIM_OC1PreloadConfig(TIM1, TIM_OCPreload_Enable); // 启动定时器 TIM_Cmd(TIM1, ENABLE); } int main(void) { TIM_PWM_Init(); while (1) { // 主循环中可以进行其他操作 } } ``` 以上代码中,TIM_PWM_Init()函数用于初始化高级定时器输出PWM信号。在主函数中,可以添加其他代码来实现你的需求。 请根据你的具体需求修改代码中的引脚、定时器等参数。希望这能对你有所帮助!如果你有任何问题,请随时提问。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值