CO_driver.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. /*
  2. * Copyright (c) 2019 Vestas Wind Systems A/S
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <zephyr.h>
  7. #include <drivers/can.h>
  8. #include <init.h>
  9. #include <sys/util.h>
  10. #include <canopennode.h>
  11. #define LOG_LEVEL CONFIG_CANOPEN_LOG_LEVEL
  12. #include <logging/log.h>
  13. LOG_MODULE_REGISTER(canopen_driver);
  14. K_KERNEL_STACK_DEFINE(canopen_tx_workq_stack,
  15. CONFIG_CANOPENNODE_TX_WORKQUEUE_STACK_SIZE);
  16. struct k_work_q canopen_tx_workq;
  17. struct canopen_tx_work_container {
  18. struct k_work work;
  19. CO_CANmodule_t *CANmodule;
  20. };
  21. struct canopen_tx_work_container canopen_tx_queue;
  22. K_MUTEX_DEFINE(canopen_send_mutex);
  23. K_MUTEX_DEFINE(canopen_emcy_mutex);
  24. K_MUTEX_DEFINE(canopen_co_mutex);
  25. inline void canopen_send_lock(void)
  26. {
  27. k_mutex_lock(&canopen_send_mutex, K_FOREVER);
  28. }
  29. inline void canopen_send_unlock(void)
  30. {
  31. k_mutex_unlock(&canopen_send_mutex);
  32. }
  33. inline void canopen_emcy_lock(void)
  34. {
  35. k_mutex_lock(&canopen_emcy_mutex, K_FOREVER);
  36. }
  37. inline void canopen_emcy_unlock(void)
  38. {
  39. k_mutex_unlock(&canopen_emcy_mutex);
  40. }
  41. inline void canopen_od_lock(void)
  42. {
  43. k_mutex_lock(&canopen_co_mutex, K_FOREVER);
  44. }
  45. inline void canopen_od_unlock(void)
  46. {
  47. k_mutex_unlock(&canopen_co_mutex);
  48. }
  49. static void canopen_detach_all_rx_filters(CO_CANmodule_t *CANmodule)
  50. {
  51. uint16_t i;
  52. if (!CANmodule || !CANmodule->rx_array || !CANmodule->configured) {
  53. return;
  54. }
  55. for (i = 0U; i < CANmodule->rx_size; i++) {
  56. if (CANmodule->rx_array[i].filter_id != CAN_NO_FREE_FILTER) {
  57. can_detach(CANmodule->dev,
  58. CANmodule->rx_array[i].filter_id);
  59. CANmodule->rx_array[i].filter_id = CAN_NO_FREE_FILTER;
  60. }
  61. }
  62. }
  63. static void canopen_rx_isr_callback(struct zcan_frame *msg, void *arg)
  64. {
  65. CO_CANrx_t *buffer = (CO_CANrx_t *)arg;
  66. CO_CANrxMsg_t rxMsg;
  67. if (!buffer || !buffer->pFunct) {
  68. LOG_ERR("failed to process CAN rx isr callback");
  69. return;
  70. }
  71. rxMsg.ident = msg->id;
  72. rxMsg.DLC = msg->dlc;
  73. memcpy(rxMsg.data, msg->data, msg->dlc);
  74. buffer->pFunct(buffer->object, &rxMsg);
  75. }
  76. static void canopen_tx_isr_callback(uint32_t error_flags, void *arg)
  77. {
  78. CO_CANmodule_t *CANmodule = arg;
  79. if (!CANmodule) {
  80. LOG_ERR("failed to process CAN tx isr callback");
  81. return;
  82. }
  83. if (error_flags == CAN_TX_OK) {
  84. CANmodule->first_tx_msg = false;
  85. }
  86. k_work_submit_to_queue(&canopen_tx_workq, &canopen_tx_queue.work);
  87. }
  88. static void canopen_tx_retry(struct k_work *item)
  89. {
  90. struct canopen_tx_work_container *container =
  91. CONTAINER_OF(item, struct canopen_tx_work_container, work);
  92. CO_CANmodule_t *CANmodule = container->CANmodule;
  93. struct zcan_frame msg;
  94. CO_CANtx_t *buffer;
  95. int err;
  96. uint16_t i;
  97. CO_LOCK_CAN_SEND();
  98. for (i = 0; i < CANmodule->tx_size; i++) {
  99. buffer = &CANmodule->tx_array[i];
  100. if (buffer->bufferFull) {
  101. msg.id_type = CAN_STANDARD_IDENTIFIER;
  102. msg.id = buffer->ident;
  103. msg.dlc = buffer->DLC;
  104. msg.rtr = (buffer->rtr ? 1 : 0);
  105. memcpy(msg.data, buffer->data, buffer->DLC);
  106. err = can_send(CANmodule->dev, &msg, K_NO_WAIT,
  107. canopen_tx_isr_callback, CANmodule);
  108. if (err == CAN_TIMEOUT) {
  109. break;
  110. } else if (err != CAN_TX_OK) {
  111. LOG_ERR("failed to send CAN frame (err %d)",
  112. err);
  113. CO_errorReport(CANmodule->em,
  114. CO_EM_GENERIC_SOFTWARE_ERROR,
  115. CO_EMC_COMMUNICATION, 0);
  116. }
  117. buffer->bufferFull = false;
  118. }
  119. }
  120. CO_UNLOCK_CAN_SEND();
  121. }
  122. void CO_CANsetConfigurationMode(void *CANdriverState)
  123. {
  124. /* No operation */
  125. }
  126. void CO_CANsetNormalMode(CO_CANmodule_t *CANmodule)
  127. {
  128. CANmodule->CANnormal = true;
  129. }
  130. CO_ReturnError_t CO_CANmodule_init(CO_CANmodule_t *CANmodule,
  131. void *CANdriverState,
  132. CO_CANrx_t rxArray[], uint16_t rxSize,
  133. CO_CANtx_t txArray[], uint16_t txSize,
  134. uint16_t CANbitRate)
  135. {
  136. struct canopen_context *ctx = (struct canopen_context *)CANdriverState;
  137. uint16_t i;
  138. int err;
  139. LOG_DBG("rxSize = %d, txSize = %d", rxSize, txSize);
  140. if (!CANmodule || !rxArray || !txArray || !CANdriverState) {
  141. LOG_ERR("failed to initialize CAN module");
  142. return CO_ERROR_ILLEGAL_ARGUMENT;
  143. }
  144. if (rxSize > CONFIG_CAN_MAX_FILTER) {
  145. LOG_ERR("insufficient number of concurrent CAN RX filters"
  146. " (needs %d, %d available)", rxSize,
  147. CONFIG_CAN_MAX_FILTER);
  148. return CO_ERROR_OUT_OF_MEMORY;
  149. } else if (rxSize < CONFIG_CAN_MAX_FILTER) {
  150. LOG_DBG("excessive number of concurrent CAN RX filters enabled"
  151. " (needs %d, %d available)", rxSize,
  152. CONFIG_CAN_MAX_FILTER);
  153. }
  154. canopen_detach_all_rx_filters(CANmodule);
  155. canopen_tx_queue.CANmodule = CANmodule;
  156. CANmodule->dev = ctx->dev;
  157. CANmodule->rx_array = rxArray;
  158. CANmodule->rx_size = rxSize;
  159. CANmodule->tx_array = txArray;
  160. CANmodule->tx_size = txSize;
  161. CANmodule->CANnormal = false;
  162. CANmodule->first_tx_msg = true;
  163. CANmodule->errors = 0;
  164. CANmodule->em = NULL;
  165. for (i = 0U; i < rxSize; i++) {
  166. rxArray[i].ident = 0U;
  167. rxArray[i].pFunct = NULL;
  168. rxArray[i].filter_id = CAN_NO_FREE_FILTER;
  169. }
  170. for (i = 0U; i < txSize; i++) {
  171. txArray[i].bufferFull = false;
  172. }
  173. err = can_set_bitrate(CANmodule->dev, KHZ(CANbitRate), 0);
  174. if (err) {
  175. LOG_ERR("failed to configure CAN bitrate (err %d)", err);
  176. return CO_ERROR_ILLEGAL_ARGUMENT;
  177. }
  178. err = can_set_mode(CANmodule->dev, CAN_NORMAL_MODE);
  179. if (err) {
  180. LOG_ERR("failed to configure CAN interface (err %d)", err);
  181. return CO_ERROR_ILLEGAL_ARGUMENT;
  182. }
  183. CANmodule->configured = true;
  184. return CO_ERROR_NO;
  185. }
  186. void CO_CANmodule_disable(CO_CANmodule_t *CANmodule)
  187. {
  188. int err;
  189. if (!CANmodule || !CANmodule->dev) {
  190. return;
  191. }
  192. canopen_detach_all_rx_filters(CANmodule);
  193. err = can_configure(CANmodule->dev, CAN_SILENT_MODE, 0);
  194. if (err) {
  195. LOG_ERR("failed to disable CAN interface (err %d)", err);
  196. }
  197. }
  198. uint16_t CO_CANrxMsg_readIdent(const CO_CANrxMsg_t *rxMsg)
  199. {
  200. return rxMsg->ident;
  201. }
  202. CO_ReturnError_t CO_CANrxBufferInit(CO_CANmodule_t *CANmodule, uint16_t index,
  203. uint16_t ident, uint16_t mask, bool_t rtr,
  204. void *object,
  205. CO_CANrxBufferCallback_t pFunct)
  206. {
  207. struct zcan_filter filter;
  208. CO_CANrx_t *buffer;
  209. if (CANmodule == NULL) {
  210. return CO_ERROR_ILLEGAL_ARGUMENT;
  211. }
  212. if (!pFunct || (index >= CANmodule->rx_size)) {
  213. LOG_ERR("failed to initialize CAN rx buffer, illegal argument");
  214. CO_errorReport(CANmodule->em, CO_EM_GENERIC_SOFTWARE_ERROR,
  215. CO_EMC_SOFTWARE_INTERNAL, 0);
  216. return CO_ERROR_ILLEGAL_ARGUMENT;
  217. }
  218. buffer = &CANmodule->rx_array[index];
  219. buffer->object = object;
  220. buffer->pFunct = pFunct;
  221. filter.id_type = CAN_STANDARD_IDENTIFIER;
  222. filter.id = ident;
  223. filter.id_mask = mask;
  224. filter.rtr = (rtr ? 1 : 0);
  225. filter.rtr_mask = 1;
  226. if (buffer->filter_id != CAN_NO_FREE_FILTER) {
  227. can_detach(CANmodule->dev, buffer->filter_id);
  228. }
  229. buffer->filter_id = can_attach_isr(CANmodule->dev,
  230. canopen_rx_isr_callback,
  231. buffer, &filter);
  232. if (buffer->filter_id == CAN_NO_FREE_FILTER) {
  233. LOG_ERR("failed to attach CAN rx isr, no free filter");
  234. CO_errorReport(CANmodule->em, CO_EM_MEMORY_ALLOCATION_ERROR,
  235. CO_EMC_SOFTWARE_INTERNAL, 0);
  236. return CO_ERROR_OUT_OF_MEMORY;
  237. }
  238. return CO_ERROR_NO;
  239. }
  240. CO_CANtx_t *CO_CANtxBufferInit(CO_CANmodule_t *CANmodule, uint16_t index,
  241. uint16_t ident, bool_t rtr, uint8_t noOfBytes,
  242. bool_t syncFlag)
  243. {
  244. CO_CANtx_t *buffer;
  245. if (CANmodule == NULL) {
  246. return NULL;
  247. }
  248. if (index >= CANmodule->tx_size) {
  249. LOG_ERR("failed to initialize CAN rx buffer, illegal argument");
  250. CO_errorReport(CANmodule->em, CO_EM_GENERIC_SOFTWARE_ERROR,
  251. CO_EMC_SOFTWARE_INTERNAL, 0);
  252. return NULL;
  253. }
  254. buffer = &CANmodule->tx_array[index];
  255. buffer->ident = ident;
  256. buffer->rtr = rtr;
  257. buffer->DLC = noOfBytes;
  258. buffer->bufferFull = false;
  259. buffer->syncFlag = syncFlag;
  260. return buffer;
  261. }
  262. CO_ReturnError_t CO_CANsend(CO_CANmodule_t *CANmodule, CO_CANtx_t *buffer)
  263. {
  264. CO_ReturnError_t ret = CO_ERROR_NO;
  265. struct zcan_frame msg;
  266. int err;
  267. if (!CANmodule || !CANmodule->dev || !buffer) {
  268. return CO_ERROR_ILLEGAL_ARGUMENT;
  269. }
  270. CO_LOCK_CAN_SEND();
  271. if (buffer->bufferFull) {
  272. if (!CANmodule->first_tx_msg) {
  273. CO_errorReport(CANmodule->em, CO_EM_CAN_TX_OVERFLOW,
  274. CO_EMC_CAN_OVERRUN, buffer->ident);
  275. }
  276. buffer->bufferFull = false;
  277. ret = CO_ERROR_TX_OVERFLOW;
  278. }
  279. msg.id_type = CAN_STANDARD_IDENTIFIER;
  280. msg.id = buffer->ident;
  281. msg.dlc = buffer->DLC;
  282. msg.rtr = (buffer->rtr ? 1 : 0);
  283. memcpy(msg.data, buffer->data, buffer->DLC);
  284. err = can_send(CANmodule->dev, &msg, K_NO_WAIT, canopen_tx_isr_callback,
  285. CANmodule);
  286. if (err == CAN_TIMEOUT) {
  287. buffer->bufferFull = true;
  288. } else if (err != CAN_TX_OK) {
  289. LOG_ERR("failed to send CAN frame (err %d)", err);
  290. CO_errorReport(CANmodule->em, CO_EM_GENERIC_SOFTWARE_ERROR,
  291. CO_EMC_COMMUNICATION, 0);
  292. ret = CO_ERROR_TX_UNCONFIGURED;
  293. }
  294. CO_UNLOCK_CAN_SEND();
  295. return ret;
  296. }
  297. void CO_CANclearPendingSyncPDOs(CO_CANmodule_t *CANmodule)
  298. {
  299. bool_t tpdoDeleted = false;
  300. CO_CANtx_t *buffer;
  301. uint16_t i;
  302. if (!CANmodule) {
  303. return;
  304. }
  305. CO_LOCK_CAN_SEND();
  306. for (i = 0; i < CANmodule->tx_size; i++) {
  307. buffer = &CANmodule->tx_array[i];
  308. if (buffer->bufferFull && buffer->syncFlag) {
  309. buffer->bufferFull = false;
  310. tpdoDeleted = true;
  311. }
  312. }
  313. CO_UNLOCK_CAN_SEND();
  314. if (tpdoDeleted) {
  315. CO_errorReport(CANmodule->em, CO_EM_TPDO_OUTSIDE_WINDOW,
  316. CO_EMC_COMMUNICATION, 0);
  317. }
  318. }
  319. void CO_CANverifyErrors(CO_CANmodule_t *CANmodule)
  320. {
  321. CO_EM_t *em = (CO_EM_t *)CANmodule->em;
  322. struct can_bus_err_cnt err_cnt;
  323. enum can_state state;
  324. uint8_t rx_overflows;
  325. uint32_t errors;
  326. /*
  327. * TODO: Zephyr lacks an API for reading the rx mailbox
  328. * overflow counter.
  329. */
  330. rx_overflows = 0;
  331. state = can_get_state(CANmodule->dev, &err_cnt);
  332. errors = ((uint32_t)err_cnt.tx_err_cnt << 16) |
  333. ((uint32_t)err_cnt.rx_err_cnt << 8) |
  334. rx_overflows;
  335. if (errors != CANmodule->errors) {
  336. CANmodule->errors = errors;
  337. if (state == CAN_BUS_OFF) {
  338. /* Bus off */
  339. CO_errorReport(em, CO_EM_CAN_TX_BUS_OFF,
  340. CO_EMC_BUS_OFF_RECOVERED, errors);
  341. } else {
  342. /* Bus not off */
  343. CO_errorReset(em, CO_EM_CAN_TX_BUS_OFF, errors);
  344. if ((err_cnt.rx_err_cnt >= 96U) ||
  345. (err_cnt.tx_err_cnt >= 96U)) {
  346. /* Bus warning */
  347. CO_errorReport(em, CO_EM_CAN_BUS_WARNING,
  348. CO_EMC_NO_ERROR, errors);
  349. } else {
  350. /* Bus not warning */
  351. CO_errorReset(em, CO_EM_CAN_BUS_WARNING,
  352. errors);
  353. }
  354. if (err_cnt.rx_err_cnt >= 128U) {
  355. /* Bus rx passive */
  356. CO_errorReport(em, CO_EM_CAN_RX_BUS_PASSIVE,
  357. CO_EMC_CAN_PASSIVE, errors);
  358. } else {
  359. /* Bus not rx passive */
  360. CO_errorReset(em, CO_EM_CAN_RX_BUS_PASSIVE,
  361. errors);
  362. }
  363. if (err_cnt.tx_err_cnt >= 128U &&
  364. !CANmodule->first_tx_msg) {
  365. /* Bus tx passive */
  366. CO_errorReport(em, CO_EM_CAN_TX_BUS_PASSIVE,
  367. CO_EMC_CAN_PASSIVE, errors);
  368. } else if (CO_isError(em, CO_EM_CAN_TX_BUS_PASSIVE)) {
  369. /* Bus not tx passive */
  370. CO_errorReset(em, CO_EM_CAN_TX_BUS_PASSIVE,
  371. errors);
  372. CO_errorReset(em, CO_EM_CAN_TX_OVERFLOW,
  373. errors);
  374. }
  375. }
  376. /* This code can be activated if we can read the overflows*/
  377. if (false && rx_overflows != 0U) {
  378. CO_errorReport(em, CO_EM_CAN_RXB_OVERFLOW,
  379. CO_EMC_CAN_OVERRUN, errors);
  380. }
  381. }
  382. }
  383. static int canopen_init(const struct device *dev)
  384. {
  385. ARG_UNUSED(dev);
  386. k_work_queue_start(&canopen_tx_workq, canopen_tx_workq_stack,
  387. K_KERNEL_STACK_SIZEOF(canopen_tx_workq_stack),
  388. CONFIG_CANOPENNODE_TX_WORKQUEUE_PRIORITY, NULL);
  389. k_work_init(&canopen_tx_queue.work, canopen_tx_retry);
  390. return 0;
  391. }
  392. SYS_INIT(canopen_init, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);