#include "selfdrive/boardd/panda.h" #include #include #include #include #include "cereal/messaging/messaging.h" #include "selfdrive/common/gpio.h" #include "selfdrive/common/swaglog.h" #include "selfdrive/common/util.h" static int init_usb_ctx(libusb_context **context) { assert(context != nullptr); int err = libusb_init(context); if (err != 0) { LOGE("libusb initialization error"); return err; } #if LIBUSB_API_VERSION >= 0x01000106 libusb_set_option(*context, LIBUSB_OPTION_LOG_LEVEL, LIBUSB_LOG_LEVEL_INFO); #else libusb_set_debug(*context, 3); #endif return err; } Panda::Panda(std::string serial, uint32_t bus_offset) : bus_offset(bus_offset) { // init libusb ssize_t num_devices; libusb_device **dev_list = NULL; int err = init_usb_ctx(&ctx); if (err != 0) { goto fail; } // connect by serial num_devices = libusb_get_device_list(ctx, &dev_list); if (num_devices < 0) { goto fail; } for (size_t i = 0; i < num_devices; ++i) { libusb_device_descriptor desc; libusb_get_device_descriptor(dev_list[i], &desc); if (desc.idVendor == 0xbbaa && desc.idProduct == 0xddcc) { libusb_open(dev_list[i], &dev_handle); if (dev_handle == NULL) { goto fail; } unsigned char desc_serial[26] = { 0 }; int ret = libusb_get_string_descriptor_ascii(dev_handle, desc.iSerialNumber, desc_serial, std::size(desc_serial)); if (ret < 0) { goto fail; } usb_serial = std::string((char *)desc_serial, ret).c_str(); if (serial.empty() || serial == usb_serial) { break; } libusb_close(dev_handle); dev_handle = NULL; } } if (dev_handle == NULL) goto fail; libusb_free_device_list(dev_list, 1); dev_list = nullptr; if (libusb_kernel_driver_active(dev_handle, 0) == 1) { libusb_detach_kernel_driver(dev_handle, 0); } err = libusb_set_configuration(dev_handle, 1); if (err != 0) { goto fail; } err = libusb_claim_interface(dev_handle, 0); if (err != 0) { goto fail; } hw_type = get_hw_type(); assert((hw_type != cereal::PandaState::PandaType::WHITE_PANDA) && (hw_type != cereal::PandaState::PandaType::GREY_PANDA)); has_rtc = (hw_type == cereal::PandaState::PandaType::UNO) || (hw_type == cereal::PandaState::PandaType::DOS); return; fail: if (dev_list != NULL) { libusb_free_device_list(dev_list, 1); } cleanup(); throw std::runtime_error("Error connecting to panda"); } Panda::~Panda() { std::lock_guard lk(usb_lock); cleanup(); connected = false; } void Panda::cleanup() { if (dev_handle) { libusb_release_interface(dev_handle, 0); libusb_close(dev_handle); } if (ctx) { libusb_exit(ctx); } } std::vector Panda::list() { // init libusb ssize_t num_devices; libusb_context *context = NULL; libusb_device **dev_list = NULL; std::vector serials; int err = init_usb_ctx(&context); if (err != 0) { return serials; } num_devices = libusb_get_device_list(context, &dev_list); if (num_devices < 0) { LOGE("libusb can't get device list"); goto finish; } for (size_t i = 0; i < num_devices; ++i) { libusb_device *device = dev_list[i]; libusb_device_descriptor desc; libusb_get_device_descriptor(device, &desc); if (desc.idVendor == 0xbbaa && desc.idProduct == 0xddcc) { libusb_device_handle *handle = NULL; libusb_open(device, &handle); unsigned char desc_serial[26] = { 0 }; int ret = libusb_get_string_descriptor_ascii(handle, desc.iSerialNumber, desc_serial, std::size(desc_serial)); libusb_close(handle); if (ret < 0) { goto finish; } serials.push_back(std::string((char *)desc_serial, ret).c_str()); } } finish: if (dev_list != NULL) { libusb_free_device_list(dev_list, 1); } if (context) { libusb_exit(context); } return serials; } void Panda::handle_usb_issue(int err, const char func[]) { LOGE_100("usb error %d \"%s\" in %s", err, libusb_strerror((enum libusb_error)err), func); if (err == LIBUSB_ERROR_NO_DEVICE) { LOGE("lost connection"); connected = false; } // TODO: check other errors, is simply retrying okay? } int Panda::usb_write(uint8_t bRequest, uint16_t wValue, uint16_t wIndex, unsigned int timeout) { int err; const uint8_t bmRequestType = LIBUSB_ENDPOINT_OUT | LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE; if (!connected) { return LIBUSB_ERROR_NO_DEVICE; } std::lock_guard lk(usb_lock); do { err = libusb_control_transfer(dev_handle, bmRequestType, bRequest, wValue, wIndex, NULL, 0, timeout); if (err < 0) handle_usb_issue(err, __func__); } while (err < 0 && connected); return err; } int Panda::usb_read(uint8_t bRequest, uint16_t wValue, uint16_t wIndex, unsigned char *data, uint16_t wLength, unsigned int timeout) { int err; const uint8_t bmRequestType = LIBUSB_ENDPOINT_IN | LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE; if (!connected) { return LIBUSB_ERROR_NO_DEVICE; } std::lock_guard lk(usb_lock); do { err = libusb_control_transfer(dev_handle, bmRequestType, bRequest, wValue, wIndex, data, wLength, timeout); if (err < 0) handle_usb_issue(err, __func__); } while (err < 0 && connected); return err; } int Panda::usb_bulk_write(unsigned char endpoint, unsigned char* data, int length, unsigned int timeout) { int err; int transferred = 0; if (!connected) { return 0; } std::lock_guard lk(usb_lock); do { // Try sending can messages. If the receive buffer on the panda is full it will NAK // and libusb will try again. After 5ms, it will time out. We will drop the messages. err = libusb_bulk_transfer(dev_handle, endpoint, data, length, &transferred, timeout); if (err == LIBUSB_ERROR_TIMEOUT) { LOGW("Transmit buffer full"); break; } else if (err != 0 || length != transferred) { handle_usb_issue(err, __func__); } } while(err != 0 && connected); return transferred; } int Panda::usb_bulk_read(unsigned char endpoint, unsigned char* data, int length, unsigned int timeout) { int err; int transferred = 0; if (!connected) { return 0; } std::lock_guard lk(usb_lock); do { err = libusb_bulk_transfer(dev_handle, endpoint, data, length, &transferred, timeout); if (err == LIBUSB_ERROR_TIMEOUT) { break; // timeout is okay to exit, recv still happened } else if (err == LIBUSB_ERROR_OVERFLOW) { comms_healthy = false; LOGE_100("overflow got 0x%x", transferred); } else if (err != 0) { handle_usb_issue(err, __func__); } } while(err != 0 && connected); return transferred; } void Panda::set_safety_model(cereal::CarParams::SafetyModel safety_model, int safety_param) { usb_write(0xdc, (uint16_t)safety_model, safety_param); } void Panda::set_unsafe_mode(uint16_t unsafe_mode) { usb_write(0xdf, unsafe_mode, 0); } cereal::PandaState::PandaType Panda::get_hw_type() { unsigned char hw_query[1] = {0}; usb_read(0xc1, 0, 0, hw_query, 1); return (cereal::PandaState::PandaType)(hw_query[0]); } void Panda::set_rtc(struct tm sys_time) { // tm struct has year defined as years since 1900 usb_write(0xa1, (uint16_t)(1900 + sys_time.tm_year), 0); usb_write(0xa2, (uint16_t)(1 + sys_time.tm_mon), 0); usb_write(0xa3, (uint16_t)sys_time.tm_mday, 0); // usb_write(0xa4, (uint16_t)(1 + sys_time.tm_wday), 0); usb_write(0xa5, (uint16_t)sys_time.tm_hour, 0); usb_write(0xa6, (uint16_t)sys_time.tm_min, 0); usb_write(0xa7, (uint16_t)sys_time.tm_sec, 0); } struct tm Panda::get_rtc() { struct __attribute__((packed)) timestamp_t { uint16_t year; // Starts at 0 uint8_t month; uint8_t day; uint8_t weekday; uint8_t hour; uint8_t minute; uint8_t second; } rtc_time = {0}; usb_read(0xa0, 0, 0, (unsigned char*)&rtc_time, sizeof(rtc_time)); struct tm new_time = { 0 }; new_time.tm_year = rtc_time.year - 1900; // tm struct has year defined as years since 1900 new_time.tm_mon = rtc_time.month - 1; new_time.tm_mday = rtc_time.day; new_time.tm_hour = rtc_time.hour; new_time.tm_min = rtc_time.minute; new_time.tm_sec = rtc_time.second; return new_time; } void Panda::set_fan_speed(uint16_t fan_speed) { usb_write(0xb1, fan_speed, 0); } uint16_t Panda::get_fan_speed() { uint16_t fan_speed_rpm = 0; usb_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) { usb_write(0xb0, ir_pwr, 0); } health_t Panda::get_state() { health_t health {0}; usb_read(0xd2, 0, 0, (unsigned char*)&health, sizeof(health)); return health; } void Panda::set_loopback(bool loopback) { usb_write(0xe5, loopback, 0); } std::optional> Panda::get_firmware_version() { std::vector fw_sig_buf(128); int read_1 = usb_read(0xd3, 0, 0, &fw_sig_buf[0], 64); int read_2 = usb_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 = usb_read(0xd0, 0, 0, (uint8_t*)serial_buf, 16); return err >= 0 ? std::make_optional(serial_buf) : std::nullopt; } void Panda::set_power_saving(bool power_saving) { usb_write(0xe7, power_saving, 0); } void Panda::set_usb_power_mode(cereal::PeripheralState::UsbPowerMode power_mode) { usb_write(0xe6, (uint16_t)power_mode, 0); } void Panda::send_heartbeat() { usb_write(0xf3, 1, 0); } void Panda::can_send(capnp::List::Reader can_data_list) { send.resize(4 * can_data_list.size()); uint32_t msg_cnt = 0; for (int i = 0; i < can_data_list.size(); i++) { auto cmsg = can_data_list[i]; // check if the message is intended for this panda uint8_t bus = cmsg.getSrc(); if (bus < bus_offset || bus >= (bus_offset + PANDA_BUS_CNT)) { continue; } if (cmsg.getAddress() >= 0x800) { // extended send[msg_cnt*4] = (cmsg.getAddress() << 3) | 5; } else { // normal send[msg_cnt*4] = (cmsg.getAddress() << 21) | 1; } auto can_data = cmsg.getDat(); assert(can_data.size() <= 8); send[msg_cnt*4+1] = can_data.size() | ((bus - bus_offset) << 4); memcpy(&send[msg_cnt*4+2], can_data.begin(), can_data.size()); msg_cnt++; } usb_bulk_write(3, (unsigned char*)send.data(), msg_cnt * 0x10, 5); } bool Panda::can_receive(std::vector& out_vec) { uint32_t data[RECV_SIZE/4]; int recv = usb_bulk_read(0x81, (unsigned char*)data, RECV_SIZE); // Not sure if this can happen if (recv < 0) recv = 0; if (recv == RECV_SIZE) { LOGW("Receive buffer full"); } if (!comms_healthy) { return false; } // Append to the end of the out_vec, such that we can pass it to multiple pandas // We already insert space for all the messages here for speed size_t num_msg = recv / 0x10; out_vec.reserve(out_vec.size() + num_msg); // Populate messages for (int i = 0; i < num_msg; i++) { can_frame canData; if (data[i*4] & 4) { // extended canData.address = data[i*4] >> 3; //printf("got extended: %x\n", data[i*4] >> 3); } else { // normal canData.address = data[i*4] >> 21; } canData.busTime = data[i*4+1] >> 16; int len = data[i*4+1] & 0xF; canData.dat.assign((char *)&data[i*4+2], len); canData.src = ((data[i*4+1] >> 4) & 0xff) + bus_offset; // add to vector out_vec.push_back(canData); } return true; }