253 lines
13 KiB
C++
253 lines
13 KiB
C++
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
#ifndef AC_POSCONTROL_H
|
|
#define AC_POSCONTROL_H
|
|
|
|
#include <inttypes.h>
|
|
#include <AP_Common.h>
|
|
#include <AP_Param.h>
|
|
#include <AP_Math.h>
|
|
#include <AC_PID.h> // PID library
|
|
#include <APM_PI.h> // PID library
|
|
#include <AP_InertialNav.h> // Inertial Navigation library
|
|
#include <AC_AttitudeControl.h> // Attitude control library
|
|
#include <AP_Motors.h> // motors library
|
|
|
|
|
|
// position controller default definitions
|
|
#define POSCONTROL_THROTTLE_HOVER 450.0f // default throttle required to maintain hover
|
|
#define POSCONTROL_LEASH_Z 750.0f // leash length for z-axis altitude controller. To-Do: replace this with calculation based on alt_pos.kP()?
|
|
#define POSCONTROL_ACCELERATION 100.0f // defines the default velocity vs distant curve. maximum acceleration in cm/s/s that position controller asks for from acceleration controller
|
|
#define POSCONTROL_ACCELERATION_MIN 50.0f // minimum acceleration in cm/s/s - used for sanity checking _wp_accel parameter
|
|
#define POSCONTROL_ACCEL_XY_MAX 980.0f // max horizontal acceleration in cm/s/s that the position velocity controller will ask from the lower accel controller
|
|
#define POSCONTROL_STOPPING_DIST_Z_MAX 200.0f // max stopping distance vertically
|
|
// should be 1.5 times larger than POSCONTROL_ACCELERATION.
|
|
// max acceleration = max lean angle * 980 * pi / 180. i.e. 23deg * 980 * 3.141 / 180 = 393 cm/s/s
|
|
#define POSCONTROL_TAKEOFF_JUMP_CM 20.0f // during take-off altitude target is set to current altitude + this value
|
|
|
|
#define POSCONTROL_SPEED 500.0f // maximum default loiter speed in cm/s
|
|
#define POSCONTROL_VEL_Z_MIN -150.0f // default minimum climb velocity (i.e. max descent rate). To-Do: subtract 250 from this?
|
|
#define POSCONTROL_VEL_Z_MAX 250.0f // default maximum climb velocity. To-Do: add 250 to this?
|
|
#define POSCONTROL_SPEED_MAX_TO_CORRECT_ERROR 200.0f // maximum speed used to correct position error (i.e. not including feed forward)
|
|
|
|
#define POSCONTROL_ALT_HOLD_ACCEL_MAX 250.0f // hard coded copy of throttle controller's maximum acceleration in cm/s. To-Do: remove duplication with throttle controller definition
|
|
|
|
#define POSCONTROL_LEASH_LENGTH_MIN 100.0f // minimum leash lengths in cm
|
|
|
|
#define POSCONTROL_DT_10HZ 0.10f // time difference in seconds for 10hz update rate
|
|
|
|
class AC_PosControl
|
|
{
|
|
public:
|
|
|
|
/// Constructor
|
|
AC_PosControl(const AP_AHRS& ahrs, const AP_InertialNav& inav,
|
|
const AP_Motors& motors, AC_AttitudeControl& attitude_control,
|
|
APM_PI& pi_alt_pos, AC_PID& pid_alt_rate, AC_PID& pid_alt_accel,
|
|
APM_PI& pi_pos_lat, APM_PI& pi_pos_lon, AC_PID& pid_rate_lat, AC_PID& pid_rate_lon);
|
|
|
|
///
|
|
/// initialisation functions
|
|
///
|
|
|
|
/// set_dt - sets time delta in seconds for all controllers (i.e. 100hz = 0.01, 400hz = 0.0025)
|
|
void set_dt(float delta_sec) { _dt = delta_sec; }
|
|
float get_dt() const { return _dt; }
|
|
|
|
/// set_alt_max - sets maximum altitude above home in cm
|
|
/// set to zero to disable limit
|
|
/// To-Do: update this intermittantly from main code after checking if fence is enabled/disabled
|
|
void set_alt_max(float alt) { _alt_max = alt; }
|
|
|
|
/// set_vel_z_limits - sets maximum climb and descent rates
|
|
/// To-Do: call this in the main code as part of flight mode initialisation
|
|
void set_vel_z_limits(float vel_min, float vel_max) { _vel_z_min = vel_min; _vel_z_max = vel_max;}
|
|
|
|
///
|
|
/// z position controller
|
|
///
|
|
|
|
void set_alt_target(float alt_cm) { _pos_target.z = alt_cm; }
|
|
float get_alt_target() const { return _pos_target.z; }
|
|
/// get_desired_alt - get desired altitude (in cm above home) from loiter or wp controller which should be fed into throttle controller
|
|
float get_desired_alt() const { return _pos_target.z; }
|
|
|
|
/// set_target_to_stopping_point_z - sets altitude target to reasonable stopping altitude in cm above home
|
|
void set_target_to_stopping_point_z();
|
|
|
|
/// init_takeoff - initialises target altitude if we are taking off
|
|
void init_takeoff();
|
|
|
|
/// fly_to_z - fly to altitude in cm above home
|
|
void fly_to_target_z();
|
|
|
|
/// climb_at_rate - climb at rate provided in cm/s
|
|
void climb_at_rate(const float rate_target_cms);
|
|
|
|
/*
|
|
///
|
|
/// xy position controller
|
|
///
|
|
|
|
/// get_pos_target - get target as position vector (from home in cm)
|
|
const Vector3f &get_pos_target() const { return _target; }
|
|
|
|
/// set_pos_target in cm from home
|
|
void set_pos_target(const Vector3f& position);
|
|
|
|
/// init_pos_target - set initial loiter target based on current position and velocity
|
|
void init_pos_target(const Vector3f& position, const Vector3f& velocity);
|
|
|
|
/// get_distance_to_target - get horizontal distance to loiter target in cm
|
|
float get_distance_to_target() const;
|
|
|
|
/// get_bearing_to_target - get bearing to loiter target in centi-degrees
|
|
int32_t get_bearing_to_target() const;
|
|
|
|
/// update_loiter - run the loiter controller - should be called at 10hz
|
|
void update_pos_controller();
|
|
|
|
/// get_stopping_point - returns vector to stopping point based on a horizontal position and velocity
|
|
void get_stopping_point(const Vector3f& position, const Vector3f& velocity, Vector3f &target) const;
|
|
|
|
///
|
|
/// shared methods
|
|
///
|
|
|
|
/// get desired roll, pitch which should be fed into stabilize controllers
|
|
int32_t get_desired_roll() const { return _desired_roll; };
|
|
int32_t get_desired_pitch() const { return _desired_pitch; };
|
|
*/
|
|
|
|
/// set_cos_sin_yaw - short-cut to save on calculations to convert from roll-pitch frame to lat-lon frame
|
|
void set_cos_sin_yaw(float cos_yaw, float sin_yaw, float cos_pitch) {
|
|
_cos_yaw = cos_yaw;
|
|
_sin_yaw = sin_yaw;
|
|
_cos_pitch = cos_pitch;
|
|
}
|
|
|
|
// set_throttle_hover - update estimated throttle required to maintain hover
|
|
void set_throttle_hover(float throttle) { _throttle_hover = throttle; }
|
|
|
|
/*
|
|
/// set_horizontal_velocity - allows main code to pass target horizontal velocity for wp navigation
|
|
void set_horizontal_velocity(float velocity_cms) { _wp_speed_cms = velocity_cms; };
|
|
|
|
/// get_horizontal_velocity - allows main code to retrieve target horizontal velocity for wp navigation
|
|
float get_horizontal_velocity() { return _wp_speed_cms; };
|
|
|
|
/// get_climb_velocity - returns target climb speed in cm/s during missions
|
|
float get_climb_velocity() const { return _wp_speed_up_cms; };
|
|
|
|
/// get_descent_velocity - returns target descent speed in cm/s during missions. Note: always positive
|
|
float get_descent_velocity() const { return _wp_speed_down_cms; };
|
|
|
|
/// get_waypoint_radius - access for waypoint radius in cm
|
|
float get_waypoint_radius() const { return _wp_radius_cm; }
|
|
|
|
/// get_waypoint_acceleration - returns acceleration in cm/s/s during missions
|
|
float get_waypoint_acceleration() const { return _wp_accel_cms.get(); }
|
|
*/
|
|
|
|
static const struct AP_Param::GroupInfo var_info[];
|
|
|
|
private:
|
|
// flags structure
|
|
struct poscontroller_flags {
|
|
uint8_t pos_up : 1; // 1 if we have hit the vertical position leash limit while going up
|
|
uint8_t pos_down : 1; // 1 if we have hit the vertical position leash limit while going down
|
|
uint8_t vel_up : 1; // 1 if we have hit the vertical velocity limit going up
|
|
uint8_t vel_down : 1; // 1 if we have hit the vertical velocity limit going down
|
|
} _limit;
|
|
|
|
// pos_to_rate_z - position to rate controller for Z axis
|
|
// target altitude should be placed into _pos_target.z using or set with one of these functions
|
|
// set_alt_target
|
|
// set_target_to_stopping_point_z
|
|
// init_takeoff
|
|
void pos_to_rate_z();
|
|
|
|
// rate_to_accel_z - calculates desired accel required to achieve the velocity target
|
|
void rate_to_accel_z(float vel_target_z);
|
|
|
|
// accel_to_throttle - alt hold's acceleration controller
|
|
void accel_to_throttle(float accel_target_z);
|
|
|
|
/*
|
|
/// get_loiter_position_to_velocity - loiter position controller
|
|
/// converts desired position held in _target vector to desired velocity
|
|
void get_position_to_velocity(float dt, float max_speed_cms);
|
|
|
|
/// get_loiter_velocity_to_acceleration - loiter velocity controller
|
|
/// converts desired velocities in lat/lon directions to accelerations in lat/lon frame
|
|
void get_velocity_to_acceleration(float vel_lat_cms, float vel_lon_cms, float dt);
|
|
|
|
/// get_loiter_acceleration_to_lean_angles - loiter acceleration controller
|
|
/// converts desired accelerations provided in lat/lon frame to roll/pitch angles
|
|
void get_acceleration_to_lean_angles(float accel_lat_cmss, float accel_lon_cmss);
|
|
|
|
/// get_bearing_cd - return bearing in centi-degrees between two positions
|
|
float get_bearing_cd(const Vector3f &origin, const Vector3f &destination) const;
|
|
|
|
/// reset_I - clears I terms from position PID controller
|
|
void reset_I();
|
|
|
|
/// calculate_leash_length - calculates the maximum distance in cm that the target position may be from the current location
|
|
void calculate_leash_length();
|
|
*/
|
|
|
|
// references to inertial nav and ahrs libraries
|
|
const AP_AHRS& _ahrs;
|
|
const AP_InertialNav& _inav;
|
|
const AP_Motors& _motors;
|
|
AC_AttitudeControl& _attitude_control;
|
|
|
|
// references to pid controllers and motors
|
|
APM_PI& _pi_alt_pos;
|
|
AC_PID& _pid_alt_rate;
|
|
AC_PID& _pid_alt_accel;
|
|
APM_PI& _pi_pos_lat;
|
|
APM_PI& _pi_pos_lon;
|
|
AC_PID& _pid_rate_lat;
|
|
AC_PID& _pid_rate_lon;
|
|
|
|
// parameters
|
|
AP_Float _throttle_hover; // estimated throttle required to maintain a level hover
|
|
|
|
// internal variables
|
|
float _dt; // time difference (in seconds) between calls from the main program
|
|
uint32_t _last_update_ms; // system time of last update_position_controller call
|
|
uint32_t _last_update_rate_ms; // system time of last call to rate_to_accel_z (alt hold accel controller)
|
|
uint32_t _last_update_accel_ms; // system time of last call to accel_to_throttle (alt hold accel controller)
|
|
uint8_t _step; // used to decide which portion of position controller to run during this iteration
|
|
float _speed_cms; // max horizontal speed in cm/s
|
|
float _vel_z_min; // min climb rate (i.e. max descent rate) in cm/s
|
|
float _vel_z_max; // max climb rate in cm/s
|
|
float _accel_cms; // max horizontal acceleration in cm/s/s
|
|
float _cos_yaw; // short-cut to save on calcs required to convert roll-pitch frame to lat-lon frame
|
|
float _sin_yaw;
|
|
float _cos_pitch;
|
|
|
|
// output from controller
|
|
float _desired_roll; // desired roll angle in centi-degrees calculated by position controller
|
|
float _desired_pitch; // desired roll pitch in centi-degrees calculated by position controller
|
|
|
|
// position controller internal variables
|
|
Vector3f _pos_target; // target location in cm from home
|
|
Vector3f _pos_error; // error between desired and actual position in cm
|
|
Vector3f _vel_target; // desired velocity in cm/s
|
|
Vector3f _vel_error; // error between desired and actual acceleration in cm/s. To-Do: x & y actually used?
|
|
Vector3f _vel_last; // previous iterations velocity in cm/s
|
|
float _vel_target_filt_z; // filtered target vertical velocity
|
|
Vector3f _accel_target; // desired acceleration in cm/s/s // To-Do: are xy actually required?
|
|
Vector3f _accel_error; // desired acceleration in cm/s/s // To-Do: are xy actually required?
|
|
float _leash; // horizontal leash length in cm. used to stop the pilot from pushing the target location too far from the current location
|
|
float _alt_max; // max altitude - should be updated from the main code with altitude limit from fence
|
|
|
|
public:
|
|
// for logging purposes
|
|
Vector2f dist_error; // distance error calculated by loiter controller
|
|
Vector2f desired_vel; // loiter controller desired velocity
|
|
Vector2f desired_accel; // the resulting desired acceleration
|
|
};
|
|
#endif // AC_POSCONTROL_H
|