STM32F103 使用TIM3产生四路PWM

程序如下:

/*******************************************************************************
* 程序说明 : 思路PWM波生成函数
* 函数功能 : 使用TIM3的PWM功能生成思路PWM,
* 输 入 : 无
* 输 出 : 四路PWM,通过GPIO引脚复用,对TIM3的四个输出通道引脚重映射为PC6、PC7、PC8、PC9
*******************************************************************************/ #include"stm32f10x.h" void RCC_Cfg(void);
void GPIO_Cfg(void);
void TIM_Cfg(void);
void NVIC_Cfg(void);
void delay_ms(u32 i);
void PWM_Cfg(float dutyfactor1,float dutyfactor2,float dutyfactor3,float dutyfactor4); int main()
{
u8 flag = ;
float ooo=0.5;
RCC_Cfg();
NVIC_Cfg();
GPIO_Cfg();
TIM_Cfg(); //开启定时器2
TIM_Cmd(TIM3,ENABLE); //呼吸灯
while(){
PWM_Cfg(ooo,,+0.5*ooo,-*ooo); if(flag == )
{
ooo=ooo+0.002;
}
if(flag == )
{
ooo=ooo-0.002;
}
if(ooo>){
flag = ;
}
if(ooo<0.5)
{
flag = ;
} }
} void GPIO_Cfg(void)
{ GPIO_InitTypeDef GPIO_InitStructure;
//RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO,ENABLE); //全部映射,将TIM3_CH2映射到PB5
//根据STM32中文参考手册2010中第第119页可知:
//当没有重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PA6,PA7,PB0,PB1
//当部分重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PB4,PB5,PB0,PB1
//当完全重映射时,TIM3的四个通道CH1,CH2,CH3,CH4分别对应PC6,PC7,PC8,PC9
//也即是说,完全重映射之后,四个通道的PWM输出引脚分别为PC6,PC7,PC8,PC9,我们用到了通道1和通道2,所以对应引脚为PC6,PC7,PC8,PC9,我们用到了通道1和通道2,所以对应引脚为
GPIO_PinRemapConfig(GPIO_FullRemap_TIM3, ENABLE); //部分重映射的参数
//GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE); //设置PC6、PC7、PC8、PC9为复用输出,输出4路PWM
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7|GPIO_Pin_8|GPIO_Pin_9;
GPIO_Init(GPIOC,&GPIO_InitStructure); } void TIM_Cfg(void)
{
//定义结构体
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; //重新将Timer设置为缺省值
TIM_DeInit(TIM3);
//采用内部时钟给TIM2提供时钟源
TIM_InternalClockConfig(TIM3); //预分频系数为0,即不进行预分频,此时TIMER的频率为72MHzre.TIM_Prescaler =0;
TIM_TimeBaseStructure.TIM_Prescaler = ;
//设置计数溢出大小,每计20000个数就产生一个更新事件
TIM_TimeBaseStructure.TIM_Period = - ;
//设置时钟分割
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
//设置计数器模式为向上计数模式
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //将配置应用到TIM2中
TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);
//清除溢出中断标志
//TIM_ClearFlag(TIM2, TIM_FLAG_Update);
//禁止ARR预装载缓冲器
//TIM_ARRPreloadConfig(TIM2, DISABLE);
//开启TIM2的中断
//TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE); } /*******************************************************************************
* 函 数 名 : PWM波产生配置函数
* 函数功能 : PWM_Cfg
* 输 入 : dutyfactor 占空比数值,大小从0.014到100
* 输 出 : 无
*******************************************************************************/
void PWM_Cfg(float dutyfactor1,float dutyfactor2,float dutyfactor3,float dutyfactor4)
{
TIM_OCInitTypeDef TIM_OCInitStructure;
//设置缺省值
TIM_OCStructInit(&TIM_OCInitStructure); //TIM3的CH1输出
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
//设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
TIM_OCInitStructure.TIM_Pulse = dutyfactor1 * / ;
TIM_OC1Init(TIM3, &TIM_OCInitStructure); //TIM3的CH2输出
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
//设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
TIM_OCInitStructure.TIM_Pulse = dutyfactor2 * / ;
TIM_OC2Init(TIM3, &TIM_OCInitStructure); //TIM3的CH3输出
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
//设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
TIM_OCInitStructure.TIM_Pulse = dutyfactor3 * / ;
TIM_OC3Init(TIM3, &TIM_OCInitStructure); //TIM3的CH4输出
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //设置是PWM模式还是比较模式
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能,使能PWM输出到端口
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //设置极性是高还是低
//设置占空比,占空比=(CCRx/ARR)*100%或(TIM_Pulse/TIM_Period)*100%
TIM_OCInitStructure.TIM_Pulse = dutyfactor4 * / ;
TIM_OC4Init(TIM3, &TIM_OCInitStructure); //使能输出状态
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //设置TIM3的PWM输出为使能
TIM_CtrlPWMOutputs(TIM3,ENABLE);
} void NVIC_Cfg(void)
{
//定义结构体
NVIC_InitTypeDef NVIC_InitStructure; //选择中断分组1
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1); //选择TIM2的中断通道
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
//抢占式中断优先级设置为0
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = ;
//响应式中断优先级设置为0
NVIC_InitStructure.NVIC_IRQChannelSubPriority = ;
//使能中断
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure);
} void RCC_Cfg(void)
{
//定义错误状态变量
ErrorStatus HSEStartUpStatus; //将RCC寄存器重新设置为默认值
RCC_DeInit(); //打开外部高速时钟晶振
RCC_HSEConfig(RCC_HSE_ON); //等待外部高速时钟晶振工作
HSEStartUpStatus = RCC_WaitForHSEStartUp(); if(HSEStartUpStatus == SUCCESS)
{
//设置AHB时钟(HCLK)为系统时钟
RCC_HCLKConfig(RCC_SYSCLK_Div1); //设置高速AHB时钟(APB2)为HCLK时钟
RCC_PCLK2Config(RCC_HCLK_Div1); //设置低速AHB时钟(APB1)为HCLK的2分频
RCC_PCLK1Config(RCC_HCLK_Div2); //设置FLASH代码延时
FLASH_SetLatency(FLASH_Latency_2); //使能预取指缓存
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //设置PLL时钟,为HSE的9倍频 8MHz * 9 = 72MHz
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9); //使能PLL
RCC_PLLCmd(ENABLE); //等待PLL准备就绪
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //设置PLL为系统时钟源
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //判断PLL是否是系统时钟
while(RCC_GetSYSCLKSource() != 0x08); } //允许TIM2的时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE); //允许GPIO的时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC|RCC_APB2Periph_AFIO,ENABLE); } void TIM2_IRQHandler(void)
{
u16 aa=;
if(TIM_GetFlagStatus(TIM2,TIM_IT_Update)!=RESET)
{
//清除TIM2的中断待处理位
TIM_ClearITPendingBit(TIM2 , TIM_FLAG_Update); TIM_Cmd(TIM2,DISABLE);
//通过循环让灯闪烁
while (aa){
GPIO_SetBits(GPIOC,GPIO_Pin_3);
delay_ms();
GPIO_ResetBits(GPIOC,GPIO_Pin_3);
delay_ms();
aa--;
}
//使灯的状态为灭
GPIO_SetBits(GPIOC,GPIO_Pin_3);
TIM_Cmd(TIM2,ENABLE);
}
} void delay_ms(u32 i)
{
u32 temp;
SysTick->LOAD=*i; //设置重装数值, 72MHZ时
SysTick->CTRL=0X01; //使能,减到零是无动作,采用外部时钟源
SysTick->VAL=; //清零计数器
do
{
temp=SysTick->CTRL; //读取当前倒计数值
}
while((temp&0x01)&&(!(temp&(<<)))); //等待时间到达
SysTick->CTRL=; //关闭计数器
SysTick->VAL=; //清空计数器
}

在产生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

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

GPIO_PinRemapConfig(GPIO_FullRemap_TIM3, ENABLE);

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

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. Cocos2d 使用控制台打印的方法

    1.打开当前项目的win32解决方案. 2.在解决方案管理器的win32文件夹下打开main.cpp 3.增加以下代码: #define USE_WIN32_CONSOLE //以下加到入口函数 #i ...

  2. 转:Selenium Grid深入学习

    应网友要求写一个用Selenium Grid控制多系统多浏览器并行执行test case的例子. 因为我这里有两台机子,我打算这样演示: 一台机子启动一个作为主点节的hub 和 一个作为次节点的hub ...

  3. Adobe Acrobat 9 Pro 注册码

    来自百度知道,记录与此,以备后用http://zhidao.baidu.com/question/177914535.html 如果你的系统盘是C盘,那么就删除:c:/Documents and Se ...

  4. VirtualBox中CentOS通过Host-Only方式实现虚拟机主机互相访问、共享上网

    VirtualBox常用的网络配置如下: 连接方式 主机访问虚拟机 虚拟机访问主机 虚拟机访问虚拟机 虚拟机访问外网 说明 网络地址转换(NAT) 不支持 支持 不支持 支持 默认连接方式,虚拟IP, ...

  5. linux下如何修改iptables开启80端口

    linux下如何修改iptables开启80端口   最近在做本地服务器的环境,发现网站localhost能正常访问,用ip访问就访问不了,经常使用CentOS的朋友,可能会遇到和我一样的问题.开启了 ...

  6. AngularJS中的$http.post与jQuery.post的区别

    原文:http://my.oschina.net/tommyfok/blog/287748 很多时候我们需要用ajax提交post数据,angularjs与jq类似,也有封装好的post. 但是jQu ...

  7. SPOJ Count on a tree

    Count on a tree Time Limit:129MS     Memory Limit:1572864KB     64bit IO Format:%lld & %llu Subm ...

  8. The Triangle 经典DP

    题意:数塔问题 思路:1:递归.2:递推.3:记忆化搜索.<刘汝佳,第九章> #include<iostream> #include<cstdio> #includ ...

  9. Android在一个APP中通过包名或类名启动另一个APP

    开发有时需要在一个应用中启动另一个应用,比如Launcher加载所有的已安装的程序的列表,当点击图标时可以启动另一个应用.一般我们知道了另一个应用的包名和MainActivity的名字之后便可以直接通 ...

  10. java中从含反斜杠路径截取文件名的方法

    例如:获取到的文件路径为C:\Documents and Settings\Leeo\My Documents\logo.gif现在想要取得图片的名称logo.gif,我们知道反斜杠“\”是转义字符, ...