STM32输出可控数量与频率的脉冲


关于脉冲输出的控制我查阅网上资料后发现有五种方法

    1、单脉冲法,需要一个脉冲中断一次,中断次数多,影响效率

    2、一个定时器输出PWM,另一定时器进行中断计数,与方法1一样,同样需要频繁的中断

    3、用主从定时器门控方式,比较繁琐

    4、用一个定时器(从)作为另一个定时器(主)的外部时钟触发源

    5、高级定时器T1、T8的重复计数方式,RCR计数中断,看手册好像这种方式最简单,能满足一部分人要求,缺点是寄存器只有8位,最多实现255个脉冲计数输出。

    这里我使用了第四个方法。

    pulse.h

#ifndef __PUSEL_H
#define __PUSEL_H
#include "sys.h"
 
void TIM1_config(u32 Cycle);
void TIM2_config(u32 PulseNum);
void Pulse_output(u32 Cycle,u32 PulseNum);
 
 
#endif
 
    pulse.c

#include "pulse.h"
 
 
/***********************TIM1初始化函数*************************/
/****参数:****************************************************/
/******u32 Cycle用于设定计数频率(计算公式:Cycle=1Mhz/目标频率)/
/****返回值:**************************************************/
/******无*****************************************************/
void TIM1_config(u32 Cycle)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    TIM_OCInitTypeDef  TIM_OCInitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_TIM1 , ENABLE); //时钟使能
 
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;                   //TIM1_CH4 PA11
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;             //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
 
    TIM_TimeBaseStructure.TIM_Period = Cycle-1;                 //使用Cycle来控制频率(f=72/(71+1)/Cycle)  当Cycle为100时脉冲频率为10KHZ                           
    TIM_TimeBaseStructure.TIM_Prescaler =71;                    //设置用来作为TIMx时钟频率除数的预分频值                                                     
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;     //设置时钟分割:TDTS= Tck_tim            
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
    TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;            //重复计数,一定要=0!!!(高级定时器特有)
    TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);                                       
 
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;                  //选择定时器模式:TIM脉冲宽度调制模式1       
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;     //比较输出使能
    TIM_OCInitStructure.TIM_Pulse = Cycle/2-1;                        //设置待装入捕获寄存器的脉冲值(占空比:默认50%,这可也可以调节如果需要的话将它作为一个参数传入即可)                                   
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;          //输出极性       
 
    TIM_OC4Init(TIM1, &TIM_OCInitStructure);                        //使能通道4                                                 
 
    TIM_SelectMasterSlaveMode(TIM1, TIM_MasterSlaveMode_Enable);    //设置为主从模式
    TIM_SelectOutputTrigger(TIM1, TIM_TRGOSource_Update);            //选择定时器1的触发方式(使用更新事件作为触发输出)
    
 
    TIM_OC4PreloadConfig(TIM1, TIM_OCPreload_Enable);               //使能通道4预装载寄存器               
    TIM_ARRPreloadConfig(TIM1, ENABLE);                             //使能TIM1在ARR上的预装载寄存器       
}
/***********************TIM2初始化函数*************************/
/****参数:****************************************************/
/******u32 PulseNum用于设定脉冲数量****************************/
/****返回值:*************************************************/
/******无*****************************************************/
void TIM2_config(u32 PulseNum)
{
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    NVIC_InitTypeDef NVIC_InitStructure; 
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);        //使能定时器2的时钟
 
    TIM_TimeBaseStructure.TIM_Period = PulseNum-1;               //脉冲数
    TIM_TimeBaseStructure.TIM_Prescaler =0;    
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;     
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
    TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);  
 
    TIM_SelectInputTrigger(TIM2, TIM_TS_ITR0);                    //选择定时器2的输入触发源(内部触发(TIM1))
 
    TIM2->SMCR|=0x07;                                              //设置从模式寄存器(SMS[2:0]:111 外部时钟模式1) 
 
    TIM_ITConfig(TIM2,TIM_IT_Update,DISABLE);                    //更新中断失能
 
    NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;        
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;     
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
    NVIC_Init(&NVIC_InitStructure);                                //定时器2中断初始化
}
/************************脉冲输出函数**************************/
/****参数:****************************************************/
/******u32 Cycle用于设定计数频率(计算公式:Cycle=1Mhz/目标频率)/
/******u32 PulseNum用于设定输出脉冲的数量(单位:个)************/
/****返回值:**************************************************/
/******无*****************************************************/
void Pulse_output(u32 Cycle,u32 PulseNum)
{
    TIM2_config(PulseNum);                        //设置脉冲数量
    TIM_Cmd(TIM2, ENABLE);                        //使能TIM2(从定时器)
    TIM_ClearITPendingBit(TIM2,TIM_IT_Update);    //清除中断标志位
    TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);    //使能更新中断
    TIM1_config(Cycle);                            //使能定时器1(主定时器)
    
    TIM_Cmd(TIM1, ENABLE);                        //使能定时器1
    TIM_CtrlPWMOutputs(TIM1, ENABLE);           //高级定时器一定要加上,主输出使能
}
 
 
 
/********************定时器2的中断服务函数**********************/
/****参数:****************************************************/
/******u32 PulseNum用于设定脉冲数量****************************/
/****返回值:*************************************************/
/******无*****************************************************/
/****函数说明:************************************************/
/*当TIM的CNT寄存器的值到达设定的Update值会触发更新中断,此时设定的脉冲数已输出完毕,关闭TIM1和TIM2*/
void TIM2_IRQHandler(void) 

    if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)     //TIM_IT_Update
    { 
        TIM_ClearITPendingBit(TIM2, TIM_IT_Update);     // 清除中断标志位 
        TIM_CtrlPWMOutputs(TIM1, DISABLE);              //主输出使能
        TIM_Cmd(TIM1, DISABLE);                         //关闭定时器 
        TIM_Cmd(TIM2, DISABLE);                         //关闭定时器 
        TIM_ITConfig(TIM2, TIM_IT_Update, DISABLE);     //关闭TIM2更新中断
        
    } 

 
    main.c

#include "sys.h"
#include "delay.h"
#include "led.h"
#include "usart.h"
#include "pulse.h"
 
 
int main()
{
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);// 设置中断优先级分组2
    delay_init();             //延时函数初始化    
    uart_init(9600);                 //9600     
    led_init();
    while(1)
    {
        LED=1;
        delay_ms(500);
        LED=0;
        delay_ms(500);
        Pulse_output(100,3200);
    }
}
 
    脉冲频率10KHz,每经过1s会输出3200个脉冲,步进电机会转1周。
————————————————

具体操作要求如下: 以给定频率输出脉冲脉冲数无限制 以给定频率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公司能够能够简化器件的使用。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值