#include "bsp_can.h" #include "main.h" #include #include "memory.h" /* can instance ptrs storage, used for recv callback */ static can_instance *instance[MX_REGISTER_DEVICE_CNT] = {NULL}; /* ----------------two static function called by CANRegister()-------------------- */ /** * @brief add filter to receive mesg with specific ID,called by CANRegister() * * @note there are total 28 filter and 2 FIFO in bxCAN of STM32F4 series product. * here, we assign the former 14 to CAN1 and the rest for CAN2 * when initializing, module with odd ID will be assigned to FIFO0 while even one to FIFO1 * those modules which registered in CAN1 would use Filter0-13, while CAN2 use Filter14-27 * * @attention you don't have to fully understand what this function done, cause it is basically * for initialization.Enjoy developing without caring about the infrastructure! * if you really want to know what is happeng, contact author. * * @param _instance can instance owned by specific module */ static void CANAddFilter(can_instance *_instance) { CAN_FilterTypeDef can_filter_conf; static uint8_t can1_filter_idx = 0, can2_filter_idx = 14; can_filter_conf.FilterMode = CAN_FILTERMODE_IDLIST; can_filter_conf.FilterScale = CAN_FILTERSCALE_16BIT; can_filter_conf.FilterFIFOAssignment = (_instance->tx_id & 1) ? CAN_RX_FIFO0 : CAN_RX_FIFO1; can_filter_conf.SlaveStartFilterBank = 14; can_filter_conf.FilterIdLow = _instance->rx_id << 5; can_filter_conf.FilterBank = _instance->can_handle == &hcan1 ? (can1_filter_idx++) : (can2_filter_idx++); can_filter_conf.FilterActivation = CAN_FILTER_ENABLE; HAL_CAN_ConfigFilter(_instance->can_handle, &can_filter_conf); } /** * @brief called by CANRegister before the first module being registered * * @note this func will handle all these thing automatically * there is no need to worry about hardware initialization, we do these for you! * */ static void CANServiceInit() { HAL_CAN_Start(&hcan1); HAL_CAN_ActivateNotification(&hcan1, CAN_IT_RX_FIFO0_MSG_PENDING); HAL_CAN_ActivateNotification(&hcan1, CAN_IT_RX_FIFO1_MSG_PENDING); HAL_CAN_Start(&hcan2); HAL_CAN_ActivateNotification(&hcan2, CAN_IT_RX_FIFO0_MSG_PENDING); HAL_CAN_ActivateNotification(&hcan2, CAN_IT_RX_FIFO1_MSG_PENDING); } /* ----------------------- two extern callable function -----------------------*/ can_instance* CANRegister(can_instance_config_s* config) { static uint8_t idx; if (!idx) { CANServiceInit(); } instance[idx] = (can_instance*)malloc(sizeof(can_instance)); memset(instance[idx],0,sizeof(can_instance)); instance[idx]->txconf.StdId = config->tx_id; instance[idx]->txconf.IDE = CAN_ID_STD; instance[idx]->txconf.RTR = CAN_RTR_DATA; instance[idx]->txconf.DLC = 0x08; // 默认发送长度为8 instance[idx]->can_handle = config->can_handle; instance[idx]->tx_id = config->tx_id; instance[idx]->rx_id = config->rx_id; instance[idx]->can_module_callback = config->can_module_callback; CANAddFilter(instance[idx]); return instance[idx++]; } void CANTransmit(can_instance *_instance) { while (HAL_CAN_GetTxMailboxesFreeLevel(_instance->can_handle) == 0) ; HAL_CAN_AddTxMessage(_instance->can_handle, &_instance->txconf, _instance->tx_buff, &_instance->tx_mailbox); } void CANSetDLC(can_instance *_instance, uint8_t length) { if (length > 8) while (1) ; _instance->txconf.DLC = length; } /* -----------------------belows are callback definitions--------------------------*/ /** * @brief this func will recv data from @param:fifox to a tmp can_rx_buff * then, all the instances will be polling to check which should recv this pack of data * * @param _hcan * @param fifox passed to HAL_CAN_GetRxMessage() to get mesg from a specific fifo */ static void CANFIFOxCallback(CAN_HandleTypeDef *_hcan, uint32_t fifox) { uint8_t can_rx_buff[8]; CAN_RxHeaderTypeDef rxconf; HAL_CAN_GetRxMessage(_hcan, fifox, &rxconf, can_rx_buff); for (size_t i = 0; i < DEVICE_CAN_CNT; i++) { if (instance[i] != NULL) { if (_hcan == instance[i]->can_handle && rxconf.StdId == instance[i]->rx_id) { instance[i]->rx_len=rxconf.DLC; memcpy(instance[i]->rx_buff, can_rx_buff, rxconf.DLC); instance[i]->can_module_callback(instance[i]); break; } } else break; } } /* ATTENTION: two CAN devices in STM32 share two FIFOs */ /* functions below will call CANFIFOxCallback() to further process message from a specific CAN device */ /** * @brief rx fifo callback. Once FIFO_0 is full,this func would be called * * @param hcan CAN handle indicate which device the oddest mesg in FIFO_0 comes from */ void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan) { CANFIFOxCallback(hcan, CAN_RX_FIFO0); } /** * @brief rx fifo callback. Once FIFO_1 is full,this func would be called * * @param hcan CAN handle indicate which device the oddest mesg in FIFO_1 comes from */ void HAL_CAN_RxFifo1MsgPendingCallback(CAN_HandleTypeDef *hcan) { CANFIFOxCallback(hcan, CAN_RX_FIFO1); }