ardupilot/libraries/AP_Scripting/examples/Flip_Mode.lua

245 lines
6.8 KiB
Lua

-- This script replicates the behavior of flip mode
local MODE_NUMBER = 100
-- Register flip as mode 100
local FLIP_MODE_STATE = assert(vehicle:register_custom_mode(MODE_NUMBER, "Flip 2", "FLI2"))
-- Get input channels
local THROTTLE_CHAN = math.floor(assert(param:get("RCMAP_THROTTLE")))
local pilot_throttle = assert(rc:get_channel(THROTTLE_CHAN))
local pilot_pitch = assert(rc:get_channel(assert(param:get("RCMAP_PITCH"))))
local pilot_roll = assert(rc:get_channel(assert(param:get("RCMAP_ROLL"))))
local HOVER_THROTTLE = Parameter("MOT_THST_HOVER")
local THROTTLE_MIN = Parameter("RC" .. THROTTLE_CHAN .. "_MIN")
local THROTTLE_MAX = Parameter("RC" .. THROTTLE_CHAN .. "_MAX")
local THROTTLE_DZ = Parameter("RC" .. THROTTLE_CHAN .. "_DZ")
-- Replication of the copter function
local function get_throttle_mid()
local r_in = (THROTTLE_MIN:get() + THROTTLE_MAX:get()) * 0.5
local radio_trim_low = THROTTLE_MIN:get() + THROTTLE_DZ:get()
return 1000.0 * ((r_in - radio_trim_low) / (THROTTLE_MAX:get() - radio_trim_low))
end
-- Replication of the copter function
local function get_pilot_desired_throttle()
local thr_mid = HOVER_THROTTLE:get()
local throttle_control = (pilot_throttle:norm_input_ignore_trim() + 1.0) * (1000.0 / 2.0)
local mid_stick = get_throttle_mid()
local throttle_in
if throttle_control < mid_stick then
throttle_in = throttle_control * 0.5 / mid_stick
else
throttle_in = 0.5 + ((throttle_control - mid_stick) * 0.5 / (1000.0 - mid_stick))
end
local expo = -(thr_mid - 0.5) / 0.375
expo = math.max(expo, -0.5)
expo = math.min(expo, 1.0)
return throttle_in * (1.0-expo) + expo*throttle_in*throttle_in*throttle_in
end
-- Used to check for mode changes
local last_mode_number
local MODES = {
STABILIZE = 0,
ALT_HOLD = 2,
}
-- Check if we should be allowed to enter flip mode
local function allow_enter(previous_mode)
-- Only allow flip from stabilize and alt hold
if (previous_mode ~= MODES.STABILIZE) and (previous_mode ~= MODES.ALT_HOLD) then
return false
end
-- Must be armed and flying
if (not arming:is_armed()) or (not vehicle:get_likely_flying()) then
return false
end
return true
end
local FLIP_STATE = {
START = 0,
ROLL = 1,
PITCH_A = 2,
PITCH_B = 3,
RECOVER = 4,
}
local state
local start_ms
local roll_dir
local pitch_dir
local start_attitude = {
roll = 0,
pitch = 0,
yaw = 0,
}
-- Init on first call
local previous_mode
local function init()
-- Record the previous mode, this is returned to on completion of the flip
previous_mode = last_mode_number
-- Return to previous mode immediately if flip cannot be performed
if not allow_enter(previous_mode) then
vehicle:set_mode(previous_mode)
return
end
state = FLIP_STATE.START
start_ms = millis()
roll_dir = 0.0
pitch_dir = 0.0
-- choose direction based on pilot's roll and pitch sticks
if pilot_pitch:norm_input_dz() > 0.1 then
pitch_dir = 1.0
elseif pilot_pitch:norm_input_dz() < -0.1 then
pitch_dir = -1.0
elseif pilot_roll:norm_input_dz() >= 0 then
roll_dir = 1.0
else
roll_dir = -1.0
end
-- Record start attitude to be used in recovery stage
start_attitude.roll = math.deg(ahrs:get_roll())
start_attitude.pitch = math.deg(ahrs:get_pitch())
start_attitude.yaw = math.deg(ahrs:get_yaw())
end
local FLIP_THR_INC = 0.2
local FLIP_THR_DEC = 0.24
local FLIP_ROTATION_RATE = 400
local function run()
local NOW = millis()
-- Disarmed, pilot input or timeout then return to previous mode
local PILOT_INPUT = (math.abs(pilot_roll:norm_input_dz()) > 0.85) or (math.abs(pilot_pitch:norm_input_dz()) > 0.85)
if (not arming:is_armed()) or PILOT_INPUT or ((NOW - start_ms) > 2500) then
vehicle:set_mode(previous_mode)
return
end
local roll_deg = math.deg(ahrs:get_roll())
local pitch_deg = math.deg(ahrs:get_pitch())
if state == FLIP_STATE.RECOVER then
-- Target original attitude with 0 climb rate
vehicle:set_target_angle_and_climbrate(start_attitude.roll, start_attitude.pitch, start_attitude.yaw, 0.0, false, 0.0)
-- See if we have returned to the desired angle
local recovery_angle = math.abs((roll_deg - start_attitude.roll) * roll_dir) + math.abs((pitch_deg - start_attitude.pitch) * pitch_dir)
if recovery_angle < 5.0 then
-- Complete, return to original mode
vehicle:set_mode(previous_mode)
end
return
end
local throttle_out = get_pilot_desired_throttle()
local flip_angle = roll_deg * roll_dir + pitch_deg * pitch_dir
if state == FLIP_STATE.START then
-- Increase throttle
throttle_out = math.min(throttle_out + FLIP_THR_INC, 1.0)
-- Check for next stage
if flip_angle >= 45.0 then
if roll_dir ~= 0 then
-- Rolling flip
state = FLIP_STATE.ROLL
else
-- Pitching flip
state = FLIP_STATE.PITCH_A
end
end
elseif state == FLIP_STATE.ROLL then
-- decrease throttle
throttle_out = math.max(throttle_out - FLIP_THR_DEC, 0.0)
-- beyond 90deg move on to recovery
if (flip_angle < 45.0) and (flip_angle > -90.0) then
state = FLIP_STATE.RECOVER
end
elseif state == FLIP_STATE.PITCH_A then
-- decrease throttle
throttle_out = math.max(throttle_out - FLIP_THR_DEC, 0.0)
-- check roll for inversion
if (math.abs(roll_deg) > 90.0) and (flip_angle > 45.0) then
state = FLIP_STATE.PITCH_B
end
elseif state == FLIP_STATE.PITCH_B then
-- decrease throttle
throttle_out = math.max(throttle_out - FLIP_THR_DEC, 0.0)
-- check roll for un-inversion
if (math.abs(roll_deg) < 90.0) and (flip_angle > -45.0) then
state = FLIP_STATE.RECOVER
end
end
-- Send rate and throttle command to vehicle
local roll_rate = FLIP_ROTATION_RATE * roll_dir
local pitch_rate = FLIP_ROTATION_RATE * pitch_dir
vehicle:set_target_rate_and_throttle(roll_rate, pitch_rate, 0.0, throttle_out)
end
local function exit()
-- Nothing to do here
end
local function update()
local mode = vehicle:get_mode()
-- Update entry state
FLIP_MODE_STATE:allow_entry(allow_enter(mode))
if mode == MODE_NUMBER then
if last_mode_number ~= MODE_NUMBER then
-- Fist call after entering
init()
else
-- Runtime function
run()
end
elseif last_mode_number == MODE_NUMBER then
-- Exit mode
exit()
end
last_mode_number = mode
-- Run at 100Hz
return update, 10
end
return update()