直流电机驱动器设计的应用程序
来源:互联网 发布:图片音乐合成软件 编辑:程序博客网 时间:2024/06/11 06:18
#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[i];}
}
//**************************读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[i]=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[i]=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();
}
- 直流电机驱动器设计的应用程序
- 直流电机驱动器
- 控制直流电机设计常用方法
- 直流电机的工作原理
- 直流电机的PWM调速
- 直流电机
- 有刷直流电机H桥功率驱动电路的设计心得
- 单片机驱动直流电机的例程
- 电动机的工作原理--直流电机
- 基于TMS320F2812无刷直流电机控制系统设计
- 密切关注你的NTFS驱动器(二):创建一个更改日志的应用程序
- 如何在应用程序中映射网络驱动器
- Python 应用程序的设计
- 磁带驱动器的使用
- tap驱动器的编译
- 限制驱动器的使用
- 限制驱动器的使用
- 隐藏驱动器的方法
- SD2303的具体应用
- typedef struct的一些资料(搜集)
- 条码基础知识
- 思念的碎片-close to- 错误总汇及解决方法
- 编写shell (一)
- 直流电机驱动器设计的应用程序
- JavaScript split() 方法
- UEFI 正常启动过程--与EDK为例,大家一起讨论吧!
- 检索 COM 类工厂中 CLSID 为 ???的组件时失败,原因是出现以下错误: 80080005。
- java 判断是否是周末
- USB电源
- 程序员工作效率
- C#,.Net经典面试题目及答案
- 寻线机器人系统设计程序