2013-05-29 20:51:34 -03:00
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
2013-08-29 02:34:34 -03:00
/*
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 , see < http : //www.gnu.org/licenses/>.
*/
2012-04-02 05:26:37 -03:00
/*
2012-08-17 03:20:26 -03:00
* AP_Motors . cpp - ArduCopter motors library
* Code by RandyMackay . DIYDrones . com
*
*/
2012-04-02 05:26:37 -03:00
2012-10-26 20:59:07 -03:00
# include "AP_Motors_Class.h"
# include <AP_HAL.h>
extern const AP_HAL : : HAL & hal ;
2012-04-02 05:26:37 -03:00
2014-02-10 02:58:24 -04:00
// initialise motor map
# if CONFIG_HAL_BOARD == HAL_BOARD_APM1
const uint8_t AP_Motors : : _motor_to_channel_map [ AP_MOTORS_MAX_NUM_MOTORS ] PROGMEM = { APM1_MOTOR_TO_CHANNEL_MAP } ;
# else
const uint8_t AP_Motors : : _motor_to_channel_map [ AP_MOTORS_MAX_NUM_MOTORS ] PROGMEM = { APM2_MOTOR_TO_CHANNEL_MAP } ;
# endif
2012-04-02 05:26:37 -03:00
// parameters for the motor class
const AP_Param : : GroupInfo AP_Motors : : var_info [ ] PROGMEM = {
2013-05-14 04:56:55 -03:00
// 0 was used by TB_RATIO
2015-02-21 04:33:13 -04:00
// 1,2,3 were used by throttle curve
2012-12-10 08:37:20 -04:00
2013-07-16 00:45:40 -03:00
// @Param: SPIN_ARMED
// @DisplayName: Motors always spin when armed
2013-09-18 01:28:01 -03:00
// @Description: Controls whether motors always spin when armed (must be below THR_MIN)
// @Values: 0:Do Not Spin,70:VerySlow,100:Slow,130:Medium,150:Fast
2013-11-12 23:28:59 -04:00
// @User: Standard
2013-09-18 01:28:01 -03:00
AP_GROUPINFO ( " SPIN_ARMED " , 5 , AP_Motors , _spin_when_armed , AP_MOTORS_SPIN_WHEN_ARMED ) ,
2012-12-10 08:37:20 -04:00
2015-02-11 08:03:39 -04:00
// @Param: YAW_HEADROOM
// @DisplayName: Matrix Yaw Min
// @Description: Yaw control is given at least this pwm range
// @Range: 0 500
// @Units: pwm
// @User: Advanced
AP_GROUPINFO ( " YAW_HEADROOM " , 6 , AP_Motors , _yaw_headroom , AP_MOTORS_YAW_HEADROOM_DEFAULT ) ,
2015-03-09 10:23:16 -03:00
// 7 was THR_LOW_CMP
2015-02-21 02:06:56 -04:00
2015-02-23 02:35:10 -04:00
// @Param: THST_EXPO
// @DisplayName: Thrust Curve Expo
// @Description: Motor thrust curve exponent (from 0 for linear to 1.0 for second order curve)
// @Range: 0.25 0.8
// @User: Advanced
AP_GROUPINFO ( " THST_EXPO " , 8 , AP_Motors , _thrust_curve_expo , AP_MOTORS_THST_EXPO_DEFAULT ) ,
// @Param: THST_MAX
// @DisplayName: Thrust Curve Max
// @Description: Point at which the thrust saturates
// @Values: 0.9:Low, 1.0:High
// @User: Advanced
AP_GROUPINFO ( " THST_MAX " , 9 , AP_Motors , _thrust_curve_max , AP_MOTORS_THST_MAX_DEFAULT ) ,
// @Param: THST_BAT_MAX
// @DisplayName: Battery voltage compensation maximum voltage
// @Description: Battery voltage compensation maximum voltage (voltage above this will have no additional scaling effect on thrust). Recommend 4.4 * cell count, 0 = Disabled
// @Range: 6 35
// @Units: Volts
// @User: Advanced
AP_GROUPINFO ( " THST_BAT_MAX " , 10 , AP_Motors , _batt_voltage_max , AP_MOTORS_THST_BAT_MAX_DEFAULT ) ,
// @Param: THST_BAT_MIN
// @DisplayName: Battery voltage compensation minimum voltage
// @Description: Battery voltage compensation minimum voltage (voltage below this will have no additional scaling effect on thrust). Recommend 3.5 * cell count, 0 = Disabled
// @Range: 6 35
// @Units: Volts
// @User: Advanced
AP_GROUPINFO ( " THST_BAT_MIN " , 11 , AP_Motors , _batt_voltage_min , AP_MOTORS_THST_BAT_MIN_DEFAULT ) ,
2015-02-23 02:36:23 -04:00
// @Param: CURR_MAX
// @DisplayName: Motor Current Max
// @Description: Maximum current over which maximum throttle is limited (0 = Disabled)
// @Range: 0 200
// @Units: Amps
// @User: Advanced
AP_GROUPINFO ( " CURR_MAX " , 12 , AP_Motors , _batt_current_max , AP_MOTORS_CURR_MAX_DEFAULT ) ,
2015-05-03 11:21:05 -03:00
// @Param: THR_MIX_MIN
// @DisplayName: Throttle Mix Minimum
// @Description: Minimum ratio that the average throttle can increase above the desired throttle after roll, pitch and yaw are mixed
// @Range: 0.1 0.25
// @User: Advanced
AP_GROUPINFO ( " THR_MIX_MIN " , 13 , AP_Motors , _thr_mix_min , AP_MOTORS_THR_MIX_MIN_DEFAULT ) ,
2012-04-02 05:26:37 -03:00
AP_GROUPEND
} ;
2012-08-17 03:20:26 -03:00
// Constructor
2015-05-14 22:00:46 -03:00
AP_Motors : : AP_Motors ( uint16_t loop_rate , uint16_t speed_hz ) :
2015-05-25 08:17:35 -03:00
_roll_control_input ( 0.0f ) ,
_pitch_control_input ( 0.0f ) ,
_throttle_control_input ( 0.0f ) ,
_yaw_control_input ( 0.0f ) ,
_throttle_pwm_scalar ( 1.0f ) ,
2015-02-21 02:45:56 -04:00
_loop_rate ( loop_rate ) ,
2012-08-17 03:20:26 -03:00
_speed_hz ( speed_hz ) ,
_min_throttle ( AP_MOTORS_DEFAULT_MIN_THROTTLE ) ,
2013-05-26 23:21:31 -03:00
_max_throttle ( AP_MOTORS_DEFAULT_MAX_THROTTLE ) ,
2015-05-25 08:17:35 -03:00
_throttle_radio_min ( 1100 ) ,
_throttle_radio_max ( 1900 ) ,
2013-10-20 02:51:35 -03:00
_hover_out ( AP_MOTORS_DEFAULT_MID_THROTTLE ) ,
2015-02-23 02:35:10 -04:00
_spin_when_armed_ramped ( 0 ) ,
2015-05-03 11:21:05 -03:00
_throttle_thr_mix ( AP_MOTORS_THR_LOW_CMP_DEFAULT ) ,
_throttle_thr_mix_desired ( AP_MOTORS_THR_LOW_CMP_DEFAULT ) ,
2015-02-23 02:35:10 -04:00
_batt_voltage ( 0.0f ) ,
_batt_voltage_resting ( 0.0f ) ,
_batt_current ( 0.0f ) ,
_batt_current_resting ( 0.0f ) ,
_batt_resistance ( 0.0f ) ,
_batt_timer ( 0 ) ,
2015-04-28 15:19:50 -03:00
_air_density_ratio ( 1.0f ) ,
2015-02-23 02:36:23 -04:00
_lift_max ( 1.0f ) ,
2015-04-01 00:16:43 -03:00
_throttle_limit ( 1.0f ) ,
_throttle_in ( 0.0f ) ,
_throttle_filter ( )
2012-04-02 05:26:37 -03:00
{
2012-12-12 17:52:23 -04:00
AP_Param : : setup_object_defaults ( this , var_info ) ;
2013-10-20 02:51:35 -03:00
// slow start motors from zero to min throttle
2015-06-08 00:18:40 -03:00
_flags . slow_start = true ;
2013-10-20 02:51:35 -03:00
_flags . slow_start_low_end = true ;
2015-02-23 00:51:54 -04:00
2015-06-08 00:18:40 -03:00
// init other flags
_flags . armed = false ;
_flags . stabilizing = false ;
_flags . frame_orientation = AP_MOTORS_X_FRAME ;
_flags . interlock = false ;
2015-02-23 00:51:54 -04:00
// setup battery voltage filtering
2015-04-16 16:36:02 -03:00
_batt_voltage_filt . set_cutoff_frequency ( AP_MOTORS_BATT_VOLT_FILT_HZ ) ;
2015-02-23 00:51:54 -04:00
_batt_voltage_filt . reset ( 1.0f ) ;
2015-04-01 00:16:43 -03:00
2015-04-16 01:55:12 -03:00
// setup throttle filtering
2015-04-16 16:36:02 -03:00
_throttle_filter . set_cutoff_frequency ( 0.0f ) ;
2015-04-01 00:16:43 -03:00
_throttle_filter . reset ( 0.0f ) ;
2015-06-08 00:18:40 -03:00
// init limit flags
limit . roll_pitch = true ;
limit . yaw = true ;
limit . throttle_lower = true ;
limit . throttle_upper = true ;
2012-04-02 05:26:37 -03:00
} ;
2013-08-08 10:15:04 -03:00
void AP_Motors : : armed ( bool arm )
{
2013-09-12 10:27:44 -03:00
_flags . armed = arm ;
2013-10-20 02:51:35 -03:00
if ( ! _flags . armed ) {
_flags . slow_start_low_end = true ;
}
2013-09-12 10:27:44 -03:00
AP_Notify : : flags . armed = arm ;
2013-08-08 10:15:04 -03:00
} ;
2015-05-14 22:00:46 -03:00
// set_throttle_range - sets the minimum throttle that will be sent to the engines when they're not off (i.e. to prevents issues with some motors spinning and some not at very low throttle)
// also sets throttle channel minimum and maximum pwm
void AP_Motors : : set_throttle_range ( uint16_t min_throttle , int16_t radio_min , int16_t radio_max )
2013-05-26 23:21:31 -03:00
{
2015-05-14 22:00:46 -03:00
_throttle_radio_min = radio_min ;
_throttle_radio_max = radio_max ;
_throttle_pwm_scalar = ( _throttle_radio_max - _throttle_radio_min ) / 1000.0f ;
_min_throttle = ( float ) min_throttle * _throttle_pwm_scalar ;
2013-05-26 23:21:31 -03:00
}
2015-02-22 22:27:51 -04:00
// get_hover_throttle_as_pwm - converts hover throttle to pwm (i.e. range 1000 ~ 2000)
int16_t AP_Motors : : get_hover_throttle_as_pwm ( ) const
2013-05-26 23:21:31 -03:00
{
2015-05-14 22:00:46 -03:00
return ( _throttle_radio_min + ( float ) ( _throttle_radio_max - _throttle_radio_min ) * _hover_out / 1000.0f ) ;
2013-05-26 23:21:31 -03:00
}
2014-09-19 10:06:29 -03:00
// throttle_pass_through - passes provided pwm directly to all motors - dangerous but used for initialising ESCs
// pwm value is an actual pwm value that will be output, normally in the range of 1000 ~ 2000
void AP_Motors : : throttle_pass_through ( int16_t pwm )
2012-09-18 11:05:08 -03:00
{
2013-05-14 05:07:36 -03:00
if ( armed ( ) ) {
// send the pilot's input directly to each enabled motor
for ( int16_t i = 0 ; i < AP_MOTORS_MAX_NUM_MOTORS ; i + + ) {
if ( motor_enabled [ i ] ) {
2014-09-19 10:06:29 -03:00
hal . rcout - > write ( pgm_read_byte ( & _motor_to_channel_map [ i ] ) , pwm ) ;
2013-05-14 05:07:36 -03:00
}
2012-08-17 03:20:26 -03:00
}
}
2012-04-02 05:26:37 -03:00
}
2012-09-18 11:05:08 -03:00
2013-09-12 10:27:44 -03:00
// output - sends commands to the motors
void AP_Motors : : output ( )
{
2015-04-01 00:16:43 -03:00
// update throttle filter
update_throttle_filter ( ) ;
2013-09-12 10:27:44 -03:00
// update max throttle
update_max_throttle ( ) ;
2015-02-23 00:52:39 -04:00
// update battery resistance
update_battery_resistance ( ) ;
2015-02-21 05:15:11 -04:00
// calc filtered battery voltage and lift_max
update_lift_max_from_batt_voltage ( ) ;
2015-03-08 07:17:28 -03:00
// move throttle_low_comp towards desired throttle low comp
2015-05-03 11:21:05 -03:00
update_throttle_thr_mix ( ) ;
2015-03-08 07:17:28 -03:00
2015-04-02 17:54:15 -03:00
if ( _flags . armed ) {
2015-04-17 12:58:26 -03:00
if ( ! _flags . interlock ) {
output_armed_zero_throttle ( ) ;
} else if ( _flags . stabilizing ) {
2015-04-02 17:54:15 -03:00
output_armed_stabilizing ( ) ;
} else {
output_armed_not_stabilizing ( ) ;
}
} else {
2013-09-12 10:27:44 -03:00
output_disarmed ( ) ;
}
} ;
// slow_start - set to true to slew motors from current speed to maximum
// Note: this must be set immediately before a step up in throttle
void AP_Motors : : slow_start ( bool true_false )
{
// set slow_start flag
_flags . slow_start = true ;
// initialise maximum throttle to current throttle
2015-05-14 22:00:46 -03:00
_max_throttle = constrain_int16 ( _throttle_control_input , 0 , AP_MOTORS_DEFAULT_MAX_THROTTLE ) ;
2013-09-12 10:27:44 -03:00
}
2015-04-01 00:16:43 -03:00
// update the throttle input filter
void AP_Motors : : update_throttle_filter ( )
{
2015-04-02 17:54:15 -03:00
if ( armed ( ) ) {
2015-04-24 01:35:33 -03:00
_throttle_filter . apply ( constrain_float ( _throttle_in , - 100 , 1100 ) , 1.0f / _loop_rate ) ;
2015-04-01 00:16:43 -03:00
} else {
_throttle_filter . reset ( 0.0f ) ;
}
2015-05-14 22:00:46 -03:00
// prevent _throttle_control_input from wrapping at int16 max or min
_throttle_control_input = constrain_float ( _throttle_filter . get ( ) , - 32000 , 32000 ) ;
2015-04-01 00:16:43 -03:00
}
2013-09-12 10:27:44 -03:00
// update_max_throttle - updates the limits on _max_throttle if necessary taking into account slow_start_throttle flag
void AP_Motors : : update_max_throttle ( )
{
2013-10-20 02:51:35 -03:00
// ramp up minimum spin speed if necessary
if ( _flags . slow_start_low_end ) {
_spin_when_armed_ramped + = AP_MOTOR_SLOW_START_LOW_END_INCREMENT ;
if ( _spin_when_armed_ramped > = _spin_when_armed ) {
_spin_when_armed_ramped = _spin_when_armed ;
_flags . slow_start_low_end = false ;
}
}
2015-02-23 05:09:08 -04:00
// implement slow start
if ( _flags . slow_start ) {
2013-09-12 10:27:44 -03:00
// increase slow start throttle
_max_throttle + = AP_MOTOR_SLOW_START_INCREMENT ;
// turn off slow start if we've reached max throttle
2015-05-14 22:00:46 -03:00
if ( _max_throttle > = _throttle_control_input ) {
2013-09-12 10:27:44 -03:00
_max_throttle = AP_MOTORS_DEFAULT_MAX_THROTTLE ;
_flags . slow_start = false ;
}
2015-02-23 05:09:08 -04:00
return ;
}
// current limit throttle
current_limit_max_throttle ( ) ;
}
// current_limit_max_throttle - limits maximum throttle based on current
void AP_Motors : : current_limit_max_throttle ( )
{
// return maximum if current limiting is disabled
if ( _batt_current_max < = 0 ) {
_throttle_limit = 1.0f ;
_max_throttle = AP_MOTORS_DEFAULT_MAX_THROTTLE ;
return ;
}
// remove throttle limit if throttle is at zero or disarmed
2015-05-14 22:00:46 -03:00
if ( _throttle_control_input < = 0 | | ! _flags . armed ) {
2015-02-23 05:09:08 -04:00
_throttle_limit = 1.0f ;
}
// limit throttle if over current
if ( _batt_current > _batt_current_max * 1.25f ) {
// Fast drop for extreme over current (1 second)
_throttle_limit - = 1.0f / _loop_rate ;
} else if ( _batt_current > _batt_current_max ) {
2015-04-14 09:55:15 -03:00
// Slow drop for extreme over current (5 second)
_throttle_limit - = 0.2f / _loop_rate ;
2015-02-23 05:09:08 -04:00
} else {
// Increase throttle limit (2 second)
_throttle_limit + = 0.5f / _loop_rate ;
}
// throttle limit drops to 20% between hover and full throttle
_throttle_limit = constrain_float ( _throttle_limit , 0.2f , 1.0f ) ;
// limit max throttle
_max_throttle = _hover_out + ( ( 1000 - _hover_out ) * _throttle_limit ) ;
2014-02-09 04:47:14 -04:00
}
2015-02-21 03:55:13 -04:00
// apply_thrust_curve_and_volt_scaling - returns throttle curve adjusted pwm value (i.e. 1000 ~ 2000)
int16_t AP_Motors : : apply_thrust_curve_and_volt_scaling ( int16_t pwm_out , int16_t pwm_min , int16_t pwm_max ) const
{
2015-04-28 15:17:28 -03:00
// convert to 0.0 to 1.0 ratio
float throttle_ratio = ( ( float ) ( pwm_out - pwm_min ) ) / ( ( float ) ( pwm_max - pwm_min ) ) ;
// apply thrust curve - domain 0.0 to 1.0, range 0.0 to 1.0
2015-02-21 03:55:13 -04:00
if ( _thrust_curve_expo > 0.0f ) {
2015-04-28 15:17:28 -03:00
throttle_ratio = ( ( _thrust_curve_expo - 1.0f ) + safe_sqrt ( ( 1.0f - _thrust_curve_expo ) * ( 1.0f - _thrust_curve_expo ) + 4.0f * _thrust_curve_expo * _lift_max * throttle_ratio ) ) / ( 2.0f * _thrust_curve_expo * _batt_voltage_filt . get ( ) ) ;
2015-02-21 03:55:13 -04:00
}
2015-04-28 15:17:28 -03:00
// scale to maximum thrust point
throttle_ratio * = _thrust_curve_max ;
// convert back to pwm range, constrain and return
return ( int16_t ) constrain_float ( throttle_ratio * ( pwm_max - pwm_min ) + pwm_min , pwm_min , ( pwm_max - pwm_min ) * _thrust_curve_max + pwm_min ) ;
2015-02-21 03:55:13 -04:00
}
2015-02-21 05:15:11 -04:00
// update_lift_max from battery voltage - used for voltage compensation
void AP_Motors : : update_lift_max_from_batt_voltage ( )
{
// sanity check battery_voltage_min is not too small
// if disabled or misconfigured exit immediately
2015-04-24 00:51:54 -03:00
if ( ( _batt_voltage_max < = 0 ) | | ( _batt_voltage_min > = _batt_voltage_max ) | | ( _batt_voltage < 0.25f * _batt_voltage_min ) ) {
2015-02-23 00:51:54 -04:00
_batt_voltage_filt . reset ( 1.0f ) ;
2015-02-21 05:15:11 -04:00
_lift_max = 1.0f ;
return ;
}
2015-03-09 03:23:33 -03:00
_batt_voltage_min = max ( _batt_voltage_min , _batt_voltage_max * 0.6f ) ;
2015-02-21 05:15:11 -04:00
// add current based voltage sag to battery voltage
float batt_voltage = _batt_voltage + _batt_current * _batt_resistance ;
batt_voltage = constrain_float ( batt_voltage , _batt_voltage_min , _batt_voltage_max ) ;
// filter at 0.5 Hz
2015-04-16 16:36:02 -03:00
float bvf = _batt_voltage_filt . apply ( batt_voltage / _batt_voltage_max , 1.0f / _loop_rate ) ;
2015-02-23 00:51:54 -04:00
// calculate lift max
_lift_max = bvf * ( 1 - _thrust_curve_expo ) + _thrust_curve_expo * bvf * bvf ;
2015-02-21 05:15:11 -04:00
}
2015-02-23 00:52:39 -04:00
// update_battery_resistance - calculate battery resistance when throttle is above hover_out
void AP_Motors : : update_battery_resistance ( )
{
// if motors are stopped, reset resting voltage and current
2015-05-14 22:00:46 -03:00
if ( _throttle_control_input < = 0 | | ! _flags . armed ) {
2015-02-23 00:52:39 -04:00
_batt_voltage_resting = _batt_voltage ;
_batt_current_resting = _batt_current ;
_batt_timer = 0 ;
} else {
// update battery resistance when throttle is over hover throttle
if ( ( _batt_timer < 400 ) & & ( ( _batt_current_resting * 2.0f ) < _batt_current ) ) {
2015-05-14 22:00:46 -03:00
if ( _throttle_control_input > = _hover_out ) {
2015-02-23 00:52:39 -04:00
// filter reaches 90% in 1/4 the test time
_batt_resistance + = 0.05f * ( ( ( _batt_voltage_resting - _batt_voltage ) / ( _batt_current - _batt_current_resting ) ) - _batt_resistance ) ;
_batt_timer + = 1 ;
} else {
// initialize battery resistance to prevent change in resting voltage estimate
_batt_resistance = ( ( _batt_voltage_resting - _batt_voltage ) / ( _batt_current - _batt_current_resting ) ) ;
}
}
}
}
2015-03-08 07:17:28 -03:00
2015-05-03 11:21:05 -03:00
// update_throttle_thr_mix - slew set_throttle_thr_mix to requested value
void AP_Motors : : update_throttle_thr_mix ( )
2015-03-08 07:17:28 -03:00
{
2015-05-03 11:21:05 -03:00
// slew _throttle_thr_mix to _throttle_thr_mix_desired
if ( _throttle_thr_mix < _throttle_thr_mix_desired ) {
2015-04-14 09:55:15 -03:00
// increase quickly (i.e. from 0.1 to 0.9 in 0.4 seconds)
2015-05-03 11:21:05 -03:00
_throttle_thr_mix + = min ( 2.0f / _loop_rate , _throttle_thr_mix_desired - _throttle_thr_mix ) ;
} else if ( _throttle_thr_mix > _throttle_thr_mix_desired ) {
2015-04-14 09:55:15 -03:00
// reduce more slowly (from 0.9 to 0.1 in 1.6 seconds)
2015-05-03 11:21:05 -03:00
_throttle_thr_mix - = min ( 0.5f / _loop_rate , _throttle_thr_mix - _throttle_thr_mix_desired ) ;
2015-03-08 07:17:28 -03:00
}
2015-05-03 11:21:05 -03:00
_throttle_thr_mix = constrain_float ( _throttle_thr_mix , 0.1f , 1.0f ) ;
2015-03-08 07:17:28 -03:00
}
2015-04-02 17:54:15 -03:00
2015-04-28 15:19:50 -03:00
float AP_Motors : : get_compensation_gain ( ) const
{
2015-04-29 02:13:07 -03:00
// avoid divide by zero
if ( _lift_max < = 0.0f ) {
return 1.0f ;
}
2015-04-28 15:19:50 -03:00
float ret = 1.0f / _lift_max ;
// air density ratio is increasing in density / decreasing in altitude
if ( _air_density_ratio > 0.3f & & _air_density_ratio < 1.5f ) {
ret * = 1.0f / constrain_float ( _air_density_ratio , 0.5f , 1.25f ) ;
}
return ret ;
}
2015-04-02 17:54:15 -03:00
float AP_Motors : : rel_pwm_to_thr_range ( float pwm ) const
{
2015-05-14 22:00:46 -03:00
return pwm / _throttle_pwm_scalar ;
2015-04-02 17:54:15 -03:00
}
float AP_Motors : : thr_range_to_rel_pwm ( float thr ) const
{
2015-05-14 22:00:46 -03:00
return _throttle_pwm_scalar * thr ;
2015-04-02 17:54:15 -03:00
}