STM32一定时器产生不同频率的PWM

发布者:EnigmaticCharm最新更新时间:2015-10-23 来源: eefocus关键字:STM32  定时器  不同频率  PWM 手机看文章 扫描二维码
随时随地手机看文章
平时记性实在太差,调试完的程序,过两天又忘了,往往需要一阵子才能想起来,有时以前的资料找不到了,更是恼火,不得不重复到网上搜索。刚刚调试成功了一个类型的程序,立刻记下来,呵呵,不要又忘记了。


    STM32产生PWM是非常的方便的,要需要简单的设置定时器,即刻产生!当然,简单的设置对于新手来产,也是麻烦的,主要包括:

(1)使能定时器时钟:RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);

(2)定义相应的GPIO:


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //下拉接地,检测输入的高电平
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //50M时钟速度
GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //50M时钟速度
GPIO_Init(GPIOA, &GPIO_InitStructure);

(3)如果是产生PWM(频率不变,占空比可变),记得打开PWM控制,在TIM_Configuration()中。

TIM_Cmd(TIM3,ENABLE);


TIM_CtrlPWMOutputs(TIM1,ENABLE);

利用定时器产生不同频率的PWM

  有时候,需要产生不同频率的PWM,这个时候,设置与产生相同PWM的程序,有关键的不一样。

(一) 设置的原理

    利用改变定时器输出比较通道的捕获值,当输出通道捕获值产生中断时,在中断中将捕获值改变,这时,输出的I/O会产生一个电平翻转,利用这种办法,实现不同频率的PWM输出。

(二)关键设置

在定时器设置中:TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Disable);

在中断函数中: if (TIM_GetITStatus(TIM3, TIM_IT_CC2) != RESET)
    {
    TIM_ClearITPendingBit(TIM3, TIM_IT_CC2);
  capture = TIM_GetCapture2(TIM3);
    TIM_SetCompare2(TIM3, capture + Key_Value);
}

一个定时器四个通道,分别产生不同频率(这个例子网上也有)


vu16 CCR1_Val = 32768;
vu16 CCR2_Val = 16384;
vu16 CCR3_Val = 8192;
vu16 CCR4_Val = 4096;

void TIM_Configuration(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef TIM_OCInitStructure;

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);



TIM_TimeBaseStructure.TIM_Period = 65535;       
TIM_TimeBaseStructure.TIM_Prescaler = 2;     
TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;

TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);


TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;      //PWM模式2
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //正向通道有效
TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Disable;//反向通道无效
TIM_OCInitStructure.TIM_Pulse = CCR1_Val;        //占空时间
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;    //输出极性
TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_High;    //互补端的极性
TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

TIM_OC1Init(TIM2,&TIM_OCInitStructure);        //通道1
TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Disable);

TIM_OCInitStructure.TIM_Pulse = CCR2_Val;        //占空时间
TIM_OC2Init(TIM2,&TIM_OCInitStructure);        //通道2
TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Disable);

TIM_OCInitStructure.TIM_Pulse = CCR3_Val;        //占空时间
TIM_OC3Init(TIM2,&TIM_OCInitStructure);        //通道3
TIM_OC3PreloadConfig(TIM2, TIM_OCPreload_Disable);

TIM_OCInitStructure.TIM_Pulse = CCR4_Val;        //占空时间
TIM_OC4Init(TIM2,&TIM_OCInitStructure);        //通道4
TIM_OC4PreloadConfig(TIM2, TIM_OCPreload_Disable);



TIM_Cmd(TIM2,ENABLE);


//TIM_CtrlPWMOutputs(TIM2,ENABLE);

 
TIM_ITConfig(TIM2, TIM_IT_CC1 | TIM_IT_CC2 | TIM_IT_CC3 | TIM_IT_CC4, ENABLE);

}

void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;



RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD, ENABLE);



GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; //开漏输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //50M时钟速度
GPIO_Init(GPIOA, &GPIO_InitStructure);



GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; //开漏输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //50M时钟速度
GPIO_Init(GPIOB, &GPIO_InitStructure);


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  //上拉输入
GPIO_Init(GPIOA, &GPIO_InitStructure);


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  //上拉输入
GPIO_Init(GPIOC, &GPIO_InitStructure);


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  //上拉输入
GPIO_Init(GPIOB, &GPIO_InitStructure);


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void NVIC_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure;


NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

NVIC_InitStructure.NVIC_IRQChannel=TIM2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;
NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
NVIC_Init(&NVIC_InitStructure);
}

u16 capture = 0;
extern vu16 CCR1_Val;
extern vu16 CCR2_Val;
extern vu16 CCR3_Val;
extern vu16 CCR4_Val;

void TIM2_IRQHandler(void)
{


if (TIM_GetITStatus(TIM2, TIM_IT_CC1) != RESET)
{
    TIM_ClearITPendingBit(TIM2, TIM_IT_CC1 );
capture = TIM_GetCapture1(TIM2);
TIM_SetCompare1(TIM2, capture + CCR1_Val );
}


if (TIM_GetITStatus(TIM2, TIM_IT_CC2) != RESET)
{
    TIM_ClearITPendingBit(TIM2, TIM_IT_CC2);
capture = TIM_GetCapture2(TIM2);
    TIM_SetCompare2(TIM2, capture + CCR2_Val);
}


if (TIM_GetITStatus(TIM2, TIM_IT_CC3) != RESET)
{
    TIM_ClearITPendingBit(TIM2, TIM_IT_CC3);
capture = TIM_GetCapture3(TIM2);
    TIM_SetCompare3(TIM2, capture + CCR3_Val);
}


if (TIM_GetITStatus(TIM2, TIM_IT_CC4) != RESET)
{
    TIM_ClearITPendingBit(TIM2, TIM_IT_CC4);
capture = TIM_GetCapture4(TIM2);
    TIM_SetCompare4(TIM2, capture + CCR4_Val);
}

}
一个定时器一个通道,产生不同频率

其它的设置都一样,就是在主函数中修改一个参数,然后在定时器中断中,根据这个参数,改变频率。

#include "stm32libstm32f10x.h"
#include "hal.h"
volatile u16 Key_Value=1000;  //用于保存按键相应的PWM波占空比值
int main(void)
{
ChipHalInit();
ChipOutHalInit();
while(1)

  if( (!Get_Key_Up)&(!Get_Key_Down)&(!Get_Key_Left)&(!Get_Key_Right)&(!Get_Key_Ctrl) )
  {
    Key_Value=12000;
  }
  else
  {
    if(Get_Key_Up)    //按键前进按下 ,对应1kHz
    {
    Key_Value=6000;
    }
    else if(Get_Key_Down)  //按键后退按下 ,对应2kHz
    {
      Key_Value=3000;
    }
    Delay_Ms(20);      //10ms延时

    if(Get_Key_Left)    //按键左转按下,对应3kHz
    {
    Key_Value=2000;
    }
    else if(Get_Key_Right) //按键右转按下,对应4kHz
    {
      Key_Value=1500;
   
    Delay_Ms(20);      //10ms延时

    if(Get_Key_Ctrl)    //按键控制按下,对应5kHz
    {
      Key_Value=1200;
    }
    Delay_Ms(20);      //10ms延时
  }
}
}
extern volatile u16 Key_Value;
u16 capture=0;
void TIM3_IRQHandler(void)
{

if (TIM_GetITStatus(TIM3, TIM_IT_CC2) != RESET)
{
    TIM_ClearITPendingBit(TIM3, TIM_IT_CC2);
capture = TIM_GetCapture2(TIM3);
    TIM_SetCompare2(TIM3, capture + Key_Value);
}
}
void TIM3_Configuration(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef TIM_OCInitStructure;


RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);


TIM_TimeBaseStructure.TIM_Prescaler = 5;      //预分频(时钟分频)72M/6=12M
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //向上计数
TIM_TimeBaseStructure.TIM_Period = 65535;        //装载值选择最大
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0x0;
TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);


TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;      //PWM模式2
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //正向通道有效
TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Disable;//反向通道无效
TIM_OCInitStructure.TIM_Pulse = Key_Value;        //占空时间
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;    //输出极性
TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_High;    //互补端的极性
TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

TIM_OC2Init(TIM3,&TIM_OCInitStructure);        //通道2
TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Disable);


TIM_Cmd(TIM3,ENABLE);


//TIM_CtrlPWMOutputs(TIM1,ENABLE);
TIM_ITConfig(TIM3, TIM_IT_CC2 , ENABLE);
}

注意:在计算PWM频率的时候,TIMx的时钟都是72Mhz,分频后,因为翻转两次才能形成一个PWM波,因为,PWM的频率是捕获改变频率的1/2。

关键字:STM32  定时器  不同频率  PWM 引用地址:STM32一定时器产生不同频率的PWM

上一篇:STM32的SPI接口
下一篇:STM32笔记(五)RTC的初始化

推荐阅读最新更新时间:2024-03-16 14:37

STM32 关闭JTAG 使用相应GPIO口 简单记录
STM32 的PA13-PA14-PA15-PB3-PB4-PB5主要是用来JTAG调试用的,于是在默认下是启动后为JTAG模式,但是对于不需要JTAG而需要充分利用GPIO口时,就需要将JTAG关闭,设置为GPIO模式。 GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //打开PA时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //打开PB时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFI
[单片机]
stm32 时钟设置函数分析
__IO uint32_t StartUpCounter = 0, HSEStatus = 0; /* SYSCLK, HCLK, PCLK2 and PCLK1 configuration ---------------------------*/ 这个函数的主要目的是 设置以上4个时钟的,那这四个时钟主要是用来干什么的? 这个后面再分析。 先看怎么获取72M的频率 /* Enable HSE */ 打开外部高速时钟 RCC- CR |= ((uint32_t)RCC_CR_HSEON); 等待时钟稳定 /* Wait till HSE is ready and if Time out is r
[单片机]
<font color='red'>stm32</font> 时钟设置函数分析
MSP430用按键中断控制定时器产生持续 1s 的周期信号
////////////////////////////////////////////////////////////////////////////////////////////////////// // 2012 年吉林省大学生电子设计大赛 C题 声源定位系统 ----(声源部分) // 要求: 有信号产生电路, 放大电路, 声音频率 800Hz 左右, 按一次键响声持续 1 秒. // 声源模块体积不超过 4cm*4cm*4cm. //==========================================================================================
[单片机]
STM32F3-PWM输入捕获测量频率脉宽
利用STM32的PWM输入捕获功能,可以测方波的占空比和(或)频率 使用时将相应的输入配置为对应定时器对应的复用功能,外部待测量波形从该引脚输入 再配置定时器输入捕获功能相应参数,选择主从模式,最后打开中断或者DMA读取测量数据 1. Enable TIM clock 2. Configure the TIM pins by configuring the corresponding GPIO pins 3. Fill the TIM_ICInitStruct 5. Call TIM_ICInit(TIMx, &TIM_ICInitStruct) ;. Call TIM_PWMIConfig(TIMx, &TIM_ICInit
[单片机]
stm32产生随机数
stm32使用rand()函数产生随机数,在OS编程中会设置srand()随机种子,其实随机种子就是某一时候的time()的值,因为这个值不同,所以每次的随机数都不同。 那么我们需要在stm32中模拟这样的一个随机种子 1——可以采用ADC采样值,必须是一个空脚(没接GND&VCC) 2——可以采用定时器的计数值 上传部分代码 /* 必须要每次刷新种子。每次的随机数才会不一样 srand()在#include stdlib.h rand()在#include time.h */ srand(Timer_Count_Value); //随机种子Timer_Count_Value是定时器
[单片机]
STM32的DMA串口直通
搞这个,起因呢,公司产品里ZigBee模块固件版本不对,一大批产品需要二次升级。 需要升级的ZigBee模块是直接与stm32相连的,没有引出升级口,所以得搞个串口直通程序,通过232转MCU再转ZigBee,来进行升级。 一开始直接中断互通,结果呢,通倒是可以通,一升级固件就不行了,大概是因为数据量较大吧。 所以用了DMA,之前也没用过DMA,就慢慢学慢慢模仿着改呗。 这里要涉及到一个点,DMA的外设到外设,经常写的是,DMA只有外设到内存、内存到外设、内存到内存三种模式。 实际上,DMA是可以外设直接到外设的,而且用法非常非常简单,直接设置成外设到内存或内存到外设模式,然后把内存直接改成外设的地址,就可以了。
[单片机]
<font color='red'>STM32</font>的DMA串口直通
STM32的DMA控制器
  DMA(直接存储器存取)用来提供在外设和存储器之间或者存储器与存储器之间的高速数据传输。无须CPU干预,这就可以节省CPU的资源来做其它操作。   两个DMA控制器有12个通道(DMA1有7通道,DMA2有5通道),还有一个仲裁器协调各个DMA请求的估先权。   有关DMA控制器的特性、框图、寄存器地址及含义等参考STN32资料手册,在此就不多说了,只讲述具体的设置应用。   结构体(DMA_InitTypeDef)说明:   PeripheralBaseAddr:外设地址 CPAR   MemoryBaseAddr: 存储器地址 CMAR   DIR:        传输方向(从外设读/从存储器读)   Buffe
[单片机]
三相高频PWM整流器的预测电流控制
引言       传统的相控整流器和二级管整流器存在功率因数低、电流谐波含量高、对电网污染严重等缺点。高频PWM整流器功率因数可达1,输入电流为正弦,且可向电网回馈能量,克服了传统整流器的缺点。高频PWM整流器在控制算法上一般采用电压、电流双环设计,以控制直流输出电压的稳定并使输入电流为正弦。在电流控制算法上,常常采用将模型转换到同步旋转的dq坐标系的方法,以实现d、q轴电流的解耦控制为目标,这种算法常常需要锁相环等环节实现d、q轴的定位,比较复杂。本文研究了一种预测电流控制法,能实现对电流的快速响应,且实现简单。     1 三相高频PWM整流器模型和预测电流控制的基本原理   三相电压型
[电源管理]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved