mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-04 15:08:28 -04:00
8b9c96814b
Further to refactor of RC_Channel class which included adding get_xx set_xx methods, change reads and writes to the public members to calls to get and set functionsss old public member(int16_t) get function -> int16_t set function (int16_t) (expression where c is an object of type RC_Channel) c.radio_in c.get_radio_in() c.set_radio_in(v) c.control_in c.get_control_in() c.set_control_in(v) c.servo_out c.get_servo_out() c.set_servo_out(v) c.pwm_out c.get_pwm_out() // use existing c.radio_out c.get_radio_out() c.set_radio_out(v) c.radio_max c.get_radio_max() c.set_radio_max(v) c.radio_min c.get_radio_min() c.set_radio_min(v) c.radio_trim c.get_radio_trim() c.set_radio_trim(v); c.min_max_configured() // return true if min and max are configured Because data members of RC_Channels are now private and so cannot be written directly some overloads are provided in the Plane classes to provide the old functionality new overload Plane::stick_mix_channel(RC_Channel *channel) which forwards to the previously existing void stick_mix_channel(RC_Channel *channel, int16_t &servo_out); new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const which forwards to (uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const; Rename functions RC_Channel_aux::set_radio_trim(Aux_servo_function_t function) to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function) RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value) to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value) Rationale: RC_Channel is a complicated class, which combines several functionalities dealing with stick inputs in pwm and logical units, logical and actual actuator outputs, unit conversion etc, etc The intent of this PR is to clarify existing use of the class. At the basic level it should now be possible to grep all places where private variable is set by searching for the set_xx function. (The wider purpose is to provide a more generic and logically simpler method of output mixing. This is a small step)
158 lines
6.8 KiB
C++
158 lines
6.8 KiB
C++
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
#include "AP_Mount_Backend.h"
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
// set_angle_targets - sets angle targets in degrees
|
|
void AP_Mount_Backend::set_angle_targets(float roll, float tilt, float pan)
|
|
{
|
|
// set angle targets
|
|
_angle_ef_target_rad.x = radians(roll);
|
|
_angle_ef_target_rad.y = radians(tilt);
|
|
_angle_ef_target_rad.z = radians(pan);
|
|
|
|
// set the mode to mavlink targeting
|
|
_frontend.set_mode(_instance, MAV_MOUNT_MODE_MAVLINK_TARGETING);
|
|
}
|
|
|
|
// set_roi_target - sets target location that mount should attempt to point towards
|
|
void AP_Mount_Backend::set_roi_target(const struct Location &target_loc)
|
|
{
|
|
// set the target gps location
|
|
_state._roi_target = target_loc;
|
|
|
|
// set the mode to GPS tracking mode
|
|
_frontend.set_mode(_instance, MAV_MOUNT_MODE_GPS_POINT);
|
|
}
|
|
|
|
// configure_msg - process MOUNT_CONFIGURE messages received from GCS. deprecated.
|
|
void AP_Mount_Backend::configure_msg(mavlink_message_t* msg)
|
|
{
|
|
__mavlink_mount_configure_t packet;
|
|
mavlink_msg_mount_configure_decode(msg, &packet);
|
|
|
|
set_mode((MAV_MOUNT_MODE)packet.mount_mode);
|
|
}
|
|
|
|
// control_msg - process MOUNT_CONTROL messages received from GCS. deprecated.
|
|
void AP_Mount_Backend::control_msg(mavlink_message_t *msg)
|
|
{
|
|
__mavlink_mount_control_t packet;
|
|
mavlink_msg_mount_control_decode(msg, &packet);
|
|
|
|
control((int32_t)packet.input_a, (int32_t)packet.input_b, (int32_t)packet.input_c, _state._mode);
|
|
}
|
|
|
|
void AP_Mount_Backend::control(int32_t pitch_or_lat, int32_t roll_or_lon, int32_t yaw_or_alt, MAV_MOUNT_MODE mount_mode)
|
|
{
|
|
_frontend.set_mode(_instance, mount_mode);
|
|
|
|
// interpret message fields based on mode
|
|
switch (_frontend.get_mode(_instance)) {
|
|
case MAV_MOUNT_MODE_RETRACT:
|
|
case MAV_MOUNT_MODE_NEUTRAL:
|
|
// do nothing with request if mount is retracted or in neutral position
|
|
break;
|
|
|
|
// set earth frame target angles from mavlink message
|
|
case MAV_MOUNT_MODE_MAVLINK_TARGETING:
|
|
set_angle_targets(roll_or_lon*0.01f, pitch_or_lat*0.01f, yaw_or_alt*0.01f);
|
|
break;
|
|
|
|
// Load neutral position and start RC Roll,Pitch,Yaw control with stabilization
|
|
case MAV_MOUNT_MODE_RC_TARGETING:
|
|
// do nothing if pilot is controlling the roll, pitch and yaw
|
|
break;
|
|
|
|
// set lat, lon, alt position targets from mavlink message
|
|
case MAV_MOUNT_MODE_GPS_POINT:
|
|
Location target_location;
|
|
memset(&target_location, 0, sizeof(target_location));
|
|
target_location.lat = pitch_or_lat;
|
|
target_location.lng = roll_or_lon;
|
|
target_location.alt = yaw_or_alt;
|
|
target_location.flags.relative_alt = true;
|
|
set_roi_target(target_location);
|
|
break;
|
|
|
|
default:
|
|
// do nothing
|
|
break;
|
|
}
|
|
}
|
|
|
|
// update_targets_from_rc - updates angle targets using input from receiver
|
|
void AP_Mount_Backend::update_targets_from_rc()
|
|
{
|
|
#define rc_ch(i) RC_Channel::rc_channel(i-1)
|
|
|
|
uint8_t roll_rc_in = _state._roll_rc_in;
|
|
uint8_t tilt_rc_in = _state._tilt_rc_in;
|
|
uint8_t pan_rc_in = _state._pan_rc_in;
|
|
|
|
// if joystick_speed is defined then pilot input defines a rate of change of the angle
|
|
if (_frontend._joystick_speed) {
|
|
// allow pilot speed position input to come directly from an RC_Channel
|
|
if (roll_rc_in && rc_ch(roll_rc_in)) {
|
|
_angle_ef_target_rad.x += rc_ch(roll_rc_in)->norm_input_dz() * 0.0001f * _frontend._joystick_speed;
|
|
_angle_ef_target_rad.x = constrain_float(_angle_ef_target_rad.x, radians(_state._roll_angle_min*0.01f), radians(_state._roll_angle_max*0.01f));
|
|
}
|
|
if (tilt_rc_in && (rc_ch(tilt_rc_in))) {
|
|
_angle_ef_target_rad.y += rc_ch(tilt_rc_in)->norm_input_dz() * 0.0001f * _frontend._joystick_speed;
|
|
_angle_ef_target_rad.y = constrain_float(_angle_ef_target_rad.y, radians(_state._tilt_angle_min*0.01f), radians(_state._tilt_angle_max*0.01f));
|
|
}
|
|
if (pan_rc_in && (rc_ch(pan_rc_in))) {
|
|
_angle_ef_target_rad.z += rc_ch(pan_rc_in)->norm_input_dz() * 0.0001f * _frontend._joystick_speed;
|
|
_angle_ef_target_rad.z = constrain_float(_angle_ef_target_rad.z, radians(_state._pan_angle_min*0.01f), radians(_state._pan_angle_max*0.01f));
|
|
}
|
|
} else {
|
|
// allow pilot position input to come directly from an RC_Channel
|
|
if (roll_rc_in && (rc_ch(roll_rc_in))) {
|
|
_angle_ef_target_rad.x = angle_input_rad(rc_ch(roll_rc_in), _state._roll_angle_min, _state._roll_angle_max);
|
|
}
|
|
if (tilt_rc_in && (rc_ch(tilt_rc_in))) {
|
|
_angle_ef_target_rad.y = angle_input_rad(rc_ch(tilt_rc_in), _state._tilt_angle_min, _state._tilt_angle_max);
|
|
}
|
|
if (pan_rc_in && (rc_ch(pan_rc_in))) {
|
|
_angle_ef_target_rad.z = angle_input_rad(rc_ch(pan_rc_in), _state._pan_angle_min, _state._pan_angle_max);
|
|
}
|
|
}
|
|
}
|
|
|
|
// returns the angle (degrees*100) that the RC_Channel input is receiving
|
|
int32_t AP_Mount_Backend::angle_input(RC_Channel* rc, int16_t angle_min, int16_t angle_max)
|
|
{
|
|
return (rc->get_reverse() ? -1 : 1) * (rc->get_radio_in() - rc->get_radio_min())
|
|
* (int32_t)(angle_max - angle_min) / (rc->get_radio_max() - rc->get_radio_min()) + (rc->get_reverse() ? angle_max : angle_min);
|
|
}
|
|
|
|
// returns the angle (radians) that the RC_Channel input is receiving
|
|
float AP_Mount_Backend::angle_input_rad(RC_Channel* rc, int16_t angle_min, int16_t angle_max)
|
|
{
|
|
return radians(angle_input(rc, angle_min, angle_max)*0.01f);
|
|
}
|
|
|
|
// calc_angle_to_location - calculates the earth-frame roll, tilt and pan angles (and radians) to point at the given target
|
|
void AP_Mount_Backend::calc_angle_to_location(const struct Location &target, Vector3f& angles_to_target_rad, bool calc_tilt, bool calc_pan)
|
|
{
|
|
float GPS_vector_x = (target.lng-_frontend._current_loc.lng)*cosf(ToRad((_frontend._current_loc.lat+target.lat)*0.00000005f))*0.01113195f;
|
|
float GPS_vector_y = (target.lat-_frontend._current_loc.lat)*0.01113195f;
|
|
float GPS_vector_z = (target.alt-_frontend._current_loc.alt); // baro altitude(IN CM) should be adjusted to known home elevation before take off (Set altimeter).
|
|
float target_distance = 100.0f*pythagorous2(GPS_vector_x, GPS_vector_y); // Careful , centimeters here locally. Baro/alt is in cm, lat/lon is in meters.
|
|
|
|
// initialise all angles to zero
|
|
angles_to_target_rad.zero();
|
|
|
|
// tilt calcs
|
|
if (calc_tilt) {
|
|
angles_to_target_rad.y = atan2f(GPS_vector_z, target_distance);
|
|
}
|
|
|
|
// pan calcs
|
|
if (calc_pan) {
|
|
// calc absolute heading and then onvert to vehicle relative yaw
|
|
angles_to_target_rad.z = wrap_PI(atan2f(GPS_vector_x, GPS_vector_y) - _frontend._ahrs.yaw);
|
|
}
|
|
}
|