你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

TIM输出比较的三种模式

[复制链接]
Savy1314 发布时间:2017-7-13 17:16
TIM输出比较的三种模式

此项功能是用来控制一个输出波形,或者指示一段给定的的时间已经到时。

当计数器与捕获/比较寄存器的内容相同时,输出比较功能做如下操作:

● 将输出比较模式(TIMx_CCMRx寄存器中的OCxM位)和输出极性(TIMx_CCER寄存器中的

CCxP位)定义的值输出到对应的引脚上。在比较匹配时,输出引脚可以保持它的电平

(OCxM=000)、被设置成有效电平(OCxM=001)、被设置成无效电平(OCxM=010)或进行翻

转(OCxM=011)。

● 设置中断状态寄存器中的标志位(TIMx_SR寄存器中的CCxIF位)。

● 若设置了相应的中断屏蔽(TIMx_DIER寄存器中的CCxIE位),则产生一个中断。

● 若设置了相应的使能位(TIMx_DIER寄存器中的CCxDE位,TIMx_CR2寄存器中的CCDS位

选择DMA请求功能),则产生一个DMA请求。

TIMx_CCMRx中的OCxPE位选择TIMx_CCRx寄存器是否需要使用预装载寄存器。

●设置中断状态寄存器中的标志位(TIMx_SR寄存器中的CCxIF位)。
●若设置了相应的中断屏蔽(TIMx_DIER寄存器中的CCXIE位),则产生一个中断。
●若设置了相应的使能位(TIMx_DIER寄存器中的CCxDE位,TIMx_CR2寄存器中的CCDS位选择DMA请求功能),则产生一个DMA请求。
TIMx_CCMRx中的OCxPE位选择TIMx_CCRx寄存器是否需要使用预装载寄存器。
2.jpg

在输出比较模式下,更新事件UEV对OCxREF和OCx输出没有影响。
同步的精度可以达到计数器的一个计数周期。输出比较模式(在单脉冲模式下)也能用来输出一个单脉冲。
输出比较模式的配置步骤:

1.选择计数器时钟(内部,外部,预分频器)
2.将相应的数据写入TIMx_ARR和TIMx_CCRx寄存器中
3.如果要产生一个中断请求和/或一个DMA请求,设置CCxIE位和/或CCxDE位。
4.选择输出模式,例如:必须设置OCxM=’011’、OCxPE=’0’、CCxP=’0’和CCxE=’1’,当计数器CNT与CCRx匹配时翻转OCx的输出管脚,CCRx预装载未用,开启OCx输出且高电平有效。
5.设置TIMx_CR1寄存器的CEN位启动计数器
TIMx_CCRx寄存器能够在任何时候通过软件进行更新以控制输出波形,条件是未使用预装载寄存器(OCxPE=’0’,否则TIMx_CCRx影子寄存器只能在发生下一次更新事件时被更新)

程序如下:
  1. <font size="2"> TIM_TimeBaseStructure.TIM_Period = 65535;         
  2. TIM_TimeBaseStructure.TIM_Prescaler = 0;      
  3. TIM_TimeBaseStructure.TIM_ClockDivision = 0;   
  4. TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  5. TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  6. TIM_PrescalerConfig(TIM2, 35999, TIM_PSCReloadMode_Immediate);
  7. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Timing;//这个地方就是改比较模式的</font>
复制代码

但是由于比较模式无论选哪个对于产生中断的作用是一样的,所以选TIMING都可以   
  1. TIM_OCInitStructure.TIM_Channel = TIM_Channel_1;         
  2.   TIM_OCInitStructure.TIM_Pulse = CCR1_Val;
  3.   TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
  4.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  5.   TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Disable);//
复制代码



TIMx_CCRx寄存器能够在任何时候通过软件进行更新以控制输出波形,条件是未使用预装载寄存器(OCxPE=’0’,否则TIMx_CCRx影子寄存器只能在发生下一次更新事件时被更新)。这里设置为Disable

就是为了后面在中断服务子程序可以修改TIMx_CCR实时起作用~
  1. TIM_OCInitStructure.TIM_Channel = TIM_Channel_2;         
  2.   TIM_OCInitStructure.TIM_Pulse = CCR2_Val;
  3.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  4.   TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Disable);
  5.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_3;         
  6.   TIM_OCInitStructure.TIM_Pulse = CCR3_Val;
  7.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  8.   TIM_OC3PreloadConfig(TIM2, TIM_OCPreload_Disable);
  9.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_4;         
  10.   TIM_OCInitStructure.TIM_Pulse = CCR4_Val;
  11.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  12.   TIM_OC4PreloadConfig(TIM2, TIM_OCPreload_Disable);
  13.   TIM_ARRPreloadConfig(TIM2, ENABLE);//TIM_OCPreload_Enable
  14.   TIM_ITConfig(TIM2, TIM_IT_CC1 | TIM_IT_CC2 | TIM_IT_CC3 | TIM_IT_CC4, ENABLE);
  15.     // STM3210B-LK1,  set PC.04 - PC.07
  16.   GPIO_SetBits(GPIOC, GPIO_Pin_4 |GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7);
  17.   TIM_Cmd(TIM2, ENABLE);

  18.   while (1)
  19.   {
  20.   }
  21. }
复制代码


中断服务子程序:
  1. <font size="2">void TIM2_IRQHandler(void)
  2.   {  u16 capture;
  3.      u16 CCR1_Val = 1000;
  4.      u16 CCR2_Val = 500;
  5.      u16 CCR3_Val = 250;
  6.      u16 CCR4_Val = 125;

  7.   if (TIM_GetITStatus(TIM2, TIM_IT_CC1) != RESET)
  8.   {
  9.    
  10.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC1);
  11.    
  12.     capture = TIM_GetCapture1(TIM2);
  13.     TIM_SetCompare1(TIM2, capture + CCR1_Val);

  14.    ////设置TIMx捕获比较1寄存器值然后动态修改其CCR的值 使整个程序一直进行下去
  15.    
  16.     // PC.04
  17.     GPIO_WriteBit(GPIOC, GPIO_Pin_4, (BitAction)(1 - GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_4)));

  18.   }
  19.   else if (TIM_GetITStatus(TIM2, TIM_IT_CC2) != RESET)
  20.   {
  21.    
  22.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC2);
  23.     capture = TIM_GetCapture2(TIM2);
  24.     TIM_SetCompare2(TIM2, capture + CCR2_Val);

  25.     // PC.05
  26.     GPIO_WriteBit(GPIOC, GPIO_Pin_5, (BitAction)(1 - GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_5)));
  27.    
  28.   }
  29.   else if (TIM_GetITStatus(TIM2, TIM_IT_CC3) != RESET)
  30.   {
  31.    
  32.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC3);
  33.     capture = TIM_GetCapture3(TIM2);
  34.     TIM_SetCompare3(TIM2, capture + CCR3_Val);
  35.   
  36.     // PC.06
  37.     //GPIO_ResetBits(GPIOC, GPIO_Pin_6);
  38.      GPIO_WriteBit(GPIOC, GPIO_Pin_6, (BitAction)(1 - GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_6)));
  39.   }
  40.   else
  41.   {
  42.    
  43.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC4);
  44.     capture = TIM_GetCapture4(TIM2);
  45.     TIM_SetCompare4(TIM2, capture + CCR4_Val);
  46.      
  47.     // PC.07
  48.    // GPIO_ResetBits(GPIOC, GPIO_Pin_7);
  49.      GPIO_WriteBit(GPIOC, GPIO_Pin_7, (BitAction)(1 - GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_7)));

  50.   }
  51. }</font>
复制代码
在STM32的某些应用中,用户有周期性执行某些程序的要求,使用定时器可以产生固定的时间周期,满足 这样的需求。

STM32相关特征:
STM32高级定时器TIM1、TIM8,通用定时器TIM2、TIM3、TIM4、TIM5;
定时器最大时钟72MHz,配合预分频,提供灵活的时钟周期;
每个TIM有4个独立捕获/比较通道,DMA/中断功能;
通道工作在输出比较定时模式,一个TIM至多可以提供4个不同的定时周期。

原理
TIM某输出/捕获通道工作在输出比较定时模式
计数器计数至比较值时产生中断,在中断中刷新捕获比较寄存器,这样在相同时间间隔后可产生下一次中断
TIM2时钟设置为36MHz,预分频设置为2,使用输出比较-翻转模式(Output Compare Toggle Mode)。
TIM2计数器时钟可表达为:TIM2 counter clock = TIMxCLK / (Prescaler +1) = 12 MHz。
设置TIM2_CCR1寄存器值为32768,则CC1更新频率为TIM2计数器时钟频率除以CCR1寄存器值,为366.2 Hz。因此,TIM2通道1可产生一个频率为183.1 Hz的周期信号。

同理,根据寄存器TIM2_CCR2 、TIM2_CCR3和 TIM2_CCR4的值,TIM2通道2可产生一个频率为366.3 Hz的周期信号;TIM2通道3可产生一个频率为732.4 Hz的周期信号;TIM2通道4可产生一个频率为1464.8 Hz的周期信号。
  1. <font size="2">#include "stm32f10x_lib.h"

  2. TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
  3. TIM_OCInitTypeDef  TIM_OCInitStructure;
  4. vu16 CCR1_Val = 32768;
  5. vu16 CCR2_Val = 16384;
  6. vu16 CCR3_Val = 8192;
  7. vu16 CCR4_Val = 4096;
  8. ErrorStatus HSEStartUpStatus;

  9. void RCC_Configuration(void);
  10. void GPIO_Configuration(void);
  11. void NVIC_Configuration(void);
  12.    
  13. int main(void)
  14. {
  15. #ifdef DEBUG
  16.   debug();
  17. #endif

  18.   RCC_Configuration();
  19.   NVIC_Configuration();
  20.   GPIO_Configuration();

  21.   TIM_TimeBaseStructure.TIM_Period = 65535;   //这里必须是65535
  22.   TIM_TimeBaseStructure.TIM_Prescaler = 2;      
  23.   TIM_TimeBaseStructure.TIM_ClockDivision = 0;   
  24.   TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  25.   TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  26.   TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;  //管脚输出模式:翻转(TIM输出比较触发模式)
  27.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_1;         
  28.   TIM_OCInitStructure.TIM_Pulse = CCR1_Val;   //翻转周期
  29.   TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;//TIM输出比较极性低
  30.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  31.   TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Disable);//失能TIMx在CCR1上的预装载寄存器
  32.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_2;         
  33.   TIM_OCInitStructure.TIM_Pulse = CCR2_Val;
  34.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  35.   TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Disable);
  36.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_3;         
  37.   TIM_OCInitStructure.TIM_Pulse = CCR3_Val;
  38.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  39.   TIM_OC3PreloadConfig(TIM2, TIM_OCPreload_Disable);
  40.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_4;         
  41.   TIM_OCInitStructure.TIM_Pulse = CCR4_Val;
  42.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  43.   TIM_OC4PreloadConfig(TIM2, TIM_OCPreload_Disable);
  44.   TIM_Cmd(TIM2, ENABLE);
  45.   TIM_ITConfig(TIM2, TIM_IT_CC1 | TIM_IT_CC2 | TIM_IT_CC3 | TIM_IT_CC4, ENABLE);

  46.   while (1)
  47.   {
  48.   }
  49. }

  50. void RCC_Configuration(void)
  51. {  
  52.   RCC_DeInit();
  53.   RCC_HSEConfig(RCC_HSE_ON);
  54.   HSEStartUpStatus = RCC_WaitForHSEStartUp();

  55.   if(HSEStartUpStatus == SUCCESS)
  56.   {
  57.    
  58.     FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
  59.     FLASH_SetLatency(FLASH_Latency_2);</font>
复制代码

中断服务:
  1. <font size="2">void TIM2_IRQHandler(void)
  2. {

  3.   if (TIM_GetITStatus(TIM2, TIM_IT_CC1) != RESET)//检查指定的TIM中断发生与否
  4.   {
  5.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC1 );//清除TIMx的中断待处理位
  6. capture = TIM_GetCapture1(TIM2);
  7. TIM_SetCompare1(TIM2, capture + CCR1_Val );//设置TIMx自动重装载寄存器值
  8.         //将TIM2_CC1的值增加CCR1_Val,使得下一个TIM事件也需要CCR1_Val个脉冲,
  9.   }

  10.   if (TIM_GetITStatus(TIM2, TIM_IT_CC2) != RESET)
  11.   {
  12.      TIM_ClearITPendingBit(TIM2, TIM_IT_CC2);
  13. capture = TIM_GetCapture2(TIM2);
  14.     TIM_SetCompare2(TIM2, capture + CCR2_Val);
  15.   }

  16.   if (TIM_GetITStatus(TIM2, TIM_IT_CC3) != RESET)
  17.   {
  18.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC3);
  19. capture = TIM_GetCapture3(TIM2);
  20.     TIM_SetCompare3(TIM2, capture + CCR3_Val);
  21.   }
  22.   if (TIM_GetITStatus(TIM2, TIM_IT_CC4) != RESET)
  23.   {
  24.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC4);
  25. capture = TIM_GetCapture4(TIM2);
  26.     TIM_SetCompare4(TIM2, capture + CCR4_Val);
  27.   }  
  28. }</font>
复制代码

TIMx的输出比较模式是用来控制一个输出波形或者指示何时一段给定的的时间已经到时。
当计数器与捕获/比较寄存器的内容相同时,输出比较功能做如下操作:
将输出比较模式(TIMx_CCMRx寄存器中的OCxM位)和输出极性(TIMx_CCER寄存器中的CCxP位)定义的值输出到对应的管脚上。在比较匹配时,输出管脚可以保持它的电平(OCxM=000)、
被设置成有效电平(OCxM=001)、
被设置成无有效电平(OCxM=010)
或进行翻转(OCxM=011)。


这几种模式对应定义在库中

  1. <font size="2">TIM_OCInitTypeDef structure
  2. TIM_OCInitTypeDef定义于文件“stm32f10x_tim.h”:
  3. typedef struct
  4. {
  5. u16 TIM_OCMode;
  6. u16 TIM_Channel;
  7. u16 TIM_Pulse;
  8. u16 TIM_OCPolarity;
  9. } TIM_OCInitTypeDef;</font>
复制代码
TIM_OCMode选择定时器模式。该参数取值见下表。
3.jpg
●设置中断状态寄存器中的标志位(TIMx_SR寄存器中的CCxIF位)。
●若设置了相应的中断屏蔽(TIMx_DIER寄存器中的CCXIE位),则产生一个中断。
●若设置了相应的使能位(TIMx_DIER寄存器中的CCxDE位,TIMx_CR2寄存器中的CCDS位选择DMA请求功能),则产生一个DMA请求。
TIMx_CCMRx中的OCxPE位选择TIMx_CCRx寄存器是否需要使用预装载寄存器。
在输出比较模式下,更新事件UEV对OCxREF和OCx输出没有影响。
同步的精度可以达到计数器的一个计数周期。输出比较模式(在单脉冲模式下)也能用来输出一个单脉冲。
输出比较模式的配置步骤:

1.选择计数器时钟(内部,外部,预分频器)
2.将相应的数据写入TIMx_ARR和TIMx_CCRx寄存器中
3.如果要产生一个中断请求和/或一个DMA请求,设置CCxIE位和/或CCxDE位。
4.选择输出模式,例如:必须设置OCxM=’011’、OCxPE=’0’、CCxP=’0’和CCxE=’1’,当计数器CNT与CCRx匹配时翻转OCx的输出管脚,CCRx预装载未用,开启OCx输出且高电平有效。
5.设置TIMx_CR1寄存器的CEN位启动计数器
TIMx_CCRx寄存器能够在任何时候通过软件进行更新以控制输出波形,条件是未使用预装载寄存器(OCxPE=’0’,否则TIMx_CCRx影子寄存器只能在发生下一次更新事件时被更新)。

这三种模式我都拿板子的例程改了一下 ~

000:冻结。输出比较寄存器TIMx_CCR1与计数器TIMx_CNT间的比较对OC1REF不起作用;也就是Timing模式
001:匹配时设置通道1为有效电平。当计数器TIMx_CNT的值与捕获/比较寄存器1(TIMx_CCR1)相同时,强制OC1REF为高。这个也就是ACTIVE模式
010:匹配时设置通道1为无效电平。当计数器TIMx_CNT的值与捕获/比较寄存器1(TIMx_CCR1)相同时,强制OC1REF为低。这个也就是INACTIVE模式
011:翻转。当TIMx_CCR1=TIMx_CNT时,翻转OC1REF的电平。也就是toggle模式。

程序如下:
  1. TIM_TimeBaseStructure.TIM_Period = 65535;         
  2.   TIM_TimeBaseStructure.TIM_Prescaler = 0;      
  3.   TIM_TimeBaseStructure.TIM_ClockDivision = 0;   
  4.   TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  5.   TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  6.   TIM_PrescalerConfig(TIM2, 35999, TIM_PSCReloadMode_Immediate);
  7.   TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Timing;//这个地方就是改比较模式的
复制代码

但是由于比较模式无论选哪个对于产生中断的作用是一样的,所以选TIMING都可以
  1. TIM_OCInitStructure.TIM_Channel = TIM_Channel_1;         
  2. TIM_OCInitStructure.TIM_Pulse = CCR1_Val;
  3. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
  4. TIM_OCInit(TIM2, &TIM_OCInitStructure);
  5. TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Disable);//
复制代码
TIMx_CCRx寄存器能够在任何时候通过软件进行更新以控制输出波形,条件是未使用预装载寄存器(OCxPE=’0’,否则TIMx_CCRx影子寄存器只能在发生下一次更新事件时被更新)。这里设置为Disable

就是为了后面在中断服务子程序可以修改TIMx_CCR实时起作用~
  1. <font size="4"> </font><font size="2"> TIM_OCInitStructure.TIM_Channel = TIM_Channel_2;         
  2.   TIM_OCInitStructure.TIM_Pulse = CCR2_Val;
  3.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  4.   TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Disable);
  5.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_3;         
  6.   TIM_OCInitStructure.TIM_Pulse = CCR3_Val;
  7.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  8.   TIM_OC3PreloadConfig(TIM2, TIM_OCPreload_Disable);
  9.   TIM_OCInitStructure.TIM_Channel = TIM_Channel_4;         
  10.   TIM_OCInitStructure.TIM_Pulse = CCR4_Val;
  11.   TIM_OCInit(TIM2, &TIM_OCInitStructure);
  12.   TIM_OC4PreloadConfig(TIM2, TIM_OCPreload_Disable);
  13.   TIM_ARRPreloadConfig(TIM2, ENABLE);//TIM_OCPreload_Enable
  14.   TIM_ITConfig(TIM2, TIM_IT_CC1 | TIM_IT_CC2 | TIM_IT_CC3 | TIM_IT_CC4, ENABLE);
  15.     // STM3210B-LK1,  set PC.04 - PC.07
  16.   GPIO_SetBits(GPIOC, GPIO_Pin_4 |GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7);
  17.   TIM_Cmd(TIM2, ENABLE);
  18.   while (1)
  19.   {
  20.   }
  21. }</font>
复制代码

评分

参与人数 1 ST金币 +2 收起 理由
MrJiu + 2 赞一个!

查看全部评分

收藏 2 评论6 发布时间:2017-7-13 17:16

举报

6个回答
野火_firege 回答时间:2017-7-13 17:39:09
帮你顶一下,看你写了那么多。
wolfgang 回答时间:2017-7-13 23:37:59
好贴,很详细
shuolang126 回答时间:2017-7-14 09:03:28
挺详细的!
MrJiu 回答时间:2017-7-14 09:25:10
还不错的样子1!!!!!!!!
Savy1314 回答时间:2017-7-14 14:37:39
野火_firege 发表于 2017-7-13 17:39
帮你顶一下,看你写了那么多。

哈哈
qwerdfbas 回答时间:2020-9-15 23:47:57
照着仿真还是不行

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版