/* ESC Telemetry for Hobbywing Pro 80A HV ESC. This will be incorporated into a broader ESC telemetry library in ArduPilot master in the future This protocol only allows for one ESC per UART RX line, so using a CAN node per ESC works well. */ #include "AP_Periph.h" #include "hwing_esc.h" #include #include #ifdef HAL_PERIPH_ENABLE_HWESC extern const AP_HAL::HAL& hal; #define TELEM_HEADER 0x9B #define TELEM_LEN 0x16 // constructor HWESC_Telem::HWESC_Telem(void) { } void HWESC_Telem::init(AP_HAL::UARTDriver *_uart) { uart = _uart; uart->begin(19200); uart->set_options(AP_HAL::UARTDriver::OPTION_PULLDOWN_RX); } /* update ESC telemetry */ bool HWESC_Telem::update() { uint32_t n = uart->available(); if (n == 0) { return false; } // we expect at least 50ms idle between frames uint32_t now = AP_HAL::millis(); bool frame_gap = (now - last_read_ms) > 10; last_read_ms = now; // don't read too much in one loop to prevent too high CPU load if (n > 500) { n = 500; } if (len == 0 && !frame_gap) { uart->discard_input(); return false; } if (frame_gap) { len = 0; } bool ret = false; while (n--) { uint8_t b = uart->read(); //hal.console->printf("t=%u 0x%02x\n", now, b); if (len == 0 && b != TELEM_HEADER) { continue; } if (len == 1 && b != TELEM_LEN) { continue; } uint8_t *buf = (uint8_t *)&pkt; buf[len++] = b; if (len == sizeof(pkt)) { ret = parse_packet(); len = 0; } } return ret; } static uint16_t calc_crc(const uint8_t *buf, uint8_t len) { uint16_t crc = 0; while (len--) { crc += *buf++; } return crc; } static const struct { uint8_t adc_temp; uint8_t temp_C; } temp_table[] = { { 241, 0}, { 240, 1}, { 239, 2}, { 238, 3}, { 237, 4}, { 236, 5}, { 235, 6}, { 234, 7}, { 233, 8}, { 232, 9}, { 231, 10}, { 230, 11}, { 229, 12}, { 228, 13}, { 227, 14}, { 226, 15}, { 224, 16}, { 223, 17}, { 222, 18}, { 220, 19}, { 219, 20}, { 217, 21}, { 216, 22}, { 214, 23}, { 213, 24}, { 211, 25}, { 209, 26}, { 208, 27}, { 206, 28}, { 204, 29}, { 202, 30}, { 201, 31}, { 199, 32}, { 197, 33}, { 195, 34}, { 193, 35}, { 191, 36}, { 189, 37}, { 187, 38}, { 185, 39}, { 183, 40}, { 181, 41}, { 179, 42}, { 177, 43}, { 174, 44}, { 172, 45}, { 170, 46}, { 168, 47}, { 166, 48}, { 164, 49}, { 161, 50}, { 159, 51}, { 157, 52}, { 154, 53}, { 152, 54}, { 150, 55}, { 148, 56}, { 146, 57}, { 143, 58}, { 141, 59}, { 139, 60}, { 136, 61}, { 134, 62}, { 132, 63}, { 130, 64}, { 128, 65}, { 125, 66}, { 123, 67}, { 121, 68}, { 119, 69}, { 117, 70}, { 115, 71}, { 113, 72}, { 111, 73}, { 109, 74}, { 106, 75}, { 105, 76}, { 103, 77}, { 101, 78}, { 99, 79}, { 97, 80}, { 95, 81}, { 93, 82}, { 91, 83}, { 90, 84}, { 88, 85}, { 85, 86}, { 84, 87}, { 82, 88}, { 81, 89}, { 79, 90}, { 77, 91}, { 76, 92}, { 74, 93}, { 73, 94}, { 72, 95}, { 69, 96}, { 68, 97}, { 66, 98}, { 65, 99}, { 64, 100}, { 62, 101}, { 62, 102}, { 61, 103}, { 59, 104}, { 58, 105}, { 56, 106}, { 54, 107}, { 54, 108}, { 53, 109}, { 51, 110}, { 51, 111}, { 50, 112}, { 48, 113}, { 48, 114}, { 46, 115}, { 46, 116}, { 44, 117}, { 43, 118}, { 43, 119}, { 41, 120}, { 41, 121}, { 39, 122}, { 39, 123}, { 39, 124}, { 37, 125}, { 37, 126}, { 35, 127}, { 35, 128}, { 33, 129}, }; uint8_t HWESC_Telem::temperature_decode(uint8_t temp_raw) const { for (uint8_t i=0; i