stm32 精确个数高速脉冲输出

/****************************
**TIM2的通道1使用单脉冲模式
**TIM3使用门控模式用来输出PWM
**
****************************/
//TIM2per:重装值
//Compare1:比较捕获1的预装载值
void Motor_Init(u16 TIM2per, u16 TIM3per, u16 TIM3Compare1)
{
  TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
  TIM_OCInitTypeDef TIM_OCInitStruct;
  GPIO_InitTypeDef GPIO_InitStruct;
  
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);//使能定时器2的时钟
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);//使能定时器3的时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//使能GPIOA时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);//使能复用IO时钟
  
http://write.blog.csdn.net/postedit  
 
 
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽输出
  GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_6;//PA0
  GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOA,&GPIO_InitStruct);
  
  //TIM2工作在单脉冲下
  TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;//没有时钟分割
  TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
  TIM_TimeBaseInitStruct.TIM_Prescaler = 7200;//预分频值,每100us计数一次
  TIM_TimeBaseInitStruct.TIM_Period = TIM2per;//重装值
  TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);
  
  TIM_SelectOnePulseMode(TIM2,TIM_OPMode_Single);//设置TIM2在单脉冲模式,且是单一的脉冲,在下一个更新事件停止
  TIM_OC1PreloadConfig(TIM2,TIM_OCPreload_Enable);//使能定时器2的通道1预装载寄存器
  TIM_SelectOutputTrigger(TIM2,TIM_TRGOSource_OC1Ref);
  
  TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM2;//在向上计数时,一旦TIMx_CNT<TIMx_CCR1时通道1为无效电平,否则为有效电平
  TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;//OC1输出使能
  TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//有效电平为高
  TIM_OCInitStruct.TIM_Pulse = 1;//比较捕获1的预装载值
  TIM_OC1Init(TIM2,&TIM_OCInitStruct);
  
  TIM_Cmd(TIM2,DISABLE);//先不使能能TIM2
  
  
  //TIM3工作在从模式的门控模式下的PWM输出模式
  TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;//没有时钟分割
  TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
  TIM_TimeBaseInitStruct.TIM_Prescaler = 720;//预分频值,10us计数一次
  TIM_TimeBaseInitStruct.TIM_Period = TIM3per;//重装值
  TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStruct);
  
  TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Gated);//TIM3为门控模式
  TIM_SelectMasterSlaveMode(TIM3,TIM_MasterSlaveMode_Enable);//使能TIM3的主从模式
  TIM_SelectInputTrigger(TIM3,TIM_TS_ITR1);//内部触发,从TIM2触发
  
  TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM2;//在向上计数时,一旦TIMx_CNT<TIMx_CCR1时通道1为无效电平,否则为有效电平
  TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;//OC1输出使能
  TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//有效电平为高
  TIM_OCInitStruct.TIM_Pulse = TIM3Compare1;//比较捕获1的预装载值
  TIM_OC1Init(TIM3,&TIM_OCInitStruct);
  
  TIM_Cmd(TIM3,ENABLE);//使能TIM3
}

//设置PWM的输出
//Cycle:为周期,单位(us)
//Pulse_Num:为脉冲个数(小于3200)
void TIM2_TIM3_PWM(u16 Cycle, u16 Pulse_Num)
{
  u16 TIM3per = 0;
  u32 Time = 0;
  //改变TIM3的重装值改变脉冲频率这里设置脉冲占空比为50%
  //改变TIM2的预分频器和重装值改变脉冲个数
    
  Time = Cycle * Pulse_Num;
  Time /= 100;              //预分频为7200,100us计数一次
  TIM3per = Cycle/10;       //预分频为720,10us计数一次
  
  TIM_SetAutoreload(TIM2, Time+1);//设置TIM2的重装值
  TIM_SetAutoreload(TIM3, TIM3per-1);//设置TIM3的重装值
  TIM_SetCompare1(TIM3,TIM3per/2);//设置占空比为50%
  TIM_Cmd(TIM2,ENABLE);//使能TIM2
}

具体操作要求如下: 以给定频率输出脉冲脉冲数无限制 以给定频率f、输出n个脉冲 相对定位 相对定位+绝对定位 脉冲输出PORTA.0 方向信号输出PORTB.5 模仿PLC定位指令 可以作为简易运动控制器控制伺服电机 发脉冲两种目的 1)速度控制 2)位置控制 速度控制目的和模拟量一样,没有什么需要关注的地方 发送脉冲方式为PWM,速率稳定而且资源占用少 stm32位置控制需要获得发送的脉冲数,有下面4种手段 1)每发送一个脉冲,做一次中断计数 2)根据发送的频率×发送的时间,获得脉冲数量,对于变速的脉冲,可以累计积分的方法来获得总脉冲 3)一个定时器作为主发送脉冲,另外一个定时器作为从,对发送的脉冲计数 4)使用DMA方式,例如共发送1000个脉冲,那么定义u16 per[1001],每发送一个脉冲,dma会从数组中更新下一个占空比字,数组最后一个字为0,表示停发脉冲 上面4种方法的用途和特点 1)对于低速率脉冲比较好,可以说低速发脉冲的首选,例如10Khz以下的,否则中断占用太多的cpu,这种方法要注意将中断优先级提高,否则会丢计数, 2)用作定时的计时精确高,可以允许有脉冲计数丢失的情况 3)主从方式,需额外的定时器来计数,例如tim1发脉冲 tim2计数,最方便的方式,无论高速低速即可,同时占用cpu最低,只是要占用多一个定时器 4)DMA方式也算是一个很确定的方式,不会丢失脉冲,但是高速的时候,会较多的占用内部总线同时会使用一个多余的DMA控制器,而且有个缺点,就是使用起来比较复杂,没有达到KISS原则 个人推荐方式,低速时中断方式,如果不知高速还是低速,则使用主从方式。具体的方式需要根据资源和需求来确定。 stm32定时器算是比较复杂的器件,而且用户要较多的介入底层,希望将来st公司能够能够简化器件的使用。
### 回答1: STM32是一款常用的微控制器系列,其具备强大的PWM功能和DMA技术,可以实现高速输出。 PWM(脉宽调制)是一种常用的控制信号技术,它通过调整信号的占空比来控制输出电平,常用于驱动电机、控制LED亮度等应用。STM32系列微控制器内置有多个PWM输出通道,并且可以通过寄存器配置和编程实现灵活的PWM控制。 DMA(直接存储器访问)是一种用于数据传输的硬件技术,它可以实现数据的高速、自动传输,减轻CPU的负载,提高系统的效率。STM32系列微控制器配备有多个DMA通道,可以与PWM功能相结合,实现高速的PWM信号输出。 通过配置DMA通道,可以将PWM输出数据直接从内存传输到PWM寄存器,从而实现高速的PWM输出。这种方式避免了CPU的干预,大大提高了输出速度和精确度,适用于对PWM输出要求较高的应用。 使用STM32的PWM和DMA功能,可以实现高速精确的PWM输出,适用于各种应用领域,如工业控制、机器人、无线通信等。同时,STM32系列还提供了丰富的开发工具和软件库,方便开发者进行PWM和DMA的配置和编程。 总之,STM32的PWM和DMA功能结合,可以实现高速输出,满足各种应用需求,并且提高了系统的性能和效率。 ### 回答2: STM32是一款广泛应用于嵌入式系统开发的微控制器系列,其中的STM32PWM模块可以实现高速脉冲宽度调制(PWM)信号输出。而使用DMA技术可以进一步提高PWM输出的效率和精度。 在STM32中,PWM输出可以通过先配置TIM定时器的相关寄存器来实现。通过与DMA控制器配合使用,可以将要输出的PWM数据存储在一个内存缓冲区中,然后通过DMA通道将数据传输到TIM的寄存器中,从而实现高速输出。 使用DMA的主要好处是减少了CPU的负担,从而使CPU能够更多地处理其他任务。DMA通过直接在外设和内存之间传输数据,无需CPU进行介入,因此能够以更高的速度完成数据传输。这对于实现高速PWM输出非常有帮助。 此外,DMA还可以实现多通道的PWM输出,以满足系统对多个PWM信号的需求。通过配置多个DMA通道,每个通道负责一个PWM信号的传输,可以同时输出多个PWM信号,从而提高系统的扩展能力和灵活性。 总之,使用STM32的PWM模块配合DMA技术,可以实现高速、高效、多通道的PWM信号输出,适用于各种需要PWM输出的应用领域,如电机控制、LED调光、音频处理等。 ### 回答3: STM32是一种微控制器系列,通过配置PWM(脉冲宽度调制)和DMA(直接内存访问)功能,可以实现高速输出。 PWM是一种常见的数字信号产生方式,通过控制脉冲的宽度,可以间接控制输出的电平。在STM32微控制器中,可以通过配置定时器的输出比较模式生成PWM信号。对于高速PWM输出,可以选择适当的定时器和精确的计数值,以实现较高的输出频率。 DMA是一种优化的数据传输方式,通过直接访问内存,将数据从一个地方传输到另一个地方,减少了CPU的负担。在STM32中,可以通过配置DMA通道,将要输出的PWM数据存储在内存中,然后通过DMA传输到定时器输出比较寄存器中,从而实现高速的PWM输出。 通过结合PWM和DMA功能,可以实现高速的PWM输出。具体步骤如下: 1. 初始化定时器:选择合适的定时器和计数值,并设置输出比较模式为PWM模式。 2. 初始化GPIO:将定时器的输出引脚与特定的GPIO引脚相连。 3. 配置DMA通道:选择合适的DMA通道,并设置传输方向为从内存到外设,传输数据源为存储PWM数据的内存地址,传输目的地为定时器的输出比较寄存器地址,设置传输长度为PWM数据的长度。 4. 配置DMA中断:如果需要在DMA传输完成后进行某些操作,可以配置DMA传输完成的中断。 5. 设置PWM数据:将要输出的PWM数据存储在内存中。 6. 启动DMA传输:通过启动DMA传输,将PWM数据从内存传输到定时器的输出比较寄存器中。 7. 开始PWM输出:启动定时器,开始输出PWM信号。 通过以上步骤,可以实现STM32的PWM DMA高速输出。通过合理选择定时器和DMA配置,可以实现较高的PWM输出频率,并减少CPU的负担,提高系统效率。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值