stm32控制电机

一、总体思路

      使用端口GPIOA来连接电机,所以给GPIOA编程就可以控制电机。使用系统时钟SysTick来周期性的给电机发送脉冲。用四个按钮来控制需要发送脉冲的个数,每个按钮被按下就设置给电机发送脉冲的个数,如果上一次给电机发送的脉冲没有发送完成,这次按钮发送的脉冲将不被响应。
 

二、GPIOA端口的设置

      由于需要控制两个电机,所以将GPIOA端口的1,2,3号引脚与电机0相连(分别控制电机的使能,旋转方向和脉冲),GPIOA的4,5,6号引脚与电机1相连。具体对端口的初始化代码为:
复制代码
GPIO_InitTypeDef GPIO_InitStruct;  
//开启电机0外设时钟
DJ_EnablePeriphClock_0();
//初始化电机0
GPIO_InitStruct.GPIO_Pin = DJ_EN_0 | DJ_DR_0 | DJ_MC_0;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(DJ_GPIO_0, &GPIO_InitStruct);
//设置电机0的初始化状态
DJ_DisEnable(DJ_GPIO_0, DJ_EN_0);  //关闭电机0
复制代码
上面的代码是对与电机0连接的引脚的初始化,电机1的初始化是一样的,只是引脚不同了。从上面的代码可以看到引脚的输出模式是推挽的(为了做Debug),实际应该使用开漏的,由于我们要给电机输入5V的高电平,所以我们应该在外部接一个上拉电阻,电源为5V。
 

三、SysTick设置

     SysTick是一个系统定时器,系统的滴答是可以配置的,在控制电机的程序中我们将系统滴答设置为100us,理论上可以将系统滴答设置为1/72000000s,由于系统的时钟为72MHz。每个脉冲间隔为5个滴答。也就是说每隔500us发送一个脉冲,脉冲周期为1ms。
     设置系统滴答通过宏:
     #define TICK 10000  //100us一个滴答
     实际的配置是通过下面代码:
    SysTick_Config(SystemCoreClock / TICK);
    为了实现每隔5个系统滴答发送一个脉冲,定义了两个全局变量TimingDelay和TimingLoad ,可以通过函数Timer来设置这个变量的值:
      void Timer(__IO uint32_t nTime)
      {
        TimingDelay = nTime;
        TimingLoad = nTime;
      }
     TimingDelay表示当前距离发送下一个脉冲还需要等待的滴答数,TimingLoad 表示发送脉冲的间隔,如果每个脉冲间隔为5个滴答,则TimingLoad =5。 这样在系统时钟的每次中断代码中将TimingDelay减1,当TimingDelay为0时就向电机发送脉冲(将对应电机脉冲的引脚的值变反就可以了),然后重新将TimingLoad赋值给TimingDelay来准备下一个脉冲的发送。具体代码如下:
复制代码
if (TimingDelay)
        TimingDelay--;
    else
    {
        TimingDelay = TimingLoad; 
        if (dj_GetMc())  //判断是否还需要发送脉冲
        {
            printfd("\r\nsend %dth pulse, %d", dj_GetMc(), 1 - GPIO_ReadOutputDataBit (DJ_GPIO_0, DJ_MC_0));  //用于调试
            DJ_IO(1 - GPIO_ReadOutputDataBit (DJ_GPIO_0, DJ_MC_0), DJ_GPIO_0, DJ_MC_0);   //用于发送脉冲
            dj_DesMc();  
        }
        else
            SysTick_Shutdown();  //关闭系统时钟
    }
复制代码
      dj_GetMc函数和dj_DesMc函数分别获得全局变量dj_McCount的值和对该全局变量减1,这个全局变量代表总共需要发送几个脉冲。正如开头说的,按下一个键就设置需要发送的脉冲数,当脉冲发完了,就关闭系统定时器。
 

四、按键设置

     使用ARM板自带的WAKEUP,TAMPER,USER1,USER2四个键来配置需要发送脉冲的个数,它们分别配置为需要发送2,20,200,2000个脉冲。我们通过中断的方式来检查哪个键被按下了。配置将四个按键与EXTI相连,让EXTI产生中断到中断控制器NVIC。首先需要配置NVIC的抢占优先级和响应优先级,直接调用库函数就可以了:

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

然后是配置按键与EXTI相连,具体配置代码如下:
复制代码
static void NVIC_SetVector( IRQn_Type IRQn, uint8_t PreemptionPriority, uint8_t SubPriority)
{
     NVIC_InitTypeDef NVIC_InitStruct;     
     NVIC_InitStruct.NVIC_IRQChannel = IRQn;//EXTI0_IRQn | EXTI9_5_IRQn | EXTI3_IRQn |  EXTI15_10_IRQn;
     NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
     NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = PreemptionPriority;
     NVIC_InitStruct.NVIC_IRQChannelSubPriority = SubPriority;
     NVIC_Init(&NVIC_InitStruct);
}
//WAKEUP键
static void EXTI_PA0_Config(void)
{
     GPIO_InitTypeDef  GPIO_InitStruct;
     EXTI_InitTypeDef  EXTI_InitStruct;
     RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO |  RCC_APB2Periph_GPIOA, ENABLE);
     NVIC_SetVector(EXTI0_IRQn, 0, 0);  //配置NVIC
     GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IPU;
     GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0;
     GPIO_Init(GPIOA, &GPIO_InitStruct);     
     GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0);   //配置GPIOA_Pin0为EXTI0线
     EXTI_InitStruct.EXTI_Line = EXTI_Line0;
     EXTI_InitStruct.EXTI_LineCmd = ENABLE;
     EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;   //中断模式
     EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising ;    //上升沿触发
     EXTI_Init(&EXTI_InitStruct);
}
复制代码

     上面的代码显示了配置WAKEUP键,其它键也是同样的配置。当按下一个键时,对应的中断响应函数就会执行,我们在响应函数中判断全局变量dj_McCount是否为0,如果不为0,说明上一次按键的脉冲还没有发生完成,则直接退出中断响应函数;如果为0,说明当前没有在发生脉冲,则设置dj_McCount为2,开启系统定时器来发送脉冲。代码如下:
复制代码
//WAKEUP
void EXTI0_IRQHandler(void)
{    
    if (EXTI_GetFlagStatus(EXTI_Line0) != RESET)  //看是否产生了EXTI_Line0中断
    {
        printfd("\r\nexti0");
        if (dj_GetMc() == 0)
        {
            dj_SetMc(1);
            SysTick_Startup();
        }
        EXTI_ClearFlag(EXTI_Line0);  //清除中断标志位
    }
}
复制代码

 

源代码:http://pan.baidu.com/s/1uy6PK

posted @ 2017-11-16 19:18  专注it  阅读(3022)  评论(0编辑  收藏  举报