当前位置:   article > 正文

stm32-定时器输出比较PWM_stm32 输出比较

stm32 输出比较

目录

一、输出比较简介

二、PWM简介

三、输出比较模式实现

1.输出比较框图(以通用定时器为例)

2.PWM基本结构

四、固件库实现

1.程序1:PWM呼吸灯

 2.程序2:PWM驱动直流电机

3.程序3:控制舵机 


一、输出比较简介

6427569e13dc4058bc4108d2252324d8.png

死区生成和互补输出一般用于对电机的控制  

二、PWM简介

217520c3ee6d4309a335940969b560c2.png

 惯性系统:即要能使人眼产生视觉停留的系统

三、输出比较模式实现

4df2aa6225314807bd9e305e6156d983.png

1.输出比较框图(以通用定时器为例)

b20f276000924d83b123d9bcd4402f15.pngd5b1cc384dcb4bff8e3172aa26bb5cdb.pngebc01af3e04148d988d242ccd5663b17.png 

 高级定时器比通用定时器多了个互补输出和死区生成852fe456c92146febbc5a922cc620899.png

如右图,两个mos管构成了推挽电路,上管导通下管关闭输出高电平,下管导通上管关闭输出低电平,两管都关闭为高阻态,两管都导通为短路,会对元器件造成损伤,所以两管不能同时导通

互补输出:当单片机要控制这个电路时,就需要两个输出端口,且二者电平要相反,即互补,而这里OC1和OC1N就是互补的两个端口,即互补输出

死区发生器:但是若在上管关闭的瞬间下管就导通,很可能由于器件的不理想而出现上下管都导通的情况,为了避免这种情况发生,于是又是死区发生器,即他可以在上管关闭后延迟一段时间再导通下管,避免同时导通

2.PWM基本结构

10788d1f0c4c4fc9bf92ee1f9302d992.png

1e024aa74b8a49988ef208aef673b9f7.png

四、固件库实现

1.程序1:PWM呼吸灯

1.我使用的是TIM3的CH2的重定义引脚PB5,所以要打开AFIO时钟

2.开启TIM3和GPIO的时钟

3.初始化GPIO结构体。注意:要记得使用库函数进行重定义

4.选择TIM3的时钟输入,可以为内部时钟,外部时钟模式1和2,

        外部时钟模式1:来源可以是ITR(其他定时器,多用于定时器级联),可以是ETR(外部时  钟),可以是CH1引脚的边沿,CH1引脚和CH2引脚(多用于输入捕获测频率)

        外部时钟模式2:ETR的触发控制模式

5.初始化TIM3结构体--配置PSC,ARR,计数模式,等等

6.初始化OC结构体--PWM模式选择,CCR,输出ref的有效电平

7.定时器使能

  1. //使用TIM3的通道2的重定义引脚PB5-指南者上面是红灯
  2. //我们配置
  3. void PWM_Config()
  4. {
  5. //首先开启GPIO时钟
  6. //开启AFIO时钟,因为用到了重定义
  7. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO,ENABLE);
  8. //开启定时器的时钟
  9. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
  10. //重定义引脚
  11. //选择部分重定义
  12. GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3,ENABLE);
  13. //初始化GPIO结构体-PB5-输出比较
  14. GPIO_InitTypeDef GPIO_InitStruct;
  15. GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
  16. GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽,手册可看
  17. GPIO_InitStruct.GPIO_Pin = GPIO_Pin_5;
  18. GPIO_Init(GPIOB,&GPIO_InitStruct);
  19. //选择TIM3的时钟输入
  20. TIM_InternalClockConfig(TIM3);//我们直接使用内部时钟
  21. //配置TIM结构体
  22. TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
  23. TIM_TimeBaseInitStruct.TIM_Prescaler = 72-1;//PSC
  24. TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//计数模式
  25. TIM_TimeBaseInitStruct.TIM_Period = 100-1;//ARR
  26. TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;//输入滤波器的分频
  27. TIM_TimeBaseInitStruct.TIM_RepetitionCounter = 0;//重复计数器,只有高级定时器才有
  28. TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStruct);
  29. //初始化OC结构体
  30. TIM_OCInitTypeDef TIM_OCInitStruct;
  31. TIM_OCStructInit(&TIM_OCInitStruct);//先赋初值,因为我们没有把结构体配置完全
  32. TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1;//模式选择-PWM1
  33. TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;//输出使能
  34. TIM_OCInitStruct.TIM_Pulse = 0;//CCR,这里我们不配置CCR的值,后面用一个库函数直接在主函数
  35. 里面配置,实现呼吸灯
  36. TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//输出ref极性,选择有效电平,即ref有效时输出高电平
  37. //TIM_OCInitStruct.TIM_OCNPolarity = ;
  38. //TIM_OCInitStruct.TIM_OCIdleState = ;
  39. //TIM_OCInitStruct.TIM_OCNIdleState = ;
  40. //TIM_OCInitStruct.TIM_OutputNState = ;
  41. TIM_OC2Init(TIM3,&TIM_OCInitStruct);
  42. //启动定时器
  43. TIM_Cmd(TIM3,ENABLE);
  44. }
  45. void PWM_SetCompare2(uint16_t Compare)
  46. {
  47. TIM_SetCompare2(TIM3,Compare);
  48. }
  1. main.c
  2. #include "stm32f10x.h" // Device header
  3. #include "bsp_led.h"
  4. #include ".\tim\bsp_tim.h"
  5. extern uint16_t Num;//定时器都是16位的
  6. int i;
  7. void Delay(u32 i)
  8. {
  9. u32 temp;
  10. SysTick->LOAD=9000*i; //设置重装数值, 72MHZ时
  11. SysTick->CTRL=0X01; //使能,减到零是无动作,采用外部时钟源
  12. SysTick->VAL=0; //清零计数器
  13. do
  14. {
  15. temp=SysTick->CTRL; //读取当前倒计数值
  16. }
  17. while((temp&0x01)&&(!(temp&(1<<16)))); //等待时间到达
  18. SysTick->CTRL=0; //关闭计数器
  19. SysTick->VAL=0; //清空计数器
  20. }
  21. //ARR=99-->PWM一个周期是100,那么分辨率为1%
  22. //占空比 = CCR/(ARR+1)
  23. //频率 = 计数器溢出频率 = CK_PSC/(PSC+1)/(ARR+1) = 72M/720/100= 1000HZ ->1ms
  24. int main()
  25. {
  26. LED_GPIO_Config();
  27. PWM_Config();
  28. while(1)
  29. {
  30. for(i=0;i<=100;i++)
  31. {
  32. PWM_SetCompare2(i);
  33. Delay(10);
  34. }
  35. for(i=100;i>=0;i--)
  36. {
  37. PWM_SetCompare2(i);
  38. Delay(10);
  39. }
  40. }
  41. }

 2.程序2:PWM驱动直流电机

具体TIM的配置过程同呼吸灯一样,同样是输出不同的PWM占空比来实现电机的不同速度

我们需要三个引脚,一个输出PWM给电机,两个接电机的控制引脚

  • 使用PA2输出PWM,AIN1/2接到PA4/5
  • 频率设置为1KHZ(可以自己随便设置)
  • 定义一个八位有符号的变量Speed,+:正转  -:反转
  • 使用GPIO_SetBits/ResetBits();来设置AIN1/2的电平高低
  • 使用按键来改变转速
  1. void PWM_Config()
  2. {
  3. //开启时钟
  4. //使用TIM2的CH3的PA2
  5. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
  6. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//开启PWM引脚
  7. //初始化GPIO
  8. GPIO_InitTypeDef GPIO_InitStruct;
  9. GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
  10. GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽,手册可看
  11. GPIO_InitStruct.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_4 | GPIO_Pin_5 ;
  12. GPIO_Init(GPIOA,&GPIO_InitStruct);
  13. //选择时基单元的时钟-为内部时钟--定时器上电后默认是内部时钟,故不写这一个也行
  14. TIM_InternalClockConfig(TIM2);
  15. //初始化时基单元
  16. TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
  17. //配置1KHZ的PWM
  18. TIM_TimeBaseInitStruct.TIM_Prescaler = 720-1;//PSC-预分频器
  19. TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//向上计数
  20. TIM_TimeBaseInitStruct.TIM_Period = 100-1;//ARR寄存器-重装载寄存器
  21. TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;/*不分频----滤波器的采样频率,可以由内部时钟直接提供,
  22. 也可以由内部时钟加一个时钟分频而来,
  23. 分频系数就是由TIM_ClockDivision决定*/
  24. TIM_TimeBaseInitStruct.TIM_RepetitionCounter = 0;//重复计数器,只有高级定时器才有
  25. TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);
  26. //初始化OC-输出比较结构体
  27. TIM_OCInitTypeDef TIM_OCInitStruct;
  28. TIM_OCStructInit(&TIM_OCInitStruct);//因为结构体里面的成员有些是高级定时器采用得到,所以这里就先全部初始化一遍,然后再配置具体的值
  29. TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1;//输出比较模式
  30. TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;
  31. //TIM_OCInitStruct.TIM_Pulse = 50;//CRR
  32. TIM_OCInitStruct.TIM_Pulse = 0;//输出要求波型,这里的CRR就不需要了,用固件库的一个函数 TIM_SetCompare3 直接配置CRR
  33. TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//输出比较极性
  34. TIM_OC3Init(TIM2,&TIM_OCInitStruct);//CH3通道
  35. //启动定时器
  36. TIM_Cmd(TIM2,ENABLE);
  37. }
  1. //电机控制函数
  2. void Motor_SetSpeed(int8_t Speed)//+:正转 -:反转
  3. {
  4. if(Speed >= 0)
  5. {
  6. GPIO_SetBits(GPIOA,GPIO_Pin_4);
  7. GPIO_ResetBits(GPIOA,GPIO_Pin_5);
  8. TIM_SetCompare3(TIM3,Speed);
  9. }
  10. else
  11. {
  12. GPIO_ResetBits(GPIOA,GPIO_Pin_4);
  13. GPIO_SetBits(GPIOA,GPIO_Pin_5);
  14. TIM_SetCompare3(TIM3,-Speed);
  15. }
  16. }
  1. //主函数
  2. #include "stm32f10x.h" // Device header
  3. #include ".\tim\bsp_tim.h"
  4. #include ".\KEY\bsp_key.h"
  5. uint8_t KeyNum;
  6. int8_t Speed;
  7. int main()
  8. {
  9. KEY_GPIO_Config();
  10. PWM_Config();
  11. while(1)
  12. {
  13. KeyNum = Key_Scan();
  14. if(KeyNum == 0)
  15. {
  16. Speed += 20;
  17. if(Speed > 100)
  18. {
  19. Speed = -100;
  20. }
  21. }
  22. else if(KeyNum == 1)
  23. {
  24. Speed -= 20;
  25. if(Speed < -100)
  26. {
  27. Speed = 100;
  28. }
  29. }
  30. Motor_SetSpeed(Speed);
  31. }
  32. }

3.程序3:控制舵机 

0ee84118f09c434898681decee12dc1c.png

-TB6612驱动板

VM->STLINK的5v

VCC->面包板3.3v

GND->面包板负极

AO1 AO2 接电机

STBY->待机控制引脚,这里不需要待机,接面包3.3v

AIN1/2-> 任意接两个引脚

PWMA->PWM输出控制引脚

驱动VM放在左下角使用PA2输出PWM,AIN1/2接到PA4/5

  • 要点:输出如上图右侧所示的PWM波型
  • 指南者的PA0引脚为按键1,所以使用TIM2的CH3通道的PA2
  • PWM要求频率为50HZ,即总时间20ms,高电平占0.5~2.5ms,这里我们可以给ARR配置20000-1,PSC配置72-1
  • 封装Angle转换函数0-50  180->2500   -->y=Angle/180*2500+50, Angle使用浮点型,利于计算
  • 使用按键来改变角度

  1. #include "bsp_tim.h"
  2. void PWM_Config()
  3. {
  4. //开启时钟
  5. //使用TIM2的CH3的PA2
  6. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
  7. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//开启PWM引脚
  8. //初始化GPIO
  9. GPIO_InitTypeDef GPIO_InitStruct;
  10. GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
  11. GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽,手册可看
  12. GPIO_InitStruct.GPIO_Pin = GPIO_Pin_2;
  13. GPIO_Init(GPIOA,&GPIO_InitStruct);
  14. //选择时基单元的时钟-为内部时钟--定时器上电后默认是内部时钟,故不写这一个也行
  15. TIM_InternalClockConfig(TIM2);
  16. //初始化时基单元
  17. TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
  18. TIM_TimeBaseInitStruct.TIM_Prescaler = 72-1;//PSC-预分频器
  19. TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//向上计数
  20. TIM_TimeBaseInitStruct.TIM_Period = 20000-1;//ARR寄存器-重装载寄存器
  21. TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;/*不分频----滤波器的采样频率,可以由内部时钟直接提供,
  22. 也可以由内部时钟加一个时钟分频而来,
  23. 分频系数就是由TIM_ClockDivision决定*/
  24. TIM_TimeBaseInitStruct.TIM_RepetitionCounter = 0;//重复计数器,只有高级定时器才有
  25. TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);
  26. //初始化OC-输出比较结构体
  27. TIM_OCInitTypeDef TIM_OCInitStruct;
  28. TIM_OCStructInit(&TIM_OCInitStruct);//因为结构体里面的成员有些是高级定时器采用得到,所以这里就先全部初始化一遍,然后再配置具体的值
  29. TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1;//输出比较模式
  30. TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;
  31. //TIM_OCInitStruct.TIM_Pulse = 50;//CRR --舵机要求设置频率50HZ-即总时间20ms,高电平时间在0.5~2.5ms之间的PWM波型即CCR->500~2500
  32. TIM_OCInitStruct.TIM_Pulse = 0;//输出要求波型,这里的CRR就不需要了,用固件库的一个函数 TIM_SetCompare3 直接配置CRR
  33. TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//输出比较极性
  34. TIM_OC3Init(TIM2,&TIM_OCInitStruct);//CH3通道
  35. //启动定时器
  36. TIM_Cmd(TIM2,ENABLE);
  37. }
  38. //0.5ms-0度 2.5ms-180度
  39. //0度 CCR=500
  40. //180度 CCR=2500
  41. //舵机设置角度,范围0~180
  42. void Servo_SetAngle(float Angle)
  43. {
  44. TIM_SetCompare3(TIM2,Angle*2000/180+500);
  45. }
  46. void PWM_SetCompare3(uint16_t Compare)//设置CRR
  47. {
  48. TIM_SetCompare3(TIM2,Compare);
  49. }
  1. #include "stm32f10x.h" // Device header
  2. #include ".\tim\bsp_tim.h"
  3. #include ".\KEY\bsp_key.h"
  4. uint8_t KeyNum;
  5. float Angle;
  6. int main()
  7. {
  8. KEY_GPIO_Config();
  9. LED_GPIO_Config();
  10. PWM_Config();
  11. while(1)
  12. {
  13. KeyNum = Key_Scan();
  14. if(KeyNum == 0)
  15. {
  16. Angle+=30;
  17. if(Angle>180)
  18. {
  19. Angle = 0;
  20. }
  21. }
  22. else if(KeyNum == 1)
  23. {
  24. Angle-=30;
  25. if(Angle<0)
  26. {
  27. Angle = 180;
  28. }
  29. }
  30. Servo_SetAngle(Angle);
  31. }
  32. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Guff_9hys/article/detail/791324
推荐阅读
相关标签
  

闽ICP备14008679号