IO_Firmware: update IOMCU firmware with profiled LED support over safety pins
This commit is contained in:
parent
93d99ece0a
commit
4cbc4dfe34
@ -43,6 +43,7 @@ enum ioevents {
|
|||||||
IOEVENT_SET_DSHOT_PERIOD,
|
IOEVENT_SET_DSHOT_PERIOD,
|
||||||
IOEVENT_SET_CHANNEL_MASK,
|
IOEVENT_SET_CHANNEL_MASK,
|
||||||
IOEVENT_DSHOT,
|
IOEVENT_DSHOT,
|
||||||
|
IOEVENT_PROFILED,
|
||||||
};
|
};
|
||||||
|
|
||||||
// max number of consecutve protocol failures we accept before raising
|
// max number of consecutve protocol failures we accept before raising
|
||||||
@ -89,7 +90,9 @@ void AP_IOMCU::init(void)
|
|||||||
crc_is_ok = true;
|
crc_is_ok = true;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
use_safety_as_led = boardconfig->use_safety_as_led();
|
||||||
|
#endif
|
||||||
if (!hal.scheduler->thread_create(FUNCTOR_BIND_MEMBER(&AP_IOMCU::thread_main, void), "IOMCU",
|
if (!hal.scheduler->thread_create(FUNCTOR_BIND_MEMBER(&AP_IOMCU::thread_main, void), "IOMCU",
|
||||||
1024, AP_HAL::Scheduler::PRIORITY_BOOST, 1)) {
|
1024, AP_HAL::Scheduler::PRIORITY_BOOST, 1)) {
|
||||||
AP_HAL::panic("Unable to allocate IOMCU thread");
|
AP_HAL::panic("Unable to allocate IOMCU thread");
|
||||||
@ -300,6 +303,16 @@ void AP_IOMCU::thread_main(void)
|
|||||||
}
|
}
|
||||||
mask &= ~EVENT_MASK(IOEVENT_DSHOT);
|
mask &= ~EVENT_MASK(IOEVENT_DSHOT);
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
if (mask & EVENT_MASK(IOEVENT_PROFILED)) {
|
||||||
|
if (!write_registers(PAGE_PROFILED, 0, sizeof(profiled)/sizeof(uint16_t), (const uint16_t*)&profiled)) {
|
||||||
|
event_failed(mask);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mask &= ~EVENT_MASK(IOEVENT_PROFILED);
|
||||||
|
#endif
|
||||||
|
|
||||||
// check for regular timed events
|
// check for regular timed events
|
||||||
uint32_t now = AP_HAL::millis();
|
uint32_t now = AP_HAL::millis();
|
||||||
if (now - last_rc_read_ms > 20) {
|
if (now - last_rc_read_ms > 20) {
|
||||||
@ -1439,6 +1452,23 @@ void AP_IOMCU::toggle_GPIO(uint8_t pin)
|
|||||||
trigger_event(IOEVENT_GPIO);
|
trigger_event(IOEVENT_GPIO);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
// set profiled R G B values
|
||||||
|
void AP_IOMCU::set_profiled(uint8_t r, uint8_t g, uint8_t b)
|
||||||
|
{
|
||||||
|
if (!use_safety_as_led) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (r == profiled.red && g == profiled.green && b == profiled.blue) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
profiled.magic = PROFILED_ENABLE_MAGIC;
|
||||||
|
profiled.red = r;
|
||||||
|
profiled.green = g;
|
||||||
|
profiled.blue = b;
|
||||||
|
trigger_event(IOEVENT_PROFILED);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace AP {
|
namespace AP {
|
||||||
AP_IOMCU *iomcu(void) {
|
AP_IOMCU *iomcu(void) {
|
||||||
|
@ -176,6 +176,11 @@ public:
|
|||||||
// toggle a output pin
|
// toggle a output pin
|
||||||
void toggle_GPIO(uint8_t pin);
|
void toggle_GPIO(uint8_t pin);
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
// set profiled values
|
||||||
|
void set_profiled(uint8_t r, uint8_t g, uint8_t b);
|
||||||
|
#endif
|
||||||
|
|
||||||
// channel group masks
|
// channel group masks
|
||||||
const uint8_t ch_masks[3] = { 0x03,0x0C,0xF0 };
|
const uint8_t ch_masks[3] = { 0x03,0x0C,0xF0 };
|
||||||
|
|
||||||
@ -297,6 +302,11 @@ private:
|
|||||||
// output mode values
|
// output mode values
|
||||||
struct page_mode_out mode_out;
|
struct page_mode_out mode_out;
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
// profiled control
|
||||||
|
struct page_profiled profiled {0, 255, 255, 255}; // by default, white
|
||||||
|
#endif
|
||||||
|
|
||||||
// IMU heater duty cycle
|
// IMU heater duty cycle
|
||||||
uint8_t heater_duty_cycle;
|
uint8_t heater_duty_cycle;
|
||||||
|
|
||||||
@ -310,6 +320,9 @@ private:
|
|||||||
bool detected_io_reset;
|
bool detected_io_reset;
|
||||||
bool initialised;
|
bool initialised;
|
||||||
bool is_chibios_backend;
|
bool is_chibios_backend;
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
bool use_safety_as_led;
|
||||||
|
#endif
|
||||||
|
|
||||||
uint32_t protocol_fail_count;
|
uint32_t protocol_fail_count;
|
||||||
uint32_t protocol_count;
|
uint32_t protocol_count;
|
||||||
|
@ -489,6 +489,10 @@ void AP_IOMCU_FW::update()
|
|||||||
last_status_ms = now;
|
last_status_ms = now;
|
||||||
page_status_update();
|
page_status_update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
profiled_update();
|
||||||
|
#endif
|
||||||
#ifdef HAL_WITH_BIDIR_DSHOT
|
#ifdef HAL_WITH_BIDIR_DSHOT
|
||||||
// EDT updates are semt at ~1Hz per ESC, but we want to make sure
|
// EDT updates are semt at ~1Hz per ESC, but we want to make sure
|
||||||
// that we don't delay updates unduly so sample at 5Hz
|
// that we don't delay updates unduly so sample at 5Hz
|
||||||
@ -1029,6 +1033,20 @@ bool AP_IOMCU_FW::handle_code_write()
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
case PAGE_PROFILED:
|
||||||
|
if (rx_io_packet.count != 2 || (rx_io_packet.regs[0] & 0xFF) != PROFILED_ENABLE_MAGIC) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
profiled_brg[0] = rx_io_packet.regs[0] >> 8;
|
||||||
|
profiled_brg[1] = rx_io_packet.regs[1] & 0xFF;
|
||||||
|
profiled_brg[2] = rx_io_packet.regs[1] >> 8;
|
||||||
|
// push new led data
|
||||||
|
profiled_num_led_pushed = 0;
|
||||||
|
profiled_control_enabled = true;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1063,6 +1081,48 @@ void AP_IOMCU_FW::calculate_fw_crc(void)
|
|||||||
reg_setup.crc[1] = sum >> 16;
|
reg_setup.crc[1] = sum >> 16;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
// bitbang profiled bitstream, 8-10 chunks at a time
|
||||||
|
// Max time taken per call is ~7us
|
||||||
|
void AP_IOMCU_FW::profiled_update(void)
|
||||||
|
{
|
||||||
|
if (profiled_num_led_pushed > PROFILED_LED_LEN) {
|
||||||
|
profiled_byte_index = 0;
|
||||||
|
profiled_leading_zeros = PROFILED_LEADING_ZEROS;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// push 10 zero leading bits at a time
|
||||||
|
if (profiled_leading_zeros != 0) {
|
||||||
|
for (uint8_t i = 0; i < 10; i++) {
|
||||||
|
palClearLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
|
palSetLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
|
profiled_leading_zeros--;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((profiled_byte_index == 0) ||
|
||||||
|
(profiled_byte_index == PROFILED_OUTPUT_BYTE_LEN)) {
|
||||||
|
// start bit
|
||||||
|
palClearLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
|
palSetLine(HAL_GPIO_PIN_SAFETY_LED);
|
||||||
|
palSetLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
|
profiled_byte_index = 0;
|
||||||
|
profiled_num_led_pushed++;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t byte_val = profiled_brg[profiled_byte_index];
|
||||||
|
for (uint8_t i = 0; i < 8; i++) {
|
||||||
|
palClearLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
|
palWriteLine(HAL_GPIO_PIN_SAFETY_LED, byte_val & 1);
|
||||||
|
byte_val >>= 1;
|
||||||
|
palSetLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
profiled_byte_index++;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
update safety state
|
update safety state
|
||||||
@ -1076,6 +1136,15 @@ void AP_IOMCU_FW::safety_update(void)
|
|||||||
}
|
}
|
||||||
safety_update_ms = now;
|
safety_update_ms = now;
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
if (profiled_control_enabled) {
|
||||||
|
// set line mode to output for safety input pin
|
||||||
|
palSetLineMode(HAL_GPIO_PIN_SAFETY_INPUT, PAL_MODE_OUTPUT_PUSHPULL);
|
||||||
|
palSetLineMode(HAL_GPIO_PIN_SAFETY_LED, PAL_MODE_OUTPUT_PUSHPULL);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
bool safety_pressed = palReadLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
bool safety_pressed = palReadLine(HAL_GPIO_PIN_SAFETY_INPUT);
|
||||||
if (safety_pressed) {
|
if (safety_pressed) {
|
||||||
if (reg_status.flag_safety_off && (reg_setup.arming & P_SETUP_ARMING_SAFETY_DISABLE_ON)) {
|
if (reg_status.flag_safety_off && (reg_setup.arming & P_SETUP_ARMING_SAFETY_DISABLE_ON)) {
|
||||||
|
@ -17,6 +17,10 @@
|
|||||||
#define PWM_IGNORE_THIS_CHANNEL UINT16_MAX
|
#define PWM_IGNORE_THIS_CHANNEL UINT16_MAX
|
||||||
#define SERVO_COUNT 8
|
#define SERVO_COUNT 8
|
||||||
|
|
||||||
|
#define PROFILED_LED_LEN 2
|
||||||
|
#define PROFILED_OUTPUT_BYTE_LEN 3
|
||||||
|
#define PROFILED_LEADING_ZEROS 50
|
||||||
|
|
||||||
class AP_IOMCU_FW {
|
class AP_IOMCU_FW {
|
||||||
public:
|
public:
|
||||||
void process_io_packet();
|
void process_io_packet();
|
||||||
@ -36,6 +40,9 @@ public:
|
|||||||
bool handle_code_write();
|
bool handle_code_write();
|
||||||
bool handle_code_read();
|
bool handle_code_read();
|
||||||
void schedule_reboot(uint32_t time_ms);
|
void schedule_reboot(uint32_t time_ms);
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
void profiled_update();
|
||||||
|
#endif
|
||||||
void safety_update();
|
void safety_update();
|
||||||
void rcout_config_update();
|
void rcout_config_update();
|
||||||
void rcin_serial_init();
|
void rcin_serial_init();
|
||||||
@ -81,6 +88,14 @@ public:
|
|||||||
|
|
||||||
uint16_t last_channel_mask;
|
uint16_t last_channel_mask;
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
uint8_t profiled_byte_index;
|
||||||
|
uint8_t profiled_leading_zeros;
|
||||||
|
uint8_t profiled_num_led_pushed;
|
||||||
|
uint8_t profiled_brg[3];
|
||||||
|
bool profiled_control_enabled;
|
||||||
|
#endif
|
||||||
|
|
||||||
// CONFIG values
|
// CONFIG values
|
||||||
struct page_config config;
|
struct page_config config;
|
||||||
|
|
||||||
@ -117,6 +132,11 @@ public:
|
|||||||
// output mode values
|
// output mode values
|
||||||
struct page_mode_out mode_out;
|
struct page_mode_out mode_out;
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
// profiled control values
|
||||||
|
struct page_profiled profiled;
|
||||||
|
#endif
|
||||||
|
|
||||||
uint16_t last_output_mode_mask;
|
uint16_t last_output_mode_mask;
|
||||||
uint16_t last_output_bdmask;
|
uint16_t last_output_bdmask;
|
||||||
uint16_t last_output_esc_type;
|
uint16_t last_output_esc_type;
|
||||||
|
@ -7,6 +7,10 @@
|
|||||||
common protocol definitions between AP_IOMCU and iofirmware
|
common protocol definitions between AP_IOMCU and iofirmware
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#ifndef AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
#define AP_IOMCU_PROFILED_SUPPORT_ENABLED 0
|
||||||
|
#endif
|
||||||
|
|
||||||
// 22 is enough for the rc_input page in one transfer
|
// 22 is enough for the rc_input page in one transfer
|
||||||
#define PKT_MAX_REGS 22
|
#define PKT_MAX_REGS 22
|
||||||
// The number of channels that can be propagated - due to SBUS_OUT is higher than the physical channels
|
// The number of channels that can be propagated - due to SBUS_OUT is higher than the physical channels
|
||||||
@ -68,6 +72,9 @@ enum iopage {
|
|||||||
PAGE_RAW_DSHOT_TELEM_5_8 = 205,
|
PAGE_RAW_DSHOT_TELEM_5_8 = 205,
|
||||||
PAGE_RAW_DSHOT_TELEM_9_12 = 206,
|
PAGE_RAW_DSHOT_TELEM_9_12 = 206,
|
||||||
PAGE_RAW_DSHOT_TELEM_13_16 = 207,
|
PAGE_RAW_DSHOT_TELEM_13_16 = 207,
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
PAGE_PROFILED = 208,
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
// setup page registers
|
// setup page registers
|
||||||
@ -114,6 +121,8 @@ enum iopage {
|
|||||||
#define PAGE_REG_SETUP_FORCE_SAFETY_ON 14
|
#define PAGE_REG_SETUP_FORCE_SAFETY_ON 14
|
||||||
#define FORCE_SAFETY_MAGIC 22027
|
#define FORCE_SAFETY_MAGIC 22027
|
||||||
|
|
||||||
|
#define PROFILED_ENABLE_MAGIC 123
|
||||||
|
|
||||||
struct page_config {
|
struct page_config {
|
||||||
uint16_t protocol_version;
|
uint16_t protocol_version;
|
||||||
uint16_t protocol_version2;
|
uint16_t protocol_version2;
|
||||||
@ -230,3 +239,12 @@ struct page_dshot_telem {
|
|||||||
uint8_t edt2_stress[4];
|
uint8_t edt2_stress[4];
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if AP_IOMCU_PROFILED_SUPPORT_ENABLED
|
||||||
|
struct __attribute__((packed, aligned(2))) page_profiled {
|
||||||
|
uint8_t magic;
|
||||||
|
uint8_t blue;
|
||||||
|
uint8_t red;
|
||||||
|
uint8_t green;
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
Loading…
Reference in New Issue
Block a user