123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484 |
- /*
- * Copyright (c) 2019 Vestas Wind Systems A/S
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <zephyr.h>
- #include <drivers/can.h>
- #include <init.h>
- #include <sys/util.h>
- #include <canopennode.h>
- #define LOG_LEVEL CONFIG_CANOPEN_LOG_LEVEL
- #include <logging/log.h>
- LOG_MODULE_REGISTER(canopen_driver);
- K_KERNEL_STACK_DEFINE(canopen_tx_workq_stack,
- CONFIG_CANOPENNODE_TX_WORKQUEUE_STACK_SIZE);
- struct k_work_q canopen_tx_workq;
- struct canopen_tx_work_container {
- struct k_work work;
- CO_CANmodule_t *CANmodule;
- };
- struct canopen_tx_work_container canopen_tx_queue;
- K_MUTEX_DEFINE(canopen_send_mutex);
- K_MUTEX_DEFINE(canopen_emcy_mutex);
- K_MUTEX_DEFINE(canopen_co_mutex);
- inline void canopen_send_lock(void)
- {
- k_mutex_lock(&canopen_send_mutex, K_FOREVER);
- }
- inline void canopen_send_unlock(void)
- {
- k_mutex_unlock(&canopen_send_mutex);
- }
- inline void canopen_emcy_lock(void)
- {
- k_mutex_lock(&canopen_emcy_mutex, K_FOREVER);
- }
- inline void canopen_emcy_unlock(void)
- {
- k_mutex_unlock(&canopen_emcy_mutex);
- }
- inline void canopen_od_lock(void)
- {
- k_mutex_lock(&canopen_co_mutex, K_FOREVER);
- }
- inline void canopen_od_unlock(void)
- {
- k_mutex_unlock(&canopen_co_mutex);
- }
- static void canopen_detach_all_rx_filters(CO_CANmodule_t *CANmodule)
- {
- uint16_t i;
- if (!CANmodule || !CANmodule->rx_array || !CANmodule->configured) {
- return;
- }
- for (i = 0U; i < CANmodule->rx_size; i++) {
- if (CANmodule->rx_array[i].filter_id != CAN_NO_FREE_FILTER) {
- can_detach(CANmodule->dev,
- CANmodule->rx_array[i].filter_id);
- CANmodule->rx_array[i].filter_id = CAN_NO_FREE_FILTER;
- }
- }
- }
- static void canopen_rx_isr_callback(struct zcan_frame *msg, void *arg)
- {
- CO_CANrx_t *buffer = (CO_CANrx_t *)arg;
- CO_CANrxMsg_t rxMsg;
- if (!buffer || !buffer->pFunct) {
- LOG_ERR("failed to process CAN rx isr callback");
- return;
- }
- rxMsg.ident = msg->id;
- rxMsg.DLC = msg->dlc;
- memcpy(rxMsg.data, msg->data, msg->dlc);
- buffer->pFunct(buffer->object, &rxMsg);
- }
- static void canopen_tx_isr_callback(uint32_t error_flags, void *arg)
- {
- CO_CANmodule_t *CANmodule = arg;
- if (!CANmodule) {
- LOG_ERR("failed to process CAN tx isr callback");
- return;
- }
- if (error_flags == CAN_TX_OK) {
- CANmodule->first_tx_msg = false;
- }
- k_work_submit_to_queue(&canopen_tx_workq, &canopen_tx_queue.work);
- }
- static void canopen_tx_retry(struct k_work *item)
- {
- struct canopen_tx_work_container *container =
- CONTAINER_OF(item, struct canopen_tx_work_container, work);
- CO_CANmodule_t *CANmodule = container->CANmodule;
- struct zcan_frame msg;
- CO_CANtx_t *buffer;
- int err;
- uint16_t i;
- CO_LOCK_CAN_SEND();
- for (i = 0; i < CANmodule->tx_size; i++) {
- buffer = &CANmodule->tx_array[i];
- if (buffer->bufferFull) {
- msg.id_type = CAN_STANDARD_IDENTIFIER;
- msg.id = buffer->ident;
- msg.dlc = buffer->DLC;
- msg.rtr = (buffer->rtr ? 1 : 0);
- memcpy(msg.data, buffer->data, buffer->DLC);
- err = can_send(CANmodule->dev, &msg, K_NO_WAIT,
- canopen_tx_isr_callback, CANmodule);
- if (err == CAN_TIMEOUT) {
- break;
- } else if (err != CAN_TX_OK) {
- LOG_ERR("failed to send CAN frame (err %d)",
- err);
- CO_errorReport(CANmodule->em,
- CO_EM_GENERIC_SOFTWARE_ERROR,
- CO_EMC_COMMUNICATION, 0);
- }
- buffer->bufferFull = false;
- }
- }
- CO_UNLOCK_CAN_SEND();
- }
- void CO_CANsetConfigurationMode(void *CANdriverState)
- {
- /* No operation */
- }
- void CO_CANsetNormalMode(CO_CANmodule_t *CANmodule)
- {
- CANmodule->CANnormal = true;
- }
- CO_ReturnError_t CO_CANmodule_init(CO_CANmodule_t *CANmodule,
- void *CANdriverState,
- CO_CANrx_t rxArray[], uint16_t rxSize,
- CO_CANtx_t txArray[], uint16_t txSize,
- uint16_t CANbitRate)
- {
- struct canopen_context *ctx = (struct canopen_context *)CANdriverState;
- uint16_t i;
- int err;
- LOG_DBG("rxSize = %d, txSize = %d", rxSize, txSize);
- if (!CANmodule || !rxArray || !txArray || !CANdriverState) {
- LOG_ERR("failed to initialize CAN module");
- return CO_ERROR_ILLEGAL_ARGUMENT;
- }
- if (rxSize > CONFIG_CAN_MAX_FILTER) {
- LOG_ERR("insufficient number of concurrent CAN RX filters"
- " (needs %d, %d available)", rxSize,
- CONFIG_CAN_MAX_FILTER);
- return CO_ERROR_OUT_OF_MEMORY;
- } else if (rxSize < CONFIG_CAN_MAX_FILTER) {
- LOG_DBG("excessive number of concurrent CAN RX filters enabled"
- " (needs %d, %d available)", rxSize,
- CONFIG_CAN_MAX_FILTER);
- }
- canopen_detach_all_rx_filters(CANmodule);
- canopen_tx_queue.CANmodule = CANmodule;
- CANmodule->dev = ctx->dev;
- CANmodule->rx_array = rxArray;
- CANmodule->rx_size = rxSize;
- CANmodule->tx_array = txArray;
- CANmodule->tx_size = txSize;
- CANmodule->CANnormal = false;
- CANmodule->first_tx_msg = true;
- CANmodule->errors = 0;
- CANmodule->em = NULL;
- for (i = 0U; i < rxSize; i++) {
- rxArray[i].ident = 0U;
- rxArray[i].pFunct = NULL;
- rxArray[i].filter_id = CAN_NO_FREE_FILTER;
- }
- for (i = 0U; i < txSize; i++) {
- txArray[i].bufferFull = false;
- }
- err = can_set_bitrate(CANmodule->dev, KHZ(CANbitRate), 0);
- if (err) {
- LOG_ERR("failed to configure CAN bitrate (err %d)", err);
- return CO_ERROR_ILLEGAL_ARGUMENT;
- }
- err = can_set_mode(CANmodule->dev, CAN_NORMAL_MODE);
- if (err) {
- LOG_ERR("failed to configure CAN interface (err %d)", err);
- return CO_ERROR_ILLEGAL_ARGUMENT;
- }
- CANmodule->configured = true;
- return CO_ERROR_NO;
- }
- void CO_CANmodule_disable(CO_CANmodule_t *CANmodule)
- {
- int err;
- if (!CANmodule || !CANmodule->dev) {
- return;
- }
- canopen_detach_all_rx_filters(CANmodule);
- err = can_configure(CANmodule->dev, CAN_SILENT_MODE, 0);
- if (err) {
- LOG_ERR("failed to disable CAN interface (err %d)", err);
- }
- }
- uint16_t CO_CANrxMsg_readIdent(const CO_CANrxMsg_t *rxMsg)
- {
- return rxMsg->ident;
- }
- CO_ReturnError_t CO_CANrxBufferInit(CO_CANmodule_t *CANmodule, uint16_t index,
- uint16_t ident, uint16_t mask, bool_t rtr,
- void *object,
- CO_CANrxBufferCallback_t pFunct)
- {
- struct zcan_filter filter;
- CO_CANrx_t *buffer;
- if (CANmodule == NULL) {
- return CO_ERROR_ILLEGAL_ARGUMENT;
- }
- if (!pFunct || (index >= CANmodule->rx_size)) {
- LOG_ERR("failed to initialize CAN rx buffer, illegal argument");
- CO_errorReport(CANmodule->em, CO_EM_GENERIC_SOFTWARE_ERROR,
- CO_EMC_SOFTWARE_INTERNAL, 0);
- return CO_ERROR_ILLEGAL_ARGUMENT;
- }
- buffer = &CANmodule->rx_array[index];
- buffer->object = object;
- buffer->pFunct = pFunct;
- filter.id_type = CAN_STANDARD_IDENTIFIER;
- filter.id = ident;
- filter.id_mask = mask;
- filter.rtr = (rtr ? 1 : 0);
- filter.rtr_mask = 1;
- if (buffer->filter_id != CAN_NO_FREE_FILTER) {
- can_detach(CANmodule->dev, buffer->filter_id);
- }
- buffer->filter_id = can_attach_isr(CANmodule->dev,
- canopen_rx_isr_callback,
- buffer, &filter);
- if (buffer->filter_id == CAN_NO_FREE_FILTER) {
- LOG_ERR("failed to attach CAN rx isr, no free filter");
- CO_errorReport(CANmodule->em, CO_EM_MEMORY_ALLOCATION_ERROR,
- CO_EMC_SOFTWARE_INTERNAL, 0);
- return CO_ERROR_OUT_OF_MEMORY;
- }
- return CO_ERROR_NO;
- }
- CO_CANtx_t *CO_CANtxBufferInit(CO_CANmodule_t *CANmodule, uint16_t index,
- uint16_t ident, bool_t rtr, uint8_t noOfBytes,
- bool_t syncFlag)
- {
- CO_CANtx_t *buffer;
- if (CANmodule == NULL) {
- return NULL;
- }
- if (index >= CANmodule->tx_size) {
- LOG_ERR("failed to initialize CAN rx buffer, illegal argument");
- CO_errorReport(CANmodule->em, CO_EM_GENERIC_SOFTWARE_ERROR,
- CO_EMC_SOFTWARE_INTERNAL, 0);
- return NULL;
- }
- buffer = &CANmodule->tx_array[index];
- buffer->ident = ident;
- buffer->rtr = rtr;
- buffer->DLC = noOfBytes;
- buffer->bufferFull = false;
- buffer->syncFlag = syncFlag;
- return buffer;
- }
- CO_ReturnError_t CO_CANsend(CO_CANmodule_t *CANmodule, CO_CANtx_t *buffer)
- {
- CO_ReturnError_t ret = CO_ERROR_NO;
- struct zcan_frame msg;
- int err;
- if (!CANmodule || !CANmodule->dev || !buffer) {
- return CO_ERROR_ILLEGAL_ARGUMENT;
- }
- CO_LOCK_CAN_SEND();
- if (buffer->bufferFull) {
- if (!CANmodule->first_tx_msg) {
- CO_errorReport(CANmodule->em, CO_EM_CAN_TX_OVERFLOW,
- CO_EMC_CAN_OVERRUN, buffer->ident);
- }
- buffer->bufferFull = false;
- ret = CO_ERROR_TX_OVERFLOW;
- }
- msg.id_type = CAN_STANDARD_IDENTIFIER;
- msg.id = buffer->ident;
- msg.dlc = buffer->DLC;
- msg.rtr = (buffer->rtr ? 1 : 0);
- memcpy(msg.data, buffer->data, buffer->DLC);
- err = can_send(CANmodule->dev, &msg, K_NO_WAIT, canopen_tx_isr_callback,
- CANmodule);
- if (err == CAN_TIMEOUT) {
- buffer->bufferFull = true;
- } else if (err != CAN_TX_OK) {
- LOG_ERR("failed to send CAN frame (err %d)", err);
- CO_errorReport(CANmodule->em, CO_EM_GENERIC_SOFTWARE_ERROR,
- CO_EMC_COMMUNICATION, 0);
- ret = CO_ERROR_TX_UNCONFIGURED;
- }
- CO_UNLOCK_CAN_SEND();
- return ret;
- }
- void CO_CANclearPendingSyncPDOs(CO_CANmodule_t *CANmodule)
- {
- bool_t tpdoDeleted = false;
- CO_CANtx_t *buffer;
- uint16_t i;
- if (!CANmodule) {
- return;
- }
- CO_LOCK_CAN_SEND();
- for (i = 0; i < CANmodule->tx_size; i++) {
- buffer = &CANmodule->tx_array[i];
- if (buffer->bufferFull && buffer->syncFlag) {
- buffer->bufferFull = false;
- tpdoDeleted = true;
- }
- }
- CO_UNLOCK_CAN_SEND();
- if (tpdoDeleted) {
- CO_errorReport(CANmodule->em, CO_EM_TPDO_OUTSIDE_WINDOW,
- CO_EMC_COMMUNICATION, 0);
- }
- }
- void CO_CANverifyErrors(CO_CANmodule_t *CANmodule)
- {
- CO_EM_t *em = (CO_EM_t *)CANmodule->em;
- struct can_bus_err_cnt err_cnt;
- enum can_state state;
- uint8_t rx_overflows;
- uint32_t errors;
- /*
- * TODO: Zephyr lacks an API for reading the rx mailbox
- * overflow counter.
- */
- rx_overflows = 0;
- state = can_get_state(CANmodule->dev, &err_cnt);
- errors = ((uint32_t)err_cnt.tx_err_cnt << 16) |
- ((uint32_t)err_cnt.rx_err_cnt << 8) |
- rx_overflows;
- if (errors != CANmodule->errors) {
- CANmodule->errors = errors;
- if (state == CAN_BUS_OFF) {
- /* Bus off */
- CO_errorReport(em, CO_EM_CAN_TX_BUS_OFF,
- CO_EMC_BUS_OFF_RECOVERED, errors);
- } else {
- /* Bus not off */
- CO_errorReset(em, CO_EM_CAN_TX_BUS_OFF, errors);
- if ((err_cnt.rx_err_cnt >= 96U) ||
- (err_cnt.tx_err_cnt >= 96U)) {
- /* Bus warning */
- CO_errorReport(em, CO_EM_CAN_BUS_WARNING,
- CO_EMC_NO_ERROR, errors);
- } else {
- /* Bus not warning */
- CO_errorReset(em, CO_EM_CAN_BUS_WARNING,
- errors);
- }
- if (err_cnt.rx_err_cnt >= 128U) {
- /* Bus rx passive */
- CO_errorReport(em, CO_EM_CAN_RX_BUS_PASSIVE,
- CO_EMC_CAN_PASSIVE, errors);
- } else {
- /* Bus not rx passive */
- CO_errorReset(em, CO_EM_CAN_RX_BUS_PASSIVE,
- errors);
- }
- if (err_cnt.tx_err_cnt >= 128U &&
- !CANmodule->first_tx_msg) {
- /* Bus tx passive */
- CO_errorReport(em, CO_EM_CAN_TX_BUS_PASSIVE,
- CO_EMC_CAN_PASSIVE, errors);
- } else if (CO_isError(em, CO_EM_CAN_TX_BUS_PASSIVE)) {
- /* Bus not tx passive */
- CO_errorReset(em, CO_EM_CAN_TX_BUS_PASSIVE,
- errors);
- CO_errorReset(em, CO_EM_CAN_TX_OVERFLOW,
- errors);
- }
- }
- /* This code can be activated if we can read the overflows*/
- if (false && rx_overflows != 0U) {
- CO_errorReport(em, CO_EM_CAN_RXB_OVERFLOW,
- CO_EMC_CAN_OVERRUN, errors);
- }
- }
- }
- static int canopen_init(const struct device *dev)
- {
- ARG_UNUSED(dev);
- k_work_queue_start(&canopen_tx_workq, canopen_tx_workq_stack,
- K_KERNEL_STACK_SIZEOF(canopen_tx_workq_stack),
- CONFIG_CANOPENNODE_TX_WORKQUEUE_PRIORITY, NULL);
- k_work_init(&canopen_tx_queue.work, canopen_tx_retry);
- return 0;
- }
- SYS_INIT(canopen_init, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|