4.15. CAN

4.15.1. Module Files

4.15.1.1. Driver

  • src/app/driver/can/can.c (API, source)

  • src/app/driver/can/can.h (API, source)

  • src/app/driver/can/cbs/can_helper.c (API, source)

  • src/app/driver/can/cbs/can_helper.h (API, source)

  • src/app/driver/can/cbs/can_cbs.h (API, source)

4.15.1.2. Configuration

  • src/app/driver/config/can_cfg.c (API, source)

  • src/app/driver/config/can_cfg.h (API, source)

4.15.1.3. Unit Test

  • tests/unit/app/driver/can/test_can.c (API, source)

  • tests/unit/app/driver/config/test_can_cfg.c (API, source)

4.15.2. Description

In can_cfg.c, CAN messages are defined in two tables:

  • can_txMessages[] for messages to be sent.

  • can_rxMessages[] for messages to be received.

4.15.2.1. Messages to send

The sent message parameters are:

  • CAN ID of message to be sent.

  • data length code, number of bytes to send. Default 8, maximum 8.

  • repetition time, period of transmission in ms. Must be a multiple of 10.

  • repetition phase, delay for the first transmission. Avoids sending all messages with same period at the same time.

  • byte order, endianness (big or little endian) of CAN data.

  • callback function, pointer to the function that is called when the message has to be sent.

  • multiplexer, pointer to a number. This is used to multiplex data in CAN messages. A static variable must be defined to be used as multiplexer.

The data of the CAN message is divided into signals. Data for each signal is prepared within the callback function. The developer must implement the signals as needed by the application.

Two helper functions are defined for this task, CAN_TxSetMessageDataWithSignalData() and CAN_TxSetCanDataWithMessageData(). In the callback function, a uint64_t variable must be defined, which represents the CAN message. With the function CAN_TxSetMessageDataWithSignalData(), the signal data can be stored in the message data. The parameters are

  • pointer to 64-bit CAN message.

  • bit position in CAN message where the signal data must be stored.

  • length of the signal data in the CAN message.

  • signal data. If data is e.g. a float, it must be cast to an integer before being passed to the function.

  • endianness (big or little endian) of CAN data.

Once the CAN message is ready, the function CAN_TxSetCanDataWithMessageData() must be called. It will store the CAN message in the variable used by the low-level driver for the actual transmission.

The function CAN_PeriodicTransmit() is called every 10ms by the 10ms task. It parses all the elements of can_txMessages[]. If the time has been reached to send the messages, the corresponding callback function is called.

The message is then sent with the function CAN_DataSend(). The function CAN_DataSend() can also be used to send a CAN message directly anywhere else in the code.

4.15.2.2. Messages to receive

The received message parameters are:

  • CAN ID of message to be received.

  • data length code, number of bytes to receive. Default 8, maximum 8.

  • byte order, endianness (big or little endian) of CAN data.

  • callback function: pointer to the function that is called when the message is received. The data of the CAN message is available within this function.

A receive queue called ftsk_canRxQueue is used as shown in Queue handle for CAN receive and Supporting variables for the queue of the CAN receive module.

Listing 4.5 Queue handle for CAN receive
1QueueHandle_t ftsk_canRxQueue = NULL_PTR;
Listing 4.6 Supporting variables for the queue of the CAN receive module
1    static StaticQueue_t ftsk_canRxQueueStructure = {0}; /*!< structure for static data queue */
2    /**
3     * @brief   size of storage area for the CAN Rx queue
4     * @details The array that is used for the queue's storage area.
5     *          This must be at least
6     *          #FTSK_CAN_RX_QUEUE_LENGTH * #FTSK_CAN_RX_QUEUE_ITEM_SIZE
7     */
8    static uint8_t ftsk_canRxQueueStorageArea[FTSK_CAN_RX_QUEUE_LENGTH * FTSK_CAN_RX_QUEUE_ITEM_SIZE] = {0};

When CAN messages are received, the CAN interrupt callback calls CAN_RxInterrupt(). The message received is sent to the queue. The function CAN_ReadRxBuffer() is called every 1ms by the 1ms task. For each element in the queue, it checks if the CAN message ID matches an ID of the RX message list can_rxMessages[]. If this is the case, the corresponding callback function is called.

In the callback function, a uint64_t variable must be defined, which represents the CAN message. The helper function CAN_RxGetMessageDataFromCanData() MUST be called at the beginning. It copies the CAN data retrieved by the low level driver into the message variable. If necessary, the other helper function CAN_RxGetSignalDataFromMessageData() can be used to retrieve signal data easily from the message. The parameters are

  • 64-bit CAN message.

  • bit position in CAN message where the signal data must be stored.

  • length of the signal data in the CAN message.

  • pointer to signal data.

  • endianness (big or little endian) of CAN data.

The signal data is then available in the variable pointed to by the signal data pointer.

A receive queue was used because usually the developer needs to access the database in the callback of the receive function, but this must not be done in an interrupt routine. With the current implementation, the receive interrupt routine sends the received data to the queue. The CAN_ReadRxBuffer() function retrieves the messages from the queue and calls the callbacks outside of the interrupt routine.

4.15.2.3. Configuration in HALCoGen

64 messages boxes are available for each CAN interface. 32 of them are configured as transmit message boxes, the other half as receive message boxes (with HALCoGen), so the macro CAN_NR_OF_TX_MESSAGE_BOX is set to 32. It must be adapted if the number of transmit message boxes is changed.

When using the CAN interface CAN4, special care has to be taken because of a bug in HALCoGen. For more information, refer to HALCoGen tool documentation.

4.15.2.4. Callback definition

If a new file is needed for a new callback, it must be added in the directory ./src/app/driver/can/cbs. The corresponding header is the file can_cbs.h in the same directory. The new callback file must also be added to the wscript file in ./src/app/driver. For instance, if the file can_my_callback.c is added, the line

os.path.join("can", "cbs", "can_my_callback.c"),

must be added.