drv_can.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594
  1. /*
  2. * Copyright (c) 2006-2023, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Email: opensource_embedded@phytium.com.cn
  7. *
  8. * Change Logs:
  9. * Date Author Notes
  10. * 2023-03-20 zhangyan first version
  11. *
  12. */
  13. #include "drv_can.h"
  14. #include "sdkconfig.h"
  15. #ifdef RT_USING_CAN
  16. #include "fdebug.h"
  17. #include "fpinctrl.h"
  18. #define FCAN_TEST_DEBUG_TAG "FCAN_TEST"
  19. #define FCAN_TEST_DEBUG(format, ...) FT_DEBUG_PRINT_D(FCAN_TEST_DEBUG_TAG, format, ##__VA_ARGS__)
  20. #define FCAN_TEST_INFO(format, ...) FT_DEBUG_PRINT_I(FCAN_TEST_DEBUG_TAG, format, ##__VA_ARGS__)
  21. #define FCAN_TEST_WARN(format, ...) FT_DEBUG_PRINT_W(FCAN_TEST_DEBUG_TAG, format, ##__VA_ARGS__)
  22. #define FCAN_TEST_ERROR(format, ...) FT_DEBUG_PRINT_E(FCAN_TEST_DEBUG_TAG, format, ##__VA_ARGS__)
  23. struct phytium_can
  24. {
  25. const char *name;
  26. FCanCtrl can_handle;
  27. FCanIdMaskConfig filter;
  28. struct rt_can_device device; /* inherit from can device */
  29. };
  30. static struct phytium_can drv_can[FCAN_NUM] =
  31. {
  32. {
  33. .name = "CAN0",
  34. .can_handle.config.instance_id = 0,
  35. },
  36. {
  37. .name = "CAN1",
  38. .can_handle.config.instance_id = 1,
  39. },
  40. #if defined(CONFIG_TARGET_F2000_4) || defined(CONFIG_TARGET_D2000)
  41. {
  42. .name = "CAN2",
  43. .can_handle.config.instance_id = 2,
  44. },
  45. #endif
  46. };
  47. static void CanRxIrqCallback(void *args)
  48. {
  49. FCanCtrl *instance_p = (FCanCtrl *)args;
  50. rt_hw_can_isr(&drv_can[instance_p->config.instance_id].device, RT_CAN_EVENT_RX_IND);
  51. FCAN_TEST_DEBUG("CAN%d irq recv frame callback.", instance_p->config.instance_id);
  52. }
  53. static void CanErrorCallback(void *args)
  54. {
  55. FCanCtrl *instance_p = (FCanCtrl *)args;
  56. uintptr base_addr = instance_p->config.base_address;
  57. FCAN_TEST_DEBUG("CAN %d is under error.", instance_p->config.instance_id);
  58. FCAN_TEST_DEBUG("error_status is %x.", FCAN_READ_REG32(base_addr, FCAN_INTR_OFFSET));
  59. FCAN_TEST_DEBUG("rxerr_cnt is %x.", FCAN_ERR_CNT_RFN_GET(FCAN_READ_REG32(base_addr, FCAN_ERR_CNT_OFFSET)));
  60. FCAN_TEST_DEBUG("txerr_cnt is %x.", FCAN_ERR_CNT_TFN_GET(FCAN_READ_REG32(base_addr, FCAN_ERR_CNT_OFFSET)));
  61. }
  62. static void CanTxIrqCallback(void *args)
  63. {
  64. FCanCtrl *instance_p = (FCanCtrl *)args;
  65. rt_hw_can_isr(&drv_can[instance_p->config.instance_id].device, RT_CAN_EVENT_TX_DONE);
  66. FCAN_TEST_DEBUG("CAN%d irq send frame callback.", instance_p->config.instance_id);
  67. }
  68. static rt_err_t _can_config(struct rt_can_device *can, struct can_configure *cfg)
  69. {
  70. RT_ASSERT(can);
  71. RT_ASSERT(cfg);
  72. struct phytium_can *drv_can;
  73. drv_can = (struct phytium_can *)can->parent.user_data;
  74. RT_ASSERT(drv_can);
  75. FError status = FT_SUCCESS;
  76. rt_kprintf("CAN%d begin to config.\n", drv_can->can_handle.config.instance_id);
  77. #if defined(CONFIG_TARGET_F2000_4) || defined(CONFIG_TARGET_D2000)
  78. if(drv_can->can_handle.config.instance_id == FCAN_INSTANCE_0)
  79. {
  80. FPinSetFunc(FIOCTRL_TJTAG_TDI_PAD, FPIN_FUNC1); /* can0-tx: func 1 */
  81. FPinSetFunc(FIOCTRL_SWDITMS_SWJ_PAD, FPIN_FUNC1); /* can0-rx: func 1 */
  82. }
  83. else if(drv_can->can_handle.config.instance_id == FCAN_INSTANCE_1)
  84. {
  85. FPinSetFunc(FIOCTRL_NTRST_SWJ_PAD, FPIN_FUNC1); /* can1-tx: func 1 */
  86. FPinSetFunc(FIOCTRL_SWDO_SWJ_PAD, FPIN_FUNC1); /* can1-rx: func 1 */
  87. }
  88. else
  89. {
  90. FCAN_TEST_ERROR("CAN id is under error.");
  91. return RT_ERROR;
  92. }
  93. #elif defined(CONFIG_TARGET_E2000)
  94. FIOPadSetCanMux(drv_can->can_handle.config.instance_id);
  95. #endif
  96. /*CAN config init*/
  97. status = FCanCfgInitialize(&(drv_can->can_handle), FCanLookupConfig(drv_can->can_handle.config.instance_id));
  98. if (status != FT_SUCCESS)
  99. {
  100. FCAN_TEST_DEBUG("CAN %d initialize error, status = %#x.", drv_can->can_handle.config.instance_id, status);
  101. return RT_ERROR;
  102. }
  103. /*Set the baudrate*/
  104. FCanBaudrateConfig arb_segment_config;
  105. FCanBaudrateConfig data_segment_config;
  106. memset(&arb_segment_config, 0, sizeof(arb_segment_config));
  107. memset(&data_segment_config, 0, sizeof(data_segment_config));
  108. #if defined(RT_CAN_USING_CANFD)
  109. arb_segment_config.auto_calc = TRUE;
  110. arb_segment_config.baudrate = CAN1MBaud; /*CANFD arb baud defaults to 1M ,allowed to be modified*/
  111. arb_segment_config.segment = FCAN_ARB_SEGMENT;
  112. status = FCanBaudrateSet(&(drv_can->can_handle), &arb_segment_config);
  113. if (status != RT_EOK)
  114. {
  115. FCAN_TEST_DEBUG("CAN%d set arb segment baudrate failed.", drv_can->can_handle.config.instance_id);
  116. return RT_ERROR;
  117. }
  118. data_segment_config.auto_calc = TRUE;
  119. data_segment_config.baudrate = cfg->baud_rate_fd;
  120. data_segment_config.segment = FCAN_DATA_SEGMENT;
  121. status = FCanBaudrateSet(&(drv_can->can_handle), &data_segment_config);
  122. if (status != RT_EOK)
  123. {
  124. FCAN_TEST_DEBUG("CAN%d set data segment baudrate failed.", drv_can->can_handle.config.instance_id);
  125. return RT_ERROR;
  126. }
  127. #else
  128. arb_segment_config.auto_calc = TRUE;
  129. arb_segment_config.baudrate = cfg->baud_rate;
  130. arb_segment_config.segment = FCAN_ARB_SEGMENT;
  131. status = FCanBaudrateSet(&(drv_can->can_handle), &arb_segment_config);
  132. if (status != FT_SUCCESS)
  133. {
  134. FCAN_TEST_DEBUG("CAN%d set arb segment baudrate failed.", drv_can->can_handle.config.instance_id);
  135. return RT_ERROR;
  136. }
  137. data_segment_config.auto_calc = TRUE;
  138. data_segment_config.baudrate = cfg->baud_rate;
  139. data_segment_config.segment = FCAN_DATA_SEGMENT;
  140. status = FCanBaudrateSet(&(drv_can->can_handle), &data_segment_config);
  141. if (status != FT_SUCCESS)
  142. {
  143. FCAN_TEST_DEBUG("CAN%d set data segment baudrate failed.", drv_can->can_handle.config.instance_id);
  144. return RT_ERROR;
  145. }
  146. #endif
  147. /*CAN filter function init*/
  148. for (int i = 0; i < FCAN_ACC_ID_REG_NUM; i++)
  149. {
  150. drv_can->filter.filter_index = i;
  151. drv_can->filter.id = 0;
  152. drv_can->filter.mask = FCAN_ACC_IDN_MASK;
  153. status |= FCanIdMaskFilterSet(&(drv_can->can_handle), &(drv_can->filter));
  154. }
  155. if (status != FT_SUCCESS)
  156. {
  157. FCAN_TEST_ERROR("CAN%d set mask filter failed.", drv_can->can_handle.config.instance_id);
  158. return RT_ERROR;
  159. }
  160. /* Identifier mask enable */
  161. FCanIdMaskFilterEnable(&(drv_can->can_handle));
  162. /* Transmit mode init , the default setting is normal mode */
  163. FCanSetMode(&(drv_can->can_handle), FCAN_PROBE_NORMAL_MODE);
  164. /* enable can transfer */
  165. FCanEnable(&(drv_can->can_handle), RT_TRUE);
  166. return RT_EOK;
  167. }
  168. static rt_err_t _can_control(struct rt_can_device *can, int cmd, void *arg)
  169. {
  170. RT_ASSERT(can);
  171. rt_uint32_t argval;
  172. struct phytium_can *drv_can;
  173. drv_can = (struct phytium_can *)can->parent.user_data;
  174. RT_ASSERT(drv_can != RT_NULL);
  175. rt_uint32_t cpu_id;
  176. FCanIntrEventConfig intr_event;
  177. FError status = FT_SUCCESS;
  178. #ifdef RT_CAN_USING_HDR
  179. struct rt_can_filter_config *filter_cfg;
  180. #endif
  181. switch (cmd)
  182. {
  183. case RT_DEVICE_CTRL_SET_INT:
  184. GetCpuId(&cpu_id);
  185. rt_hw_interrupt_set_target_cpus(drv_can->can_handle.config.irq_num, cpu_id);
  186. argval = (rt_uint32_t) arg;
  187. /*Open different interrupts*/
  188. if (argval == RT_DEVICE_CAN_INT_ERR)
  189. {
  190. intr_event.type = FCAN_INTR_EVENT_ERROR;
  191. intr_event.handler = CanErrorCallback;
  192. intr_event.param = (void *)(&(drv_can->can_handle));
  193. FCanRegisterInterruptHandler(&(drv_can->can_handle), &intr_event);
  194. FCanInterruptEnable(&(drv_can->can_handle), intr_event.type);
  195. }
  196. if (argval == RT_DEVICE_FLAG_INT_TX)
  197. {
  198. intr_event.type = FCAN_INTR_EVENT_SEND;
  199. intr_event.handler = CanTxIrqCallback;
  200. intr_event.param = (void *)(&(drv_can->can_handle));
  201. FCanRegisterInterruptHandler(&(drv_can->can_handle), &intr_event);
  202. FCanInterruptEnable(&(drv_can->can_handle), intr_event.type);
  203. }
  204. if (argval == RT_DEVICE_FLAG_INT_RX)
  205. {
  206. intr_event.type = FCAN_INTR_EVENT_RECV;
  207. intr_event.handler = CanRxIrqCallback;
  208. intr_event.param = (void *)(&(drv_can->can_handle));
  209. FCanRegisterInterruptHandler(&(drv_can->can_handle), &intr_event);
  210. FCanInterruptEnable(&(drv_can->can_handle), intr_event.type);
  211. }
  212. rt_hw_interrupt_set_priority(drv_can->can_handle.config.irq_num, 16);
  213. rt_hw_interrupt_install(drv_can->can_handle.config.irq_num, FCanIntrHandler, &(drv_can->can_handle), drv_can->name);
  214. rt_hw_interrupt_umask(drv_can->can_handle.config.irq_num);
  215. break;
  216. case RT_CAN_CMD_SET_MODE:
  217. argval = (rt_uint32_t) arg;
  218. FCanEnable(&(drv_can->can_handle), RT_FALSE);
  219. if (argval == RT_CAN_MODE_LISTEN)
  220. {
  221. FCanSetMode(&(drv_can->can_handle), FCAN_PROBE_MONITOR_MODE);
  222. drv_can->device.config.mode = RT_CAN_MODE_LISTEN;
  223. }
  224. else if (argval == RT_CAN_MODE_NORMAL)
  225. {
  226. FCanSetMode(&(drv_can->can_handle), FCAN_PROBE_NORMAL_MODE);
  227. drv_can->device.config.mode = RT_CAN_MODE_NORMAL;
  228. }
  229. FCanEnable(&(drv_can->can_handle), RT_TRUE);
  230. break;
  231. case RT_CAN_CMD_SET_BAUD:
  232. argval = (rt_uint32_t) arg;
  233. if (argval != CAN1MBaud &&
  234. argval != CAN800kBaud &&
  235. argval != CAN500kBaud &&
  236. argval != CAN250kBaud &&
  237. argval != CAN125kBaud &&
  238. argval != CAN100kBaud &&
  239. argval != CAN50kBaud &&
  240. argval != CAN20kBaud &&
  241. argval != CAN10kBaud)
  242. {
  243. return RT_ERROR;
  244. }
  245. if (argval != drv_can->device.config.baud_rate)
  246. {
  247. FCanBaudrateConfig arb_segment_config;
  248. FCanBaudrateConfig data_segment_config;
  249. memset(&arb_segment_config, 0, sizeof(arb_segment_config));
  250. memset(&data_segment_config, 0, sizeof(data_segment_config));
  251. drv_can->device.config.baud_rate = argval;
  252. FCanEnable(&(drv_can->can_handle), RT_FALSE);
  253. arb_segment_config.auto_calc = TRUE;
  254. arb_segment_config.baudrate = drv_can->device.config.baud_rate;
  255. arb_segment_config.segment = FCAN_ARB_SEGMENT;
  256. status = FCanBaudrateSet(&(drv_can->can_handle), &arb_segment_config);
  257. if (status != FT_SUCCESS)
  258. {
  259. FCAN_TEST_DEBUG("CAN%d set arb segment baudrate failed.", drv_can->can_handle.config.instance_id);
  260. return RT_ERROR;
  261. }
  262. data_segment_config.auto_calc = TRUE;
  263. data_segment_config.baudrate = drv_can->device.config.baud_rate;
  264. data_segment_config.segment = FCAN_DATA_SEGMENT;
  265. status = FCanBaudrateSet(&(drv_can->can_handle), &data_segment_config);
  266. if (status != FT_SUCCESS)
  267. {
  268. FCAN_TEST_DEBUG("CAN%d set data segment baudrate failed.", drv_can->can_handle.config.instance_id);
  269. return RT_ERROR;
  270. }
  271. FCanEnable(&(drv_can->can_handle), RT_TRUE);
  272. }
  273. break;
  274. case RT_CAN_CMD_SET_BAUD_FD:
  275. #if defined RT_CAN_USING_CANFD
  276. argval = (rt_uint32_t) arg;
  277. if (argval != drv_can->device.config.baud_rate_fd)
  278. {
  279. FCanBaudrateConfig arb_segment_config;
  280. FCanBaudrateConfig data_segment_config;
  281. memset(&arb_segment_config, 0, sizeof(arb_segment_config));
  282. memset(&data_segment_config, 0, sizeof(data_segment_config));
  283. drv_can->device.config.baud_rate = argval;
  284. FCanEnable(&(drv_can->can_handle), RT_FALSE);
  285. arb_segment_config.auto_calc = TRUE;
  286. arb_segment_config.baudrate = CAN1MBaud;
  287. arb_segment_config.segment = FCAN_ARB_SEGMENT;
  288. status = FCanBaudrateSet(&(drv_can->can_handle), &arb_segment_config);
  289. if (status != FT_SUCCESS)
  290. {
  291. FCAN_TEST_DEBUG("CAN%d set arb segment baudrate failed.", drv_can->can_handle.config.instance_id);
  292. return RT_ERROR;
  293. }
  294. data_segment_config.auto_calc = TRUE;
  295. data_segment_config.baudrate = drv_can->device.config.baud_rate_fd;
  296. data_segment_config.segment = FCAN_DATA_SEGMENT;
  297. status = FCanBaudrateSet(&(drv_can->can_handle), &data_segment_config);
  298. if (status != FT_SUCCESS)
  299. {
  300. FCAN_TEST_DEBUG("CAN%d set data segment baudrate failed.", drv_can->can_handle.config.instance_id);
  301. return RT_ERROR;
  302. }
  303. FCanEnable(&(drv_can->can_handle), RT_TRUE);
  304. }
  305. #endif
  306. break;
  307. case RT_CAN_CMD_SET_FILTER:
  308. #ifdef RT_CAN_USING_HDR
  309. filter_cfg = (struct rt_can_filter_config *)arg;
  310. FCanEnable(&(drv_can->can_handle), RT_FALSE);
  311. for (int i = 0; i < filter_cfg->count; i++)
  312. {
  313. drv_can->filter.filter_index = i;
  314. drv_can->filter.mask = filter_cfg->items[i].mask;
  315. drv_can->filter.id = filter_cfg->items[i].id;
  316. drv_can->filter.type = FCAN_STANDARD_FRAME;
  317. status = FCanIdMaskFilterSet(&(drv_can->can_handle), &(drv_can->filter));
  318. if (status != FT_SUCCESS)
  319. {
  320. FCAN_TEST_ERROR("CAN%d set mask filter failed.", drv_can->can_handle.config.instance_id);
  321. return RT_ERROR;
  322. }
  323. }
  324. FCanEnable(&(drv_can->can_handle), RT_TRUE);
  325. #endif
  326. break;
  327. }
  328. return RT_EOK;
  329. }
  330. static int _can_sendmsg(struct rt_can_device *can, const void *buf, rt_uint32_t box_num)
  331. {
  332. RT_ASSERT(can);
  333. RT_ASSERT(buf);
  334. struct phytium_can *drv_can;
  335. drv_can = (struct phytium_can *)can->parent.user_data;
  336. RT_ASSERT(drv_can);
  337. struct rt_can_msg *pmsg = (struct rt_can_msg *)buf;
  338. FCanFrame can_frame = {0};
  339. /* Check the parameters */
  340. RT_ASSERT(pmsg->len <= 8U);
  341. if (RT_CAN_STDID == pmsg->ide)
  342. {
  343. can_frame.canid = pmsg->id;
  344. }
  345. else
  346. {
  347. can_frame.canid = pmsg->id;
  348. can_frame.canid |= CAN_EFF_FLAG;
  349. }
  350. if (RT_CAN_DTR == pmsg->rtr)
  351. {
  352. }
  353. else
  354. {
  355. can_frame.canid |= CAN_RTR_FLAG;
  356. }
  357. can_frame.candlc = pmsg->len ;
  358. memcpy(can_frame.data, pmsg->data, 8);
  359. return (FCanSend(&drv_can->can_handle, &can_frame) == RT_EOK) ? RT_EOK : -RT_ERROR;
  360. }
  361. static int _can_recvmsg(struct rt_can_device *can, void *buf, rt_uint32_t fifo)
  362. {
  363. RT_ASSERT(can);
  364. RT_ASSERT(buf);
  365. struct phytium_can *drv_can;
  366. drv_can = (struct phytium_can *)can->parent.user_data;
  367. RT_ASSERT(drv_can);
  368. struct rt_can_msg *pmsg = (struct rt_can_msg *)buf;
  369. FCanFrame recv_frame;
  370. FError status = FT_SUCCESS;
  371. status = FCanRecv(&(drv_can->can_handle), &recv_frame);
  372. if (status != FT_SUCCESS)
  373. {
  374. FCAN_TEST_DEBUG("CAN%d recv data failed.", drv_can->can_handle.config.instance_id);
  375. return RT_ERROR;
  376. }
  377. if (CAN_EFF_FLAG & recv_frame.canid)
  378. {
  379. pmsg->ide = RT_CAN_EXTID;
  380. pmsg->id = (recv_frame.canid & ~(RT_CAN_EXTID));
  381. }
  382. else
  383. {
  384. pmsg->ide = RT_CAN_STDID;
  385. pmsg->id = recv_frame.canid;
  386. }
  387. if (CAN_RTR_FLAG & recv_frame.canid)
  388. {
  389. pmsg->id &= ~CAN_RTR_FLAG;
  390. pmsg->rtr = RT_CAN_RTR;
  391. }
  392. else
  393. {
  394. pmsg->rtr = RT_CAN_DTR;
  395. }
  396. /* get len */
  397. pmsg->len = recv_frame.candlc;
  398. for (int i = 0; i < pmsg->len; i++)
  399. {
  400. pmsg->data[i] = recv_frame.data[i];
  401. }
  402. /* get hdr */
  403. pmsg->hdr = 0;
  404. return RT_EOK;
  405. }
  406. static const struct rt_can_ops _can_ops =
  407. {
  408. _can_config,
  409. _can_control,
  410. _can_sendmsg,
  411. _can_recvmsg,
  412. };
  413. int rt_hw_can_init(void)
  414. {
  415. rt_err_t ret = RT_EOK;
  416. for (int i = 0; i < (u32)FCAN_NUM; i++)
  417. {
  418. drv_can[i].device.config.ticks = 20000;
  419. drv_can[i].device.config.baud_rate = 800000;
  420. #ifdef RT_CAN_USING_CANFD
  421. drv_can[i].device.config.baud_rate_fd = 800000;
  422. #endif
  423. drv_can[i].device.config.mode = RT_CAN_MODE_NORMAL;
  424. drv_can[i].device.config.sndboxnumber = 1;
  425. drv_can[i].device.config.msgboxsz = 1;
  426. #ifdef RT_CAN_USING_HDR
  427. drv_can[i].device.config.maxhdr = 1;
  428. #endif
  429. ret = rt_hw_can_register(&drv_can[i].device,
  430. drv_can[i].name,
  431. &_can_ops,
  432. &drv_can[i]);
  433. RT_ASSERT(ret == RT_EOK);
  434. }
  435. return (int)ret;
  436. }
  437. INIT_BOARD_EXPORT(rt_hw_can_init);
  438. /*can test example*/
  439. static rt_device_t can_dev; /* CAN device handle */
  440. static struct rt_semaphore rx_sem;
  441. static rt_err_t can_rx_call(rt_device_t dev, rt_size_t size)
  442. {
  443. /* The CAN generates an interrupt after receiving data, calls this callback function, and then sends the received semaphore */
  444. rt_sem_release(&rx_sem);
  445. return RT_EOK;
  446. }
  447. static void can_rx_thread(void *parameter)
  448. {
  449. int i;
  450. rt_err_t res = RT_EOK;
  451. struct rt_can_msg rxmsg = {0};
  452. rt_device_set_rx_indicate(can_dev, can_rx_call);
  453. while (1)
  454. {
  455. /* The hdr value is - 1, which means reading data directly from the uselist */
  456. rxmsg.hdr = -1;
  457. /* Blocking waiting to receive semaphore */
  458. res = rt_sem_take(&rx_sem, RT_WAITING_FOREVER);
  459. RT_ASSERT(res == RT_EOK);
  460. /* Read a frame of data from CAN */
  461. rt_device_read(can_dev, 0, &rxmsg, sizeof(rxmsg));
  462. /* Print data ID and conten */
  463. rt_kprintf("ID:%x\n", rxmsg.id);
  464. rt_kprintf("DATA: ");
  465. for (i = 0; i < 8; i++)
  466. {
  467. rt_kprintf("%2x ", rxmsg.data[i]);
  468. }
  469. rt_kprintf("\n");
  470. }
  471. }
  472. int can_sample(int argc, char *argv[])
  473. {
  474. struct rt_can_msg msg = {0};
  475. rt_err_t res = RT_EOK;;
  476. rt_size_t size;
  477. rt_thread_t thread;
  478. char can_name[RT_NAME_MAX];
  479. if (argc == 2)
  480. {
  481. rt_strncpy(can_name, argv[1], RT_NAME_MAX);
  482. }
  483. else
  484. {
  485. rt_strncpy(can_name, "CAN0", RT_NAME_MAX);
  486. }
  487. /* Find CAN device */
  488. can_dev = rt_device_find(can_name);
  489. if (!can_dev)
  490. {
  491. rt_kprintf("Find %s failed.\n", can_name);
  492. return RT_ERROR;
  493. }
  494. /* Initialize CAN receive signal quantity */
  495. res = rt_sem_init(&rx_sem, "rx_sem", 0, RT_IPC_FLAG_FIFO);
  496. RT_ASSERT(res == RT_EOK);
  497. /* Open the CAN device in the way of interrupt reception and transmission */
  498. res = rt_device_open(can_dev, RT_DEVICE_FLAG_INT_TX | RT_DEVICE_FLAG_INT_RX);
  499. rt_device_control(can_dev,RT_CAN_CMD_SET_BAUD, CAN1MBaud);
  500. RT_ASSERT(res == RT_EOK);
  501. #ifdef RT_CAN_USING_HDR
  502. struct rt_can_filter_item items[4] =
  503. {
  504. RT_CAN_FILTER_ITEM_INIT(0x3, 0, 0, 0, 0, RT_NULL, RT_NULL),
  505. RT_CAN_FILTER_ITEM_INIT(0x3, 0, 0, 0, 0, RT_NULL, RT_NULL),
  506. RT_CAN_FILTER_ITEM_INIT(0x3, 0, 0, 0, 0, RT_NULL, RT_NULL),
  507. RT_CAN_FILTER_ITEM_INIT(0x3, 0, 0, 0, 0, RT_NULL, RT_NULL)
  508. };
  509. struct rt_can_filter_config cfg = {4, 1, items}; /* There are 4 filter tables in total */
  510. /* Set the hardware filter table. After setting, only frames with id=0x03 can be received*/
  511. res = rt_device_control(can_dev, RT_CAN_CMD_SET_FILTER, &cfg);
  512. RT_ASSERT(res == RT_EOK);
  513. #endif
  514. /* Create data receiving thread */
  515. thread = rt_thread_create("can_rx", can_rx_thread, RT_NULL, 1024, 25, 10);
  516. if (thread != RT_NULL)
  517. {
  518. res = rt_thread_startup(thread);
  519. RT_ASSERT(res == RT_EOK);
  520. }
  521. else
  522. {
  523. rt_kprintf("Create can_rx thread failed.\n");
  524. }
  525. msg.id = 0x78; /* ID = 0x78 */
  526. msg.ide = RT_CAN_STDID; /* Standard format */
  527. msg.rtr = RT_CAN_RTR; /* Data frame */
  528. msg.len = 8; /* Data length is 8 */
  529. /* Send CAN data */
  530. for (int i = 0; i < 10; i++)
  531. {
  532. /* 8-byte data to be sent */
  533. msg.data[0] = 0x00+i;
  534. msg.data[1] = 0x11+i;
  535. msg.data[2] = 0x22+i;
  536. msg.data[3] = 0x33+i;
  537. msg.data[4] = 0x44+i;
  538. msg.data[5] = 0x55+i;
  539. msg.data[6] = 0x66+i;
  540. msg.data[7] = 0x77+i;
  541. rt_device_write(can_dev, 0, &msg, sizeof(msg));
  542. }
  543. return res;
  544. }
  545. /* Enter can_sample command for testing */
  546. MSH_CMD_EXPORT(can_sample, can device sample);
  547. #endif