rusefi/firmware/console/binary/serial_can.cpp

419 lines
13 KiB
C++
Raw Normal View History

2020-09-07 09:10:39 -07:00
/**
* @file serial_can.cpp
*
* This code is a bridge between a serial streaming used by TS and a packet-frame CAN-bus, using the ISO-TP protocol.
* ISO 15765-2, or ISO-TP (Transport Layer), which is an international standard for sending data packets over a CAN-Bus.
* https://en.wikipedia.org/wiki/ISO_15765-2
*
* @date Aug 1, 2020
* @author andreika <prometheus.pcb@gmail.com>
* @author Andrey Belomutskiy, (c) 2012-2020
*/
#include "pch.h"
2022-09-07 12:56:45 -07:00
2020-09-07 09:10:39 -07:00
#if EFI_UNIT_TEST
#define PRINT printf
#define PRINT_EOL "\n"
#else
#define PRINT efiPrintf
#define PRINT_EOL ""
#endif
// todo: this file is asking to improve conditional compilation. unit_tests and cypress/kinetis are both special cases
#if HAL_USE_CAN || EFI_UNIT_TEST
2021-12-08 14:01:53 -08:00
#include "serial_can.h"
#include "can.h"
#include "can_msg_tx.h"
#endif // HAL_USE_CAN || EFI_UNIT_TEST
2021-12-08 14:01:53 -08:00
#if HAL_USE_CAN
static CanStreamer streamer;
2021-12-01 08:13:36 -08:00
static CanStreamerState state(&streamer);
static CanTsListener listener;
2021-12-08 09:01:28 -08:00
#endif // HAL_USE_CAN
2020-09-07 09:10:39 -07:00
#if HAL_USE_CAN || EFI_UNIT_TEST
2021-12-01 08:13:36 -08:00
int CanStreamerState::sendFrame(const IsoTpFrameHeader & header, const uint8_t *data, int num, can_sysinterval_t timeout) {
int dlc = 8; // standard 8 bytes
2023-02-20 08:29:25 -08:00
CanTxMessage txmsg(CanCategory::SERIAL, CAN_ECU_SERIAL_TX_ID, dlc, 0, false);
2020-09-07 09:10:39 -07:00
// fill the frame data according to the CAN-TP protocol (ISO 15765-2)
2021-12-01 08:13:36 -08:00
txmsg[0] = (uint8_t)((header.frameType & 0xf) << 4);
2020-09-07 09:10:39 -07:00
int offset, maxNumBytes;
switch (header.frameType) {
case ISO_TP_FRAME_SINGLE:
offset = 1;
2021-12-01 08:13:36 -08:00
maxNumBytes = minI(header.numBytes, dlc - offset);
txmsg[0] |= maxNumBytes;
2020-09-07 09:10:39 -07:00
break;
case ISO_TP_FRAME_FIRST:
2021-12-01 08:13:36 -08:00
txmsg[0] |= (header.numBytes >> 8) & 0xf;
txmsg[1] = (uint8_t)(header.numBytes & 0xff);
2020-09-07 09:10:39 -07:00
offset = 2;
2021-12-01 08:13:36 -08:00
maxNumBytes = minI(header.numBytes, dlc - offset);
2020-09-07 09:10:39 -07:00
break;
case ISO_TP_FRAME_CONSECUTIVE:
2021-12-01 08:13:36 -08:00
txmsg[0] |= header.index & 0xf;
2020-09-07 09:10:39 -07:00
offset = 1;
2021-12-01 08:13:36 -08:00
// todo: is it correct?
maxNumBytes = dlc - offset;
2020-09-07 09:10:39 -07:00
break;
case ISO_TP_FRAME_FLOW_CONTROL:
2021-12-01 08:13:36 -08:00
txmsg[0] |= header.fcFlag & 0xf;
txmsg[1] = (uint8_t)(header.blockSize);
txmsg[2] = (uint8_t)(header.separationTime);
2020-09-07 09:10:39 -07:00
offset = 3;
maxNumBytes = 0; // no data is sent with 'flow control' frame
break;
default:
// bad frame type
return 0;
2020-09-07 09:10:39 -07:00
}
int numBytes = minI(maxNumBytes, num);
// copy the contents
if (data != nullptr) {
for (int i = 0; i < numBytes; i++) {
2021-12-01 08:13:36 -08:00
txmsg[i + offset] = data[i];
2020-09-07 09:10:39 -07:00
}
}
2021-12-01 08:13:36 -08:00
2020-09-07 09:10:39 -07:00
// send the frame!
2021-12-01 08:13:36 -08:00
if (streamer->transmit(CAN_ANY_MAILBOX, &txmsg, timeout) == CAN_MSG_OK)
2020-09-07 09:10:39 -07:00
return numBytes;
return 0;
}
// returns the number of copied bytes
2021-12-01 08:13:36 -08:00
int CanStreamerState::receiveFrame(CANRxFrame *rxmsg, uint8_t *buf, int num, can_sysinterval_t timeout) {
2020-09-07 09:10:39 -07:00
if (rxmsg == nullptr || rxmsg->DLC < 1)
return 0;
2021-12-05 18:20:54 -08:00
engine->pauseCANdueToSerial = true;
2020-09-07 09:10:39 -07:00
int frameType = (rxmsg->data8[0] >> 4) & 0xf;
int numBytesAvailable, frameIdx;
uint8_t *srcBuf = rxmsg->data8;
switch (frameType) {
case ISO_TP_FRAME_SINGLE:
numBytesAvailable = rxmsg->data8[0] & 0xf;
srcBuf = rxmsg->data8 + 1;
this->waitingForNumBytes = -1;
break;
case ISO_TP_FRAME_FIRST:
this->waitingForNumBytes = ((rxmsg->data8[0] & 0xf) << 8) | rxmsg->data8[1];
this->waitingForFrameIndex = 1;
numBytesAvailable = minI(this->waitingForNumBytes, 6);
srcBuf = rxmsg->data8 + 2;
break;
case ISO_TP_FRAME_CONSECUTIVE:
frameIdx = rxmsg->data8[0] & 0xf;
if (this->waitingForNumBytes < 0 || this->waitingForFrameIndex != frameIdx) {
// todo: that's an abnormal situation, and we probably should react?
return 0;
}
numBytesAvailable = minI(this->waitingForNumBytes, 7);
srcBuf = rxmsg->data8 + 1;
this->waitingForFrameIndex = (this->waitingForFrameIndex + 1) & 0xf;
break;
case ISO_TP_FRAME_FLOW_CONTROL:
// todo: currently we just ignore the FC frame
return 0;
default:
// bad frame type
return 0;
2020-09-07 09:10:39 -07:00
}
#if defined(TS_CAN_DEVICE_SHORT_PACKETS_IN_ONE_FRAME)
if (frameType == ISO_TP_FRAME_SINGLE) {
// restore the CRC on the whole packet
2021-12-01 08:13:36 -08:00
uint32_t crc = crc32((void *) srcBuf, numBytesAvailable);
2020-09-07 09:10:39 -07:00
// we need a separate buffer for crc because srcBuf may not be word-aligned for direct copy
uint8_t crcBuffer[sizeof(uint32_t)];
*(uint32_t *) (crcBuffer) = SWAP_UINT32(crc);
2021-12-01 08:13:36 -08:00
// now set the packet size
*(uint16_t *) tmpRxBuf = SWAP_UINT16(numBytesAvailable);
2020-09-07 09:10:39 -07:00
// copy the data
if (numBytesAvailable > 0)
2021-12-01 08:13:36 -08:00
memcpy(tmpRxBuf + sizeof(uint16_t), srcBuf, numBytesAvailable);
// copy the crc to the end
memcpy(tmpRxBuf + sizeof(uint16_t) + numBytesAvailable, crcBuffer, sizeof(crcBuffer));
// use the reconstructed tmp buffer as a source buffer
srcBuf = tmpRxBuf;
// we added the 16-bit size & 32-bit crc bytes
numBytesAvailable += sizeof(uint16_t) + sizeof(crcBuffer);
2020-09-07 09:10:39 -07:00
}
#endif /* TS_CAN_DEVICE_SHORT_PACKETS_IN_ONE_FRAME */
int numBytesToCopy = minI(num, numBytesAvailable);
if (buf != nullptr) {
memcpy(buf, srcBuf, numBytesToCopy);
}
srcBuf += numBytesToCopy;
waitingForNumBytes -= numBytesAvailable;
2020-09-07 09:10:39 -07:00
numBytesAvailable -= numBytesToCopy;
// if there are some more bytes left, we save them for the next time
for (int i = 0; i < numBytesAvailable; i++) {
rxFifoBuf.put(srcBuf[i]);
}
// according to the specs, we need to acknowledge the received multi-frame start frame
if (frameType == ISO_TP_FRAME_FIRST) {
IsoTpFrameHeader header;
header.frameType = ISO_TP_FRAME_FLOW_CONTROL;
header.fcFlag = 0; // = "continue to send"
header.blockSize = 0; // = the remaining "frames" to be sent without flow control or delay
header.separationTime = 0; // = wait 0 milliseconds, send immediately
2021-12-01 08:13:36 -08:00
sendFrame(header, nullptr, 0, timeout);
2020-09-07 09:10:39 -07:00
}
return numBytesToCopy;
}
2021-12-01 08:13:36 -08:00
int CanStreamerState::sendDataTimeout(const uint8_t *txbuf, int numBytes, can_sysinterval_t timeout) {
2020-09-07 09:10:39 -07:00
int offset = 0;
2021-12-01 08:13:36 -08:00
if (engineConfiguration->verboseIsoTp) {
PRINT("*** INFO: sendDataTimeout %d" PRINT_EOL, numBytes);
}
2021-12-01 08:13:36 -08:00
if (numBytes < 1)
return 0;
2020-09-07 09:10:39 -07:00
// 1 frame
if (numBytes <= 7) {
IsoTpFrameHeader header;
header.frameType = ISO_TP_FRAME_SINGLE;
header.numBytes = numBytes;
2021-12-01 08:13:36 -08:00
return sendFrame(header, txbuf, numBytes, timeout);
2020-09-07 09:10:39 -07:00
}
// multiple frames
2021-12-01 08:13:36 -08:00
// send the first header frame (FF)
2020-09-07 09:10:39 -07:00
IsoTpFrameHeader header;
header.frameType = ISO_TP_FRAME_FIRST;
header.numBytes = numBytes;
2021-12-01 08:13:36 -08:00
int numSent = sendFrame(header, txbuf + offset, numBytes, timeout);
2020-09-07 09:10:39 -07:00
offset += numSent;
numBytes -= numSent;
int totalNumSent = numSent;
2021-12-01 08:13:36 -08:00
// get a flow control (FC) frame
#if !EFI_UNIT_TEST // todo: add FC to unit-tests?
2020-09-07 09:10:39 -07:00
CANRxFrame rxmsg;
2021-12-01 08:13:36 -08:00
for (int numFcReceived = 0; ; numFcReceived++) {
if (streamer->receive(CAN_ANY_MAILBOX, &rxmsg, timeout) != CAN_MSG_OK) {
#ifdef SERIAL_CAN_DEBUG
PRINT("*** ERROR: CAN Flow Control frame not received" PRINT_EOL);
2021-12-01 08:13:36 -08:00
#endif /* SERIAL_CAN_DEBUG */
//warning(ObdCode::CUSTOM_ERR_CAN_COMMUNICATION, "CAN Flow Control frame not received");
2021-12-01 08:13:36 -08:00
return 0;
}
receiveFrame(&rxmsg, nullptr, 0, timeout);
int flowStatus = rxmsg.data8[0] & 0xf;
// if something is not ok
if (flowStatus != CAN_FLOW_STATUS_OK) {
// if the receiver is not ready yet and asks to wait for the next FC frame (give it 3 attempts)
if (flowStatus == CAN_FLOW_STATUS_WAIT_MORE && numFcReceived < 3) {
continue;
}
#ifdef SERIAL_CAN_DEBUG
efiPrintf("*** ERROR: CAN Flow Control mode not supported");
#endif /* SERIAL_CAN_DEBUG */
//warning(ObdCode::CUSTOM_ERR_CAN_COMMUNICATION, "CAN Flow Control mode not supported");
2021-12-01 08:13:36 -08:00
return 0;
}
int blockSize = rxmsg.data8[1];
int minSeparationTime = rxmsg.data8[2];
if (blockSize != 0 || minSeparationTime != 0) {
// todo: process other Flow Control fields (see ISO 15765-2)
#ifdef SERIAL_CAN_DEBUG
efiPrintf("*** ERROR: CAN Flow Control fields not supported");
#endif /* SERIAL_CAN_DEBUG */
//warning(ObdCode::CUSTOM_ERR_CAN_COMMUNICATION, "CAN Flow Control fields not supported");
2021-12-01 08:13:36 -08:00
}
break;
2020-09-07 09:10:39 -07:00
}
#endif /* EFI_UNIT_TEST */
2020-09-07 09:10:39 -07:00
// send the rest of the data
int idx = 1;
while (numBytes > 0) {
int len = minI(numBytes, 7);
// send the consecutive frames
IsoTpFrameHeader header;
header.frameType = ISO_TP_FRAME_CONSECUTIVE;
header.index = ((idx++) & 0x0f);
2021-12-01 08:13:36 -08:00
header.numBytes = len;
int numSent = sendFrame(header, txbuf + offset, len, timeout);
2020-09-07 09:10:39 -07:00
if (numSent < 1)
break;
totalNumSent += numSent;
offset += numSent;
numBytes -= numSent;
}
return totalNumSent;
}
int CanStreamerState::getDataFromFifo(uint8_t *rxbuf, size_t &numBytes) {
if (rxFifoBuf.isEmpty())
return 0;
int numReadFromFifo = minI(numBytes, rxFifoBuf.getCount());
// move bytes from the FIFO buffer
int i;
for (i = 0; !rxFifoBuf.isEmpty() && i < numReadFromFifo; i++) {
rxbuf[i] = rxFifoBuf.get();
numBytes--;
}
return i;
}
2021-12-01 08:13:36 -08:00
can_msg_t CanStreamerState::streamAddToTxTimeout(size_t *np, const uint8_t *txbuf, can_sysinterval_t timeout) {
2020-09-07 09:10:39 -07:00
int numBytes = *np;
int offset = 0;
if (engineConfiguration->verboseIsoTp) {
PRINT("*** INFO: streamAddToTxTimeout adding %d, in buffer %d" PRINT_EOL, numBytes, txFifoBuf.getCount());
}
// we send here only if the TX FIFO buffer is getting overflowed
while (numBytes >= txFifoBuf.getSize() - txFifoBuf.getCount()) {
int numBytesToAdd = txFifoBuf.getSize() - txFifoBuf.getCount();
txFifoBuf.put(txbuf + offset, numBytesToAdd);
2021-12-01 08:13:36 -08:00
int numSent = sendDataTimeout((const uint8_t *)txFifoBuf.getElements(), txFifoBuf.getCount(), timeout);
if (engineConfiguration->verboseIsoTp) {
PRINT("*** INFO: streamAddToTxTimeout numBytesToAdd %d / numSent %d / numBytes %d" PRINT_EOL, numBytesToAdd, numSent, numBytes);
}
2020-09-07 09:10:39 -07:00
if (numSent < 1)
break;
2021-12-01 08:13:36 -08:00
txFifoBuf.clear();
offset += numBytesToAdd;
numBytes -= numBytesToAdd;
2020-09-07 09:10:39 -07:00
}
if (engineConfiguration->verboseIsoTp) {
PRINT("*** INFO: streamAddToTxTimeout remaining goes to buffer %d" PRINT_EOL, numBytes);
}
2020-09-07 09:10:39 -07:00
// now we put the rest on hold
2021-12-01 08:13:36 -08:00
txFifoBuf.put(txbuf + offset, numBytes);
2020-09-07 09:10:39 -07:00
if (engineConfiguration->verboseIsoTp) {
PRINT("*** INFO: in buffer %d" PRINT_EOL, txFifoBuf.getCount());
}
2021-12-01 08:13:36 -08:00
return CAN_MSG_OK;
2020-09-07 09:10:39 -07:00
}
2021-12-01 08:13:36 -08:00
can_msg_t CanStreamerState::streamFlushTx(can_sysinterval_t timeout) {
int numSent = sendDataTimeout((const uint8_t *)txFifoBuf.getElements(), txFifoBuf.getCount(), timeout);
if (numSent != txFifoBuf.getCount()) {
//warning(ObdCode::CUSTOM_ERR_CAN_COMMUNICATION, "CAN sendDataTimeout() problems");
2021-12-01 08:13:36 -08:00
}
txFifoBuf.clear();
2020-09-07 09:10:39 -07:00
2021-12-01 08:13:36 -08:00
return CAN_MSG_OK;
2020-09-07 09:10:39 -07:00
}
2021-12-01 08:13:36 -08:00
can_msg_t CanStreamerState::streamReceiveTimeout(size_t *np, uint8_t *rxbuf, can_sysinterval_t timeout) {
2023-03-16 10:09:36 -07:00
size_t availableBufferSpace = *np;
2021-12-01 08:13:36 -08:00
2020-09-07 09:10:39 -07:00
// first, fill the data from the stored buffer (saved from the previous CAN frame)
2023-03-16 10:09:36 -07:00
int receivedSoFar = getDataFromFifo(rxbuf, availableBufferSpace);
2020-09-07 09:10:39 -07:00
// if even more data is needed, then we receive more CAN frames
2023-03-16 10:09:36 -07:00
while (availableBufferSpace > 0) {
2020-09-07 09:10:39 -07:00
CANRxFrame rxmsg;
2021-12-01 08:13:36 -08:00
if (streamer->receive(CAN_ANY_MAILBOX, &rxmsg, timeout) == CAN_MSG_OK) {
2023-03-16 10:09:36 -07:00
int numReceived = receiveFrame(&rxmsg, rxbuf + receivedSoFar, availableBufferSpace, timeout);
2021-12-01 08:13:36 -08:00
2020-09-07 09:10:39 -07:00
if (numReceived < 1)
break;
2023-03-16 10:09:36 -07:00
availableBufferSpace -= numReceived;
2023-03-16 09:54:59 -07:00
receivedSoFar += numReceived;
2021-12-01 08:13:36 -08:00
} else {
break;
2020-09-07 09:10:39 -07:00
}
}
2023-03-16 10:09:36 -07:00
*np -= availableBufferSpace;
2021-12-01 08:13:36 -08:00
#ifdef SERIAL_CAN_DEBUG
2023-03-16 10:09:36 -07:00
efiPrintf("* ret: %d %d (%d)", i, *np, availableBufferSpace);
2023-03-16 09:54:59 -07:00
for (int j = 0; j < receivedSoFar; j++) {
2021-12-01 08:13:36 -08:00
efiPrintf("* [%d]: %02x", j, rxbuf[j]);
}
#endif /* SERIAL_CAN_DEBUG */
return CAN_MSG_OK;
}
2022-08-09 10:12:51 -07:00
static int isoTpPacketCounter = 0;
2021-12-01 08:13:36 -08:00
2022-08-09 10:12:51 -07:00
/**
* incoming data main entry point
*/
2021-12-01 08:13:36 -08:00
void CanTsListener::decodeFrame(const CANRxFrame& frame, efitick_t /*nowNt*/) {
2022-08-09 10:12:51 -07:00
// CAN ID filtering happens in base class, by the time we are here we know it's the CAN_ECU_SERIAL_RX_ID packet
2021-12-01 08:13:36 -08:00
// todo: what if the FIFO is full?
CanRxMessage msg(frame);
2022-08-09 10:12:51 -07:00
if (engineConfiguration->verboseIsoTp) {
PRINT("*** INFO: CanTsListener decodeFrame %d" PRINT_EOL, isoTpPacketCounter++);
}
2021-12-01 08:13:36 -08:00
if (!rxFifo.put(msg)) {
warning(ObdCode::CUSTOM_ERR_CAN_COMMUNICATION, "CAN sendDataTimeout() problems");
2021-12-01 08:13:36 -08:00
}
}
2021-12-08 09:01:28 -08:00
#if HAL_USE_CAN
2021-12-01 08:13:36 -08:00
void CanStreamer::init() {
registerCanListener(listener);
}
can_msg_t CanStreamer::transmit(canmbx_t /*mailbox*/, const CanTxMessage */*ctfp*/, can_sysinterval_t /*timeout*/) {
// we do nothing here - see CanTxMessage::~CanTxMessage()
return CAN_MSG_OK;
}
can_msg_t CanStreamer::receive(canmbx_t /*mailbox*/, CANRxFrame *crfp, can_sysinterval_t timeout) {
// see CanTsListener and processCanRxMessage()
CanRxMessage msg;
if (listener.get(msg, timeout)) {
*crfp = msg.frame;
return CAN_MSG_OK;
}
return CAN_MSG_TIMEOUT;
}
void canStreamInit(void) {
streamer.init();
}
msg_t canStreamAddToTxTimeout(size_t *np, const uint8_t *txbuf, sysinterval_t timeout) {
return state.streamAddToTxTimeout(np, txbuf, timeout);
}
msg_t canStreamFlushTx(sysinterval_t timeout) {
return state.streamFlushTx(timeout);
}
2023-03-16 10:27:22 -07:00
// np uses in/out parameter approach. Yes ChibiOS does same but still evil!
// in entry: number of data frames to receive
// on exit the number of frames actually received
2021-12-01 08:13:36 -08:00
msg_t canStreamReceiveTimeout(size_t *np, uint8_t *rxbuf, sysinterval_t timeout) {
return state.streamReceiveTimeout(np, rxbuf, timeout);
2020-09-07 09:10:39 -07:00
}
2021-12-01 08:13:36 -08:00
#endif /* HAL_USE_CAN */
#endif // HAL_USE_CAN || EFI_UNIT_TEST