SWM341_can.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442
  1. /******************************************************************************************************************************************
  2. * 文件名称: SWM341_can.c
  3. * 功能说明: SWM341单片机的CAN模块驱动库
  4. * 技术支持: http://www.synwit.com.cn/e/tool/gbook/?bid=1
  5. * 注意事项:
  6. * 版本日期: V1.1.0 2017年10月25日
  7. * 升级记录:
  8. *
  9. *
  10. *******************************************************************************************************************************************
  11. * @attention
  12. *
  13. * THE PRESENT FIRMWARE WHICH IS FOR GUIDANCE ONLY AIMS AT PROVIDING CUSTOMERS WITH CODING INFORMATION
  14. * REGARDING THEIR PRODUCTS IN ORDER FOR THEM TO SAVE TIME. AS A RESULT, SYNWIT SHALL NOT BE HELD LIABLE
  15. * FOR ANY DIRECT, INDIRECT OR CONSEQUENTIAL DAMAGES WITH RESPECT TO ANY CLAIMS ARISING FROM THE CONTENT
  16. * OF SUCH FIRMWARE AND/OR THE USE MADE BY CUSTOMERS OF THE CODING INFORMATION CONTAINED HEREIN IN CONN-
  17. * -ECTION WITH THEIR PRODUCTS.
  18. *
  19. * COPYRIGHT 2012 Synwit Technology
  20. *******************************************************************************************************************************************/
  21. #include "SWM341.h"
  22. #include "SWM341_can.h"
  23. /******************************************************************************************************************************************
  24. * 函数名称: CAN_Init()
  25. * 功能说明: CAN接口初始化
  26. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  27. * CAN_InitStructure * initStruct 包含CAN接口相关设定值的结构体
  28. * 输 出: 无
  29. * 注意事项: 无
  30. ******************************************************************************************************************************************/
  31. void CAN_Init(CAN_TypeDef * CANx, CAN_InitStructure * initStruct)
  32. {
  33. uint32_t brp = (SystemCoreClock/2)/2/initStruct->Baudrate/(1 + (initStruct->CAN_bs1 + 1) + (initStruct->CAN_bs2 + 1)) - 1;
  34. switch((uint32_t)CANx)
  35. {
  36. case ((uint32_t)CAN0):
  37. SYS->CLKEN0 |= (0x01 << SYS_CLKEN0_CAN0_Pos);
  38. break;
  39. case ((uint32_t)CAN1):
  40. SYS->CLKEN1 |= (0x01 << SYS_CLKEN1_CAN1_Pos);
  41. break;
  42. }
  43. CAN_Close(CANx); //一些关键寄存器只能在CAN关闭时设置
  44. CANx->CR &= ~(CAN_CR_LOM_Msk | CAN_CR_STM_Msk);
  45. CANx->CR |= (initStruct->Mode << CAN_CR_LOM_Pos);
  46. CANx->BT1 = (0 << CAN_BT1_SAM_Pos) |
  47. (initStruct->CAN_bs1 << CAN_BT1_TSEG1_Pos) |
  48. (initStruct->CAN_bs2 << CAN_BT1_TSEG2_Pos);
  49. CANx->BT0 = (initStruct->CAN_sjw << CAN_BT0_SJW_Pos) |
  50. ((brp & 0x3F) << CAN_BT0_BRP_Pos);
  51. CANx->BT2 = ((brp >> 6) << CAN_BT2_BRP_Pos);
  52. CANx->RXERR = 0; //只能在复位模式下清除
  53. CANx->TXERR = 0;
  54. CANx->IE = (initStruct->RXNotEmptyIEn << CAN_IE_RXDA_Pos) |
  55. (initStruct->ArbitrLostIEn << CAN_IE_ARBLOST_Pos) |
  56. (initStruct->ErrPassiveIEn << CAN_IE_ERRPASS_Pos);
  57. switch((uint32_t)CANx)
  58. {
  59. case ((uint32_t)CAN0):
  60. if(initStruct->RXNotEmptyIEn | initStruct->ArbitrLostIEn | initStruct->ErrPassiveIEn)
  61. {
  62. NVIC_EnableIRQ(CAN0_IRQn);
  63. }
  64. else
  65. {
  66. NVIC_DisableIRQ(CAN0_IRQn);
  67. }
  68. break;
  69. case ((uint32_t)CAN1):
  70. if(initStruct->RXNotEmptyIEn | initStruct->ArbitrLostIEn | initStruct->ErrPassiveIEn)
  71. {
  72. NVIC_EnableIRQ(CAN1_IRQn);
  73. }
  74. else
  75. {
  76. NVIC_DisableIRQ(CAN1_IRQn);
  77. }
  78. break;
  79. }
  80. }
  81. /******************************************************************************************************************************************
  82. * 函数名称: CAN_Open()
  83. * 功能说明: CAN接口打开
  84. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  85. * 输 出: 无
  86. * 注意事项: 无
  87. ******************************************************************************************************************************************/
  88. void CAN_Open(CAN_TypeDef * CANx)
  89. {
  90. CANx->CR &= ~(0x01 << CAN_CR_RST_Pos); //退出复位模式,进入工作模式
  91. }
  92. /******************************************************************************************************************************************
  93. * 函数名称: CAN_Close()
  94. * 功能说明: CAN接口关闭
  95. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  96. * 输 出: 无
  97. * 注意事项: 无
  98. ******************************************************************************************************************************************/
  99. void CAN_Close(CAN_TypeDef * CANx)
  100. {
  101. CANx->CR |= (0x01 << CAN_CR_RST_Pos); //进入复位模式,不能发送和接收数据
  102. }
  103. /******************************************************************************************************************************************
  104. * 函数名称: CAN_Transmit()
  105. * 功能说明: CAN发送数据
  106. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  107. * uint32_t format CAN_FRAME_STD 标准帧 CAN_FRAME_EXT 扩展帧
  108. * uint32_t id 消息ID
  109. * uint8_t data[] 要发送的数据
  110. * uint32_t size 要发送的数据的个数
  111. * uint32_t once 只发送一次,即使发送失败(仲裁丢失、发送出错、NAK)也不尝试重发
  112. * 输 出: 无
  113. * 注意事项: 无
  114. ******************************************************************************************************************************************/
  115. void CAN_Transmit(CAN_TypeDef * CANx, uint32_t format, uint32_t id, uint8_t data[], uint32_t size, uint32_t once)
  116. {
  117. uint32_t i;
  118. if(format == CAN_FRAME_STD)
  119. {
  120. CANx->FRAME.INFO = (0 << CAN_INFO_FF_Pos) |
  121. (0 << CAN_INFO_RTR_Pos) |
  122. (size << CAN_INFO_DLC_Pos);
  123. CANx->FRAME.DATA[0] = id >> 3;
  124. CANx->FRAME.DATA[1] = id << 5;
  125. for(i = 0; i < size; i++)
  126. {
  127. CANx->FRAME.DATA[i+2] = data[i];
  128. }
  129. }
  130. else //if(format == CAN_FRAME_EXT)
  131. {
  132. CANx->FRAME.INFO = (1 << CAN_INFO_FF_Pos) |
  133. (0 << CAN_INFO_RTR_Pos) |
  134. (size << CAN_INFO_DLC_Pos);
  135. CANx->FRAME.DATA[0] = id >> 21;
  136. CANx->FRAME.DATA[1] = id >> 13;
  137. CANx->FRAME.DATA[2] = id >> 5;
  138. CANx->FRAME.DATA[3] = id << 3;
  139. for(i = 0; i < size; i++)
  140. {
  141. CANx->FRAME.DATA[i+4] = data[i];
  142. }
  143. }
  144. if(CANx->CR & CAN_CR_STM_Msk)
  145. {
  146. CANx->CMD = (1 << CAN_CMD_SRR_Pos);
  147. }
  148. else
  149. {
  150. if(once == 0)
  151. {
  152. CANx->CMD = (1 << CAN_CMD_TXREQ_Pos);
  153. }
  154. else
  155. {
  156. CANx->CMD = (1 << CAN_CMD_TXREQ_Pos) | (1 << CAN_CMD_ABTTX_Pos);
  157. }
  158. }
  159. }
  160. /******************************************************************************************************************************************
  161. * 函数名称: CAN_TransmitRequest()
  162. * 功能说明: CAN发送远程请求,请求远程节点发送数据
  163. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  164. * uint32_t format CAN_FRAME_STD 标准帧 CAN_FRAME_EXT 扩展帧
  165. * uint32_t id 消息ID
  166. * uint32_t once 只发送一次,即使发送失败(仲裁丢失、发送出错、NAK)也不尝试重发
  167. * 输 出: 无
  168. * 注意事项: 无
  169. ******************************************************************************************************************************************/
  170. void CAN_TransmitRequest(CAN_TypeDef * CANx, uint32_t format, uint32_t id, uint32_t once)
  171. {
  172. if(format == CAN_FRAME_STD)
  173. {
  174. CANx->FRAME.INFO = (0 << CAN_INFO_FF_Pos) |
  175. (1 << CAN_INFO_RTR_Pos) |
  176. (0 << CAN_INFO_DLC_Pos);
  177. CANx->FRAME.DATA[0] = id >> 3;
  178. CANx->FRAME.DATA[1] = id << 5;
  179. }
  180. else //if(format == CAN_FRAME_EXT)
  181. {
  182. CANx->FRAME.INFO = (1 << CAN_INFO_FF_Pos) |
  183. (1 << CAN_INFO_RTR_Pos) |
  184. (0 << CAN_INFO_DLC_Pos);
  185. CANx->FRAME.DATA[0] = id >> 21;
  186. CANx->FRAME.DATA[1] = id >> 13;
  187. CANx->FRAME.DATA[2] = id >> 5;
  188. CANx->FRAME.DATA[3] = id << 3;
  189. }
  190. if(once == 0)
  191. {
  192. CANx->CMD = (1 << CAN_CMD_TXREQ_Pos);
  193. }
  194. else
  195. {
  196. CANx->CMD = (1 << CAN_CMD_TXREQ_Pos) | (1 << CAN_CMD_ABTTX_Pos);
  197. }
  198. }
  199. /******************************************************************************************************************************************
  200. * 函数名称: CAN_Receive()
  201. * 功能说明: CAN接收数据
  202. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  203. * CAN_RXMessage *msg 接收到的消息存储在此结构体变量中
  204. * 输 出: 无
  205. * 注意事项: 无
  206. ******************************************************************************************************************************************/
  207. void CAN_Receive(CAN_TypeDef * CANx, CAN_RXMessage *msg)
  208. {
  209. uint32_t i;
  210. msg->format = (CANx->FRAME.INFO & CAN_INFO_FF_Msk) >> CAN_INFO_FF_Pos;
  211. msg->remote = (CANx->FRAME.INFO & CAN_INFO_RTR_Msk) >> CAN_INFO_RTR_Pos;
  212. msg->size = (CANx->FRAME.INFO & CAN_INFO_DLC_Msk) >> CAN_INFO_DLC_Pos;
  213. if(msg->format == CAN_FRAME_STD)
  214. {
  215. msg->id = (CANx->FRAME.DATA[0] << 3) | (CANx->FRAME.DATA[1] >> 5);
  216. for(i = 0; i < msg->size; i++)
  217. {
  218. msg->data[i] = CANx->FRAME.DATA[i+2];
  219. }
  220. }
  221. else //if(msg->format == CAN_FRAME_EXT)
  222. {
  223. msg->id = (CANx->FRAME.DATA[0] << 21) | (CANx->FRAME.DATA[1] << 13) | (CANx->FRAME.DATA[2] << 5) | (CANx->FRAME.DATA[3] >> 3);
  224. for(i = 0; i < msg->size; i++)
  225. {
  226. msg->data[i] = CANx->FRAME.DATA[i+4];
  227. }
  228. }
  229. CANx->CMD = (1 << CAN_CMD_RRB_Pos);
  230. }
  231. /******************************************************************************************************************************************
  232. * 函数名称: CAN_TXComplete()
  233. * 功能说明: 发送是否完成
  234. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  235. * 输 出: uint32_t 1 已经完成 0 还未完成
  236. * 注意事项: 发送被Abort也会触发发送完成,但不会触发发送成功
  237. ******************************************************************************************************************************************/
  238. uint32_t CAN_TXComplete(CAN_TypeDef * CANx)
  239. {
  240. return (CANx->SR & CAN_SR_TXBR_Msk) ? 1 : 0;
  241. }
  242. /******************************************************************************************************************************************
  243. * 函数名称: CAN_TXSuccess()
  244. * 功能说明: 发送是否成功
  245. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  246. * 输 出: uint32_t 1 发送成功 0 发送失败
  247. * 注意事项: 无
  248. ******************************************************************************************************************************************/
  249. uint32_t CAN_TXSuccess(CAN_TypeDef * CANx)
  250. {
  251. return (CANx->SR & CAN_SR_TXOK_Msk) ? 1 : 0;
  252. }
  253. /******************************************************************************************************************************************
  254. * 函数名称: CAN_AbortTransmit()
  255. * 功能说明: 终止发送
  256. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  257. * 输 出: 无
  258. * 注意事项: 正在进行的发送无法终止,但执行此命令后若发送失败不会再重发
  259. ******************************************************************************************************************************************/
  260. void CAN_AbortTransmit(CAN_TypeDef * CANx)
  261. {
  262. CANx->CMD = (1 << CAN_CMD_ABTTX_Pos);
  263. }
  264. /******************************************************************************************************************************************
  265. * 函数名称: CAN_TXBufferReady()
  266. * 功能说明: TX Buffer是否准备好可以写入消息
  267. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  268. * 输 出: uint32_t 1 已准备好 0 未准备好
  269. * 注意事项: 无
  270. ******************************************************************************************************************************************/
  271. uint32_t CAN_TXBufferReady(CAN_TypeDef * CANx)
  272. {
  273. return (CANx->SR & CAN_SR_TXBR_Msk) ? 1 : 0;
  274. }
  275. /******************************************************************************************************************************************
  276. * 函数名称: CAN_RXDataAvailable()
  277. * 功能说明: RX FIFO中是否有数据可读出
  278. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  279. * 输 出: uint32_t 1 有数据可读出 0 没有数据
  280. * 注意事项: 无
  281. ******************************************************************************************************************************************/
  282. uint32_t CAN_RXDataAvailable(CAN_TypeDef * CANx)
  283. {
  284. return (CANx->SR & CAN_SR_RXDA_Msk) ? 1 : 0;
  285. }
  286. /******************************************************************************************************************************************
  287. * 函数名称: CAN_SetBaudrate()
  288. * 功能说明: 设置波特率
  289. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  290. * uint32_t baudrate 波特率,即位传输速率
  291. * uint32_t CAN_bs1 CAN_BS1_1tq、CAN_BS1_2tq、... ... 、CAN_BS1_16tq
  292. * uint32_t CAN_bs2 CAN_BS2_1tq、CAN_BS2_2tq、... ... 、CAN_BS2_8tq
  293. * uint32_t CAN_sjw CAN_SJW_1tq、CAN_SJW_2tq、CAN_SJW_3tq、CAN_SJW_4tq
  294. * 输 出: 无
  295. * 注意事项: 设置前需要先调用CAN_Close()关闭CAN模块
  296. ******************************************************************************************************************************************/
  297. void CAN_SetBaudrate(CAN_TypeDef * CANx, uint32_t baudrate, uint32_t CAN_bs1, uint32_t CAN_bs2, uint32_t CAN_sjw)
  298. {
  299. uint32_t brp = (SystemCoreClock/2)/2/baudrate/(1 + (CAN_bs1 + 1) + (CAN_bs2 + 1)) - 1;
  300. CANx->BT1 = (0 << CAN_BT1_SAM_Pos) |
  301. (CAN_bs1 << CAN_BT1_TSEG1_Pos) |
  302. (CAN_bs2 << CAN_BT1_TSEG2_Pos);
  303. CANx->BT0 = (CAN_sjw << CAN_BT0_SJW_Pos) |
  304. ((brp & 0x3F) << CAN_BT0_BRP_Pos);
  305. CANx->BT2 = ((brp >> 6) << CAN_BT2_BRP_Pos);
  306. }
  307. /******************************************************************************************************************************************
  308. * 函数名称: CAN_SetFilter32b()
  309. * 功能说明: 设置接收滤波器,模式为1个32位滤波器
  310. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  311. * uint32_t filter 要设置的滤波器,有效值有CAN_FILTER_1、CAN_FILTER_2、...、CAN_FILTER_16
  312. * uint32_t check 与mask一起决定了接收到的Message是否是自己需要的:check & mask == ID & mask的Message通过过滤
  313. * uint32_t mask
  314. * 输 出: 无
  315. * 注意事项: 只能在关闭时设置
  316. ******************************************************************************************************************************************/
  317. void CAN_SetFilter32b(CAN_TypeDef * CANx, uint32_t filter, uint32_t check, uint32_t mask)
  318. {
  319. CANx->AFM |= (1 << filter);
  320. CANx->ACR[filter] = __REV(check << 3); // 高29位
  321. CANx->AMR[filter] = __REV(~(mask << 3));
  322. CANx->AFE |= (1 << filter);
  323. }
  324. /******************************************************************************************************************************************
  325. * 函数名称: CAN_SetFilter16b()
  326. * 功能说明: 设置接收滤波器,模式为2个16位滤波器
  327. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  328. * uint32_t filter 要设置的滤波器,有效值有CAN_FILTER_1、CAN_FILTER_2、...、CAN_FILTER_16
  329. * uint16_t check1 与mask一起决定了接收到的Message是否是自己需要的:check & mask == ID & mask的Message通过过滤
  330. * uint16_t mask1
  331. * uint16_t check2
  332. * uint16_t mask2
  333. * 输 出: 无
  334. * 注意事项: 只能在关闭时设置
  335. ******************************************************************************************************************************************/
  336. void CAN_SetFilter16b(CAN_TypeDef * CANx, uint32_t filter, uint16_t check1, uint16_t mask1, uint16_t check2, uint16_t mask2)
  337. {
  338. CANx->AFM &= ~(1 << filter);
  339. CANx->ACR[filter] = __REV((check1 << 5) | (check2 << 21)); // 高11位
  340. CANx->AMR[filter] = __REV(~((mask1 << 5) | (mask2 << 21)));
  341. CANx->AFE |= (1 << filter);
  342. }
  343. /******************************************************************************************************************************************
  344. * 函数名称: CAN_INTEn()
  345. * 功能说明: 使能指定中断
  346. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  347. * uint32_t it interrupt type,有效值包括 CAN_IT_RX_NOTEMPTY、CAN_IT_RX_OVERFLOW、CAN_IT_TX_EMPTY、CAN_IT_ARBLOST、
  348. * CAN_IT_ERR、CAN_IT_ERR_WARN、CAN_IT_ERR_PASS、CAN_IT_WAKEUP 及其“或”
  349. * 输 出: 无
  350. * 注意事项: 无
  351. ******************************************************************************************************************************************/
  352. void CAN_INTEn(CAN_TypeDef * CANx, uint32_t it)
  353. {
  354. CANx->IE |= it;
  355. }
  356. /******************************************************************************************************************************************
  357. * 函数名称: CAN_INTDis()
  358. * 功能说明: 关闭指定中断
  359. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  360. * uint32_t it interrupt type,有效值包括 CAN_IT_RX_NOTEMPTY、CAN_IT_RX_OVERFLOW、CAN_IT_TX_EMPTY、CAN_IT_ARBLOST、
  361. * CAN_IT_ERR、CAN_IT_ERR_WARN、CAN_IT_ERR_PASS、CAN_IT_WAKEUP 及其“或”
  362. * 输 出: 无
  363. * 注意事项: 无
  364. ******************************************************************************************************************************************/
  365. void CAN_INTDis(CAN_TypeDef * CANx, uint32_t it)
  366. {
  367. CANx->IE &= ~it;
  368. }
  369. /******************************************************************************************************************************************
  370. * 函数名称: CAN_INTClr()
  371. * 功能说明: 清除中断标志
  372. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  373. * uint32_t it interrupt type,有效值包括 CAN_IT_RX_OVERFLOW
  374. * 输 出: 无
  375. * 注意事项: 无
  376. ******************************************************************************************************************************************/
  377. void CAN_INTClr(CAN_TypeDef * CANx, uint32_t it)
  378. {
  379. CANx->CMD = (1 << CAN_CMD_CLROV_Pos);
  380. }
  381. /******************************************************************************************************************************************
  382. * 函数名称: CAN_INTStat()
  383. * 功能说明: 查询中断状态
  384. * 输 入: CAN_TypeDef * CANx 指定要被设置的CAN接口,有效值包括CAN0、CAN1
  385. * 输 出: uint32_t 当前中断状态
  386. * 注意事项: CANx->IF读取清零,因此在中断ISR中只能读取一次,不能多次读取
  387. ******************************************************************************************************************************************/
  388. uint32_t CAN_INTStat(CAN_TypeDef * CANx)
  389. {
  390. return CANx->IF;
  391. }