2017-08-25 08:43:19 -03:00
from math import sqrt
2022-07-18 11:09:04 -03:00
import DataflashLog
2017-02-07 02:47:57 -04:00
import matplotlib . pyplot as plt
2022-07-18 11:09:04 -03:00
import numpy as np
from LogAnalyzer import Test , TestResult
2017-02-07 02:47:57 -04:00
class TestFlow ( Test ) :
''' test optical flow sensor scale factor calibration '''
2022-07-18 11:09:04 -03:00
2017-02-07 02:47:57 -04:00
#
# Use the following procedure to log the calibration data. is assumed that the optical flow sensor has been
# correctly aligned, is focussed and the test is performed over a textured surface with adequate lighting.
# Note that the strobing effect from non incandescent artifical lighting can produce poor optical flow measurements.
#
2019-04-04 02:12:01 -03:00
# 1) Set LOG_DISARMED and FLOW_TYPE to 10 and verify that ATT and OF messages are being logged onboard
2017-02-07 02:47:57 -04:00
# 2) Place on level ground, apply power and wait for EKF to complete attitude alignment
# 3) Keeping the copter level, lift it to shoulder height and rock between +-20 and +-30 degrees
# in roll about an axis that passes through the flow sensor lens assembly. The time taken to rotate from
# maximum left roll to maximum right roll should be about 1 second.
# 4) Repeat 3) about the pitch axis
# 5) Holding the copter level, lower it to the ground and remove power
# 6) Transfer the logfile from the sdcard.
# 7) Open a terminal and cd to the ardupilot/Tools/LogAnalyzer directory
# 8) Enter to run the analysis 'python LogAnalyzer.py <log file name including full path>'
# 9) Check the OpticalFlow test status printed to the screen. The analysis plots are saved to
# flow_calibration.pdf and the recommended scale factors to flow_calibration.param
def __init__ ( self ) :
Test . __init__ ( self )
self . name = " OpticalFlow "
def run ( self , logdata , verbose ) :
self . result = TestResult ( )
self . result . status = TestResult . StatusType . GOOD
def FAIL ( ) :
self . result . status = TestResult . StatusType . FAIL
def WARN ( ) :
if self . result . status != TestResult . StatusType . FAIL :
self . result . status = TestResult . StatusType . WARN
try :
# tuning parameters used by the algorithm
2022-07-18 11:09:04 -03:00
tilt_threshold = 15 # roll and pitch threshold used to start and stop calibration (deg)
quality_threshold = 124 # minimum flow quality required for data to be used by the curve fit (N/A)
min_rate_threshold = (
0.0 # if the gyro rate is less than this, the data will not be used by the curve fit (rad/sec)
)
max_rate_threshold = (
2.0 # if the gyro rate is greter than this, the data will not be used by the curve fit (rad/sec)
)
param_std_threshold = 5.0 # maximum allowable 1-std uncertainty in scaling parameter (scale factor * 1000)
param_abs_threshold = 200 # max/min allowable scale factor parameter. Values of FLOW_FXSCALER and FLOW_FYSCALER outside the range of +-param_abs_threshold indicate a sensor configuration problem.
min_num_points = 100 # minimum number of points required for a curve fit - this is necessary, but not sufficient condition - the standard deviation estimate of the fit gradient is also important.
2017-02-07 02:47:57 -04:00
# get the existing scale parameters
flow_fxscaler = logdata . parameters [ " FLOW_FXSCALER " ]
flow_fyscaler = logdata . parameters [ " FLOW_FYSCALER " ]
# load required optical flow data
if " OF " in logdata . channels :
flowX = np . zeros ( len ( logdata . channels [ " OF " ] [ " flowX " ] . listData ) )
for i in range ( len ( logdata . channels [ " OF " ] [ " flowX " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , flowX [ i ] ) = logdata . channels [ " OF " ] [ " flowX " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
bodyX = np . zeros ( len ( logdata . channels [ " OF " ] [ " bodyX " ] . listData ) )
for i in range ( len ( logdata . channels [ " OF " ] [ " bodyX " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , bodyX [ i ] ) = logdata . channels [ " OF " ] [ " bodyX " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
flowY = np . zeros ( len ( logdata . channels [ " OF " ] [ " flowY " ] . listData ) )
for i in range ( len ( logdata . channels [ " OF " ] [ " flowY " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , flowY [ i ] ) = logdata . channels [ " OF " ] [ " flowY " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
bodyY = np . zeros ( len ( logdata . channels [ " OF " ] [ " bodyY " ] . listData ) )
for i in range ( len ( logdata . channels [ " OF " ] [ " bodyY " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , bodyY [ i ] ) = logdata . channels [ " OF " ] [ " bodyY " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
flow_time_us = np . zeros ( len ( logdata . channels [ " OF " ] [ " TimeUS " ] . listData ) )
for i in range ( len ( logdata . channels [ " OF " ] [ " TimeUS " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , flow_time_us [ i ] ) = logdata . channels [ " OF " ] [ " TimeUS " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
flow_qual = np . zeros ( len ( logdata . channels [ " OF " ] [ " Qual " ] . listData ) )
for i in range ( len ( logdata . channels [ " OF " ] [ " Qual " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , flow_qual [ i ] ) = logdata . channels [ " OF " ] [ " Qual " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
else :
FAIL ( )
self . result . statusMessage = " FAIL: no optical flow data \n "
return
# load required attitude data
if " ATT " in logdata . channels :
Roll = np . zeros ( len ( logdata . channels [ " ATT " ] [ " Roll " ] . listData ) )
for i in range ( len ( logdata . channels [ " ATT " ] [ " Roll " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , Roll [ i ] ) = logdata . channels [ " ATT " ] [ " Roll " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
Pitch = np . zeros ( len ( logdata . channels [ " ATT " ] [ " Pitch " ] . listData ) )
for i in range ( len ( logdata . channels [ " ATT " ] [ " Pitch " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , Pitch [ i ] ) = logdata . channels [ " ATT " ] [ " Pitch " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
att_time_us = np . zeros ( len ( logdata . channels [ " ATT " ] [ " TimeUS " ] . listData ) )
for i in range ( len ( logdata . channels [ " ATT " ] [ " TimeUS " ] . listData ) ) :
2022-07-18 11:09:04 -03:00
( line , att_time_us [ i ] ) = logdata . channels [ " ATT " ] [ " TimeUS " ] . listData [ i ]
2017-02-07 02:47:57 -04:00
else :
FAIL ( )
self . result . statusMessage = " FAIL: no attitude data \n "
return
# calculate the start time for the roll calibration
startTime = int ( 0 )
startRollIndex = int ( 0 )
for i in range ( len ( Roll ) ) :
if abs ( Roll [ i ] ) > tilt_threshold :
startTime = att_time_us [ i ]
break
for i in range ( len ( flow_time_us ) ) :
if flow_time_us [ i ] > startTime :
startRollIndex = i
break
# calculate the end time for the roll calibration
endTime = int ( 0 )
endRollIndex = int ( 0 )
2022-07-18 11:09:04 -03:00
for i in range ( len ( Roll ) - 1 , - 1 , - 1 ) :
2017-02-07 02:47:57 -04:00
if abs ( Roll [ i ] ) > tilt_threshold :
endTime = att_time_us [ i ]
break
2022-07-18 11:09:04 -03:00
for i in range ( len ( flow_time_us ) - 1 , - 1 , - 1 ) :
2017-02-07 02:47:57 -04:00
if flow_time_us [ i ] < endTime :
endRollIndex = i
break
# check we have enough roll data points
2022-07-18 11:09:04 -03:00
if endRollIndex - startRollIndex < = min_num_points :
2017-02-07 02:47:57 -04:00
FAIL ( )
self . result . statusMessage = " FAIL: insufficient roll data pointsa \n "
return
# resample roll test data excluding data before first movement and after last movement
# also exclude data where there is insufficient angular rate
flowX_resampled = [ ]
bodyX_resampled = [ ]
flowX_time_us_resampled = [ ]
for i in range ( len ( Roll ) ) :
2022-07-18 11:09:04 -03:00
if (
( i > = startRollIndex )
and ( i < = endRollIndex )
and ( abs ( bodyX [ i ] ) > min_rate_threshold )
and ( abs ( bodyX [ i ] ) < max_rate_threshold )
and ( flow_qual [ i ] > quality_threshold )
) :
2017-02-07 02:47:57 -04:00
flowX_resampled . append ( flowX [ i ] )
bodyX_resampled . append ( bodyX [ i ] )
flowX_time_us_resampled . append ( flow_time_us [ i ] )
# calculate the start time for the pitch calibration
startTime = 0
startPitchIndex = int ( 0 )
for i in range ( len ( Pitch ) ) :
if abs ( Pitch [ i ] ) > tilt_threshold :
startTime = att_time_us [ i ]
break
for i in range ( len ( flow_time_us ) ) :
if flow_time_us [ i ] > startTime :
startPitchIndex = i
break
# calculate the end time for the pitch calibration
endTime = 0
endPitchIndex = int ( 0 )
2022-07-18 11:09:04 -03:00
for i in range ( len ( Pitch ) - 1 , - 1 , - 1 ) :
2017-02-07 02:47:57 -04:00
if abs ( Pitch [ i ] ) > tilt_threshold :
endTime = att_time_us [ i ]
break
2022-07-18 11:09:04 -03:00
for i in range ( len ( flow_time_us ) - 1 , - 1 , - 1 ) :
2017-02-07 02:47:57 -04:00
if flow_time_us [ i ] < endTime :
endPitchIndex = i
break
# check we have enough pitch data points
2022-07-18 11:09:04 -03:00
if endPitchIndex - startPitchIndex < = min_num_points :
2017-02-07 02:47:57 -04:00
FAIL ( )
self . result . statusMessage = " FAIL: insufficient pitch data pointsa \n "
return
# resample pitch test data excluding data before first movement and after last movement
# also exclude data where there is insufficient or too much angular rate
flowY_resampled = [ ]
bodyY_resampled = [ ]
flowY_time_us_resampled = [ ]
for i in range ( len ( Roll ) ) :
2022-07-18 11:09:04 -03:00
if (
( i > = startPitchIndex )
and ( i < = endPitchIndex )
and ( abs ( bodyY [ i ] ) > min_rate_threshold )
and ( abs ( bodyY [ i ] ) < max_rate_threshold )
and ( flow_qual [ i ] > quality_threshold )
) :
2017-02-07 02:47:57 -04:00
flowY_resampled . append ( flowY [ i ] )
bodyY_resampled . append ( bodyY [ i ] )
flowY_time_us_resampled . append ( flow_time_us [ i ] )
# fit a straight line to the flow vs body rate data and calculate the scale factor parameter required to achieve a slope of 1
2022-07-18 11:09:04 -03:00
coef_flow_x , cov_x = np . polyfit (
bodyX_resampled , flowX_resampled , 1 , rcond = None , full = False , w = None , cov = True
)
coef_flow_y , cov_y = np . polyfit (
bodyY_resampled , flowY_resampled , 1 , rcond = None , full = False , w = None , cov = True
)
2017-02-07 02:47:57 -04:00
# taking the exisiting scale factor parameters into account, calculate the parameter values reequired to achieve a unity slope
2022-07-18 11:09:04 -03:00
flow_fxscaler_new = int ( 1000 * ( ( ( 1 + 0.001 * float ( flow_fxscaler ) ) / coef_flow_x [ 0 ] - 1 ) ) )
flow_fyscaler_new = int ( 1000 * ( ( ( 1 + 0.001 * float ( flow_fyscaler ) ) / coef_flow_y [ 0 ] - 1 ) ) )
2017-02-07 02:47:57 -04:00
# Do a sanity check on the scale factor variance
if sqrt ( cov_x [ 0 ] [ 0 ] ) > param_std_threshold or sqrt ( cov_y [ 0 ] [ 0 ] ) > param_std_threshold :
FAIL ( )
2022-07-18 11:09:04 -03:00
self . result . statusMessage = (
" FAIL: inaccurate fit - poor quality or insufficient data \n FLOW_FXSCALER 1STD = %u \n FLOW_FYSCALER 1STD = %u \n "
% ( round ( 1000 * sqrt ( cov_x [ 0 ] [ 0 ] ) ) , round ( 1000 * sqrt ( cov_y [ 0 ] [ 0 ] ) ) )
)
2017-02-07 02:47:57 -04:00
# Do a sanity check on the scale factors
if abs ( flow_fxscaler_new ) > param_abs_threshold or abs ( flow_fyscaler_new ) > param_abs_threshold :
FAIL ( )
2022-07-18 11:09:04 -03:00
self . result . statusMessage = (
" FAIL: required scale factors are excessive \n FLOW_FXSCALER= %i \n FLOW_FYSCALER= %i \n "
% ( flow_fxscaler , flow_fyscaler )
)
2017-02-07 02:47:57 -04:00
# display recommended scale factors
2022-07-18 11:09:04 -03:00
self . result . statusMessage = (
" Set FLOW_FXSCALER to %i \n Set FLOW_FYSCALER to %i \n \n Cal plots saved to flow_calibration.pdf \n Cal parameters saved to flow_calibration.param \n \n FLOW_FXSCALER 1STD = %u \n FLOW_FYSCALER 1STD = %u \n "
% (
flow_fxscaler_new ,
flow_fyscaler_new ,
round ( 1000 * sqrt ( cov_x [ 0 ] [ 0 ] ) ) ,
round ( 1000 * sqrt ( cov_y [ 0 ] [ 0 ] ) ) ,
)
)
2017-02-07 02:47:57 -04:00
# calculate fit display data
2022-07-18 11:09:04 -03:00
body_rate_display = [ - max_rate_threshold , max_rate_threshold ]
2017-02-07 02:47:57 -04:00
fit_coef_x = np . poly1d ( coef_flow_x )
flowX_display = fit_coef_x ( body_rate_display )
fit_coef_y = np . poly1d ( coef_flow_y )
flowY_display = fit_coef_y ( body_rate_display )
# plot and save calibration test points to PDF
from matplotlib . backends . backend_pdf import PdfPages
2022-07-18 11:09:04 -03:00
2017-02-07 02:47:57 -04:00
output_plot_filename = " flow_calibration.pdf "
pp = PdfPages ( output_plot_filename )
2022-07-18 11:09:04 -03:00
plt . figure ( 1 , figsize = ( 20 , 13 ) )
plt . subplot ( 2 , 1 , 1 )
plt . plot ( bodyX_resampled , flowX_resampled , ' b ' , linestyle = ' ' , marker = ' o ' , label = " test points " )
plt . plot ( body_rate_display , flowX_display , ' r ' , linewidth = 2.5 , label = " linear fit " )
2017-02-07 02:47:57 -04:00
plt . title ( ' X axis flow rate vs gyro rate ' )
plt . ylabel ( ' flow rate (rad/s) ' )
plt . xlabel ( ' gyro rate (rad/sec) ' )
plt . grid ( )
plt . legend ( loc = ' upper left ' )
# draw plots
2022-07-18 11:09:04 -03:00
plt . subplot ( 2 , 1 , 2 )
plt . plot ( bodyY_resampled , flowY_resampled , ' b ' , linestyle = ' ' , marker = ' o ' , label = " test points " )
plt . plot ( body_rate_display , flowY_display , ' r ' , linewidth = 2.5 , label = " linear fit " )
2017-02-07 02:47:57 -04:00
plt . title ( ' Y axis flow rate vs gyro rate ' )
plt . ylabel ( ' flow rate (rad/s) ' )
plt . xlabel ( ' gyro rate (rad/sec) ' )
plt . grid ( )
plt . legend ( loc = ' upper left ' )
pp . savefig ( )
2022-07-18 11:09:04 -03:00
plt . figure ( 2 , figsize = ( 20 , 13 ) )
plt . subplot ( 2 , 1 , 1 )
plt . plot ( flow_time_us , flowX , ' b ' , label = " flow rate - all " )
plt . plot ( flow_time_us , bodyX , ' r ' , label = " gyro rate - all " )
plt . plot ( flowX_time_us_resampled , flowX_resampled , ' c ' , linestyle = ' ' , marker = ' o ' , label = " flow rate - used " )
plt . plot ( flowX_time_us_resampled , bodyX_resampled , ' m ' , linestyle = ' ' , marker = ' o ' , label = " gyro rate - used " )
2017-02-07 02:47:57 -04:00
plt . title ( ' X axis flow and body rate vs time ' )
plt . ylabel ( ' rate (rad/s) ' )
plt . xlabel ( ' time (usec) ' )
plt . grid ( )
plt . legend ( loc = ' upper left ' )
# draw plots
2022-07-18 11:09:04 -03:00
plt . subplot ( 2 , 1 , 2 )
plt . plot ( flow_time_us , flowY , ' b ' , label = " flow rate - all " )
plt . plot ( flow_time_us , bodyY , ' r ' , label = " gyro rate - all " )
plt . plot ( flowY_time_us_resampled , flowY_resampled , ' c ' , linestyle = ' ' , marker = ' o ' , label = " flow rate - used " )
plt . plot ( flowY_time_us_resampled , bodyY_resampled , ' m ' , linestyle = ' ' , marker = ' o ' , label = " gyro rate - used " )
2017-02-07 02:47:57 -04:00
plt . title ( ' Y axis flow and body rate vs time ' )
plt . ylabel ( ' rate (rad/s) ' )
plt . xlabel ( ' time (usec) ' )
plt . grid ( )
plt . legend ( loc = ' upper left ' )
pp . savefig ( )
# close the pdf file
pp . close ( )
# close all figures
plt . close ( " all " )
# write correction parameters to file
test_results_filename = " flow_calibration.param "
2022-07-18 11:09:04 -03:00
file = open ( test_results_filename , " w " )
file . write ( " FLOW_FXSCALER " + " " + str ( flow_fxscaler_new ) + " \n " )
file . write ( " FLOW_FYSCALER " + " " + str ( flow_fyscaler_new ) + " \n " )
2017-02-07 02:47:57 -04:00
file . close ( )
except KeyError as e :
self . result . status = TestResult . StatusType . FAIL
self . result . statusMessage = str ( e ) + ' not found '