搜索
bottom↓
回复: 48

我公司产品成熟使用的AVR电机控制算法

[复制链接]

出0入0汤圆

发表于 2010-4-7 20:44:18 | 显示全部楼层 |阅读模式
//***************************FileName:Motor.C************************
//***************************ICCAVR6.30编译**************************
#include <io8535v.h>
#include <macros.h>
#include <eeprom.h>

//***************************全局变量定义**************************//
#define U1 5
char Data[10];               //接收数据数组
int adress;                  //数据地址
int a;                       //采样时间次数
float t;                     //采样时间.s
int itime;                   //给定采样时间的次数
int Ek=0;                    //本次偏差
int Ek_1=0;                  //上次偏差
int Ek_2=0;                  //上上次偏差
char flage=0;                //监控标志
union chang1                 //float数据与char数据转换
{char c[4];   
float x;
}floatxin;
union chang2                 //int数据与char数据转换           
{char c[2];
int x;
}intxin;

//**************************串口数据发送函数***********************//
//说明:串口数据发送采用查询方式,每一帧发送10个字节
void Putdata(char data[10])
{int i;
for(i=0;i<=9;i++)
{while(!(USR&(1<<UDRE)));
  UDR=data;}
}

//**************************读AD转换结果***************************//
int GetOutputADC()
{int temp;
temp=ADCH;
temp=temp<<8;
return(temp+ADCL);
}

//***************************电机加速函数**************************//
void Upspeed()
{EEPROMReadBytes(0x0010,intxin.c,2);
intxin.c[0]+=Data[6];
intxin.c[1]+=Data[7];
EEPROMWriteBytes(0x0010,intxin.c,2);
}

//***************************电机减速函数**************************//
void Downspeed()
{
  EEPROMReadBytes(0x0010,intxin.c,2);
  intxin.c[0]-=Data[6];
  intxin.c[1]-=Data[7];
  EEPROMWriteBytes(0x0010,intxin.c,2);
}

//***************************PID调节函数***************************//
void PIDB()
{int y;           //本次速度值
float u;         //电压差值
int z;           //输出增量
int t;           //采样时间
char temp1;      //暂存
int i;
int speed;       //速度设定
float k[3];      //PID参数记录
for (i=0;i<=2;i++)
  {EEPROMReadBytes(i*4,floatxin.c,4);
   k=floatxin.x;}
  EEPROMReadBytes(0x0010,intxin.c,1);   //读EEPROM
  speed=intxin.x;
  t=itime*0.0001275;
  y = TCNT1;
  Ek=y-speed;
  u=k[0]*((Ek-Ek_1)+(t/k[1])*Ek+(k[2]/t)*(Ek-2*Ek_1+Ek_2));
  z=u/U1*0xFF;
  temp1=OCR2;
  if(flage==1)                            //监控状态
  {Data[4]=0xD0;
   Data[5]=temp1;
   Data[6]=y>>8;
   Data[7]=y;
   Putdata(Data);}
  temp1=temp1+z;
  if(temp1<=0)                           //结果小于0时输出0
   temp1 = 0x00;
  if(temp1>=0xF0)                        //结果大于0xF0时输出0xF0
   temp1 = 0xF0;
  OCR2=temp1;
  Ek_2 = Ek_1;
  Ek_1 = Ek;
  TCNT1 = 0x0000;                          //计数器清零
}

//***************************串口初始化函数************************//                  
void USARTInit(int baudrate)   
{UCR = (1<<TXEN)|(1<<RXCIE)|(1<<RXEN);    //设置收发使能,接受中断允许
  UBRR = baudrate;                        //设置波特率寄存器
}

//***************************读EEPROM函数**************************//
void Eepromdata(void)
{if((adress==0x0000)||(adress==0x0004)||(adress==0x0008)) //地址判断
EEPROMReadBytes(adress,&Data[4],4);
else
EEPROMReadBytes(adress,&Data[6],2);
switch(adress)
{case 0x0000:Data[3]=0xE4;break;
  case 0x0004:Data[3]=0xE5;break;
  case 0x0008:Data[3]=0xE6;break;
  case 0x000C:Data[3]=0xE7;break;
  case 0x000E:Data[3]=0xE8;break;
  case 0x0010:Data[3]=0xE9;break;
  default :break;}
}

//***************************读数据函数****************************//
void readdata(char data)
{int trandata;
char tempdata;
if(data==0xE)
Eepromdata();
else
{switch(data)
  {case 1:trandata=TCNT1;break; //读TCNT1
   case 2:trandata=(int)(tempdata=TCNT0);break;//读TCNT0
   case 3:trandata=GetOutputADC();break;//读A/D转换结果
   default :break;}
Data[6]=(char)trandata;//数据低八位
Data[7]=(char)(trandata>>8);//数据高八位
}
Putdata(Data);
}

//***************************命令函数******************************//
void order(void)
{switch (Data[4])
{case 0xC0:PORTB=PORTB|0x04;break;    //停止
  case 0xC1:PORTB=PORTB&0xFB;break;    //启动
  case 0xC2:                           //正转
           {PORTB=PORTB&0xFC|0x02;
                    PORTD=PORTD&0x3F|0x40;
                        break;}
  case 0xC3:                           //反转
           {PORTB=PORTB&0xFC|0x01;
                    PORTD=PORTD&0x3F|0x80;
                        break;}
  case 0xC4:Upspeed();break;           //加速
  case 0xC5:Downspeed();break;         //减速
  case 0xC6:                           //制动
           {PORTB=PORTB|0x03;
                    PORTD=PORTD&0x3F;
                        break;}
  case 0xC8:flage=1;break;             //监控
  case 0xC9:flage=0;break;             //退出监控
  default :break;
}
}


//***************************数据处理函数**************************//
void Datadisposal(void)
{char operation;
int temp;
operation=Data[3];
temp=Data[4];
temp=temp<<8;
adress=temp+Data[5];
switch(operation&0x0F)
{case 0:readdata(operation>>4);break;
  case 1:EEPROMWriteBytes(adress,&Data[6],2);break;
  case 2:order();break;
  default :break;
}
}

//***************************主函数********************************//
void main()
{USARTInit(25);              //初始化串口            
TCNT1=00;                   //定时器1初始化
TCCR1B=0x06;
DDRB=0x0F;                  //I/O口初始化
DDRD=0xC0;     
TIMSK=0x40;                 //中断初始化
OCR2 = 0x0F;                //PWM初始化
TCCR2=0x72;
PORTB=PORTB&0xFC|0x01;
PORTD=PORTD&0x3F|0x80;
SEI();
do{} while(1);              //等待中断
}

//***************************串行接收中断服务程序******************//
#pragma interrupt_handler UART_RXC:12
void UART_RXC(void)
{char i=0;
CLI();
Data[0]=UDR;
if(Data[0]==0x55)           //起始位检测
{do
  {while(!(USR&(1<<RXC)));
   i++;
   Data=UDR;
   if(i==9)
   break;} while(1);
  if((Data[8]==0xFE)&&(Data[9]==0xFF))  //检测结束位
  Datadisposal();}
SEI();
}

//***************************定时器2中断服务程序*******************//
#pragma interrupt_handler TIM2_OVF:5
void TIM2_OVF(void)   
{CLI();
EEPROMReadBytes(0x000C,&itime,2);//读采样次数
a++;
if(a==itime)                //时间窗口判断,
  {a = 0;
   PIDB();}
  SEI();
}
点击此处下载 ourdev_544343.rar(文件大小:48K) (原文件名:我公司产品成熟使用的AVR电机控制算法.rar)

阿莫论坛20周年了!感谢大家的支持与爱护!!

知道什么是神吗?其实神本来也是人,只不过神做了人做不到的事情 所以才成了神。 (头文字D, 杜汶泽)

出0入0汤圆

发表于 2010-4-7 20:48:02 | 显示全部楼层
mark,说不定什么时候用到

出0入0汤圆

发表于 2010-4-7 20:50:48 | 显示全部楼层
留下了

出0入0汤圆

发表于 2010-4-7 21:20:57 | 显示全部楼层
成熟吗?

出0入0汤圆

发表于 2010-4-7 21:23:40 | 显示全部楼层
爪印..

出150入0汤圆

发表于 2010-4-7 21:26:31 | 显示全部楼层

出0入0汤圆

发表于 2010-4-7 21:41:36 | 显示全部楼层
mark

出0入0汤圆

 楼主| 发表于 2010-4-7 22:05:35 | 显示全部楼层
使用iccavr编译。参考价值很高。

出0入0汤圆

发表于 2010-4-7 22:14:36 | 显示全部楼层
mark

出0入264汤圆

发表于 2010-4-7 22:16:18 | 显示全部楼层
谢谢

出0入0汤圆

发表于 2010-4-7 22:19:48 | 显示全部楼层
不错

出0入0汤圆

发表于 2010-4-7 22:28:07 | 显示全部楼层
没找到使用说明, 介绍一下硬件环境好吗?

出0入0汤圆

发表于 2010-4-7 22:31:33 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-7 22:35:35 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-7 22:44:54 | 显示全部楼层
记下

出0入4汤圆

发表于 2010-4-7 22:47:57 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-7 22:50:53 | 显示全部楼层
配点说明就更好了。

出0入10汤圆

发表于 2010-4-7 23:01:00 | 显示全部楼层
有用

出0入0汤圆

发表于 2010-4-7 23:15:07 | 显示全部楼层
配点说明就更好了

出0入0汤圆

发表于 2010-4-7 23:20:08 | 显示全部楼层
来点接口说明就好了!

出0入0汤圆

发表于 2010-4-7 23:22:06 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 08:40:52 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 09:12:56 | 显示全部楼层
mark,谢谢。

出0入0汤圆

发表于 2010-4-8 09:16:23 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 09:18:45 | 显示全部楼层
回复【3楼】ywl0409 老黄牛
成熟吗?
-----------------------------------------------------------------------

熟了,可以吃了...

出610入42汤圆

发表于 2010-4-8 09:19:57 | 显示全部楼层
记号

出0入0汤圆

发表于 2010-4-8 09:41:52 | 显示全部楼层
仅看一下串口接收部分,问题不小。
假如某个干扰引起中断,不知道要等到啥时候才能退出中断了?
既已用中断接收,在中断里面为何又去用查询接收?
.....

出0入0汤圆

发表于 2010-4-8 09:53:28 | 显示全部楼层
收下看看先

出0入0汤圆

发表于 2010-4-8 10:06:00 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 10:12:33 | 显示全部楼层
Mark。

出0入0汤圆

发表于 2010-4-8 10:18:39 | 显示全部楼层
有参考价值,谢了~

出0入0汤圆

发表于 2010-4-8 10:20:02 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 10:25:27 | 显示全部楼层
收藏

出0入0汤圆

发表于 2010-4-8 11:27:36 | 显示全部楼层
收藏

出0入0汤圆

发表于 2010-4-8 11:34:51 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 12:44:36 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 12:57:18 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 13:29:29 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 13:38:44 | 显示全部楼层
mark

出0入0汤圆

发表于 2010-4-8 13:38:55 | 显示全部楼层
mark!

出0入0汤圆

发表于 2012-5-8 21:56:16 | 显示全部楼层
mark  不多 多谢楼主了

出0入0汤圆

发表于 2013-11-11 09:55:51 | 显示全部楼层
就只是PID调速吗?

出0入0汤圆

发表于 2014-6-9 19:39:34 | 显示全部楼层
mark~~~~~~

出0入0汤圆

发表于 2014-6-9 20:31:37 | 显示全部楼层
一个标准的PID,参数都是要从串口输入的

出0入0汤圆

发表于 2014-6-10 07:59:52 | 显示全部楼层
标记,我公司产品成熟使用的AVR电机控制算法

出0入0汤圆

发表于 2014-6-10 08:22:19 | 显示全部楼层
mark,谢谢分享!

出0入0汤圆

发表于 2014-6-10 10:29:51 | 显示全部楼层
MARK         

出85入4汤圆

发表于 2014-6-10 11:10:19 | 显示全部楼层
请问,这是控制什么电机的?

出0入0汤圆

发表于 2014-6-11 10:25:03 | 显示全部楼层
和张军写的《AVR系统典型开发实例》这本书里面的循迹机器人案列电机控制部分一模一样
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片。注意:要连续压缩2次才能满足要求!!】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-7-23 12:25

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表