2020-02-10 00:13:22 -04:00
/*
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "AP_ESC_Telem.h"
# include <AP_HAL/AP_HAL.h>
2021-02-23 18:04:06 -04:00
# include <GCS_MAVLink/GCS.h>
# include <AP_Logger/AP_Logger.h>
2020-02-10 00:13:22 -04:00
2021-02-23 18:04:06 -04:00
# if HAL_WITH_ESC_TELEM
# include <AP_BoardConfig/AP_BoardConfig.h>
2023-01-19 13:09:29 -04:00
# include <AP_TemperatureSensor/AP_TemperatureSensor_config.h>
2021-02-23 18:04:06 -04:00
2024-01-10 01:11:28 -04:00
# include <AP_Math/AP_Math.h>
2021-02-23 18:04:06 -04:00
//#define ESC_TELEM_DEBUG
2020-02-10 00:13:22 -04:00
2023-05-25 16:17:35 -03:00
# define ESC_RPM_CHECK_TIMEOUT_US 210000UL // timeout for motor running validity
2022-05-25 08:26:02 -03:00
2020-02-10 00:13:22 -04:00
extern const AP_HAL : : HAL & hal ;
2022-06-06 01:40:25 -03:00
// table of user settable parameters
const AP_Param : : GroupInfo AP_ESC_Telem : : var_info [ ] = {
// @Param: _MAV_OFS
// @DisplayName: ESC Telemetry mavlink offset
// @Description: Offset to apply to ESC numbers when reporting as ESC_TELEMETRY packets over MAVLink. This allows high numbered motors to be displayed as low numbered ESCs for convenience on GCS displays. A value of 4 would send ESC on output 5 as ESC number 1 in ESC_TELEMETRY packets
// @Increment: 1
// @Range: 0 31
// @User: Standard
AP_GROUPINFO ( " _MAV_OFS " , 1 , AP_ESC_Telem , mavlink_offset , 0 ) ,
AP_GROUPEND
} ;
2020-02-10 00:13:22 -04:00
AP_ESC_Telem : : AP_ESC_Telem ( )
{
if ( _singleton ) {
AP_HAL : : panic ( " Too many AP_ESC_Telem instances " ) ;
}
_singleton = this ;
2023-10-03 16:25:00 -03:00
# if !defined(IOMCU_FW)
2022-06-06 01:40:25 -03:00
AP_Param : : setup_object_defaults ( this , var_info ) ;
2023-10-03 16:25:00 -03:00
# endif
2020-02-10 00:13:22 -04:00
}
2021-07-05 19:31:40 -03:00
// return the average motor RPM
float AP_ESC_Telem : : get_average_motor_rpm ( uint32_t servo_channel_mask ) const
2021-02-23 18:04:06 -04:00
{
2021-07-05 19:31:40 -03:00
float rpm_avg = 0.0f ;
2021-02-23 18:04:06 -04:00
uint8_t valid_escs = 0 ;
2021-07-05 19:31:40 -03:00
// average the rpm of each motor
2021-02-23 18:04:06 -04:00
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
2021-06-03 04:27:27 -03:00
if ( BIT_IS_SET ( servo_channel_mask , i ) ) {
float rpm ;
if ( get_rpm ( i , rpm ) ) {
2021-07-05 19:31:40 -03:00
rpm_avg + = rpm ;
2021-06-03 04:27:27 -03:00
valid_escs + + ;
}
2021-02-23 18:04:06 -04:00
}
}
2021-07-05 19:31:40 -03:00
2021-02-23 18:04:06 -04:00
if ( valid_escs > 0 ) {
2021-07-05 19:31:40 -03:00
rpm_avg / = valid_escs ;
2021-02-23 18:04:06 -04:00
}
2021-07-05 19:31:40 -03:00
return rpm_avg ;
2021-06-03 04:27:27 -03:00
}
2021-02-23 18:04:06 -04:00
// return all the motor frequencies in Hz for dynamic filtering
uint8_t AP_ESC_Telem : : get_motor_frequencies_hz ( uint8_t nfreqs , float * freqs ) const
{
uint8_t valid_escs = 0 ;
// average the rpm of each motor as reported by BLHeli and convert to Hz
2022-08-20 07:06:58 -03:00
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS & & valid_escs < nfreqs ; i + + ) {
2021-02-23 18:04:06 -04:00
float rpm ;
if ( get_rpm ( i , rpm ) ) {
2022-08-20 07:06:58 -03:00
freqs [ valid_escs + + ] = rpm * ( 1.0f / 60.0f ) ;
2023-07-21 14:07:13 -03:00
} else if ( was_rpm_data_ever_reported ( _rpm_data [ i ] ) ) {
2022-08-20 07:06:58 -03:00
// if we have ever received data on an ESC, mark it as valid but with no data
// this prevents large frequency shifts when ESCs disappear
freqs [ valid_escs + + ] = 0.0f ;
2021-02-23 18:04:06 -04:00
}
}
return MIN ( valid_escs , nfreqs ) ;
}
2022-09-06 11:55:43 -03:00
// get mask of ESCs that sent valid telemetry and/or rpm data in the last
// ESC_TELEM_DATA_TIMEOUT_MS/ESC_RPM_DATA_TIMEOUT_US
2022-05-15 18:30:16 -03:00
uint32_t AP_ESC_Telem : : get_active_esc_mask ( ) const {
uint32_t ret = 0 ;
2021-06-22 14:26:49 -03:00
const uint32_t now = AP_HAL : : millis ( ) ;
2022-09-06 11:55:43 -03:00
uint32_t now_us = AP_HAL : : micros ( ) ;
2021-02-23 18:04:06 -04:00
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
2023-07-21 14:07:13 -03:00
if ( _telem_data [ i ] . last_update_ms = = 0 & & ! was_rpm_data_ever_reported ( _rpm_data [ i ] ) ) {
// have never seen telem from this ESC
2021-06-22 14:26:49 -03:00
continue ;
2021-02-23 18:04:06 -04:00
}
2023-10-25 03:48:48 -03:00
if ( _telem_data [ i ] . stale ( now )
2023-08-16 15:29:32 -03:00
& & ! rpm_data_within_timeout ( _rpm_data [ i ] , now_us , ESC_RPM_DATA_TIMEOUT_US ) ) {
2021-06-22 14:26:49 -03:00
continue ;
}
ret | = ( 1U < < i ) ;
2021-02-23 18:04:06 -04:00
}
2021-06-22 14:26:49 -03:00
return ret ;
}
2024-05-22 13:20:05 -03:00
// return an active ESC for the purposes of reporting (e.g. in the OSD)
uint8_t AP_ESC_Telem : : get_max_rpm_esc ( ) const
{
uint32_t ret = 0 ;
float max_rpm = 0 ;
const uint32_t now = AP_HAL : : millis ( ) ;
const uint32_t now_us = AP_HAL : : micros ( ) ;
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
if ( _telem_data [ i ] . last_update_ms = = 0 & & ! was_rpm_data_ever_reported ( _rpm_data [ i ] ) ) {
// have never seen telem from this ESC
continue ;
}
if ( _telem_data [ i ] . stale ( now )
& & ! rpm_data_within_timeout ( _rpm_data [ i ] , now_us , ESC_RPM_DATA_TIMEOUT_US ) ) {
continue ;
}
if ( _rpm_data [ i ] . rpm > max_rpm ) {
max_rpm = _rpm_data [ i ] . rpm ;
ret = i ;
}
}
return ret ;
}
2021-06-22 14:26:49 -03:00
// return number of active ESCs present
uint8_t AP_ESC_Telem : : get_num_active_escs ( ) const {
2022-05-15 18:30:16 -03:00
uint32_t active = get_active_esc_mask ( ) ;
2021-06-22 14:26:49 -03:00
return __builtin_popcount ( active ) ;
2021-02-23 18:04:06 -04:00
}
2022-05-25 08:26:02 -03:00
// return the whether all the motors in servo_channel_mask are running
2023-04-28 17:02:55 -03:00
bool AP_ESC_Telem : : are_motors_running ( uint32_t servo_channel_mask , float min_rpm , float max_rpm ) const
2022-05-25 08:26:02 -03:00
{
const uint32_t now = AP_HAL : : micros ( ) ;
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
if ( BIT_IS_SET ( servo_channel_mask , i ) ) {
const volatile AP_ESC_Telem_Backend : : RpmData & rpmdata = _rpm_data [ i ] ;
// we choose a relatively strict measure of health so that failsafe actions can rely on the results
2023-07-21 14:07:13 -03:00
if ( ! rpm_data_within_timeout ( rpmdata , now , ESC_RPM_CHECK_TIMEOUT_US ) ) {
2022-05-25 08:26:02 -03:00
return false ;
}
if ( rpmdata . rpm < min_rpm ) {
return false ;
}
2023-04-28 17:02:55 -03:00
if ( ( max_rpm > 0 ) & & ( rpmdata . rpm > max_rpm ) ) {
return false ;
}
2022-05-25 08:26:02 -03:00
}
}
return true ;
}
// is telemetry active for the provided channel mask
bool AP_ESC_Telem : : is_telemetry_active ( uint32_t servo_channel_mask ) const
{
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
if ( BIT_IS_SET ( servo_channel_mask , i ) ) {
// no data received
2023-07-21 14:07:13 -03:00
if ( get_last_telem_data_ms ( i ) = = 0 & & ! was_rpm_data_ever_reported ( _rpm_data [ i ] ) ) {
2022-05-25 08:26:02 -03:00
return false ;
}
}
}
return true ;
}
2021-02-23 18:04:06 -04:00
// get an individual ESC's slewed rpm if available, returns true on success
bool AP_ESC_Telem : : get_rpm ( uint8_t esc_index , float & rpm ) const
{
2021-08-10 19:14:19 -03:00
if ( esc_index > = ESC_TELEM_MAX_ESCS ) {
return false ;
}
2021-02-23 18:04:06 -04:00
const volatile AP_ESC_Telem_Backend : : RpmData & rpmdata = _rpm_data [ esc_index ] ;
2021-08-10 19:14:19 -03:00
if ( is_zero ( rpmdata . update_rate_hz ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
const uint32_t now = AP_HAL : : micros ( ) ;
2023-07-21 14:07:13 -03:00
if ( rpm_data_within_timeout ( rpmdata , now , ESC_RPM_DATA_TIMEOUT_US ) ) {
2021-02-23 18:04:06 -04:00
const float slew = MIN ( 1.0f , ( now - rpmdata . last_update_us ) * rpmdata . update_rate_hz * ( 1.0f / 1e6 f ) ) ;
rpm = ( rpmdata . prev_rpm + ( rpmdata . rpm - rpmdata . prev_rpm ) * slew ) ;
2022-09-01 05:02:33 -03:00
# if AP_SCRIPTING_ENABLED
if ( ( 1U < < esc_index ) & rpm_scale_mask ) {
rpm * = rpm_scale_factor [ esc_index ] ;
}
# endif
2021-02-23 18:04:06 -04:00
return true ;
}
return false ;
}
// get an individual ESC's raw rpm if available, returns true on success
bool AP_ESC_Telem : : get_raw_rpm ( uint8_t esc_index , float & rpm ) const
{
2021-08-10 19:14:19 -03:00
if ( esc_index > = ESC_TELEM_MAX_ESCS ) {
return false ;
}
2021-02-23 18:04:06 -04:00
const volatile AP_ESC_Telem_Backend : : RpmData & rpmdata = _rpm_data [ esc_index ] ;
const uint32_t now = AP_HAL : : micros ( ) ;
2023-07-21 14:07:13 -03:00
if ( ! rpm_data_within_timeout ( rpmdata , now , ESC_RPM_DATA_TIMEOUT_US ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
rpm = rpmdata . rpm ;
return true ;
}
// get an individual ESC's temperature in centi-degrees if available, returns true on success
bool AP_ESC_Telem : : get_temperature ( uint8_t esc_index , int16_t & temp ) const
{
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-02-23 18:04:06 -04:00
if ( esc_index > = ESC_TELEM_MAX_ESCS
2024-02-20 08:45:41 -04:00
| | telemdata . stale ( )
| | ! ( telemdata . types & ( AP_ESC_Telem_Backend : : TelemetryType : : TEMPERATURE | AP_ESC_Telem_Backend : : TelemetryType : : TEMPERATURE_EXTERNAL ) ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
2024-02-20 08:45:41 -04:00
temp = telemdata . temperature_cdeg ;
2021-02-23 18:04:06 -04:00
return true ;
}
// get an individual motor's temperature in centi-degrees if available, returns true on success
bool AP_ESC_Telem : : get_motor_temperature ( uint8_t esc_index , int16_t & temp ) const
{
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-02-23 18:04:06 -04:00
if ( esc_index > = ESC_TELEM_MAX_ESCS
2024-02-20 08:45:41 -04:00
| | telemdata . stale ( )
| | ! ( telemdata . types & ( AP_ESC_Telem_Backend : : TelemetryType : : MOTOR_TEMPERATURE | AP_ESC_Telem_Backend : : TelemetryType : : MOTOR_TEMPERATURE_EXTERNAL ) ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
2024-02-20 08:45:41 -04:00
temp = telemdata . motor_temp_cdeg ;
2021-02-23 18:04:06 -04:00
return true ;
}
2021-07-05 19:37:01 -03:00
// get the highest ESC temperature in centi-degrees if available, returns true if there is valid data for at least one ESC
2024-08-06 15:29:41 -03:00
bool AP_ESC_Telem : : get_highest_temperature ( int16_t & temp ) const
2021-07-05 19:37:01 -03:00
{
uint8_t valid_escs = 0 ;
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
int16_t temp_temp ;
2024-08-06 15:29:41 -03:00
if ( get_temperature ( i , temp_temp ) ) {
2021-07-05 19:37:01 -03:00
temp = MAX ( temp , temp_temp ) ;
valid_escs + + ;
}
}
return valid_escs > 0 ;
}
2021-02-23 18:04:06 -04:00
// get an individual ESC's current in Ampere if available, returns true on success
bool AP_ESC_Telem : : get_current ( uint8_t esc_index , float & amps ) const
{
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-02-23 18:04:06 -04:00
if ( esc_index > = ESC_TELEM_MAX_ESCS
2024-02-20 08:45:41 -04:00
| | telemdata . stale ( )
| | ! ( telemdata . types & AP_ESC_Telem_Backend : : TelemetryType : : CURRENT ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
2024-02-20 08:45:41 -04:00
amps = telemdata . current ;
2021-02-23 18:04:06 -04:00
return true ;
}
// get an individual ESC's voltage in Volt if available, returns true on success
bool AP_ESC_Telem : : get_voltage ( uint8_t esc_index , float & volts ) const
{
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-02-23 18:04:06 -04:00
if ( esc_index > = ESC_TELEM_MAX_ESCS
2024-02-20 08:45:41 -04:00
| | telemdata . stale ( )
| | ! ( telemdata . types & AP_ESC_Telem_Backend : : TelemetryType : : VOLTAGE ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
2024-02-20 08:45:41 -04:00
volts = telemdata . voltage ;
2021-02-23 18:04:06 -04:00
return true ;
}
// get an individual ESC's energy consumption in milli-Ampere.hour if available, returns true on success
bool AP_ESC_Telem : : get_consumption_mah ( uint8_t esc_index , float & consumption_mah ) const
{
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-02-23 18:04:06 -04:00
if ( esc_index > = ESC_TELEM_MAX_ESCS
2024-02-20 08:45:41 -04:00
| | telemdata . stale ( )
| | ! ( telemdata . types & AP_ESC_Telem_Backend : : TelemetryType : : CONSUMPTION ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
2024-02-20 08:45:41 -04:00
consumption_mah = telemdata . consumption_mah ;
2021-02-23 18:04:06 -04:00
return true ;
}
2020-02-10 00:13:22 -04:00
// get an individual ESC's usage time in seconds if available, returns true on success
2021-02-23 18:04:06 -04:00
bool AP_ESC_Telem : : get_usage_seconds ( uint8_t esc_index , uint32_t & usage_s ) const
{
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-02-23 18:04:06 -04:00
if ( esc_index > = ESC_TELEM_MAX_ESCS
2024-02-20 08:45:41 -04:00
| | telemdata . stale ( )
| | ! ( telemdata . types & AP_ESC_Telem_Backend : : TelemetryType : : USAGE ) ) {
2021-02-23 18:04:06 -04:00
return false ;
}
2024-02-20 08:45:41 -04:00
usage_s = telemdata . usage_s ;
2021-02-23 18:04:06 -04:00
return true ;
}
// send ESC telemetry messages over MAVLink
void AP_ESC_Telem : : send_esc_telemetry_mavlink ( uint8_t mav_chan )
{
2021-12-06 22:46:49 -04:00
# if HAL_GCS_ENABLED
2021-06-06 22:02:35 -03:00
if ( ! _have_data ) {
// we've never had any data
return ;
}
2023-07-21 14:07:13 -03:00
const uint32_t now = AP_HAL : : millis ( ) ;
const uint32_t now_us = AP_HAL : : micros ( ) ;
2022-06-06 01:11:14 -03:00
// loop through groups of 4 ESCs
2023-09-30 16:27:55 -03:00
const uint8_t esc_offset = constrain_int16 ( mavlink_offset , 0 , ESC_TELEM_MAX_ESCS - 1 ) ;
// ensure we send out partially-full groups:
const uint8_t num_idx = ( ESC_TELEM_MAX_ESCS + 3 ) / 4 ;
2022-06-06 01:11:14 -03:00
for ( uint8_t idx = 0 ; idx < num_idx ; idx + + ) {
const uint8_t i = ( next_idx + idx ) % num_idx ;
2021-02-23 18:04:06 -04:00
// return if no space in output buffer to send mavlink messages
if ( ! HAVE_PAYLOAD_SPACE ( ( mavlink_channel_t ) mav_chan , ESC_TELEMETRY_1_TO_4 ) ) {
2022-06-06 01:11:14 -03:00
// not enough mavlink buffer space, start at this index next time
next_idx = i ;
2021-02-23 18:04:06 -04:00
return ;
}
2022-06-06 01:40:25 -03:00
bool all_stale = true ;
for ( uint8_t j = 0 ; j < 4 ; j + + ) {
const uint8_t esc_id = ( i * 4 + j ) + esc_offset ;
if ( esc_id < ESC_TELEM_MAX_ESCS & &
2023-10-25 03:48:48 -03:00
( ! _telem_data [ esc_id ] . stale ( now ) | |
2023-07-21 14:07:13 -03:00
rpm_data_within_timeout ( _rpm_data [ esc_id ] , now_us , ESC_RPM_DATA_TIMEOUT_US ) ) ) {
2022-06-06 01:40:25 -03:00
all_stale = false ;
break ;
}
}
if ( all_stale ) {
// skip this group of ESCs if no data to send
2021-02-23 18:04:06 -04:00
continue ;
}
2022-06-06 01:11:14 -03:00
2021-02-23 18:04:06 -04:00
// arrays to hold output
2022-06-10 04:16:45 -03:00
mavlink_esc_telemetry_1_to_4_t s { } ;
2021-02-23 18:04:06 -04:00
// fill in output arrays
for ( uint8_t j = 0 ; j < 4 ; j + + ) {
2022-06-06 01:40:25 -03:00
const uint8_t esc_id = ( i * 4 + j ) + esc_offset ;
if ( esc_id > = ESC_TELEM_MAX_ESCS ) {
continue ;
}
2024-02-20 08:45:41 -04:00
volatile AP_ESC_Telem_Backend : : TelemetryData const & telemdata = _telem_data [ esc_id ] ;
2022-06-06 01:40:25 -03:00
2024-02-20 08:45:41 -04:00
s . temperature [ j ] = telemdata . temperature_cdeg / 100 ;
s . voltage [ j ] = constrain_float ( telemdata . voltage * 100.0f , 0 , UINT16_MAX ) ;
s . current [ j ] = constrain_float ( telemdata . current * 100.0f , 0 , UINT16_MAX ) ;
s . totalcurrent [ j ] = constrain_float ( telemdata . consumption_mah , 0 , UINT16_MAX ) ;
2024-02-20 08:27:09 -04:00
float rpmf ;
2021-02-23 18:04:06 -04:00
if ( get_rpm ( esc_id , rpmf ) ) {
2022-06-10 04:16:45 -03:00
s . rpm [ j ] = constrain_float ( rpmf , 0 , UINT16_MAX ) ;
2020-02-10 00:13:22 -04:00
}
2024-02-20 08:45:41 -04:00
s . count [ j ] = telemdata . count ;
2020-02-10 00:13:22 -04:00
}
2021-02-23 18:04:06 -04:00
2022-06-10 04:16:45 -03:00
// make sure a msg hasn't been extended
static_assert ( MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_5_TO_8_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_9_TO_12_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_13_TO_16_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_17_TO_20_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_21_TO_24_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_21_TO_24_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_25_TO_28_LEN & &
MAVLINK_MSG_ID_ESC_TELEMETRY_1_TO_4_LEN = = MAVLINK_MSG_ID_ESC_TELEMETRY_29_TO_32_LEN ,
" telem messages not compatible " ) ;
const mavlink_channel_t chan = ( mavlink_channel_t ) mav_chan ;
2021-02-23 18:04:06 -04:00
// send messages
switch ( i ) {
case 0 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_1_to_4_send_struct ( chan , & s ) ;
2021-02-23 18:04:06 -04:00
break ;
case 1 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_5_to_8_send_struct ( chan , ( const mavlink_esc_telemetry_5_to_8_t * ) & s ) ;
2021-02-23 18:04:06 -04:00
break ;
case 2 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_9_to_12_send_struct ( chan , ( const mavlink_esc_telemetry_9_to_12_t * ) & s ) ;
2022-06-06 01:11:14 -03:00
break ;
case 3 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_13_to_16_send_struct ( chan , ( const mavlink_esc_telemetry_13_to_16_t * ) & s ) ;
2021-02-23 18:04:06 -04:00
break ;
2022-06-06 01:11:14 -03:00
# if ESC_TELEM_MAX_ESCS > 16
case 4 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_17_to_20_send_struct ( chan , ( const mavlink_esc_telemetry_17_to_20_t * ) & s ) ;
2021-02-23 18:04:06 -04:00
break ;
2022-06-06 01:11:14 -03:00
case 5 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_21_to_24_send_struct ( chan , ( const mavlink_esc_telemetry_21_to_24_t * ) & s ) ;
2022-06-06 01:11:14 -03:00
break ;
case 6 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_25_to_28_send_struct ( chan , ( const mavlink_esc_telemetry_25_to_28_t * ) & s ) ;
2022-06-06 01:11:14 -03:00
break ;
case 7 :
2022-06-10 04:16:45 -03:00
mavlink_msg_esc_telemetry_29_to_32_send_struct ( chan , ( const mavlink_esc_telemetry_29_to_32_t * ) & s ) ;
2022-06-06 01:11:14 -03:00
break ;
# endif
}
2021-02-23 18:04:06 -04:00
}
2022-06-06 01:11:14 -03:00
// we checked for all sends without running out of buffer space,
// start at zero next time
next_idx = 0 ;
2021-12-06 22:46:49 -04:00
# endif // HAL_GCS_ENABLED
2021-02-23 18:04:06 -04:00
}
// record an update to the telemetry data together with timestamp
// this should be called by backends when new telemetry values are available
void AP_ESC_Telem : : update_telem_data ( const uint8_t esc_index , const AP_ESC_Telem_Backend : : TelemetryData & new_data , const uint16_t data_mask )
{
// rpm and telemetry data are not protected by a semaphore even though updated from different threads
// all data is per-ESC and only written from the update thread and read by the user thread
// each element is a primitive type and the timestamp is only updated at the end, thus a caller
// can only get slightly more up-to-date information that perhaps they were expecting or might
// read data that has just gone stale - both of these are safe and avoid the overhead of locking
2023-10-03 16:25:00 -03:00
if ( esc_index > = ESC_TELEM_MAX_ESCS | | data_mask = = 0 ) {
2021-02-23 18:04:06 -04:00
return ;
}
2021-06-06 22:02:35 -03:00
_have_data = true ;
2024-02-20 08:45:41 -04:00
volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ esc_index ] ;
2021-06-06 22:02:35 -03:00
2022-10-11 16:34:32 -03:00
# if AP_TEMPERATURE_SENSOR_ENABLED
2023-01-19 13:09:29 -04:00
// always allow external data. Block "internal" if external has ever its ever been set externally then ignore normal "internal" updates
const bool has_temperature = ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : TEMPERATURE_EXTERNAL ) | |
2024-02-20 08:45:41 -04:00
( ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : TEMPERATURE ) & & ! ( telemdata . types & AP_ESC_Telem_Backend : : TelemetryType : : TEMPERATURE_EXTERNAL ) ) ;
2023-01-19 13:09:29 -04:00
const bool has_motor_temperature = ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : MOTOR_TEMPERATURE_EXTERNAL ) | |
2024-02-20 08:45:41 -04:00
( ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : MOTOR_TEMPERATURE ) & & ! ( telemdata . types & AP_ESC_Telem_Backend : : TelemetryType : : MOTOR_TEMPERATURE_EXTERNAL ) ) ;
2022-10-11 16:34:32 -03:00
# else
2023-01-19 13:09:29 -04:00
const bool has_temperature = ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : TEMPERATURE ) ;
const bool has_motor_temperature = ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : MOTOR_TEMPERATURE ) ;
# endif
if ( has_temperature ) {
2024-02-20 08:45:41 -04:00
telemdata . temperature_cdeg = new_data . temperature_cdeg ;
2020-02-10 00:13:22 -04:00
}
2023-01-19 13:09:29 -04:00
if ( has_motor_temperature ) {
2024-02-20 08:45:41 -04:00
telemdata . motor_temp_cdeg = new_data . motor_temp_cdeg ;
2021-02-23 18:04:06 -04:00
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : VOLTAGE ) {
2024-02-20 08:45:41 -04:00
telemdata . voltage = new_data . voltage ;
2021-02-23 18:04:06 -04:00
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : CURRENT ) {
2024-02-20 08:45:41 -04:00
telemdata . current = new_data . current ;
2021-02-23 18:04:06 -04:00
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : CONSUMPTION ) {
2024-02-20 08:45:41 -04:00
telemdata . consumption_mah = new_data . consumption_mah ;
2021-02-23 18:04:06 -04:00
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : USAGE ) {
2024-02-20 08:45:41 -04:00
telemdata . usage_s = new_data . usage_s ;
2021-02-23 18:04:06 -04:00
}
2024-07-27 16:44:13 -03:00
# if AP_EXTENDED_ESC_TELEM_ENABLED
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : INPUT_DUTY ) {
_telem_data [ esc_index ] . input_duty = new_data . input_duty ;
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : OUTPUT_DUTY ) {
_telem_data [ esc_index ] . output_duty = new_data . output_duty ;
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : FLAGS ) {
_telem_data [ esc_index ] . flags = new_data . flags ;
}
2024-07-27 16:03:36 -03:00
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : POWER_PERCENTAGE ) {
_telem_data [ esc_index ] . power_percentage = new_data . power_percentage ;
}
2024-07-27 16:44:13 -03:00
# endif //AP_EXTENDED_ESC_TELEM_ENABLED
2024-05-13 20:16:21 -03:00
# if AP_EXTENDED_DSHOT_TELEM_V2_ENABLED
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : EDT2_STATUS ) {
telemdata . edt2_status = merge_edt2_status ( telemdata . edt2_status , new_data . edt2_status ) ;
}
if ( data_mask & AP_ESC_Telem_Backend : : TelemetryType : : EDT2_STRESS ) {
telemdata . edt2_stress = merge_edt2_stress ( telemdata . edt2_stress , new_data . edt2_stress ) ;
}
# endif
2024-02-20 08:45:41 -04:00
telemdata . count + + ;
telemdata . types | = data_mask ;
telemdata . last_update_ms = AP_HAL : : millis ( ) ;
2021-02-23 18:04:06 -04:00
}
// record an update to the RPM together with timestamp, this allows the notch values to be slewed
// this should be called by backends when new telemetry values are available
2022-05-25 08:26:02 -03:00
void AP_ESC_Telem : : update_rpm ( const uint8_t esc_index , const float new_rpm , const float error_rate )
2021-02-23 18:04:06 -04:00
{
2021-08-10 19:12:06 -03:00
if ( esc_index > = ESC_TELEM_MAX_ESCS ) {
2021-02-23 18:04:06 -04:00
return ;
}
2021-06-06 22:02:35 -03:00
_have_data = true ;
2023-07-21 14:07:13 -03:00
const uint32_t now = MAX ( 1U , AP_HAL : : micros ( ) ) ; // don't allow a value of 0 in, as we use this as a flag in places
2021-02-23 18:04:06 -04:00
volatile AP_ESC_Telem_Backend : : RpmData & rpmdata = _rpm_data [ esc_index ] ;
2022-07-07 22:59:59 -03:00
const auto last_update_us = rpmdata . last_update_us ;
2021-02-23 18:04:06 -04:00
rpmdata . prev_rpm = rpmdata . rpm ;
rpmdata . rpm = new_rpm ;
2023-07-21 14:07:13 -03:00
rpmdata . update_rate_hz = 1.0e6 f / constrain_uint32 ( ( now - last_update_us ) , 100 , 1000000U * 10U ) ; // limit the update rate 0.1Hz to 10KHz
2021-02-23 18:04:06 -04:00
rpmdata . last_update_us = now ;
rpmdata . error_rate = error_rate ;
2023-07-21 14:07:13 -03:00
rpmdata . data_valid = true ;
2021-02-23 18:04:06 -04:00
# ifdef ESC_TELEM_DEBUG
2023-01-14 08:18:54 -04:00
hal . console - > printf ( " RPM: rate=%.1fhz, rpm=%f) \n " , rpmdata . update_rate_hz , new_rpm ) ;
2020-02-10 00:13:22 -04:00
# endif
2021-02-23 18:04:06 -04:00
}
2024-05-13 20:16:21 -03:00
# if AP_EXTENDED_DSHOT_TELEM_V2_ENABLED
// The following is based on https://github.com/bird-sanctuary/extended-dshot-telemetry.
// For the following part we explain the bits of Extended DShot Telemetry v2 status telemetry:
// - bits 0-3: the "stress level"
// - bit 5: the "error" bit (e.g. the stall event in Bluejay)
// - bit 6: the "warning" bit (e.g. the desync event in Bluejay)
// - bit 7: the "alert" bit (e.g. the demag event in Bluejay)
// Since logger can read out telemetry values less frequently than they are updated,
// it makes sense to aggregate these status bits, and to collect the maximum observed stress level.
// To reduce the logging rate of the EDT2 messages, we will try to log them only once a new frame comes.
// To track this, we are going to (ab)use bit 15 of the field: 1 means there is something to write.
// EDTv2 also features separate "stress" messages.
// These come more frequently, and are scaled differently (the allowed range is from 0 to 255),
// so we have to log them separately.
constexpr uint16_t EDT2_TELEM_UPDATED = 0x8000U ;
constexpr uint16_t EDT2_STRESS_0F_MASK = 0xfU ;
constexpr uint16_t EDT2_STRESS_FF_MASK = 0xffU ;
constexpr uint16_t EDT2_ERROR_MASK = 0x20U ;
constexpr uint16_t EDT2_WARNING_MASK = 0x40U ;
constexpr uint16_t EDT2_ALERT_MASK = 0x80U ;
constexpr uint16_t EDT2_ALL_BITS = EDT2_ERROR_MASK | EDT2_WARNING_MASK | EDT2_ALERT_MASK ;
# define EDT2_HAS_NEW_DATA(status) bool((status) & EDT2_TELEM_UPDATED)
# define EDT2_STRESS_FROM_STATUS(status) uint8_t((status) & EDT2_STRESS_0F_MASK)
# define EDT2_ERROR_BIT_FROM_STATUS(status) bool((status) & EDT2_ERROR_MASK)
# define EDT2_WARNING_BIT_FROM_STATUS(status) bool((status) & EDT2_WARNING_MASK)
# define EDT2_ALERT_BIT_FROM_STATUS(status) bool((status) & EDT2_ALERT_MASK)
# define EDT2_STRESS_FROM_STRESS(stress) uint8_t((stress) & EDT2_STRESS_FF_MASK)
uint16_t AP_ESC_Telem : : merge_edt2_status ( uint16_t old_status , uint16_t new_status )
{
if ( EDT2_HAS_NEW_DATA ( old_status ) ) {
new_status = uint16_t (
( old_status & ~ EDT2_STRESS_0F_MASK ) | // old status except for the stress is preserved
( new_status & EDT2_ALL_BITS ) | // all new status bits are included
MAX ( old_status & EDT2_STRESS_0F_MASK , new_status & EDT2_STRESS_0F_MASK ) // the stress is maxed out
) ;
}
return EDT2_TELEM_UPDATED | new_status ;
}
uint16_t AP_ESC_Telem : : merge_edt2_stress ( uint16_t old_stress , uint16_t new_stress )
{
if ( EDT2_HAS_NEW_DATA ( old_stress ) ) {
new_stress = uint16_t (
MAX ( old_stress & EDT2_STRESS_FF_MASK , new_stress & EDT2_STRESS_FF_MASK ) // the stress is maxed out
) ;
}
return EDT2_TELEM_UPDATED | new_stress ;
}
# endif // AP_EXTENDED_DSHOT_TELEM_V2_ENABLED
2021-02-23 18:04:06 -04:00
void AP_ESC_Telem : : update ( )
{
2023-07-13 21:58:05 -03:00
# if HAL_LOGGING_ENABLED
2021-02-23 18:04:06 -04:00
AP_Logger * logger = AP_Logger : : get_singleton ( ) ;
2024-02-20 08:41:03 -04:00
const uint64_t now_us64 = AP_HAL : : micros64 ( ) ;
2021-02-23 18:04:06 -04:00
2023-07-21 14:07:13 -03:00
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
2024-02-20 08:45:41 -04:00
const volatile AP_ESC_Telem_Backend : : RpmData & rpmdata = _rpm_data [ i ] ;
2024-05-13 20:16:21 -03:00
volatile AP_ESC_Telem_Backend : : TelemetryData & telemdata = _telem_data [ i ] ;
2023-07-21 14:07:13 -03:00
// Push received telemetry data into the logging system
if ( logger & & logger - > logging_enabled ( ) ) {
2024-02-20 08:45:41 -04:00
if ( telemdata . last_update_ms ! = _last_telem_log_ms [ i ]
| | rpmdata . last_update_us ! = _last_rpm_log_us [ i ] ) {
2021-02-23 18:04:06 -04:00
2024-07-27 16:44:13 -03:00
// Update last log timestamps
_last_telem_log_ms [ i ] = telemdata . last_update_ms ;
_last_rpm_log_us [ i ] = rpmdata . last_update_us ;
2024-02-18 08:15:33 -04:00
float rpm = AP : : logger ( ) . quiet_nanf ( ) ;
2021-02-23 18:04:06 -04:00
get_rpm ( i , rpm ) ;
2024-02-18 08:15:33 -04:00
float raw_rpm = AP : : logger ( ) . quiet_nanf ( ) ;
2024-01-14 15:33:15 -04:00
get_raw_rpm ( i , raw_rpm ) ;
2021-02-23 18:04:06 -04:00
// Write ESC status messages
// id starts from 0
2024-01-14 15:33:15 -04:00
// rpm, raw_rpm is eRPM (in RPM units)
2021-02-23 18:04:06 -04:00
// voltage is in Volt
// current is in Ampere
// esc_temp is in centi-degrees Celsius
2022-01-29 20:18:52 -04:00
// current_tot is in milli-Ampere hours
2021-02-23 18:04:06 -04:00
// motor_temp is in centi-degrees Celsius
// error_rate is in percentage
const struct log_Esc pkt {
LOG_PACKET_HEADER_INIT ( uint8_t ( LOG_ESC_MSG ) ) ,
2024-02-20 08:41:03 -04:00
time_us : now_us64 ,
2021-02-23 18:04:06 -04:00
instance : i ,
2024-01-14 15:33:15 -04:00
rpm : rpm ,
raw_rpm : raw_rpm ,
2024-02-20 08:45:41 -04:00
voltage : telemdata . voltage ,
current : telemdata . current ,
esc_temp : telemdata . temperature_cdeg ,
current_tot : telemdata . consumption_mah ,
motor_temp : telemdata . motor_temp_cdeg ,
error_rate : rpmdata . error_rate
2021-02-23 18:04:06 -04:00
} ;
AP : : logger ( ) . WriteBlock ( & pkt , sizeof ( pkt ) ) ;
2024-07-27 16:44:13 -03:00
# if AP_EXTENDED_ESC_TELEM_ENABLED
// Write ESC extended status messages
// id: starts from 0
// input duty: duty cycle input to the ESC in percent
// output duty: duty cycle output to the motor in percent
// status flags: manufacurer-specific status flags
const bool has_ext_data = telemdata . types &
( AP_ESC_Telem_Backend : : TelemetryType : : INPUT_DUTY |
AP_ESC_Telem_Backend : : TelemetryType : : OUTPUT_DUTY |
2024-07-27 16:03:36 -03:00
AP_ESC_Telem_Backend : : TelemetryType : : FLAGS |
AP_ESC_Telem_Backend : : TelemetryType : : POWER_PERCENTAGE ) ;
2024-07-27 16:44:13 -03:00
if ( has_ext_data ) {
// @LoggerMessage: ESCX
// @Description: ESC extended telemetry data
// @Field: TimeUS: Time since system startup
// @Field: Instance: starts from 0
// @Field: inpct: input duty cycle in percent
// @Field: outpct: output duty cycle in percent
// @Field: flags: manufacturer-specific status flags
2024-07-27 16:03:36 -03:00
// @Field: Pwr: Power percentage
2024-07-27 16:44:13 -03:00
AP : : logger ( ) . WriteStreaming ( " ESCX " ,
2024-07-27 16:03:36 -03:00
" TimeUS,Instance,inpct,outpct,flags,Pwr " ,
" s " " # " " % " " % " " - " " % " ,
" F " " - " " - " " - " " - " " - " ,
" Q " " B " " B " " B " " I " " B " ,
2024-07-27 16:44:13 -03:00
AP_HAL : : micros64 ( ) ,
i ,
telemdata . input_duty ,
telemdata . output_duty ,
2024-07-27 16:03:36 -03:00
telemdata . flags ,
telemdata . power_percentage ) ;
2024-07-27 16:44:13 -03:00
}
# endif //AP_EXTENDED_ESC_TELEM_ENABLED
2021-02-23 18:04:06 -04:00
}
2024-05-13 20:16:21 -03:00
# if AP_EXTENDED_DSHOT_TELEM_V2_ENABLED
// Write an EDTv2 message, if there is any update
uint16_t edt2_status = telemdata . edt2_status ;
uint16_t edt2_stress = telemdata . edt2_stress ;
if ( EDT2_HAS_NEW_DATA ( edt2_status | edt2_stress ) ) {
// Could probably be faster/smaller with bitmasking, but not sure
uint8_t status = 0 ;
if ( EDT2_HAS_NEW_DATA ( edt2_stress ) ) {
status | = uint8_t ( log_Edt2_Status : : HAS_STRESS_DATA ) ;
}
if ( EDT2_HAS_NEW_DATA ( edt2_status ) ) {
status | = uint8_t ( log_Edt2_Status : : HAS_STATUS_DATA ) ;
}
if ( EDT2_ALERT_BIT_FROM_STATUS ( edt2_status ) ) {
status | = uint8_t ( log_Edt2_Status : : ALERT_BIT ) ;
}
if ( EDT2_WARNING_BIT_FROM_STATUS ( edt2_status ) ) {
status | = uint8_t ( log_Edt2_Status : : WARNING_BIT ) ;
}
if ( EDT2_ERROR_BIT_FROM_STATUS ( edt2_status ) ) {
status | = uint8_t ( log_Edt2_Status : : ERROR_BIT ) ;
}
// An EDT2 status message is:
// id: starts from 0
// stress: the current stress which comes from edt2_stress
// max_stress: the maximum stress which comes from edt2_status
// status: the status bits which come from both
const struct log_Edt2 pkt_edt2 {
LOG_PACKET_HEADER_INIT ( uint8_t ( LOG_EDT2_MSG ) ) ,
time_us : now_us64 ,
instance : i ,
stress : EDT2_STRESS_FROM_STRESS ( edt2_stress ) ,
max_stress : EDT2_STRESS_FROM_STATUS ( edt2_status ) ,
status : status ,
} ;
if ( AP : : logger ( ) . WriteBlock_first_succeed ( & pkt_edt2 , sizeof ( pkt_edt2 ) ) ) {
// Only clean the telem_updated bits if the write succeeded.
// This is important because, if rate limiting is enabled,
// the log-on-change behavior may lose a lot of entries
telemdata . edt2_status & = ~ EDT2_TELEM_UPDATED ;
telemdata . edt2_stress & = ~ EDT2_TELEM_UPDATED ;
}
}
# endif // AP_EXTENDED_DSHOT_TELEM_V2_ENABLED
2021-02-23 18:04:06 -04:00
}
2024-02-20 08:41:03 -04:00
}
2023-07-13 21:58:05 -03:00
# endif // HAL_LOGGING_ENABLED
2023-07-21 14:07:13 -03:00
2024-02-20 08:41:03 -04:00
const uint32_t now_us = AP_HAL : : micros ( ) ;
for ( uint8_t i = 0 ; i < ESC_TELEM_MAX_ESCS ; i + + ) {
// Invalidate RPM data if not received for too long
2023-07-21 14:07:13 -03:00
if ( ( now_us - _rpm_data [ i ] . last_update_us ) > ESC_RPM_DATA_TIMEOUT_US ) {
_rpm_data [ i ] . data_valid = false ;
}
2021-02-23 18:04:06 -04:00
}
2020-02-10 00:13:22 -04:00
}
2024-02-20 08:26:24 -04:00
bool AP_ESC_Telem : : rpm_data_within_timeout ( const volatile AP_ESC_Telem_Backend : : RpmData & instance , const uint32_t now_us , const uint32_t timeout_us )
2023-07-21 14:07:13 -03:00
{
// easy case, has the time window been crossed so it's invalid
if ( ( now_us - instance . last_update_us ) > timeout_us ) {
return false ;
}
// we never got a valid data, to it's invalid
if ( instance . last_update_us = = 0 ) {
return false ;
}
// check if things generally expired on us, this is done to handle time wrapping
return instance . data_valid ;
}
2024-02-20 08:26:24 -04:00
bool AP_ESC_Telem : : was_rpm_data_ever_reported ( const volatile AP_ESC_Telem_Backend : : RpmData & instance )
2023-07-21 14:07:13 -03:00
{
return instance . last_update_us > 0 ;
}
2022-09-01 05:02:33 -03:00
# if AP_SCRIPTING_ENABLED
/*
set RPM scale factor from script
*/
void AP_ESC_Telem : : set_rpm_scale ( const uint8_t esc_index , const float scale_factor )
{
if ( esc_index < ESC_TELEM_MAX_ESCS ) {
rpm_scale_factor [ esc_index ] = scale_factor ;
rpm_scale_mask | = ( 1U < < esc_index ) ;
}
}
# endif
2020-02-10 00:13:22 -04:00
AP_ESC_Telem * AP_ESC_Telem : : _singleton = nullptr ;
/*
2021-02-23 12:16:37 -04:00
* Get the AP_ESC_Telem singleton
2020-02-10 00:13:22 -04:00
*/
AP_ESC_Telem * AP_ESC_Telem : : get_singleton ( )
{
return AP_ESC_Telem : : _singleton ;
}
namespace AP {
AP_ESC_Telem & esc_telem ( )
{
return * AP_ESC_Telem : : get_singleton ( ) ;
}
} ;
2021-02-23 18:04:06 -04:00
2024-05-13 20:16:21 -03:00
# endif // HAL_WITH_ESC_TELEM