Malc 发表于 2013-10-7 19:24:32

2013电子设计竞赛-旋转倒立摆-整体方案-2014.10.26更新

本帖最后由 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={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=armSpeedArr;
    armSpeedArr=armSpeedArr;
    armSpeedArr=armSpeed;
    armSpeed=mid(armSpeedArr,armSpeedArr,armSpeedArr);
    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);
}



视频、图片在这里:
http://v.youku.com/v_show/id_XNjA2MzY3NTQw.html


附件:




115:
http://115.com/lb/5lbawy4ygq2#
倒立摆1.rar
115网盘礼包码:5lbawy4ygq2

TB6560 资料:

Malc 发表于 2014-8-4 14:16:57

gyq 发表于 2014-8-4 10:40 static/image/common/back.gif
你说的用502固定在步进电机的轴上么,还是底座上,要是在底座上的话,编码器怎么采集转轴的速度数据呀,这 ...

用502来粘合步进电机的轴与编码器的轴,用胶带固定编码器,防止自转

弘毅 发表于 2013-10-7 21:04:09

{:soso_e102:}终于看到源码了。。。仔细学习

幻生幻灭 发表于 2013-10-7 21:15:30

这效果图让我想起了那个小蜘蛛

Malc 发表于 2013-10-7 22:16:25

弘毅 发表于 2013-10-7 21:04 static/image/common/back.gif
终于看到源码了。。。仔细学习

时间紧迫,代码写的比较乱,算法有待改进。。
ps:那两个nano偷偷拆下来自己用了{:soso_e113:}

Malc 发表于 2013-10-7 22:17:22

幻生幻灭 发表于 2013-10-7 21:15 static/image/common/back.gif
这效果图让我想起了那个小蜘蛛

不玩蜘蛛了,雕刻机更好玩~

ZYLLOVE 发表于 2013-10-7 22:21:29

关注你很久了,好喜欢你的文章,终于知道你的倒立摆是咋做的的啦~学习,谢谢~~~

lmb312 发表于 2013-10-8 10:54:19

Uno的话程序能用么

小黑爱GEEK 发表于 2013-10-9 11:34:45

我们做的也是这个,不过跟您的比起来简直就是个渣啊,这会真的学习了,膜拜啊

tangmao48 发表于 2013-10-9 21:42:59

这架势应该有国一吧。我们的学生一组国一一组国二。没有您这位大神做得好。

惠普彩色打印机 发表于 2013-10-11 15:51:16

大神。。。。肯定是国一,一个地区三等奖给跪了

惠普彩色打印机 发表于 2013-10-11 15:52:04

做D题的孩纸快要被吓死了

wisology 发表于 2013-10-11 21:52:58

编码器放在哪儿了?硬件及机械部分是不是没讲完整,我个人理解是放到倒立摆转轴那块了。

Malc 发表于 2013-10-12 16:40:45

wisology 发表于 2013-10-11 21:52 static/image/common/back.gif
编码器放在哪儿了?硬件及机械部分是不是没讲完整,我个人理解是放到倒立摆转轴那块了。

两个编码器,一个装在摆臂末端,检测摆杆的转角,另一个装在步进底下,步进没有闭环的话摆杆会一直在水平方向移动停不下来,跟平衡车一样

龙翔竞天 发表于 2013-10-12 23:17:53

这个厉害。。。。。想都没想过。。

青青木麻黄 发表于 2013-10-14 13:51:56

太赞了!!!
页: [1] 2 3 4 5 6 7 8 9 10
查看完整版本: 2013电子设计竞赛-旋转倒立摆-整体方案-2014.10.26更新