2017-02-21 13:32:26 -04:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
2016-07-31 07:22:06 -03:00
|
|
|
# Fly ArduPlane in SITL
|
2016-11-08 07:06:05 -04:00
|
|
|
from __future__ import print_function
|
2016-07-31 07:22:06 -03:00
|
|
|
import math
|
|
|
|
import os
|
|
|
|
|
|
|
|
import pexpect
|
2018-05-31 07:27:41 -03:00
|
|
|
from pymavlink import quaternion
|
2013-06-18 03:27:21 -03:00
|
|
|
from pymavlink import mavutil
|
2016-07-31 07:22:06 -03:00
|
|
|
|
|
|
|
from pysim import util
|
2011-11-12 07:13:17 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
from common import AutoTest
|
2018-04-27 15:21:53 -03:00
|
|
|
from common import NotAchievedException
|
2018-08-03 07:50:15 -03:00
|
|
|
from common import PreconditionFailedException
|
2018-03-14 08:08:53 -03:00
|
|
|
|
2011-11-12 07:13:17 -04:00
|
|
|
# get location of scripts
|
2016-07-31 07:22:06 -03:00
|
|
|
testdir = os.path.dirname(os.path.realpath(__file__))
|
2018-03-05 11:14:34 -04:00
|
|
|
HOME = mavutil.location(-35.362938, 149.165085, 585, 354)
|
2016-07-31 07:22:06 -03:00
|
|
|
WIND = "0,180,0.2" # speed,direction,variance
|
2011-11-12 07:13:17 -04:00
|
|
|
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
class AutoTestPlane(AutoTest):
|
|
|
|
def __init__(self,
|
|
|
|
binary,
|
|
|
|
valgrind=False,
|
|
|
|
gdb=False,
|
|
|
|
speedup=10,
|
|
|
|
frame=None,
|
|
|
|
params=None,
|
2018-03-15 08:31:19 -03:00
|
|
|
gdbserver=False,
|
2018-07-23 04:06:00 -03:00
|
|
|
breakpoints=[],
|
2018-03-15 08:31:19 -03:00
|
|
|
**kwargs):
|
|
|
|
super(AutoTestPlane, self).__init__(**kwargs)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.binary = binary
|
|
|
|
self.valgrind = valgrind
|
|
|
|
self.gdb = gdb
|
|
|
|
self.frame = frame
|
|
|
|
self.params = params
|
|
|
|
self.gdbserver = gdbserver
|
2018-07-23 04:06:00 -03:00
|
|
|
self.breakpoints = breakpoints
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.home = "%f,%f,%u,%u" % (HOME.lat,
|
|
|
|
HOME.lng,
|
|
|
|
HOME.alt,
|
|
|
|
HOME.heading)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.homeloc = None
|
|
|
|
self.speedup = speedup
|
|
|
|
|
|
|
|
self.sitl = None
|
|
|
|
self.hasInit = False
|
|
|
|
|
2018-03-15 08:54:34 -03:00
|
|
|
self.log_name = "ArduPlane"
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def init(self):
|
|
|
|
if self.frame is None:
|
|
|
|
self.frame = 'plane-elevrev'
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
defaults_file = os.path.join(testdir,
|
|
|
|
'default_params/plane-jsbsim.parm')
|
|
|
|
self.sitl = util.start_SITL(self.binary,
|
|
|
|
wipe=True,
|
|
|
|
model=self.frame,
|
|
|
|
home=self.home,
|
|
|
|
speedup=self.speedup,
|
|
|
|
defaults_file=defaults_file,
|
|
|
|
valgrind=self.valgrind,
|
|
|
|
gdb=self.gdb,
|
2018-07-23 04:06:00 -03:00
|
|
|
gdbserver=self.gdbserver,
|
|
|
|
breakpoints=self.breakpoints)
|
2018-03-15 08:31:19 -03:00
|
|
|
self.mavproxy = util.start_MAVProxy_SITL(
|
|
|
|
'ArduPlane', options=self.mavproxy_options())
|
2018-03-20 08:34:59 -03:00
|
|
|
self.mavproxy.expect('Telemetry log: (\S+)\r\n')
|
2018-07-31 06:49:22 -03:00
|
|
|
self.logfile = self.mavproxy.match.group(1)
|
|
|
|
self.progress("LOGFILE %s" % self.logfile)
|
|
|
|
self.try_symlink_tlog()
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
self.mavproxy.expect('Received [0-9]+ parameters')
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
util.expect_setup_callback(self.mavproxy, self.expect_callback)
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.expect_list_clear()
|
|
|
|
self.expect_list_extend([self.sitl, self.mavproxy])
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Started simulator")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-08-03 20:20:15 -03:00
|
|
|
self.get_mavlink_connection_going()
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
self.hasInit = True
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Ready to start testing!")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def takeoff(self):
|
|
|
|
"""Takeoff get to 30m altitude."""
|
|
|
|
|
|
|
|
self.mavproxy.send('switch 4\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
|
2018-08-13 08:07:34 -03:00
|
|
|
self.wait_ready_to_arm()
|
|
|
|
self.arm_vehicle()
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
# some rudder to counteract the prop torque
|
|
|
|
self.set_rc(4, 1700)
|
|
|
|
|
|
|
|
# some up elevator to keep the tail down
|
|
|
|
self.set_rc(2, 1200)
|
|
|
|
|
|
|
|
# get it moving a bit first
|
|
|
|
self.set_rc(3, 1300)
|
|
|
|
self.mav.recv_match(condition='VFR_HUD.groundspeed>6', blocking=True)
|
|
|
|
|
|
|
|
# a bit faster again, straighten rudder
|
|
|
|
self.set_rc(3, 1600)
|
|
|
|
self.set_rc(4, 1500)
|
|
|
|
self.mav.recv_match(condition='VFR_HUD.groundspeed>12', blocking=True)
|
|
|
|
|
|
|
|
# hit the gas harder now, and give it some more elevator
|
|
|
|
self.set_rc(2, 1100)
|
|
|
|
self.set_rc(3, 2000)
|
|
|
|
|
|
|
|
# gain a bit of altitude
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_altitude(self.homeloc.alt+150,
|
2018-05-09 00:32:23 -03:00
|
|
|
self.homeloc.alt+180,
|
|
|
|
timeout=30)
|
2016-07-31 07:22:06 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
# level off
|
|
|
|
self.set_rc(2, 1500)
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("TAKEOFF COMPLETE")
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def fly_left_circuit(self):
|
|
|
|
"""Fly a left circuit, 200m on a side."""
|
|
|
|
self.mavproxy.send('switch 4\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 2000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_level_flight()
|
2018-03-14 08:08:53 -03:00
|
|
|
|
|
|
|
self.progress("Flying left circuit")
|
2018-03-05 11:14:34 -04:00
|
|
|
# do 4 turns
|
|
|
|
for i in range(0, 4):
|
|
|
|
# hard left
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting turn %u" % i)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_heading(270 - (90*i), accuracy=10)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1500)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting leg %u" % i)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_distance(100, accuracy=20)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Circuit complete")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def fly_RTL(self):
|
|
|
|
"""Fly to home."""
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Flying home in RTL")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('switch 2\n')
|
|
|
|
self.wait_mode('RTL')
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_location(self.homeloc,
|
2018-05-09 00:32:23 -03:00
|
|
|
accuracy=120,
|
|
|
|
target_altitude=self.homeloc.alt+100,
|
|
|
|
height_accuracy=20,
|
|
|
|
timeout=180)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("RTL Complete")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def fly_LOITER(self, num_circles=4):
|
|
|
|
"""Loiter where we are."""
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Testing LOITER for %u turns" % num_circles)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('loiter\n')
|
|
|
|
self.wait_mode('LOITER')
|
|
|
|
|
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
initial_alt = m.alt
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Initial altitude %u\n" % initial_alt)
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
while num_circles > 0:
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_heading(0, accuracy=10, timeout=60)
|
|
|
|
self.wait_heading(180, accuracy=10, timeout=60)
|
2018-03-05 11:14:34 -04:00
|
|
|
num_circles -= 1
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Loiter %u circles left" % num_circles)
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
final_alt = m.alt
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Final altitude %u initial %u\n" %
|
|
|
|
(final_alt, initial_alt))
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
|
|
|
|
if abs(final_alt - initial_alt) > 20:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Failed to maintain altitude")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Completed Loiter OK")
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def fly_CIRCLE(self, num_circles=1):
|
|
|
|
"""Circle where we are."""
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Testing CIRCLE for %u turns" % num_circles)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('mode CIRCLE\n')
|
|
|
|
self.wait_mode('CIRCLE')
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
initial_alt = m.alt
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Initial altitude %u\n" % initial_alt)
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
while num_circles > 0:
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_heading(0, accuracy=10, timeout=60)
|
|
|
|
self.wait_heading(180, accuracy=10, timeout=60)
|
2018-03-05 11:14:34 -04:00
|
|
|
num_circles -= 1
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("CIRCLE %u circles left" % num_circles)
|
2011-11-12 07:13:17 -04:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
final_alt = m.alt
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Final altitude %u initial %u\n" %
|
|
|
|
(final_alt, initial_alt))
|
2011-11-12 07:13:17 -04:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
2011-11-12 07:13:17 -04:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
if abs(final_alt - initial_alt) > 20:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Failed to maintain altitude")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2011-11-13 21:59:59 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Completed CIRCLE OK")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def wait_level_flight(self, accuracy=5, timeout=30):
|
|
|
|
"""Wait for level flight."""
|
|
|
|
tstart = self.get_sim_time()
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Waiting for level flight")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1500)
|
|
|
|
self.set_rc(2, 1500)
|
|
|
|
self.set_rc(4, 1500)
|
|
|
|
while self.get_sim_time() < tstart + timeout:
|
|
|
|
m = self.mav.recv_match(type='ATTITUDE', blocking=True)
|
|
|
|
roll = math.degrees(m.roll)
|
|
|
|
pitch = math.degrees(m.pitch)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Roll=%.1f Pitch=%.1f" % (roll, pitch))
|
2018-03-05 11:14:34 -04:00
|
|
|
if math.fabs(roll) <= accuracy and math.fabs(pitch) <= accuracy:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Attained level flight")
|
2018-04-27 15:21:53 -03:00
|
|
|
return
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Failed to attain level flight")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2011-11-13 21:59:59 -04:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def change_altitude(self, altitude, accuracy=30):
|
|
|
|
"""Get to a given altitude."""
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
alt_error = self.mav.messages['VFR_HUD'].alt - altitude
|
|
|
|
if alt_error > 0:
|
|
|
|
self.set_rc(2, 2000)
|
|
|
|
else:
|
|
|
|
self.set_rc(2, 1000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_altitude(altitude-accuracy/2, altitude+accuracy/2)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(2, 1500)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Reached target altitude at %u" %
|
|
|
|
self.mav.messages['VFR_HUD'].alt)
|
2018-03-05 11:14:34 -04:00
|
|
|
return self.wait_level_flight()
|
|
|
|
|
|
|
|
def axial_left_roll(self, count=1):
|
|
|
|
"""Fly a left axial roll."""
|
|
|
|
# full throttle!
|
|
|
|
self.set_rc(3, 2000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.change_altitude(self.homeloc.alt+300)
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
# fly the roll in manual
|
|
|
|
self.mavproxy.send('switch 6\n')
|
|
|
|
self.wait_mode('MANUAL')
|
|
|
|
|
|
|
|
while count > 0:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting roll")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1000)
|
2018-04-27 15:21:53 -03:00
|
|
|
try:
|
|
|
|
self.wait_roll(-150, accuracy=90)
|
|
|
|
self.wait_roll(150, accuracy=90)
|
|
|
|
self.wait_roll(0, accuracy=90)
|
|
|
|
except Exception as e:
|
2018-05-09 00:32:23 -03:00
|
|
|
self.set_rc(1, 1500)
|
2018-04-27 15:21:53 -03:00
|
|
|
raise e
|
2018-03-05 11:14:34 -04:00
|
|
|
count -= 1
|
|
|
|
|
|
|
|
# back to FBWA
|
|
|
|
self.set_rc(1, 1500)
|
|
|
|
self.mavproxy.send('switch 4\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
return self.wait_level_flight()
|
|
|
|
|
|
|
|
def inside_loop(self, count=1):
|
|
|
|
"""Fly a inside loop."""
|
|
|
|
# full throttle!
|
|
|
|
self.set_rc(3, 2000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.change_altitude(self.homeloc.alt+300)
|
2018-03-05 11:14:34 -04:00
|
|
|
# fly the loop in manual
|
|
|
|
self.mavproxy.send('switch 6\n')
|
|
|
|
self.wait_mode('MANUAL')
|
|
|
|
|
|
|
|
while count > 0:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting loop")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(2, 1000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_pitch(-60, accuracy=20)
|
|
|
|
self.wait_pitch(0, accuracy=20)
|
2018-03-05 11:14:34 -04:00
|
|
|
count -= 1
|
|
|
|
|
|
|
|
# back to FBWA
|
|
|
|
self.set_rc(2, 1500)
|
|
|
|
self.mavproxy.send('switch 4\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
return self.wait_level_flight()
|
|
|
|
|
2018-05-31 07:27:41 -03:00
|
|
|
def set_attitude_target(self):
|
|
|
|
"""Test setting of attitude target in guided mode."""
|
|
|
|
# mode guided:
|
|
|
|
self.mavproxy.send('mode GUIDED\n')
|
|
|
|
self.wait_mode('GUIDED')
|
|
|
|
|
|
|
|
target_roll_degrees = 70
|
|
|
|
state_roll_over = "roll-over"
|
|
|
|
state_stabilize_roll = "stabilize-roll"
|
|
|
|
state_hold = "hold"
|
|
|
|
state_roll_back = "roll-back"
|
|
|
|
state_done = "done"
|
|
|
|
|
|
|
|
tstart = self.get_sim_time()
|
|
|
|
|
|
|
|
try:
|
|
|
|
state = state_roll_over
|
|
|
|
while state != state_done:
|
|
|
|
if self.get_sim_time() - tstart > 20:
|
|
|
|
raise NotAchievedException()
|
|
|
|
|
|
|
|
m = self.mav.recv_match(type='ATTITUDE',
|
|
|
|
blocking=True,
|
|
|
|
timeout=0.1)
|
|
|
|
if m is None:
|
|
|
|
continue
|
|
|
|
|
|
|
|
r = math.degrees(m.roll)
|
|
|
|
if state == state_roll_over:
|
|
|
|
target_roll_degrees = 70
|
|
|
|
if abs(r - target_roll_degrees) < 10:
|
|
|
|
state = state_stabilize_roll
|
|
|
|
stabilize_start = self.get_sim_time()
|
|
|
|
elif state == state_stabilize_roll:
|
|
|
|
# just give it a little time to sort it self out
|
|
|
|
if self.get_sim_time() - stabilize_start > 2:
|
|
|
|
state = state_hold
|
|
|
|
hold_start = self.get_sim_time()
|
|
|
|
elif state == state_hold:
|
|
|
|
target_roll_degrees = 70
|
|
|
|
if self.get_sim_time() - hold_start > 10:
|
|
|
|
state = state_roll_back
|
|
|
|
if abs(r - target_roll_degrees) > 10:
|
|
|
|
self.progress("Failed to hold attitude")
|
|
|
|
raise NotAchievedException()
|
|
|
|
elif state == state_roll_back:
|
|
|
|
target_roll_degrees = 0
|
|
|
|
if abs(r - target_roll_degrees) < 10:
|
|
|
|
state = state_done
|
|
|
|
else:
|
|
|
|
raise ValueError()
|
|
|
|
|
|
|
|
self.progress("%s Roll: %f desired=%f" %
|
|
|
|
(state, r, target_roll_degrees))
|
|
|
|
|
|
|
|
time_boot_millis = 0 # FIXME
|
|
|
|
target_system = 1 # FIXME
|
|
|
|
target_component = 1 # FIXME
|
|
|
|
type_mask = 0b10000001 ^ 0xFF # FIXME
|
|
|
|
# attitude in radians:
|
|
|
|
q = quaternion.Quaternion([math.radians(target_roll_degrees),
|
|
|
|
0 ,
|
|
|
|
0])
|
|
|
|
roll_rate_radians = 0.5
|
|
|
|
pitch_rate_radians = 0
|
|
|
|
yaw_rate_radians = 0
|
|
|
|
thrust = 1.0
|
|
|
|
self.mav.mav.set_attitude_target_send(time_boot_millis,
|
|
|
|
target_system,
|
|
|
|
target_component,
|
|
|
|
type_mask,
|
|
|
|
q,
|
|
|
|
roll_rate_radians,
|
|
|
|
pitch_rate_radians,
|
|
|
|
yaw_rate_radians,
|
|
|
|
thrust)
|
|
|
|
except Exception as e:
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
raise e
|
|
|
|
|
|
|
|
# back to FBWA
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
self.wait_level_flight()
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def test_stabilize(self, count=1):
|
|
|
|
"""Fly stabilize mode."""
|
|
|
|
# full throttle!
|
|
|
|
self.set_rc(3, 2000)
|
|
|
|
self.set_rc(2, 1300)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.change_altitude(self.homeloc.alt+300)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(2, 1500)
|
|
|
|
|
|
|
|
self.mavproxy.send("mode STABILIZE\n")
|
|
|
|
self.wait_mode('STABILIZE')
|
|
|
|
|
|
|
|
while count > 0:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting roll")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 2000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_roll(-150, accuracy=90)
|
|
|
|
self.wait_roll(150, accuracy=90)
|
|
|
|
self.wait_roll(0, accuracy=90)
|
2018-03-05 11:14:34 -04:00
|
|
|
count -= 1
|
|
|
|
|
|
|
|
self.set_rc(1, 1500)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_roll(0, accuracy=5)
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
# back to FBWA
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
return self.wait_level_flight()
|
|
|
|
|
|
|
|
def test_acro(self, count=1):
|
|
|
|
"""Fly ACRO mode."""
|
|
|
|
# full throttle!
|
|
|
|
self.set_rc(3, 2000)
|
|
|
|
self.set_rc(2, 1300)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.change_altitude(self.homeloc.alt+300)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(2, 1500)
|
|
|
|
|
|
|
|
self.mavproxy.send("mode ACRO\n")
|
|
|
|
self.wait_mode('ACRO')
|
|
|
|
|
|
|
|
while count > 0:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting roll")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_roll(-150, accuracy=90)
|
|
|
|
self.wait_roll(150, accuracy=90)
|
|
|
|
self.wait_roll(0, accuracy=90)
|
2018-03-05 11:14:34 -04:00
|
|
|
count -= 1
|
|
|
|
self.set_rc(1, 1500)
|
|
|
|
|
|
|
|
# back to FBWA
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
|
|
|
|
self.wait_level_flight()
|
|
|
|
|
|
|
|
self.mavproxy.send("mode ACRO\n")
|
|
|
|
self.wait_mode('ACRO')
|
|
|
|
|
|
|
|
count = 2
|
|
|
|
while count > 0:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting loop")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(2, 1000)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_pitch(-60, accuracy=20)
|
|
|
|
self.wait_pitch(0, accuracy=20)
|
2018-03-05 11:14:34 -04:00
|
|
|
count -= 1
|
|
|
|
|
|
|
|
self.set_rc(2, 1500)
|
|
|
|
|
|
|
|
# back to FBWA
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
return self.wait_level_flight()
|
|
|
|
|
2018-07-31 06:47:49 -03:00
|
|
|
def test_FBWB(self, mode='FBWB'):
|
2018-03-05 11:14:34 -04:00
|
|
|
"""Fly FBWB or CRUISE mode."""
|
|
|
|
self.mavproxy.send("mode %s\n" % mode)
|
|
|
|
self.wait_mode(mode)
|
|
|
|
self.set_rc(3, 1700)
|
|
|
|
self.set_rc(2, 1500)
|
|
|
|
|
|
|
|
# lock in the altitude by asking for an altitude change then releasing
|
|
|
|
self.set_rc(2, 1000)
|
|
|
|
self.wait_distance(50, accuracy=20)
|
|
|
|
self.set_rc(2, 1500)
|
|
|
|
self.wait_distance(50, accuracy=20)
|
|
|
|
|
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
initial_alt = m.alt
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Initial altitude %u\n" % initial_alt)
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Flying right circuit")
|
2018-03-05 11:14:34 -04:00
|
|
|
# do 4 turns
|
|
|
|
for i in range(0, 4):
|
|
|
|
# hard left
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting turn %u" % i)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1800)
|
2018-04-27 15:21:53 -03:00
|
|
|
try:
|
|
|
|
self.wait_heading(0 + (90*i), accuracy=20, timeout=60)
|
|
|
|
except Exception as e:
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1500)
|
2018-04-27 15:21:53 -03:00
|
|
|
raise e
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(1, 1500)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting leg %u" % i)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_distance(100, accuracy=20)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Circuit complete")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Flying rudder left circuit")
|
2018-03-05 11:14:34 -04:00
|
|
|
# do 4 turns
|
|
|
|
for i in range(0, 4):
|
|
|
|
# hard left
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting turn %u" % i)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(4, 1900)
|
2018-04-27 15:21:53 -03:00
|
|
|
try:
|
|
|
|
self.wait_heading(360 - (90*i), accuracy=20, timeout=60)
|
|
|
|
except Exception as e:
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(4, 1500)
|
2018-04-27 15:21:53 -03:00
|
|
|
raise e
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(4, 1500)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Starting leg %u" % i)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_distance(100, accuracy=20)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Circuit complete")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
final_alt = m.alt
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Final altitude %u initial %u\n" %
|
|
|
|
(final_alt, initial_alt))
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
# back to FBWA
|
|
|
|
self.mavproxy.send('mode FBWA\n')
|
|
|
|
self.wait_mode('FBWA')
|
|
|
|
|
|
|
|
if abs(final_alt - initial_alt) > 20:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Failed to maintain altitude")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
return self.wait_level_flight()
|
|
|
|
|
2018-08-13 08:07:34 -03:00
|
|
|
def wp_load(self, filename):
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('wp load %s\n' % filename)
|
|
|
|
self.mavproxy.expect('Flight plan received')
|
|
|
|
self.mavproxy.send('wp list\n')
|
|
|
|
self.mavproxy.expect('Requesting [0-9]+ waypoints')
|
2018-08-13 08:07:34 -03:00
|
|
|
|
|
|
|
def fly_mission(self, filename):
|
|
|
|
"""Fly a mission from a file."""
|
|
|
|
self.progress("Flying mission %s" % filename)
|
|
|
|
self.wp_load(filename)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('switch 1\n') # auto mode
|
|
|
|
self.wait_mode('AUTO')
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_waypoint(1, 7, max_dist=60)
|
|
|
|
self.wait_groundspeed(0, 0.5, timeout=60)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.expect("Auto disarmed")
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Mission OK")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-08-13 08:07:34 -03:00
|
|
|
def fly_flaps(self):
|
|
|
|
"""Test flaps functionality."""
|
|
|
|
filename = os.path.join(testdir, "flaps.txt")
|
|
|
|
self.context_push()
|
|
|
|
ex = None
|
|
|
|
try:
|
|
|
|
flaps_ch = 5
|
|
|
|
servo_ch = 5
|
|
|
|
self.set_parameter("SERVO%u_FUNCTION" % servo_ch, 3) # flapsauto
|
|
|
|
self.set_parameter("FLAP_IN_CHANNEL", flaps_ch)
|
|
|
|
self.set_parameter("LAND_FLAP_PERCNT", 50)
|
|
|
|
self.set_parameter("LOG_DISARMED", 1)
|
|
|
|
flaps_ch_min = 1000
|
|
|
|
flaps_ch_trim = 1500
|
|
|
|
flaps_ch_max = 2000
|
|
|
|
self.set_parameter("RC%u_MIN" % flaps_ch, flaps_ch_min)
|
|
|
|
self.set_parameter("RC%u_MAX" % flaps_ch, flaps_ch_max)
|
|
|
|
self.set_parameter("RC%u_TRIM" % flaps_ch, flaps_ch_trim)
|
|
|
|
|
|
|
|
servo_ch_min = 1200
|
|
|
|
servo_ch_trim = 1300
|
|
|
|
servo_ch_max = 1800
|
|
|
|
self.set_parameter("SERVO%u_MIN" % servo_ch, servo_ch_min)
|
|
|
|
self.set_parameter("SERVO%u_MAX" % servo_ch, servo_ch_max)
|
|
|
|
self.set_parameter("SERVO%u_TRIM" % servo_ch, servo_ch_trim)
|
|
|
|
|
|
|
|
# check flaps are not deployed:
|
|
|
|
self.set_rc(flaps_ch, flaps_ch_min)
|
|
|
|
self.wait_servo_channel_value(servo_ch, servo_ch_min)
|
|
|
|
# deploy the flaps:
|
|
|
|
self.set_rc(flaps_ch, flaps_ch_max)
|
|
|
|
tstart = self.get_sim_time()
|
|
|
|
self.wait_servo_channel_value(servo_ch, servo_ch_max)
|
|
|
|
tstop = self.get_sim_time_cached()
|
|
|
|
delta_time = tstop - tstart
|
|
|
|
delta_time_min = 0.5
|
|
|
|
delta_time_max = 1.5
|
|
|
|
if delta_time < delta_time_min or delta_time > delta_time_max:
|
|
|
|
self.progress("Flaps Slew not working (%f seconds)" %
|
|
|
|
(delta_time,))
|
|
|
|
raise NotAchievedException()
|
|
|
|
# undeploy flaps:
|
|
|
|
self.set_rc(flaps_ch, flaps_ch_min)
|
|
|
|
self.wait_servo_channel_value(servo_ch, servo_ch_min)
|
|
|
|
|
|
|
|
self.progress("Flying mission %s" % filename)
|
|
|
|
self.wp_load(filename)
|
|
|
|
self.mavproxy.send('wp set 1\n')
|
|
|
|
self.mavproxy.send('switch 1\n') # auto mode
|
|
|
|
self.wait_mode('AUTO')
|
|
|
|
self.wait_ready_to_arm()
|
|
|
|
self.arm_vehicle()
|
2018-09-07 20:48:45 -03:00
|
|
|
tstart = self.get_sim_time_cached()
|
|
|
|
last_mission_current_msg = 0
|
|
|
|
last_seq = None
|
2018-08-13 08:07:34 -03:00
|
|
|
while self.armed():
|
|
|
|
m = self.mav.recv_match(type='MISSION_CURRENT', blocking=True)
|
2018-09-07 20:48:45 -03:00
|
|
|
time_delta = (self.get_sim_time_cached() -
|
|
|
|
last_mission_current_msg)
|
|
|
|
if (time_delta >1 or
|
|
|
|
m.seq != last_seq):
|
|
|
|
dist = None
|
|
|
|
x = self.mav.messages.get("NAV_CONTROLLER_OUTPUT", None)
|
|
|
|
if x is not None:
|
|
|
|
dist = x.wp_dist
|
|
|
|
self.progress("MISSION_CURRENT.seq=%u (dist=%s)" %
|
|
|
|
(m.seq,str(dist)))
|
|
|
|
last_mission_current_msg = self.get_sim_time_cached()
|
|
|
|
last_seq = m.seq
|
2018-08-13 08:07:34 -03:00
|
|
|
# flaps should undeploy at the end
|
|
|
|
self.wait_servo_channel_value(servo_ch, servo_ch_min, timeout=30)
|
|
|
|
|
|
|
|
# do a short flight in FBWA, watching for flaps
|
|
|
|
# self.mavproxy.send('switch 4\n')
|
|
|
|
# self.wait_mode('FBWA')
|
|
|
|
# self.wait_seconds(10)
|
|
|
|
# self.mavproxy.send('switch 6\n')
|
|
|
|
# self.wait_mode('MANUAL')
|
|
|
|
# self.wait_seconds(10)
|
|
|
|
|
|
|
|
self.progress("Flaps OK")
|
|
|
|
except Exception as e:
|
|
|
|
ex = e
|
|
|
|
self.context_pop()
|
|
|
|
if ex:
|
2018-09-07 08:15:02 -03:00
|
|
|
if self.armed():
|
|
|
|
self.disarm_vehicle()
|
2018-08-13 08:07:34 -03:00
|
|
|
raise ex
|
|
|
|
|
2018-08-03 07:16:10 -03:00
|
|
|
def test_rc_relay(self):
|
|
|
|
'''test toggling channel 12 toggles relay'''
|
|
|
|
off = self.get_parameter("SIM_PIN_MASK")
|
|
|
|
if off:
|
|
|
|
raise PreconditionFailedException()
|
|
|
|
self.set_rc(12, 2000)
|
|
|
|
self.mav.wait_heartbeat()
|
|
|
|
self.mav.wait_heartbeat()
|
|
|
|
on = self.get_parameter("SIM_PIN_MASK")
|
|
|
|
if not on:
|
|
|
|
raise NotAchievedException()
|
|
|
|
self.set_rc(12, 1000)
|
|
|
|
self.mav.wait_heartbeat()
|
|
|
|
self.mav.wait_heartbeat()
|
|
|
|
off = self.get_parameter("SIM_PIN_MASK")
|
|
|
|
if off:
|
|
|
|
raise NotAchievedException()
|
|
|
|
|
2018-08-03 07:50:15 -03:00
|
|
|
def test_rc_option_camera_trigger(self):
|
|
|
|
'''test toggling channel 12 takes picture'''
|
|
|
|
x = self.mav.messages.get("CAMERA_FEEDBACK", None)
|
|
|
|
if x is not None:
|
|
|
|
raise PreconditionFailedException()
|
|
|
|
self.set_rc(12, 2000)
|
|
|
|
tstart = self.get_sim_time()
|
2018-09-07 20:48:45 -03:00
|
|
|
while self.get_sim_time() - tstart < 10:
|
2018-08-03 07:50:15 -03:00
|
|
|
x = self.mav.messages.get("CAMERA_FEEDBACK", None)
|
|
|
|
if x is not None:
|
|
|
|
break
|
|
|
|
self.mav.wait_heartbeat()
|
|
|
|
self.set_rc(12, 1000)
|
|
|
|
if x is None:
|
|
|
|
raise NotAchievedException()
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def autotest(self):
|
|
|
|
"""Autotest ArduPlane in SITL."""
|
2018-08-20 12:47:58 -03:00
|
|
|
self.check_test_syntax(test_file=os.path.realpath(__file__))
|
2018-03-05 11:14:34 -04:00
|
|
|
if not self.hasInit:
|
|
|
|
self.init()
|
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
self.fail_list = []
|
2018-03-05 11:14:34 -04:00
|
|
|
try:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Waiting for a heartbeat with mavlink protocol %s"
|
|
|
|
% self.mav.WIRE_PROTOCOL_VERSION)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mav.wait_heartbeat()
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Setting up RC parameters")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc_default()
|
|
|
|
self.set_rc(3, 1000)
|
|
|
|
self.set_rc(8, 1800)
|
2018-08-03 07:16:10 -03:00
|
|
|
|
2018-08-03 07:50:15 -03:00
|
|
|
self.set_parameter("RC12_OPTION", 9)
|
|
|
|
self.reboot_sitl() # needed for RC12_OPTION to take effect
|
|
|
|
|
|
|
|
self.run_test("Test RC Option - Camera Trigger",
|
|
|
|
self.test_rc_option_camera_trigger)
|
|
|
|
|
2018-08-03 07:16:10 -03:00
|
|
|
self.set_parameter("RC12_OPTION", 28)
|
|
|
|
self.reboot_sitl() # needed for RC12_OPTION to take effect
|
|
|
|
|
|
|
|
self.run_test("Test Relay RC Channel Option",
|
|
|
|
self.test_rc_relay)
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Waiting for GPS fix")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mav.recv_match(condition='VFR_HUD.alt>10', blocking=True)
|
|
|
|
self.mav.wait_gps_fix()
|
|
|
|
while self.mav.location().alt < 10:
|
|
|
|
self.mav.wait_gps_fix()
|
|
|
|
self.homeloc = self.mav.location()
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Home location: %s" % self.homeloc)
|
2018-08-03 06:42:19 -03:00
|
|
|
self.wait_ready_to_arm()
|
|
|
|
self.run_test("Arm features", self.test_arm_feature)
|
2018-04-27 15:21:53 -03:00
|
|
|
|
2018-08-13 08:07:34 -03:00
|
|
|
self.run_test("Flaps", self.fly_flaps)
|
|
|
|
|
2018-09-07 08:15:02 -03:00
|
|
|
self.mavproxy.send('switch 6\n')
|
|
|
|
self.wait_mode('MANUAL')
|
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
self.run_test("Takeoff", self.takeoff)
|
|
|
|
|
2018-05-31 07:27:41 -03:00
|
|
|
self.run_test("Set Attitude Target", self.set_attitude_target)
|
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
self.run_test("Fly left circuit", self.fly_left_circuit)
|
|
|
|
|
|
|
|
self.run_test("Left roll", lambda: self.axial_left_roll(1))
|
|
|
|
|
|
|
|
self.run_test("Inside loop", self.inside_loop)
|
|
|
|
|
|
|
|
self.run_test("Stablize test", self.test_stabilize)
|
|
|
|
|
|
|
|
self.run_test("ACRO test", self.test_acro)
|
|
|
|
|
|
|
|
self.run_test("FBWB test", self.test_FBWB)
|
|
|
|
|
|
|
|
self.run_test("CRUISE test", lambda: self.test_FBWB(mode='CRUISE'))
|
|
|
|
|
|
|
|
self.run_test("RTL test", self.fly_RTL)
|
|
|
|
|
|
|
|
self.run_test("LOITER test", self.fly_LOITER)
|
|
|
|
|
|
|
|
self.run_test("CIRCLE test", self.fly_CIRCLE)
|
|
|
|
|
2018-05-09 00:32:23 -03:00
|
|
|
self.run_test("Mission test",
|
|
|
|
lambda: self.fly_mission(
|
2018-07-31 06:47:49 -03:00
|
|
|
os.path.join(testdir, "ap1.txt")))
|
2018-04-27 15:21:53 -03:00
|
|
|
|
2018-05-09 00:32:23 -03:00
|
|
|
self.run_test("Log download",
|
|
|
|
lambda: self.log_download(
|
|
|
|
self.buildlogs_path("ArduPlane-log.bin")))
|
2018-04-27 15:21:53 -03:00
|
|
|
|
2018-07-31 06:50:02 -03:00
|
|
|
except pexpect.TIMEOUT:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Failed with timeout")
|
2018-04-27 15:21:53 -03:00
|
|
|
self.fail_list.append("timeout")
|
2013-07-16 00:23:48 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
self.close()
|
2011-11-12 07:13:17 -04:00
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
if len(self.fail_list):
|
|
|
|
self.progress("FAILED: %s" % self.fail_list)
|
2018-03-05 11:14:34 -04:00
|
|
|
return False
|
2018-09-07 20:48:45 -03:00
|
|
|
|
|
|
|
self.progress("Max set_rc_timeout=%s" % self.max_set_rc_timeout);
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
return True
|