STM32F103 使用TIM3产生四路PWM

程序如下:

  1. /*******************************************************************************
  2. * 程序说明 : 思路PWM波生成函数
  3. * 函数功能 : 使用TIM3的PWM功能生成思路PWM,
  4. * 输 入 : 无
  5. * 输 出 : 四路PWM,通过GPIO引脚复用,对TIM3的四个输出通道引脚重映射为PC6、PC7、PC8、PC9
  6. *******************************************************************************/
  7.  
  8. #include"stm32f10x.h"
  9.  
  10. void RCC_Cfg(void);
  11. void GPIO_Cfg(void);
  12. void TIM_Cfg(void);
  13. void NVIC_Cfg(void);
  14. void delay_ms(u32 i);
  15. void PWM_Cfg(float dutyfactor1,float dutyfactor2,float dutyfactor3,float dutyfactor4);
  16.  
  17. int main()
  18. {
  19. u8 flag = ;
  20. float ooo=0.5;
  21. RCC_Cfg();
  22. NVIC_Cfg();
  23. GPIO_Cfg();
  24. TIM_Cfg();
  25.  
  26. //开启定时器2
  27. TIM_Cmd(TIM3,ENABLE);
  28.  
  29. //呼吸灯
  30. while(){
  31. PWM_Cfg(ooo,,+0.5*ooo,-*ooo);
  32.  
  33. if(flag == )
  34. {
  35. ooo=ooo+0.002;
  36. }
  37. if(flag == )
  38. {
  39. ooo=ooo-0.002;
  40. }
  41. if(ooo>){
  42. flag = ;
  43. }
  44. if(ooo<0.5)
  45. {
  46. flag = ;
  47. }
  48.  
  49. }
  50. }
  51.  
  52. void GPIO_Cfg(void)
  53. {
  54.  
  55. GPIO_InitTypeDef GPIO_InitStructure;
  56. //RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO,ENABLE);
  57.  
  58. //全部映射,将TIM3_CH2映射到PB5
  59. //根据STM32中文参考手册2010中第第119页可知:
  60. //当没有重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PA6,PA7,PB0,PB1
  61. //当部分重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PB4,PB5,PB0,PB1
  62. //当完全重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PC6,PC7,PC8,PC9
  63. //也即是说,完全重映射之后,四个通道的PWM输出引脚分别为PC6,PC7,PC8,PC9,我们用到了通道1和通道2,所以对应引脚为PC6,PC7,PC8,PC9,我们用到了通道1和通道2,所以对应引脚为
  64. GPIO_PinRemapConfig(GPIO_FullRemap_TIM3, ENABLE);
  65.  
  66. //部分重映射的参数
  67. //GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE);
  68.  
  69. //设置PC6、PC7、PC8、PC9为复用输出,输出4路PWM
  70. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  71. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  72. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7|GPIO_Pin_8|GPIO_Pin_9;
  73. GPIO_Init(GPIOC,&GPIO_InitStructure);
  74.  
  75. }
  76.  
  77. void TIM_Cfg(void)
  78. {
  79. //定义结构体
  80. TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  81.  
  82. //重新将Timer设置为缺省值
  83. TIM_DeInit(TIM3);
  84. //采用内部时钟给TIM2提供时钟源
  85. TIM_InternalClockConfig(TIM3);
  86.  
  87. //预分频系数为0,即不进行预分频,此时TIMER的频率为72MHzre.TIM_Prescaler =0;
  88. TIM_TimeBaseStructure.TIM_Prescaler = ;
  89. //设置计数溢出大小,每计20000个数就产生一个更新事件
  90. TIM_TimeBaseStructure.TIM_Period = - ;
  91. //设置时钟分割
  92. TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
  93. //设置计数器模式为向上计数模式
  94. TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  95.  
  96. //将配置应用到TIM2中
  97. TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);
  98. //清除溢出中断标志
  99. //TIM_ClearFlag(TIM2, TIM_FLAG_Update);
  100. //禁止ARR预装载缓冲器
  101. //TIM_ARRPreloadConfig(TIM2, DISABLE);
  102. //开启TIM2的中断
  103. //TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);
  104.  
  105. }
  106.  
  107. /*******************************************************************************
  108. * 函 数 名 : PWM波产生配置函数
  109. * 函数功能 : PWM_Cfg
  110. * 输 入 : dutyfactor 占空比数值,大小从0.014到100
  111. * 输 出 : 无
  112. *******************************************************************************/
  113. void PWM_Cfg(float dutyfactor1,float dutyfactor2,float dutyfactor3,float dutyfactor4)
  114. {
  115. TIM_OCInitTypeDef TIM_OCInitStructure;
  116. //设置缺省值
  117. TIM_OCStructInit(&TIM_OCInitStructure);
  118.  
  119. //TIM3的CH1输出
  120. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
  121. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
  122. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
  123. //设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
  124. TIM_OCInitStructure.TIM_Pulse = dutyfactor1 * / ;
  125. TIM_OC1Init(TIM3, &TIM_OCInitStructure);
  126.  
  127. //TIM3的CH2输出
  128. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
  129. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
  130. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
  131. //设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
  132. TIM_OCInitStructure.TIM_Pulse = dutyfactor2 * / ;
  133. TIM_OC2Init(TIM3, &TIM_OCInitStructure);
  134.  
  135. //TIM3的CH3输出
  136. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
  137. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
  138. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
  139. //设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
  140. TIM_OCInitStructure.TIM_Pulse = dutyfactor3 * / ;
  141. TIM_OC3Init(TIM3, &TIM_OCInitStructure);
  142.  
  143. //TIM3的CH4输出
  144. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
  145. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
  146. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
  147. //设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
  148. TIM_OCInitStructure.TIM_Pulse = dutyfactor4 * / ;
  149. TIM_OC4Init(TIM3, &TIM_OCInitStructure);
  150.  
  151. //使能输出状态
  152. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  153.  
  154. //设置TIM3的PWM输出为使能
  155. TIM_CtrlPWMOutputs(TIM3,ENABLE);
  156. }
  157.  
  158. void NVIC_Cfg(void)
  159. {
  160. //定义结构体
  161. NVIC_InitTypeDef NVIC_InitStructure;
  162.  
  163. //选择中断分组1
  164. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  165.  
  166. //选择TIM2的中断通道
  167. NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
  168. //抢占式中断优先级设置为0
  169. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = ;
  170. //响应式中断优先级设置为0
  171. NVIC_InitStructure.NVIC_IRQChannelSubPriority = ;
  172. //使能中断
  173. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  174.  
  175. NVIC_Init(&NVIC_InitStructure);
  176. }
  177.  
  178. void RCC_Cfg(void)
  179. {
  180. //定义错误状态变量
  181. ErrorStatus HSEStartUpStatus;
  182.  
  183. //将RCC寄存器重新设置为默认值
  184. RCC_DeInit();
  185.  
  186. //打开外部高速时钟晶振
  187. RCC_HSEConfig(RCC_HSE_ON);
  188.  
  189. //等待外部高速时钟晶振工作
  190. HSEStartUpStatus = RCC_WaitForHSEStartUp();
  191.  
  192. if(HSEStartUpStatus == SUCCESS)
  193. {
  194. //设置AHB时钟(HCLK)为系统时钟
  195. RCC_HCLKConfig(RCC_SYSCLK_Div1);
  196.  
  197. //设置高速AHB时钟(APB2)为HCLK时钟
  198. RCC_PCLK2Config(RCC_HCLK_Div1);
  199.  
  200. //设置低速AHB时钟(APB1)为HCLK的2分频
  201. RCC_PCLK1Config(RCC_HCLK_Div2);
  202.  
  203. //设置FLASH代码延时
  204. FLASH_SetLatency(FLASH_Latency_2);
  205.  
  206. //使能预取指缓存
  207. FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
  208.  
  209. //设置PLL时钟,为HSE的9倍频 8MHz * 9 = 72MHz
  210. RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
  211.  
  212. //使能PLL
  213. RCC_PLLCmd(ENABLE);
  214.  
  215. //等待PLL准备就绪
  216. while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
  217.  
  218. //设置PLL为系统时钟源
  219. RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
  220.  
  221. //判断PLL是否是系统时钟
  222. while(RCC_GetSYSCLKSource() != 0x08);
  223.  
  224. }
  225.  
  226. //允许TIM2的时钟
  227. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
  228.  
  229. //允许GPIO的时钟
  230. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC|RCC_APB2Periph_AFIO,ENABLE);
  231.  
  232. }
  233.  
  234. void TIM2_IRQHandler(void)
  235. {
  236. u16 aa=;
  237. if(TIM_GetFlagStatus(TIM2,TIM_IT_Update)!=RESET)
  238. {
  239. //清除TIM2的中断待处理位
  240. TIM_ClearITPendingBit(TIM2 , TIM_FLAG_Update);
  241.  
  242. TIM_Cmd(TIM2,DISABLE);
  243. //通过循环让灯闪烁
  244. while (aa){
  245. GPIO_SetBits(GPIOC,GPIO_Pin_3);
  246. delay_ms();
  247. GPIO_ResetBits(GPIOC,GPIO_Pin_3);
  248. delay_ms();
  249. aa--;
  250. }
  251. //使灯的状态为灭
  252. GPIO_SetBits(GPIOC,GPIO_Pin_3);
  253. TIM_Cmd(TIM2,ENABLE);
  254. }
  255. }
  256.  
  257. void delay_ms(u32 i)
  258. {
  259. u32 temp;
  260. SysTick->LOAD=*i; //设置重装数值, 72MHZ时
  261. SysTick->CTRL=0X01; //使能,减到零是无动作,采用外部时钟源
  262. SysTick->VAL=; //清零计数器
  263. do
  264. {
  265. temp=SysTick->CTRL; //读取当前倒计数值
  266. }
  267. while((temp&0x01)&&(!(temp&(<<)))); //等待时间到达
  268. SysTick->CTRL=; //关闭计数器
  269. SysTick->VAL=; //清空计数器
  270. }

在产生PWM时,如果输出引脚已经被使用,就要对引脚进行重映射,阅读《STM32中文参考手册2010》第119页可知:

对TIM3而言:

1、当没有重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PA6,PA7,PB0,PB1
2、当部分重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PB4,PB5,PB0,PB1
3、当完全重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PC6,PC7,PC8,PC9

为了整齐,我们选择完全重映射,使用的函数是:

  1. GPIO_PinRemapConfig(GPIO_FullRemap_TIM3, ENABLE);

如果想使用部分映射,参数用GPIO_PartialRemap_TIM3:

  1. GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE);

STM32F103 使用TIM3产生四路PWM的更多相关文章

  1. stm32之PWM博客好文收藏

    https://www.cnblogs.com/jiwangbujiu/p/5616376.html STM32F103 使用TIM3产生四路PWM https://www.cnblogs.com/c ...

  2. 增量式PID的stm32实现(转)

    源:增量式PID的stm32实现,整定过程 首先说说增量式PID的公式,这个关系到MCU算法公式的书写,实际上两个公式的写法是同一个公式变换来得,不同的是系数的差异. 资料上比较多的是: 还有一种是: ...

  3. STM32F103定时器输出PWM波控制直流电机

    这个暑假没有回家,在学校准备九月份的电子设计竞赛.今天想给大家分享一下STM32高级定时器输出PWM波驱动直流电机的问题.. 要想用定时器输出的PWM控制直流电机,,首先要理解“通道”的概念..一个定 ...

  4. STM32之PWM君

    PWM..英语好的人估计又知道这三个大写字母代表哪三个英语单词了.小弟不才,就说中文意思好了:脉冲宽度调制,玩过飞思卡尔的人估计对PWM非常的不陌生吧.电机驱动需要PWM,控制舵机的转向需要PWM,总 ...

  5. STM32中的PWM的频率和占空比的设置

    转于http://blog.csdn.net/liming0931/article/details/8491468 下面的这个是stm32的定时器逻辑图,上来有助于理解:   TIM3的ARR寄存器和 ...

  6. pwm最后的解释

    之前学东西总是模模糊糊,前几天看了pwm,虽然知道怎么配置,但是如果让我自己去写一个pwm的程序,我却不知如何下手. 不知道如何配置他的频率和占空比.今天痛定思痛,决定彻底搞懂pwm. 百度给 的答案 ...

  7. (五)转载:通用定时器PWM输出

    1.     TIMER输出PWM基本概念 脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非常有 ...

  8. 【转载】 stm32之PWM

    发现这位博主的博客被大量的转发,我也转载一篇,谁叫人家写的好呢. 原文地址:http://blog.sina.com.cn/s/blog_49cb42490100s6uh.html 脉冲宽度调制(PW ...

  9. STM32(7)——通用定时器PWM输出

    1.TIMER输出PWM基本概念         脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种 ...

随机推荐

  1. 使用HttpUtils 上传视频文件

    private void shangchuan(){                 //文件的路径        //File file=new File(path);        File fi ...

  2. sql 指定范围 获取随机数

    DECLARE @nMinimumCount INT= 1DECLARE @nMaximumCount INT= 100SELECT abs(CHECKSUM(NEWID()))%(@nMaximum ...

  3. hadoop中联结不同来源数据

    装载自http://www.cnblogs.com/dandingyy/archive/2013/03/01/2938462.html 有时可能需要对来自不同源的数据进行综合分析: 如下例子: 有Cu ...

  4. java 常见异常总结

    异常1:java.util.NoSuchElementException: No line found 原因:Java 是顺序执行的 你执行到.close() 后就代表 你关闭了 流,你再去调用已经被 ...

  5. gen_create_syn.sql

    set echo off feedback off verify off pagesize 0 linesize 120 ---变量从 sqlplus 的 call代码 传递过来 . -- 1 表示连 ...

  6. iOS开发中控制器切换方式Modal

    简介 在iPhone开发中 Modal是一种常见的切换控制器的方式 默认是从屏幕底部往上弹出,直到完全盖住后面的内容为止 在iPad开发中 Modal的使用频率也是非常高的 对比iPhone开发,Mo ...

  7. PAT (Advanced Level) 1071. Speech Patterns (25)

    简单题. #include<cstdio> #include<cstring> #include<cmath> #include<vector> #in ...

  8. UML类图基本画法

    转自:http://www.cnblogs.com/import-this/p/5330501.html 1. 类 类包括属性方法 下图的生命就是属性 代谢就是方法 2.基本关系 a.泛化(Gener ...

  9. 要你的祝福.lrc

    要你的祝福(电影<我是路人甲>插曲 试听版) - 李潇潇 午夜的温度慢慢起舞 穿梭的人潮有些荒芜 开始欢呼 开始麻木 谁被谁在安抚 落单的幸福变得模糊 孤单的城市独自起舞 也许满足 也许糊 ...

  10. Struts2--带参数的结果集

    带参数的结果集: 配置文件: <result type="redirect">/user_success.jsp?t=${type}</result> js ...