sentry_chassis_hzz/bsp/bsp_can.c

148 lines
5.1 KiB
C
Raw Normal View History

2022-10-20 17:13:02 +08:00
#include "bsp_can.h"
#include "main.h"
#include <stdlib.h>
#include "memory.h"
2022-10-30 22:19:13 +08:00
/* can instance ptrs storage, used for recv callback */
2022-11-11 21:20:58 +08:00
static can_instance *instance[MX_REGISTER_DEVICE_CNT] = {NULL};
2022-10-31 12:14:47 +08:00
/* ----------------two static function called by CANRegister()-------------------- */
2022-10-20 17:13:02 +08:00
/**
* @brief add filter to receive mesg with specific ID,called by CANRegister()
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
* @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
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
* @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.
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
* @param _instance can instance owned by specific module
*/
2022-10-31 12:14:47 +08:00
static void CANAddFilter(can_instance *_instance)
2022-10-20 17:13:02 +08:00
{
CAN_FilterTypeDef can_filter_conf;
2022-10-31 12:14:47 +08:00
static uint8_t can1_filter_idx = 0, can2_filter_idx = 14;
2022-10-20 17:13:02 +08:00
can_filter_conf.FilterMode = CAN_FILTERMODE_IDLIST;
can_filter_conf.FilterScale = CAN_FILTERSCALE_16BIT;
2022-11-23 22:23:58 +08:00
can_filter_conf.FilterFIFOAssignment = (_instance->tx_id & 1) ? CAN_RX_FIFO0 : CAN_RX_FIFO1;
2022-10-20 17:13:02 +08:00
can_filter_conf.SlaveStartFilterBank = 14;
can_filter_conf.FilterIdLow = _instance->rx_id << 5;
2022-10-31 12:14:47 +08:00
can_filter_conf.FilterBank = _instance->can_handle == &hcan1 ? (can1_filter_idx++) : (can2_filter_idx++);
2022-10-20 17:13:02 +08:00
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
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
* @note this func will handle all these thing automatically
* there is no need to worry about hardware initialization, we do these for you!
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
*/
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);
}
2022-10-31 12:14:47 +08:00
/* -----------------------two extern callable function -----------------------*/
2022-10-30 22:19:13 +08:00
void CANRegister(can_instance *ins, can_instance_config_s* config)
2022-10-20 17:13:02 +08:00
{
static uint8_t idx;
2022-10-31 12:14:47 +08:00
if (!idx)
2022-10-20 17:13:02 +08:00
{
CANServiceInit();
}
2022-11-09 20:24:22 +08:00
instance[idx] = ins;
2022-10-30 22:19:13 +08:00
instance[idx]->txconf.StdId = config->tx_id;
2022-10-31 12:14:47 +08:00
instance[idx]->txconf.IDE = CAN_ID_STD;
instance[idx]->txconf.RTR = CAN_RTR_DATA;
instance[idx]->txconf.DLC = 0x08;
2022-10-30 22:19:13 +08:00
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;
2022-10-30 22:19:13 +08:00
2022-11-09 20:24:22 +08:00
CANAddFilter(instance[idx++]);
2022-10-20 17:13:02 +08:00
}
2022-10-31 12:14:47 +08:00
void CANTransmit(can_instance *_instance)
2022-10-20 17:13:02 +08:00
{
2022-10-31 12:14:47 +08:00
while (HAL_CAN_GetTxMailboxesFreeLevel(_instance->can_handle) == 0)
;
2022-10-30 22:19:13 +08:00
HAL_CAN_AddTxMessage(_instance->can_handle, &_instance->txconf, _instance->tx_buff, &_instance->tx_mailbox);
2022-10-20 17:13:02 +08:00
}
void CANSetDLC(can_instance *_instance, uint8_t length)
{
if (length > 8)
while (1)
;
_instance->txconf.DLC = length;
}
2022-10-30 22:19:13 +08:00
/* -----------------------belows are callback definitions--------------------------*/
2022-10-20 17:13:02 +08:00
/**
* @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
2022-10-31 12:14:47 +08:00
*
* @param _hcan
2022-10-20 17:13:02 +08:00
* @param fifox passed to HAL_CAN_GetRxMessage() to get mesg from a specific fifo
*/
2022-10-31 12:14:47 +08:00
static void CANFIFOxCallback(CAN_HandleTypeDef *_hcan, uint32_t fifox)
2022-10-20 17:13:02 +08:00
{
uint8_t can_rx_buff[8];
CAN_RxHeaderTypeDef rxconf;
2022-10-31 12:14:47 +08:00
HAL_CAN_GetRxMessage(_hcan, fifox, &rxconf, can_rx_buff);
2022-10-20 17:13:02 +08:00
for (size_t i = 0; i < DEVICE_CAN_CNT; i++)
{
if (instance[i] != NULL)
2022-10-20 17:13:02 +08:00
{
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;
}
2022-10-20 17:13:02 +08:00
}
else
break;
2022-10-31 12:14:47 +08:00
}
2022-10-20 17:13:02 +08:00
}
2022-10-30 22:19:13 +08:00
/* ATTENTION: two CAN devices in STM32 share two FIFOs */
2022-10-20 17:13:02 +08:00
/* 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
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
* @param hcan CAN handle indicate which device the oddest mesg in FIFO_0 comes from
*/
2022-10-31 12:14:47 +08:00
void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
2022-10-20 17:13:02 +08:00
{
CANFIFOxCallback(hcan, CAN_RX_FIFO0);
}
/**
* @brief rx fifo callback. Once FIFO_1 is full,this func would be called
2022-10-31 12:14:47 +08:00
*
2022-10-20 17:13:02 +08:00
* @param hcan CAN handle indicate which device the oddest mesg in FIFO_1 comes from
*/
2022-10-31 12:14:47 +08:00
void HAL_CAN_RxFifo1MsgPendingCallback(CAN_HandleTypeDef *hcan)
2022-10-20 17:13:02 +08:00
{
CANFIFOxCallback(hcan, CAN_RX_FIFO1);
}