AP_FETtecOneWire: complete rewrite of the ESC-configuration state machine

Co-authored-by: Dr.-Ing. Amilcar do Carmo Lucas <amilcar.lucas@iav.de>

tidy message sending using templates
Calculate and enforce the minimum update period.
Disable unused features to save flash
forced time gaps between all transmits
correct ESC reset functionality
Avoid re-initialization repeatition
Make sure we stop FETtec if safety is on (ignore reverse) this reduces duplicated code
Error count calculation changed
   as the telemetry error count is absolute only the overflow status can be safed and used for the percentage calculation
Update the README to add autotests information
FETtec needs a time gap between frames
  This allows running at high fast_loop_rates
do not send fast_throttle data if a configuration command just got sent
Example parameter configuration file is for a Quadcopter with ESCs connected to Telem2
remove two FIXME
fix compilation in master
Fix the ESC not re-initializing issue.
  Now we re-init whenever we loose connection
RVMASK parameter changes only take effect when not armed
Improve documentation
Always use the same wording when referring to fast-throttle commands
fix pre-arm check message
assure the length of the memmove is positive
Set HAL_AP_FETTEC_CONFIGURE_ESCS to 0 when no ESC hardware is available and you want to test the UART send function
This commit is contained in:
Peter Barker 2021-08-13 10:48:19 +10:00 committed by Peter Barker
parent e263063600
commit ea7c7f9d19
4 changed files with 1111 additions and 941 deletions

File diff suppressed because it is too large Load Diff

View File

@ -46,10 +46,19 @@
#if HAL_AP_FETTEC_ONEWIRE_ENABLED #if HAL_AP_FETTEC_ONEWIRE_ENABLED
#define FTW_DEBUGGING 0
#if FTW_DEBUGGING
#include <stdio.h>
#define fet_debug(fmt, args ...) do {::fprintf(stderr,"FETtec: %s:%d: " fmt "\n", __FUNCTION__, __LINE__, ## args); } while(0)
#else
#define fet_debug(fmt, args ...)
#endif
#include <AP_ESC_Telem/AP_ESC_Telem.h> #include <AP_ESC_Telem/AP_ESC_Telem.h>
#include <AP_Param/AP_Param.h> #include <AP_Param/AP_Param.h>
#include <AP_Math/AP_Math.h>
#include <AP_Math/crc.h>
class AP_FETtecOneWire : public AP_ESC_Telem_Backend class AP_FETtecOneWire : public AP_ESC_Telem_Backend
{ {
@ -63,14 +72,17 @@ public:
static const struct AP_Param::GroupInfo var_info[]; static const struct AP_Param::GroupInfo var_info[];
bool pre_arm_check(char *failure_msg, const uint8_t failure_msg_len) const;
/// periodically called from SRV_Channels::push()
void update();
static AP_FETtecOneWire *get_singleton() { static AP_FETtecOneWire *get_singleton() {
return _singleton; return _singleton;
} }
/// periodically called from SRV_Channels::push()
void update();
/// called from AP_Arming; should return false if arming should be
/// disallowed
bool pre_arm_check(char *failure_msg, const uint8_t failure_msg_len) const;
#if HAL_AP_FETTEC_ESC_BEEP #if HAL_AP_FETTEC_ESC_BEEP
/** /**
makes all connected ESCs beep makes all connected ESCs beep
@ -93,168 +105,116 @@ private:
static AP_FETtecOneWire *_singleton; static AP_FETtecOneWire *_singleton;
AP_HAL::UARTDriver *_uart; AP_HAL::UARTDriver *_uart;
#if HAL_WITH_ESC_TELEM
static constexpr uint8_t MOTOR_COUNT_MAX = ESC_TELEM_MAX_ESCS; ///< OneWire supports up-to 15 ESCs, but Ardupilot only supports 12
#else
static constexpr uint8_t MOTOR_COUNT_MAX = 12; ///< OneWire supports up-to 15 ESCs, but Ardupilot only supports 12
#endif
AP_Int32 _motor_mask_parameter; AP_Int32 _motor_mask_parameter;
AP_Int32 _reverse_mask_parameter; AP_Int32 _reverse_mask_parameter;
#if HAL_WITH_ESC_TELEM #if HAL_WITH_ESC_TELEM
AP_Int8 _pole_count_parameter; AP_Int8 _pole_count_parameter;
#endif #endif
static constexpr uint8_t FRAME_OVERHEAD = 6; static constexpr uint8_t FRAME_OVERHEAD = 6; ///< OneWire message frame overhead (header+tail bytes)
static constexpr uint8_t MAX_TRANSMIT_LENGTH = 4; static constexpr uint8_t MAX_RECEIVE_LENGTH = 12; ///< OneWire max receive message payload length in bytes
static constexpr uint8_t MAX_RECEIVE_LENGTH = 12; static constexpr uint8_t SERIAL_NUMBER_LENGTH = 12; ///< ESC serial number length in bytes
/** /**
initialize the serial port, scan the OneWire bus, setup the found ESCs initialize the device driver: configure serial port, wake-up and configure ESCs
*/ */
void init(); void init();
/** /**
check if the current configuration is OK initialize the serial port
*/ */
void configuration_check(); void init_uart();
/** /**
transmits a FETtec OneWire frame to an ESC scan the OneWire bus, configure the ESCs requested in the _motor_mask_parameter
@param esc_id id of the ESC
@param bytes 8 bit array of bytes. Where byte 1 contains the command, and all following bytes can be the payload
@param length length of the bytes array
@return false if length is bigger than MAX_TRANSMIT_LENGTH, true on write success
*/ */
bool transmit(const uint8_t esc_id, const uint8_t *bytes, uint8_t length); void configure_escs();
enum class return_type : uint8_t // states configured ESCs can be in:
{ enum class ESCState : uint8_t {
RESPONSE, UNINITIALISED = 5, // when we haven't tried to send anything to the ESC
FULL_FRAME
};
enum class receive_response : uint8_t WANT_SEND_OK_TO_GET_RUNNING_SW_TYPE = 10,
{ WAITING_OK_FOR_RUNNING_SW_TYPE = 11,
NO_ANSWER_YET,
ANSWER_VALID,
CRC_MISSMATCH,
REQ_OVERLENGTH
};
/** WANT_SEND_START_FW = 20,
reads the FETtec OneWire answer frame of an ESC WAITING_OK_FOR_START_FW = 21,
@param bytes 8 bit byte array, where the received answer gets stored in
@param length the expected answer length
@param return_full_frame can be return_type::RESPONSE or return_type::FULL_FRAME
@return receive_response enum
*/
receive_response receive(uint8_t *bytes, uint8_t length, return_type return_full_frame);
uint8_t receive_buf[FRAME_OVERHEAD + MAX_RECEIVE_LENGTH];
uint8_t receive_buf_used;
void move_preamble_in_receive_buffer(uint8_t search_start_pos = 0);
void consume_bytes(uint8_t n);
enum class pull_state : uint8_t {
BUSY,
COMPLETED,
FAILED
};
/**
Pulls a complete request between flight controller and ESC
@param esc_id id of the ESC
@param command 8bit array containing the command that should be send including the possible payload
@param response 8bit array where the response will be stored in
@param return_full_frame can be return_type::RESPONSE or return_type::FULL_FRAME
@param req_len transmit request length
@return pull_state enum
*/
pull_state pull_command(const uint8_t esc_id, const uint8_t *command, uint8_t *response, return_type return_full_frame, const uint8_t req_len);
/**
Scans for all ESCs in bus. Configures fast-throttle and telemetry for the ones found.
Should be periodically called until _scan.state == scan_state_t::DONE
*/
void scan_escs();
/**
configure the fast-throttle command.
Should be called once after scan_escs() is completted and before config_escs()
*/
void config_fast_throttle();
#if HAL_WITH_ESC_TELEM
/**
increment message packet count for every ESC
*/
void inc_sent_msg_count();
/**
calculates tx (outgoing packets) error-rate by converting the CRC error counts reported by the ESCs into percentage
@param esc_id id of ESC, that the error is calculated for
@param esc_error_count the error count given by the esc
@return the error in percent
*/
float calc_tx_crc_error_perc(const uint8_t esc_id, uint16_t esc_error_count);
/**
if init is complete checks if the requested telemetry is available.
@param t telemetry datastructure where the read telemetry will be stored in.
@param centi_erpm 16bit centi-eRPM value returned from the ESC
@param tx_err_count Ardupilot->ESC communication CRC error counter
@param tlm_from_id receives the ID from the ESC that has respond with its telemetry
@return receive_response enum
*/
receive_response decode_single_esc_telemetry(TelemetryData& t, int16_t& centi_erpm, uint16_t& tx_err_count, uint8_t &tlm_from_id);
#endif
/**
if init is complete sends a single fast-throttle frame containing the throttle for all found OneWire ESCs.
@param motor_values a 16bit array containing the throttle values that should be sent to the motors. 0-2000 where 1001-2000 is positive rotation and 0-999 reversed rotation
@param tlm_request the ESC to request telemetry from (-1 for no telemetry, 0 for ESC1, 1 for ESC2, 2 for ESC3, ...)
*/
void escs_set_values(const uint16_t *motor_values, const int8_t tlm_request);
static constexpr uint8_t SERIAL_NR_BITWIDTH = 12;
class FETtecOneWireESC
{
public:
#if HAL_WITH_ESC_TELEM
uint16_t error_count; ///< error counter from the ESCs.
uint16_t error_count_since_overflow; ///< error counter from the ESCs to pass the overflow.
#endif
bool active;
#if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO #if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO
uint8_t firmware_version; WANT_SEND_REQ_TYPE = 30,
uint8_t firmware_sub_version; WAITING_ESC_TYPE = 31,
uint8_t esc_type;
uint8_t serial_number[SERIAL_NR_BITWIDTH]; WANT_SEND_REQ_SW_VER = 40,
#endif WAITING_SW_VER = 41,
} _found_escs[MOTOR_COUNT_MAX]; ///< Zero-indexed array
WANT_SEND_REQ_SN = 50,
WAITING_SN = 51,
#endif
uint32_t _last_config_check_ms;
#if HAL_WITH_ESC_TELEM #if HAL_WITH_ESC_TELEM
float _crc_error_rate_factor; ///< multiply factor. Used to avoid division operations WANT_SEND_SET_TLM_TYPE = 60,
uint16_t _sent_msg_count; ///< number of fast-throttle commands sent by the flight controller WAITING_SET_TLM_TYPE_OK = 61,
uint16_t _update_rate_hz;
#endif #endif
uint16_t _motor_mask;
uint16_t _reverse_mask;
uint8_t _nr_escs_in_bitmask; ///< number of ESCs set on the FTW_MASK parameter
uint8_t _found_escs_count; ///< number of ESCs auto-scanned in the bus by the scan_escs() function
uint8_t _configured_escs; ///< number of ESCs fully configured by the scan_escs() function, might be smaller than _found_escs_count
int8_t _requested_telemetry_from_esc; ///< the ESC to request telemetry from (-1 for no telemetry, 0 for ESC1, 1 for ESC2, 2 for ESC3, ...) WANT_SEND_SET_FAST_COM_LENGTH = 70,
WAITING_SET_FAST_COM_LENGTH_OK = 71,
RUNNING = 100,
};
class ESC {
public:
#if HAL_WITH_ESC_TELEM
uint32_t last_telem_us; ///< last time we got telemetry from this ESC
uint32_t last_reset_us;
uint16_t error_count_at_throttle_count_overflow; ///< overflow counter for error counter from the ESCs.
bool telem_expected; ///< this ESC is fully configured and is now expected to send us telemetry
#endif
uint8_t id; ///< FETtec ESC ID
uint8_t servo_ofs; ///< offset into ArduPilot servo array
bool is_awake;
void set_state(ESCState _state) {
fet_debug("Moving ESC.id=%u from state=%u to state=%u", (unsigned)id, (unsigned)state, (unsigned)_state);
state = _state;
};
ESCState state = ESCState::UNINITIALISED;
#if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO
uint8_t serial_number[SERIAL_NUMBER_LENGTH];
uint8_t firmware_version;
uint8_t firmware_subversion;
uint8_t type;
#endif
};
uint32_t _min_fast_throttle_period_us; ///< minimum allowed fast-throttle command transmit period
uint32_t _last_not_running_warning_ms; ///< last time we warned the user their ESCs are stuffed
int32_t _motor_mask; ///< an un-mutable copy of the _motor_mask_parameter taken before _init_done goes true
int32_t _reverse_mask; ///< a copy of the _reverse_mask_parameter taken while not armed
int32_t _running_mask; ///< a bitmask of the actively running ESCs
uint32_t _last_transmit_us; ///< last time the transmit() function sent data
ESC *_escs;
uint8_t _esc_count; ///< number of allocated ESCs
uint8_t _fast_throttle_byte_count; ///< pre-calculated number of bytes required to send an entire packed throttle message
#if HAL_AP_FETTEC_HALF_DUPLEX #if HAL_AP_FETTEC_HALF_DUPLEX
uint8_t _ignore_own_bytes; ///< bytes to ignore while receiving, because we have transmitted them ourselves uint8_t _ignore_own_bytes; ///< bytes to ignore while receiving, because we have transmitted them ourselves
uint8_t _last_crc; ///< the CRC from the last sent fast-throttle command uint8_t _last_crc; ///< the CRC from the last sent fast-throttle command
bool _use_hdplex; ///< use asynchronous half-duplex serial communication bool _use_hdplex; ///< use asynchronous half-duplex serial communication
#endif #endif
bool _initialised; ///< device driver and ESCs are fully initialized
bool _pull_busy; ///< request-reply transaction is busy
enum class msg_type : uint8_t bool _init_done; ///< device driver is initialized; ESCs may still need to be configured
bool _invalid_mask; ///< true if the mask parameter is invalid
enum class FrameSource : uint8_t {
MASTER = 0x01, ///< master is always 0x01
BOOTLOADER = 0x02,
ESC = 0x03,
};
enum class MsgType : uint8_t
{ {
OK = 0, OK = 0,
BL_PAGE_CORRECT = 1, ///< Bootloader only BL_PAGE_CORRECT = 1, ///< Bootloader only
@ -271,53 +231,270 @@ private:
#endif #endif
SET_FAST_COM_LENGTH = 26, ///< configure fast-throttle command SET_FAST_COM_LENGTH = 26, ///< configure fast-throttle command
SET_TLM_TYPE = 27, ///< telemetry operation mode SET_TLM_TYPE = 27, ///< telemetry operation mode
SIZEOF_RESPONSE_LENGTH, ///< size of the _response_length array used in the pull_command() function, you can move this one around
#if HAL_AP_FETTEC_ESC_LIGHT #if HAL_AP_FETTEC_ESC_LIGHT
SET_LED_TMP_COLOR = 51, ///< msg_type::SET_LED_TMP_COLOR is ignored here. You must update this if you add new msg_type cases SET_LED_TMP_COLOR = 51, ///< set ESC's LED color
#endif #endif
}; };
enum class scan_state_t : uint8_t { /**
WAIT_FOR_BOOT, ///< initial state, wait for a ESC(s) cold-start a frame looks like:
IN_BOOTLOADER, ///< in bootloader? byte 1 = frame header (master is always 0x01)
START_FW, ///< start the firmware byte 2 = target ID (5bit)
WAIT_START_FW, ///< wait for the firmware to start byte 3 & 4 = frame type (always 0x00, 0x00 used for bootloader. here just for compatibility)
byte 5 = frame length over all bytes
byte 6 - X = request type, followed by the payload
byte X+1 = 8bit CRC
*/
template <typename T>
class PACKED PackedMessage {
public:
PackedMessage(uint8_t _esc_id, T _msg) :
esc_id(_esc_id),
msg(_msg)
{
update_checksum();
}
uint8_t frame_source { (uint8_t)FrameSource::MASTER };
uint8_t esc_id;
uint16_t frame_type { 0 }; // bootloader only, always zero
uint8_t frame_length {sizeof(T) + FRAME_OVERHEAD}; // all bytes including frame_source and checksum
T msg;
uint8_t checksum;
void update_checksum() {
checksum = crc8_dvb_update(0, (const uint8_t*)this, frame_length-1);
}
};
class PACKED OK {
public:
uint8_t msgid { (uint8_t)MsgType::OK };
};
class PACKED START_FW {
public:
uint8_t msgid { (uint8_t)MsgType::BL_START_FW };
};
class PACKED SET_FAST_COM_LENGTH {
public:
SET_FAST_COM_LENGTH(uint8_t _byte_count, uint8_t _min_esc_id, uint8_t _esc_count) :
byte_count{_byte_count},
min_esc_id{_min_esc_id},
esc_count{_esc_count}
{ }
uint8_t msgid { (uint8_t)MsgType::SET_FAST_COM_LENGTH };
uint8_t byte_count;
uint8_t min_esc_id;
uint8_t esc_count;
};
#if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO #if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO
ESC_TYPE, ///< ask the ESC type class PACKED REQ_TYPE {
SW_VER, ///< ask the software version public:
SN, ///< ask the serial number uint8_t msgid { (uint8_t)MsgType::REQ_TYPE };
#endif
NEXT_ID, ///< increment ESC ID and jump to IN_BOOTLOADER
CONFIG_FAST_THROTTLE, ///< configure fast-throttle command header
#if HAL_WITH_ESC_TELEM
CONFIG_TLM, ///< configure telemetry mode
#endif
CONFIG_NEXT_ACTIVE_ESC, ///< increment ESC ID and jump to CONFIG_FAST_THROTTLE
DONE ///< configuration done
}; };
/// presistent scan state data (only used inside scan_escs() function) class PACKED REQ_SW_VER {
struct scan_state public:
{ uint8_t msgid { (uint8_t)MsgType::REQ_SW_VER };
uint32_t last_us; ///< last transaction time in microseconds };
uint8_t id; ///< Zero-indexed ID of the used ESC
scan_state_t state; ///< scan state-machine state
uint8_t rx_try_cnt; ///< receive try counter
uint8_t trans_try_cnt; ///< transaction (transmit and response) try counter
} _scan;
/// fast-throttle command configuration class PACKED REQ_SN {
struct fast_throttle_config public:
{ uint8_t msgid { (uint8_t)MsgType::REQ_SN };
uint16_t bits_to_add_left; ///< bits to add after the header };
uint8_t command[4]; ///< fast-throttle command frame header bytes
uint8_t byte_count; ///< nr bytes in a fast throttle command
uint8_t min_id; ///< Zero-indexed ESC ID
uint8_t max_id; ///< Zero-indexed ESC ID
} _fast_throttle;
/// response length lookup table, saves 104 bytes of flash and speeds up the pull_command() function class PACKED ESC_TYPE {
uint8_t _response_length[uint8_t(msg_type::SIZEOF_RESPONSE_LENGTH)]; public:
ESC_TYPE(uint8_t _type) :
type{_type} { }
uint8_t type;
};
class PACKED SW_VER {
public:
SW_VER(uint8_t _version, uint8_t _subversion) :
version{_version},
subversion{_subversion}
{ }
uint8_t version;
uint8_t subversion;
};
class PACKED SN {
public:
SN(uint8_t *_sn, uint8_t snlen) {
memcpy(sn, _sn, ARRAY_SIZE(sn));
}
uint8_t sn[SERIAL_NUMBER_LENGTH];
};
#endif // HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO
void pack_fast_throttle_command(const uint16_t *motor_values, uint8_t *buffer, const uint8_t length, const uint8_t esc_id_to_request_telem_from);
/*
* Messages, methods and states for dealing with ESC telemetry
*/
#if HAL_WITH_ESC_TELEM
void handle_message_telem(ESC &esc);
uint16_t _fast_throttle_cmd_count; ///< number of fast-throttle commands sent by the flight controller
/// the ESC at this offset into _escs should be the next to send a
/// telemetry request for:
uint8_t _esc_ofs_to_request_telem_from;
class PACKED SET_TLM_TYPE {
public:
SET_TLM_TYPE(uint8_t _tlm_type) :
tlm_type{_tlm_type}
{ }
uint8_t msgid { (uint8_t)MsgType::SET_TLM_TYPE };
uint8_t tlm_type;
};
class PACKED TLM {
public:
TLM(int8_t _temp, uint16_t _voltage, uint16_t _current, int16_t _rpm, uint16_t _consumption_mah, uint16_t _tx_err_count) :
temp{_temp},
voltage{_voltage},
current{_current},
rpm{_rpm},
consumption_mah{_consumption_mah},
tx_err_count{_tx_err_count}
{ }
int8_t temp; // centi-degrees
uint16_t voltage; // centi-Volt
uint16_t current; // centi-Ampere (signed?)
int16_t rpm; // centi-rpm
uint16_t consumption_mah; // mili-Ampere.hour
uint16_t tx_err_count; // CRC error count, as perceived from the ESC receiving side
};
#endif // HAL_WITH_ESC_TELEM
#if HAL_AP_FETTEC_ESC_BEEP
class PACKED Beep {
public:
Beep(uint8_t _beep_frequency) :
beep_frequency{_beep_frequency}
{ }
uint8_t msgid { (uint8_t)MsgType::BEEP };
uint8_t beep_frequency;
// add two zeros to make sure all ESCs can catch their command as we don't wait for a response here (don't blame me --pb)
uint16_t spacer = 0;
};
#endif // HAL_AP_FETTEC_ESC_BEEP
#if HAL_AP_FETTEC_ESC_LIGHT
class PACKED LEDColour {
public:
LEDColour(uint8_t _r, uint8_t _g, uint8_t _b) :
r{_r},
g{_g},
b{_b}
{ }
uint8_t msgid { (uint8_t)MsgType::SET_LED_TMP_COLOR };
uint8_t r;
uint8_t g;
uint8_t b;
// add two zeros to make sure all ESCs can catch their command as we don't wait for a response here (don't blame me --pb)
uint16_t spacer = 0;
};
#endif // HAL_AP_FETTEC_ESC_LIGHT
/*
* Methods and data for transmitting data to the ESCSs:
*/
/**
transmits data to ESCs
@param bytes bytes to transmit
@param length number of bytes to transmit
@return false there's no space in the UART for this message
*/
bool transmit(const uint8_t* bytes, const uint8_t length);
template <typename T>
bool transmit(const PackedMessage<T> &msg) {
return transmit((const uint8_t*)&msg, sizeof(msg));
}
/**
transmits configuration request data to ESCs
@param bytes bytes to transmit
@param length number of bytes to transmit
@return false if vehicle armed or there's no space in the UART for this message
*/
bool transmit_config_request(const uint8_t* bytes, const uint8_t length);
template <typename T>
bool transmit_config_request(const PackedMessage<T> &msg) {
return transmit_config_request((const uint8_t*)&msg, sizeof(msg));
}
/**
sends a single fast-throttle frame containing the throttle for all configured OneWire ESCs.
@param motor_values a 16bit array containing the throttle values that should be sent to the motors. 0-2000 where 1001-2000 is positive rotation and 999-0 reversed rotation
*/
void escs_set_values(const uint16_t *motor_values);
/*
* Methods and data for receiving data from the ESCs:
*/
// FIXME: this should be tighter - and probably calculated. Note
// that we can't request telemetry faster than the loop interval,
// which is 20ms on Plane, so that puts a constraint here. When
// using fast-throttle with 12 ESCs on Plane you could expect
// 240ms between telem updates. Why you have a Plane with 12 ESCs
// is a bit of a puzzle.
static const uint32_t max_telem_interval_us = 100000;
void handle_message(ESC &esc, const uint8_t length);
/**
reads data from the UART, calling handle_message on any message found
*/
void read_data_from_uart();
union MessageUnion {
MessageUnion() { }
PackedMessage<OK> packed_ok;
#if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO
PackedMessage<ESC_TYPE> packed_esc_type;
PackedMessage<SW_VER> packed_sw_ver;
PackedMessage<SN> packed_sn;
#endif
#if HAL_WITH_ESC_TELEM
PackedMessage<TLM> packed_tlm;
#endif
uint8_t receive_buf[FRAME_OVERHEAD + MAX_RECEIVE_LENGTH];
} u;
static_assert(sizeof(u.packed_ok) <= sizeof(u.receive_buf),"packed_ok does not fit in receive_buf. MAX_RECEIVE_LENGTH too small?");
#if HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO
static_assert(sizeof(u.packed_esc_type) <= sizeof(u.receive_buf),"packed_esc_type does not fit in receive_buf. MAX_RECEIVE_LENGTH too small?");
static_assert(sizeof(u.packed_sw_ver) <= sizeof(u.receive_buf),"packed_sw_ver does not fit in receive_buf. MAX_RECEIVE_LENGTH too small?");
static_assert(sizeof(u.packed_sn) <= sizeof(u.receive_buf),"packed_sn does not fit in receive_buf. MAX_RECEIVE_LENGTH too small?");
#endif
#if HAL_WITH_ESC_TELEM
static_assert(sizeof(u.packed_tlm) <= sizeof(u.receive_buf),"packed_tlm does not fit in receive_buf. MAX_RECEIVE_LENGTH too small?");
#endif
uint16_t _unknown_esc_message;
uint16_t _message_invalid_in_state_count;
uint16_t _period_too_short;
uint8_t _receive_buf_used;
/// shifts data to start of buffer based on magic header bytes
void move_frame_source_in_receive_buffer(const uint8_t search_start_pos = 0);
/// cut n bytes from start of buffer
void consume_bytes(const uint8_t n);
/// returns true if the first message in the buffer is OK
bool buffer_contains_ok(const uint8_t length);
}; };
#endif // HAL_AP_FETTEC_ONEWIRE_ENABLED #endif // HAL_AP_FETTEC_ONEWIRE_ENABLED

View File

@ -10,46 +10,74 @@ For purchase, connection and configuration information please see the [Ardupilot
- use ArduPilot's coding guidelines and naming conventions - use ArduPilot's coding guidelines and naming conventions
- control motor speed - control motor speed
- copy ESC telemetry data into MAVLink telemetry - Use the AP_ESC_Telem base class to:
- save ESC telemetry data in dataflash logs - copy ESC telemetry data into MAVLink telemetry
- use RPM telemetry for dynamic notch filter frequencies - save ESC telemetry data in dataflash logs
- sum the current telemetry info from all ESCs and use it as virtual battery current monitor sensor - use RPM telemetry for dynamic notch filter frequencies
- average the voltage telemetry info and use it as virtual battery voltage monitor sensor - sum the current telemetry info from all ESCs and use it as virtual battery current monitor sensor
- average the temperature telemetry info and use it as virtual battery temperature monitor sensor - average the voltage telemetry info and use it as virtual battery voltage monitor sensor
- report telemetry communication error rate in the dataflash logs - average the temperature telemetry info and use it as virtual battery temperature monitor sensor
- warn the user if there is a gap in the bitmask parameter. - ~~report telemetry communication error rate in the dataflash logs~~
- Obey the safety switch. Keeps motors from turning
- Use `SERVO_FWT_MASK` to define which servo output should be routed to FETtec ESCs
- Use `SERVO_FWT_RVMASK` to define the rotation direction of the motors
- `SERVO_FWT_RVMASK` changes only take effect when disarmed
- Can be compiled when `HAL_WITH_ESC_TELEM` is disabled. No telemetry data will be available but it saves a lot of memory
- pre-arm checks:
- Check that UART is available
- check that the desired motor channels parameter (`SERVO_FWT_MASK`) is valid
- check that the desired motor poles parameter (`SERVO_FWT_POLES`) is valid
- check that the all desired ESCs are found and configured
- check that the ESCs are periodically sending telemetry data
- re-enumerate all ESCs if not armed (motors not spinning) when - re-enumerate all ESCs if not armed (motors not spinning) when
- there is a gap in their address space IDs
- communication with one of the ESCs is lost - communication with one of the ESCs is lost
- some of the configured ESCs are not found - adds a serial simulator (--uartF=sim:fetteconewireesc) of FETtec OneWire ESCs
- some of the configured ESCs are not correctly configured - adds autotest (using the simulator) to:
- allows the user to configure motor rotation direction per ESC (only gets updated if not armed) - simulate telemetry voltage, current, temperature, RPM data using SITL internal variables
- adds a serial simulator of FETtec OneWire ESCs - test the safety switch functionality
- adds autotest (using the simulator) to fly a copter over a simulated serial link connection - test ESC power outages
- test `SERVO_FWT_MASK` parameter validation
- fly a copter over a simulated serial link connection
## Ardupilot to ESC protocol ## Ardupilot to ESC protocol
The FETtec OneWire protocol supports up to 24 ESCs. As most copters only use at most 12 motors, Ardupilot's implementation supports only 12 to save memory. The FETtec OneWire protocol supports up to 24 ESCs. As most copters only use at most 12 motors, Ardupilot's implementation supports only 12 (`ESC_TELEM_MAX_ESCS`)to save memory.
On this device driver the `SERVO_FTW_MASK` parameter must contain a single contiguous block of bits set, and bit0 (zero-indexed) must be a part of that set. There are two types of messages sent to the ESCs configuration and fast-throttle messages:
The ESC IDs (one-indexed) set on the ESCs must also be a single contiguous block, but it can start at an ID different from 1. The max ID must be lower than 12.
There are two types of messages sent to the ESCs: ### Configuration message frame
### Configuration message
Consists of six frame bytes + payload bytes. Consists of six frame bytes + payload bytes.
``` ```
Byte 0 is the transfer direction (e.g. 0x01 Master to Slave) Byte 0 is the source type
Byte 1 is the ID Byte 1 is the ID
Byte 2 is the frame type (Low byte) Byte 2 is the frame type (Low byte)
Byte 3 is the frame type (High byte) Byte 3 is the frame type (High byte)
Byte 4 is the frame length Byte 4 is the frame length
Byte 5-254 is the payload Byte 5-254 is the payload
Byte 6-255 is CRC (last Byte after the Payload). It uses the same CRC algorithm as Dshot. Byte 6-255 is CRC (last Byte after the Payload). It uses the same CRC algorithm as Dshot.
``` ```
### Fast Throttle Signal #### Check if bootloader or ESC firmware is running
To check which firmware is running on the FETtec an ESCs `PackedMessage<OK>` is sent to each ESC.
If the answer frame `frame_source` is FrameSource::BOOTLOADER we are in bootloader and need to send an `PackedMessage<START_FW>` to start the ESC firmware.
If the answer frame `frame_source` is FrameSource::ESC we are already running the correct firmware and can directly configure the telemetry.
#### Start the ESC firmware
If the ESC is running on bootloader firmware we need to send an `PackedMessage<START_FW>` to start the ESC firmware.
The answer must be a `PackedMessage<OK>` with `frame_source` equal to FrameSource::ESC. If not, we need to repeat the command.
#### Configure Full/Alternative Telemetry
The telemetry can be switched to "per ESC" Mode, where one ESC answers with it's full telemetry as oneWire package including CRC and additionally the CRC Errors counted by the ESC.
To use this mode, `PackedMessage<SET_TLM_TYPE>` is send to each ESC while initializing.
If this was successful the ESC responds with `PackedMessage<OK>`.
#### Configure Fast throttle messages
To configure the fast-throttle frame structure a `PackedMessage<SET_FAST_COM_LENGTH>` is send to each ESC while initializing.
If this was successful the ESC responds with `PackedMessage<OK>`.
### Fast-throttle message frame
``` ```
Byte 0 is the frame header Byte 0 is the frame header
@ -73,11 +101,14 @@ See *ESC to Ardupilot Protocol* section below and comments in `FETtecOneWire.cpp
### Timing ### Timing
Four ESCs need 90uS for the throttle request and telemetry reception. With four ESCs 11kHz are possible. Four ESCs need 90us for the fast-throttle request and telemetry reception. With four ESCs 11kHz update would be possible.
As each additional ESC adds 11 extra fast-throttle command bits, so the rate is lowered by each ESC. Each additional ESC adds 11 extra fast-throttle command bits, so the update rate is lowered by each additional ESC.
If you use 8 ESCs, it needs 160uS including telemetry response, so 5.8kHz are possible. If you use 8 ESCs, it needs 160us including telemetry response, so 5.8kHz update rate would be possible.
The FETtec Ardupilot device driver limits the message transmit period to `_min_fast_throttle_period_us` according to the number of ESCs used.
The update() function has an extra invocation period limit so that even at very high loop rates the the ESCs will still operate correctly albeit doing some decimation.
The current update rate for Copter is 400Hz (~2500us) and for other vehicles is 50Hz (~20000us) so we are bellow device driver limit.
**Note:** You need at least a 4Hz motor signal (max 250ms between messages) before the motors disarm. **Note:** The FETtec ESCs firmware requires at least a 4Hz fast-throttle update rate (max. 250ms between messages) otherwise the FETtec ESC disarm (stop) the motors.
## ESC to Ardupilot protocol ## ESC to Ardupilot protocol
@ -98,26 +129,50 @@ This information is used by Ardupilot to:
- Optionally measure battery voltage and power consumption - Optionally measure battery voltage and power consumption
## Full/Alternative Telemetry ### Full/Alternative Telemetry
The telemetry can be switched to "per ESC" Mode, where one ESC answers with it's full telemetry as oneWire package including CRC and additionally the CRC Errors counted by the ESC. The answer to a fast-throttle command frame is a `PackedMessage<TLM>` message frame, received and decoded in the `FETtecOneWire::handle_message_telem()` function.
To use this mode, `msg_type::SET_TLM_TYPE` is send to each ESC while initializing. The data is forwarded to the `AP_ESC_Telem` class that distributes it to other parts of the ArduPilot code.
If this was successful set the ESC response with `msg_type::OK`.
The answer is packed inside a OneWire package, that can be received with the `FETtecOneWire::receive()` function, that also checks the CRC.
As the packages are send in an uInt8_t array the values must be restored like as only temp is one byte long:
```C++
Telemetry[0]= telem[0]; // Temperature [°C/10]
Telemetry[1]=(telem[1]<<8)|telem[2]; // Voltage [V/10]
Telemetry[2]=(telem[3]<<8)|telem[4]; // Current [A/10]
Telemetry[3]=(telem[5]<<8)|telem[6]; // ERPM/100 (must be divided by number of motor poles to translate to propeller RPM)
Telemetry[4]=(telem[7]<<8)|telem[8]; // Consumption [mA.h]
Telemetry[5]=(telem[9]<<8)|telem[10];// CRC error (ArduPilot->ESC) counter
```
## Function structure
There are two public top level functions `update()` and `pre_arm_check()`. And these two call all other private internal functions:
update()
init()
init_uart()
read_data_from_uart()
move_frame_source_in_receive_buffer()
consume_bytes()
consume_bytes()
handle_message() <-- RX state machine
buffer_contains_ok()
handle_message_telem()
configure_escs() <-- TX state machine
transmit_config_request()
transmit()
escs_set_values()
pack_fast_throttle_command()
transmit()
pre_arm_check()
## Device driver parameters
The `SERVO_FTW_MASK` parameter selects which servo outputs, if any, will be routed to FETtec ESCs.
You need to reboot after changing this parameter.
When `HAL_WITH_ESC_TELEM` is active (default) only the first 12 (`ESC_TELEM_MAX_ESCS`) can be used.
The FETtec ESC IDs set inside the FETtec firmware by the FETtec configuration tool are one-indexed.
These IDs must start at ID 1 and be in a single contiguous block.
You do not need to change these FETtec IDs if you change the servo output assignments inside ArduPilot the using the `SERVO_FTW_MASK` parameter
The `SERVO_FTW_RVMASK` parameter selects which servo outputs, if any, will reverse their rotation.
This parameter is only visible if the `SERVO_FTW_MASK` parameter has at least one bit set.
This parameter effects the outputs immediately when changed and the motors are not armed.
The `SERVO_FTW_POLES` parameter selects Number of motor electrical poles.
It is used to calculate the motors RPM
This parameter effects the RPM calculation immediately when changed.
## Extra features ## Extra features
@ -130,7 +185,7 @@ To control this you must activate the code in the header file:
#endif #endif
``` ```
Or just set the `HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO` macro in the compiler toolchain. Or just set the `HAL_AP_FETTEC_ONEWIRE_GET_STATIC_INFO` macro in the compiler toolchain.
After that you will be able to access this information in the `_found_escs[]` datastructure. After that you will be able to access this information in the `_escs[]` datastructure.
### Beep ### Beep
To control this you must activate the code in the header file: To control this you must activate the code in the header file:

View File

@ -0,0 +1,31 @@
# Example parameter configuration file is for a Quadcopter with ESCs connected to Telem2
# FETtec OneWire ESCs are connected on Telem2 port
SERIAL2_PROTOCOL 38
SERIAL2_OPTIONS 0
# Use the full digital range supported by the ESCs
MOT_PWM_MAX 2000
MOT_PWM_MIN 1000
# Make sure the correct functions get routed out
SERVO1_FUNCTION 33
SERVO2_FUNCTION 34
SERVO3_FUNCTION 35
SERVO4_FUNCTION 36
# Activate FETtec OneWire support
SERVO_FTW_MASK 15
SERVO_FTW_RVMASK 0
SERVO_FTW_POLES 14
# Use RPM telemetry data to dynamically configure one notch filter per motor
INS_HNTCH_ENABLE 1
INS_HNTCH_MODE 3
INS_HNTCH_OPTS 2
INS_HNTCH_REF 1
INS_HNTCH_FREQ 60
INS_HNTCH_BW 30
# Use ESC telemetry data as a "virtual" battery monitor
BATT2_MONITOR 9