2020-07-24 05:29:43 -03:00
|
|
|
#include <AP_Winch/AP_Winch_Daiwa.h>
|
2022-03-03 23:55:36 -04:00
|
|
|
|
2023-03-02 20:48:38 -04:00
|
|
|
#if AP_WINCH_DAIWA_ENABLED
|
|
|
|
|
2022-03-03 23:55:36 -04:00
|
|
|
#include <AP_Logger/AP_Logger.h>
|
2020-07-24 05:29:43 -03:00
|
|
|
#include <GCS_MAVLink/GCS.h>
|
2023-01-24 17:39:44 -04:00
|
|
|
#include <SRV_Channel/SRV_Channel.h>
|
2020-07-24 05:29:43 -03:00
|
|
|
|
2023-10-02 02:02:40 -03:00
|
|
|
#define AP_WINCH_DAIWA_STUCK_TIMEOUT_MS 1000 // winch is considered stuck if unmoving for this many milliseconds
|
|
|
|
#define AP_WINCH_DAIWA_STUCK_CENTER_MS 1000 // stuck protection outputs zero rate for this many milliseconds
|
|
|
|
#define AP_WINCH_DAIWA_STUCK_LENGTH_MIN 0.1 // stuck protection active when line length is more than this many meters
|
|
|
|
#define AP_WINCH_DAIWA_STUCK_RATE_MIN 0.2 // stuck protection active when desired rate is at least this value (+ve or -ve)
|
|
|
|
|
2020-07-24 05:29:43 -03:00
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
|
2023-09-25 04:35:54 -03:00
|
|
|
const char* AP_Winch_Daiwa::send_text_prefix = "Winch:";
|
|
|
|
|
2020-07-24 05:29:43 -03:00
|
|
|
// true if winch is healthy
|
|
|
|
bool AP_Winch_Daiwa::healthy() const
|
|
|
|
{
|
|
|
|
// healthy if we have received data within 3 seconds
|
|
|
|
return (AP_HAL::millis() - data_update_ms < 3000);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AP_Winch_Daiwa::init()
|
|
|
|
{
|
2020-08-06 22:24:36 -03:00
|
|
|
// call superclass init
|
|
|
|
AP_Winch_Backend::init();
|
2020-07-24 05:29:43 -03:00
|
|
|
|
|
|
|
// initialise serial connection to winch
|
|
|
|
const AP_SerialManager &serial_manager = AP::serialmanager();
|
|
|
|
uart = serial_manager.find_serial(AP_SerialManager::SerialProtocol_Winch, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AP_Winch_Daiwa::update()
|
|
|
|
{
|
|
|
|
// return immediately if no servo is assigned to control the winch
|
|
|
|
if (!SRV_Channels::function_assigned(SRV_Channel::k_winch)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// read latest data from winch
|
|
|
|
read_data_from_winch();
|
|
|
|
|
|
|
|
// read pilot input
|
|
|
|
read_pilot_desired_rate();
|
|
|
|
|
|
|
|
// send outputs to winch
|
|
|
|
control_winch();
|
2023-09-25 04:35:54 -03:00
|
|
|
|
|
|
|
// update_user
|
|
|
|
update_user();
|
2020-07-24 05:29:43 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
//send generator status
|
|
|
|
void AP_Winch_Daiwa::send_status(const GCS_MAVLINK &channel)
|
|
|
|
{
|
|
|
|
// prepare status bitmask
|
|
|
|
uint32_t status_bitmask = 0;
|
|
|
|
if (healthy()) {
|
|
|
|
status_bitmask |= MAV_WINCH_STATUS_HEALTHY;
|
|
|
|
}
|
|
|
|
if (latest.thread_end) {
|
|
|
|
status_bitmask |= MAV_WINCH_STATUS_FULLY_RETRACTED;
|
|
|
|
}
|
|
|
|
if (latest.moving > 0) {
|
|
|
|
status_bitmask |= MAV_WINCH_STATUS_MOVING;
|
|
|
|
}
|
|
|
|
if (latest.clutch > 0) {
|
|
|
|
status_bitmask |= MAV_WINCH_STATUS_CLUTCH_ENGAGED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// convert speed percentage to absolute speed
|
2020-09-07 04:23:44 -03:00
|
|
|
const float speed_ms = fabsf(config.rate_max) * (float)latest.speed_pct * 0.01f;
|
2020-07-24 05:29:43 -03:00
|
|
|
|
|
|
|
// send status
|
|
|
|
mavlink_msg_winch_status_send(
|
|
|
|
channel.get_chan(),
|
|
|
|
AP_HAL::micros64(),
|
|
|
|
latest.line_length,
|
|
|
|
speed_ms,
|
|
|
|
(float)latest.tension_corrected * 0.01f,
|
|
|
|
latest.voltage,
|
|
|
|
latest.current,
|
|
|
|
latest.motor_temp,
|
|
|
|
status_bitmask);
|
|
|
|
}
|
|
|
|
|
|
|
|
// write log
|
|
|
|
void AP_Winch_Daiwa::write_log()
|
|
|
|
{
|
|
|
|
AP::logger().Write_Winch(
|
|
|
|
healthy(),
|
|
|
|
latest.thread_end,
|
|
|
|
latest.moving,
|
|
|
|
latest.clutch,
|
|
|
|
(uint8_t)config.control_mode,
|
|
|
|
config.length_desired,
|
|
|
|
get_current_length(),
|
|
|
|
config.rate_desired,
|
|
|
|
latest.tension_corrected,
|
|
|
|
latest.voltage,
|
|
|
|
constrain_int16(latest.motor_temp, INT8_MIN, INT8_MAX));
|
|
|
|
}
|
|
|
|
|
|
|
|
// read incoming data from winch and update intermediate and latest structures
|
|
|
|
void AP_Winch_Daiwa::read_data_from_winch()
|
|
|
|
{
|
|
|
|
// return immediately if serial port is not configured
|
|
|
|
if (uart == nullptr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// read any available characters from serial port and send to GCS
|
|
|
|
int16_t nbytes = uart->available();
|
|
|
|
while (nbytes-- > 0) {
|
|
|
|
int16_t b = uart->read();
|
|
|
|
|
|
|
|
if ((b >= '0' && b <= '9') || (b >= 'A' && b <= 'F') || (b >= 'a' && b <= 'f')) {
|
|
|
|
// add digits to buffer
|
|
|
|
buff[buff_len] = b;
|
|
|
|
buff_len++;
|
2020-08-06 22:24:36 -03:00
|
|
|
if (buff_len >= ARRAY_SIZE(buff)) {
|
2020-07-24 05:29:43 -03:00
|
|
|
buff_len = 0;
|
|
|
|
parse_state = ParseState::WAITING_FOR_TIME;
|
|
|
|
}
|
|
|
|
} else if (b == ',' || b == '\r') {
|
|
|
|
// comma or carriage return signals end of current value
|
2020-08-06 22:24:36 -03:00
|
|
|
// parse number received and empty buffer
|
2020-07-24 05:29:43 -03:00
|
|
|
buff[buff_len] = '\0';
|
2020-08-06 22:24:36 -03:00
|
|
|
long int value = (int32_t)strtol(buff, nullptr, 16);
|
|
|
|
buff_len = 0;
|
|
|
|
switch (parse_state) {
|
|
|
|
case ParseState::WAITING_FOR_TIME:
|
|
|
|
intermediate.time_ms = (uint32_t)value;
|
|
|
|
parse_state = ParseState::WAITING_FOR_SPOOL;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_SPOOL:
|
|
|
|
intermediate.line_length = (int32_t)value * line_length_correction_factor;
|
|
|
|
parse_state = ParseState::WAITING_FOR_TENSION1;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_TENSION1:
|
|
|
|
intermediate.tension_uncorrected = (uint16_t)value;
|
|
|
|
parse_state = ParseState::WAITING_FOR_TENSION2;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_TENSION2:
|
|
|
|
intermediate.tension_corrected = (uint16_t)value;
|
|
|
|
parse_state = ParseState::WAITING_FOR_THREAD_END;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_THREAD_END:
|
|
|
|
intermediate.thread_end = (value > 0);
|
|
|
|
parse_state = ParseState::WAITING_FOR_MOVING;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_MOVING:
|
|
|
|
intermediate.moving = constrain_int32(value, 0, UINT8_MAX);
|
|
|
|
parse_state = ParseState::WAITING_FOR_CLUTCH;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_CLUTCH:
|
|
|
|
intermediate.clutch = constrain_int32(value, 0, UINT8_MAX);
|
|
|
|
parse_state = ParseState::WAITING_FOR_SPEED;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_SPEED:
|
|
|
|
intermediate.speed_pct = constrain_int32(value, 0, UINT8_MAX);
|
|
|
|
parse_state = ParseState::WAITING_FOR_VOLTAGE;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_VOLTAGE:
|
|
|
|
intermediate.voltage = (float)value * 0.1f;
|
|
|
|
parse_state = ParseState::WAITING_FOR_CURRENT;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_CURRENT:
|
|
|
|
intermediate.current = (float)value * 0.1f;
|
|
|
|
parse_state = ParseState::WAITING_FOR_PCB_TEMP;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_PCB_TEMP:
|
|
|
|
intermediate.pcb_temp = (float)value * 0.1f;
|
|
|
|
parse_state = ParseState::WAITING_FOR_MOTOR_TEMP;
|
|
|
|
break;
|
|
|
|
case ParseState::WAITING_FOR_MOTOR_TEMP:
|
|
|
|
intermediate.motor_temp = (float)value * 0.1f;
|
|
|
|
// successfully parsed a complete message
|
|
|
|
latest = intermediate;
|
|
|
|
data_update_ms = AP_HAL::millis();
|
2020-07-24 05:29:43 -03:00
|
|
|
parse_state = ParseState::WAITING_FOR_TIME;
|
2020-08-06 22:24:36 -03:00
|
|
|
break;
|
2020-07-24 05:29:43 -03:00
|
|
|
}
|
|
|
|
} else {
|
2020-08-06 22:24:36 -03:00
|
|
|
// line feed or unexpected characters
|
2020-07-24 05:29:43 -03:00
|
|
|
buff_len = 0;
|
|
|
|
parse_state = ParseState::WAITING_FOR_TIME;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// update pwm outputs to control winch
|
|
|
|
void AP_Winch_Daiwa::control_winch()
|
|
|
|
{
|
2020-08-06 22:24:36 -03:00
|
|
|
const uint32_t now_ms = AP_HAL::millis();
|
2022-03-11 13:35:51 -04:00
|
|
|
float dt = (now_ms - control_update_ms) * 0.001f;
|
2020-07-24 05:29:43 -03:00
|
|
|
if (dt > 1.0f) {
|
|
|
|
dt = 0.0f;
|
|
|
|
}
|
|
|
|
control_update_ms = now_ms;
|
|
|
|
|
|
|
|
// if real doing any control output trim value
|
|
|
|
if (config.control_mode == AP_Winch::ControlMode::RELAXED) {
|
|
|
|
// if not doing any control output release clutch and move winch to trim
|
|
|
|
SRV_Channels::set_output_limit(SRV_Channel::k_winch_clutch, SRV_Channel::Limit::MAX);
|
|
|
|
SRV_Channels::set_output_scaled(SRV_Channel::k_winch, 0);
|
|
|
|
|
|
|
|
// rate used for acceleration limiting reset to zero
|
|
|
|
set_previous_rate(0.0f);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// release clutch
|
|
|
|
SRV_Channels::set_output_limit(SRV_Channel::k_winch_clutch, SRV_Channel::Limit::MIN);
|
|
|
|
|
|
|
|
// if doing position control, calculate position error to desired rate
|
|
|
|
if ((config.control_mode == AP_Winch::ControlMode::POSITION) && healthy()) {
|
|
|
|
float position_error = config.length_desired - latest.line_length;
|
|
|
|
config.rate_desired = constrain_float(position_error * config.pos_p, -config.rate_max, config.rate_max);
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply acceleration limited to rate
|
2023-10-02 02:02:40 -03:00
|
|
|
float rate_limited = get_rate_limited_by_accel(config.rate_desired, dt);
|
|
|
|
|
|
|
|
// apply stuck protection to rate
|
|
|
|
rate_limited = get_stuck_protected_rate(now_ms, rate_limited);
|
2020-07-24 05:29:43 -03:00
|
|
|
|
|
|
|
// use linear interpolation to calculate output to move winch at desired rate
|
2021-09-18 14:57:26 -03:00
|
|
|
float scaled_output = 0;
|
2020-07-24 05:29:43 -03:00
|
|
|
if (!is_zero(rate_limited)) {
|
|
|
|
scaled_output = linear_interpolate(output_dz, 1000, fabsf(rate_limited), 0, config.rate_max) * (is_positive(rate_limited) ? 1.0f : -1.0f);
|
|
|
|
}
|
|
|
|
SRV_Channels::set_output_scaled(SRV_Channel::k_winch, scaled_output);
|
|
|
|
}
|
2023-03-02 20:48:38 -04:00
|
|
|
|
2023-10-02 02:02:40 -03:00
|
|
|
// returns the rate which may be modified to unstick the winch
|
|
|
|
// if the winch stops, the rate is temporarily set to zero
|
|
|
|
// now_ms should be set to the current system time
|
|
|
|
// rate should be the rate used to calculate the final PWM output to the winch
|
|
|
|
float AP_Winch_Daiwa::get_stuck_protected_rate(uint32_t now_ms, float rate)
|
|
|
|
{
|
|
|
|
// exit immediately if stuck protection disabled
|
|
|
|
if (!option_enabled(AP_Winch::Options::RetryIfStuck)) {
|
|
|
|
return rate;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for timeout
|
|
|
|
bool timeout = (now_ms - stuck_protection.last_update_ms) > 1000;
|
|
|
|
stuck_protection.last_update_ms = now_ms;
|
|
|
|
|
|
|
|
// check if winch is nearly fully pulled in
|
|
|
|
const bool near_thread_start = (latest.line_length < AP_WINCH_DAIWA_STUCK_LENGTH_MIN) && is_negative(rate);
|
|
|
|
|
|
|
|
// check if rate is near zero (winch may not move with very low desired rates)
|
|
|
|
const bool rate_near_zero = fabsf(rate) < AP_WINCH_DAIWA_STUCK_RATE_MIN;
|
|
|
|
|
|
|
|
// return rate unchanged if this protection has not been called recently or winch is unhealthy
|
|
|
|
// or if winch is moving, desired rate is near zero or winch has stopped at thread start or thread end
|
|
|
|
if (timeout || !healthy() || latest.moving || rate_near_zero || near_thread_start || latest.thread_end) {
|
|
|
|
// notify user when winch becomes unstuck
|
|
|
|
if (option_enabled(AP_Winch::Options::VerboseOutput) && (stuck_protection.stuck_start_ms != 0) && (stuck_protection.user_notified)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s unstuck", send_text_prefix);
|
|
|
|
}
|
|
|
|
// reset stuck protection state
|
|
|
|
stuck_protection.stuck_start_ms = 0;
|
|
|
|
return rate;
|
|
|
|
}
|
|
|
|
|
|
|
|
// winch is healthy, with non-zero requested rate but not moving
|
|
|
|
// record when winch became stuck
|
|
|
|
if (stuck_protection.stuck_start_ms == 0) {
|
|
|
|
stuck_protection.stuck_start_ms = now_ms;
|
|
|
|
stuck_protection.user_notified = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if stuck for between 1 to 2 seconds return zero rate
|
|
|
|
const uint32_t stuck_time_ms = (now_ms - stuck_protection.stuck_start_ms);
|
|
|
|
if (stuck_time_ms > AP_WINCH_DAIWA_STUCK_TIMEOUT_MS) {
|
|
|
|
// notify user
|
|
|
|
if (!stuck_protection.user_notified) {
|
|
|
|
stuck_protection.user_notified = true;
|
|
|
|
if (option_enabled(AP_Winch::Options::VerboseOutput)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_WARNING, "%s stuck", send_text_prefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return zero rate for 1 second
|
|
|
|
if (stuck_time_ms <= (AP_WINCH_DAIWA_STUCK_TIMEOUT_MS+AP_WINCH_DAIWA_STUCK_CENTER_MS)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// rate has been set to zero for 1 sec so release and restart stuck detection
|
|
|
|
stuck_protection.stuck_start_ms = 0;
|
|
|
|
|
|
|
|
// rate used for acceleration limiting also reset to zero
|
|
|
|
set_previous_rate(0.0f);
|
|
|
|
|
|
|
|
// update user
|
|
|
|
if (option_enabled(AP_Winch::Options::VerboseOutput)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s retrying", send_text_prefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// give winch more time to start moving
|
|
|
|
return rate;
|
|
|
|
}
|
|
|
|
|
2023-09-25 04:35:54 -03:00
|
|
|
// update user with changes to winch state via send text messages
|
|
|
|
void AP_Winch_Daiwa::update_user()
|
|
|
|
{
|
|
|
|
// exit immediately if verbose output disabled
|
|
|
|
if (!option_enabled(AP_Winch::Options::VerboseOutput)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// send updates at no more than 2hz
|
|
|
|
uint32_t now_ms = AP_HAL::millis();
|
|
|
|
if (now_ms - user_update.last_ms < 500) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bool update_sent = false;
|
|
|
|
|
|
|
|
// health change
|
|
|
|
const bool now_healthy = healthy();
|
|
|
|
if (user_update.healthy != now_healthy) {
|
|
|
|
user_update.healthy = now_healthy;
|
2023-10-09 03:29:39 -03:00
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s %shealthy", send_text_prefix, now_healthy ? "" : "not ");
|
2023-09-25 04:35:54 -03:00
|
|
|
update_sent = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// thread end
|
|
|
|
if (latest.thread_end && (user_update.thread_end != latest.thread_end)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s thread end", send_text_prefix);
|
|
|
|
update_sent = true;
|
|
|
|
}
|
|
|
|
user_update.thread_end = latest.thread_end;
|
|
|
|
|
|
|
|
// moving state
|
|
|
|
if (user_update.moving != latest.moving) {
|
|
|
|
// 0:stopped, 1:retracting line, 2:extending line, 3:clutch engaged, 4:zero reset
|
|
|
|
static const char* moving_str[] = {"stopped", "raising", "lowering", "free spinning", "zero reset"};
|
|
|
|
if (latest.moving < ARRAY_SIZE(moving_str)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s %s", send_text_prefix, moving_str[latest.moving]);
|
|
|
|
} else {
|
2023-10-08 21:14:24 -03:00
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s move state unknown", send_text_prefix);
|
2023-09-25 04:35:54 -03:00
|
|
|
}
|
|
|
|
update_sent = true;
|
|
|
|
}
|
|
|
|
user_update.moving = latest.moving;
|
|
|
|
|
|
|
|
// clutch state
|
|
|
|
if (user_update.clutch != latest.clutch) {
|
|
|
|
// 0:clutch off, 1:clutch engaged weakly, 2:clutch engaged strongly, motor can spin freely
|
|
|
|
static const char* clutch_str[] = {"off", "weak", "strong (free)"};
|
|
|
|
if (user_update.clutch < ARRAY_SIZE(clutch_str)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s clutch %s", send_text_prefix, clutch_str[latest.moving]);
|
|
|
|
} else {
|
2023-10-08 21:14:24 -03:00
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s clutch state unknown", send_text_prefix);
|
2023-09-25 04:35:54 -03:00
|
|
|
}
|
|
|
|
update_sent = true;
|
|
|
|
}
|
|
|
|
user_update.clutch = latest.clutch;
|
|
|
|
|
|
|
|
// length in meters
|
|
|
|
const float latest_line_length_rounded = roundf(latest.line_length);
|
|
|
|
if (!is_equal(user_update.line_length, latest_line_length_rounded)) {
|
|
|
|
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "%s %dm", send_text_prefix, (int)latest_line_length_rounded);
|
|
|
|
update_sent = true;
|
|
|
|
}
|
|
|
|
user_update.line_length = latest_line_length_rounded;
|
|
|
|
|
|
|
|
// record time message last sent to user
|
|
|
|
if (update_sent) {
|
|
|
|
user_update.last_ms = now_ms;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-02 20:48:38 -04:00
|
|
|
#endif // AP_WINCH_DAIWA_ENABLED
|