STM32F0xx定时器输出PWM配置

#include “stm32f0xx.h”

#define FCY_DIV 1//
#define FCY_DIV_1 2//FCY_DIV_1=FCY_DIV+1//2分频比
#define PWM_FREQ 500
#define PWM_CYCLE ((SystemCoreClock/FCY_DIV_1/ PWM_FREQ ) - 1)

#define V_BAOHU_ADC 400
u32 w_speed=0;
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef TIM_OCInitStructure;

//uint16_t TimerPeriod = 0;
uint16_t Channel1Pulse = 0, Channel2Pulse = 0, Channel3Pulse = 0, Channel4Pulse = 0;
int16_t Current_Filter_Data[4]={0,0,0,0};
int16_t Voltage_Filter_Data[4]={0,0,0,0};
u8 bao_hu_sta=0;
u8 I_ADC_IN_BUF=0;
u8 pid_chu_shi=0;

extern int16_t PID_Data;
extern u16 Current_Check_Data;
extern u16 Voltage_Check_Data;

void PWM_IO_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

/* 使能GPIO时钟 */
RCC_AHBPeriphClockCmd( RCC_AHBPeriph_GPIOA, ENABLE);
RCC_AHBPeriphClockCmd( RCC_AHBPeriph_GPIOB, ENABLE);

/* 配置GPIO管脚复用*/

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_9|GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP ;
GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource0, GPIO_AF_2);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource1, GPIO_AF_2);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_2);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_2);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_2);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_2);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP ;
GPIO_Init(GPIOB, &GPIO_InitStructure);

GPIO_PinAFConfig(GPIOB, GPIO_PinSource1, GPIO_AF_1);
}

void PWM_TIM1_Config(void)
{
/* TIM1 的配置 ---------------------------------------------------
TIM1 输入时钟(TIM1CLK) 设置为 APB2 时钟 (PCLK2)
=> TIM1CLK = PCLK2 = SystemCoreClock
TIM1CLK = SystemCoreClock, Prescaler = 0, TIM1 counter clock = SystemCoreClock
SystemCoreClock 为48 MHz

我们的目标产生 4 路PWM 信号在17.57 KHz:
- TIM1_Period = (SystemCoreClock / 17570) - 1
信道1设置的占空比为 50%
信道2设置的占空比为 37.5%
信道3设置的占空比为 25%
信道4设置的占空比为 12.5%
定时器脉冲的计算方式如下:
- ChannelxPulse = DutyCycle * (TIM1_Period - 1) / 100
*/
/计算预定表的值,也就是多少个时钟计数为一个周期/
//TimerPeriod = (SystemCoreClock / PWM_FREQ ) - 1;
//TimerPeriod = PWM_CYCLE;
//PWM_FREQ=SystemCoreClock/(PWM_CYCLE+1)

/计算CCR1 跳转值 在占空比为50%时/
Channel1Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
/计算CCR2 跳转值 在占空比为37.5%时/
Channel2Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
/计算CCR3 跳转值 在占空比为25%时/
Channel3Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
/计算CCR4跳转值 在占空比为12.5%时/
Channel4Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);

/* TIM1 时钟使能 */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1 , ENABLE);

/* Time 定时基础设置*/
TIM_TimeBaseStructure.TIM_Prescaler = FCY_DIV;//2分频
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; /* Time 定时设置为上升沿计算模式*/
TIM_TimeBaseStructure.TIM_Period = PWM_CYCLE;
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;

TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);

/* 频道1,2,3,4的PWM 模式设置 */
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Enable;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;//
TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_Low;
TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

TIM_OCInitStructure.TIM_Pulse = Channel1Pulse;//使能频道1配置
TIM_OC1Init(TIM1, &TIM_OCInitStructure);

TIM_OCInitStructure.TIM_Pulse = Channel2Pulse;//使能频道2配置
TIM_OC2Init(TIM1, &TIM_OCInitStructure);

TIM_OCInitStructure.TIM_Pulse = Channel3Pulse;//使能频道3配置
TIM_OC3Init(TIM1, &TIM_OCInitStructure);

TIM_OCInitStructure.TIM_Pulse = Channel4Pulse;//使能频道4配置
TIM_OC4Init(TIM1, &TIM_OCInitStructure);

/* TIM1 计算器使能*/
TIM_Cmd(TIM1, ENABLE);

/* TIM1 主输出使能 */
TIM_CtrlPWMOutputs(TIM1, ENABLE);
}

void PWM_TIM2_Config(void)
{

/* TIM1 的配置 ---------------------------------------------------

TIM1 输入时钟(TIM1CLK) 设置为 APB2 时钟 (PCLK2)
=> TIM1CLK = PCLK2 = SystemCoreClock
TIM1CLK = SystemCoreClock, Prescaler = 0, TIM1 counter clock = SystemCoreClock
SystemCoreClock 为48 MHz

我们的目标产生 4 路PWM 信号在17.57 KHz:
- TIM1_Period = (SystemCoreClock / 17570) - 1
信道1设置的占空比为 50%
信道2设置的占空比为 37.5%
信道3设置的占空比为 25%
信道4设置的占空比为 12.5%
定时器脉冲的计算方式如下:
- ChannelxPulse = DutyCycle * (TIM1_Period - 1) / 100
*/
/计算预定表的值,也就是多少个时钟计数为一个周期/
//TimerPeriod = (SystemCoreClock / PWM_FREQ ) - 1;
//TimerPeriod = PWM_CYCLE;

/计算CCR1 跳转值 在占空比为50%时/
Channel1Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
/计算CCR2 跳转值 在占空比为37.5%时/
Channel2Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
/计算CCR3 跳转值 在占空比为25%时/
Channel3Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
/计算CCR4跳转值 在占空比为12.5%时/
Channel4Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);

/* TIM1 时钟使能 */
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2 , ENABLE);

/* Time 定时基础设置*/
TIM_TimeBaseStructure.TIM_Prescaler = FCY_DIV;//2分频
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; /* Time 定时设置为上升沿计算模式*/
TIM_TimeBaseStructure.TIM_Period = PWM_CYCLE;
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;

TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);

/* 频道1,2,3,4的PWM 模式设置 */
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Enable;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;//TIM_OCPolarity_Low
TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_Low;
TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

TIM_OCInitStructure.TIM_Pulse = Channel1Pulse;//使能频道1配置
TIM_OC1Init(TIM2, &TIM_OCInitStructure);

TIM_OCInitStructure.TIM_Pulse = Channel2Pulse;//使能频道2配置
TIM_OC2Init(TIM2, &TIM_OCInitStructure);

TIM_OCInitStructure.TIM_Pulse = Channel3Pulse;//使能频道3配置
TIM_OC3Init(TIM2, &TIM_OCInitStructure);

TIM_OCInitStructure.TIM_Pulse = Channel4Pulse;//使能频道4配置
TIM_OC4Init(TIM2, &TIM_OCInitStructure);

/* TIM1 计算器使能*/
TIM_Cmd(TIM2, ENABLE);

/* TIM1 主输出使能 */
TIM_CtrlPWMOutputs(TIM2, ENABLE);
}

void PWM_TIM3_Config(void)
{
/* TIM1 时钟使能 */
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3 , ENABLE);
/计算CCR1 跳转值 在占空比为50%时/
Channel1Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
/计算CCR2 跳转值 在占空比为37.5%时/
Channel2Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
/计算CCR3 跳转值 在占空比为25%时/
Channel3Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
/计算CCR4跳转值 在占空比为12.5%时/
Channel4Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);

/* Time 定时基础设置*/
TIM_TimeBaseStructure.TIM_Prescaler = FCY_DIV;//2分频
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; /* Time 定时设置为上升沿计算模式*/
TIM_TimeBaseStructure.TIM_Period = PWM_CYCLE;
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;

TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);

/* 频道1,2,3,4的PWM 模式设置 */
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Enable;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;//TIM_OCPolarity_Low
TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_Low;
TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

TIM_OCInitStructure.TIM_Pulse = Channel4Pulse;//使能频道1配置
TIM_OC4Init(TIM3, &TIM_OCInitStructure);
TIM_OC3Init(TIM3, &TIM_OCInitStructure);
TIM_OC2Init(TIM3, &TIM_OCInitStructure);
TIM_OC1Init(TIM3, &TIM_OCInitStructure);

/* TIM1 计算器使能*/
TIM_Cmd(TIM3, ENABLE);

/* TIM1 主输出使能 */
TIM_CtrlPWMOutputs(TIM3, ENABLE);
}

void PWM_Init(void)
{
PWM_IO_Init();
PWM1_Config();
PWM2_Config();
PWM3_Config();
}

void PWM_TIM1_Out(void)
{
int16_t w_data_pid=0;

Channel1Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
Channel2Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
Channel3Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
Channel4Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
TIM_SetCompare1(TIM1,Channel1Pulse);
TIM_SetCompare2(TIM1,Channel2Pulse);
TIM_SetCompare3(TIM1,Channel3Pulse);
TIM_SetCompare4(TIM1,Channel4Pulse);

}

void PWM_TIM2_Out(void)
{
int16_t w_data_pid=0;

Channel1Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
Channel2Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
Channel3Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
Channel4Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
TIM_SetCompare1(TIM2,Channel1Pulse);
TIM_SetCompare2(TIM2,Channel2Pulse);
TIM_SetCompare3(TIM2,Channel3Pulse);
TIM_SetCompare4(TIM2,Channel4Pulse);

}

void PWM_TIM3_Out(void)
{
Channel1Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
Channel2Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
Channel3Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);
Channel4Pulse = (uint16_t) (((uint32_t) 1000 * (PWM_CYCLE - 1)) / 2000);//0-2000
TIM_SetCompare1(TIM3,Channel4Pulse);
TIM_SetCompare2(TIM3,Channel3Pulse);
TIM_SetCompare3(TIM3,Channel2Pulse);
TIM_SetCompare4(TIM3,Channel1Pulse);
}

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
根据提供的引用内容,可以看出PA9并没有被用作PWM输出。但是,我们可以通过修改引用中的代码来将PA9设置为PWM输出。具体步骤如下: 1. 配置PA9引脚为复用推挽输出模式,具体代码如下: ```c GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(GPIOA, &GPIO_InitStructure); ``` 2. 配置TIM1_CH2通道为PWM输出模式,具体代码如下: ```c TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse = 0; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC2Init(TIM1, &TIM_OCInitStructure); TIM_OC2PreloadConfig(TIM1, TIM_OCPreload_Enable); ``` 3. 启动TIM1定时器,具体代码如下: ```c TIM_Cmd(TIM1, ENABLE); ``` 4. 启动TIM1_CH2通道的PWM输出,具体代码如下: ```c TIM_CtrlPWMOutputs(TIM1, ENABLE); ``` 完整的代码如下: ```c #include "stm32f0xx.h" void TIM_IO_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_2); TIM_TimeBaseStructure.TIM_Period = 999; TIM_TimeBaseStructure.TIM_Prescaler = 47; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure); TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse = 0; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC2Init(TIM1, &TIM_OCInitStructure); TIM_OC2PreloadConfig(TIM1, TIM_OCPreload_Enable); TIM_Cmd(TIM1, ENABLE); TIM_CtrlPWMOutputs(TIM1, ENABLE); } int main(void) { TIM_IO_Config(); while (1) { } } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wxd82331

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值