本帖最后由 Malc 于 2015-6-10 13:15 编辑
电赛结束了一段时间了,趁这段时间,总结、整理一下自己的方案
------------------------------------------------------------------------------------------------------------
先来看看题目要求吧:
看完题目,经过一番思索和百度之后,有了大概的方案,于是乎开始准备材料,设计机械(我负责),同时其他队员开始设计tb6560驱动、电源模块等等,当然如果你本来就打算选择控制类题目,步进、直流电机及其驱动最好在比赛之前就有充分的准备,我们是使用现成模块的同时开始制作tb6560(比赛的时候其实评委不看我们做的电路,只看我们能否完成题目要求)
用到的材料大部分都是以前的雕刻机项目剩下的材料,倒是省了不少时间,时间宝贵,当然怎么简单有效怎么来了,编码器连接件等东西是这几天加工出来的,abs材料
来张机械结构渲染图,比赛的时候并没有把全部结果画出来,画出个大概就开始做了,那些细节都是比赛后加上去的:
具体的solidworks文件、工程图全部打包在附件中
接下来是电路部分,由于这是控制类题,所以电路比较简单,24v开关电源供电,2596-5v供系统工作,一块arduino nano,一块tb6560,一个ADC键盘用于现场演示,两个mini绝对型编码器1024线
ADC键盘,一个ADC口检测10个按键,我估计检测16个没问题:
tb6560:
2596稳压:
2014.10.26更新:
系统函数图如下:
在接下来就是程序了
总共写了14个版本程序,最后用的就是PID_1_3、PID_1_4,所以这里就给出这两个版本的注释吧:
///////////PID_1_3/////////
PID_1_3.ino :主程序,开机自动检测0°并计算出180°,之后进入循环功能演示。平衡采用PID控制,第一个PID,以摆杆角度作为作为输入,步进电机速度作为输出。第二个PID,以步进速度作为输入,平衡点作为输出(180°±8°)
当摆杆在-110°~-180°或110°~180°范围内时,进行平衡控制,否则关闭步进输出
编码器读数、PID控制放在Timer2中断中,以保证控制周期精准,控制周期5ms
串口命令读取、状态发送放在主函数循环中
起摆思路1:反复震荡摆杆使其摆角越来越大;步进电机往一个方向运动一点距离,停止,等待摆杆达到最高点,再反向运动,停止,等待摆杆达到反向最高点,如此反复,摆杆进入平衡范围后进行平衡控制(起摆时间太长,不采用)
起摆思路2:步进电机突然给一个方向速度,再急刹车,摆杆由于惯性继续圆周运动,当进入平衡范围时,进行平衡控制(时间基本1s以内)
起摆思路3:步进电机突然给一个方向速度,再急刹车,等待摆杆达到最高点,步进电机再反向快速运动,以提高摆杆动能(杆子太重可用此方法)
Stepp.ino :步进电机驱动,使用了Timer1,setStepperSpeed(long myspeed)用于更新速度,StepperEvent()根据速度值驱动步进,如果有加减速效果会更好
Command.ino :接收命令,更新参数,P100代表参数P=100,etc
Encoder.ino :绝对型编码器读数
Filter.ino :FIR低通滤波、中值滤波
function1~6.ino :分别实现基本要求1~3,提高要求1~3
///////////PID_1_4/////////
PID_1_4.ino : loop中命令值略有更改
Command.ino : 将读取串口数据改为读取ADC键盘值,ADC键盘优点:仅一个AD口可实现10个按键,缺点:读数略复杂,只能单点按键,适合atmega328管脚少的芯片
其他函数同PID_1_3
程序比较长,这里就贴出PID_1_3的主函数吧,完整代码附在附件中: - #include <TimerOne.h>
-
-
- #define setBit(val, bitn) (val |=(1<<(bitn)))
- #define clearBit(val, bitn) (val&=~(1<<(bitn)))
- #define getBit(val, bitn) (val &(1<<(bitn)) )
- int functionSpeed=0,functionFlag=0;
- int PIDSign=true,SpeedPIDSign=true;;
- int function3Sign=false;
- /////////////Angle Parameter/////////
- long setPoint=0,setPointOpp=0,originPoint=0;
- int degree=0,degreeLast=0,setPointDegree=0;
- /////////Arm Speed Parameter/////////
- #define PULSE_PER_ROUND 1600
- #define MAX_SP_SPEED (PULSE_PER_ROUND*1)
- #define MIN_SP_SPEED (-PULSE_PER_ROUND*1)
- long setPosition=0,position=0,positionLast=0;
- long absPositionLast=0,absPosition=0;
- int setSpeed=0,armSpeed=0,lastSpeed=0;
- int np=500,ni=0,nd=0;
- long sp_error_p=0,sp_error_i=0,sp_error_d=0;
- int tempSpSpeed=0,tempPosition;
- int delta=2,maxDelta=8;
- int calcAngle(int x)
- {
- if(setPointOpp<=x&&x<originPoint)
- return (x-setPointOpp)*-0.352;
- else if(0<=x&&x<setPointOpp)
- return (setPointOpp-x)*0.352;
- else
- return 180-(x-originPoint)*0.352;
-
- }
- void setup()
- {
- EncoderInit();
- StepperInit();
- Serial.begin(57600);
- Serial.print("test begin\ndelay 2s\n");
- int i,num=5,temp;
- for(i=0;i<num;i++)
- {
- temp=SSI(0);
- setPointOpp+=temp;//避免编码器0点在摆杆最低点或最高点,简化计算
- Serial.println(temp);
- delay(100);
- }
- Serial.println(setPointOpp);
- setPointOpp/=num;
- Serial.println(setPointOpp);
- if(setPointOpp<512)
- setPoint=setPointOpp+512;
- else
- setPoint=setPointOpp-512;
- originPoint=setPoint;
- setPointDegree=calcAngle(originPoint);
- Serial.print("setPointOpp=");
- Serial.println(setPointOpp);
- Serial.print("setPoint=");
- Serial.println(setPoint);
- delay(2000);
-
- /*--------------------*/
- // for Timer2
- /*--------------------*/
- // interrupts every 1 ms
-
- TIMSK2 &= ~(1<<TOIE2);
- TCCR2A &= ~((1<<WGM21) | (1<<WGM20));
- TCCR2B &= ~(1<<WGM22);
- ASSR &= ~(1<<AS2);
- TIMSK2 &= ~(1<<OCIE2A);
- TCCR2B |= (1<<CS22) | (1<<CS20);
- TCCR2B &= ~(1<<CS21);
- TCNT2 = 131;
- TIMSK2 |= (1<<TOIE2);
-
- Timer1.attachInterrupt( StepperEvent ); // attach the service routine here
- }
- int myspeed;
- unsigned long timer;
- int angle,angleLast;
- int error_p,error_i,error_d;
- int velocity;
- int kp=15,kd=0,ki=20;//p=15,i=15 works!!
- unsigned int counter=0;
- void loop()
- {
- checkParameter();
- switch(functionFlag)
- {
- case 1:
- function1();
- break;
- case 11:
- function1();
- break;
- case 2:
- function2();
- break;
- case 3:
- function3();
- break;
- case 4:
- function4();
- break;
- case 41:
- function41();
- break;
- case 6:
- function6();
- break;
-
- }
-
-
-
- Serial.print(originPoint);
- Serial.print(',');
- Serial.print(setPoint);
- Serial.print(',');
- Serial.print(angle);
- Serial.print(',');
- Serial.print(degree);
- Serial.print(',');
- Serial.print(armSpeed);//sp_error_i
- Serial.print(',');
- Serial.print(absPosition);
- Serial.print('\n');
-
- //delay(50);
- }
- int calcSpeed(int pos,int posLast)
- {
- if(abs(pos-posLast<512))
- return pos-posLast;
- int temp;
- if(pos>posLast)//counter clock
- temp=1023-pos-posLast;
- else if(pos<posLast)//clock
- temp=pos+(1023-posLast);
- if(temp>1000)
- return temp-1023;
- else if(temp<-1000)
- return 1023+temp;
- return temp;
- }
- int controlTimer=0,speedTimer=0;;
- int tempPos,tempPosLast;
- int armSpeedArr[3]={0};
- ISR(TIMER2_OVF_vect) {
- TIMSK2 |= (0<<TOIE2);
- TCNT2 = 131; // reload the timer
- controlTimer++;
- speedTimer++;
- // estimate velocity
- if (controlTimer == 5){
- controlTimer = 0;
- ////////////angle PID///////////
- angleLast=angle;
- angle=SSI(0);
- degreeLast=degree;
- degree=calcAngle(angle);
- //angle=FIR2(angle);
- velocity = (angle - angleLast);
- if(abs(angle-setPoint)<=4)
- error_p = 0;
- else
- error_p = setPoint-angle;
- error_d = velocity;
- error_i = error_i + error_p;
-
- if(PIDSign==false||abs(degree)<110)
- {
- error_p=0;
- error_d=0;
- error_i=0;
- }
- myspeed = - (kp*error_p + kd*error_d + ki*error_i/10);
- setStepperSpeed(myspeed + tempSpSpeed+functionSpeed);
- }
-
-
- if(speedTimer==5)
- {
- speedTimer=0;
- ////////Speed PID///////////////
- positionLast=position;
- lastSpeed=armSpeed;
- position=SSI(1);
-
- armSpeed=calcSpeed(position,positionLast);
- armSpeedArr[0]=armSpeedArr[1];
- armSpeedArr[1]=armSpeedArr[2];
- armSpeedArr[2]=armSpeed;
- armSpeed=mid(armSpeedArr[0],armSpeedArr[1],armSpeedArr[2]);
- if(abs(armSpeed)<=1)
- armSpeed=0;
-
- absPositionLast=absPosition;
- absPosition+=armSpeed;
-
- sp_error_p = setSpeed-armSpeed;
- sp_error_d = armSpeed-lastSpeed;
- sp_error_i = sp_error_i + sp_error_p;
- sp_error_i=constrain(sp_error_i,-1000,1000);
-
-
- tempSpSpeed = -(np*sp_error_p/10.0 + nd*sp_error_d/10.0 + ni*sp_error_i/100.0);
- tempSpSpeed=constrain(tempSpSpeed,-delta,delta);
- //if(PIDSign==false||SpeedPIDSign==false||abs(degree)>155)
- if(PIDSign==true&&SpeedPIDSign==true&&abs(degree)>155)
- {
- setPoint+=tempSpSpeed;
- if(setPoint>originPoint+maxDelta)
- setPoint=originPoint+maxDelta;
- else if(setPoint<originPoint-maxDelta)
- setPoint=originPoint-maxDelta;
- }
- else if(SpeedPIDSign==true)
- {
- setPoint=originPoint;
- }
- }
-
- //setStepperSpeed(myspeed + tempSpSpeed);
- TIMSK2 |= (1<<TOIE2);
- }
复制代码
视频、图片在这里:
附件:
115:
http://115.com/lb/5lbawy4ygq2#
倒立摆1.rar
115网盘礼包码:5lbawy4ygq2
TB6560 资料:
|