#include "selfdrive/pandad/panda.h" #include #include #include #include #include "cereal/messaging/messaging.h" #include "common/swaglog.h" #include "common/util.h" Panda::Panda(std::string serial, uint32_t bus_offset) : bus_offset(bus_offset) { // try USB first, then SPI try { handle = std::make_unique(serial); LOGW("connected to %s over USB", serial.c_str()); } catch (std::exception &e) { #ifndef __APPLE__ handle = std::make_unique(serial); LOGW("connected to %s over SPI", serial.c_str()); #else throw e; #endif } hw_type = get_hw_type(); can_reset_communications(); return; } bool Panda::connected() { return handle->connected; } bool Panda::comms_healthy() { return handle->comms_healthy; } std::string Panda::hw_serial() { return handle->hw_serial; } std::vector Panda::list(bool usb_only) { std::vector serials = PandaUsbHandle::list(); #ifndef __APPLE__ if (!usb_only) { for (auto s : PandaSpiHandle::list()) { if (std::find(serials.begin(), serials.end(), s) == serials.end()) { serials.push_back(s); } } } #endif return serials; } void Panda::set_safety_model(cereal::CarParams::SafetyModel safety_model, uint16_t safety_param) { handle->control_write(0xdc, (uint16_t)safety_model, safety_param); } void Panda::set_alternative_experience(uint16_t alternative_experience) { handle->control_write(0xdf, alternative_experience, 0); } cereal::PandaState::PandaType Panda::get_hw_type() { unsigned char hw_query[1] = {0}; handle->control_read(0xc1, 0, 0, hw_query, 1); return (cereal::PandaState::PandaType)(hw_query[0]); } void Panda::set_fan_speed(uint16_t fan_speed) { handle->control_write(0xb1, fan_speed, 0); } uint16_t Panda::get_fan_speed() { uint16_t fan_speed_rpm = 0; handle->control_read(0xb2, 0, 0, (unsigned char*)&fan_speed_rpm, sizeof(fan_speed_rpm)); return fan_speed_rpm; } void Panda::set_ir_pwr(uint16_t ir_pwr) { handle->control_write(0xb0, ir_pwr, 0); } std::optional Panda::get_state() { health_t health {0}; int err = handle->control_read(0xd2, 0, 0, (unsigned char*)&health, sizeof(health)); return err >= 0 ? std::make_optional(health) : std::nullopt; } std::optional Panda::get_can_state(uint16_t can_number) { can_health_t can_health {0}; int err = handle->control_read(0xc2, can_number, 0, (unsigned char*)&can_health, sizeof(can_health)); return err >= 0 ? std::make_optional(can_health) : std::nullopt; } void Panda::set_loopback(bool loopback) { handle->control_write(0xe5, loopback, 0); } std::optional> Panda::get_firmware_version() { std::vector fw_sig_buf(128); int read_1 = handle->control_read(0xd3, 0, 0, &fw_sig_buf[0], 64); int read_2 = handle->control_read(0xd4, 0, 0, &fw_sig_buf[64], 64); return ((read_1 == 64) && (read_2 == 64)) ? std::make_optional(fw_sig_buf) : std::nullopt; } std::optional Panda::get_serial() { char serial_buf[17] = {'\0'}; int err = handle->control_read(0xd0, 0, 0, (uint8_t*)serial_buf, 16); return err >= 0 ? std::make_optional(serial_buf) : std::nullopt; } bool Panda::up_to_date() { if (auto fw_sig = get_firmware_version()) { for (auto fn : { "panda.bin.signed", "panda_h7.bin.signed" }) { auto content = util::read_file(std::string("../../panda/board/obj/") + fn); if (content.size() >= fw_sig->size() && memcmp(content.data() + content.size() - fw_sig->size(), fw_sig->data(), fw_sig->size()) == 0) { return true; } } } return false; } void Panda::set_power_saving(bool power_saving) { handle->control_write(0xe7, power_saving, 0); } void Panda::enable_deepsleep() { handle->control_write(0xfb, 0, 0); } void Panda::send_heartbeat(bool engaged) { handle->control_write(0xf3, engaged, 0); } void Panda::set_can_speed_kbps(uint16_t bus, uint16_t speed) { handle->control_write(0xde, bus, (speed * 10)); } void Panda::set_data_speed_kbps(uint16_t bus, uint16_t speed) { handle->control_write(0xf9, bus, (speed * 10)); } void Panda::set_canfd_non_iso(uint16_t bus, bool non_iso) { handle->control_write(0xfc, bus, non_iso); } static uint8_t len_to_dlc(uint8_t len) { if (len <= 8) { return len; } if (len <= 24) { return 8 + ((len - 8) / 4) + ((len % 4) ? 1 : 0); } else { return 11 + (len / 16) + ((len % 16) ? 1 : 0); } } void Panda::pack_can_buffer(const capnp::List::Reader &can_data_list, std::function write_func) { int32_t pos = 0; uint8_t send_buf[2 * USB_TX_SOFT_LIMIT]; for (auto cmsg : can_data_list) { // check if the message is intended for this panda uint8_t bus = cmsg.getSrc(); if (bus < bus_offset || bus >= (bus_offset + PANDA_BUS_OFFSET)) { continue; } auto can_data = cmsg.getDat(); uint8_t data_len_code = len_to_dlc(can_data.size()); assert(can_data.size() <= 64); assert(can_data.size() == dlc_to_len[data_len_code]); can_header header = {}; header.addr = cmsg.getAddress(); header.extended = (cmsg.getAddress() >= 0x800) ? 1 : 0; header.data_len_code = data_len_code; header.bus = bus - bus_offset; header.checksum = 0; memcpy(&send_buf[pos], (uint8_t *)&header, sizeof(can_header)); memcpy(&send_buf[pos + sizeof(can_header)], (uint8_t *)can_data.begin(), can_data.size()); uint32_t msg_size = sizeof(can_header) + can_data.size(); // set checksum ((can_header *) &send_buf[pos])->checksum = calculate_checksum(&send_buf[pos], msg_size); pos += msg_size; if (pos >= USB_TX_SOFT_LIMIT) { write_func(send_buf, pos); pos = 0; } } // send remaining packets if (pos > 0) write_func(send_buf, pos); } void Panda::can_send(capnp::List::Reader can_data_list) { pack_can_buffer(can_data_list, [=](uint8_t* data, size_t size) { handle->bulk_write(3, data, size, 5); }); } bool Panda::can_receive(std::vector& out_vec) { // Check if enough space left in buffer to store RECV_SIZE data assert(receive_buffer_size + RECV_SIZE <= sizeof(receive_buffer)); int recv = handle->bulk_read(0x81, &receive_buffer[receive_buffer_size], RECV_SIZE); if (!comms_healthy()) { return false; } bool ret = true; if (recv > 0) { receive_buffer_size += recv; ret = unpack_can_buffer(receive_buffer, receive_buffer_size, out_vec); } return ret; } void Panda::can_reset_communications() { handle->control_write(0xc0, 0, 0); } bool Panda::unpack_can_buffer(uint8_t *data, uint32_t &size, std::vector &out_vec) { int pos = 0; while (pos <= size - sizeof(can_header)) { can_header header; memcpy(&header, &data[pos], sizeof(can_header)); const uint8_t data_len = dlc_to_len[header.data_len_code]; if (pos + sizeof(can_header) + data_len > size) { // we don't have all the data for this message yet break; } if (calculate_checksum(&data[pos], sizeof(can_header) + data_len) != 0) { LOGE("Panda CAN checksum failed"); size = 0; can_reset_communications(); return false; } can_frame &canData = out_vec.emplace_back(); canData.busTime = 0; canData.address = header.addr; canData.src = header.bus + bus_offset; if (header.rejected) { canData.src += CAN_REJECTED_BUS_OFFSET; } if (header.returned) { canData.src += CAN_RETURNED_BUS_OFFSET; } canData.dat.assign((char *)&data[pos + sizeof(can_header)], data_len); pos += sizeof(can_header) + data_len; } // move the overflowing data to the beginning of the buffer for the next round memmove(data, &data[pos], size - pos); size -= pos; return true; } uint8_t Panda::calculate_checksum(uint8_t *data, uint32_t len) { uint8_t checksum = 0U; for (uint32_t i = 0U; i < len; i++) { checksum ^= data[i]; } return checksum; }