uncrustify libraries/AP_Motors/AP_MotorsHeli.cpp

This commit is contained in:
uncrustify 2012-08-21 19:19:52 -07:00 committed by Pat Hickey
parent 4189870bfb
commit 24943e0ea7
1 changed files with 469 additions and 469 deletions

View File

@ -1,164 +1,164 @@
/* /*
AP_MotorsHeli.cpp - ArduCopter motors library * AP_MotorsHeli.cpp - ArduCopter motors library
Code by RandyMackay. DIYDrones.com * Code by RandyMackay. DIYDrones.com
*
This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public * modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either * License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version. * version 2.1 of the License, or (at your option) any later version.
*/ */
#include "AP_MotorsHeli.h" #include "AP_MotorsHeli.h"
const AP_Param::GroupInfo AP_MotorsHeli::var_info[] PROGMEM = { const AP_Param::GroupInfo AP_MotorsHeli::var_info[] PROGMEM = {
// @Param: SV1_POS // @Param: SV1_POS
// @DisplayName: Servo 1 Position // @DisplayName: Servo 1 Position
// @Description: This is the angular location of swash servo #1. // @Description: This is the angular location of swash servo #1.
// @Range: -180 180 // @Range: -180 180
// @Units: Degrees // @Units: Degrees
// @User: Standard // @User: Standard
// @Increment: 1 // @Increment: 1
AP_GROUPINFO("SV1_POS", 1, AP_MotorsHeli, servo1_pos, -60), AP_GROUPINFO("SV1_POS", 1, AP_MotorsHeli, servo1_pos, -60),
// @Param: SV2_POS // @Param: SV2_POS
// @DisplayName: Servo 2 Position // @DisplayName: Servo 2 Position
// @Description: This is the angular location of swash servo #2. // @Description: This is the angular location of swash servo #2.
// @Range: -180 180 // @Range: -180 180
// @Units: Degrees // @Units: Degrees
// @User: Standard // @User: Standard
// @Increment: 1 // @Increment: 1
AP_GROUPINFO("SV2_POS", 2, AP_MotorsHeli, servo2_pos, 60), AP_GROUPINFO("SV2_POS", 2, AP_MotorsHeli, servo2_pos, 60),
// @Param: SV3_POS // @Param: SV3_POS
// @DisplayName: Servo 3 Position // @DisplayName: Servo 3 Position
// @Description: This is the angular location of swash servo #3. // @Description: This is the angular location of swash servo #3.
// @Range: -180 180 // @Range: -180 180
// @Units: Degrees // @Units: Degrees
// @User: Standard // @User: Standard
// @Increment: 1 // @Increment: 1
AP_GROUPINFO("SV3_POS", 3, AP_MotorsHeli, servo3_pos, 180), AP_GROUPINFO("SV3_POS", 3, AP_MotorsHeli, servo3_pos, 180),
// @Param: ROL_MAX // @Param: ROL_MAX
// @DisplayName: Maximum Roll Angle // @DisplayName: Maximum Roll Angle
// @Description: This is the maximum allowable roll of the swash plate. // @Description: This is the maximum allowable roll of the swash plate.
// @Range: 0 18000 // @Range: 0 18000
// @Units: Degrees // @Units: Degrees
// @Increment: 1 // @Increment: 1
// @User: Advanced // @User: Advanced
AP_GROUPINFO("ROL_MAX", 4, AP_MotorsHeli, roll_max, 4500), AP_GROUPINFO("ROL_MAX", 4, AP_MotorsHeli, roll_max, 4500),
// @Param: PIT_MAX // @Param: PIT_MAX
// @DisplayName: Maximum Pitch Angle // @DisplayName: Maximum Pitch Angle
// @Description: This is the maximum allowable pitch of the swash plate. // @Description: This is the maximum allowable pitch of the swash plate.
// @Range: 0 18000 // @Range: 0 18000
// @Units: Degrees // @Units: Degrees
// @Increment: 1 // @Increment: 1
// @User: Advanced // @User: Advanced
AP_GROUPINFO("PIT_MAX", 5, AP_MotorsHeli, pitch_max, 4500), AP_GROUPINFO("PIT_MAX", 5, AP_MotorsHeli, pitch_max, 4500),
// @Param: COL_MIN // @Param: COL_MIN
// @DisplayName: Collective Pitch Minimum // @DisplayName: Collective Pitch Minimum
// @Description: This controls the lowest possible servo position for the swashplate. // @Description: This controls the lowest possible servo position for the swashplate.
// @Range: 1000 2000 // @Range: 1000 2000
// @Units: PWM // @Units: PWM
// @Increment: 1 // @Increment: 1
// @User: Standard // @User: Standard
AP_GROUPINFO("COL_MIN", 6, AP_MotorsHeli, collective_min, 1250), AP_GROUPINFO("COL_MIN", 6, AP_MotorsHeli, collective_min, 1250),
// @Param: COL_MAX // @Param: COL_MAX
// @DisplayName: Collective Pitch Maximum // @DisplayName: Collective Pitch Maximum
// @Description: This controls the highest possible servo position for the swashplate. // @Description: This controls the highest possible servo position for the swashplate.
// @Range: 1000 2000 // @Range: 1000 2000
// @Units: PWM // @Units: PWM
// @Increment: 1 // @Increment: 1
// @User: Standard // @User: Standard
AP_GROUPINFO("COL_MAX", 7, AP_MotorsHeli, collective_max, 1750), AP_GROUPINFO("COL_MAX", 7, AP_MotorsHeli, collective_max, 1750),
// @Param: COL_MID // @Param: COL_MID
// @DisplayName: Collective Pitch Mid-Point // @DisplayName: Collective Pitch Mid-Point
// @Description: This is the swash servo position corresponding to zero collective pitch (or zero lift for Assymetrical blades). // @Description: This is the swash servo position corresponding to zero collective pitch (or zero lift for Assymetrical blades).
// @Range: 1000 2000 // @Range: 1000 2000
// @Units: PWM // @Units: PWM
// @Increment: 1 // @Increment: 1
// @User: Standard // @User: Standard
AP_GROUPINFO("COL_MID", 8, AP_MotorsHeli, collective_mid, 1500), AP_GROUPINFO("COL_MID", 8, AP_MotorsHeli, collective_mid, 1500),
// @Param: GYR_ENABLE // @Param: GYR_ENABLE
// @DisplayName: External Gyro Enabled // @DisplayName: External Gyro Enabled
// @Description: Setting this to Enabled(1) will enable an external rudder gyro control which means outputting a gain on channel 7 and using a simpler heading control algorithm. Setting this to Disabled(0) will disable the external gyro gain on channel 7 and revert to a more complex yaw control algorithm. // @Description: Setting this to Enabled(1) will enable an external rudder gyro control which means outputting a gain on channel 7 and using a simpler heading control algorithm. Setting this to Disabled(0) will disable the external gyro gain on channel 7 and revert to a more complex yaw control algorithm.
// @Values: 0:Disabled,1:Enabled // @Values: 0:Disabled,1:Enabled
// @User: Standard // @User: Standard
AP_GROUPINFO("GYR_ENABLE", 9, AP_MotorsHeli, ext_gyro_enabled, 0), AP_GROUPINFO("GYR_ENABLE", 9, AP_MotorsHeli, ext_gyro_enabled, 0),
// @Param: SWASH_TYPE // @Param: SWASH_TYPE
// @DisplayName: Swash Plate Type // @DisplayName: Swash Plate Type
// @Description: Setting this to 0 will configure for a 3-servo CCPM. Setting this to 1 will configure for mechanically mixed "H1". // @Description: Setting this to 0 will configure for a 3-servo CCPM. Setting this to 1 will configure for mechanically mixed "H1".
// @User: Standard // @User: Standard
AP_GROUPINFO("SWASH_TYPE", 10, AP_MotorsHeli, swash_type, AP_MOTORS_HELI_SWASH_CCPM), AP_GROUPINFO("SWASH_TYPE", 10, AP_MotorsHeli, swash_type, AP_MOTORS_HELI_SWASH_CCPM),
// @Param: GYR_GAIN // @Param: GYR_GAIN
// @DisplayName: External Gyro Gain // @DisplayName: External Gyro Gain
// @Description: This is the PWM which is passed to the external gyro when external gyro is enabled. // @Description: This is the PWM which is passed to the external gyro when external gyro is enabled.
// @Range: 1000 2000 // @Range: 1000 2000
// @Units: PWM // @Units: PWM
// @Increment: 1 // @Increment: 1
// @User: Standard // @User: Standard
AP_GROUPINFO("GYR_GAIN", 11, AP_MotorsHeli, ext_gyro_gain, 1350), AP_GROUPINFO("GYR_GAIN", 11, AP_MotorsHeli, ext_gyro_gain, 1350),
// @Param: SV_MAN // @Param: SV_MAN
// @DisplayName: Manual Servo Mode // @DisplayName: Manual Servo Mode
// @Description: Setting this to Enabled(1) will pass radio inputs directly to servos. Setting this to Disabled(0) will enable Arducopter control of servos. This is only meant to be used by the Mission Planner using swash plate set-up. // @Description: Setting this to Enabled(1) will pass radio inputs directly to servos. Setting this to Disabled(0) will enable Arducopter control of servos. This is only meant to be used by the Mission Planner using swash plate set-up.
// @Values: 0:Disabled,1:Enabled // @Values: 0:Disabled,1:Enabled
// @User: Standard // @User: Standard
AP_GROUPINFO("SV_MAN", 12, AP_MotorsHeli, servo_manual, 0), AP_GROUPINFO("SV_MAN", 12, AP_MotorsHeli, servo_manual, 0),
// @Param: PHANG // @Param: PHANG
// @DisplayName: Swashplate Phase Angle Compensation // @DisplayName: Swashplate Phase Angle Compensation
// @Description: This corrects for phase angle errors of the helicopter main rotor head. For example if pitching the swash forward also induces a roll, that effect can be offset with this parameter. // @Description: This corrects for phase angle errors of the helicopter main rotor head. For example if pitching the swash forward also induces a roll, that effect can be offset with this parameter.
// @Range: -90 90 // @Range: -90 90
// @Units: Degrees // @Units: Degrees
// @User: Advanced // @User: Advanced
// @Increment: 1 // @Increment: 1
AP_GROUPINFO("PHANG", 13, AP_MotorsHeli, phase_angle, 0), AP_GROUPINFO("PHANG", 13, AP_MotorsHeli, phase_angle, 0),
// @Param: COLYAW // @Param: COLYAW
// @DisplayName: Collective-Yaw Mixing // @DisplayName: Collective-Yaw Mixing
// @Description: This is a feed-forward compensation to automatically add rudder input when collective pitch is increased. // @Description: This is a feed-forward compensation to automatically add rudder input when collective pitch is increased.
// @Range: 0 5 // @Range: 0 5
AP_GROUPINFO("COLYAW", 14, AP_MotorsHeli, collective_yaw_effect, 0), AP_GROUPINFO("COLYAW", 14, AP_MotorsHeli, collective_yaw_effect, 0),
// @Param: GOV_SETPOINT // @Param: GOV_SETPOINT
// @DisplayName: External Motor Governor Setpoint // @DisplayName: External Motor Governor Setpoint
// @Description: This is the PWM which is passed to the external motor governor when external governor is enabled. // @Description: This is the PWM which is passed to the external motor governor when external governor is enabled.
// @Range: 1000 2000 // @Range: 1000 2000
// @Units: PWM // @Units: PWM
// @Increment: 10 // @Increment: 10
// @User: Standard // @User: Standard
AP_GROUPINFO("GOV_SETPOINT", 15, AP_MotorsHeli, ext_gov_setpoint, 1500), AP_GROUPINFO("GOV_SETPOINT", 15, AP_MotorsHeli, ext_gov_setpoint, 1500),
// @Param: RSC_MODE // @Param: RSC_MODE
// @DisplayName: Rotor Speed Control Mode // @DisplayName: Rotor Speed Control Mode
// @Description: This sets which ESC control mode is active. // @Description: This sets which ESC control mode is active.
// @Range: 1 3 // @Range: 1 3
// @User: Standard // @User: Standard
AP_GROUPINFO("RSC_MODE", 16, AP_MotorsHeli, rsc_mode, 1), AP_GROUPINFO("RSC_MODE", 16, AP_MotorsHeli, rsc_mode, 1),
// @Param: RSC_RATE // @Param: RSC_RATE
// @DisplayName: RSC Ramp Rate // @DisplayName: RSC Ramp Rate
// @Description: This sets the time the RSC takes to ramp up to full speed (Soft Start). // @Description: This sets the time the RSC takes to ramp up to full speed (Soft Start).
// @Range: 0 6000 // @Range: 0 6000
// @Units: Seconds // @Units: Seconds
// @User: Standard // @User: Standard
AP_GROUPINFO("RSC_RATE", 17, AP_MotorsHeli, rsc_ramp_up_rate, 1000), AP_GROUPINFO("RSC_RATE", 17, AP_MotorsHeli, rsc_ramp_up_rate, 1000),
// @Param: FLYBAR_MODE // @Param: FLYBAR_MODE
// @DisplayName: Flybar Mode Selector // @DisplayName: Flybar Mode Selector
// @Description: This sets which acro mode is active. (0) is Flybarless (1) is Mechanical Flybar // @Description: This sets which acro mode is active. (0) is Flybarless (1) is Mechanical Flybar
// @Range: 0 1 // @Range: 0 1
// @User: Standard // @User: Standard
AP_GROUPINFO("FLYBAR_MODE", 18, AP_MotorsHeli, flybar_mode, 0), AP_GROUPINFO("FLYBAR_MODE", 18, AP_MotorsHeli, flybar_mode, 0),
AP_GROUPEND AP_GROUPEND
}; };
@ -166,39 +166,39 @@ const AP_Param::GroupInfo AP_MotorsHeli::var_info[] PROGMEM = {
// init // init
void AP_MotorsHeli::Init() void AP_MotorsHeli::Init()
{ {
// set update rate // set update rate
set_update_rate(_speed_hz); set_update_rate(_speed_hz);
} }
// set update rate to motors - a value in hertz or AP_MOTORS_SPEED_INSTANT_PWM for instant pwm // set update rate to motors - a value in hertz or AP_MOTORS_SPEED_INSTANT_PWM for instant pwm
void AP_MotorsHeli::set_update_rate( uint16_t speed_hz ) void AP_MotorsHeli::set_update_rate( uint16_t speed_hz )
{ {
// record requested speed // record requested speed
_speed_hz = speed_hz; _speed_hz = speed_hz;
// setup fast channels // setup fast channels
if( _speed_hz != AP_MOTORS_SPEED_INSTANT_PWM ) { if( _speed_hz != AP_MOTORS_SPEED_INSTANT_PWM ) {
_rc->SetFastOutputChannels(_BV(_motor_to_channel_map[AP_MOTORS_MOT_1]) | _BV(_motor_to_channel_map[AP_MOTORS_MOT_2]) | _BV(_motor_to_channel_map[AP_MOTORS_MOT_3]) | _BV(_motor_to_channel_map[AP_MOTORS_MOT_4]), _speed_hz); _rc->SetFastOutputChannels(_BV(_motor_to_channel_map[AP_MOTORS_MOT_1]) | _BV(_motor_to_channel_map[AP_MOTORS_MOT_2]) | _BV(_motor_to_channel_map[AP_MOTORS_MOT_3]) | _BV(_motor_to_channel_map[AP_MOTORS_MOT_4]), _speed_hz);
} }
} }
// enable - starts allowing signals to be sent to motors // enable - starts allowing signals to be sent to motors
void AP_MotorsHeli::enable() void AP_MotorsHeli::enable()
{ {
// enable output channels // enable output channels
_rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_1]); // swash servo 1 _rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_1]); // swash servo 1
_rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_2]); // swash servo 2 _rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_2]); // swash servo 2
_rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_3]); // swash servo 3 _rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_3]); // swash servo 3
_rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_4]); // yaw _rc->enable_out(_motor_to_channel_map[AP_MOTORS_MOT_4]); // yaw
_rc->enable_out(AP_MOTORS_HELI_EXT_GYRO); // for external gyro _rc->enable_out(AP_MOTORS_HELI_EXT_GYRO); // for external gyro
_rc->enable_out(AP_MOTORS_HELI_EXT_RSC); // for external RSC _rc->enable_out(AP_MOTORS_HELI_EXT_RSC); // for external RSC
} }
// output_min - sends minimum values out to the motors // output_min - sends minimum values out to the motors
void AP_MotorsHeli::output_min() void AP_MotorsHeli::output_min()
{ {
// move swash to mid // move swash to mid
move_swash(0,0,500,0); move_swash(0,0,500,0);
} }
// output_armed - sends commands to the motors // output_armed - sends commands to the motors
@ -206,217 +206,217 @@ void AP_MotorsHeli::output_armed()
{ {
// if manual override (i.e. when setting up swash), pass pilot commands straight through to swash // if manual override (i.e. when setting up swash), pass pilot commands straight through to swash
if( servo_manual == 1 ) { if( servo_manual == 1 ) {
_rc_roll->servo_out = _rc_roll->control_in; _rc_roll->servo_out = _rc_roll->control_in;
_rc_pitch->servo_out = _rc_pitch->control_in; _rc_pitch->servo_out = _rc_pitch->control_in;
_rc_throttle->servo_out = _rc_throttle->control_in; _rc_throttle->servo_out = _rc_throttle->control_in;
_rc_yaw->servo_out = _rc_yaw->control_in; _rc_yaw->servo_out = _rc_yaw->control_in;
} }
//static int counter = 0; //static int counter = 0;
_rc_roll->calc_pwm(); _rc_roll->calc_pwm();
_rc_pitch->calc_pwm(); _rc_pitch->calc_pwm();
_rc_throttle->calc_pwm(); _rc_throttle->calc_pwm();
_rc_yaw->calc_pwm(); _rc_yaw->calc_pwm();
move_swash( _rc_roll->servo_out, _rc_pitch->servo_out, _rc_throttle->servo_out, _rc_yaw->servo_out ); move_swash( _rc_roll->servo_out, _rc_pitch->servo_out, _rc_throttle->servo_out, _rc_yaw->servo_out );
rsc_control(); rsc_control();
} }
// output_disarmed - sends commands to the motors // output_disarmed - sends commands to the motors
void AP_MotorsHeli::output_disarmed() void AP_MotorsHeli::output_disarmed()
{ {
if(_rc_throttle->control_in > 0){ if(_rc_throttle->control_in > 0) {
// we have pushed up the throttle // we have pushed up the throttle
// remove safety // remove safety
_auto_armed = true; _auto_armed = true;
} }
// for helis - armed or disarmed we allow servos to move // for helis - armed or disarmed we allow servos to move
output_armed(); output_armed();
} }
// output_disarmed - sends commands to the motors // output_disarmed - sends commands to the motors
void AP_MotorsHeli::output_test() void AP_MotorsHeli::output_test()
{ {
int16_t i; int16_t i;
// Send minimum values to all motors // Send minimum values to all motors
output_min(); output_min();
// servo 1 // servo 1
for( i=0; i<5; i++ ) { for( i=0; i<5; i++ ) {
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_trim + 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_trim + 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_trim - 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_trim - 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_trim + 0); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_trim + 0);
delay(300); delay(300);
} }
// servo 2 // servo 2
for( i=0; i<5; i++ ) { for( i=0; i<5; i++ ) {
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_trim + 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_trim + 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_trim - 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_trim - 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_trim + 0); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_trim + 0);
delay(300); delay(300);
} }
// servo 3 // servo 3
for( i=0; i<5; i++ ) { for( i=0; i<5; i++ ) {
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_trim + 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_trim + 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_trim - 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_trim - 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_trim + 0); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_trim + 0);
delay(300); delay(300);
} }
// external gyro // external gyro
if( ext_gyro_enabled ) { if( ext_gyro_enabled ) {
_rc->OutputCh(AP_MOTORS_HELI_EXT_GYRO, ext_gyro_gain); _rc->OutputCh(AP_MOTORS_HELI_EXT_GYRO, ext_gyro_gain);
} }
// servo 4 // servo 4
for( i=0; i<5; i++ ) { for( i=0; i<5; i++ ) {
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_trim + 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_trim + 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_trim - 100); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_trim - 100);
delay(300); delay(300);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_trim + 0); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_trim + 0);
delay(300); delay(300);
} }
// Send minimum values to all motors // Send minimum values to all motors
output_min(); output_min();
} }
// reset_swash - free up swash for maximum movements. Used for set-up // reset_swash - free up swash for maximum movements. Used for set-up
void AP_MotorsHeli::reset_swash() void AP_MotorsHeli::reset_swash()
{ {
// free up servo ranges // free up servo ranges
_servo_1->radio_min = 1000; _servo_1->radio_min = 1000;
_servo_1->radio_max = 2000; _servo_1->radio_max = 2000;
_servo_2->radio_min = 1000; _servo_2->radio_min = 1000;
_servo_2->radio_max = 2000; _servo_2->radio_max = 2000;
_servo_3->radio_min = 1000; _servo_3->radio_min = 1000;
_servo_3->radio_max = 2000; _servo_3->radio_max = 2000;
if( swash_type == AP_MOTORS_HELI_SWASH_CCPM ) { //CCPM Swashplate, perform servo control mixing if( swash_type == AP_MOTORS_HELI_SWASH_CCPM ) { //CCPM Swashplate, perform servo control mixing
// roll factors // roll factors
_rollFactor[CH_1] = cos(radians(servo1_pos + 90 - phase_angle)); _rollFactor[CH_1] = cos(radians(servo1_pos + 90 - phase_angle));
_rollFactor[CH_2] = cos(radians(servo2_pos + 90 - phase_angle)); _rollFactor[CH_2] = cos(radians(servo2_pos + 90 - phase_angle));
_rollFactor[CH_3] = cos(radians(servo3_pos + 90 - phase_angle)); _rollFactor[CH_3] = cos(radians(servo3_pos + 90 - phase_angle));
// pitch factors // pitch factors
_pitchFactor[CH_1] = cos(radians(servo1_pos - phase_angle)); _pitchFactor[CH_1] = cos(radians(servo1_pos - phase_angle));
_pitchFactor[CH_2] = cos(radians(servo2_pos - phase_angle)); _pitchFactor[CH_2] = cos(radians(servo2_pos - phase_angle));
_pitchFactor[CH_3] = cos(radians(servo3_pos - phase_angle)); _pitchFactor[CH_3] = cos(radians(servo3_pos - phase_angle));
// collective factors // collective factors
_collectiveFactor[CH_1] = 1; _collectiveFactor[CH_1] = 1;
_collectiveFactor[CH_2] = 1; _collectiveFactor[CH_2] = 1;
_collectiveFactor[CH_3] = 1; _collectiveFactor[CH_3] = 1;
}else{ //H1 Swashplate, keep servo outputs seperated }else{ //H1 Swashplate, keep servo outputs seperated
// roll factors // roll factors
_rollFactor[CH_1] = 1; _rollFactor[CH_1] = 1;
_rollFactor[CH_2] = 0; _rollFactor[CH_2] = 0;
_rollFactor[CH_3] = 0; _rollFactor[CH_3] = 0;
// pitch factors // pitch factors
_pitchFactor[CH_1] = 0; _pitchFactor[CH_1] = 0;
_pitchFactor[CH_2] = 1; _pitchFactor[CH_2] = 1;
_pitchFactor[CH_3] = 0; _pitchFactor[CH_3] = 0;
// collective factors // collective factors
_collectiveFactor[CH_1] = 0; _collectiveFactor[CH_1] = 0;
_collectiveFactor[CH_2] = 0; _collectiveFactor[CH_2] = 0;
_collectiveFactor[CH_3] = 1; _collectiveFactor[CH_3] = 1;
} }
// set roll, pitch and throttle scaling // set roll, pitch and throttle scaling
_roll_scaler = 1.0; _roll_scaler = 1.0;
_pitch_scaler = 1.0; _pitch_scaler = 1.0;
_collective_scalar = ((float)(_rc_throttle->radio_max - _rc_throttle->radio_min))/1000.0; _collective_scalar = ((float)(_rc_throttle->radio_max - _rc_throttle->radio_min))/1000.0;
// we must be in set-up mode so mark swash as uninitialised // we must be in set-up mode so mark swash as uninitialised
_swash_initialised = false; _swash_initialised = false;
} }
// init_swash - initialise the swash plate // init_swash - initialise the swash plate
void AP_MotorsHeli::init_swash() void AP_MotorsHeli::init_swash()
{ {
// swash servo initialisation // swash servo initialisation
_servo_1->set_range(0,1000); _servo_1->set_range(0,1000);
_servo_2->set_range(0,1000); _servo_2->set_range(0,1000);
_servo_3->set_range(0,1000); _servo_3->set_range(0,1000);
_servo_4->set_angle(4500); _servo_4->set_angle(4500);
// ensure _coll values are reasonable // ensure _coll values are reasonable
if( collective_min >= collective_max ) { if( collective_min >= collective_max ) {
collective_min = 1000; collective_min = 1000;
collective_max = 2000; collective_max = 2000;
} }
collective_mid = constrain(collective_mid, collective_min, collective_max); collective_mid = constrain(collective_mid, collective_min, collective_max);
// calculate throttle mid point // calculate throttle mid point
throttle_mid = ((float)(collective_mid-collective_min))/((float)(collective_max-collective_min))*1000.0; throttle_mid = ((float)(collective_mid-collective_min))/((float)(collective_max-collective_min))*1000.0;
// determine roll, pitch and throttle scaling // determine roll, pitch and throttle scaling
_roll_scaler = (float)roll_max/4500.0; _roll_scaler = (float)roll_max/4500.0;
_pitch_scaler = (float)pitch_max/4500.0; _pitch_scaler = (float)pitch_max/4500.0;
_collective_scalar = ((float)(collective_max-collective_min))/1000.0; _collective_scalar = ((float)(collective_max-collective_min))/1000.0;
if( swash_type == AP_MOTORS_HELI_SWASH_CCPM ) { //CCPM Swashplate, perform control mixing if( swash_type == AP_MOTORS_HELI_SWASH_CCPM ) { //CCPM Swashplate, perform control mixing
// roll factors // roll factors
_rollFactor[CH_1] = cos(radians(servo1_pos + 90 - phase_angle)); _rollFactor[CH_1] = cos(radians(servo1_pos + 90 - phase_angle));
_rollFactor[CH_2] = cos(radians(servo2_pos + 90 - phase_angle)); _rollFactor[CH_2] = cos(radians(servo2_pos + 90 - phase_angle));
_rollFactor[CH_3] = cos(radians(servo3_pos + 90 - phase_angle)); _rollFactor[CH_3] = cos(radians(servo3_pos + 90 - phase_angle));
// pitch factors // pitch factors
_pitchFactor[CH_1] = cos(radians(servo1_pos - phase_angle)); _pitchFactor[CH_1] = cos(radians(servo1_pos - phase_angle));
_pitchFactor[CH_2] = cos(radians(servo2_pos - phase_angle)); _pitchFactor[CH_2] = cos(radians(servo2_pos - phase_angle));
_pitchFactor[CH_3] = cos(radians(servo3_pos - phase_angle)); _pitchFactor[CH_3] = cos(radians(servo3_pos - phase_angle));
// collective factors // collective factors
_collectiveFactor[CH_1] = 1; _collectiveFactor[CH_1] = 1;
_collectiveFactor[CH_2] = 1; _collectiveFactor[CH_2] = 1;
_collectiveFactor[CH_3] = 1; _collectiveFactor[CH_3] = 1;
}else{ //H1 Swashplate, keep servo outputs seperated }else{ //H1 Swashplate, keep servo outputs seperated
// roll factors // roll factors
_rollFactor[CH_1] = 1; _rollFactor[CH_1] = 1;
_rollFactor[CH_2] = 0; _rollFactor[CH_2] = 0;
_rollFactor[CH_3] = 0; _rollFactor[CH_3] = 0;
// pitch factors // pitch factors
_pitchFactor[CH_1] = 0; _pitchFactor[CH_1] = 0;
_pitchFactor[CH_2] = 1; _pitchFactor[CH_2] = 1;
_pitchFactor[CH_3] = 0; _pitchFactor[CH_3] = 0;
// collective factors // collective factors
_collectiveFactor[CH_1] = 0; _collectiveFactor[CH_1] = 0;
_collectiveFactor[CH_2] = 0; _collectiveFactor[CH_2] = 0;
_collectiveFactor[CH_3] = 1; _collectiveFactor[CH_3] = 1;
} }
// servo min/max values // servo min/max values
_servo_1->radio_min = 1000; _servo_1->radio_min = 1000;
_servo_1->radio_max = 2000; _servo_1->radio_max = 2000;
_servo_2->radio_min = 1000; _servo_2->radio_min = 1000;
_servo_2->radio_max = 2000; _servo_2->radio_max = 2000;
_servo_3->radio_min = 1000; _servo_3->radio_min = 1000;
_servo_3->radio_max = 2000; _servo_3->radio_max = 2000;
// mark swash as initialised // mark swash as initialised
_swash_initialised = true; _swash_initialised = true;
} }
// //
@ -429,146 +429,146 @@ void AP_MotorsHeli::init_swash()
// //
void AP_MotorsHeli::move_swash(int16_t roll_out, int16_t pitch_out, int16_t coll_out, int16_t yaw_out) void AP_MotorsHeli::move_swash(int16_t roll_out, int16_t pitch_out, int16_t coll_out, int16_t yaw_out)
{ {
int16_t yaw_offset = 0; int16_t yaw_offset = 0;
int16_t coll_out_scaled; int16_t coll_out_scaled;
if( servo_manual == 1 ) { // are we in manual servo mode? (i.e. swash set-up mode)? if( servo_manual == 1 ) { // are we in manual servo mode? (i.e. swash set-up mode)?
// check if we need to free up the swash // check if we need to free up the swash
if( _swash_initialised ) { if( _swash_initialised ) {
reset_swash(); reset_swash();
} }
coll_out_scaled = coll_out * _collective_scalar + _rc_throttle->radio_min - 1000; coll_out_scaled = coll_out * _collective_scalar + _rc_throttle->radio_min - 1000;
}else{ // regular flight mode }else{ // regular flight mode
// check if we need to reinitialise the swash // check if we need to reinitialise the swash
if( !_swash_initialised ) { if( !_swash_initialised ) {
init_swash(); init_swash();
} }
// rescale roll_out and pitch-out into the min and max ranges to provide linear motion // rescale roll_out and pitch-out into the min and max ranges to provide linear motion
// across the input range instead of stopping when the input hits the constrain value // across the input range instead of stopping when the input hits the constrain value
// these calculations are based on an assumption of the user specified roll_max and pitch_max // these calculations are based on an assumption of the user specified roll_max and pitch_max
// coming into this equation at 4500 or less, and based on the original assumption of the // coming into this equation at 4500 or less, and based on the original assumption of the
// total _servo_x.servo_out range being -4500 to 4500. // total _servo_x.servo_out range being -4500 to 4500.
roll_out = roll_out * _roll_scaler; roll_out = roll_out * _roll_scaler;
roll_out = constrain(roll_out, (int16_t)-roll_max, (int16_t)roll_max); roll_out = constrain(roll_out, (int16_t)-roll_max, (int16_t)roll_max);
pitch_out = pitch_out * _pitch_scaler; pitch_out = pitch_out * _pitch_scaler;
pitch_out = constrain(pitch_out, (int16_t)-pitch_max, (int16_t)pitch_max); pitch_out = constrain(pitch_out, (int16_t)-pitch_max, (int16_t)pitch_max);
// scale collective pitch // scale collective pitch
coll_out = constrain(coll_out, 0, 1000); coll_out = constrain(coll_out, 0, 1000);
coll_out_scaled = coll_out * _collective_scalar + collective_min - 1000; coll_out_scaled = coll_out * _collective_scalar + collective_min - 1000;
// rudder feed forward based on collective // rudder feed forward based on collective
if( !ext_gyro_enabled ) { if( !ext_gyro_enabled ) {
yaw_offset = collective_yaw_effect * abs(coll_out_scaled - throttle_mid); yaw_offset = collective_yaw_effect * abs(coll_out_scaled - throttle_mid);
} }
} }
// swashplate servos // swashplate servos
_servo_1->servo_out = (_rollFactor[CH_1] * roll_out + _pitchFactor[CH_1] * pitch_out)/10 + _collectiveFactor[CH_1] * coll_out_scaled + (_servo_1->radio_trim-1500); _servo_1->servo_out = (_rollFactor[CH_1] * roll_out + _pitchFactor[CH_1] * pitch_out)/10 + _collectiveFactor[CH_1] * coll_out_scaled + (_servo_1->radio_trim-1500);
_servo_2->servo_out = (_rollFactor[CH_2] * roll_out + _pitchFactor[CH_2] * pitch_out)/10 + _collectiveFactor[CH_2] * coll_out_scaled + (_servo_2->radio_trim-1500); _servo_2->servo_out = (_rollFactor[CH_2] * roll_out + _pitchFactor[CH_2] * pitch_out)/10 + _collectiveFactor[CH_2] * coll_out_scaled + (_servo_2->radio_trim-1500);
if( swash_type == AP_MOTORS_HELI_SWASH_H1 ) { if( swash_type == AP_MOTORS_HELI_SWASH_H1 ) {
_servo_1->servo_out += 500; _servo_1->servo_out += 500;
_servo_2->servo_out += 500; _servo_2->servo_out += 500;
} }
_servo_3->servo_out = (_rollFactor[CH_3] * roll_out + _pitchFactor[CH_3] * pitch_out)/10 + _collectiveFactor[CH_3] * coll_out_scaled + (_servo_3->radio_trim-1500); _servo_3->servo_out = (_rollFactor[CH_3] * roll_out + _pitchFactor[CH_3] * pitch_out)/10 + _collectiveFactor[CH_3] * coll_out_scaled + (_servo_3->radio_trim-1500);
_servo_4->servo_out = yaw_out + yaw_offset; _servo_4->servo_out = yaw_out + yaw_offset;
// use servo_out to calculate pwm_out and radio_out // use servo_out to calculate pwm_out and radio_out
_servo_1->calc_pwm(); _servo_1->calc_pwm();
_servo_2->calc_pwm(); _servo_2->calc_pwm();
_servo_3->calc_pwm(); _servo_3->calc_pwm();
_servo_4->calc_pwm(); _servo_4->calc_pwm();
// actually move the servos // actually move the servos
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_out); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_1], _servo_1->radio_out);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_out); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_2], _servo_2->radio_out);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_out); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_3], _servo_3->radio_out);
_rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_out); _rc->OutputCh(_motor_to_channel_map[AP_MOTORS_MOT_4], _servo_4->radio_out);
// to be compatible with other frame types // to be compatible with other frame types
motor_out[AP_MOTORS_MOT_1] = _servo_1->radio_out; motor_out[AP_MOTORS_MOT_1] = _servo_1->radio_out;
motor_out[AP_MOTORS_MOT_2] = _servo_2->radio_out; motor_out[AP_MOTORS_MOT_2] = _servo_2->radio_out;
motor_out[AP_MOTORS_MOT_3] = _servo_3->radio_out; motor_out[AP_MOTORS_MOT_3] = _servo_3->radio_out;
motor_out[AP_MOTORS_MOT_4] = _servo_4->radio_out; motor_out[AP_MOTORS_MOT_4] = _servo_4->radio_out;
// output gyro value // output gyro value
if( ext_gyro_enabled ) { if( ext_gyro_enabled ) {
_rc->OutputCh(AP_MOTORS_HELI_EXT_GYRO, ext_gyro_gain); _rc->OutputCh(AP_MOTORS_HELI_EXT_GYRO, ext_gyro_gain);
} }
// InstantPWM // InstantPWM
if( _speed_hz == AP_MOTORS_SPEED_INSTANT_PWM ) { if( _speed_hz == AP_MOTORS_SPEED_INSTANT_PWM ) {
_rc->Force_Out0_Out1(); _rc->Force_Out0_Out1();
_rc->Force_Out2_Out3(); _rc->Force_Out2_Out3();
} }
} }
void AP_MotorsHeli::rsc_control() void AP_MotorsHeli::rsc_control()
{ {
switch ( rsc_mode ) { switch ( rsc_mode ) {
case AP_MOTORSHELI_RSC_MODE_CH8_PASSTHROUGH: case AP_MOTORSHELI_RSC_MODE_CH8_PASSTHROUGH:
if( armed() && _rc_8->control_in > 10 ){ if( armed() && _rc_8->control_in > 10 ) {
if (rsc_ramp < rsc_ramp_up_rate){ if (rsc_ramp < rsc_ramp_up_rate) {
rsc_ramp++; rsc_ramp++;
rsc_output = map(rsc_ramp, 0, rsc_ramp_up_rate, 1000, _rc_8->control_in); rsc_output = map(rsc_ramp, 0, rsc_ramp_up_rate, 1000, _rc_8->control_in);
} else { } else {
rsc_output = _rc_8->control_in; rsc_output = _rc_8->control_in;
} }
} else if( !armed() ) { } else if( !armed() ) {
_rc->OutputCh(AP_MOTORS_HELI_EXT_RSC, _rc_8->radio_min); _rc->OutputCh(AP_MOTORS_HELI_EXT_RSC, _rc_8->radio_min);
rsc_ramp = 0; //Return RSC Ramp to 0 rsc_ramp = 0; //Return RSC Ramp to 0
} }
break; break;
case AP_MOTORSHELI_RSC_MODE_EXT_GOV: case AP_MOTORSHELI_RSC_MODE_EXT_GOV:
if( armed() && _rc_throttle->control_in > 10){ if( armed() && _rc_throttle->control_in > 10) {
if (rsc_ramp < rsc_ramp_up_rate){ if (rsc_ramp < rsc_ramp_up_rate) {
rsc_ramp++; rsc_ramp++;
rsc_output = map(rsc_ramp, 0, rsc_ramp_up_rate, 1000, ext_gov_setpoint); rsc_output = map(rsc_ramp, 0, rsc_ramp_up_rate, 1000, ext_gov_setpoint);
} else { } else {
rsc_output = ext_gov_setpoint; rsc_output = ext_gov_setpoint;
} }
} else { } else {
rsc_ramp--; //Return RSC Ramp to 0 slowly, allowing for "warm restart" rsc_ramp--; //Return RSC Ramp to 0 slowly, allowing for "warm restart"
if (rsc_ramp < 0){ if (rsc_ramp < 0) {
rsc_ramp = 0; rsc_ramp = 0;
} }
rsc_output = 1000; //Just to be sure RSC output is 0 rsc_output = 1000; //Just to be sure RSC output is 0
} }
_rc->OutputCh(AP_MOTORS_HELI_EXT_RSC, rsc_output); _rc->OutputCh(AP_MOTORS_HELI_EXT_RSC, rsc_output);
break; break;
// case 3: // Open Loop ESC Control // case 3: // Open Loop ESC Control
// //
// coll_scaled = _motors->coll_out_scaled + 1000; // coll_scaled = _motors->coll_out_scaled + 1000;
// if(coll_scaled <= _motors->collective_mid){ // if(coll_scaled <= _motors->collective_mid){
// esc_ol_output = map(coll_scaled, _motors->collective_min, _motors->collective_mid, esc_out_low, esc_out_mid); // Bottom half of V-curve // esc_ol_output = map(coll_scaled, _motors->collective_min, _motors->collective_mid, esc_out_low, esc_out_mid); // Bottom half of V-curve
// } else if (coll_scaled > _motors->collective_mid){ // } else if (coll_scaled > _motors->collective_mid){
// esc_ol_output = map(coll_scaled, _motors->collective_mid, _motors->collective_max, esc_out_mid, esc_out_high); // Top half of V-curve // esc_ol_output = map(coll_scaled, _motors->collective_mid, _motors->collective_max, esc_out_mid, esc_out_high); // Top half of V-curve
// } else { esc_ol_output = 1000; } // Just in case. // } else { esc_ol_output = 1000; } // Just in case.
// //
// if(_motors->armed() && _rc_throttle->control_in > 10){ // if(_motors->armed() && _rc_throttle->control_in > 10){
// if (ext_esc_ramp < ext_esc_ramp_up){ // if (ext_esc_ramp < ext_esc_ramp_up){
// ext_esc_ramp++; // ext_esc_ramp++;
// ext_esc_output = map(ext_esc_ramp, 0, ext_esc_ramp_up, 1000, esc_ol_output); // ext_esc_output = map(ext_esc_ramp, 0, ext_esc_ramp_up, 1000, esc_ol_output);
// } else { // } else {
// ext_esc_output = esc_ol_output; // ext_esc_output = esc_ol_output;
// } // }
// } else { // } else {
// ext_esc_ramp = 0; //Return ESC Ramp to 0 // ext_esc_ramp = 0; //Return ESC Ramp to 0
// ext_esc_output = 1000; //Just to be sure ESC output is 0 // ext_esc_output = 1000; //Just to be sure ESC output is 0
//} //}
// _rc->OutputCh(AP_MOTORS_HELI_EXT_ESC, ext_esc_output); // _rc->OutputCh(AP_MOTORS_HELI_EXT_ESC, ext_esc_output);
// break; // break;
default: default:
break; break;
} }
}; };