Skip to content

Commit

Permalink
pandad/SPI: ensure slave is in a consistent state (#32645)
Browse files Browse the repository at this point in the history
* maxout

* get ready for the next one

* really get ready

* much better

---------

Co-authored-by: Comma Device <device@comma.ai>
old-commit-hash: f8cb04e
  • Loading branch information
adeebshihadeh and Comma Device authored Jun 11, 2024
1 parent 0ee93c6 commit 50dd024
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 15 deletions.
5 changes: 5 additions & 0 deletions selfdrive/pandad/panda.cc
Original file line number Diff line number Diff line change
Expand Up @@ -221,6 +221,11 @@ bool Panda::can_receive(std::vector<can_frame>& out_vec) {
return false;
}

if (getenv("PANDAD_MAXOUT") != NULL) {
static uint8_t junk[RECV_SIZE];
handle->bulk_read(0xab, junk, RECV_SIZE - recv);
}

bool ret = true;
if (recv > 0) {
receive_buffer_size += recv;
Expand Down
45 changes: 31 additions & 14 deletions selfdrive/pandad/spi.cc
Original file line number Diff line number Diff line change
Expand Up @@ -50,8 +50,9 @@ class LockEx {

#define SPILOG(fn, fmt, ...) do { \
fn(fmt, ## __VA_ARGS__); \
fn(" %d / 0x%x / %d / %d", \
xfer_count, header.endpoint, header.tx_len, header.max_rx_len); \
fn(" %d / 0x%x / %d / %d / tx: %s", \
xfer_count, header.endpoint, header.tx_len, header.max_rx_len, \
util::hexdump(tx_buf, std::min((int)header.tx_len, 8)).c_str()); \
} while (0)

PandaSpiHandle::PandaSpiHandle(std::string serial) : PandaCommsHandle(serial) {
Expand Down Expand Up @@ -238,6 +239,7 @@ int PandaSpiHandle::spi_transfer_retry(uint8_t endpoint, uint8_t *tx_data, uint1
// due to full TX buffers
nack_count += 1;
if (nack_count > 3) {
SPILOG(LOGE, "NACK sleep %d", nack_count);
usleep(std::clamp(nack_count*10, 200, 2000));
}
}
Expand All @@ -256,14 +258,14 @@ int PandaSpiHandle::wait_for_ack(uint8_t ack, uint8_t tx, unsigned int timeout,
if (timeout == 0) {
timeout = SPI_ACK_TIMEOUT;
}
timeout = std::clamp(timeout, 100U, SPI_ACK_TIMEOUT);
timeout = std::clamp(timeout, 20U, SPI_ACK_TIMEOUT);

spi_ioc_transfer transfer = {
.tx_buf = (uint64_t)tx_buf,
.rx_buf = (uint64_t)rx_buf,
.len = length
.len = length,
};
tx_buf[0] = tx;
memset(tx_buf, tx, length);

while (true) {
int ret = lltransfer(transfer);
Expand All @@ -275,13 +277,13 @@ int PandaSpiHandle::wait_for_ack(uint8_t ack, uint8_t tx, unsigned int timeout,
if (rx_buf[0] == ack) {
break;
} else if (rx_buf[0] == SPI_NACK) {
SPILOG(LOGD, "SPI: got NACK");
SPILOG(LOGD, "SPI: got NACK, waiting for 0x%x", ack);
return SpiError::NACK;
}

// handle timeout
if (millis_since_boot() - start_millis > timeout) {
SPILOG(LOGW, "SPI: timed out waiting for ACK");
SPILOG(LOGW, "SPI: timed out waiting for ACK, waiting for 0x%x", ack);
return SpiError::ACK_TIMEOUT;
}
}
Expand Down Expand Up @@ -352,13 +354,13 @@ int PandaSpiHandle::spi_transfer(uint8_t endpoint, uint8_t *tx_data, uint16_t tx
ret = lltransfer(transfer);
if (ret < 0) {
SPILOG(LOGE, "SPI: failed to send header");
return ret;
goto fail;
}

// Wait for (N)ACK
ret = wait_for_ack(SPI_HACK, 0x11, timeout, 1);
if (ret < 0) {
return ret;
goto fail;
}

// Send data
Expand All @@ -370,38 +372,53 @@ int PandaSpiHandle::spi_transfer(uint8_t endpoint, uint8_t *tx_data, uint16_t tx
ret = lltransfer(transfer);
if (ret < 0) {
SPILOG(LOGE, "SPI: failed to send data");
return ret;
goto fail;
}

// Wait for (N)ACK
ret = wait_for_ack(SPI_DACK, 0x13, timeout, 3);
if (ret < 0) {
return ret;
goto fail;
}

// Read data
rx_data_len = *(uint16_t *)(rx_buf+1);
if (rx_data_len >= SPI_BUF_SIZE) {
SPILOG(LOGE, "SPI: RX data len larger than buf size %d", rx_data_len);
return -1;
goto fail;
}

transfer.len = rx_data_len + 1;
transfer.rx_buf = (uint64_t)(rx_buf + 2 + 1);
ret = lltransfer(transfer);
if (ret < 0) {
SPILOG(LOGE, "SPI: failed to read rx data");
return ret;
goto fail;
}
if (!check_checksum(rx_buf, rx_data_len + 4)) {
SPILOG(LOGE, "SPI: bad checksum");
return -1;
goto fail;
}

if (rx_data != NULL) {
memcpy(rx_data, rx_buf + 3, rx_data_len);
}

return rx_data_len;

fail:
// ensure slave is in a consistent state
// and ready for the next transfer
int nack_cnt = 0;
while (nack_cnt < 3) {
if (wait_for_ack(SPI_NACK, 0x14, 1, SPI_BUF_SIZE/2) == 0) {
nack_cnt += 1;
} else {
nack_cnt = 0;
}
}

if (ret > 0) ret = -1;
return ret;
}
#endif
2 changes: 1 addition & 1 deletion selfdrive/pandad/tests/test_pandad_spi.py
Original file line number Diff line number Diff line change
Expand Up @@ -97,7 +97,7 @@ def test_spi_corruption(self, subtests):
with subtests.test(msg="timing check", service=service):
edt = 1e3 / SERVICE_LIST[service].frequency
assert edt*0.9 < np.mean(dts) < edt*1.1
assert np.max(dts) < edt*20
assert np.max(dts) < edt*8
assert np.min(dts) < edt
assert len(dts) >= ((et-0.5)*SERVICE_LIST[service].frequency*0.8)

Expand Down

0 comments on commit 50dd024

Please sign in to comment.