专注电子技术学习与研究
当前位置:单片机教程网 >> MCU设计实例 >> 浏览文章

基于Mega128编码器控制步进电机的平衡系统

作者:佚名   来源:本站原创   点击数:  更新时间:2011年11月10日   【字体:

    提起编码器,可能大家并不陌生,因为这东西真的很常用,现在的主流编码器一般精度都是比较高的,基本上都是基于光栅的。毕竟用硬件用电刷做到512精度以上是很困难的,而且成本也是很高的,这里我就不多说什么了。

    编码器一般共有三个主通道,每个主通道有分为两个分支;一个VCC,一个GND,一条屏蔽线。前两个通道一般是比较精确的脉冲信号,且之间有四分之一左右的相位差用来判断正反转的,第三通道基本上是旋转一周才会有一个脉冲信号的那种。

 
    提到步进电机,就一定要有一个合适的电机驱动,个人是比较喜欢用L298n这款芯片的,因为它价格低,操作比较简单。
 
    对于这个系统,我是用128的外部中断的下降沿触发方式来捕捉编码器的脉冲的,硬件连接方面电机驱动和主控芯片一定要注意地线的连接。
 
    下面是程序的完整代码下载地址: http://www.51hei.com/f/bmma.rar         

 
这里是delay.h
====================================================================
//根据CPU时钟频率选择
#ifndef F_CPU
//#define F_CPU 7372800
//#define F_CPU 8000000
//#define F_CPU 11059200
//#define F_CPU 12000000
#define F_CPU 16000000
#endif



//------------------------------------------------------------------------------
//1、2、3、5和10us的精确延时,用于需要精确时间的场合,比如DS18B20
//------------------------------------------------------------------------------
#if   F_CPU == 7372800
#define DELAY_1US NOP();NOP();NOP();NOP();NOP();NOP();NOP()
#define DELAY_2US DELAY_1US;DELAY_1US
#define DELAY_3US DELAY_1US;DELAY_1US;DELAY_1US
#define DELAY_5US DELAY_2US;DELAY_3US
#define DELAY_10US DELAY_5US;DELAY_5US

#elif F_CPU == 8000000
#define DELAY_1US NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP()
#define DELAY_2US DELAY_1US;DELAY_1US
#define DELAY_3US DELAY_1US;DELAY_1US;DELAY_1US
#define DELAY_5US DELAY_2US;DELAY_3US
#define DELAY_10US DELAY_5US;DELAY_5US

#elif F_CPU == 11059200
#define DELAY_1US NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP()
#define DELAY_2US DELAY_1US;DELAY_1US
#define DELAY_3US DELAY_1US;DELAY_1US;DELAY_1US
#define DELAY_5US DELAY_2US;DELAY_3US
#define DELAY_10US DELAY_5US;DELAY_5US

#elif F_CPU == 12000000
#define DELAY_1US NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP()
#define DELAY_2US DELAY_1US;DELAY_1US
#define DELAY_3US DELAY_1US;DELAY_1US;DELAY_1US
#define DELAY_5US DELAY_2US;DELAY_3US
#define DELAY_10US DELAY_5US;DELAY_5US

#elif F_CPU == 16000000
#define DELAY_1US NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP();NOP()
#define DELAY_2US DELAY_1US;DELAY_1US
#define DELAY_3US DELAY_1US;DELAY_1US;DELAY_1US
#define DELAY_5US DELAY_2US;DELAY_3US
#define DELAY_10US DELAY_5US;DELAY_5US
#endif


//------------------------------------------------------------------------------
//函数声明
//------------------------------------------------------------------------------

void delay_nus(unsigned int);//<10us时误差较大,用于无须精确延时的场合

void delay_nms(unsigned int);


#endif//__DELAY_H


====================================================================
 这里是delay.c
====================================================================
//|文件名称|delay.c
//|--------|--------------------------------------------------------------------
//|描    述|延时文件
//|--------|--------------------------------------------------------------------
//|说    明|delay_us(unsigned int time)用于不需精确定时的场合,<10us时误差较大
//|        |若要精确定时用delay.h里的宏
//|--------|--------------------------------------------------------------------
//|调用文件|delay.h
//|--------|--------------------------------------------------------------------
//|作    者|
//|--------|--------------------------------------------------------------------
//|版    本|
//|--------|--------------------------------------------------------------------
//|时    间|
//|--------|--------------------------------------------------------------------
//|E-mail  |
//|--------|--------------------------------------------------------------------
//|开发环境|ICCAVR6.31
//==============================================================================

#include "delay.h"

#if F_CPU == 7372800
void delay_nus(unsigned int time)
{
  unsigned int i;
  for(i=0;i<time;i++)
  {
    NOP();
  }
}

void delay_nms(unsigned int time)
{
  unsigned int i,j;
  for(i=0;i<time;i++)
  {
    for(j=0;j<1045;j++);
  }
}

#elif F_CPU == 8000000
void delay_nus(unsigned int time)
{
  unsigned int i;
  for(i=0;i<time;i++)
  {
    NOP();NOP();
  }
}

void delay_nms(unsigned int time)
{
  unsigned int i,j;
  for(i=0;i<time;i++)
  {
    for(j=0;j<1178;j++);
  }
}

#elif F_CPU == 11059200
void delay_nus(unsigned int time)
{
  unsigned int i;
  for(i=0;i<time;i++)
  {
    NOP();NOP();NOP();NOP();NOP();
  }
}

void delay_nms(unsigned int time)
{
  unsigned int i,j;
  for(i=0;i<time;i++)
  {
    for(j=0;j<1578;j++);
  }
}

#elif F_CPU == 12000000
void delay_nus(unsigned int time)
{
  unsigned int i;
  for(i=0;i<time;i++)
  {
    NOP();NOP();NOP();NOP();NOP();
  }
}

void delay_nms(unsigned int time)
{
  unsigned int i,j;
  for(i=0;i<time;i++)
  {
    for(j=0;j<1713;j++);
  }
}

#elif F_CPU == 16000000
void delay_us(unsigned int time)
{
  unsigned int i;
  for(i=0;i<time;i++)
  {
    NOP();NOP();NOP();NOP();NOP();NOP();
NOP();NOP();NOP();NOP();
  }
}

void delay_ms(unsigned int time)
{
  unsigned int i,j;
  for(i=0;i<time;i++)
  {
    for(j=0;j<2288;j++);
  }
}

#endif
====================================================================
 这里是dianji.h
====================================================================
#ifndef __DIANJI_H
#define __DIANJI_H

#define EN_1 PORTC|=(1<<0)
#define EN_0 PORTC&=~(1<<0)

void one_circle(void);
void circle(unsigned int xcircle);

#endif

====================================================================
 这里是dianji.c
====================================================================
#include"dianji.h"
#include"delay.h"
#include<iom128v.h>


/*步进电机旋转一周子函数*/
void one_circle(void)
{
  unsigned int i;
for(i=0;i<3200;i++)
{
EN_1;
delay_us(100);
EN_0;
delay_us(100);
}
}
/*步进电机根据编码器计算的数值旋转任意角度子函数*/
void circle(unsigned int xcircle)
{
  unsigned int i,j;
for(j=0;j<xcircle;j++)
{
for(i=0;i<3;i++)
{
   EN_1;
delay_us(100);
EN_0;
delay_us(100);
}
}
}
====================================================================
 这里是caiji.h
====================================================================
#ifndef __CAIJI_H
#define __CAIJI_H

 //宏定义595
#define SCK_0 PORTB&=~(1<<PB4)
#define SCK_1 PORTB|=(1<<PB4)
#define LCK_0 PORTB&=~(1<<PB5)
#define LCK_1 PORTB|=(1<<PB5)
#define SDI_0 PORTB&=~(1<<PB6)
#define SDI_1 PORTB|=(1<<PB6) 

#define DIR_0 PORTC&=~(1<<1)
#define DIR_1 PORTC|=(1<<1)


void init_xianshi(void);
void send_595(unsigned char dat);
void jiaoduzhuanhuan(void);


#endif

====================================================================
 这里是caiji.c
====================================================================
#include"caiji.h"
#include"dianji.h"
#include<iom128v.h>
#include<macros.h>
#include<math.h>

//一些变量的定义
int k=0;
unsigned int x=1,d=0,c=0,cir=0;
unsigned int a,aa,q,angle=0,p=0;
//595(数码管)显示数组
unsigned char led[]=
{
0x3F, // 0
0x06, // 1
0x5B, // 2
0x4F, // 3
0x66, // 4
0x6D, // 5
0x7D, // 6
0x07, // 7
0x7F, // 8
0x6F, // 9
};
//外部中断0~3向量号码
#pragma interrupt_handler Zhongduan_0:2
#pragma interrupt_handler Zhongduan_1:3
#pragma interrupt_handler Zhongduan_2:4

//端口输入输出初始化
void init_xianshi(void)
{
DDRA=0XFF;
DDRB=0XFF;//595控制口
DDRC=0XFF;//正反转提示LED
PORTC=0XFF;//LED初始化(暗)
}


//595串入一字节并处发送一字节子函数
void send_595(unsigned char dat)
{
   unsigned char i;
   LCK_0;
   SDI_1;
   SCK_0;
   //上面的三条语句为了初始化端口状态
   for(i=0;i<8;i++)
   {
    LCK_0;//时钟线拉低
 
    if(dat&0x80)
        SDI_1;
     else SDI_0;
     dat=dat<<1;
   delay_us(100);  
    LCK_1;          //时钟线拉高将数据读入595的移位寄存器    
    delay_us(100); 
   }
   SCK_1; //发送数据到并行端口
   SCK_0;   
}

void jiaoduzhuanhuan(void)
{
  init_xianshi();
  EIMSK=0X0F;
  EICRA=0XAA; 
 while(1)
  { 
 if(a!=aa) /*如果有角度变化就执行下面的程序*/
 {
angle++;
 
if(!(PING&0X04))
cir++; 
while(cir)
{
if(!k)
  {
   delay_ms(800);
 if(d<=3)
     one_circle();
 d++;
  }
 }  
 
 if(!(PING&0X02))
 {
 while(PING&0X01)
 {
   DIR_0;
  circle(1);
   p++;
   while(p==angle);
 }
  while(PING&0X01);
 if(q=-1)//正转
 DIR_1;
 circle(1);

 if(q==1)  //反转
 DIR_1;
 }
 
  aa=a;
 

  if(!x)
  {
PORTC=0X00;
  }
  }
  /*下面为数码管显示编码器当前计数的数值*/
  PORTA=0X08; 
  send_595(0x00);
  send_595(led[abs(k)%10]);
  PORTA=0X01; 
  send_595(0x00);
  send_595(led[(abs(k)%100)/10]);
  PORTA=0X02; 
  send_595(0x00);
  send_595(led[(abs(k)%1000)/100]);
  PORTA=0X04; 
  send_595(0x00);
  send_595(led[abs(k)/1000]);
SEI(); /*使能中断*/
  } 
}

 void Zhongduan_0(void)
 {
  CLI();
  x=0;
 ////////////////// 
  if(PIND&0X02)
  q=-1;
  if(!(PIND&0X02))
  q=1;
  ////////////////
  k=k+q;
  SEI();
 }

 void Zhongduan_1(void)
 {
 CLI();

 if(PIND&0X01)
 q=1;
 k=k+q;
 if(abs(k)==10000)
 {
   k=0;
 }
 a=k;
 //EIMSK=0X0E;
 SEI();
 }
 void Zhongduan_2(void)
 {
  CLI();
  k=0;
  SEI();
  }
====================================================================
 这里是main.c
====================================================================
#include<iom128v.h>
#include<macros.h>
#include"delay.h"
#include"caiji.h"
#include"dianji.h"

void init(void)
{
  DDRC=0X03;
PORTC=0X00;
}

void main(void)
{
  init();     //初始化子函数
  jiaoduzhuanhuan(); //主程序的实现
}
====================================================================
程序中涉及到了一些嵌套,如果不明白的话可以给我留言哦!^_^
关闭窗口

相关文章