can.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. /**
  2. ******************************************************************************
  3. * File Name : CAN.c
  4. * Description : This file provides code for the configuration
  5. * of the CAN instances.
  6. ******************************************************************************
  7. ** This notice applies to any and all portions of this file
  8. * that are not between comment pairs USER CODE BEGIN and
  9. * USER CODE END. Other portions of this file, whether
  10. * inserted by the user or by software development tools
  11. * are owned by their respective copyright owners.
  12. *
  13. * COPYRIGHT(c) 2019 STMicroelectronics
  14. *
  15. * Redistribution and use in source and binary forms, with or without modification,
  16. * are permitted provided that the following conditions are met:
  17. * 1. Redistributions of source code must retain the above copyright notice,
  18. * this list of conditions and the following disclaimer.
  19. * 2. Redistributions in binary form must reproduce the above copyright notice,
  20. * this list of conditions and the following disclaimer in the documentation
  21. * and/or other materials provided with the distribution.
  22. * 3. Neither the name of STMicroelectronics nor the names of its contributors
  23. * may be used to endorse or promote products derived from this software
  24. * without specific prior written permission.
  25. *
  26. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  27. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  28. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  29. * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
  30. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  31. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
  32. * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  33. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  34. * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  35. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  36. *
  37. ******************************************************************************
  38. */
  39. /* Includes ------------------------------------------------------------------*/
  40. #include "can.h"
  41. #include "gpio.h"
  42. #include "can_process.h"
  43. #include "enviolo_can.h"
  44. /* USER CODE BEGIN 0 */
  45. CanTxMsgTypeDef CAN_TxMessaage;
  46. CanRxMsgTypeDef CAN_RxMessaage;
  47. uint8_t CAN_RxBuf_PBU[255];
  48. CAN_Buf_TypeDef CAN_RxBuf_Struct_PBU = {0,255,0,0,0,0,CAN_RxBuf_PBU,0,FALSE};
  49. uint8_t CAN_RxBuf_BMS[255];
  50. CAN_Buf_TypeDef CAN_RxBuf_Struct_BMS = {0,255,0,0,0,0,CAN_RxBuf_BMS,0,FALSE};
  51. uint8_t CAN_RxBuf_HMI[255];
  52. CAN_Buf_TypeDef CAN_RxBuf_Struct_HMI = {0,255,0,0,0,0,CAN_RxBuf_HMI,0,FALSE};
  53. uint8_t CAN_RxBuf_CDL[255];
  54. CAN_Buf_TypeDef CAN_RxBuf_Struct_CDL = {0,255,0,0,0,0,CAN_RxBuf_CDL,0,FALSE};
  55. /* USER CODE END 0 */
  56. CAN_HandleTypeDef hcan;
  57. /* CAN init function */
  58. void MX_CAN_Init(void)
  59. {
  60. hcan.Instance = CAN1;
  61. hcan.Init.Prescaler = 24;
  62. hcan.Init.Mode = CAN_MODE_NORMAL;
  63. hcan.Init.SJW = CAN_SJW_1TQ;
  64. hcan.Init.BS1 = CAN_BS1_4TQ;
  65. hcan.Init.BS2 = CAN_BS2_1TQ;
  66. hcan.Init.TTCM = DISABLE;
  67. hcan.Init.ABOM = ENABLE;
  68. hcan.Init.AWUM = DISABLE;
  69. hcan.Init.NART = DISABLE;
  70. hcan.Init.RFLM = DISABLE;
  71. hcan.Init.TXFP = DISABLE;
  72. if (HAL_CAN_Init(&hcan) != HAL_OK)
  73. {
  74. _Error_Handler(__FILE__, __LINE__);
  75. }
  76. }
  77. void HAL_CAN_MspInit(CAN_HandleTypeDef* canHandle)
  78. {
  79. GPIO_InitTypeDef GPIO_InitStruct;
  80. if(canHandle->Instance==CAN1)
  81. {
  82. /* USER CODE BEGIN CAN1_MspInit 0 */
  83. /* USER CODE END CAN1_MspInit 0 */
  84. /* CAN1 clock enable */
  85. __HAL_RCC_CAN1_CLK_ENABLE();
  86. /**CAN GPIO Configuration
  87. PB8 ------> CAN_RX
  88. PB9 ------> CAN_TX
  89. */
  90. GPIO_InitStruct.Pin = GPIO_PIN_8;
  91. GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
  92. GPIO_InitStruct.Pull = GPIO_NOPULL;
  93. HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
  94. GPIO_InitStruct.Pin = GPIO_PIN_9;
  95. GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
  96. GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
  97. HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
  98. __HAL_AFIO_REMAP_CAN1_2();
  99. /* CAN1 interrupt Init */
  100. HAL_NVIC_SetPriority(USB_LP_CAN1_RX0_IRQn, 1, 1);
  101. HAL_NVIC_EnableIRQ(USB_LP_CAN1_RX0_IRQn);
  102. /* USER CODE BEGIN CAN1_MspInit 1 */
  103. canHandle->pTxMsg = &CAN_TxMessaage;
  104. canHandle->pRxMsg = &CAN_RxMessaage;
  105. CANFilterConfig_Scale32_IdMask_StandardIdOnly();
  106. __HAL_CAN_ENABLE_IT(&hcan, CAN_IT_FMP0);
  107. /* USER CODE END CAN1_MspInit 1 */
  108. }
  109. }
  110. void HAL_CAN_MspDeInit(CAN_HandleTypeDef* canHandle)
  111. {
  112. if(canHandle->Instance==CAN1)
  113. {
  114. /* USER CODE BEGIN CAN1_MspDeInit 0 */
  115. /* USER CODE END CAN1_MspDeInit 0 */
  116. /* Peripheral clock disable */
  117. __HAL_RCC_CAN1_CLK_DISABLE();
  118. /**CAN GPIO Configuration
  119. PB8 ------> CAN_RX
  120. PB9 ------> CAN_TX
  121. */
  122. HAL_GPIO_DeInit(GPIOB, GPIO_PIN_8|GPIO_PIN_9);
  123. /* CAN1 interrupt Deinit */
  124. HAL_NVIC_DisableIRQ(USB_LP_CAN1_RX0_IRQn);
  125. /* USER CODE BEGIN CAN1_MspDeInit 1 */
  126. /* USER CODE END CAN1_MspDeInit 1 */
  127. }
  128. }
  129. /* USER CODE BEGIN 1 */
  130. //CAN接收数据处理
  131. void CAN_Rx_ISR(CAN_Buf_TypeDef*ptCANRx,uint8_t ucLength)
  132. {
  133. for(uint8_t i=0;i<ucLength;i++)
  134. {
  135. *((* ptCANRx).pcBufAddr + (* ptCANRx).ucBufWrInde) = hcan.pRxMsg->Data[i];
  136. if(++(* ptCANRx).ucBufWrInde >= (* ptCANRx).ucBufSize)
  137. {
  138. (* ptCANRx).ucBufWrInde = 0;
  139. }
  140. if(++(* ptCANRx).ucBufCnt > (* ptCANRx).ucBufSize)
  141. {
  142. (* ptCANRx).ucBufCnt = (* ptCANRx).ucBufSize;
  143. (* ptCANRx).ucBufOvf = 1;
  144. }
  145. }
  146. }
  147. void HAL_CAN_RxCpltCallback(CAN_HandleTypeDef *CanHandle)
  148. {
  149. if((CanHandle->pRxMsg->IDE != CAN_ID_STD)||(CanHandle->pRxMsg->DLC == 0))
  150. {
  151. __HAL_CAN_ENABLE_IT(&hcan, CAN_IT_FMP0);//开接收中断
  152. return;
  153. }
  154. switch(CanHandle->pRxMsg->StdId)
  155. {
  156. case ID_PBU_BC: case ID_PBU_TO_MC://接收PBU数据
  157. {
  158. CAN_RxBuf_Struct_PBU.ucBufID = CanHandle->pRxMsg->StdId;
  159. CAN_Rx_ISR(&CAN_RxBuf_Struct_PBU,CanHandle->pRxMsg->DLC);
  160. break;
  161. }
  162. case ID_BMS_BC: case ID_BMS_TO_MC://接收BMS数据
  163. {
  164. CAN_RxBuf_Struct_BMS.ucBufID = CanHandle->pRxMsg->StdId;
  165. CAN_Rx_ISR(&CAN_RxBuf_Struct_BMS,CanHandle->pRxMsg->DLC);
  166. break;
  167. }
  168. case ID_HMI_BC: case ID_HMI_TO_MC://接收HMI数据
  169. {
  170. CAN_RxBuf_Struct_HMI.ucBufID = CanHandle->pRxMsg->StdId;
  171. CAN_Rx_ISR(&CAN_RxBuf_Struct_HMI,CanHandle->pRxMsg->DLC);
  172. break;
  173. }
  174. case ID_CDL_BC: case ID_CDL_TO_MC:case ID_CDL_TO_MC_TE://接收CDL数据
  175. {
  176. CAN_RxBuf_Struct_CDL.ucBufID = CanHandle->pRxMsg->StdId;
  177. CAN_Rx_ISR(&CAN_RxBuf_Struct_CDL,CanHandle->pRxMsg->DLC);
  178. break;
  179. }
  180. case ID_ENVIOLO_MC:
  181. {
  182. if(CanHandle->pRxMsg->DLC == 8)//由于没有帧头帧尾和CRC,故保证数据的准确性必须为8字节才解析
  183. {
  184. Slot_EnvioloData(hcan.pRxMsg->Data, &Enviolo_Calibration);
  185. #if 0
  186. CanEnvioloData.u8Get_Gear_Pedal_RPM = hcan.pRxMsg->Data[0]; //反馈的踏频值 cadence
  187. CanEnvioloData.u8Get_Gear_CVP_Actuel_Ratio = hcan.pRxMsg->Data[1]; //实际的档位值 1bit = 0.1ratio
  188. CanEnvioloData.u16Get_Gear_Vehicle_Speed = (hcan.pRxMsg->Data[2])*5; //当前速度 乘以0.5 eg:50=25km/h 文档协议单位为0.1,故乘以5
  189. CanEnvioloData.flGet_Gear_Battery_Voltage = (hcan.pRxMsg->Data[3])*0.25;//当前电压值 乘以0.25 eg:102=25.5km/h
  190. CanEnvioloData.u16Get_Gear_Wheel_RPM = (uint16_t)(hcan.pRxMsg->Data[4]<<8)|(uint16_t)(hcan.pRxMsg->Data[5]);//车轮转速 需要设置车子周长
  191. CanEnvioloData.u16CalibrationVal = (uint16_t)(hcan.pRxMsg->Data[6]<<8)|(uint16_t)(hcan.pRxMsg->Data[7]);//获取是否需要校准
  192. if((CanEnvioloData.u16CalibrationVal&0XFFFF) == 0X1000)
  193. {
  194. if(CanEnvioloData.u8CalibrationCmd == 0)
  195. CanEnvioloData.u8CalibrationCmd = 0X01;//需要校准
  196. }
  197. else if((CanEnvioloData.u16CalibrationVal&0XFFFF) == 0X2000)//校准完成
  198. {
  199. if(CanEnvioloData.u8CalibrationCmd != 0X00)
  200. {
  201. CanEnvioloData.u8CalibrationCmd = 0X02;//进入校准
  202. }
  203. CanEnvioloData.u8CalibrationHistory = 0;
  204. //CanEnvioloData.u8CalibrationCmd = 0;
  205. }
  206. else if((CanEnvioloData.u16CalibrationVal&0XFFFF) == 0X3000)//校准完成
  207. {
  208. if(CanEnvioloData.u8CalibrationCmd != 0X00)
  209. {
  210. CanEnvioloData.u8CalibrationCmd = 0X03;//校准完成
  211. }
  212. CanEnvioloData.u8CalibrationHistory = 0;
  213. //CanEnvioloData.u8CalibrationCmd = 0;
  214. }
  215. if(MC_SpeedSensorData.IsStopFlag == TRUE)
  216. {
  217. MC_SpeedSensorData.IsStopFlag = FALSE;
  218. }
  219. CanEnvioloData.u16Timeout_Communication = 0;
  220. #endif
  221. }
  222. break;
  223. }
  224. default:break;
  225. }
  226. __HAL_CAN_ENABLE_IT(&hcan, CAN_IT_FMP0);//开接收中断
  227. }
  228. //CAN发送数据
  229. void CAN_SendData(uint16_t ID, uint8_t *Data, uint16_t Length)
  230. {
  231. uint16_t LastPacketLen, PacketNum;
  232. uint16_t i,j;
  233. if(Length > 0)
  234. {
  235. //计算分包个数
  236. LastPacketLen = Length % 8;//最后一个数据包长度
  237. if(LastPacketLen == 0)
  238. {
  239. LastPacketLen = 8;
  240. PacketNum = Length / 8;
  241. }
  242. else
  243. {
  244. PacketNum = Length / 8 + 1;
  245. }
  246. //开始发送数据
  247. hcan.pTxMsg->StdId = ID;
  248. hcan.pTxMsg->RTR = CAN_RTR_DATA;
  249. hcan.pTxMsg->IDE = CAN_ID_STD;
  250. //发送前(PacketNum - 1)个数据包
  251. for(i = 0;i < (PacketNum - 1); i++)
  252. {
  253. hcan.pTxMsg->DLC = 8;
  254. for(j = 0;j < 8;j++)
  255. {
  256. hcan.pTxMsg->Data[j] = Data[8 * i + j];
  257. }
  258. if(HAL_CAN_Transmit(&hcan, 10) == HAL_ERROR)
  259. {
  260. MX_CAN_Init();
  261. }
  262. }
  263. //发送最后一个数据包
  264. hcan.pTxMsg->DLC = LastPacketLen;
  265. for(j = 0;j < LastPacketLen;j++)
  266. {
  267. hcan.pTxMsg->Data[j] = Data[8 * i + j];
  268. }
  269. if(HAL_CAN_Transmit(&hcan, 10) == HAL_ERROR)
  270. {
  271. MX_CAN_Init();
  272. }
  273. }
  274. }
  275. //过滤器设置
  276. void CANFilterConfig_Scale32_IdMask_StandardIdOnly(void)
  277. {
  278. CAN_FilterConfTypeDef sFilterConfig;
  279. //设置过滤器组0,指定接收发送给PBU的数据
  280. sFilterConfig.FilterNumber = 0;//使用过滤器0
  281. sFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK;//配置为掩码模式
  282. sFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT;//设置为32位宽
  283. //过滤寄存器判断接收的报文ID,符合过滤寄存器的则接收
  284. //屏蔽寄存器判断接收的报文ID,符合屏蔽寄存器的则丢弃
  285. //当屏蔽寄存器的某位为0时,过滤寄存器的相应位不起作用,即信息标识符相应位无论是0还是1都会被接收
  286. //当屏蔽寄存器的某位为1时,信息标识符相应位与过滤器的相应位必须一致才能接收
  287. //设置过滤寄存器
  288. //(0x701 & 0x605)=601
  289. //0X601<<21 0110 0000 0001 <<21 --> 1100 0000 0010 0000 0000 0000 0000 0000 -->C0200000
  290. //0XC0200000&0XFFFF0000=0XC0200000
  291. //0XC0200000>>16=0XC020
  292. //0xC0200000|0x00000000U|0x00000000U=0XC0200000=0X0000
  293. sFilterConfig.FilterIdHigh = (uint16_t)((((uint32_t)ID_TO_MC_FILTER << 21) & 0xFFFF0000) >> 16); //0XC020
  294. sFilterConfig.FilterIdLow = (uint16_t)(((uint32_t)ID_TO_MC_FILTER << 21) | CAN_ID_STD | CAN_RTR_DATA) & 0xFFFF;//0X0000
  295. //(0x70F & 0x60F)=60F
  296. //60F<<21 0110 0000 1111<<21 -->1100 0001 1110 0000 0000 0000 0000 0000 -->0XC1E0 0000
  297. //设置屏蔽寄存器
  298. sFilterConfig.FilterMaskIdHigh = (uint16_t)((((uint32_t)ID_TO_MC_MASK << 21) & 0xFFFF0000) >> 16);//0XC1E0
  299. sFilterConfig.FilterMaskIdLow = 0xFFFF;
  300. //以上定义的含义为:1100 0000 0010 0000 0000 0000 0000 0000
  301. // 1100 0001 1110 0000 1111 1111 1111 1111
  302. //
  303. sFilterConfig.FilterFIFOAssignment = 0; //设置通过的数据帧进入到FIFO中
  304. sFilterConfig.FilterActivation = ENABLE;
  305. if(HAL_CAN_ConfigFilter(&hcan, &sFilterConfig) != HAL_OK)
  306. {
  307. Error_Handler();
  308. }
  309. //设置过滤器组1,指定接收广播数据
  310. sFilterConfig.FilterNumber = 1;
  311. sFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK;//配置为掩码模式
  312. sFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT;//设置为32位宽
  313. //设置过滤寄存器
  314. sFilterConfig.FilterIdHigh = (uint16_t)((((uint32_t)ID_BC_FILTER << 21) & 0xFFFF0000) >> 16);
  315. sFilterConfig.FilterIdLow = (uint16_t)(((uint32_t)ID_BC_FILTER << 21) | CAN_ID_STD | CAN_RTR_DATA) & 0xFFFF;
  316. //设置屏蔽寄存器
  317. sFilterConfig.FilterMaskIdHigh = (uint16_t)((((uint32_t)ID_BC_MASK << 21) & 0xFFFF0000) >> 16); //0x60F
  318. sFilterConfig.FilterMaskIdLow = 0xFFFF;
  319. sFilterConfig.FilterFIFOAssignment = 0;
  320. sFilterConfig.FilterActivation = ENABLE;
  321. if(HAL_CAN_ConfigFilter(&hcan, &sFilterConfig) != HAL_OK)
  322. {
  323. Error_Handler();
  324. }
  325. //设置过滤器组2,指定接收enviolo的0x250报文
  326. sFilterConfig.FilterNumber = 2;
  327. sFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK;//配置为掩码模式
  328. sFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT;//设置为32位宽
  329. //设置过滤寄存器
  330. sFilterConfig.FilterIdHigh = (uint16_t)((((uint32_t)ID_ENVIOLO_FILTER << 21) & 0xFFFF0000) >> 16); //0x700
  331. sFilterConfig.FilterIdLow = (uint16_t)(((uint32_t)ID_ENVIOLO_FILTER << 21) | CAN_ID_STD | CAN_RTR_DATA) & 0xFFFF;
  332. //设置屏蔽寄存器
  333. sFilterConfig.FilterMaskIdHigh = (uint16_t)((((uint32_t)ID_ENVIOLO_MASK << 21) & 0xFFFF0000) >> 16); //0x60F
  334. sFilterConfig.FilterMaskIdLow = 0xFFFF;
  335. sFilterConfig.FilterFIFOAssignment = 0;
  336. sFilterConfig.FilterActivation = ENABLE;
  337. if(HAL_CAN_ConfigFilter(&hcan, &sFilterConfig) != HAL_OK)
  338. {
  339. Error_Handler();
  340. }
  341. }
  342. void HAL_CAN_ErrorCallback(CAN_HandleTypeDef *hcan)
  343. {
  344. __HAL_CAN_ENABLE_IT(hcan, CAN_IT_FMP0);
  345. }
  346. /* USER CODE END 1 */
  347. /**
  348. * @}
  349. */
  350. /**
  351. * @}
  352. */
  353. /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/