2014-05-19 18:35:18 +02:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* @addtogroup OpenPilotLibraries OpenPilot Libraries
|
|
|
|
* @{
|
|
|
|
* @addtogroup Navigation
|
|
|
|
* @brief setups RTH/PH and other pathfollower/pathplanner status
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file plan.c
|
|
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2014.
|
|
|
|
*
|
|
|
|
* @see The GNU Public License (GPL) Version 3
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
/*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
|
|
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <plans.h>
|
2014-05-19 21:15:17 +02:00
|
|
|
#include <openpilot.h>
|
2014-05-19 18:35:18 +02:00
|
|
|
#include <attitudesettings.h>
|
|
|
|
#include <takeofflocation.h>
|
|
|
|
#include <pathdesired.h>
|
|
|
|
#include <positionstate.h>
|
|
|
|
#include <flightmodesettings.h>
|
2015-01-08 13:20:47 +11:00
|
|
|
#include <flightstatus.h>
|
|
|
|
#include <velocitystate.h>
|
2014-06-04 16:52:39 +02:00
|
|
|
#include <manualcontrolcommand.h>
|
|
|
|
#include <attitudestate.h>
|
2015-01-08 13:20:47 +11:00
|
|
|
#include <vtolpathfollowersettings.h>
|
2014-06-04 16:52:39 +02:00
|
|
|
#include <sin_lookup.h>
|
|
|
|
|
|
|
|
#define UPDATE_EXPECTED 0.02f
|
|
|
|
#define UPDATE_MIN 1.0e-6f
|
|
|
|
#define UPDATE_MAX 1.0f
|
|
|
|
#define UPDATE_ALPHA 1.0e-2f
|
2014-05-19 18:35:18 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief initialize UAVOs and structs used by this library
|
|
|
|
*/
|
|
|
|
void plan_initialize()
|
|
|
|
{
|
|
|
|
TakeOffLocationInitialize();
|
|
|
|
PositionStateInitialize();
|
|
|
|
PathDesiredInitialize();
|
|
|
|
FlightModeSettingsInitialize();
|
2015-01-08 13:20:47 +11:00
|
|
|
FlightStatusInitialize();
|
2014-06-04 16:52:39 +02:00
|
|
|
AttitudeStateInitialize();
|
|
|
|
ManualControlCommandInitialize();
|
2015-01-08 13:20:47 +11:00
|
|
|
VelocityStateInitialize();
|
|
|
|
VtolPathFollowerSettingsInitialize();
|
2014-05-19 18:35:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief setup pathplanner/pathfollower for positionhold
|
|
|
|
*/
|
|
|
|
void plan_setup_positionHold()
|
|
|
|
{
|
|
|
|
PositionStateData positionState;
|
|
|
|
|
|
|
|
PositionStateGet(&positionState);
|
|
|
|
PathDesiredData pathDesired;
|
|
|
|
PathDesiredGet(&pathDesired);
|
|
|
|
|
2014-06-07 23:06:11 +02:00
|
|
|
FlightModeSettingsPositionHoldOffsetData offset;
|
|
|
|
FlightModeSettingsPositionHoldOffsetGet(&offset);
|
2014-06-04 16:52:39 +02:00
|
|
|
|
2014-05-19 18:35:18 +02:00
|
|
|
pathDesired.End.North = positionState.North;
|
|
|
|
pathDesired.End.East = positionState.East;
|
|
|
|
pathDesired.End.Down = positionState.Down;
|
2014-06-07 23:06:11 +02:00
|
|
|
pathDesired.Start.North = positionState.North + offset.Horizontal; // in FlyEndPoint the direction of this vector does not matter
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.Start.East = positionState.East;
|
|
|
|
pathDesired.Start.Down = positionState.Down;
|
2014-08-15 17:56:57 +02:00
|
|
|
pathDesired.StartingVelocity = 0.0f;
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.EndingVelocity = 0.0f;
|
2014-05-19 18:35:18 +02:00
|
|
|
pathDesired.Mode = PATHDESIRED_MODE_FLYENDPOINT;
|
|
|
|
|
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
}
|
|
|
|
|
2015-01-08 13:20:47 +11:00
|
|
|
|
2014-05-19 18:35:18 +02:00
|
|
|
/**
|
|
|
|
* @brief setup pathplanner/pathfollower for return to base
|
|
|
|
*/
|
|
|
|
void plan_setup_returnToBase()
|
|
|
|
{
|
|
|
|
// Simple Return To Base mode - keep altitude the same applying configured delta, fly to takeoff position
|
|
|
|
float positionStateDown;
|
|
|
|
|
|
|
|
PositionStateDownGet(&positionStateDown);
|
|
|
|
|
|
|
|
PathDesiredData pathDesired;
|
|
|
|
PathDesiredGet(&pathDesired);
|
|
|
|
|
|
|
|
TakeOffLocationData takeoffLocation;
|
|
|
|
TakeOffLocationGet(&takeoffLocation);
|
|
|
|
|
|
|
|
// TODO: right now VTOLPF does fly straight to destination altitude.
|
|
|
|
// For a safer RTB destination altitude will be the higher between takeofflocation and current position (corrected with safety margin)
|
|
|
|
|
|
|
|
float destDown;
|
|
|
|
FlightModeSettingsReturnToBaseAltitudeOffsetGet(&destDown);
|
2014-05-19 22:21:32 +02:00
|
|
|
destDown = MIN(positionStateDown, takeoffLocation.Down) - destDown;
|
2014-06-07 23:06:11 +02:00
|
|
|
FlightModeSettingsPositionHoldOffsetData offset;
|
|
|
|
FlightModeSettingsPositionHoldOffsetGet(&offset);
|
2014-05-19 18:35:18 +02:00
|
|
|
|
|
|
|
pathDesired.End.North = takeoffLocation.North;
|
|
|
|
pathDesired.End.East = takeoffLocation.East;
|
|
|
|
pathDesired.End.Down = destDown;
|
|
|
|
|
2014-06-07 23:06:11 +02:00
|
|
|
pathDesired.Start.North = takeoffLocation.North + offset.Horizontal; // in FlyEndPoint the direction of this vector does not matter
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.Start.East = takeoffLocation.East;
|
|
|
|
pathDesired.Start.Down = destDown;
|
|
|
|
|
2014-08-15 17:56:57 +02:00
|
|
|
pathDesired.StartingVelocity = 0.0f;
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.EndingVelocity = 0.0f;
|
2014-05-19 18:35:18 +02:00
|
|
|
pathDesired.Mode = PATHDESIRED_MODE_FLYENDPOINT;
|
|
|
|
|
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
}
|
|
|
|
|
2014-07-15 09:39:13 +02:00
|
|
|
static PiOSDeltatimeConfig landdT;
|
2014-05-19 18:35:18 +02:00
|
|
|
void plan_setup_land()
|
|
|
|
{
|
2014-08-19 23:14:46 +02:00
|
|
|
float descendspeed;
|
|
|
|
|
2014-05-19 18:35:18 +02:00
|
|
|
plan_setup_positionHold();
|
2014-07-15 09:39:13 +02:00
|
|
|
|
2014-08-19 23:14:46 +02:00
|
|
|
FlightModeSettingsLandingVelocityGet(&descendspeed);
|
2014-07-15 09:39:13 +02:00
|
|
|
PathDesiredData pathDesired;
|
|
|
|
PathDesiredGet(&pathDesired);
|
2014-08-19 23:14:46 +02:00
|
|
|
pathDesired.StartingVelocity = descendspeed;
|
|
|
|
pathDesired.EndingVelocity = descendspeed;
|
2014-07-15 09:39:13 +02:00
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
PIOS_DELTATIME_Init(&landdT, UPDATE_EXPECTED, UPDATE_MIN, UPDATE_MAX, UPDATE_ALPHA);
|
2014-05-19 18:35:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief execute land
|
|
|
|
*/
|
|
|
|
void plan_run_land()
|
|
|
|
{
|
2014-07-15 09:39:13 +02:00
|
|
|
float downPos, descendspeed;
|
2014-05-19 18:35:18 +02:00
|
|
|
PathDesiredEndData pathDesiredEnd;
|
|
|
|
|
2014-07-15 09:39:13 +02:00
|
|
|
PositionStateDownGet(&downPos); // current down position
|
|
|
|
PathDesiredEndGet(&pathDesiredEnd); // desired position
|
|
|
|
PathDesiredEndingVelocityGet(&descendspeed);
|
2014-05-19 18:35:18 +02:00
|
|
|
|
2014-07-15 09:39:13 +02:00
|
|
|
// desired position is updated to match the desired descend speed but don't run ahead
|
|
|
|
// too far if the current position can't keep up. This normaly means we have landed.
|
|
|
|
if (pathDesiredEnd.Down - downPos < 10) {
|
|
|
|
pathDesiredEnd.Down += descendspeed * PIOS_DELTATIME_GetAverageSeconds(&landdT);
|
|
|
|
}
|
2014-05-19 18:35:18 +02:00
|
|
|
|
|
|
|
PathDesiredEndSet(&pathDesiredEnd);
|
|
|
|
}
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief positionvario functionality
|
|
|
|
*/
|
2014-11-08 12:56:17 +01:00
|
|
|
static bool vario_hold = true;
|
2014-06-07 15:54:48 +02:00
|
|
|
static float hold_position[3];
|
2014-11-10 19:06:27 +01:00
|
|
|
static float vario_control_lowpass[3];
|
2014-11-08 13:32:40 +01:00
|
|
|
static float vario_course = 0.0f;
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
static void plan_setup_PositionVario()
|
|
|
|
{
|
|
|
|
vario_hold = true;
|
2014-11-08 13:32:40 +01:00
|
|
|
vario_control_lowpass[0] = 0.0f;
|
|
|
|
vario_control_lowpass[1] = 0.0f;
|
|
|
|
vario_control_lowpass[2] = 0.0f;
|
2014-11-08 12:56:17 +01:00
|
|
|
AttitudeStateYawGet(&vario_course);
|
2014-06-04 16:52:39 +02:00
|
|
|
plan_setup_positionHold();
|
|
|
|
}
|
|
|
|
|
2014-11-08 12:56:17 +01:00
|
|
|
void plan_setup_CourseLock()
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
|
|
|
plan_setup_PositionVario();
|
|
|
|
}
|
|
|
|
|
2014-11-08 16:01:59 +01:00
|
|
|
void plan_setup_PositionRoam()
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
|
|
|
plan_setup_PositionVario();
|
|
|
|
}
|
|
|
|
|
2014-11-08 16:01:59 +01:00
|
|
|
void plan_setup_HomeLeash()
|
2014-11-08 12:56:17 +01:00
|
|
|
{
|
|
|
|
plan_setup_PositionVario();
|
|
|
|
}
|
|
|
|
|
|
|
|
void plan_setup_AbsolutePosition()
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
|
|
|
plan_setup_PositionVario();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-10 09:59:37 +02:00
|
|
|
#define DEADBAND 0.1f
|
2014-06-04 16:52:39 +02:00
|
|
|
static bool normalizeDeadband(float controlVector[4])
|
|
|
|
{
|
|
|
|
bool moving = false;
|
2014-06-10 09:59:37 +02:00
|
|
|
|
2014-06-04 16:52:39 +02:00
|
|
|
// roll, pitch, yaw between -1 and +1
|
2014-06-10 09:59:37 +02:00
|
|
|
// thrust between 0 and 1 mapped to -1 to +1
|
|
|
|
controlVector[3] = (2.0f * controlVector[3]) - 1.0f;
|
2014-06-04 16:52:39 +02:00
|
|
|
int t;
|
|
|
|
|
2014-06-10 09:59:37 +02:00
|
|
|
for (t = 0; t < 4; t++) {
|
|
|
|
if (controlVector[t] < -DEADBAND) {
|
|
|
|
moving = true;
|
|
|
|
controlVector[t] += DEADBAND;
|
|
|
|
} else if (controlVector[t] > DEADBAND) {
|
2014-06-04 16:52:39 +02:00
|
|
|
moving = true;
|
2014-06-10 09:59:37 +02:00
|
|
|
controlVector[t] -= DEADBAND;
|
2014-06-04 16:52:39 +02:00
|
|
|
} else {
|
|
|
|
controlVector[t] = 0.0f;
|
|
|
|
}
|
2014-06-10 09:59:37 +02:00
|
|
|
// deadband has been cut out, scale value back to [-1,+1]
|
|
|
|
controlVector[t] *= (1.0f / (1.0f - DEADBAND));
|
|
|
|
controlVector[t] = boundf(controlVector[t], -1.0f, 1.0f);
|
2014-06-04 16:52:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return moving;
|
|
|
|
}
|
|
|
|
|
2014-11-08 12:56:17 +01:00
|
|
|
typedef enum { COURSE, FPV, LOS, NSEW } vario_type;
|
2014-06-04 16:52:39 +02:00
|
|
|
|
2014-06-07 15:54:48 +02:00
|
|
|
static void getVector(float controlVector[4], vario_type type)
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
2014-06-07 23:06:11 +02:00
|
|
|
FlightModeSettingsPositionHoldOffsetData offset;
|
|
|
|
|
|
|
|
FlightModeSettingsPositionHoldOffsetGet(&offset);
|
|
|
|
|
|
|
|
// scale controlVector[3] (thrust) by vertical/horizontal to have vertical plane less sensitive
|
|
|
|
controlVector[3] *= offset.Vertical / offset.Horizontal;
|
|
|
|
|
2014-06-04 16:52:39 +02:00
|
|
|
float length = sqrtf(controlVector[0] * controlVector[0] + controlVector[1] * controlVector[1] + controlVector[3] * controlVector[3]);
|
|
|
|
|
|
|
|
if (length <= 1e-9f) {
|
|
|
|
length = 1.0f; // should never happen as getVector is not called if control within deadband
|
|
|
|
}
|
|
|
|
{
|
|
|
|
float direction[3] = {
|
|
|
|
controlVector[1] / length, // pitch is north
|
|
|
|
controlVector[0] / length, // roll is east
|
|
|
|
controlVector[3] / length // thrust is down
|
|
|
|
};
|
|
|
|
controlVector[0] = direction[0];
|
|
|
|
controlVector[1] = direction[1];
|
|
|
|
controlVector[2] = direction[2];
|
|
|
|
}
|
2014-06-07 23:06:11 +02:00
|
|
|
controlVector[3] = length * offset.Horizontal;
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
// rotate north and east - rotation angle based on type
|
|
|
|
float angle;
|
|
|
|
switch (type) {
|
2014-11-08 12:56:17 +01:00
|
|
|
case COURSE:
|
|
|
|
angle = vario_course;
|
|
|
|
break;
|
2014-06-04 16:52:39 +02:00
|
|
|
case NSEW:
|
|
|
|
angle = 0.0f;
|
|
|
|
// NSEW no rotation takes place
|
|
|
|
break;
|
|
|
|
case FPV:
|
|
|
|
// local rotation, using current yaw
|
|
|
|
AttitudeStateYawGet(&angle);
|
|
|
|
break;
|
|
|
|
case LOS:
|
|
|
|
// determine location based on vector from takeoff to current location
|
|
|
|
{
|
|
|
|
PositionStateData positionState;
|
|
|
|
PositionStateGet(&positionState);
|
|
|
|
TakeOffLocationData takeoffLocation;
|
|
|
|
TakeOffLocationGet(&takeoffLocation);
|
|
|
|
angle = RAD2DEG(atan2f(positionState.East - takeoffLocation.East, positionState.North - takeoffLocation.North));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// rotate horizontally by angle
|
|
|
|
{
|
|
|
|
float rotated[2] = {
|
|
|
|
controlVector[0] * cos_lookup_deg(angle) - controlVector[1] * sin_lookup_deg(angle),
|
|
|
|
controlVector[0] * sin_lookup_deg(angle) + controlVector[1] * cos_lookup_deg(angle)
|
|
|
|
};
|
|
|
|
controlVector[0] = rotated[0];
|
|
|
|
controlVector[1] = rotated[1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void plan_run_PositionVario(vario_type type)
|
|
|
|
{
|
2014-06-07 15:54:48 +02:00
|
|
|
float controlVector[4];
|
2014-11-08 13:32:40 +01:00
|
|
|
float alpha;
|
2014-06-04 23:22:04 +02:00
|
|
|
PathDesiredData pathDesired;
|
|
|
|
|
|
|
|
PathDesiredGet(&pathDesired);
|
2014-06-07 23:06:11 +02:00
|
|
|
FlightModeSettingsPositionHoldOffsetData offset;
|
|
|
|
FlightModeSettingsPositionHoldOffsetGet(&offset);
|
2014-06-07 15:54:48 +02:00
|
|
|
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
ManualControlCommandRollGet(&controlVector[0]);
|
|
|
|
ManualControlCommandPitchGet(&controlVector[1]);
|
|
|
|
ManualControlCommandYawGet(&controlVector[2]);
|
|
|
|
ManualControlCommandThrustGet(&controlVector[3]);
|
2014-06-04 23:22:04 +02:00
|
|
|
|
2014-11-08 13:32:40 +01:00
|
|
|
|
|
|
|
FlightModeSettingsVarioControlLowPassAlphaGet(&alpha);
|
|
|
|
vario_control_lowpass[0] = alpha * vario_control_lowpass[0] + (1.0f - alpha) * controlVector[0];
|
|
|
|
vario_control_lowpass[1] = alpha * vario_control_lowpass[1] + (1.0f - alpha) * controlVector[1];
|
|
|
|
vario_control_lowpass[2] = alpha * vario_control_lowpass[2] + (1.0f - alpha) * controlVector[2];
|
|
|
|
controlVector[0] = vario_control_lowpass[0];
|
|
|
|
controlVector[1] = vario_control_lowpass[1];
|
|
|
|
controlVector[2] = vario_control_lowpass[2];
|
|
|
|
|
2014-06-04 16:52:39 +02:00
|
|
|
// check if movement is desired
|
|
|
|
if (normalizeDeadband(controlVector) == false) {
|
2014-06-04 23:22:04 +02:00
|
|
|
// no movement desired, re-enter positionHold at current start-position
|
2014-06-04 16:52:39 +02:00
|
|
|
if (!vario_hold) {
|
|
|
|
vario_hold = true;
|
2014-06-04 23:22:04 +02:00
|
|
|
|
|
|
|
// new hold position is the position that was previously the start position
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.End.North = hold_position[0];
|
|
|
|
pathDesired.End.East = hold_position[1];
|
|
|
|
pathDesired.End.Down = hold_position[2];
|
2014-06-04 23:22:04 +02:00
|
|
|
// while the new start position has the same offset as in position hold
|
2014-06-07 23:06:11 +02:00
|
|
|
pathDesired.Start.North = pathDesired.End.North + offset.Horizontal; // in FlyEndPoint the direction of this vector does not matter
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.Start.East = pathDesired.End.East;
|
|
|
|
pathDesired.Start.Down = pathDesired.End.Down;
|
2014-06-04 23:22:04 +02:00
|
|
|
PathDesiredSet(&pathDesired);
|
2014-06-04 16:52:39 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
PositionStateData positionState;
|
|
|
|
PositionStateGet(&positionState);
|
|
|
|
|
2014-06-07 15:30:22 +02:00
|
|
|
// flip pitch to have pitch down (away) point north
|
|
|
|
controlVector[1] = -controlVector[1];
|
2014-06-04 16:52:39 +02:00
|
|
|
getVector(controlVector, type);
|
|
|
|
|
|
|
|
// layout of control Vector : unitVector in movement direction {0,1,2} vector length {3} velocity {4}
|
|
|
|
if (vario_hold) {
|
2014-06-04 23:22:04 +02:00
|
|
|
// start position is the position that was previously the hold position
|
2014-06-04 16:52:39 +02:00
|
|
|
vario_hold = false;
|
2014-06-07 15:54:48 +02:00
|
|
|
hold_position[0] = pathDesired.End.North;
|
|
|
|
hold_position[1] = pathDesired.End.East;
|
|
|
|
hold_position[2] = pathDesired.End.Down;
|
2014-06-04 16:52:39 +02:00
|
|
|
} else {
|
2014-06-04 23:22:04 +02:00
|
|
|
// start position is advanced according to movement - in the direction of ControlVector only
|
2014-06-04 16:52:39 +02:00
|
|
|
// projection using scalar product
|
2014-06-07 15:54:48 +02:00
|
|
|
float kp = (positionState.North - hold_position[0]) * controlVector[0]
|
|
|
|
+ (positionState.East - hold_position[1]) * controlVector[1]
|
|
|
|
+ (positionState.Down - hold_position[2]) * -controlVector[2];
|
2014-06-04 16:52:39 +02:00
|
|
|
if (kp > 0.0f) {
|
2014-06-07 15:54:48 +02:00
|
|
|
hold_position[0] += kp * controlVector[0];
|
|
|
|
hold_position[1] += kp * controlVector[1];
|
|
|
|
hold_position[2] += kp * -controlVector[2];
|
2014-06-04 16:52:39 +02:00
|
|
|
}
|
|
|
|
}
|
2014-06-07 15:54:48 +02:00
|
|
|
// new destination position is advanced based on controlVector
|
|
|
|
pathDesired.End.North = hold_position[0] + controlVector[0] * controlVector[3];
|
|
|
|
pathDesired.End.East = hold_position[1] + controlVector[1] * controlVector[3];
|
|
|
|
pathDesired.End.Down = hold_position[2] - controlVector[2] * controlVector[3];
|
|
|
|
// the new start position has the same offset as in position hold
|
2014-06-07 23:06:11 +02:00
|
|
|
pathDesired.Start.North = pathDesired.End.North + offset.Horizontal; // in FlyEndPoint the direction of this vector does not matter
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.Start.East = pathDesired.End.East;
|
|
|
|
pathDesired.Start.Down = pathDesired.End.Down;
|
2014-06-04 16:52:39 +02:00
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
}
|
|
|
|
}
|
2014-11-08 12:56:17 +01:00
|
|
|
|
|
|
|
void plan_run_CourseLock()
|
|
|
|
{
|
|
|
|
plan_run_PositionVario(COURSE);
|
|
|
|
}
|
|
|
|
|
2014-11-08 16:01:59 +01:00
|
|
|
void plan_run_PositionRoam()
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
|
|
|
plan_run_PositionVario(FPV);
|
|
|
|
}
|
|
|
|
|
2014-11-08 16:01:59 +01:00
|
|
|
void plan_run_HomeLeash()
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
|
|
|
plan_run_PositionVario(LOS);
|
|
|
|
}
|
|
|
|
|
2014-11-08 12:56:17 +01:00
|
|
|
void plan_run_AbsolutePosition()
|
2014-06-04 16:52:39 +02:00
|
|
|
{
|
|
|
|
plan_run_PositionVario(NSEW);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief setup pathplanner/pathfollower for AutoCruise
|
|
|
|
*/
|
|
|
|
static PiOSDeltatimeConfig actimeval;
|
|
|
|
void plan_setup_AutoCruise()
|
|
|
|
{
|
|
|
|
PositionStateData positionState;
|
|
|
|
|
|
|
|
PositionStateGet(&positionState);
|
|
|
|
PathDesiredData pathDesired;
|
|
|
|
PathDesiredGet(&pathDesired);
|
|
|
|
|
2014-06-07 23:06:11 +02:00
|
|
|
FlightModeSettingsPositionHoldOffsetData offset;
|
|
|
|
FlightModeSettingsPositionHoldOffsetGet(&offset);
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
// initialization is flight in direction of the nose.
|
|
|
|
// the velocity is not relevant, as it will be reset by the run function even during first call
|
|
|
|
float angle;
|
|
|
|
AttitudeStateYawGet(&angle);
|
|
|
|
float vector[2] = {
|
|
|
|
cos_lookup_deg(angle),
|
|
|
|
sin_lookup_deg(angle)
|
|
|
|
};
|
2014-06-07 15:54:48 +02:00
|
|
|
hold_position[0] = positionState.North;
|
|
|
|
hold_position[1] = positionState.East;
|
|
|
|
hold_position[2] = positionState.Down;
|
|
|
|
pathDesired.End.North = hold_position[0] + vector[0];
|
|
|
|
pathDesired.End.East = hold_position[1] + vector[1];
|
|
|
|
pathDesired.End.Down = hold_position[2];
|
|
|
|
// start position has the same offset as in position hold
|
2014-06-07 23:06:11 +02:00
|
|
|
pathDesired.Start.North = pathDesired.End.North + offset.Horizontal; // in FlyEndPoint the direction of this vector does not matter
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.Start.East = pathDesired.End.East;
|
|
|
|
pathDesired.Start.Down = pathDesired.End.Down;
|
2014-08-15 17:56:57 +02:00
|
|
|
pathDesired.StartingVelocity = 0.0f;
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.EndingVelocity = 0.0f;
|
|
|
|
pathDesired.Mode = PATHDESIRED_MODE_FLYENDPOINT;
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
|
|
|
|
// re-iniztializing deltatime is valid and also good practice here since
|
|
|
|
// getAverageSeconds() has not been called/updated in a long time if we were in a different flightmode.
|
|
|
|
PIOS_DELTATIME_Init(&actimeval, UPDATE_EXPECTED, UPDATE_MIN, UPDATE_MAX, UPDATE_ALPHA);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief execute autocruise
|
|
|
|
*/
|
|
|
|
void plan_run_AutoCruise()
|
|
|
|
{
|
|
|
|
PositionStateData positionState;
|
|
|
|
|
|
|
|
PositionStateGet(&positionState);
|
|
|
|
PathDesiredData pathDesired;
|
|
|
|
PathDesiredGet(&pathDesired);
|
2014-06-07 23:06:11 +02:00
|
|
|
FlightModeSettingsPositionHoldOffsetData offset;
|
|
|
|
FlightModeSettingsPositionHoldOffsetGet(&offset);
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
float controlVector[4];
|
|
|
|
ManualControlCommandRollGet(&controlVector[0]);
|
|
|
|
ManualControlCommandPitchGet(&controlVector[1]);
|
|
|
|
ManualControlCommandYawGet(&controlVector[2]);
|
|
|
|
controlVector[3] = 0.5f; // dummy, thrust is normalized separately
|
|
|
|
normalizeDeadband(controlVector); // return value ignored
|
|
|
|
ManualControlCommandThrustGet(&controlVector[3]); // no deadband as we are using thrust for velocity
|
2014-06-07 16:16:55 +02:00
|
|
|
controlVector[3] = boundf(controlVector[3], 1e-6f, 1.0f); // bound to above zero, to prevent loss of vector direction
|
2014-06-04 16:52:39 +02:00
|
|
|
|
|
|
|
// normalize old desired movement vector
|
2014-06-07 15:54:48 +02:00
|
|
|
float vector[3] = { pathDesired.End.North - hold_position[0],
|
|
|
|
pathDesired.End.East - hold_position[1],
|
|
|
|
pathDesired.End.Down - hold_position[2] };
|
2014-06-04 16:52:39 +02:00
|
|
|
float length = sqrtf(vector[0] * vector[0] + vector[1] * vector[1] + vector[2] * vector[2]);
|
|
|
|
if (length < 1e-9f) {
|
|
|
|
length = 1.0f; // should not happen since initialized properly in setup()
|
|
|
|
}
|
|
|
|
vector[0] /= length;
|
|
|
|
vector[1] /= length;
|
|
|
|
vector[2] /= length;
|
|
|
|
|
|
|
|
// start position is advanced according to actual movement - in the direction of desired vector only
|
|
|
|
// projection using scalar product
|
2014-06-07 15:54:48 +02:00
|
|
|
float kp = (positionState.North - hold_position[0]) * vector[0]
|
|
|
|
+ (positionState.East - hold_position[1]) * vector[1]
|
|
|
|
+ (positionState.Down - hold_position[2]) * vector[2];
|
2014-06-04 16:52:39 +02:00
|
|
|
if (kp > 0.0f) {
|
2014-06-07 15:54:48 +02:00
|
|
|
hold_position[0] += kp * vector[0];
|
|
|
|
hold_position[1] += kp * vector[1];
|
|
|
|
hold_position[2] += kp * vector[2];
|
2014-06-04 16:52:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// new angle is equal to old angle plus offset depending on yaw input and time
|
|
|
|
// (controlVector is normalized with a deadband, change is zero within deadband)
|
|
|
|
float angle = RAD2DEG(atan2f(vector[1], vector[0]));
|
|
|
|
float dT = PIOS_DELTATIME_GetAverageSeconds(&actimeval);
|
|
|
|
angle += 10.0f * controlVector[2] * dT; // TODO magic value could eventually end up in a to be created settings
|
|
|
|
|
|
|
|
// resulting movement vector is scaled by velocity demand in controlvector[3] [0.0-1.0]
|
2014-06-07 23:06:11 +02:00
|
|
|
vector[0] = cosf(DEG2RAD(angle)) * offset.Horizontal * controlVector[3];
|
|
|
|
vector[1] = sinf(DEG2RAD(angle)) * offset.Horizontal * controlVector[3];
|
|
|
|
vector[2] = -controlVector[1] * offset.Vertical * controlVector[3];
|
2014-06-04 16:52:39 +02:00
|
|
|
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.End.North = hold_position[0] + vector[0];
|
|
|
|
pathDesired.End.East = hold_position[1] + vector[1];
|
|
|
|
pathDesired.End.Down = hold_position[2] + vector[2];
|
|
|
|
// start position has the same offset as in position hold
|
2014-06-07 23:06:11 +02:00
|
|
|
pathDesired.Start.North = pathDesired.End.North + offset.Horizontal; // in FlyEndPoint the direction of this vector does not matter
|
2014-06-07 15:54:48 +02:00
|
|
|
pathDesired.Start.East = pathDesired.End.East;
|
|
|
|
pathDesired.Start.Down = pathDesired.End.Down;
|
2014-06-04 16:52:39 +02:00
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
}
|
2015-01-08 13:20:47 +11:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief setup pathplanner/pathfollower for braking in positionroam
|
|
|
|
* timeout_occurred = false: Attempt to enter flyvector for braking
|
|
|
|
* timeout_occurred = true: Revert to position hold
|
|
|
|
*/
|
|
|
|
#define ASSISTEDCONTROL_BRAKERATE_MINIMUM 0.2f // m/s2
|
|
|
|
#define ASSISTEDCONTROL_TIMETOSTOP_MINIMUM 0.2f // seconds
|
|
|
|
#define ASSISTEDCONTROL_TIMETOSTOP_MAXIMUM 9.0f // seconds
|
|
|
|
#define ASSISTEDCONTROL_DELAY_TO_BRAKE 1.0f // seconds
|
|
|
|
#define ASSISTEDCONTROL_TIMEOUT_MULTIPLIER 2.0f // actual deceleration rate can be 50% of desired...timeouts need to cater for this
|
|
|
|
void plan_setup_assistedcontrol(uint8_t timeout_occurred)
|
|
|
|
{
|
|
|
|
PositionStateData positionState;
|
|
|
|
|
|
|
|
PositionStateGet(&positionState);
|
|
|
|
PathDesiredData pathDesired;
|
|
|
|
PathDesiredGet(&pathDesired);
|
|
|
|
|
|
|
|
FlightStatusAssistedControlStateOptions assistedControlFlightMode;
|
|
|
|
FlightStatusAssistedControlStateGet(&assistedControlFlightMode);
|
|
|
|
|
|
|
|
if (timeout_occurred) {
|
|
|
|
pathDesired.End.North = positionState.North;
|
|
|
|
pathDesired.End.East = positionState.East;
|
|
|
|
pathDesired.End.Down = positionState.Down;
|
|
|
|
pathDesired.Start.North = positionState.North;
|
|
|
|
pathDesired.Start.East = positionState.East;
|
|
|
|
pathDesired.Start.Down = positionState.Down;
|
|
|
|
pathDesired.StartingVelocity = 0.0f;
|
|
|
|
pathDesired.EndingVelocity = 0.0f;
|
|
|
|
pathDesired.Mode = PATHDESIRED_MODE_FLYENDPOINT;
|
|
|
|
assistedControlFlightMode = FLIGHTSTATUS_ASSISTEDCONTROLSTATE_HOLD;
|
|
|
|
} else {
|
|
|
|
VelocityStateData velocityState;
|
|
|
|
VelocityStateGet(&velocityState);
|
|
|
|
float brakeRate;
|
|
|
|
VtolPathFollowerSettingsBrakeRateGet(&brakeRate);
|
|
|
|
if (brakeRate < ASSISTEDCONTROL_BRAKERATE_MINIMUM) {
|
|
|
|
brakeRate = ASSISTEDCONTROL_BRAKERATE_MINIMUM; // set a minimum to avoid a divide by zero potential below
|
|
|
|
}
|
|
|
|
// Calculate the velocity
|
|
|
|
float velocity = velocityState.North * velocityState.North + velocityState.East * velocityState.East + velocityState.Down * velocityState.Down;
|
|
|
|
velocity = sqrtf(velocity);
|
|
|
|
|
|
|
|
// Calculate the desired time to zero velocity.
|
|
|
|
float time_to_stopped = ASSISTEDCONTROL_DELAY_TO_BRAKE; // we allow at least 0.5 seconds to rotate to a brake angle.
|
|
|
|
time_to_stopped += velocity / brakeRate;
|
|
|
|
|
|
|
|
// Sanity check the brake rate by ensuring that the time to stop is within a range.
|
|
|
|
if (time_to_stopped < ASSISTEDCONTROL_TIMETOSTOP_MINIMUM) {
|
|
|
|
time_to_stopped = ASSISTEDCONTROL_TIMETOSTOP_MINIMUM;
|
|
|
|
} else if (time_to_stopped > ASSISTEDCONTROL_TIMETOSTOP_MAXIMUM) {
|
|
|
|
time_to_stopped = ASSISTEDCONTROL_TIMETOSTOP_MAXIMUM;
|
|
|
|
}
|
|
|
|
|
|
|
|
// calculate the distance we will travel
|
|
|
|
float north_delta = velocityState.North * ASSISTEDCONTROL_DELAY_TO_BRAKE; // we allow at least 0.5s to rotate to brake angle
|
|
|
|
north_delta += (time_to_stopped - ASSISTEDCONTROL_DELAY_TO_BRAKE) * 0.5f * velocityState.North; // area under the linear deceleration plot
|
|
|
|
float east_delta = velocityState.East * ASSISTEDCONTROL_DELAY_TO_BRAKE; // we allow at least 0.5s to rotate to brake angle
|
|
|
|
east_delta += (time_to_stopped - ASSISTEDCONTROL_DELAY_TO_BRAKE) * 0.5f * velocityState.East; // area under the linear deceleration plot
|
|
|
|
float down_delta = velocityState.Down * ASSISTEDCONTROL_DELAY_TO_BRAKE;
|
|
|
|
down_delta += (time_to_stopped - ASSISTEDCONTROL_DELAY_TO_BRAKE) * 0.5f * velocityState.Down; // area under the linear deceleration plot
|
|
|
|
float net_delta = east_delta * east_delta + north_delta * north_delta + down_delta * down_delta;
|
|
|
|
net_delta = sqrtf(net_delta);
|
|
|
|
|
|
|
|
pathDesired.Start.North = positionState.North;
|
|
|
|
pathDesired.Start.East = positionState.East;
|
|
|
|
pathDesired.Start.Down = positionState.Down;
|
|
|
|
pathDesired.ModeParameters[PATHDESIRED_MODEPARAMETER_BRAKE_STARTVELOCITYVECTOR_NORTH] = velocityState.North;
|
|
|
|
pathDesired.ModeParameters[PATHDESIRED_MODEPARAMETER_BRAKE_STARTVELOCITYVECTOR_EAST] = velocityState.East;
|
|
|
|
pathDesired.ModeParameters[PATHDESIRED_MODEPARAMETER_BRAKE_STARTVELOCITYVECTOR_DOWN] = velocityState.Down;
|
|
|
|
|
|
|
|
pathDesired.End.North = positionState.North + north_delta;
|
|
|
|
pathDesired.End.East = positionState.East + east_delta;
|
|
|
|
pathDesired.End.Down = positionState.Down + down_delta;
|
|
|
|
|
|
|
|
pathDesired.StartingVelocity = velocity;
|
|
|
|
pathDesired.EndingVelocity = 0.0f;
|
|
|
|
pathDesired.Mode = PATHDESIRED_MODE_BRAKE;
|
|
|
|
pathDesired.ModeParameters[PATHDESIRED_MODEPARAMETER_BRAKE_TIMEOUT] = time_to_stopped * ASSISTEDCONTROL_TIMEOUT_MULTIPLIER;
|
|
|
|
assistedControlFlightMode = FLIGHTSTATUS_ASSISTEDCONTROLSTATE_BRAKE;
|
|
|
|
}
|
|
|
|
FlightStatusAssistedControlStateSet(&assistedControlFlightMode);
|
|
|
|
PathDesiredSet(&pathDesired);
|
|
|
|
}
|