2020-02-10 00:13:22 -04:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <AP_HAL/AP_HAL.h>
|
2022-06-06 01:40:25 -03:00
|
|
|
#include <AP_Param/AP_Param.h>
|
2022-11-29 05:47:28 -04:00
|
|
|
#include <SRV_Channel/SRV_Channel_config.h>
|
2021-02-23 18:04:06 -04:00
|
|
|
#include "AP_ESC_Telem_Backend.h"
|
|
|
|
|
|
|
|
#if HAL_WITH_ESC_TELEM
|
|
|
|
|
2022-11-29 05:47:28 -04:00
|
|
|
#define ESC_TELEM_MAX_ESCS NUM_SERVO_CHANNELS
|
|
|
|
static_assert(ESC_TELEM_MAX_ESCS > 0, "Cannot have 0 ESC telemetry instances");
|
2022-06-06 01:11:14 -03:00
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
#define ESC_TELEM_DATA_TIMEOUT_MS 5000UL
|
|
|
|
#define ESC_RPM_DATA_TIMEOUT_US 1000000UL
|
2020-02-10 00:13:22 -04:00
|
|
|
|
|
|
|
class AP_ESC_Telem {
|
|
|
|
public:
|
2021-02-23 18:04:06 -04:00
|
|
|
friend class AP_ESC_Telem_Backend;
|
2020-02-10 00:13:22 -04:00
|
|
|
|
|
|
|
AP_ESC_Telem();
|
|
|
|
|
|
|
|
/* Do not allow copies */
|
2022-06-06 01:40:25 -03:00
|
|
|
CLASS_NO_COPY(AP_ESC_Telem);
|
|
|
|
|
|
|
|
static const struct AP_Param::GroupInfo var_info[];
|
2020-02-10 00:13:22 -04:00
|
|
|
|
|
|
|
static AP_ESC_Telem *get_singleton();
|
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// get an individual ESC's slewed rpm if available, returns true on success
|
|
|
|
bool get_rpm(uint8_t esc_index, float& rpm) const;
|
|
|
|
|
|
|
|
// get an individual ESC's raw rpm if available
|
|
|
|
bool get_raw_rpm(uint8_t esc_index, float& rpm) const;
|
|
|
|
|
2023-10-03 16:25:00 -03:00
|
|
|
// get raw telemetry data, used by IOMCU
|
|
|
|
const volatile AP_ESC_Telem_Backend::TelemetryData& get_telem_data(uint8_t esc_index) const {
|
|
|
|
return _telem_data[esc_index];
|
|
|
|
}
|
|
|
|
|
2021-07-05 19:31:40 -03:00
|
|
|
// return the average motor RPM
|
|
|
|
float get_average_motor_rpm(uint32_t servo_channel_mask) const;
|
|
|
|
|
|
|
|
// return the average motor RPM
|
|
|
|
float get_average_motor_rpm() const { return get_average_motor_rpm(0xFFFFFFFF); }
|
|
|
|
|
2022-05-25 08:26:02 -03:00
|
|
|
// determine whether all the motors in servo_channel_mask are running
|
2023-04-28 17:02:55 -03:00
|
|
|
bool are_motors_running(uint32_t servo_channel_mask, float min_rpm, float max_rpm) const;
|
2022-05-25 08:26:02 -03:00
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// get an individual ESC's temperature in centi-degrees if available, returns true on success
|
|
|
|
bool get_temperature(uint8_t esc_index, int16_t& temp) const;
|
|
|
|
|
|
|
|
// get an individual motor's temperature in centi-degrees if available, returns true on success
|
|
|
|
bool get_motor_temperature(uint8_t esc_index, int16_t& temp) const;
|
|
|
|
|
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
|
|
|
|
bool get_highest_motor_temperature(int16_t& temp) const;
|
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// get an individual ESC's current in Ampere if available, returns true on success
|
|
|
|
bool get_current(uint8_t esc_index, float& amps) const;
|
|
|
|
|
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 get_usage_seconds(uint8_t esc_index, uint32_t& usage_sec) const;
|
|
|
|
|
|
|
|
// get an individual ESC's voltage in Volt if available, returns true on success
|
|
|
|
bool get_voltage(uint8_t esc_index, float& volts) const;
|
|
|
|
|
|
|
|
// get an individual ESC's consumption in milli-Ampere.hour if available, returns true on success
|
|
|
|
bool get_consumption_mah(uint8_t esc_index, float& consumption_mah) const;
|
|
|
|
|
2021-06-03 04:27:27 -03:00
|
|
|
// return the average motor frequency in Hz for dynamic filtering
|
2021-07-05 19:31:40 -03:00
|
|
|
float get_average_motor_frequency_hz(uint32_t servo_channel_mask) const { return get_average_motor_rpm(servo_channel_mask) * (1.0f / 60.0f); };
|
2021-06-03 04:27:27 -03:00
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// return the average motor frequency in Hz for dynamic filtering
|
2021-07-05 19:31:40 -03:00
|
|
|
float get_average_motor_frequency_hz() const { return get_average_motor_frequency_hz(0xFFFFFFFF); }
|
2021-02-23 18:04:06 -04:00
|
|
|
|
|
|
|
// return all of the motor frequencies in Hz for dynamic filtering
|
|
|
|
uint8_t get_motor_frequencies_hz(uint8_t nfreqs, float* freqs) const;
|
|
|
|
|
2021-06-22 14:26:49 -03:00
|
|
|
// get the number of ESCs that sent valid telemetry data in the last ESC_TELEM_DATA_TIMEOUT_MS
|
2021-02-23 18:04:06 -04:00
|
|
|
uint8_t get_num_active_escs() const;
|
|
|
|
|
2021-06-22 14:26:49 -03:00
|
|
|
// get mask of ESCs that sent valid telemetry data in the last
|
|
|
|
// ESC_TELEM_DATA_TIMEOUT_MS
|
2022-05-15 18:30:16 -03:00
|
|
|
uint32_t get_active_esc_mask() const;
|
2021-06-22 14:26:49 -03:00
|
|
|
|
2024-05-22 13:20:05 -03:00
|
|
|
// return an active ESC with the highest RPM for the purposes of reporting (e.g. in the OSD)
|
|
|
|
uint8_t get_max_rpm_esc() const;
|
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// return the last time telemetry data was received in ms for the given ESC or 0 if never
|
|
|
|
uint32_t get_last_telem_data_ms(uint8_t esc_index) const {
|
2021-08-10 12:41:16 -03:00
|
|
|
if (esc_index >= ESC_TELEM_MAX_ESCS) {return 0;}
|
2021-02-23 18:04:06 -04:00
|
|
|
return _telem_data[esc_index].last_update_ms;
|
|
|
|
}
|
|
|
|
|
|
|
|
// send telemetry data to mavlink
|
|
|
|
void send_esc_telemetry_mavlink(uint8_t mav_chan);
|
|
|
|
|
2023-10-11 04:41:52 -03:00
|
|
|
// update at 10Hz to log telemetry
|
2021-02-23 18:04:06 -04:00
|
|
|
void update();
|
2020-02-10 00:13:22 -04:00
|
|
|
|
2022-05-25 08:26:02 -03:00
|
|
|
// is rpm telemetry configured for the provided channel mask
|
|
|
|
bool is_telemetry_active(uint32_t servo_channel_mask) const;
|
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// callback to update the rpm in the frontend, should be called by the driver when new data is available
|
2022-07-07 22:58:54 -03:00
|
|
|
// can also be called from scripting
|
2022-05-25 08:26:02 -03:00
|
|
|
void update_rpm(const uint8_t esc_index, const float new_rpm, const float error_rate);
|
2022-09-01 05:02:33 -03:00
|
|
|
|
2023-08-09 21:22:33 -03:00
|
|
|
// callback to update the data in the frontend, should be called by the driver when new data is available
|
|
|
|
void update_telem_data(const uint8_t esc_index, const AP_ESC_Telem_Backend::TelemetryData& new_data, const uint16_t data_mask);
|
|
|
|
|
2022-09-01 05:02:33 -03:00
|
|
|
#if AP_SCRIPTING_ENABLED
|
|
|
|
/*
|
|
|
|
set RPM scale factor from script
|
|
|
|
*/
|
|
|
|
void set_rpm_scale(const uint8_t esc_index, const float scale_factor);
|
|
|
|
#endif
|
|
|
|
|
2022-07-07 22:58:54 -03:00
|
|
|
private:
|
2022-05-25 08:26:02 -03:00
|
|
|
|
2023-07-21 14:07:13 -03:00
|
|
|
// helper that validates RPM data
|
|
|
|
static bool rpm_data_within_timeout (const volatile AP_ESC_Telem_Backend::RpmData &instance, const uint32_t now_us, const uint32_t timeout_us);
|
|
|
|
static bool was_rpm_data_ever_reported (const volatile AP_ESC_Telem_Backend::RpmData &instance);
|
|
|
|
|
2024-05-13 20:16:21 -03:00
|
|
|
#if AP_EXTENDED_DSHOT_TELEM_V2_ENABLED
|
|
|
|
// helpers that aggregate data in EDTv2 messages
|
|
|
|
static uint16_t merge_edt2_status(uint16_t old_status, uint16_t new_status);
|
|
|
|
static uint16_t merge_edt2_stress(uint16_t old_stress, uint16_t new_stress);
|
|
|
|
#endif
|
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
// rpm data
|
|
|
|
volatile AP_ESC_Telem_Backend::RpmData _rpm_data[ESC_TELEM_MAX_ESCS];
|
|
|
|
// telemetry data
|
|
|
|
volatile AP_ESC_Telem_Backend::TelemetryData _telem_data[ESC_TELEM_MAX_ESCS];
|
|
|
|
|
|
|
|
uint32_t _last_telem_log_ms[ESC_TELEM_MAX_ESCS];
|
|
|
|
uint32_t _last_rpm_log_us[ESC_TELEM_MAX_ESCS];
|
2022-06-06 01:11:14 -03:00
|
|
|
uint8_t next_idx;
|
2020-02-10 00:13:22 -04:00
|
|
|
|
2022-09-01 05:02:33 -03:00
|
|
|
#if AP_SCRIPTING_ENABLED
|
|
|
|
// allow for scaling of RPMs via lua scripts
|
|
|
|
float rpm_scale_factor[ESC_TELEM_MAX_ESCS];
|
|
|
|
uint32_t rpm_scale_mask;
|
|
|
|
#endif
|
|
|
|
|
2021-06-06 22:02:35 -03:00
|
|
|
bool _have_data;
|
|
|
|
|
2022-06-06 01:40:25 -03:00
|
|
|
AP_Int8 mavlink_offset;
|
|
|
|
|
2021-02-23 18:04:06 -04:00
|
|
|
static AP_ESC_Telem *_singleton;
|
2020-02-10 00:13:22 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
namespace AP {
|
|
|
|
AP_ESC_Telem &esc_telem();
|
|
|
|
};
|
2021-02-23 18:04:06 -04:00
|
|
|
|
2022-06-06 01:11:14 -03:00
|
|
|
#endif // HAL_WITH_ESC_TELEM
|