2012-08-04 19:12:36 -03:00
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008-2012 PX4 Development Team. All rights reserved.
|
|
|
|
* Author: @author Ivan Ovinnikov <oivan@ethz.ch>
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
* 3. Neither the name PX4 nor the names of its contributors may be
|
|
|
|
* used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @file fixedwing_control.c
|
|
|
|
* Implementation of a fixed wing attitude and position controller.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <debug.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <termios.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <arch/board/up_hrt.h>
|
|
|
|
#include <arch/board/board.h>
|
2012-08-04 19:56:30 -03:00
|
|
|
#include <drivers/drv_pwm_output.h>
|
2012-08-04 19:12:36 -03:00
|
|
|
#include <nuttx/spi.h>
|
|
|
|
#include "../mix_and_link/mix_and_link.h"
|
|
|
|
#include "fixedwing_control.h"
|
|
|
|
|
|
|
|
__EXPORT int fixedwing_control_main(int argc, char *argv[]);
|
|
|
|
|
|
|
|
#define PID_DT 0.01f
|
|
|
|
#define PID_SCALER 0.1f
|
|
|
|
#define PID_DERIVMODE_CALC 0
|
|
|
|
#define HIL_MODE 32
|
|
|
|
#define RAD2DEG ((1.0/180.0)*M_PI)
|
|
|
|
#define AUTO -1000
|
|
|
|
#define MANUAL 3000
|
|
|
|
#define SERVO_MIN 1000
|
|
|
|
#define SERVO_MAX 2000
|
|
|
|
|
|
|
|
pthread_t control_thread;
|
|
|
|
pthread_t nav_thread;
|
|
|
|
pthread_t servo_thread;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Servo channels function enumerator used for
|
|
|
|
* the servo writing part
|
|
|
|
*/
|
|
|
|
|
|
|
|
enum SERVO_CHANNELS_FUNCTION {
|
|
|
|
|
|
|
|
AIL_1 = 0,
|
|
|
|
AIL_2 = 1,
|
|
|
|
MOT = 2,
|
|
|
|
ACT_1 = 3,
|
|
|
|
ACT_2 = 4,
|
|
|
|
ACT_3 = 5,
|
|
|
|
ACT_4 = 6,
|
|
|
|
ACT_5 = 7
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The plane_data structure.
|
|
|
|
*
|
|
|
|
* The plane data structure is the local storage of all the flight information of the aircraft
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
double lat;
|
|
|
|
double lon;
|
|
|
|
float alt;
|
|
|
|
float vx;
|
|
|
|
float vy;
|
|
|
|
float vz;
|
|
|
|
float yaw;
|
|
|
|
float hdg;
|
|
|
|
float pitch;
|
|
|
|
float roll;
|
|
|
|
float yawspeed;
|
|
|
|
float pitchspeed;
|
|
|
|
float rollspeed;
|
|
|
|
float rollb; /* body frame angles */
|
|
|
|
float pitchb;
|
|
|
|
float yawb;
|
|
|
|
float p;
|
|
|
|
float q;
|
|
|
|
float r; /* body angular rates */
|
|
|
|
|
|
|
|
/* PID parameters*/
|
|
|
|
|
|
|
|
float Kp_att;
|
|
|
|
float Ki_att;
|
|
|
|
float Kd_att;
|
|
|
|
float Kp_pos;
|
|
|
|
float Ki_pos;
|
|
|
|
float Kd_pos;
|
|
|
|
float intmax_att;
|
|
|
|
float intmax_pos;
|
|
|
|
|
|
|
|
/* Next waypoint*/
|
|
|
|
|
|
|
|
float wp_x;
|
|
|
|
float wp_y;
|
|
|
|
float wp_z;
|
|
|
|
|
|
|
|
/* Setpoints */
|
|
|
|
|
|
|
|
float airspeed;
|
|
|
|
float groundspeed;
|
|
|
|
float roll_setpoint;
|
|
|
|
float pitch_setpoint;
|
|
|
|
float throttle_setpoint;
|
|
|
|
|
|
|
|
/* Navigation mode*/
|
|
|
|
int mode;
|
|
|
|
|
|
|
|
} plane_data_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The control_outputs structure.
|
|
|
|
*
|
|
|
|
* The control outputs structure contains the control outputs
|
|
|
|
* of the aircraft
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
float roll_ailerons;
|
|
|
|
float pitch_elevator;
|
|
|
|
float yaw_rudder;
|
|
|
|
float throttle;
|
|
|
|
// set the aux values to 0 per default
|
|
|
|
float aux1;
|
|
|
|
float aux2;
|
|
|
|
float aux3;
|
|
|
|
float aux4;
|
|
|
|
uint8_t mode; // HIL_ENABLED: 32
|
|
|
|
uint8_t nav_mode;
|
|
|
|
} control_outputs_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generic PID algorithm with PD scaling
|
|
|
|
*/
|
|
|
|
static float pid(float error, float error_deriv, uint16_t dt, float scaler, float K_p, float K_i, float K_d, float intmax);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Output calculations
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void calc_body_angular_rates(float roll, float pitch, float yaw, float rollspeed, float pitchspeed, float yawspeed);
|
|
|
|
static void calc_rotation_matrix(float roll, float pitch, float yaw, float x, float y, float z);
|
|
|
|
static void calc_bodyframe_angles(float roll, float pitch, float yaw);
|
|
|
|
static float calc_bearing(void);
|
|
|
|
static float calc_roll_ail(void);
|
|
|
|
static float calc_pitch_elev(void);
|
|
|
|
static float calc_yaw_rudder(float hdg);
|
|
|
|
static float calc_throttle(void);
|
|
|
|
static float calc_gnd_speed(void);
|
|
|
|
static void get_parameters(void);
|
|
|
|
static float calc_roll_setpoint(void);
|
|
|
|
static float calc_pitch_setpoint(void);
|
|
|
|
static float calc_throttle_setpoint(void);
|
|
|
|
static float calc_wp_distance(void);
|
|
|
|
static void set_plane_mode(void);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The control, navigation and servo loop threads
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void *control_loop(void *arg);
|
|
|
|
static void *nav_loop(void *arg);
|
|
|
|
static void *servo_loop(void *arg);
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
plane_data_t plane_data;
|
|
|
|
control_outputs_t control_outputs;
|
|
|
|
float scaler = 1; //M_PI;
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Calculates the PID control output given an error. Incorporates PD scaling and low-pass filter for the derivative component.
|
|
|
|
*
|
|
|
|
* @param error the input error
|
|
|
|
* @param error_deriv the derivative of the input error
|
|
|
|
* @param dt time constant
|
|
|
|
* @param scaler PD scaler
|
|
|
|
* @param K_p P gain
|
|
|
|
* @param K_i I gain
|
|
|
|
* @param K_d D gain
|
|
|
|
* @param intmax Integration limit
|
|
|
|
*
|
|
|
|
* @return the PID control output
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float pid(float error, float error_deriv, uint16_t dt, float scale, float K_p, float K_i, float K_d, float intmax)
|
|
|
|
{
|
|
|
|
float Kp = K_p;
|
|
|
|
float Ki = K_i;
|
|
|
|
float Kd = K_d;
|
|
|
|
float delta_time = dt;
|
|
|
|
float lerror;
|
|
|
|
float imax = intmax;
|
|
|
|
float integrator;
|
|
|
|
float derivative;
|
|
|
|
float lderiv;
|
|
|
|
int fCut = 20; /* anything above 20 Hz is considered noise - low pass filter for the derivative */
|
|
|
|
float output = 0;
|
|
|
|
|
|
|
|
output += error * Kp;
|
|
|
|
|
|
|
|
if ((fabs(Kd) > 0) && (dt > 0)) {
|
|
|
|
|
|
|
|
if (PID_DERIVMODE_CALC) {
|
|
|
|
derivative = (error - lerror) / delta_time;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* discrete low pass filter, cuts out the
|
|
|
|
* high frequency noise that can drive the controller crazy
|
|
|
|
*/
|
|
|
|
float RC = 1 / (2 * M_PI * fCut);
|
|
|
|
derivative = lderiv +
|
|
|
|
(delta_time / (RC + delta_time)) * (derivative - lderiv);
|
|
|
|
|
|
|
|
/* update state */
|
|
|
|
lerror = error;
|
|
|
|
lderiv = derivative;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
derivative = error_deriv;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add in derivative component */
|
|
|
|
output += Kd * derivative;
|
|
|
|
}
|
|
|
|
|
|
|
|
//printf("PID derivative %i\n", (int)(1000*derivative));
|
|
|
|
|
|
|
|
/* scale the P and D components with the PD scaler */
|
|
|
|
output *= scale;
|
|
|
|
|
|
|
|
/* Compute integral component if time has elapsed */
|
|
|
|
if ((fabs(Ki) > 0) && (dt > 0)) {
|
|
|
|
integrator += (error * Ki) * scaler * delta_time;
|
|
|
|
|
|
|
|
if (integrator < -imax) {
|
|
|
|
integrator = -imax;
|
|
|
|
|
|
|
|
} else if (integrator > imax) {
|
|
|
|
integrator = imax;
|
|
|
|
}
|
|
|
|
|
|
|
|
output += integrator;
|
|
|
|
}
|
|
|
|
|
|
|
|
//printf("PID Integrator %i\n", (int)(1000*integrator));
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load parameters from global storage.
|
|
|
|
*
|
|
|
|
* Fetches the current parameters from the global parameter storage and writes them
|
|
|
|
* to the plane_data structure
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void get_parameters()
|
|
|
|
{
|
|
|
|
plane_data.Kp_att = global_data_parameter_storage->pm.param_values[PARAM_PID_ATT_P];
|
|
|
|
plane_data.Ki_att = global_data_parameter_storage->pm.param_values[PARAM_PID_ATT_I];
|
|
|
|
plane_data.Kd_att = global_data_parameter_storage->pm.param_values[PARAM_PID_ATT_D];
|
|
|
|
plane_data.Kp_pos = global_data_parameter_storage->pm.param_values[PARAM_PID_POS_P];
|
|
|
|
plane_data.Ki_pos = global_data_parameter_storage->pm.param_values[PARAM_PID_POS_I];
|
|
|
|
plane_data.Kd_pos = global_data_parameter_storage->pm.param_values[PARAM_PID_POS_D];
|
|
|
|
plane_data.intmax_att = global_data_parameter_storage->pm.param_values[PARAM_PID_ATT_AWU];
|
|
|
|
plane_data.intmax_pos = global_data_parameter_storage->pm.param_values[PARAM_PID_POS_AWU];
|
|
|
|
plane_data.airspeed = global_data_parameter_storage->pm.param_values[PARAM_AIRSPEED];
|
|
|
|
plane_data.wp_x = global_data_parameter_storage->pm.param_values[PARAM_WPLON];
|
|
|
|
plane_data.wp_y = global_data_parameter_storage->pm.param_values[PARAM_WPLAT];
|
|
|
|
plane_data.wp_z = global_data_parameter_storage->pm.param_values[PARAM_WPALT];
|
|
|
|
plane_data.mode = global_data_parameter_storage->pm.param_values[PARAM_FLIGHTMODE];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculates the body angular rates.
|
|
|
|
*
|
|
|
|
* Calculates the rates of the plane using inertia matrix and
|
|
|
|
* writes them to the plane_data structure
|
|
|
|
*
|
|
|
|
* @param roll
|
|
|
|
* @param pitch
|
|
|
|
* @param yaw
|
|
|
|
* @param rollspeed
|
|
|
|
* @param pitchspeed
|
|
|
|
* @param yawspeed
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void calc_body_angular_rates(float roll, float pitch, float yaw, float rollspeed, float pitchspeed, float yawspeed)
|
|
|
|
{
|
|
|
|
plane_data.p = rollspeed - sinf(pitch) * yawspeed;
|
|
|
|
plane_data.q = cosf(roll) * pitchspeed + sinf(roll) * cos(pitch) * yawspeed;
|
|
|
|
plane_data.r = -sinf(roll) * pitchspeed + cosf(roll) * cos(pitch) * yawspeed;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Calculates the attitude angles in the body reference frame.
|
|
|
|
*
|
|
|
|
* Writes them to the plane data structure
|
|
|
|
*
|
|
|
|
* @param roll
|
|
|
|
* @param pitch
|
|
|
|
* @param yaw
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void calc_bodyframe_angles(float roll, float pitch, float yaw)
|
|
|
|
{
|
|
|
|
plane_data.rollb = cosf(yaw) * cosf(pitch) * roll +
|
|
|
|
(cosf(yaw) * sinf(pitch) * sinf(roll) + sinf(yaw) * cosf(roll)) * pitch
|
|
|
|
+ (-cosf(yaw) * sinf(pitch) * cosf(roll) + sinf(yaw) * sinf(roll)) * yaw;
|
|
|
|
plane_data.pitchb = -sinf(yaw) * cosf(pitch) * roll +
|
|
|
|
(-sinf(yaw) * sinf(pitch) * sinf(roll) + cosf(yaw) * cosf(roll)) * pitch
|
|
|
|
+ (sinf(yaw) * sinf(pitch) * cosf(roll) + cosf(yaw) * sinf(roll)) * yaw;
|
|
|
|
plane_data.yawb = sinf(pitch) * roll - cosf(pitch) * sinf(roll) * pitch + cosf(pitch) * cosf(roll) * yaw;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_rotation_matrix
|
|
|
|
*
|
|
|
|
* Calculates the rotation matrix
|
|
|
|
*
|
|
|
|
* @param roll
|
|
|
|
* @param pitch
|
|
|
|
* @param yaw
|
|
|
|
* @param x
|
|
|
|
* @param y
|
|
|
|
* @param z
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void calc_rotation_matrix(float roll, float pitch, float yaw, float x, float y, float z)
|
|
|
|
{
|
|
|
|
plane_data.rollb = cosf(yaw) * cosf(pitch) * x +
|
|
|
|
(cosf(yaw) * sinf(pitch) * sinf(roll) + sinf(yaw) * cosf(roll)) * y
|
|
|
|
+ (-cosf(yaw) * sinf(pitch) * cosf(roll) + sinf(yaw) * sinf(roll)) * z;
|
|
|
|
plane_data.pitchb = -sinf(yaw) * cosf(pitch) * x +
|
|
|
|
(-sinf(yaw) * sinf(pitch) * sinf(roll) + cosf(yaw) * cosf(roll)) * y
|
|
|
|
+ (sinf(yaw) * sinf(pitch) * cosf(roll) + cosf(yaw) * sinf(roll)) * z;
|
|
|
|
plane_data.yawb = sinf(pitch) * x - cosf(pitch) * sinf(roll) * y + cosf(pitch) * cosf(roll) * z;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_bearing
|
|
|
|
*
|
|
|
|
* Calculates the bearing error of the plane compared to the waypoints
|
|
|
|
*
|
|
|
|
* @return bearing Bearing error
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_bearing()
|
|
|
|
{
|
|
|
|
float bearing = 90 + atan2(-(plane_data.wp_y - plane_data.lat), (plane_data.wp_x - plane_data.lon)) * RAD2DEG;
|
|
|
|
|
|
|
|
if (bearing < 0)
|
|
|
|
bearing += 360;
|
|
|
|
|
|
|
|
return bearing;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_roll_ail
|
|
|
|
*
|
|
|
|
* Calculates the roll ailerons control output
|
|
|
|
*
|
|
|
|
* @return Roll ailerons control output (-1,1)
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_roll_ail()
|
|
|
|
{
|
|
|
|
float ret = pid((plane_data.roll_setpoint - plane_data.roll) / scaler, plane_data.rollspeed, PID_DT, PID_SCALER,
|
|
|
|
plane_data.Kp_att, plane_data.Ki_att, plane_data.Kd_att, plane_data.intmax_att);
|
|
|
|
|
|
|
|
if (ret < -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (ret > 1)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_pitch_elev
|
|
|
|
*
|
|
|
|
* Calculates the pitch elevators control output
|
|
|
|
*
|
|
|
|
* @return Pitch elevators control output (-1,1)
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_pitch_elev()
|
|
|
|
{
|
|
|
|
float ret = pid((plane_data.pitch_setpoint - plane_data.pitch) / scaler, plane_data.pitchspeed, PID_DT, PID_SCALER,
|
|
|
|
plane_data.Kp_att, plane_data.Ki_att, plane_data.Kd_att, plane_data.intmax_att);
|
|
|
|
|
|
|
|
if (ret < -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (ret > 1)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_yaw_rudder
|
|
|
|
*
|
|
|
|
* Calculates the yaw rudder control output (only if yaw rudder exists on the model)
|
|
|
|
*
|
|
|
|
* @return Yaw rudder control output (-1,1)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_yaw_rudder(float hdg)
|
|
|
|
{
|
|
|
|
float ret = pid((plane_data.yaw / RAD2DEG - abs(hdg)) / scaler, plane_data.yawspeed, PID_DT, PID_SCALER,
|
|
|
|
plane_data.Kp_pos, plane_data.Ki_pos, plane_data.Kd_pos, plane_data.intmax_pos);
|
|
|
|
|
|
|
|
if (ret < -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (ret > 1)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_throttle
|
|
|
|
*
|
|
|
|
* Calculates the throttle control output
|
|
|
|
*
|
|
|
|
* @return Throttle control output (0,1)
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_throttle()
|
|
|
|
{
|
|
|
|
float ret = pid(plane_data.throttle_setpoint - calc_gnd_speed(), 0, PID_DT, PID_SCALER,
|
|
|
|
plane_data.Kp_pos, plane_data.Ki_pos, plane_data.Kd_pos, plane_data.intmax_pos);
|
|
|
|
|
|
|
|
if (ret < 0.2)
|
|
|
|
return 0.2;
|
|
|
|
|
|
|
|
if (ret > 1)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_gnd_speed
|
|
|
|
*
|
|
|
|
* Calculates the ground speed using the x and y components
|
|
|
|
*
|
|
|
|
* Input: none (operation on global data)
|
|
|
|
*
|
|
|
|
* Output: Ground speed of the plane
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_gnd_speed()
|
|
|
|
{
|
|
|
|
float gnd_speed = sqrtf(plane_data.vx * plane_data.vx + plane_data.vy * plane_data.vy);
|
|
|
|
return gnd_speed;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_wp_distance
|
|
|
|
*
|
|
|
|
* Calculates the distance to the next waypoint
|
|
|
|
*
|
|
|
|
* @return the distance to the next waypoint
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_wp_distance()
|
|
|
|
{
|
|
|
|
return sqrtf((plane_data.lat - plane_data.wp_y) * (plane_data.lat - plane_data.wp_y) +
|
|
|
|
(plane_data.lon - plane_data.wp_x) * (plane_data.lon - plane_data.wp_x));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_roll_setpoint
|
|
|
|
*
|
|
|
|
* Calculates the offset angle for the roll plane,
|
|
|
|
* saturates at +- 35 deg.
|
|
|
|
*
|
|
|
|
* @return setpoint on which attitude control should stabilize while changing heading
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_roll_setpoint()
|
|
|
|
{
|
|
|
|
float setpoint = 0;
|
|
|
|
|
|
|
|
if (plane_data.mode == TAKEOFF) {
|
|
|
|
setpoint = 0;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
setpoint = calc_bearing() - plane_data.yaw;
|
|
|
|
|
|
|
|
if (setpoint < -35)
|
|
|
|
return -35;
|
|
|
|
|
|
|
|
if (setpoint > 35)
|
|
|
|
return 35;
|
|
|
|
}
|
|
|
|
|
|
|
|
return setpoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_pitch_setpoint
|
|
|
|
*
|
|
|
|
* Calculates the offset angle for the pitch plane
|
|
|
|
* saturates at +- 35 deg.
|
|
|
|
*
|
|
|
|
* @return setpoint on which attitude control should stabilize while changing altitude
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_pitch_setpoint()
|
|
|
|
{
|
|
|
|
float setpoint = 0;
|
|
|
|
|
|
|
|
if (plane_data.mode == TAKEOFF) {
|
|
|
|
setpoint = 35;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
setpoint = atanf((plane_data.wp_z - plane_data.alt) / calc_wp_distance()) * RAD2DEG;
|
|
|
|
|
|
|
|
if (setpoint < -35)
|
|
|
|
return -35;
|
|
|
|
|
|
|
|
if (setpoint > 35)
|
|
|
|
return 35;
|
|
|
|
}
|
|
|
|
|
|
|
|
return setpoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calc_throttle_setpoint
|
|
|
|
*
|
|
|
|
* Calculates the throttle setpoint for different flight modes
|
|
|
|
*
|
|
|
|
* @return throttle output setpoint
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static float calc_throttle_setpoint()
|
|
|
|
{
|
|
|
|
float setpoint = 0;
|
|
|
|
|
|
|
|
// if TAKEOFF full throttle
|
|
|
|
if (plane_data.mode == TAKEOFF) {
|
|
|
|
setpoint = 60;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if CRUISE - parameter value
|
|
|
|
if (plane_data.mode == CRUISE) {
|
|
|
|
setpoint = plane_data.airspeed;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if LAND no throttle
|
|
|
|
if (plane_data.mode == LAND) {
|
|
|
|
setpoint = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return setpoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set_plane_mode
|
|
|
|
*
|
|
|
|
* Sets the plane mode
|
|
|
|
* (TAKEOFF, CRUISE, LOITER or LAND)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void set_plane_mode()
|
|
|
|
{
|
|
|
|
if (plane_data.alt < 10) {
|
|
|
|
plane_data.mode = TAKEOFF;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
plane_data.mode = CRUISE;
|
|
|
|
// TODO: if reached waypoint and no further waypoint exists, go to LOITER mode
|
|
|
|
}
|
|
|
|
|
|
|
|
// Debug override - don't need TAKEOFF mode for now
|
|
|
|
plane_data.mode = CRUISE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* fixedwing_control_main
|
|
|
|
*
|
|
|
|
* @param argc number of arguments
|
|
|
|
* @param argv argument array
|
|
|
|
*
|
|
|
|
* @return 0
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
int fixedwing_control_main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
/* print text */
|
|
|
|
printf("Fixedwing control started\n");
|
|
|
|
usleep(100000);
|
|
|
|
|
|
|
|
/* default values for arguments */
|
|
|
|
char *fixedwing_uart_name = "/dev/ttyS1";
|
|
|
|
char *commandline_usage = "\tusage: fixedwing_control -d fixedwing-devicename\n";
|
|
|
|
|
|
|
|
/* read arguments */
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (strcmp(argv[i], "-d") == 0 || strcmp(argv[i], "--device") == 0) {
|
|
|
|
if (argc > i + 1) {
|
|
|
|
fixedwing_uart_name = argv[i + 1];
|
|
|
|
|
|
|
|
} else {
|
|
|
|
printf(commandline_usage);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set up to publish fixed wing control messages */
|
|
|
|
struct fixedwing_control_s control;
|
|
|
|
int fixedwing_control_pub = orb_advertise(ORB_ID(fixedwing_control), &control);
|
|
|
|
|
|
|
|
/* Subscribe to global position, attitude and rc */
|
|
|
|
struct vehicle_global_position_s global_pos;
|
|
|
|
int global_pos_sub = orb_subscribe(ORB_ID(vehicle_global_position));
|
|
|
|
struct vehicle_attitude_s att;
|
|
|
|
int attitude_sub = orb_subscribe(ORB_ID(vehicle_attitude));
|
|
|
|
struct rc_channels_s rc;
|
|
|
|
int rc_sub = orb_subscribe(ORB_ID(rc_channels));
|
|
|
|
|
|
|
|
/* Control constants */
|
|
|
|
control_outputs.mode = HIL_MODE;
|
|
|
|
control_outputs.nav_mode = 0;
|
|
|
|
|
|
|
|
/* Servo setup */
|
|
|
|
|
|
|
|
int fd;
|
2012-08-04 19:56:30 -03:00
|
|
|
servo_position_t data[PWM_OUTPUT_MAX_CHANNELS];
|
2012-08-04 19:12:36 -03:00
|
|
|
|
|
|
|
fd = open("/dev/pwm_servo", O_RDWR);
|
|
|
|
|
|
|
|
if (fd < 0) {
|
|
|
|
printf("failed opening /dev/pwm_servo\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
ioctl(fd, PWM_SERVO_ARM, 0);
|
|
|
|
|
|
|
|
int16_t buffer_rc[3];
|
|
|
|
int16_t buffer_servo[3];
|
|
|
|
mixer_data_t mixer_buffer;
|
|
|
|
mixer_buffer.input = buffer_rc;
|
|
|
|
mixer_buffer.output = buffer_servo;
|
|
|
|
|
|
|
|
mixer_conf_t mixers[3];
|
|
|
|
|
|
|
|
mixers[0].source = PITCH;
|
|
|
|
mixers[0].nr_actuators = 2;
|
|
|
|
mixers[0].dest[0] = AIL_1;
|
|
|
|
mixers[0].dest[1] = AIL_2;
|
|
|
|
mixers[0].dual_rate[0] = 1;
|
|
|
|
mixers[0].dual_rate[1] = 1;
|
|
|
|
|
|
|
|
mixers[1].source = ROLL;
|
|
|
|
mixers[1].nr_actuators = 2;
|
|
|
|
mixers[1].dest[0] = AIL_1;
|
|
|
|
mixers[1].dest[1] = AIL_2;
|
|
|
|
mixers[1].dual_rate[0] = 1;
|
|
|
|
mixers[1].dual_rate[1] = -1;
|
|
|
|
|
|
|
|
mixers[2].source = THROTTLE;
|
|
|
|
mixers[2].nr_actuators = 1;
|
|
|
|
mixers[2].dest[0] = MOT;
|
|
|
|
mixers[2].dual_rate[0] = 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Main control, navigation and servo routine
|
|
|
|
*/
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* DATA Handling
|
|
|
|
* Fetch current flight data
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* get position, attitude and rc inputs */
|
|
|
|
// XXX add error checking
|
|
|
|
orb_copy(ORB_ID(vehicle_global_position), global_pos_sub, &global_pos);
|
|
|
|
orb_copy(ORB_ID(vehicle_attitude), attitude_sub, &att);
|
|
|
|
orb_copy(ORB_ID(rc_channels), rc_sub, &rc);
|
|
|
|
|
|
|
|
/* scaling factors are defined by the data from the APM Planner
|
|
|
|
* TODO: ifdef for other parameters (HIL/Real world switch)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* position values*/
|
|
|
|
plane_data.lat = global_pos.lat / 10000000;
|
|
|
|
plane_data.lon = global_pos.lon / 10000000;
|
|
|
|
plane_data.alt = global_pos.alt / 1000;
|
|
|
|
plane_data.vx = global_pos.vx / 100;
|
|
|
|
plane_data.vy = global_pos.vy / 100;
|
|
|
|
plane_data.vz = global_pos.vz / 100;
|
|
|
|
|
|
|
|
/* attitude values*/
|
|
|
|
plane_data.roll = att.roll;
|
|
|
|
plane_data.pitch = att.pitch;
|
|
|
|
plane_data.yaw = att.yaw;
|
|
|
|
plane_data.rollspeed = att.rollspeed;
|
|
|
|
plane_data.pitchspeed = att.pitchspeed;
|
|
|
|
plane_data.yawspeed = att.yawspeed;
|
|
|
|
|
|
|
|
/* parameter values */
|
|
|
|
get_parameters();
|
|
|
|
|
|
|
|
/* Attitude control part */
|
|
|
|
|
|
|
|
//#define MUTE
|
|
|
|
#ifndef MUTE
|
|
|
|
/******************************** DEBUG OUTPUT ************************************************************/
|
|
|
|
|
|
|
|
printf("Parameter: %i,%i,%i,%i,%i,%i,%i,%i,%i,%i,%i,%i \n", (int)plane_data.Kp_att, (int)plane_data.Ki_att,
|
|
|
|
(int)plane_data.Kd_att, (int)plane_data.intmax_att, (int)plane_data.Kp_pos, (int)plane_data.Ki_pos,
|
|
|
|
(int)plane_data.Kd_pos, (int)plane_data.intmax_pos, (int)plane_data.airspeed,
|
|
|
|
(int)plane_data.wp_x, (int)plane_data.wp_y, (int)plane_data.wp_z);
|
|
|
|
|
|
|
|
// printf("PITCH SETPOINT: %i\n", (int)plane_data.pitch_setpoint);
|
|
|
|
// printf("ROLL SETPOINT: %i\n", (int)plane_data.roll_setpoint);
|
|
|
|
// printf("THROTTLE SETPOINT: %i\n", (int)calc_throttle_setpoint());
|
|
|
|
|
|
|
|
// printf("\n\nVx: %i\t Vy: %i\t Current speed:%i\n\n", (int)plane_data.vx, (int)plane_data.vy, (int)(calc_gnd_speed()));
|
|
|
|
|
|
|
|
// printf("Current Altitude: %i\n\n", (int)plane_data.alt);
|
|
|
|
|
|
|
|
printf("\nAttitude values: \n R:%i \n P: %i \n Y: %i \n\n RS: %i \n PS: %i \n YS: %i \n ",
|
|
|
|
(int)(1000 * plane_data.roll), (int)(1000 * plane_data.pitch), (int)(1000 * plane_data.yaw),
|
|
|
|
(int)(100 * plane_data.rollspeed), (int)(100 * plane_data.pitchspeed), (int)(100 * plane_data.yawspeed));
|
|
|
|
|
|
|
|
// printf("\nBody Rates: \n P: %i \n Q: %i \n R: %i \n",
|
|
|
|
// (int)(10000 * plane_data.p), (int)(10000 * plane_data.q), (int)(10000 * plane_data.r));
|
|
|
|
|
|
|
|
printf("\nCalculated outputs: \n R: %i\n P: %i\n Y: %i\n T: %i \n",
|
|
|
|
(int)(10000 * control_outputs.roll_ailerons), (int)(10000 * control_outputs.pitch_elevator),
|
|
|
|
(int)(10000 * control_outputs.yaw_rudder), (int)(10000 * control_outputs.throttle));
|
|
|
|
|
|
|
|
/************************************************************************************************************/
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Computation section
|
|
|
|
*
|
|
|
|
* The function calls to compute the required control values happen
|
|
|
|
* in this section.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Set plane mode */
|
|
|
|
set_plane_mode();
|
|
|
|
|
|
|
|
/* Calculate the P,Q,R body rates of the aircraft */
|
|
|
|
calc_body_angular_rates(plane_data.roll / RAD2DEG, plane_data.pitch / RAD2DEG, plane_data.yaw / RAD2DEG,
|
|
|
|
plane_data.rollspeed, plane_data.pitchspeed, plane_data.yawspeed);
|
|
|
|
|
|
|
|
/* Calculate the body frame angles of the aircraft */
|
|
|
|
//calc_bodyframe_angles(plane_data.roll/RAD2DEG,plane_data.pitch/RAD2DEG,plane_data.yaw/RAD2DEG);
|
|
|
|
|
|
|
|
/* Calculate the output values */
|
|
|
|
control_outputs.roll_ailerons = calc_roll_ail();
|
|
|
|
control_outputs.pitch_elevator = calc_pitch_elev();
|
|
|
|
//control_outputs.yaw_rudder = calc_yaw_rudder();
|
|
|
|
control_outputs.throttle = calc_throttle();
|
|
|
|
|
|
|
|
if (rc.chan[rc.function[OVERRIDE]].scale < MANUAL) { // if we're flying in automated mode
|
|
|
|
|
|
|
|
if (plane_data.mode == TAKEOFF) {
|
|
|
|
control.attitude_control_output[ROLL] = 0;
|
|
|
|
control.attitude_control_output[PITCH] = 5000;
|
|
|
|
control.attitude_control_output[THROTTLE] = 10000;
|
|
|
|
//global_data_fixedwing_control->attitude_control_output[YAW] = (int16_t)(control_outputs.yaw_rudder);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (plane_data.mode == CRUISE) {
|
|
|
|
control.attitude_control_output[ROLL] = (int16_t)(10000 * control_outputs.roll_ailerons);
|
|
|
|
control.attitude_control_output[PITCH] = (int16_t)(10000 * control_outputs.pitch_elevator);
|
|
|
|
control.attitude_control_output[THROTTLE] = (int16_t)(10000 * control_outputs.throttle);
|
|
|
|
//control->attitude_control_output[YAW] = (int16_t)(control_outputs.yaw_rudder);
|
|
|
|
}
|
|
|
|
|
|
|
|
control.counter++;
|
|
|
|
control.timestamp = hrt_absolute_time();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Navigation part */
|
|
|
|
|
|
|
|
// Get GPS Waypoint
|
|
|
|
|
|
|
|
// if(global_data_wait(&global_data_position_setpoint->access_conf) == 0)
|
|
|
|
// {
|
|
|
|
// plane_data.wp_x = global_data_position_setpoint->x;
|
|
|
|
// plane_data.wp_y = global_data_position_setpoint->y;
|
|
|
|
// plane_data.wp_z = global_data_position_setpoint->z;
|
|
|
|
// }
|
|
|
|
// global_data_unlock(&global_data_position_setpoint->access_conf);
|
|
|
|
|
|
|
|
if (rc.chan[rc.function[OVERRIDE]].scale < MANUAL) { // AUTO mode
|
|
|
|
// AUTO/HYBRID switch
|
|
|
|
|
|
|
|
if (rc.chan[rc.function[OVERRIDE]].scale < AUTO) {
|
|
|
|
plane_data.roll_setpoint = calc_roll_setpoint();
|
|
|
|
plane_data.pitch_setpoint = calc_pitch_setpoint();
|
|
|
|
plane_data.throttle_setpoint = calc_throttle_setpoint();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
plane_data.roll_setpoint = rc.chan[rc.function[ROLL]].scale / 200;
|
|
|
|
plane_data.pitch_setpoint = rc.chan[rc.function[PITCH]].scale / 200;
|
|
|
|
plane_data.throttle_setpoint = rc.chan[rc.function[THROTTLE]].scale / 200;
|
|
|
|
}
|
|
|
|
|
|
|
|
//control_outputs.yaw_rudder = calc_yaw_rudder(plane_data.hdg);
|
|
|
|
|
|
|
|
// 10 Hz loop
|
|
|
|
usleep(100000);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
control.attitude_control_output[ROLL] = rc.chan[rc.function[ROLL]].scale;
|
|
|
|
control.attitude_control_output[PITCH] = rc.chan[rc.function[PITCH]].scale;
|
|
|
|
control.attitude_control_output[THROTTLE] = rc.chan[rc.function[THROTTLE]].scale;
|
|
|
|
// since we don't have a yaw rudder
|
|
|
|
//control->attitude_control_output[3] = global_data_rc_channels->chan[YAW].scale;
|
|
|
|
|
|
|
|
control.counter++;
|
|
|
|
control.timestamp = hrt_absolute_time();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* publish the control data */
|
|
|
|
|
|
|
|
orb_publish(ORB_ID(fixedwing_control), fixedwing_control_pub, &control);
|
|
|
|
|
|
|
|
/* Servo part */
|
|
|
|
|
|
|
|
buffer_rc[ROLL] = control.attitude_control_output[ROLL];
|
|
|
|
buffer_rc[PITCH] = control.attitude_control_output[PITCH];
|
|
|
|
buffer_rc[THROTTLE] = control.attitude_control_output[THROTTLE];
|
|
|
|
|
|
|
|
//mix_and_link(mixers, 3, 2, &mixer_buffer);
|
|
|
|
|
|
|
|
// Scaling and saturation of servo outputs happens here
|
|
|
|
|
|
|
|
data[AIL_1] = buffer_servo[AIL_1] / global_data_parameter_storage->pm.param_values[PARAM_SERVO_SCALE]
|
|
|
|
+ global_data_parameter_storage->pm.param_values[PARAM_SERVO1_TRIM];
|
|
|
|
|
|
|
|
if (data[AIL_1] > SERVO_MAX)
|
|
|
|
data[AIL_1] = SERVO_MAX;
|
|
|
|
|
|
|
|
if (data[AIL_1] < SERVO_MIN)
|
|
|
|
data[AIL_1] = SERVO_MIN;
|
|
|
|
|
|
|
|
data[AIL_2] = buffer_servo[AIL_2] / global_data_parameter_storage->pm.param_values[PARAM_SERVO_SCALE]
|
|
|
|
+ global_data_parameter_storage->pm.param_values[PARAM_SERVO2_TRIM];
|
|
|
|
|
|
|
|
if (data[AIL_2] > SERVO_MAX)
|
|
|
|
data[AIL_2] = SERVO_MAX;
|
|
|
|
|
|
|
|
if (data[AIL_2] < SERVO_MIN)
|
|
|
|
data[AIL_2] = SERVO_MIN;
|
|
|
|
|
|
|
|
data[MOT] = buffer_servo[MOT] / global_data_parameter_storage->pm.param_values[PARAM_SERVO_SCALE]
|
|
|
|
+ global_data_parameter_storage->pm.param_values[PARAM_SERVO3_TRIM];
|
|
|
|
|
|
|
|
if (data[MOT] > SERVO_MAX)
|
|
|
|
data[MOT] = SERVO_MAX;
|
|
|
|
|
|
|
|
if (data[MOT] < SERVO_MIN)
|
|
|
|
data[MOT] = SERVO_MIN;
|
|
|
|
|
|
|
|
int result = write(fd, &data, sizeof(data));
|
|
|
|
|
|
|
|
if (result != sizeof(data)) {
|
|
|
|
printf("failed writing servo outputs\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 20Hz loop*/
|
|
|
|
usleep(50000);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|