打印

分享一个串级PID程序源代码

[复制链接]
19839|21
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
moreStrong|  楼主 | 2016-1-6 11:14 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
/**********************************************************************************************************************
******串级PID控制程序、外环采用角度控制,内环采用角速度控制(直接由陀螺仪输出) 外环角速为欧拉角(硬件DMP)*****
***********************************************************************************************************************
******本子程序为四轴飞行器核心算法 ********************************************************************************/
/****************************
                                         Y(Roll)
     顺时针转 | 逆时针转
      motor1  |  motor4
       正桨   |   反桨
    --------------------X(Pitch)         
     逆时针转 | 顺时针转
      motor2  |  motor3
       反桨   |   正桨
              |
****************************/
#include "control.h"
#include "led.h"
#include "delay.h"
#include "sys.h"
#include "usart.h"
#include "key.h"
#include "pwm_out.h"
#include "mpu6050.h"
#include "ahrs.h"
#include "inv_mpu.h"
#include "inv_mpu_dmp_motion_driver.h"

//即将要更新的ACC加速度 GYRO陀螺仪
S_INT16_XYZ MPU6050_GYRO_LAST;

T_RC_Dat  Rc_D={0,0,0,1613};//遥控通道数据;
u8 lock=0;   //飞控上锁标志位
extern u8 txbuf[4];  //发送缓冲
extern u8 rxbuf[4];  //接收缓冲
extern u16 test1[3]; //接收到NRf24L01数据

float thr=0;//油门


//PID1 PID_Motor;
/***************************************************************/
float Pitch_i,Roll_i,Yaw_i;              //积分项
float Pitch_old,Roll_old,Yaw_old;        //角度保存
float Pitch_d,Roll_d,Yaw_d;              //微分项

float RC_Pitch=0,RC_Roll=0,RC_Yaw=0;           //期望的姿态角

float Pitch_shell_out,Roll_shell_out,Yaw_shell_out;//外环总输出
        //外环PID参数
float Pitch_shell_kp=140;
float Pitch_shell_ki=0.2;
float Pitch_shell_kd=0.8;
/*********************************/
float Roll_shell_kp=150;              
float Roll_shell_ki=0.2;
float Roll_shell_kd=0.8;
/*********************************/       
float Yaw_shell_kp=15;//0.87              
float Yaw_shell_ki=0.002;
float Yaw_shell_kd=0.02;   
float Gyro_radian_old_x,Gyro_radian_old_y,Gyro_radian_old_z;//陀螺仪保存
float pitch_core_kp_out,pitch_core_ki_out,pitch_core_kd_out,Roll_core_kp_out,\
                        Roll_core_ki_out,Roll_core_kd_out,Roll_core_ki_out,Yaw_core_kp_out,\
                        Yaw_core_ki_out,Yaw_core_kd_out;//内环单项输出

float Pitch_core_out,Roll_core_out,Yaw_core_out;//内环总输出      

//内环PID参数
float Pitch_core_kp=0.02;
float Pitch_core_ki=0;
float Pitch_core_kd=0.001;//0.001

float Roll_core_kp=0.02;
float Roll_core_ki=0;
float Roll_core_kd=0.001;

float Yaw_core_kp=0.001;
float Yaw_core_ki=0;
float Yaw_core_kd=0.001;


int16_t moto1=1613,moto2=1613,moto3=1613,moto4=1613;//电机的PWM值


/////////////////////////////////////////////////////////////////
//PID 控制函数
//输入:采样的姿态欧拉角数据
//反回:无
//备注:Pitch、Roll_i采用内环、外环PID控制 ,yaw采用内环控制
////////////////////////////////////////////////////////////////
void CONTROL_PID(float pit, float rol, float yaw)
{

  RC_Pitch=(Rc_D.pitch -1499.0f )/50;
  RC_Pitch-=3.3f;
//////////////////外环(PID)/////////////
  Pitch_i+=(pit-RC_Pitch);
//-------------Pitch积分限幅----------------//
  if(Pitch_i>300) Pitch_i=300;
  else if(Pitch_i<-300) Pitch_i=-300;
//-------------Pitch微分--------------------//
  Pitch_d=pit-Pitch_old;
//-------------Pitch  PID-------------------//
  Pitch_shell_out = Pitch_shell_kp*(pit-RC_Pitch) + Pitch_shell_ki*Pitch_i + Pitch_shell_kd*Pitch_d;
//保存角度
  Pitch_old=pit;
/*********************************************************/      

        RC_Roll=(Rc_D.roll-1502)/50;
        Roll_i+=(rol-RC_Roll);
//-------------Roll积分限幅----------------//
  if(Roll_i>300) Roll_i=300;
  else if(Roll_i<-300) Roll_i=-300;
//-------------Roll微分--------------------//
  Roll_d=rol-Roll_old;
//-------------Roll  PID-------------------//
  Roll_shell_out  = Roll_shell_kp*(rol-RC_Roll) + Roll_shell_ki*Roll_i + Roll_shell_kd*Roll_d;
//------------Roll保存角度------------------//
  Roll_old=rol;


  //RC_Yaw=(Rc_D.yaw-1501)*10;
//-------------Yaw微分--------------------//
  Yaw_d=MPU6050_GYRO_LAST.Z-Yaw_old;
//-------------Roll  PID-------------------//
  Yaw_shell_out  = Yaw_shell_kp*(MPU6050_GYRO_LAST.Z-RC_Yaw) + Yaw_shell_ki*Yaw_i + Yaw_shell_kd*Yaw_d;
//------------Roll保存角度------------------//
  Yaw_old=MPU6050_GYRO_LAST.Z;

////////////////////////内环角速度环(PD)///////////////////////////////
//陀螺仪        测得是角速度
  pitch_core_kp_out = Pitch_core_kp * (Pitch_shell_out + MPU6050_GYRO_LAST.Y );
  pitch_core_kd_out = Pitch_core_kd * (MPU6050_GYRO_LAST.Y   - Gyro_radian_old_y);

  Roll_core_kp_out  = Roll_core_kp  * (Roll_shell_out  + MPU6050_GYRO_LAST.X );
  Roll_core_kd_out  = Roll_core_kd  * (MPU6050_GYRO_LAST.X   - Gyro_radian_old_x);

  Yaw_core_kp_out  = Yaw_core_kp  * (Yaw_shell_out  + MPU6050_GYRO_LAST.Z);
  Yaw_core_kd_out  = Yaw_core_kd  * (MPU6050_GYRO_LAST.Z   - Gyro_radian_old_z);


  Pitch_core_out = pitch_core_kp_out + pitch_core_kd_out;
  Roll_core_out  = Roll_core_kp_out  + Roll_core_kd_out;
  Yaw_core_out   = Yaw_core_kp_out   + Yaw_core_kd_out;

  Gyro_radian_old_y = MPU6050_GYRO_LAST.X;
  Gyro_radian_old_x = MPU6050_GYRO_LAST.Y;
  Gyro_radian_old_z = MPU6050_GYRO_LAST.Z;   


//--------------------将输出值融合到四个电机--------------------------------//
                                       
        if(Rc_D.THROTTLE>1613)
        {
                                        thr=Rc_D.THROTTLE;               
                                        moto1=(int16_t)(thr - Roll_core_out - Pitch_core_out- Yaw_core_out);
                                        moto2=(int16_t)(thr + Roll_core_out - Pitch_core_out+ Yaw_core_out);      
                                        moto3=(int16_t)(thr + Roll_core_out + Pitch_core_out- Yaw_core_out);
                                        moto4=(int16_t)(thr - Roll_core_out + Pitch_core_out+ Yaw_core_out);  
                                               
                                }
        else
        {
                                        moto1 = 1613;
                                        moto2 = 1613;
                                        moto3 = 1613;
                                        moto4 = 1613;
        }
        Motor_PWM_Update(moto1,moto2,moto3,moto4);
}


void ahrs_control_PID_moto(void)//关键字:航姿 控制 串级PID 电机
{
        if(lock==1)
                {

                  if(mpu_dmp_get_data(&Angle.pitch ,&Angle.roll,&Angle.yaw)==0)//得到欧拉角
                        {
                                //MPU_Get_Accelerometer(&Acc.X ,&Acc.Y ,&Acc.Z );        //得到加速度传感器数据
        MPU_Get_Gyroscope(&Gyro.X ,&Gyro.Y ,&Gyro.Z);//读取陀螺仪数据
                                MPU_Get_GYRO();//陀螺仪数据更新函数
                                CONTROL_PID(Angle_SET.pitch,Angle_SET.roll,Angle_SET.yaw);//PID计算pwm值       
                          if(report) Data_Exchange();//航姿数据更新到匿名上位机
                        }
                               
                }
                else//飞机上锁
                {
                             Motor_PWM_Update(moto1-100,moto2-100,moto3-100,moto4-100);//飞控锁定,还没完善
                }
}




网上的串级串级PID控制算法都大同小异。









control.zip

3.04 KB

串级PID

评论
visitor99 2022-9-17 00:20 回复TA
亲测可用 ,非常感谢博主 

相关帖子

沙发
中西部牛仔| | 2016-1-6 20:59 | 只看该作者
谢谢楼主分享

使用特权

评论回复
板凳
ltc1994| | 2016-2-17 22:38 | 只看该作者
我看你的积分项里的累加之后没有乘采样时间和微分项相减后没有除采样时间,这个没事吧?
还有一个就是我在有些地方看到的说外环算出来的数据是角速度的期望值,但是我看楼主的算法里是将外环算出来的用作误差值的,是不是我看到的那个**说错了呢?

使用特权

评论回复
地板
ltc1994| | 2016-2-17 22:53 | 只看该作者
楼主这个代码写的真的蛮不错的,有逻辑易懂

使用特权

评论回复
5
虾米一代| | 2016-2-20 19:31 | 只看该作者
能问个问题吗?请问下串级PID调试,只需要内环就可以稳定的飞起来,加上外环反而容易震荡,于是减小内环P、I,始终找不到合适的值,有种说法“当飞机偏离30°,则需要以90°/s的速度修正”的意思是外环的P是3吗?在此基础调内环?

使用特权

评论回复
6
moreStrong|  楼主 | 2016-3-3 15:29 | 只看该作者
虾米一代 发表于 2016-2-20 19:31
能问个问题吗?请问下串级PID调试,只需要内环就可以稳定的飞起来,加上外环反而容易震荡,于是减小内环P、 ...

其实只要内环既可以离开地面,但是会往一边倾斜,这个时候就需要加入外环,我在调试过程中,发现内环参数一定要调好,不然完全没法调。基本把内环参数确定后,可以稍微修改一下,

使用特权

评论回复
7
肖志宏| | 2016-3-10 23:43 | 只看该作者
楼主  我觉得你的内环没写对  不知道我讲的对不对
pitch_core_kp_out = Pitch_core_kp * (Pitch_shell_out + MPU6050_GYRO_LAST.Y );
  pitch_core_kd_out = Pitch_core_kd * (MPU6050_GYRO_LAST.Y   - Gyro_radian_old_y);

  Roll_core_kp_out  = Roll_core_kp  * (Roll_shell_out  + MPU6050_GYRO_LAST.X );
  Roll_core_kd_out  = Roll_core_kd  * (MPU6050_GYRO_LAST.X   - Gyro_radian_old_x);

  Yaw_core_kp_out  = Yaw_core_kp  * (Yaw_shell_out  + MPU6050_GYRO_LAST.Z);
  Yaw_core_kd_out  = Yaw_core_kd  * (MPU6050_GYRO_LAST.Z   - Gyro_radian_old_z);

外环给定的输入不应该是与实际角速度做差吗?得到偏差  而楼主你却把内环的期望输入和实际角速度相加了。。。。。。。。。。。。
还有微分项 应该是这一次的偏差减去上一次的偏差  这才能算微分吧  微分项不是应该是偏差的微分么?
希望可以交流下   QQ1292938163

使用特权

评论回复
8
四轴flay| | 2016-4-12 14:20 | 只看该作者
融合电机那几个错了吧?

使用特权

评论回复
9
mig2126| | 2016-4-23 13:32 | 只看该作者
正需要这个,太感谢了

使用特权

评论回复
10
cana_xzy| | 2016-4-30 01:08 | 只看该作者
下载来看看

使用特权

评论回复
11
nokia698| | 2016-9-1 17:52 | 只看该作者
想和楼主探讨一个问题,在外环控制里头,我想实现姿态控制翻跟头,怎么添加代码(比如在哪修改)?

使用特权

评论回复
12
keigo2008| | 2016-9-28 20:13 | 只看该作者

使用特权

评论回复
13
springvirus| | 2016-9-30 10:02 | 只看该作者
感谢楼主分享,后面准备啃四轴这块硬骨头~~

使用特权

评论回复
14
zhaoguangdong| | 2017-8-10 10:05 | 只看该作者
感谢楼主分享,后面准备啃四轴这块硬骨头~~

使用特权

评论回复
15
sum123456| | 2018-4-12 11:57 | 只看该作者

感谢楼主分享,后面准备啃四轴这块硬骨头~~

使用特权

评论回复
16
Assassin007| | 2018-5-17 16:09 | 只看该作者
楼主 敢问如果砍掉Rc_D这个结构体 就是说不加遥控器 这代码要怎么改啊

使用特权

评论回复
17
yutaylor| | 2018-10-26 22:00 | 只看该作者
三克油

使用特权

评论回复
18
yuchl| | 2018-11-5 18:12 | 只看该作者
感谢楼主分享

使用特权

评论回复
19
hjl2832| | 2018-11-15 08:52 | 只看该作者
好东西,谢谢分享。。。。。。

使用特权

评论回复
20
TT1000| | 2021-5-17 14:03 | 只看该作者
谢谢楼主分享的资料

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

3

主题

6

帖子

2

粉丝