STM32F103ZET6 用定时器级联方式输出特定数目的PWM(转载)

STM32F103ZET6里共有8个定时器,其中高级定时器有TIM1-TIM5、TIM8,共6个。

这里需要使用定时器的级联功能,ST的RM0008 REV12的P388和P399页上有说明对于特定的定时器,怎么去选择级联功能,参见表86。

我这里输出PWM的定时器是TIM2,空闲的定时器是TIM3。以TIM2为主定时器,TIM3为从定时器对TIM2的输出脉冲数进行计数。查表可知,TIM3为从定时器选择TIM2为触发源,需要配置TS=001,即选择ITR1。

实现通过定时器控制输出PWM个数的功能,可以有如下一种配置方式:

void TIM2_Master__TIM3_Slave_Configuration(u32 PulseFrequency)
{
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    TIM_OCInitTypeDef       TIM_OCInitStructure;

u16 nPDTemp ;
    /* -----------------------------------------------------------------------
    TIMx Configuration: generate 4 PWM signals with 4 different duty cycles:
    TIMxCLK = 72 MHz, Prescaler = 0x0, TIMx counter clock = 72 MHz
    TIMx ARR Register = 0 => TIMx Frequency = TIMx counter clock/(ARR + 1)
    TIMx Frequency = 72MHz.
    ----------------------------------------------------------------------- */
    TIM_Cmd(TIM2, DISABLE);
    nPDTemp = 72000000UL/PulseFrequency;

// 时基配置:配置PWM输出定时器——TIM2
    /* Time base configuration */
    TIM_TimeBaseStructure.TIM_Period          = nPDTemp-1;
    TIM_TimeBaseStructure.TIM_Prescaler       = 0;
    TIM_TimeBaseStructure.TIM_ClockDivision   = 0;
    TIM_TimeBaseStructure.TIM_CounterMode     = TIM_CounterMode_Up;
    TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);

// 输出配置:配置PWM输出定时器——TIM2
    /* PWM1 Mode configuration: Channel1 */
    TIM_OCInitStructure.TIM_OCMode            = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OCPolarity        = TIM_OCPolarity_High;    
    TIM_OCInitStructure.TIM_OutputState       = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse             = nPDTemp>>1;//50%

TIM_OC1Init(TIM2, &TIM_OCInitStructure);
    TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Enable);
    
// 时基配置:配置脉冲计数寄存器——TIM3
    TIM_TimeBaseStructure.TIM_Period        = 0xFFFF;
    TIM_TimeBaseStructure.TIM_Prescaler     = 1;
    TIM_TimeBaseStructure.TIM_ClockDivision = 0;
    TIM_TimeBaseStructure.TIM_CounterMode   = TIM_CounterMode_Up;
    TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);    
    
    /* Output Compare Active Mode configuration: Channel1 */
    TIM_OCInitStructure.TIM_OCMode                        = TIM_OCMode_Inactive;
    TIM_OCInitStructure.TIM_OCPolarity                = TIM_OCPolarity_High;
    TIM_OCInitStructure.TIM_OutputState                = TIM_OutputState_Enable;
    TIM_OCInitStructure.TIM_Pulse                        = 0xFFFF;                                        // 这里的配置值意义不大
    
    TIM_OC1Init(TIM3, &TIM_OCInitStructure);
    
// 配置TIM2为主定时器
    /* Select the Master Slave Mode */
    TIM_SelectMasterSlaveMode(TIM2, TIM_MasterSlaveMode_Enable);
    /* Master Mode selection */
    TIM_SelectOutputTrigger(TIM2, TIM_TRGOSource_Update);
    
// 配置TIM3为从定时器
    /* Slave Mode selection: TIM3 */
    TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Gated);
    TIM_SelectInputTrigger(TIM3, TIM_TS_ITR1);
    
    TIM_ITConfig(TIM3, TIM_IT_CC1, ENABLE);
   
    TIM_Cmd(TIM2, DISABLE);
    TIM_Cmd(TIM3, DISABLE);
}

中断服务程序如下:

u8  TIM2_Pulse_TIM3_Counter_OK = 0;
void TIM3_IRQHandler(void)
{
    if (TIM_GetITStatus(TIM3, TIM_IT_CC1) != RESET)
    {
        TIM_ClearITPendingBit(TIM3, TIM_IT_CC1);    // 清除中断标志位
        
        TIM_Cmd(TIM2, DISABLE); // 关闭定时器
        TIM_Cmd(TIM3, DISABLE); // 关闭定时器
        
        TIM2_Pulse_TIM3_Counter_OK = 1;
    }
}

应用程序为:

u16 pulsecnt = 10000;
void main(void)
{
    SystemSetup();        // 初始化内核和外设
    TIM2_Master__TIM3_Slave_Configuration(10000);//配置TIM2的脉冲输出为10k
    while(1)
    {
        TIM_ITConfig(TIM3, TIM_IT_CC1, DISABLE);        /* TIM enable counter */
        TIM_Cmd(TIM3, ENABLE);
        TIM3->CCR1  = pulsecnt;
        TIM3->CNT   = 0;
        TIM_ITConfig(TIM3, TIM_IT_CC1, ENABLE);
        TIM_Cmd(TIM2, ENABLE);                                                /* TIM enable counter */
        
        while(TIM2_Pulse_TIM3_Counter_OK  == 0);
    }
}

这种配置方式下,使用的是TIM3的比较中断,我还没试验过其他的方式,想来应该也是可以的,比如用定时器更新中断……

时间: 2024-12-14 17:41:16

STM32F103ZET6 用定时器级联方式输出特定数目的PWM(转载)的相关文章

STM32F103ZET6 用定时器级联方式输出特定数目的PWM

STM32F103ZET6 用定时器级联方式输出特定数目的PWM STM32F103ZET6里共有8个定时器,其中高级定时器有TIM1-TIM5.TIM8,共6个. 这里需要使用定时器的级联功能,ST的RM0008 REV12的P388和P399页上有说明对于特定的定时器,怎么去选择级联功能,参见表86. 我这里输出PWM的定时器是TIM2,空闲的定时器是TIM3.以TIM2为主定时器,TIM3为从定时器对TIM2的输出脉冲数进行计数. 查表可知,TIM3为从定时器选择TIM2为触发源,需要配置

STM32 定时器级联

根据参考手册给出的主/ 从定时器的例子 其实就是主定时器产生一个触发信号,让从定时器去接收这个触发信号,通过这个触发信号来让从定时器工作. 下面我们来看看我设置的从定时器 只需要配置 TIMx->SMCR 的TS[2:0] (Trigger selection)   选择是哪一个触发信号来触发从定时器工作 和位 SMS[2:0](Slave mode Selection)  选择外部时钟模式1-选中(TRGI)的上升沿驱动计数器 下面是STM32规定的内部触发连接的表格,如果某个产品中没有 相应

eclipse中console的输出行数控制

eclipse中console的输出行数控制 开发中,会遇到当输出大量的sql语句或者错误的时候,往往会因为console输出的限制而不能完整显示,所以我们自己就需要迫切的增加显示的行数,这样 就可以通过控制台输出的信息来查看错误的原因,从而解决问题. 第一步,当项目运行时,点击console输出框,右键选择"preferences"如下图所示 第二步,选择了之后,在弹出框中,找到"limit console output",如果是选中的话就把前面的选中取消掉. 综

【C语言】输入一个整数,输出该数二进制表示中1的个数(三种方法)

输入一个整数,输出该数二进制表示中1的个数.如输入32,输出1. 代码实现: 方法1:与运算 #define _CRT_SECURE_NO_WARNINGS 1 #include<iostream> using namespace std; int FindOneNumber(unsigned int num) {     int numberofOne = 0;     while (num)     {         num = num & (num - 1);         

10.16输入一个字符串,内有数字和非数字字符,如: a123x456 17960? 302tab5876 将其中连续的数字作为一个整数,依次存放到一数组num中。例如123放在num[0]中,456放在num[1]中……统计共有多少个整数,并输出这些数。

10.16输入一个字符串,内有数字和非数字字符,如: a123x456 17960? 302tab5876 将其中连续的数字作为一个整数,依次存放到一数组num中.例如123放在num[0]中,456放在num[1]中--统计共有多少个整数,并输出这些数. #include <stdio.h> int main(){ void search(char * parr, int * pnum); char arr[100],* parr; int num[30],* pnum; parr=arr;

MVC三层级联方式

三层级联方式,比如常见的乘法口诀表,这里使用ListBox,因为需要同时多选. [Get] [Post]后,保存当时选择的效果,同时为防止页面显示的效果和浏览器中的参数不一致,所以Post后同时回到HttpGet. 分析: 采用三个ListBox,分别为L1,L2,L3 操作1:L1选择后,需要在L2中显示对应的可选结果,并且L1可多选,在L2 中显示的结果中需要去重复. 比如: (1)L1选中6,那么在L2中需要显示6,7,8,9 (2)L1选中6,7,那么在L2中需要显示6,7,8,9和7,

输入一个浮点数,并输出该数的整数部分和小数部分

package javaapplication29; import java.util.Scanner;import java.util.StringTokenizer; /** * * @author qingzhu */public class JavaApplication29 { /** * @param args the command line arguments */ public static void main(String[] args) { String[] mess={"

编程题:统计1~20之间不能被3整除的数的个数并输出这些数

以下程序是用break和continue语句来实现的. #include<stdio.h> void main() { int n,s; for(n=1,s=0;n<=20;n=n+1) { if(n%3==0) continue; printf("%d\t",n); s=s+1; } printf("\ntotal:%d\n",s); } 算法分析与流程图: 运行结果: 编程题:统计1~20之间不能被3整除的数的个数并输出这些数,布布扣,bubu

输入一个数,输出该数从低位到高位的序列

★输入一个数,输出该数从低位到高位的序列 #include<stdio.h> int main(){     int m,n;     printf("请输入一个数:\n");     scanf("%d",&m);     printf("该数从低位到高位的输出序列为:\n");     printf("%d",m);     while(m>0)     {     n=m%10;