2017-02-21 13:32:26 -04:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
# Drive APMrover2 in SITL
|
2016-11-08 07:06:05 -04:00
|
|
|
from __future__ import print_function
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
import os
|
|
|
|
import pexpect
|
|
|
|
import time
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
from common import AutoTest
|
2016-07-31 07:22:06 -03:00
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
from common import MsgRcvTimeoutException
|
|
|
|
from common import NotAchievedException
|
|
|
|
from common import PreconditionFailedException
|
|
|
|
|
2016-07-31 07:22:06 -03:00
|
|
|
from pysim import util
|
2012-11-27 19:43:11 -04:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
from pymavlink import mavutil
|
|
|
|
|
2012-11-27 19:43:11 -04:00
|
|
|
# get location of scripts
|
2016-07-31 07:22:06 -03:00
|
|
|
testdir = os.path.dirname(os.path.realpath(__file__))
|
2012-11-27 19:43:11 -04:00
|
|
|
|
2017-08-16 10:55:21 -03:00
|
|
|
# HOME = mavutil.location(-35.362938, 149.165085, 584, 270)
|
2018-03-14 08:08:53 -03:00
|
|
|
HOME = mavutil.location(40.071374969556928,
|
|
|
|
-105.22978898137808,
|
|
|
|
1583.702759,
|
|
|
|
246)
|
|
|
|
|
|
|
|
|
|
|
|
class AutoTestRover(AutoTest):
|
|
|
|
def __init__(self,
|
|
|
|
binary,
|
|
|
|
valgrind=False,
|
|
|
|
gdb=False,
|
2018-07-27 03:48:18 -03:00
|
|
|
speedup=8,
|
2018-03-14 08:08:53 -03:00
|
|
|
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(AutoTestRover, 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 = "APMrover2"
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def init(self):
|
|
|
|
if self.frame is None:
|
|
|
|
self.frame = 'rover'
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.sitl = util.start_SITL(self.binary,
|
|
|
|
model=self.frame,
|
|
|
|
home=self.home,
|
|
|
|
speedup=self.speedup,
|
|
|
|
valgrind=self.valgrind,
|
|
|
|
gdb=self.gdb,
|
2018-06-26 21:56:01 -03:00
|
|
|
gdbserver=self.gdbserver,
|
2018-07-23 04:06:00 -03:00
|
|
|
breakpoints=self.breakpoints,
|
2018-06-26 21:56:01 -03:00
|
|
|
wipe=True)
|
2018-03-15 08:31:19 -03:00
|
|
|
self.mavproxy = util.start_MAVProxy_SITL(
|
|
|
|
'APMrover2', 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
|
|
|
|
2018-06-26 21:56:01 -03:00
|
|
|
self.progress("WAITING FOR PARAMETERS")
|
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-06-26 21:56:01 -03:00
|
|
|
|
|
|
|
self.apply_defaultfile_parameters()
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Ready to start testing!")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
# def reset_and_arm(self):
|
|
|
|
# """Reset RC, set to MANUAL and arm."""
|
|
|
|
# self.mav.wait_heartbeat()
|
|
|
|
# # ensure all sticks in the middle
|
|
|
|
# self.set_rc_default()
|
|
|
|
# self.mavproxy.send('switch 1\n')
|
|
|
|
# self.mav.wait_heartbeat()
|
|
|
|
# self.disarm_vehicle()
|
|
|
|
# self.mav.wait_heartbeat()
|
|
|
|
# self.arm_vehicle()
|
|
|
|
#
|
2018-08-03 06:42:19 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
# # TEST RC OVERRIDE
|
|
|
|
# # TEST RC OVERRIDE TIMEOUT
|
|
|
|
# def test_rtl(self, home, distance_min=5, timeout=250):
|
|
|
|
# """Return, land."""
|
|
|
|
# super(AutotestRover, self).test_rtl(home, distance_min, timeout)
|
|
|
|
#
|
|
|
|
# def test_mission(self, filename):
|
|
|
|
# """Test a mission from a file."""
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("Test mission %s" % filename)
|
2018-03-05 11:14:34 -04:00
|
|
|
# num_wp = self.load_mission_from_file(filename)
|
|
|
|
# self.mavproxy.send('wp set 1\n')
|
|
|
|
# self.mav.wait_heartbeat()
|
|
|
|
# self.mavproxy.send('switch 4\n') # auto mode
|
|
|
|
# self.wait_mode('AUTO')
|
|
|
|
# ret = self.wait_waypoint(0, num_wp-1, max_dist=5, timeout=500)
|
|
|
|
#
|
|
|
|
# if ret:
|
|
|
|
# self.mavproxy.expect("Mission Complete")
|
|
|
|
# self.mav.wait_heartbeat()
|
|
|
|
# self.wait_mode('HOLD')
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("test: MISSION COMPLETE: passed=%s" % ret)
|
2018-03-05 11:14:34 -04:00
|
|
|
# return ret
|
|
|
|
|
|
|
|
##########################################################
|
|
|
|
# TESTS DRIVE
|
|
|
|
##########################################################
|
|
|
|
# Drive a square in manual mode
|
|
|
|
def drive_square(self, side=50):
|
|
|
|
"""Drive a square, Driving N then E ."""
|
|
|
|
|
2018-08-16 23:48:21 -03:00
|
|
|
self.context_push()
|
|
|
|
ex = None
|
|
|
|
try:
|
|
|
|
self.progress("TEST SQUARE")
|
|
|
|
self.set_parameter("RC7_OPTION", 7)
|
|
|
|
self.set_parameter("RC8_OPTION", 58)
|
|
|
|
|
|
|
|
# use LEARNING Mode
|
|
|
|
self.mavproxy.send('switch 5\n')
|
|
|
|
self.wait_mode('MANUAL')
|
|
|
|
|
|
|
|
# first aim north
|
|
|
|
self.progress("\nTurn right towards north")
|
|
|
|
self.reach_heading_manual(10)
|
|
|
|
# save bottom left corner of box as home AND waypoint
|
|
|
|
self.progress("Save HOME")
|
|
|
|
self.save_wp()
|
|
|
|
|
|
|
|
self.progress("Save WP")
|
|
|
|
self.save_wp()
|
|
|
|
|
|
|
|
# pitch forward to fly north
|
|
|
|
self.progress("\nGoing north %u meters" % side)
|
|
|
|
self.reach_distance_manual(side)
|
|
|
|
# save top left corner of square as waypoint
|
|
|
|
self.progress("Save WP")
|
|
|
|
self.save_wp()
|
|
|
|
|
|
|
|
# roll right to fly east
|
|
|
|
self.progress("\nGoing east %u meters" % side)
|
|
|
|
self.reach_heading_manual(100)
|
|
|
|
self.reach_distance_manual(side)
|
|
|
|
# save top right corner of square as waypoint
|
|
|
|
self.progress("Save WP")
|
|
|
|
self.save_wp()
|
|
|
|
|
|
|
|
# pitch back to fly south
|
|
|
|
self.progress("\nGoing south %u meters" % side)
|
|
|
|
self.reach_heading_manual(190)
|
|
|
|
self.reach_distance_manual(side)
|
|
|
|
# save bottom right corner of square as waypoint
|
|
|
|
self.progress("Save WP")
|
|
|
|
self.save_wp()
|
|
|
|
|
|
|
|
# roll left to fly west
|
|
|
|
self.progress("\nGoing west %u meters" % side)
|
|
|
|
self.reach_heading_manual(280)
|
|
|
|
self.reach_distance_manual(side)
|
|
|
|
# save bottom left corner of square (should be near home) as waypoint
|
|
|
|
self.progress("Save WP")
|
|
|
|
self.save_wp()
|
|
|
|
|
|
|
|
self.progress("Checking number of saved waypoints")
|
|
|
|
num_wp = self.save_mission_to_file(
|
|
|
|
os.path.join(testdir, "rover-ch7_mission.txt"))
|
|
|
|
if num_wp != 6:
|
|
|
|
raise NotAchievedException()
|
|
|
|
|
|
|
|
# TODO: actually drive the mission
|
|
|
|
self.progress("Clearing waypoints")
|
|
|
|
self.clear_wp()
|
|
|
|
self.mavproxy.send('wp list\n')
|
|
|
|
self.mavproxy.expect('Requesting 0 waypoints')
|
|
|
|
except Exception as e:
|
|
|
|
self.progress("Caught exception: %s" % str(e))
|
|
|
|
ex = e
|
|
|
|
self.context_pop()
|
|
|
|
if ex:
|
|
|
|
raise ex
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def drive_left_circuit(self):
|
|
|
|
"""Drive a left circuit, 50m on a side."""
|
|
|
|
self.mavproxy.send('switch 6\n')
|
|
|
|
self.wait_mode('MANUAL')
|
|
|
|
self.set_rc(3, 2000)
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Driving 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(50, accuracy=7)
|
2018-03-05 11:14:34 -04:00
|
|
|
self.set_rc(3, 1500)
|
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
|
|
|
# def test_throttle_failsafe(self, home, distance_min=10, side=60,
|
|
|
|
# timeout=300):
|
2018-03-05 11:14:34 -04:00
|
|
|
# """Fly east, Failsafe, return, land."""
|
|
|
|
#
|
|
|
|
# self.mavproxy.send('switch 6\n') # manual mode
|
|
|
|
# self.wait_mode('MANUAL')
|
|
|
|
# self.mavproxy.send("param set FS_ACTION 1\n")
|
|
|
|
#
|
|
|
|
# # first aim east
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("turn east")
|
2018-03-05 11:14:34 -04:00
|
|
|
# if not self.reach_heading_manual(135):
|
|
|
|
# return False
|
|
|
|
#
|
|
|
|
# # fly east 60 meters
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("# Going forward %u meters" % side)
|
2018-03-05 11:14:34 -04:00
|
|
|
# if not self.reach_distance_manual(side):
|
|
|
|
# return False
|
|
|
|
#
|
|
|
|
# # pull throttle low
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("# Enter Failsafe")
|
2018-03-05 11:14:34 -04:00
|
|
|
# self.mavproxy.send('rc 3 900\n')
|
|
|
|
#
|
|
|
|
# tstart = self.get_sim_time()
|
|
|
|
# success = False
|
|
|
|
# while self.get_sim_time() < tstart + timeout and not success:
|
|
|
|
# m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
# pos = self.mav.location()
|
|
|
|
# home_distance = self.get_distance(home, pos)
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("Alt: %u HomeDistance: %.0f" %
|
|
|
|
# (m.alt, home_distance))
|
2018-03-05 11:14:34 -04:00
|
|
|
# # check if we've reached home
|
|
|
|
# if home_distance <= distance_min:
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("RTL Complete")
|
2018-03-05 11:14:34 -04:00
|
|
|
# success = True
|
|
|
|
#
|
|
|
|
# # reduce throttle
|
|
|
|
# self.mavproxy.send('rc 3 1500\n')
|
|
|
|
# self.mavproxy.expect('APM: Failsafe ended')
|
|
|
|
# self.mavproxy.send('switch 2\n') # manual mode
|
|
|
|
# self.mav.wait_heartbeat()
|
|
|
|
# self.wait_mode('MANUAL')
|
|
|
|
#
|
|
|
|
# if success:
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("Reached failsafe home OK")
|
2018-03-05 11:14:34 -04:00
|
|
|
# return True
|
|
|
|
# else:
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("Failed to reach Home on failsafe RTL - "
|
|
|
|
# "timed out after %u seconds" % timeout)
|
2018-03-05 11:14:34 -04:00
|
|
|
# return False
|
|
|
|
|
2018-08-15 04:48:54 -03:00
|
|
|
def test_sprayer(self):
|
|
|
|
"""Test sprayer functionality."""
|
|
|
|
self.context_push()
|
|
|
|
ex = None
|
|
|
|
try:
|
|
|
|
rc_ch = 5
|
|
|
|
pump_ch = 5
|
|
|
|
spinner_ch = 6
|
|
|
|
pump_ch_min = 1050
|
|
|
|
pump_ch_trim = 1520
|
|
|
|
pump_ch_max = 1950
|
|
|
|
spinner_ch_min = 975
|
|
|
|
spinner_ch_trim = 1510
|
|
|
|
spinner_ch_max = 1975
|
|
|
|
|
|
|
|
self.set_parameter("SPRAY_ENABLE", 1)
|
|
|
|
|
|
|
|
self.set_parameter("SERVO%u_FUNCTION" % pump_ch, 22)
|
|
|
|
self.set_parameter("SERVO%u_MIN" % pump_ch, pump_ch_min)
|
|
|
|
self.set_parameter("SERVO%u_TRIM" % pump_ch, pump_ch_trim)
|
|
|
|
self.set_parameter("SERVO%u_MAX" % pump_ch, pump_ch_max)
|
|
|
|
|
|
|
|
self.set_parameter("SERVO%u_FUNCTION" % spinner_ch, 23)
|
|
|
|
self.set_parameter("SERVO%u_MIN" % spinner_ch, spinner_ch_min)
|
|
|
|
self.set_parameter("SERVO%u_TRIM" % spinner_ch, spinner_ch_trim)
|
|
|
|
self.set_parameter("SERVO%u_MAX" % spinner_ch, spinner_ch_max)
|
|
|
|
|
|
|
|
self.set_parameter("SIM_SPR_ENABLE", 1)
|
|
|
|
self.fetch_parameters()
|
|
|
|
self.set_parameter("SIM_SPR_PUMP", pump_ch)
|
|
|
|
self.set_parameter("SIM_SPR_SPIN", spinner_ch)
|
|
|
|
|
|
|
|
self.set_parameter("RC%u_OPTION" % rc_ch, 15)
|
|
|
|
self.set_parameter("LOG_DISARMED", 1)
|
|
|
|
|
|
|
|
self.reboot_sitl()
|
|
|
|
|
|
|
|
self.wait_ready_to_arm()
|
|
|
|
self.arm_vehicle()
|
|
|
|
|
|
|
|
self.progress("test bootup state - it's zero-output!")
|
|
|
|
self.wait_servo_channel_value(spinner_ch, 0)
|
|
|
|
self.wait_servo_channel_value(pump_ch, 0)
|
|
|
|
|
|
|
|
self.progress("Enable sprayer")
|
|
|
|
self.set_rc(rc_ch, 2000)
|
|
|
|
|
|
|
|
self.progress("Testing zero-speed state")
|
|
|
|
self.wait_servo_channel_value(spinner_ch, spinner_ch_min)
|
|
|
|
self.wait_servo_channel_value(pump_ch, pump_ch_min)
|
|
|
|
|
|
|
|
self.progress("Testing turning it off")
|
|
|
|
self.set_rc(rc_ch, 1000)
|
|
|
|
self.wait_servo_channel_value(spinner_ch, spinner_ch_min)
|
|
|
|
self.wait_servo_channel_value(pump_ch, pump_ch_min)
|
|
|
|
|
|
|
|
self.progress("Testing turning it back on")
|
|
|
|
self.set_rc(rc_ch, 2000)
|
|
|
|
self.wait_servo_channel_value(spinner_ch, spinner_ch_min)
|
|
|
|
self.wait_servo_channel_value(pump_ch, pump_ch_min)
|
|
|
|
|
|
|
|
self.progress("Testing speed-ramping")
|
|
|
|
self.set_rc(3, 1700) # start driving forward
|
|
|
|
|
|
|
|
# this is somewhat empirical...
|
|
|
|
self.wait_servo_channel_value(pump_ch, 1695, timeout=60)
|
|
|
|
|
|
|
|
self.progress("Sprayer OK")
|
|
|
|
except Exception as e:
|
|
|
|
ex = e
|
|
|
|
self.context_pop()
|
|
|
|
self.reboot_sitl()
|
|
|
|
if ex:
|
|
|
|
raise ex
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
#################################################
|
|
|
|
# AUTOTEST ALL
|
|
|
|
#################################################
|
|
|
|
def drive_mission(self, filename):
|
|
|
|
"""Drive a mission from a file."""
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Driving mission %s" % 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')
|
|
|
|
self.mavproxy.send('switch 4\n') # auto mode
|
|
|
|
self.set_rc(3, 1500)
|
|
|
|
self.wait_mode('AUTO')
|
2018-04-27 15:21:53 -03:00
|
|
|
self.wait_waypoint(1, 4, max_dist=5)
|
2018-04-14 08:31:22 -03:00
|
|
|
self.wait_mode('HOLD', timeout=300)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Mission OK")
|
2018-04-27 15:21:53 -03:00
|
|
|
|
|
|
|
def drive_mission_rover1(self):
|
|
|
|
self.drive_mission(os.path.join(testdir, "rover1.txt"))
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def do_get_banner(self):
|
|
|
|
self.mavproxy.send("long DO_SEND_BANNER 1\n")
|
|
|
|
start = time.time()
|
|
|
|
while True:
|
2018-03-14 08:08:53 -03:00
|
|
|
m = self.mav.recv_match(type='STATUSTEXT',
|
|
|
|
blocking=True,
|
|
|
|
timeout=1)
|
2018-03-05 11:14:34 -04:00
|
|
|
if m is not None and "ArduRover" in m.text:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("banner received: %s" % m.text)
|
2018-04-27 15:21:53 -03:00
|
|
|
return
|
2018-03-05 11:14:34 -04:00
|
|
|
if time.time() - start > 10:
|
|
|
|
break
|
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("banner not received")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise MsgRcvTimeoutException()
|
2017-12-09 00:32:34 -04:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def drive_brake_get_stopping_distance(self, speed):
|
|
|
|
# measure our stopping distance:
|
|
|
|
old_cruise_speed = self.get_parameter('CRUISE_SPEED')
|
|
|
|
old_accel_max = self.get_parameter('ATC_ACCEL_MAX')
|
|
|
|
|
|
|
|
# controller tends not to meet cruise speed (max of ~14 when 15
|
|
|
|
# set), thus *1.2
|
|
|
|
self.set_parameter('CRUISE_SPEED', speed*1.2)
|
|
|
|
# at time of writing, the vehicle is only capable of 10m/s/s accel
|
|
|
|
self.set_parameter('ATC_ACCEL_MAX', 15)
|
|
|
|
self.mavproxy.send("mode STEERING\n")
|
|
|
|
self.wait_mode('STEERING')
|
|
|
|
self.set_rc(3, 2000)
|
|
|
|
self.wait_groundspeed(15, 100)
|
|
|
|
initial = self.mav.location()
|
|
|
|
initial_time = time.time()
|
|
|
|
while time.time() - initial_time < 2:
|
|
|
|
# wait for a position update from the autopilot
|
|
|
|
start = self.mav.location()
|
|
|
|
if start != initial:
|
|
|
|
break
|
|
|
|
self.set_rc(3, 1500)
|
|
|
|
self.wait_groundspeed(0, 0.2) # why do we not stop?!
|
|
|
|
initial = self.mav.location()
|
|
|
|
initial_time = time.time()
|
|
|
|
while time.time() - initial_time < 2:
|
|
|
|
# wait for a position update from the autopilot
|
|
|
|
stop = self.mav.location()
|
|
|
|
if stop != initial:
|
|
|
|
break
|
|
|
|
delta = self.get_distance(start, stop)
|
|
|
|
|
|
|
|
self.set_parameter('CRUISE_SPEED', old_cruise_speed)
|
|
|
|
self.set_parameter('ATC_ACCEL_MAX', old_accel_max)
|
|
|
|
|
|
|
|
return delta
|
|
|
|
|
|
|
|
def drive_brake(self):
|
|
|
|
old_using_brake = self.get_parameter('ATC_BRAKE')
|
|
|
|
old_cruise_speed = self.get_parameter('CRUISE_SPEED')
|
|
|
|
|
|
|
|
self.set_parameter('CRUISE_SPEED', 15)
|
|
|
|
self.set_parameter('ATC_BRAKE', 0)
|
|
|
|
|
|
|
|
distance_without_brakes = self.drive_brake_get_stopping_distance(15)
|
|
|
|
|
|
|
|
# brakes on:
|
|
|
|
self.set_parameter('ATC_BRAKE', 1)
|
|
|
|
distance_with_brakes = self.drive_brake_get_stopping_distance(15)
|
|
|
|
# revert state:
|
|
|
|
self.set_parameter('ATC_BRAKE', old_using_brake)
|
|
|
|
self.set_parameter('CRUISE_SPEED', old_cruise_speed)
|
|
|
|
|
|
|
|
delta = distance_without_brakes - distance_with_brakes
|
|
|
|
if delta < distance_without_brakes * 0.05: # 5% isn't asking for much
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Brakes have negligible effect"
|
|
|
|
"(with=%0.2fm without=%0.2fm delta=%0.2fm)" %
|
|
|
|
(distance_with_brakes,
|
|
|
|
distance_without_brakes,
|
|
|
|
delta))
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
self.progress(
|
|
|
|
"Brakes work (with=%0.2fm without=%0.2fm delta=%0.2fm)" %
|
|
|
|
(distance_with_brakes, distance_without_brakes, delta))
|
2018-03-05 11:14:34 -04:00
|
|
|
|
|
|
|
def drive_rtl_mission(self):
|
2018-03-14 08:08:53 -03:00
|
|
|
mission_filepath = os.path.join(testdir,
|
|
|
|
"ArduRover-Missions",
|
|
|
|
"rtl.txt")
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('wp load %s\n' % mission_filepath)
|
|
|
|
self.mavproxy.expect('Flight plan received')
|
|
|
|
self.mavproxy.send('switch 4\n') # auto mode
|
|
|
|
self.set_rc(3, 1500)
|
|
|
|
self.wait_mode('AUTO')
|
|
|
|
self.mavproxy.expect('Executing RTL')
|
|
|
|
|
|
|
|
m = self.mav.recv_match(type='NAV_CONTROLLER_OUTPUT',
|
|
|
|
blocking=True,
|
|
|
|
timeout=0.1)
|
|
|
|
if m is None:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Did not receive NAV_CONTROLLER_OUTPUT message")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise MsgRcvTimeoutException()
|
2017-08-16 10:55:21 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
wp_dist_min = 5
|
|
|
|
if m.wp_dist < wp_dist_min:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Did not start at least 5 metres from destination")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise PreconditionFailedException()
|
2017-08-16 10:55:21 -03:00
|
|
|
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("NAV_CONTROLLER_OUTPUT.wp_dist looks good (%u >= %u)" %
|
|
|
|
(m.wp_dist, wp_dist_min,))
|
2017-08-27 21:57:59 -03:00
|
|
|
|
2018-08-15 22:46:16 -03:00
|
|
|
self.wait_mode('HOLD', timeout=600) # balancebot can take a long time!
|
2017-09-16 08:47:46 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
pos = self.mav.location()
|
|
|
|
home_distance = self.get_distance(HOME, pos)
|
|
|
|
home_distance_max = 5
|
|
|
|
if home_distance > home_distance_max:
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Did not get home (%u metres distant > %u)" %
|
|
|
|
(home_distance, home_distance_max))
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('switch 6\n')
|
|
|
|
self.wait_mode('MANUAL')
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("RTL Mission OK")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
2018-04-18 01:02:38 -03:00
|
|
|
def test_servorelayevents(self):
|
|
|
|
self.mavproxy.send("relay set 0 0\n")
|
|
|
|
off = self.get_parameter("SIM_PIN_MASK")
|
|
|
|
self.mavproxy.send("relay set 0 1\n")
|
|
|
|
on = self.get_parameter("SIM_PIN_MASK")
|
|
|
|
if on == off:
|
|
|
|
self.progress("Pin mask unchanged after relay command")
|
2018-04-27 15:21:53 -03:00
|
|
|
raise NotAchievedException()
|
2018-04-18 01:02:38 -03:00
|
|
|
self.progress("Pin mask changed after relay command")
|
|
|
|
|
2018-04-14 08:31:22 -03:00
|
|
|
def test_setting_modes_via_mavproxy_switch(self):
|
|
|
|
fnoo = [(1, 'MANUAL'),
|
|
|
|
(2, 'MANUAL'),
|
|
|
|
(3, 'RTL'),
|
2018-07-31 06:50:02 -03:00
|
|
|
# (4, 'AUTO'), # no mission, can't set auto
|
|
|
|
(5, 'RTL'), # non-existant mode, should stay in RTL
|
2018-04-14 08:31:22 -03:00
|
|
|
(6, 'MANUAL')]
|
|
|
|
for (num, expected) in fnoo:
|
|
|
|
self.mavproxy.send('switch %u\n' % num)
|
|
|
|
self.wait_mode(expected)
|
|
|
|
|
2018-08-16 00:07:15 -03:00
|
|
|
def test_setting_modes_via_mavproxy_mode_command(self):
|
|
|
|
fnoo = [(1, 'ACRO'),
|
|
|
|
(3, 'STEERING'),
|
|
|
|
(4, 'HOLD'),
|
|
|
|
]
|
|
|
|
for (num, expected) in fnoo:
|
|
|
|
self.mavproxy.send('mode manual\n')
|
|
|
|
self.wait_mode("MANUAL")
|
|
|
|
self.mavproxy.send('mode %u\n' % num)
|
|
|
|
self.wait_mode(expected)
|
|
|
|
self.mavproxy.send('mode manual\n')
|
|
|
|
self.wait_mode("MANUAL")
|
|
|
|
self.mavproxy.send('mode %s\n' % expected)
|
|
|
|
self.wait_mode(expected)
|
|
|
|
|
2018-04-14 08:31:22 -03:00
|
|
|
def test_setting_modes_via_modeswitch(self):
|
|
|
|
# test setting of modes through mode switch
|
2018-07-31 06:50:02 -03:00
|
|
|
self.context_push()
|
2018-04-14 08:31:22 -03:00
|
|
|
ex = None
|
|
|
|
try:
|
|
|
|
self.set_parameter("MODE_CH", 8)
|
|
|
|
self.set_rc(8, 1000)
|
|
|
|
# mavutil.mavlink.ROVER_MODE_HOLD:
|
|
|
|
self.set_parameter("MODE6", 4)
|
|
|
|
# mavutil.mavlink.ROVER_MODE_ACRO
|
|
|
|
self.set_parameter("MODE5", 1)
|
|
|
|
self.set_rc(8, 1800) # PWM for mode6
|
|
|
|
self.wait_mode("HOLD")
|
|
|
|
self.set_rc(8, 1700) # PWM for mode5
|
|
|
|
self.wait_mode("ACRO")
|
|
|
|
self.set_rc(8, 1800) # PWM for mode6
|
|
|
|
self.wait_mode("HOLD")
|
|
|
|
self.set_rc(8, 1700) # PWM for mode5
|
|
|
|
self.wait_mode("ACRO")
|
|
|
|
except Exception as e:
|
|
|
|
self.progress("Exception caught")
|
|
|
|
ex = e
|
|
|
|
|
2018-07-31 06:50:02 -03:00
|
|
|
self.context_pop()
|
2018-04-14 08:31:22 -03:00
|
|
|
|
|
|
|
if ex is not None:
|
|
|
|
raise ex
|
|
|
|
|
|
|
|
def test_setting_modes_via_auxswitches(self):
|
2018-07-31 06:50:02 -03:00
|
|
|
self.context_push()
|
2018-04-14 08:31:22 -03:00
|
|
|
ex = None
|
|
|
|
try:
|
|
|
|
self.set_parameter("MODE5", 1)
|
|
|
|
self.mavproxy.send('switch 5\n') # acro mode
|
|
|
|
self.wait_mode("ACRO")
|
|
|
|
self.set_rc(9, 1000)
|
|
|
|
self.set_rc(10, 1000)
|
|
|
|
self.set_parameter("RC9_OPTION", 53) # steering
|
|
|
|
self.set_parameter("RC10_OPTION", 54) # hold
|
|
|
|
self.set_rc(9, 1900)
|
|
|
|
self.wait_mode("STEERING")
|
|
|
|
self.set_rc(10, 1900)
|
|
|
|
self.wait_mode("HOLD")
|
|
|
|
|
|
|
|
# reset both switches - should go back to ACRO
|
|
|
|
self.set_rc(9, 1000)
|
|
|
|
self.set_rc(10, 1000)
|
|
|
|
self.wait_mode("ACRO")
|
|
|
|
|
|
|
|
self.set_rc(9, 1900)
|
|
|
|
self.wait_mode("STEERING")
|
|
|
|
self.set_rc(10, 1900)
|
|
|
|
self.wait_mode("HOLD")
|
|
|
|
|
|
|
|
self.set_rc(10, 1000) # this re-polls the mode switch
|
|
|
|
self.wait_mode("ACRO")
|
|
|
|
self.set_rc(9, 1000)
|
|
|
|
except Exception as e:
|
|
|
|
self.progress("Exception caught")
|
|
|
|
ex = e
|
|
|
|
|
2018-07-31 06:50:02 -03:00
|
|
|
self.context_pop()
|
2018-04-14 08:31:22 -03:00
|
|
|
|
|
|
|
if ex is not None:
|
|
|
|
raise ex
|
|
|
|
|
2018-08-02 07:46:58 -03:00
|
|
|
def test_rc_overrides(self):
|
|
|
|
self.context_push();
|
|
|
|
ex = None
|
|
|
|
try:
|
2018-08-03 03:20:25 -03:00
|
|
|
self.set_parameter("RC12_OPTION", 46)
|
|
|
|
self.reboot_sitl()
|
|
|
|
|
2018-08-02 07:46:58 -03:00
|
|
|
self.mavproxy.send('switch 6\n') # Manual mode
|
|
|
|
self.wait_mode('MANUAL')
|
|
|
|
self.wait_ready_to_arm()
|
|
|
|
self.mavproxy.send('rc 3 1500\n') # throttle at zero
|
|
|
|
self.arm_vehicle()
|
|
|
|
# start moving forward a little:
|
|
|
|
normal_rc_throttle = 1700
|
|
|
|
self.mavproxy.send('rc 3 %u\n' % normal_rc_throttle)
|
|
|
|
self.wait_groundspeed(5, 100)
|
2018-08-03 03:20:25 -03:00
|
|
|
|
|
|
|
# allow overrides:
|
|
|
|
self.set_rc(12, 2000)
|
|
|
|
|
|
|
|
# now override to stop:
|
|
|
|
throttle_override = 1500
|
2018-08-02 07:46:58 -03:00
|
|
|
while True:
|
|
|
|
print("Sending throttle of %u" % (throttle_override,))
|
|
|
|
self.mav.mav.rc_channels_override_send(
|
|
|
|
1, # target system
|
|
|
|
1, # targe component
|
|
|
|
65535, # chan1_raw
|
|
|
|
65535, # chan2_raw
|
|
|
|
throttle_override, # chan3_raw
|
|
|
|
65535, # chan4_raw
|
|
|
|
65535, # chan5_raw
|
|
|
|
65535, # chan6_raw
|
|
|
|
65535, # chan7_raw
|
|
|
|
65535) # chan8_raw
|
|
|
|
|
2018-08-03 03:20:25 -03:00
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
want_speed = 2.0
|
|
|
|
print("Speed=%f want=<%f" % (m.groundspeed, want_speed))
|
|
|
|
if m.groundspeed < want_speed:
|
|
|
|
break
|
2018-08-02 07:46:58 -03:00
|
|
|
|
2018-08-03 03:20:25 -03:00
|
|
|
# now override to stop - but set the switch on the RC
|
|
|
|
# transmitter to deny overrides; this should send the
|
|
|
|
# speed back up to 5 metres/second:
|
|
|
|
self.set_rc(12, 1000)
|
|
|
|
|
|
|
|
throttle_override = 1500
|
|
|
|
while True:
|
|
|
|
print("Sending throttle of %u" % (throttle_override,))
|
|
|
|
self.mav.mav.rc_channels_override_send(
|
|
|
|
1, # target system
|
|
|
|
1, # targe component
|
|
|
|
65535, # chan1_raw
|
|
|
|
65535, # chan2_raw
|
|
|
|
throttle_override, # chan3_raw
|
|
|
|
65535, # chan4_raw
|
|
|
|
65535, # chan5_raw
|
|
|
|
65535, # chan6_raw
|
|
|
|
65535, # chan7_raw
|
|
|
|
65535) # chan8_raw
|
|
|
|
|
|
|
|
m = self.mav.recv_match(type='VFR_HUD', blocking=True)
|
|
|
|
want_speed = 5.0
|
|
|
|
print("Speed=%f want=>%f" % (m.groundspeed, want_speed))
|
|
|
|
|
|
|
|
if m.groundspeed > want_speed:
|
2018-08-02 07:46:58 -03:00
|
|
|
break
|
2018-08-03 03:20:25 -03:00
|
|
|
|
|
|
|
# re-enable RC overrides
|
|
|
|
self.set_rc(12, 2000)
|
|
|
|
|
2018-08-02 07:46:58 -03:00
|
|
|
# check we revert to normal RC inputs when gcs overrides cease:
|
|
|
|
self.progress("Waiting for RC to revert to normal RC input")
|
|
|
|
while True:
|
|
|
|
m = self.mav.recv_match(type='RC_CHANNELS_RAW', blocking=True)
|
|
|
|
print("%s" % m)
|
|
|
|
if m.chan3_raw == normal_rc_throttle:
|
|
|
|
break
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
self.progress("Exception caught")
|
|
|
|
ex = e
|
|
|
|
|
|
|
|
self.context_pop();
|
2018-08-03 03:20:25 -03:00
|
|
|
self.reboot_sitl()
|
2018-08-02 07:46:58 -03:00
|
|
|
|
|
|
|
if ex is not None:
|
|
|
|
raise ex
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
def autotest(self):
|
|
|
|
"""Autotest APMrover2 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-03-14 08:08:53 -03:00
|
|
|
self.progress("Started simulator")
|
2018-03-05 11:14:34 -04:00
|
|
|
|
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(8, 1800)
|
2018-03-14 08:08:53 -03:00
|
|
|
self.progress("Waiting for GPS fix")
|
2018-03-05 11:14:34 -04:00
|
|
|
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-02 07:46:58 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
self.mavproxy.send('switch 6\n') # Manual mode
|
|
|
|
self.wait_mode('MANUAL')
|
2018-08-15 04:48:54 -03:00
|
|
|
|
|
|
|
self.run_test("Test Sprayer", self.test_sprayer)
|
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
self.wait_ready_to_arm()
|
2018-08-03 06:42:19 -03:00
|
|
|
self.run_test("Arm features", self.test_arm_feature)
|
2018-04-27 15:21:53 -03:00
|
|
|
self.arm_vehicle()
|
|
|
|
|
2018-04-14 08:31:22 -03:00
|
|
|
self.run_test("Set modes via mavproxy switch",
|
|
|
|
self.test_setting_modes_via_mavproxy_switch)
|
|
|
|
|
2018-08-16 00:07:15 -03:00
|
|
|
self.run_test("Set modes via mavproxy mode command",
|
|
|
|
self.test_setting_modes_via_mavproxy_mode_command)
|
|
|
|
|
2018-04-14 08:31:22 -03:00
|
|
|
self.run_test("Set modes via modeswitch",
|
|
|
|
self.test_setting_modes_via_modeswitch)
|
|
|
|
|
|
|
|
self.run_test("Set modes via auxswitches",
|
|
|
|
self.test_setting_modes_via_auxswitches)
|
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
self.run_test("Drive an RTL Mission", self.drive_rtl_mission)
|
|
|
|
|
|
|
|
self.run_test("Learn/Drive Square with Ch7 option",
|
|
|
|
self.drive_square)
|
|
|
|
|
|
|
|
self.run_test("Drive Mission %s" % "rover1.txt",
|
|
|
|
self.drive_mission_rover1)
|
|
|
|
|
2018-07-31 23:31:12 -03:00
|
|
|
# disabled due to frequent failures in travis. This test needs re-writing
|
2018-07-31 06:50:02 -03:00
|
|
|
# self.run_test("Drive Brake", self.drive_brake)
|
2018-04-27 15:21:53 -03:00
|
|
|
|
|
|
|
self.run_test("Disarm Vehicle", self.disarm_vehicle)
|
|
|
|
|
|
|
|
self.run_test("Get Banner", self.do_get_banner)
|
|
|
|
|
|
|
|
self.run_test("Get Capabilities",
|
|
|
|
self.do_get_autopilot_capabilities)
|
|
|
|
|
|
|
|
self.run_test("Set mode via MAV_COMMAND_DO_SET_MODE",
|
|
|
|
self.do_set_mode_via_command_long)
|
|
|
|
|
|
|
|
self.run_test("Test ServoRelayEvents",
|
|
|
|
self.test_servorelayevents)
|
|
|
|
|
2018-08-02 07:46:58 -03:00
|
|
|
self.disarm_vehicle()
|
|
|
|
|
|
|
|
self.run_test("Test RC overrides", self.test_rc_overrides)
|
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
self.run_test("Download logs", lambda:
|
2018-05-09 00:32:23 -03:00
|
|
|
self.log_download(
|
|
|
|
self.buildlogs_path("APMrover2-log.bin")))
|
2018-03-05 11:14:34 -04:00
|
|
|
# if not drive_left_circuit(self):
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("Failed left circuit")
|
2018-03-05 11:14:34 -04:00
|
|
|
# failed = True
|
|
|
|
# if not drive_RTL(self):
|
2018-03-14 08:08:53 -03:00
|
|
|
# self.progress("Failed RTL")
|
2018-03-05 11:14:34 -04:00
|
|
|
# failed = True
|
|
|
|
|
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-05-09 00:32:23 -03:00
|
|
|
self.fail_list.append(("*timeout*", None))
|
2017-09-16 08:47:46 -03:00
|
|
|
|
2018-03-05 11:14:34 -04:00
|
|
|
self.close()
|
2017-09-16 08:47:46 -03:00
|
|
|
|
2018-04-27 15:21:53 -03:00
|
|
|
if len(self.fail_list):
|
|
|
|
self.progress("FAILED STEPS: %s" % self.fail_list)
|
2018-03-05 11:14:34 -04:00
|
|
|
return False
|
|
|
|
return True
|