mirror of https://github.com/ArduPilot/ardupilot
153 lines
4.8 KiB
C++
153 lines
4.8 KiB
C++
#include "Blimp.h"
|
|
|
|
// This is the scale used for RC inputs so that they can be scaled to the float point values used in the sin wave code.
|
|
#define FIN_SCALE_MAX 1000
|
|
|
|
/*
|
|
2nd group of parameters
|
|
*/
|
|
const AP_Param::GroupInfo Fins::var_info[] = {
|
|
|
|
// @Param: FREQ_HZ
|
|
// @DisplayName: Fins frequency
|
|
// @Description: This is the oscillation frequency of the fins
|
|
// @Range: 1 10
|
|
// @User: Standard
|
|
AP_GROUPINFO("FREQ_HZ", 1, Fins, freq_hz, 3),
|
|
|
|
// @Param: TURBO_MODE
|
|
// @DisplayName: Enable turbo mode
|
|
// @Description: Enables double speed on high offset.
|
|
// @Range: 0 1
|
|
// @User: Standard
|
|
AP_GROUPINFO("TURBO_MODE", 2, Fins, turbo_mode, 0),
|
|
|
|
AP_GROUPEND
|
|
};
|
|
|
|
//constructor
|
|
Fins::Fins(uint16_t loop_rate) :
|
|
_loop_rate(loop_rate)
|
|
{
|
|
AP_Param::setup_object_defaults(this, var_info);
|
|
}
|
|
|
|
void Fins::setup_fins()
|
|
{
|
|
//amp r f d y,off r f d y right, front, down, yaw
|
|
add_fin(0, 0, -1, 0.5, 0, 0, 0, -0.5, 0); //Back(?)
|
|
add_fin(1, 0, 1, 0.5, 0, 0, 0, -0.5, 0); //Front(?)
|
|
add_fin(2, -1, 0, 0, 0.5, 0, 0, 0, 0.5); //Right
|
|
add_fin(3, 1, 0, 0, 0.5, 0, 0, 0, -0.5); //Left
|
|
|
|
SRV_Channels::set_angle(SRV_Channel::k_motor1, FIN_SCALE_MAX);
|
|
SRV_Channels::set_angle(SRV_Channel::k_motor2, FIN_SCALE_MAX);
|
|
SRV_Channels::set_angle(SRV_Channel::k_motor3, FIN_SCALE_MAX);
|
|
SRV_Channels::set_angle(SRV_Channel::k_motor4, FIN_SCALE_MAX);
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_WARNING, "MIR: All fins have been added.");
|
|
}
|
|
|
|
void Fins::add_fin(int8_t fin_num, float right_amp_fac, float front_amp_fac, float down_amp_fac, float yaw_amp_fac,
|
|
float right_off_fac, float front_off_fac, float down_off_fac, float yaw_off_fac)
|
|
{
|
|
|
|
// ensure valid fin number is provided
|
|
if (fin_num >= 0 && fin_num < NUM_FINS) {
|
|
|
|
// set amplitude factors
|
|
_right_amp_factor[fin_num] = right_amp_fac;
|
|
_front_amp_factor[fin_num] = front_amp_fac;
|
|
_down_amp_factor[fin_num] = down_amp_fac;
|
|
_yaw_amp_factor[fin_num] = yaw_amp_fac;
|
|
|
|
// set offset factors
|
|
_right_off_factor[fin_num] = right_off_fac;
|
|
_front_off_factor[fin_num] = front_off_fac;
|
|
_down_off_factor[fin_num] = down_off_fac;
|
|
_yaw_off_factor[fin_num] = yaw_off_fac;
|
|
}
|
|
}
|
|
|
|
//B,F,R,L = 0,1,2,3
|
|
void Fins::output()
|
|
{
|
|
if (!_armed) {
|
|
// set everything to zero so fins stop moving
|
|
right_out = 0;
|
|
front_out = 0;
|
|
down_out = 0;
|
|
yaw_out = 0;
|
|
}
|
|
|
|
right_out /= RC_SCALE;
|
|
front_out /= RC_SCALE;
|
|
down_out /= RC_SCALE;
|
|
yaw_out /= RC_SCALE;
|
|
|
|
_time = AP_HAL::micros() * 1.0e-6;
|
|
|
|
for (int8_t i=0; i<NUM_FINS; i++) {
|
|
_amp[i] = max(0,_right_amp_factor[i]*right_out) + max(0,_front_amp_factor[i]*front_out) +
|
|
fabsf(_down_amp_factor[i]*down_out) + fabsf(_yaw_amp_factor[i]*yaw_out);
|
|
_off[i] = _right_off_factor[i]*right_out + _front_off_factor[i]*front_out +
|
|
_down_off_factor[i]*down_out + _yaw_off_factor[i]*yaw_out;
|
|
_omm[i] = 1;
|
|
|
|
_num_added = 0;
|
|
if (max(0,_right_amp_factor[i]*right_out) > 0.0f) {
|
|
_num_added++;
|
|
}
|
|
if (max(0,_front_amp_factor[i]*front_out) > 0.0f) {
|
|
_num_added++;
|
|
}
|
|
if (fabsf(_down_amp_factor[i]*down_out) > 0.0f) {
|
|
_num_added++;
|
|
}
|
|
if (fabsf(_yaw_amp_factor[i]*yaw_out) > 0.0f) {
|
|
_num_added++;
|
|
}
|
|
|
|
if (_num_added > 0) {
|
|
_off[i] = _off[i]/_num_added; //average the offsets
|
|
}
|
|
|
|
if ((_amp[i]+fabsf(_off[i])) > 1) {
|
|
_amp[i] = 1 - fabsf(_off[i]);
|
|
}
|
|
|
|
if (turbo_mode) {
|
|
//double speed fins if offset at max...
|
|
if (_amp[i] <= 0.6 && fabsf(_off[i]) >= 0.4) {
|
|
_omm[i] = 2;
|
|
}
|
|
}
|
|
|
|
// finding and outputting current position for each servo from sine wave
|
|
_pos[i]= _amp[i]*cosf(freq_hz * _omm[i] * _time * 2 * M_PI) + _off[i];
|
|
SRV_Channels::set_output_scaled(SRV_Channels::get_motor_function(i), _pos[i] * FIN_SCALE_MAX);
|
|
}
|
|
|
|
//For debugging purposes. Displays in the debug terminal so it doesn't flood the GCS.
|
|
::printf("FINS (amp %.1f %.1f %.1f %.1f off %.1f %.1f %.1f %.1f omm %.1f %.1f %.1f %.1f)\n",
|
|
_amp[0], _amp[1], _amp[2], _amp[3], _off[0], _off[1], _off[2], _off[3], _omm[0], _omm[1], _omm[2], _omm[3]);
|
|
}
|
|
|
|
void Fins::output_min()
|
|
{
|
|
right_out = 0;
|
|
front_out = 0;
|
|
down_out = 0;
|
|
yaw_out = 0;
|
|
Fins::output();
|
|
}
|
|
|
|
// TODO - Probably want to completely get rid of the desired spool state thing.
|
|
void Fins::set_desired_spool_state(DesiredSpoolState spool)
|
|
{
|
|
if (_armed || (spool == DesiredSpoolState::SHUT_DOWN)) {
|
|
// Set DesiredSpoolState only if it is either armed or it wants to shut down.
|
|
_spool_desired = spool;
|
|
}
|
|
};
|