/********************************************************************** * $Id$ lpc17xx_can.c 2011-03-09 *//** * @file lpc17xx_can.c * @brief Contains all functions support for CAN firmware library on LPC17xx * @version 3.3 * @date 09. March. 2011 * @author NXP MCU SW Application Team * * Copyright(C) 2011, NXP Semiconductor * All rights reserved. * *********************************************************************** * Software that is described herein is for illustrative purposes only * which provides customers with programming information regarding the * products. This software is supplied "AS IS" without any warranties. * NXP Semiconductors assumes no responsibility or liability for the * use of the software, conveys no license or title under any patent, * copyright, or mask work right to the product. NXP Semiconductors * reserves the right to make changes in the software without * notification. NXP Semiconductors also make no representation or * warranty that such application will be suitable for the specified * use without further testing or modification. * Permission to use, copy, modify, and distribute this software and its * documentation is hereby granted, under NXP Semiconductors' * relevant copyright in the software, without fee, provided that it * is used in conjunction with NXP Semiconductors microcontrollers. This * copyright, permission, and disclaimer notice must appear in all copies of * this code. **********************************************************************/ /* Peripheral group ----------------------------------------------------------- */ /** @addtogroup CAN * @{ */ /* Includes ------------------------------------------------------------------- */ #include "lpc17xx_can.h" #include "lpc17xx_clkpwr.h" /* If this source file built with example, the LPC17xx FW library configuration * file in each example directory ("lpc17xx_libcfg.h") must be included, * otherwise the default FW library configuration file must be included instead */ #ifdef __BUILD_WITH_EXAMPLE__ #include "lpc17xx_libcfg.h" #else #include "lpc17xx_libcfg_default.h" #endif /* __BUILD_WITH_EXAMPLE__ */ #ifdef _CAN /* Private Variables ---------------------------------------------------------- */ /** @defgroup CAN_Private_Variables CAN Private Variables * @{ */ FunctionalState FULLCAN_ENABLE; /* Counts number of filters (CAN message objects) used */ uint16_t CANAF_FullCAN_cnt = 0; uint16_t CANAF_std_cnt = 0; uint16_t CANAF_gstd_cnt = 0; uint16_t CANAF_ext_cnt = 0; uint16_t CANAF_gext_cnt = 0; /* End of Private Variables ----------------------------------------------------*/ /** * @} */ /* Private Variables ---------------------------------------------------------- */ static void can_SetBaudrate (LPC_CAN_TypeDef *CANx, uint32_t baudrate); /*********************************************************************//** * @brief Setting CAN baud rate (bps) * @param[in] CANx point to LPC_CAN_TypeDef object, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] baudrate: is the baud rate value will be set * @return None ***********************************************************************/ static void can_SetBaudrate (LPC_CAN_TypeDef *CANx, uint32_t baudrate) { uint32_t result = 0; uint8_t NT, TSEG1, TSEG2, BRFail; uint32_t CANPclk = 0; uint32_t BRP; CHECK_PARAM(PARAM_CANx(CANx)); if (CANx == LPC_CAN1) { CANPclk = CLKPWR_GetPCLK (CLKPWR_PCLKSEL_CAN1); } else { CANPclk = CLKPWR_GetPCLK (CLKPWR_PCLKSEL_CAN2); } result = CANPclk / baudrate; /* Calculate suitable nominal time value * NT (nominal time) = (TSEG1 + TSEG2 + 3) * NT <= 24 * TSEG1 >= 2*TSEG2 */ BRFail = 1; for(NT=24;NT>0;NT=NT-2) { if ((result%NT)==0) { BRP = result / NT - 1; NT--; TSEG2 = (NT/3) - 1; TSEG1 = NT -(NT/3) - 1; BRFail = 0; break; } } if(BRFail) while(1); // Failed to calculate exact CAN baud rate /* Enter reset mode */ CANx->MOD = 0x01; /* Set bit timing * Default: SAM = 0x00; * SJW = 0x03; */ CANx->BTR = (TSEG2<<20)|(TSEG1<<16)|(3<<14)|BRP; /* Return to normal operating */ CANx->MOD = 0; } /* End of Private Functions ----------------------------------------------------*/ /* Public Functions ----------------------------------------------------------- */ /** @addtogroup CAN_Public_Functions * @{ */ /********************************************************************//** * @brief Initialize CAN peripheral with given baudrate * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] baudrate: the value of CAN baudrate will be set (bps) * @return None *********************************************************************/ void CAN_Init(LPC_CAN_TypeDef *CANx, uint32_t baudrate) { uint16_t i; CHECK_PARAM(PARAM_CANx(CANx)); if(CANx == LPC_CAN1) { /* Turn on power and clock for CAN1 */ CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCAN1, ENABLE); /* Set clock divide for CAN1 */ } else { /* Turn on power and clock for CAN1 */ CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCAN2, ENABLE); /* Set clock divide for CAN2 */ } CLKPWR_SetPCLKDiv (CLKPWR_PCLKSEL_CAN1, CLKPWR_PCLKSEL_CCLK_DIV_2); CLKPWR_SetPCLKDiv (CLKPWR_PCLKSEL_CAN2, CLKPWR_PCLKSEL_CCLK_DIV_2); CLKPWR_SetPCLKDiv (CLKPWR_PCLKSEL_ACF, CLKPWR_PCLKSEL_CCLK_DIV_2); CANx->MOD = 1; // Enter Reset Mode CANx->IER = 0; // Disable All CAN Interrupts CANx->GSR = 0; /* Request command to release Rx, Tx buffer and clear data overrun */ //CANx->CMR = CAN_CMR_AT | CAN_CMR_RRB | CAN_CMR_CDO; CANx->CMR = (1<<1)|(1<<2)|(1<<3); /* Read to clear interrupt pending in interrupt capture register */ i = CANx->ICR; CANx->MOD = 0;// Return Normal operating //Reset CANAF value LPC_CANAF->AFMR = 0x01; //clear ALUT RAM for (i = 0; i < 512; i++) { LPC_CANAF_RAM->mask[i] = 0x00; } LPC_CANAF->SFF_sa = 0x00; LPC_CANAF->SFF_GRP_sa = 0x00; LPC_CANAF->EFF_sa = 0x00; LPC_CANAF->EFF_GRP_sa = 0x00; LPC_CANAF->ENDofTable = 0x00; LPC_CANAF->AFMR = 0x00; /* Set baudrate */ can_SetBaudrate (CANx, baudrate); } /********************************************************************//** * @brief CAN deInit * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @return None *********************************************************************/ void CAN_DeInit(LPC_CAN_TypeDef *CANx) { CHECK_PARAM(PARAM_CANx(CANx)); if(CANx == LPC_CAN1) { /* Turn on power and clock for CAN1 */ CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCAN1, DISABLE); } else { /* Turn on power and clock for CAN1 */ CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCAN2, DISABLE); } } /********************************************************************//** * @brief Setup Acceptance Filter Look-Up Table * @param[in] CANAFx pointer to LPC_CANAF_TypeDef * Should be: LPC_CANAF * @param[in] AFSection the pointer to AF_SectionDef structure * It contain information about 5 sections will be install in AFLUT * @return CAN Error could be: * - CAN_OBJECTS_FULL_ERROR: No more rx or tx objects available * - CAN_AF_ENTRY_ERROR: table error-violation of ascending numerical order * - CAN_OK: ID is added into table successfully *********************************************************************/ CAN_ERROR CAN_SetupAFLUT(LPC_CANAF_TypeDef* CANAFx, AF_SectionDef* AFSection) { uint8_t ctrl1,ctrl2; uint8_t dis1, dis2; uint16_t SID, ID_temp,i, count = 0; uint32_t EID, entry, buf; uint16_t lowerSID, upperSID; uint32_t lowerEID, upperEID; AF_SectionDef afSecDef; CHECK_PARAM(PARAM_CANAFx(CANAFx)); afSecDef = *AFSection; CANAFx->AFMR = 0x01; /***** setup FullCAN Table *****/ if(afSecDef.FullCAN_Sec == NULL) { FULLCAN_ENABLE = DISABLE; } else { FULLCAN_ENABLE = ENABLE; for(i=0;i<(afSecDef.FC_NumEntry);i++) { if(count + 1 > 64) { return CAN_OBJECTS_FULL_ERROR; } ctrl1 = afSecDef.FullCAN_Sec->controller; SID = afSecDef.FullCAN_Sec->id_11; dis1 = afSecDef.FullCAN_Sec->disable; CHECK_PARAM(PARAM_CTRL(ctrl1)); CHECK_PARAM(PARAM_ID_11(SID)); CHECK_PARAM(PARAM_MSG_DISABLE(dis1)); entry = 0x00; //reset entry value if((CANAF_FullCAN_cnt & 0x00000001)==0) { if(count!=0x00) { buf = LPC_CANAF_RAM->mask[count-1]; ID_temp = (buf & 0xE7FF); //mask controller & identifier bits if(ID_temp > ((ctrl1<<13)|SID)) { return CAN_AF_ENTRY_ERROR; } } entry = (ctrl1<<29)|(dis1<<28)|(SID<<16)|(1<<27); LPC_CANAF_RAM->mask[count] &= 0x0000FFFF; LPC_CANAF_RAM->mask[count] |= entry; CANAF_FullCAN_cnt++; if(CANAF_FullCAN_cnt == afSecDef.FC_NumEntry) //this is the lastest FullCAN entry count++; } else { buf = LPC_CANAF_RAM->mask[count]; ID_temp = (buf >>16) & 0xE7FF; if(ID_temp > ((ctrl1<<13)|SID)) { return CAN_AF_ENTRY_ERROR; } entry = (ctrl1<<13)|(dis1<<12)|(SID<<0)|(1<<11); LPC_CANAF_RAM->mask[count] &= 0xFFFF0000; LPC_CANAF_RAM->mask[count]|= entry; count++; CANAF_FullCAN_cnt++; } afSecDef.FullCAN_Sec = (FullCAN_Entry *)((uint32_t)(afSecDef.FullCAN_Sec)+ sizeof(FullCAN_Entry)); } } /***** Setup Explicit Standard Frame Format Section *****/ if(afSecDef.SFF_Sec != NULL) { for(i=0;i<(afSecDef.SFF_NumEntry);i++) { if(count + 1 > 512) { return CAN_OBJECTS_FULL_ERROR; } ctrl1 = afSecDef.SFF_Sec->controller; SID = afSecDef.SFF_Sec->id_11; dis1 = afSecDef.SFF_Sec->disable; //check parameter CHECK_PARAM(PARAM_CTRL(ctrl1)); CHECK_PARAM(PARAM_ID_11(SID)); CHECK_PARAM(PARAM_MSG_DISABLE(dis1)); entry = 0x00; //reset entry value if((CANAF_std_cnt & 0x00000001)==0) { if(CANAF_std_cnt !=0 ) { buf = LPC_CANAF_RAM->mask[count-1]; ID_temp = (buf & 0xE7FF); //mask controller & identifier bits if(ID_temp > ((ctrl1<<13)|SID)) { return CAN_AF_ENTRY_ERROR; } } entry = (ctrl1<<29)|(dis1<<28)|(SID<<16); LPC_CANAF_RAM->mask[count] &= 0x0000FFFF; LPC_CANAF_RAM->mask[count] |= entry; CANAF_std_cnt++; if(CANAF_std_cnt == afSecDef.SFF_NumEntry)//if this is the last SFF entry count++; } else { buf = LPC_CANAF_RAM->mask[count]; ID_temp = (buf >>16) & 0xE7FF; if(ID_temp > ((ctrl1<<13)|SID)) { return CAN_AF_ENTRY_ERROR; } entry = (ctrl1<<13)|(dis1<<12)|(SID<<0); LPC_CANAF_RAM->mask[count] &= 0xFFFF0000; LPC_CANAF_RAM->mask[count] |= entry; count++; CANAF_std_cnt++; } afSecDef.SFF_Sec = (SFF_Entry *)((uint32_t)(afSecDef.SFF_Sec)+ sizeof(SFF_Entry)); } } /***** Setup Group of Standard Frame Format Identifier Section *****/ if(afSecDef.SFF_GPR_Sec != NULL) { for(i=0;i<(afSecDef.SFF_GPR_NumEntry);i++) { if(count + 1 > 512) { return CAN_OBJECTS_FULL_ERROR; } ctrl1 = afSecDef.SFF_GPR_Sec->controller1; ctrl2 = afSecDef.SFF_GPR_Sec->controller2; dis1 = afSecDef.SFF_GPR_Sec->disable1; dis2 = afSecDef.SFF_GPR_Sec->disable2; lowerSID = afSecDef.SFF_GPR_Sec->lowerID; upperSID = afSecDef.SFF_GPR_Sec->upperID; /* check parameter */ CHECK_PARAM(PARAM_CTRL(ctrl1)); CHECK_PARAM(PARAM_CTRL(ctrl2)); CHECK_PARAM(PARAM_MSG_DISABLE(dis1)); CHECK_PARAM(PARAM_MSG_DISABLE(dis2)); CHECK_PARAM(PARAM_ID_11(lowerSID)); CHECK_PARAM(PARAM_ID_11(upperSID)); entry = 0x00; if(CANAF_gstd_cnt!=0) { buf = LPC_CANAF_RAM->mask[count-1]; ID_temp = buf & 0xE7FF; if((ctrl1 != ctrl2)||(lowerSID > upperSID)||(ID_temp > ((ctrl1<<13)|lowerSID))) { return CAN_AF_ENTRY_ERROR; } } entry = (ctrl1 << 29)|(dis1 << 28)|(lowerSID << 16)| \ (ctrl2 << 13)|(dis2 << 12)|(upperSID << 0); LPC_CANAF_RAM->mask[count] = entry; CANAF_gstd_cnt++; count++; afSecDef.SFF_GPR_Sec = (SFF_GPR_Entry *)((uint32_t)(afSecDef.SFF_GPR_Sec)+ sizeof(SFF_GPR_Entry)); } } /***** Setup Explicit Extend Frame Format Identifier Section *****/ if(afSecDef.EFF_Sec != NULL) { for(i=0;i<(afSecDef.EFF_NumEntry);i++) { if(count + 1 > 512) { return CAN_OBJECTS_FULL_ERROR; } EID = afSecDef.EFF_Sec->ID_29; ctrl1 = afSecDef.EFF_Sec->controller; // check parameter CHECK_PARAM(PARAM_ID_29(EID)); CHECK_PARAM(PARAM_CTRL(ctrl1)); entry = (ctrl1 << 29)|(EID << 0); if(CANAF_ext_cnt != 0) { buf = LPC_CANAF_RAM->mask[count-1]; // EID_temp = buf & 0x0FFFFFFF; if(buf > entry) { return CAN_AF_ENTRY_ERROR; } } LPC_CANAF_RAM->mask[count] = entry; CANAF_ext_cnt ++; count++; afSecDef.EFF_Sec = (EFF_Entry *)((uint32_t)(afSecDef.EFF_Sec)+ sizeof(EFF_Entry)); } } /***** Setup Group of Extended Frame Format Identifier Section *****/ if(afSecDef.EFF_GPR_Sec != NULL) { for(i=0;i<(afSecDef.EFF_GPR_NumEntry);i++) { if(count + 2 > 512) { return CAN_OBJECTS_FULL_ERROR; } ctrl1 = afSecDef.EFF_GPR_Sec->controller1; ctrl2 = afSecDef.EFF_GPR_Sec->controller2; lowerEID = afSecDef.EFF_GPR_Sec->lowerEID; upperEID = afSecDef.EFF_GPR_Sec->upperEID; //check parameter CHECK_PARAM(PARAM_CTRL(ctrl1)); CHECK_PARAM(PARAM_CTRL(ctrl2)); CHECK_PARAM(PARAM_ID_29(lowerEID)); CHECK_PARAM(PARAM_ID_29(upperEID)); entry = 0x00; if(CANAF_gext_cnt != 0) { buf = LPC_CANAF_RAM->mask[count-1]; // EID_temp = buf & 0x0FFFFFFF; if((ctrl1 != ctrl2) || (lowerEID > upperEID) || (buf > ((ctrl1 << 29)|(lowerEID << 0)))) { return CAN_AF_ENTRY_ERROR; } } entry = (ctrl1 << 29)|(lowerEID << 0); LPC_CANAF_RAM->mask[count++] = entry; entry = (ctrl2 << 29)|(upperEID << 0); LPC_CANAF_RAM->mask[count++] = entry; CANAF_gext_cnt++; afSecDef.EFF_GPR_Sec = (EFF_GPR_Entry *)((uint32_t)(afSecDef.EFF_GPR_Sec)+ sizeof(EFF_GPR_Entry)); } } //update address values LPC_CANAF->SFF_sa = ((CANAF_FullCAN_cnt + 1)>>1)<<2; LPC_CANAF->SFF_GRP_sa = LPC_CANAF->SFF_sa + (((CANAF_std_cnt+1)>>1)<< 2); LPC_CANAF->EFF_sa = LPC_CANAF->SFF_GRP_sa + (CANAF_gstd_cnt << 2); LPC_CANAF->EFF_GRP_sa = LPC_CANAF->EFF_sa + (CANAF_ext_cnt << 2); LPC_CANAF->ENDofTable = LPC_CANAF->EFF_GRP_sa + (CANAF_gext_cnt << 3); if(FULLCAN_ENABLE == DISABLE) { LPC_CANAF->AFMR = 0x00; // Normal mode } else { LPC_CANAF->AFMR = 0x04; } return CAN_OK; } /********************************************************************//** * @brief Add Explicit ID into AF Look-Up Table dynamically. * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] id: The ID of entry will be added * @param[in] format: is the type of ID Frame Format, should be: * - STD_ID_FORMAT: 11-bit ID value * - EXT_ID_FORMAT: 29-bit ID value * @return CAN Error, could be: * - CAN_OBJECTS_FULL_ERROR: No more rx or tx objects available * - CAN_ID_EXIT_ERROR: ID exited in table * - CAN_OK: ID is added into table successfully *********************************************************************/ CAN_ERROR CAN_LoadExplicitEntry(LPC_CAN_TypeDef* CANx, uint32_t id, CAN_ID_FORMAT_Type format) { uint32_t tmp0 = 0; uint32_t buf0=0, buf1=0; int16_t cnt1=0, cnt2=0, bound1=0, total=0; CHECK_PARAM(PARAM_CANx(CANx)); CHECK_PARAM(PARAM_ID_FORMAT(format)); if (CANx == LPC_CAN1) { tmp0 = 0; } else if (CANx == LPC_CAN2) { tmp0 = 1; } /* Acceptance Filter Memory full - return */ total =((CANAF_FullCAN_cnt+1)>>1)+ CANAF_FullCAN_cnt*3 +((CANAF_std_cnt + 1) >> 1)+ \ CANAF_gstd_cnt + CANAF_ext_cnt + (CANAF_gext_cnt<<1); if (total >= 512){ //don't have enough space return CAN_OBJECTS_FULL_ERROR; } /* Setup Acceptance Filter Configuration Acceptance Filter Mode Register = Off */ LPC_CANAF->AFMR = 0x00000001; /*********** Add Explicit Standard Identifier Frame Format entry *********/ if(format == STD_ID_FORMAT) { id &= 0x07FF; id |= (tmp0 << 13); /* Add controller number */ /* Move all remaining sections one place up if new entry will increase FullCAN list */ if ((CANAF_std_cnt & 0x0001) == 0) { cnt1 = ((CANAF_FullCAN_cnt+1)>>1)+((CANAF_std_cnt+1)>>1); bound1 = total - cnt1; buf0 = LPC_CANAF_RAM->mask[cnt1]; while(bound1--) { cnt1++; buf1 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = buf0; buf0 = buf1; } } if (CANAF_std_cnt == 0) { cnt2 = (CANAF_FullCAN_cnt + 1)>>1; /* For entering first ID */ LPC_CANAF_RAM->mask[cnt2] = 0x0000FFFF | (id << 16); } else if (CANAF_std_cnt == 1) { cnt2 = (CANAF_FullCAN_cnt + 1)>>1; /* For entering second ID */ if (((LPC_CANAF_RAM->mask[cnt2] >> 16)& 0xE7FF) > id) { LPC_CANAF_RAM->mask[cnt2] = (LPC_CANAF_RAM->mask[cnt2] >> 16) | (id << 16); } else { LPC_CANAF_RAM->mask[cnt2] = (LPC_CANAF_RAM->mask[cnt2] & 0xFFFF0000) | id; } } else { /* Find where to insert new ID */ cnt1 = (CANAF_FullCAN_cnt+1)>>1; cnt2 = CANAF_std_cnt; bound1 = ((CANAF_FullCAN_cnt+1)>>1)+((CANAF_std_cnt+1)>>1); while (cnt1 < bound1) { /* Loop through standard existing IDs */ if (((LPC_CANAF_RAM->mask[cnt1] >> 16) & 0xE7FF) > id) { cnt2 = cnt1 * 2; break; } if ((LPC_CANAF_RAM->mask[cnt1] & 0x0000E7FF) > id) { cnt2 = cnt1 * 2 + 1; break; } cnt1++; } /* cnt1 = U32 where to insert new ID */ /* cnt2 = U16 where to insert new ID */ if (cnt1 == bound1) { /* Adding ID as last entry */ /* Even number of IDs exists */ if ((CANAF_std_cnt & 0x0001) == 0) { LPC_CANAF_RAM->mask[cnt1] = 0x0000FFFF | (id << 16); } /* Odd number of IDs exists */ else { LPC_CANAF_RAM->mask[cnt1] = (LPC_CANAF_RAM->mask[cnt1] & 0xFFFF0000) | id; } } else { buf0 = LPC_CANAF_RAM->mask[cnt1]; /* Remember current entry */ if ((cnt2 & 0x0001) == 0) { /* Insert new mask to even address*/ buf1 = (id << 16) | (buf0 >> 16); } else { /* Insert new mask to odd address */ buf1 = (buf0 & 0xFFFF0000) | id; } LPC_CANAF_RAM->mask[cnt1] = buf1;/* Insert mask */ bound1 = ((CANAF_FullCAN_cnt+1)>>1)+((CANAF_std_cnt+1)>>1)-1; /* Move all remaining standard mask entries one place up */ while (cnt1 < bound1) { cnt1++; buf1 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = (buf1 >> 16) | (buf0 << 16); buf0 = buf1; } if ((CANAF_std_cnt & 0x0001) == 0) { /* Even number of IDs exists */ LPC_CANAF_RAM->mask[cnt1+1] = (buf0 <<16) |(0x0000FFFF); } } } CANAF_std_cnt++; //update address values LPC_CANAF->SFF_GRP_sa +=0x04 ; LPC_CANAF->EFF_sa +=0x04 ; LPC_CANAF->EFF_GRP_sa +=0x04; LPC_CANAF->ENDofTable +=0x04; } /*********** Add Explicit Extended Identifier Frame Format entry *********/ else { /* Add controller number */ id |= (tmp0) << 29; cnt1 = ((CANAF_FullCAN_cnt+1)>>1)+(((CANAF_std_cnt + 1) >> 1) + CANAF_gstd_cnt); cnt2 = 0; while (cnt2 < CANAF_ext_cnt) { /* Loop through extended existing masks*/ if (LPC_CANAF_RAM->mask[cnt1] > id) { break; } cnt1++;/* cnt1 = U32 where to insert new mask */ cnt2++; } buf0 = LPC_CANAF_RAM->mask[cnt1]; /* Remember current entry */ LPC_CANAF_RAM->mask[cnt1] = id; /* Insert mask */ CANAF_ext_cnt++; bound1 = total; /* Move all remaining extended mask entries one place up*/ while (cnt2 < bound1) { cnt1++; cnt2++; buf1 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = buf0; buf0 = buf1; } /* update address values */ LPC_CANAF->EFF_GRP_sa += 4; LPC_CANAF->ENDofTable += 4; } if(CANAF_FullCAN_cnt == 0) //not use FullCAN mode { LPC_CANAF->AFMR = 0x00;//not use FullCAN mode } else { LPC_CANAF->AFMR = 0x04; } return CAN_OK; } /********************************************************************//** * @brief Load FullCAN entry into AFLUT * @param[in] CANx: CAN peripheral selected, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] id: identifier of entry that will be added * @return CAN_ERROR, could be: * - CAN_OK: loading is successful * - CAN_ID_EXIT_ERROR: ID exited in FullCAN Section * - CAN_OBJECTS_FULL_ERROR: no more space available *********************************************************************/ CAN_ERROR CAN_LoadFullCANEntry (LPC_CAN_TypeDef* CANx, uint16_t id) { uint32_t ctrl0 = 0; uint32_t buf0=0, buf1=0, buf2=0; uint32_t tmp0=0, tmp1=0, tmp2=0; int16_t cnt1=0, cnt2=0, bound1=0, total=0; CHECK_PARAM(PARAM_CANx(CANx)); if (CANx == LPC_CAN1) { ctrl0 = 0; } else if (CANx == LPC_CAN2) { ctrl0 = 1; } /* Acceptance Filter Memory full - return */ total =((CANAF_FullCAN_cnt+1)>>1)+ CANAF_FullCAN_cnt*3 +((CANAF_std_cnt + 1) >> 1)+ \ CANAF_gstd_cnt + CANAF_ext_cnt + (CANAF_gext_cnt<<1); //don't have enough space for this fullCAN Entry and its Object(3*32 bytes) if ((total >=508)||(CANAF_FullCAN_cnt>=64)){ return CAN_OBJECTS_FULL_ERROR; } /* Setup Acceptance Filter Configuration Acceptance Filter Mode Register = Off */ LPC_CANAF->AFMR = 0x00000001; /* Add mask for standard identifiers */ id &= 0x07FF; id |= (ctrl0 << 13) | (1 << 11); /* Add controller number */ // total = ((CANAF_std_cnt + 1) >> 1)+ CANAF_gstd_cnt + CANAF_ext_cnt + (CANAF_gext_cnt<<1); /* Move all remaining sections one place up if new entry will increase FullCAN list */ if (((CANAF_FullCAN_cnt & 0x0001) == 0)&&(total!=0)) { //then remove remaining section cnt1 = (CANAF_FullCAN_cnt >> 1); bound1 = total; buf0 = LPC_CANAF_RAM->mask[cnt1]; while (bound1--) { cnt1++; buf1 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = buf0; buf0 = buf1; } } if (CANAF_FullCAN_cnt == 0) { /* For entering first ID */ LPC_CANAF_RAM->mask[0] = 0x0000FFFF | (id << 16); } else if (CANAF_FullCAN_cnt == 1) { /* For entering second ID */ if (((LPC_CANAF_RAM->mask[0] >> 16)& 0xE7FF) > id) { LPC_CANAF_RAM->mask[0] = (LPC_CANAF_RAM->mask[0] >> 16) | (id << 16); } else { LPC_CANAF_RAM->mask[0] = (LPC_CANAF_RAM->mask[0] & 0xFFFF0000) | id; } } else { /* Find where to insert new ID */ cnt1 = 0; cnt2 = CANAF_FullCAN_cnt; bound1 = (CANAF_FullCAN_cnt - 1) >> 1; while (cnt1 <= bound1) { /* Loop through standard existing IDs */ if (((LPC_CANAF_RAM->mask[cnt1] >> 16) & 0xE7FF) > (id & 0xE7FF)) { cnt2 = cnt1 * 2; break; } if ((LPC_CANAF_RAM->mask[cnt1] & 0x0000E7FF) > (id & 0xE7FF)) { cnt2 = cnt1 * 2 + 1; break; } cnt1++; } /* cnt1 = U32 where to insert new ID */ /* cnt2 = U16 where to insert new ID */ if (cnt1 > bound1) { /* Adding ID as last entry */ /* Even number of IDs exists */ if ((CANAF_FullCAN_cnt & 0x0001) == 0) { LPC_CANAF_RAM->mask[cnt1] = 0x0000FFFF | (id << 16); } /* Odd number of IDs exists */ else { LPC_CANAF_RAM->mask[cnt1] = (LPC_CANAF_RAM->mask[cnt1] & 0xFFFF0000) | id; } } else { buf0 = LPC_CANAF_RAM->mask[cnt1]; /* Remember current entry */ if ((cnt2 & 0x0001) == 0) { /* Insert new mask to even address*/ buf1 = (id << 16) | (buf0 >> 16); } else { /* Insert new mask to odd address */ buf1 = (buf0 & 0xFFFF0000) | id; } LPC_CANAF_RAM->mask[cnt1] = buf1;/* Insert mask */ bound1 = CANAF_FullCAN_cnt >> 1; /* Move all remaining standard mask entries one place up */ while (cnt1 < bound1) { cnt1++; buf1 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = (buf1 >> 16) | (buf0 << 16); buf0 = buf1; } if ((CANAF_FullCAN_cnt & 0x0001) == 0) { /* Even number of IDs exists */ LPC_CANAF_RAM->mask[cnt1] = (LPC_CANAF_RAM->mask[cnt1] & 0xFFFF0000) | (0x0000FFFF); } } } //restruct FulCAN Object Section bound1 = CANAF_FullCAN_cnt - cnt2; cnt1 = total - (CANAF_FullCAN_cnt)*3 + cnt2*3 + 1; buf0 = LPC_CANAF_RAM->mask[cnt1]; buf1 = LPC_CANAF_RAM->mask[cnt1+1]; buf2 = LPC_CANAF_RAM->mask[cnt1+2]; LPC_CANAF_RAM->mask[cnt1]=LPC_CANAF_RAM->mask[cnt1+1]= LPC_CANAF_RAM->mask[cnt1+2]=0x00; cnt1+=3; while(bound1--) { tmp0 = LPC_CANAF_RAM->mask[cnt1]; tmp1 = LPC_CANAF_RAM->mask[cnt1+1]; tmp2 = LPC_CANAF_RAM->mask[cnt1+2]; LPC_CANAF_RAM->mask[cnt1]= buf0; LPC_CANAF_RAM->mask[cnt1+1]= buf1; LPC_CANAF_RAM->mask[cnt1+2]= buf2; buf0 = tmp0; buf1 = tmp1; buf2 = tmp2; cnt1+=3; } CANAF_FullCAN_cnt++; //update address values LPC_CANAF->SFF_sa +=0x04; LPC_CANAF->SFF_GRP_sa +=0x04 ; LPC_CANAF->EFF_sa +=0x04 ; LPC_CANAF->EFF_GRP_sa +=0x04; LPC_CANAF->ENDofTable +=0x04; LPC_CANAF->AFMR = 0x04; return CAN_OK; } /********************************************************************//** * @brief Load Group entry into AFLUT * @param[in] CANx: CAN peripheral selected, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] lowerID, upperID: lower and upper identifier of entry * @param[in] format: type of ID format, should be: * - STD_ID_FORMAT: Standard ID format (11-bit value) * - EXT_ID_FORMAT: Extended ID format (29-bit value) * @return CAN_ERROR, could be: * - CAN_OK: loading is successful * - CAN_CONFLICT_ID_ERROR: Conflict ID occurs * - CAN_OBJECTS_FULL_ERROR: no more space available *********************************************************************/ CAN_ERROR CAN_LoadGroupEntry(LPC_CAN_TypeDef* CANx, uint32_t lowerID, \ uint32_t upperID, CAN_ID_FORMAT_Type format) { uint16_t tmp = 0; uint32_t buf0, buf1, entry1, entry2, LID,UID; int16_t cnt1, bound1, total; //LPC_CANAF_RAM_TypeDef *AFLUTTest = LPC_CANAF_RAM; CHECK_PARAM(PARAM_CANx(CANx)); CHECK_PARAM(PARAM_ID_FORMAT(format)); if(lowerID > upperID) return CAN_CONFLICT_ID_ERROR; if(CANx == LPC_CAN1) { tmp = 0; } else { tmp = 1; } total =((CANAF_FullCAN_cnt+1)>>1)+ CANAF_FullCAN_cnt*3 +((CANAF_std_cnt + 1) >> 1)+ \ CANAF_gstd_cnt + CANAF_ext_cnt + (CANAF_gext_cnt<<1); /* Setup Acceptance Filter Configuration Acceptance Filter Mode Register = Off */ LPC_CANAF->AFMR = 0x00000001; /*********Add Group of Standard Identifier Frame Format************/ if(format == STD_ID_FORMAT) { if ((total >= 512)){//don't have enough space return CAN_OBJECTS_FULL_ERROR; } lowerID &=0x7FF; //mask ID upperID &=0x7FF; entry1 = (tmp << 29)|(lowerID << 16)|(tmp << 13)|(upperID << 0); cnt1 = ((CANAF_FullCAN_cnt+1)>>1) + ((CANAF_std_cnt + 1) >> 1); //if this is the first Group standard ID entry if(CANAF_gstd_cnt == 0) { LPC_CANAF_RAM->mask[cnt1] = entry1; } else { //find the position to add new Group entry bound1 = ((CANAF_FullCAN_cnt+1)>>1) + ((CANAF_std_cnt + 1) >> 1) + CANAF_gstd_cnt; while(cnt1 < bound1) { //compare controller first while((LPC_CANAF_RAM->mask[cnt1] >> 29)< (entry1 >> 29))//increase until meet greater or equal controller cnt1++; buf0 = LPC_CANAF_RAM->mask[cnt1]; if((LPC_CANAF_RAM->mask[cnt1] >> 29)> (entry1 >> 29)) //meet greater controller { //add at this position LPC_CANAF_RAM->mask[cnt1] = entry1; break; } else //meet equal controller { LID = (buf0 >> 16)&0x7FF; UID = buf0 & 0x7FF; if (upperID <= LID) { //add new entry before this entry LPC_CANAF_RAM->mask[cnt1] = entry1; break; } else if (lowerID >= UID) { //load next entry to compare cnt1 ++; } else return CAN_CONFLICT_ID_ERROR; } } if(cnt1 >= bound1) { //add new entry at the last position in this list buf0 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = entry1; } //remove all remaining entry of this section one place up bound1 = total - cnt1; while(bound1--) { cnt1++; buf1 = LPC_CANAF_RAM->mask[cnt1]; LPC_CANAF_RAM->mask[cnt1] = buf0; buf0 = buf1; } } CANAF_gstd_cnt++; //update address values LPC_CANAF->EFF_sa +=0x04 ; LPC_CANAF->EFF_GRP_sa +=0x04; LPC_CANAF->ENDofTable +=0x04; } /*********Add Group of Extended Identifier Frame Format************/ else { if ((total >= 511)){//don't have enough space return CAN_OBJECTS_FULL_ERROR; } lowerID &= 0x1FFFFFFF; //mask ID upperID &= 0x1FFFFFFF; entry1 = (tmp << 29)|(lowerID << 0); entry2 = (tmp << 29)|(upperID << 0); cnt1 = ((CANAF_FullCAN_cnt+1)>>1) + ((CANAF_std_cnt + 1) >> 1) + CANAF_gstd_cnt + CANAF_ext_cnt; //if this is the first Group standard ID entry if(CANAF_gext_cnt == 0) { LPC_CANAF_RAM->mask[cnt1] = entry1; LPC_CANAF_RAM->mask[cnt1+1] = entry2; } else { //find the position to add new Group entry bound1 = ((CANAF_FullCAN_cnt+1)>>1) + ((CANAF_std_cnt + 1) >> 1) + CANAF_gstd_cnt \ + CANAF_ext_cnt + (CANAF_gext_cnt<<1); while(cnt1 < bound1) { while((LPC_CANAF_RAM->mask[cnt1] >>29)< tmp) //increase until meet greater or equal controller cnt1++; buf0 = LPC_CANAF_RAM->mask[cnt1]; buf1 = LPC_CANAF_RAM->mask[cnt1+1]; if((LPC_CANAF_RAM->mask[cnt1] >> 29)> (entry1 >> 29)) //meet greater controller { //add at this position LPC_CANAF_RAM->mask[cnt1] = entry1; LPC_CANAF_RAM->mask[++cnt1] = entry2; break; } else //meet equal controller { LID = buf0 & 0x1FFFFFFF; //mask ID UID = buf1 & 0x1FFFFFFF; if (upperID <= LID) { //add new entry before this entry LPC_CANAF_RAM->mask[cnt1] = entry1; LPC_CANAF_RAM->mask[++cnt1] = entry2; break; } else if (lowerID >= UID) { //load next entry to compare cnt1 +=2; } else return CAN_CONFLICT_ID_ERROR; } } if(cnt1 >= bound1) { //add new entry at the last position in this list buf0 = LPC_CANAF_RAM->mask[cnt1]; buf1 = LPC_CANAF_RAM->mask[cnt1+1]; LPC_CANAF_RAM->mask[cnt1] = entry1; LPC_CANAF_RAM->mask[++cnt1] = entry2; } //remove all remaining entry of this section two place up bound1 = total - cnt1 + 1; cnt1++; while(bound1>0) { entry1 = LPC_CANAF_RAM->mask[cnt1]; entry2 = LPC_CANAF_RAM->mask[cnt1+1]; LPC_CANAF_RAM->mask[cnt1] = buf0; LPC_CANAF_RAM->mask[cnt1+1] = buf1; buf0 = entry1; buf1 = entry2; cnt1 +=2; bound1 -=2; } } CANAF_gext_cnt++; //update address values LPC_CANAF->ENDofTable +=0x08; } LPC_CANAF->AFMR = 0x04; return CAN_OK; } /********************************************************************//** * @brief Remove AFLUT entry (FullCAN entry and Explicit Standard entry) * @param[in] EntryType: the type of entry that want to remove, should be: * - FULLCAN_ENTRY * - EXPLICIT_STANDARD_ENTRY * - GROUP_STANDARD_ENTRY * - EXPLICIT_EXTEND_ENTRY * - GROUP_EXTEND_ENTRY * @param[in] position: the position of this entry in its section * Note: the first position is 0 * @return CAN_ERROR, could be: * - CAN_OK: removing is successful * - CAN_ENTRY_NOT_EXIT_ERROR: entry want to remove is not exit *********************************************************************/ CAN_ERROR CAN_RemoveEntry(AFLUT_ENTRY_Type EntryType, uint16_t position) { uint16_t cnt, bound, total; uint32_t buf0, buf1; CHECK_PARAM(PARAM_AFLUT_ENTRY_TYPE(EntryType)); CHECK_PARAM(PARAM_POSITION(position)); /* Setup Acceptance Filter Configuration Acceptance Filter Mode Register = Off */ LPC_CANAF->AFMR = 0x00000001; total = ((CANAF_FullCAN_cnt+1)>>1)+((CANAF_std_cnt + 1) >> 1) + \ CANAF_gstd_cnt + CANAF_ext_cnt + (CANAF_gext_cnt<<1); /************** Remove FullCAN Entry *************/ if(EntryType == FULLCAN_ENTRY) { if((CANAF_FullCAN_cnt==0)||(position >= CANAF_FullCAN_cnt)) { return CAN_ENTRY_NOT_EXIT_ERROR; } else { cnt = position >> 1; buf0 = LPC_CANAF_RAM->mask[cnt]; bound = (CANAF_FullCAN_cnt - position -1)>>1; if((position & 0x0001) == 0) //event position { while(bound--) { //remove all remaining FullCAN entry one place down buf1 = LPC_CANAF_RAM->mask[cnt+1]; LPC_CANAF_RAM->mask[cnt] = (buf1 >> 16) | (buf0 << 16); buf0 = buf1; cnt++; } } else //odd position { while(bound--) { //remove all remaining FullCAN entry one place down buf1 = LPC_CANAF_RAM->mask[cnt+1]; LPC_CANAF_RAM->mask[cnt] = (buf0 & 0xFFFF0000)|(buf1 >> 16); LPC_CANAF_RAM->mask[cnt+1] = LPC_CANAF_RAM->mask[cnt+1] << 16; buf0 = buf1<<16; cnt++; } } if((CANAF_FullCAN_cnt & 0x0001) == 0) { if((position & 0x0001)==0) LPC_CANAF_RAM->mask[cnt] = (buf0 << 16) | (0x0000FFFF); else LPC_CANAF_RAM->mask[cnt] = buf0 | 0x0000FFFF; } else { //remove all remaining section one place down cnt = (CANAF_FullCAN_cnt + 1)>>1; bound = total + CANAF_FullCAN_cnt * 3; while(bound>cnt) { LPC_CANAF_RAM->mask[cnt-1] = LPC_CANAF_RAM->mask[cnt]; cnt++; } LPC_CANAF_RAM->mask[cnt-1]=0x00; //update address values LPC_CANAF->SFF_sa -=0x04; LPC_CANAF->SFF_GRP_sa -=0x04 ; LPC_CANAF->EFF_sa -=0x04 ; LPC_CANAF->EFF_GRP_sa -=0x04; LPC_CANAF->ENDofTable -=0x04; } CANAF_FullCAN_cnt--; //delete its FullCAN Object in the FullCAN Object section //remove all remaining FullCAN Object three place down cnt = total + position * 3; bound = (CANAF_FullCAN_cnt - position + 1) * 3; while(bound) { LPC_CANAF_RAM->mask[cnt]=LPC_CANAF_RAM->mask[cnt+3];; LPC_CANAF_RAM->mask[cnt+1]=LPC_CANAF_RAM->mask[cnt+4]; LPC_CANAF_RAM->mask[cnt+2]=LPC_CANAF_RAM->mask[cnt+5]; bound -=3; cnt +=3; } } } /************** Remove Explicit Standard ID Entry *************/ else if(EntryType == EXPLICIT_STANDARD_ENTRY) { if((CANAF_std_cnt==0)||(position >= CANAF_std_cnt)) { return CAN_ENTRY_NOT_EXIT_ERROR; } else { cnt = ((CANAF_FullCAN_cnt+1)>>1)+ (position >> 1); buf0 = LPC_CANAF_RAM->mask[cnt]; bound = (CANAF_std_cnt - position - 1)>>1; if((position & 0x0001) == 0) //event position { while(bound--) { //remove all remaining FullCAN entry one place down buf1 = LPC_CANAF_RAM->mask[cnt+1]; LPC_CANAF_RAM->mask[cnt] = (buf1 >> 16) | (buf0 << 16); buf0 = buf1; cnt++; } } else //odd position { while(bound--) { //remove all remaining FullCAN entry one place down buf1 = LPC_CANAF_RAM->mask[cnt+1]; LPC_CANAF_RAM->mask[cnt] = (buf0 & 0xFFFF0000)|(buf1 >> 16); LPC_CANAF_RAM->mask[cnt+1] = LPC_CANAF_RAM->mask[cnt+1] << 16; buf0 = buf1<<16; cnt++; } } if((CANAF_std_cnt & 0x0001) == 0) { if((position & 0x0001)==0) LPC_CANAF_RAM->mask[cnt] = (buf0 << 16) | (0x0000FFFF); else LPC_CANAF_RAM->mask[cnt] = buf0 | 0x0000FFFF; } else { //remove all remaining section one place down cnt = ((CANAF_FullCAN_cnt + 1)>>1) + ((CANAF_std_cnt + 1) >> 1); bound = total + CANAF_FullCAN_cnt * 3; while(bound>cnt) { LPC_CANAF_RAM->mask[cnt-1] = LPC_CANAF_RAM->mask[cnt]; cnt++; } LPC_CANAF_RAM->mask[cnt-1]=0x00; //update address value LPC_CANAF->SFF_GRP_sa -=0x04 ; LPC_CANAF->EFF_sa -=0x04 ; LPC_CANAF->EFF_GRP_sa -=0x04; LPC_CANAF->ENDofTable -=0x04; } CANAF_std_cnt--; } } /************** Remove Group of Standard ID Entry *************/ else if(EntryType == GROUP_STANDARD_ENTRY) { if((CANAF_gstd_cnt==0)||(position >= CANAF_gstd_cnt)) { return CAN_ENTRY_NOT_EXIT_ERROR; } else { cnt = ((CANAF_FullCAN_cnt + 1)>>1) + ((CANAF_std_cnt + 1) >> 1)+ position + 1; bound = total + CANAF_FullCAN_cnt * 3; while (cntmask[cnt-1] = LPC_CANAF_RAM->mask[cnt]; cnt++; } LPC_CANAF_RAM->mask[cnt-1]=0x00; } CANAF_gstd_cnt--; //update address value LPC_CANAF->EFF_sa -=0x04; LPC_CANAF->EFF_GRP_sa -=0x04; LPC_CANAF->ENDofTable -=0x04; } /************** Remove Explicit Extended ID Entry *************/ else if(EntryType == EXPLICIT_EXTEND_ENTRY) { if((CANAF_ext_cnt==0)||(position >= CANAF_ext_cnt)) { return CAN_ENTRY_NOT_EXIT_ERROR; } else { cnt = ((CANAF_FullCAN_cnt + 1)>>1) + ((CANAF_std_cnt + 1) >> 1)+ CANAF_gstd_cnt + position + 1; bound = total + CANAF_FullCAN_cnt * 3; while (cntmask[cnt-1] = LPC_CANAF_RAM->mask[cnt]; cnt++; } LPC_CANAF_RAM->mask[cnt-1]=0x00; } CANAF_ext_cnt--; LPC_CANAF->EFF_GRP_sa -=0x04; LPC_CANAF->ENDofTable -=0x04; } /************** Remove Group of Extended ID Entry *************/ else { if((CANAF_gext_cnt==0)||(position >= CANAF_gext_cnt)) { return CAN_ENTRY_NOT_EXIT_ERROR; } else { cnt = total - (CANAF_gext_cnt<<1) + (position<<1); bound = total + CANAF_FullCAN_cnt * 3; while (cntmask[cnt] = LPC_CANAF_RAM->mask[cnt+2]; LPC_CANAF_RAM->mask[cnt+1] = LPC_CANAF_RAM->mask[cnt+3]; cnt+=2; } } CANAF_gext_cnt--; LPC_CANAF->ENDofTable -=0x08; } LPC_CANAF->AFMR = 0x04; return CAN_OK; } /********************************************************************//** * @brief Send message data * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] CAN_Msg point to the CAN_MSG_Type Structure, it contains message * information such as: ID, DLC, RTR, ID Format * @return Status: * - SUCCESS: send message successfully * - ERROR: send message unsuccessfully *********************************************************************/ Status CAN_SendMsg (LPC_CAN_TypeDef *CANx, CAN_MSG_Type *CAN_Msg) { uint32_t data; CHECK_PARAM(PARAM_CANx(CANx)); CHECK_PARAM(PARAM_ID_FORMAT(CAN_Msg->format)); if(CAN_Msg->format==STD_ID_FORMAT) { CHECK_PARAM(PARAM_ID_11(CAN_Msg->id)); } else { CHECK_PARAM(PARAM_ID_29(CAN_Msg->id)); } CHECK_PARAM(PARAM_DLC(CAN_Msg->len)); CHECK_PARAM(PARAM_FRAME_TYPE(CAN_Msg->type)); //Check status of Transmit Buffer 1 if (CANx->SR & (1<<2)) { /* Transmit Channel 1 is available */ /* Write frame informations and frame data into its CANxTFI1, * CANxTID1, CANxTDA1, CANxTDB1 register */ CANx->TFI1 &= ~0x000F0000; CANx->TFI1 |= (CAN_Msg->len)<<16; if(CAN_Msg->type == REMOTE_FRAME) { CANx->TFI1 |= (1<<30); //set bit RTR } else { CANx->TFI1 &= ~(1<<30); } if(CAN_Msg->format == EXT_ID_FORMAT) { CANx->TFI1 |= (0x80000000); //set bit FF } else { CANx->TFI1 &= ~(0x80000000); } /* Write CAN ID*/ CANx->TID1 = CAN_Msg->id; /*Write first 4 data bytes*/ data = (CAN_Msg->dataA[0])|(((CAN_Msg->dataA[1]))<<8)|((CAN_Msg->dataA[2])<<16)|((CAN_Msg->dataA[3])<<24); CANx->TDA1 = data; /*Write second 4 data bytes*/ data = (CAN_Msg->dataB[0])|(((CAN_Msg->dataB[1]))<<8)|((CAN_Msg->dataB[2])<<16)|((CAN_Msg->dataB[3])<<24); CANx->TDB1 = data; /*Write transmission request*/ CANx->CMR = 0x21; return SUCCESS; } //check status of Transmit Buffer 2 else if(CANx->SR & (1<<10)) { /* Transmit Channel 2 is available */ /* Write frame informations and frame data into its CANxTFI2, * CANxTID2, CANxTDA2, CANxTDB2 register */ CANx->TFI2 &= ~0x000F0000; CANx->TFI2 |= (CAN_Msg->len)<<16; if(CAN_Msg->type == REMOTE_FRAME) { CANx->TFI2 |= (1<<30); //set bit RTR } else { CANx->TFI2 &= ~(1<<30); } if(CAN_Msg->format == EXT_ID_FORMAT) { CANx->TFI2 |= (0x80000000); //set bit FF } else { CANx->TFI2 &= ~(0x80000000); } /* Write CAN ID*/ CANx->TID2 = CAN_Msg->id; /*Write first 4 data bytes*/ data = (CAN_Msg->dataA[0])|(((CAN_Msg->dataA[1]))<<8)|((CAN_Msg->dataA[2])<<16)|((CAN_Msg->dataA[3])<<24); CANx->TDA2 = data; /*Write second 4 data bytes*/ data = (CAN_Msg->dataB[0])|(((CAN_Msg->dataB[1]))<<8)|((CAN_Msg->dataB[2])<<16)|((CAN_Msg->dataB[3])<<24); CANx->TDB2 = data; /*Write transmission request*/ CANx->CMR = 0x41; return SUCCESS; } //check status of Transmit Buffer 3 else if (CANx->SR & (1<<18)) { /* Transmit Channel 3 is available */ /* Write frame informations and frame data into its CANxTFI3, * CANxTID3, CANxTDA3, CANxTDB3 register */ CANx->TFI3 &= ~0x000F0000; CANx->TFI3 |= (CAN_Msg->len)<<16; if(CAN_Msg->type == REMOTE_FRAME) { CANx->TFI3 |= (1<<30); //set bit RTR } else { CANx->TFI3 &= ~(1<<30); } if(CAN_Msg->format == EXT_ID_FORMAT) { CANx->TFI3 |= (0x80000000); //set bit FF } else { CANx->TFI3 &= ~(0x80000000); } /* Write CAN ID*/ CANx->TID3 = CAN_Msg->id; /*Write first 4 data bytes*/ data = (CAN_Msg->dataA[0])|(((CAN_Msg->dataA[1]))<<8)|((CAN_Msg->dataA[2])<<16)|((CAN_Msg->dataA[3])<<24); CANx->TDA3 = data; /*Write second 4 data bytes*/ data = (CAN_Msg->dataB[0])|(((CAN_Msg->dataB[1]))<<8)|((CAN_Msg->dataB[2])<<16)|((CAN_Msg->dataB[3])<<24); CANx->TDB3 = data; /*Write transmission request*/ CANx->CMR = 0x81; return SUCCESS; } else { return ERROR; } } /********************************************************************//** * @brief Receive message data * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] CAN_Msg point to the CAN_MSG_Type Struct, it will contain received * message information such as: ID, DLC, RTR, ID Format * @return Status: * - SUCCESS: receive message successfully * - ERROR: receive message unsuccessfully *********************************************************************/ Status CAN_ReceiveMsg (LPC_CAN_TypeDef *CANx, CAN_MSG_Type *CAN_Msg) { uint32_t data; CHECK_PARAM(PARAM_CANx(CANx)); //check status of Receive Buffer if((CANx->SR &0x00000001)) { /* Receive message is available */ /* Read frame informations */ CAN_Msg->format = (uint8_t)(((CANx->RFS) & 0x80000000)>>31); CAN_Msg->type = (uint8_t)(((CANx->RFS) & 0x40000000)>>30); CAN_Msg->len = (uint8_t)(((CANx->RFS) & 0x000F0000)>>16); /* Read CAN message identifier */ CAN_Msg->id = CANx->RID; /* Read the data if received message was DATA FRAME */ if (CAN_Msg->type == DATA_FRAME) { /* Read first 4 data bytes */ data = CANx->RDA; *((uint8_t *) &CAN_Msg->dataA[0])= data & 0x000000FF; *((uint8_t *) &CAN_Msg->dataA[1])= (data & 0x0000FF00)>>8;; *((uint8_t *) &CAN_Msg->dataA[2])= (data & 0x00FF0000)>>16; *((uint8_t *) &CAN_Msg->dataA[3])= (data & 0xFF000000)>>24; /* Read second 4 data bytes */ data = CANx->RDB; *((uint8_t *) &CAN_Msg->dataB[0])= data & 0x000000FF; *((uint8_t *) &CAN_Msg->dataB[1])= (data & 0x0000FF00)>>8; *((uint8_t *) &CAN_Msg->dataB[2])= (data & 0x00FF0000)>>16; *((uint8_t *) &CAN_Msg->dataB[3])= (data & 0xFF000000)>>24; /*release receive buffer*/ CANx->CMR = 0x04; } else { /* Received Frame is a Remote Frame, not have data, we just receive * message information only */ CANx->CMR = 0x04; /*release receive buffer*/ return SUCCESS; } } else { // no receive message available return ERROR; } return SUCCESS; } /********************************************************************//** * @brief Receive FullCAN Object * @param[in] CANAFx: CAN Acceptance Filter register, should be: LPC_CANAF * @param[in] CAN_Msg point to the CAN_MSG_Type Struct, it will contain received * message information such as: ID, DLC, RTR, ID Format * @return CAN_ERROR, could be: * - CAN_FULL_OBJ_NOT_RCV: FullCAN Object is not be received * - CAN_OK: Received FullCAN Object successful * *********************************************************************/ CAN_ERROR FCAN_ReadObj (LPC_CANAF_TypeDef* CANAFx, CAN_MSG_Type *CAN_Msg) { uint32_t *pSrc, data; uint32_t interrut_word, msg_idx, test_bit, head_idx, tail_idx; CHECK_PARAM(PARAM_CANAFx(CANAFx)); interrut_word = 0; if (LPC_CANAF->FCANIC0 != 0) { interrut_word = LPC_CANAF->FCANIC0; head_idx = 0; tail_idx = 31; } else if (LPC_CANAF->FCANIC1 != 0) { interrut_word = LPC_CANAF->FCANIC1; head_idx = 32; tail_idx = 63; } if (interrut_word != 0) { /* Detect for interrupt pending */ msg_idx = 0; for (msg_idx = head_idx; msg_idx <= tail_idx; msg_idx++) { test_bit = interrut_word & 0x1; interrut_word = interrut_word >> 1; if (test_bit) { pSrc = (uint32_t *) (LPC_CANAF->ENDofTable + LPC_CANAF_RAM_BASE + msg_idx * 12); /* Has been finished updating the content */ if ((*pSrc & 0x03000000L) == 0x03000000L) { /*clear semaphore*/ *pSrc &= 0xFCFFFFFF; /*Set to DatA*/ pSrc++; /* Copy to dest buf */ data = *pSrc; *((uint8_t *) &CAN_Msg->dataA[0])= data & 0x000000FF; *((uint8_t *) &CAN_Msg->dataA[1])= (data & 0x0000FF00)>>8; *((uint8_t *) &CAN_Msg->dataA[2])= (data & 0x00FF0000)>>16; *((uint8_t *) &CAN_Msg->dataA[3])= (data & 0xFF000000)>>24; /*Set to DatB*/ pSrc++; /* Copy to dest buf */ data = *pSrc; *((uint8_t *) &CAN_Msg->dataB[0])= data & 0x000000FF; *((uint8_t *) &CAN_Msg->dataB[1])= (data & 0x0000FF00)>>8; *((uint8_t *) &CAN_Msg->dataB[2])= (data & 0x00FF0000)>>16; *((uint8_t *) &CAN_Msg->dataB[3])= (data & 0xFF000000)>>24; /*Back to Dat1*/ pSrc -= 2; CAN_Msg->id = *pSrc & 0x7FF; CAN_Msg->len = (uint8_t) (*pSrc >> 16) & 0x0F; CAN_Msg->format = 0; //FullCAN Object ID always is 11-bit value CAN_Msg->type = (uint8_t)(*pSrc >> 30) &0x01; /*Re-read semaphore*/ if ((*pSrc & 0x03000000L) == 0) { return CAN_OK; } } } } } return CAN_FULL_OBJ_NOT_RCV; } /********************************************************************//** * @brief Get CAN Control Status * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] arg: type of CAN status to get from CAN status register * Should be: * - CANCTRL_GLOBAL_STS: CAN Global Status * - CANCTRL_INT_CAP: CAN Interrupt and Capture * - CANCTRL_ERR_WRN: CAN Error Warning Limit * - CANCTRL_STS: CAN Control Status * @return Current Control Status that you want to get value *********************************************************************/ uint32_t CAN_GetCTRLStatus (LPC_CAN_TypeDef* CANx, CAN_CTRL_STS_Type arg) { CHECK_PARAM(PARAM_CANx(CANx)); CHECK_PARAM(PARAM_CTRL_STS_TYPE(arg)); switch (arg) { case CANCTRL_GLOBAL_STS: return CANx->GSR; case CANCTRL_INT_CAP: return CANx->ICR; case CANCTRL_ERR_WRN: return CANx->EWL; default: // CANCTRL_STS return CANx->SR; } } /********************************************************************//** * @brief Get CAN Central Status * @param[in] CANCRx point to LPC_CANCR_TypeDef, should be: LPC_CANCR * @param[in] arg: type of CAN status to get from CAN Central status register * Should be: * - CANCR_TX_STS: Central CAN Tx Status * - CANCR_RX_STS: Central CAN Rx Status * - CANCR_MS: Central CAN Miscellaneous Status * @return Current Central Status that you want to get value *********************************************************************/ uint32_t CAN_GetCRStatus (LPC_CANCR_TypeDef* CANCRx, CAN_CR_STS_Type arg) { CHECK_PARAM(PARAM_CANCRx(CANCRx)); CHECK_PARAM(PARAM_CR_STS_TYPE(arg)); switch (arg) { case CANCR_TX_STS: return CANCRx->CANTxSR; case CANCR_RX_STS: return CANCRx->CANRxSR; default: // CANCR_MS return CANCRx->CANMSR; } } /********************************************************************//** * @brief Enable/Disable CAN Interrupt * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] arg: type of CAN interrupt that you want to enable/disable * Should be: * - CANINT_RIE: CAN Receiver Interrupt Enable * - CANINT_TIE1: CAN Transmit Interrupt Enable * - CANINT_EIE: CAN Error Warning Interrupt Enable * - CANINT_DOIE: CAN Data Overrun Interrupt Enable * - CANINT_WUIE: CAN Wake-Up Interrupt Enable * - CANINT_EPIE: CAN Error Passive Interrupt Enable * - CANINT_ALIE: CAN Arbitration Lost Interrupt Enable * - CANINT_BEIE: CAN Bus Error Interrupt Enable * - CANINT_IDIE: CAN ID Ready Interrupt Enable * - CANINT_TIE2: CAN Transmit Interrupt Enable for Buffer2 * - CANINT_TIE3: CAN Transmit Interrupt Enable for Buffer3 * - CANINT_FCE: FullCAN Interrupt Enable * @param[in] NewState: New state of this function, should be: * - ENABLE * - DISABLE * @return none *********************************************************************/ void CAN_IRQCmd (LPC_CAN_TypeDef* CANx, CAN_INT_EN_Type arg, FunctionalState NewState) { CHECK_PARAM(PARAM_CANx(CANx)); CHECK_PARAM(PARAM_INT_EN_TYPE(arg)); CHECK_PARAM(PARAM_FUNCTIONALSTATE(NewState)); if(NewState == ENABLE) { if(arg==CANINT_FCE) { LPC_CANAF->AFMR = 0x01; LPC_CANAF->FCANIE = 0x01; LPC_CANAF->AFMR = 0x04; } else CANx->IER |= (1 << arg); } else { if(arg==CANINT_FCE){ LPC_CANAF->AFMR = 0x01; LPC_CANAF->FCANIE = 0x01; LPC_CANAF->AFMR = 0x00; } else CANx->IER &= ~(1 << arg); } } /********************************************************************//** * @brief Setting Acceptance Filter mode * @param[in] CANAFx point to LPC_CANAF_TypeDef object, should be: LPC_CANAF * @param[in] AFMode: type of AF mode that you want to set, should be: * - CAN_Normal: Normal mode * - CAN_AccOff: Acceptance Filter Off Mode * - CAN_AccBP: Acceptance Fileter Bypass Mode * - CAN_eFCAN: FullCAN Mode Enhancement * @return none *********************************************************************/ void CAN_SetAFMode (LPC_CANAF_TypeDef* CANAFx, CAN_AFMODE_Type AFMode) { CHECK_PARAM(PARAM_CANAFx(CANAFx)); CHECK_PARAM(PARAM_AFMODE_TYPE(AFMode)); switch(AFMode) { case CAN_Normal: CANAFx->AFMR = 0x00; break; case CAN_AccOff: CANAFx->AFMR = 0x01; break; case CAN_AccBP: CANAFx->AFMR = 0x02; break; case CAN_eFCAN: CANAFx->AFMR = 0x04; break; } } /********************************************************************//** * @brief Enable/Disable CAN Mode * @param[in] CANx pointer to LPC_CAN_TypeDef, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] mode: type of CAN mode that you want to enable/disable, should be: * - CAN_OPERATING_MODE: Normal Operating Mode * - CAN_RESET_MODE: Reset Mode * - CAN_LISTENONLY_MODE: Listen Only Mode * - CAN_SELFTEST_MODE: Self Test Mode * - CAN_TXPRIORITY_MODE: Transmit Priority Mode * - CAN_SLEEP_MODE: Sleep Mode * - CAN_RXPOLARITY_MODE: Receive Polarity Mode * - CAN_TEST_MODE: Test Mode * @param[in] NewState: New State of this function, should be: * - ENABLE * - DISABLE * @return none *********************************************************************/ void CAN_ModeConfig(LPC_CAN_TypeDef* CANx, CAN_MODE_Type mode, FunctionalState NewState) { CHECK_PARAM(PARAM_CANx(CANx)); CHECK_PARAM(PARAM_MODE_TYPE(mode)); CHECK_PARAM(PARAM_FUNCTIONALSTATE(NewState)); switch(mode) { case CAN_OPERATING_MODE: CANx->MOD = 0x00; break; case CAN_RESET_MODE: if(NewState == ENABLE) CANx->MOD |=CAN_MOD_RM; else CANx->MOD &= ~CAN_MOD_RM; break; case CAN_LISTENONLY_MODE: CANx->MOD |=CAN_MOD_RM;//Enter Reset mode if(NewState == ENABLE) CANx->MOD |=CAN_MOD_LOM; else CANx->MOD &=~CAN_MOD_LOM; CANx->MOD &=~CAN_MOD_RM;//Release Reset mode break; case CAN_SELFTEST_MODE: CANx->MOD |=CAN_MOD_RM;//Enter Reset mode if(NewState == ENABLE) CANx->MOD |=CAN_MOD_STM; else CANx->MOD &=~CAN_MOD_STM; CANx->MOD &=~CAN_MOD_RM;//Release Reset mode break; case CAN_TXPRIORITY_MODE: if(NewState == ENABLE) CANx->MOD |=CAN_MOD_TPM; else CANx->MOD &=~CAN_MOD_TPM; break; case CAN_SLEEP_MODE: if(NewState == ENABLE) CANx->MOD |=CAN_MOD_SM; else CANx->MOD &=~CAN_MOD_SM; break; case CAN_RXPOLARITY_MODE: if(NewState == ENABLE) CANx->MOD |=CAN_MOD_RPM; else CANx->MOD &=~CAN_MOD_RPM; break; case CAN_TEST_MODE: if(NewState == ENABLE) CANx->MOD |=CAN_MOD_TM; else CANx->MOD &=~CAN_MOD_TM; break; } } /*********************************************************************//** * @brief Set CAN command request * @param[in] CANx point to CAN peripheral selected, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @param[in] CMRType command request type, should be: * - CAN_CMR_TR: Transmission request * - CAN_CMR_AT: Abort Transmission request * - CAN_CMR_RRB: Release Receive Buffer request * - CAN_CMR_CDO: Clear Data Overrun request * - CAN_CMR_SRR: Self Reception request * - CAN_CMR_STB1: Select Tx Buffer 1 request * - CAN_CMR_STB2: Select Tx Buffer 2 request * - CAN_CMR_STB3: Select Tx Buffer 3 request * @return CANICR (CAN interrupt and Capture register) value **********************************************************************/ void CAN_SetCommand(LPC_CAN_TypeDef* CANx, uint32_t CMRType) { CHECK_PARAM(PARAM_CANx(CANx)); CANx->CMR |= CMRType; } /*********************************************************************//** * @brief Get CAN interrupt status * @param[in] CANx point to CAN peripheral selected, should be: * - LPC_CAN1: CAN1 peripheral * - LPC_CAN2: CAN2 peripheral * @return CANICR (CAN interrupt and Capture register) value **********************************************************************/ uint32_t CAN_IntGetStatus(LPC_CAN_TypeDef* CANx) { CHECK_PARAM(PARAM_CANx(CANx)); return CANx->ICR; } /*********************************************************************//** * @brief Check if FullCAN interrupt enable or not * @param[in] CANAFx point to LPC_CANAF_TypeDef object, should be: LPC_CANAF * @return IntStatus, could be: * - SET: if FullCAN interrupt is enable * - RESET: if FullCAN interrupt is disable **********************************************************************/ IntStatus CAN_FullCANIntGetStatus (LPC_CANAF_TypeDef* CANAFx) { CHECK_PARAM( PARAM_CANAFx(CANAFx)); if (CANAFx->FCANIE) return SET; return RESET; } /*********************************************************************//** * @brief Get value of FullCAN interrupt and capture register * @param[in] CANAFx point to LPC_CANAF_TypeDef object, should be: LPC_CANAF * @param[in] type: FullCAN IC type, should be: * - FULLCAN_IC0: FullCAN Interrupt Capture 0 * - FULLCAN_IC1: FullCAN Interrupt Capture 1 * @return FCANIC0 or FCANIC1 (FullCAN interrupt and Capture register) value **********************************************************************/ uint32_t CAN_FullCANPendGetStatus(LPC_CANAF_TypeDef* CANAFx, FullCAN_IC_Type type) { CHECK_PARAM(PARAM_CANAFx(CANAFx)); CHECK_PARAM( PARAM_FULLCAN_IC(type)); if (type == FULLCAN_IC0) return CANAFx->FCANIC0; return CANAFx->FCANIC1; } /* End of Public Variables ---------------------------------------------------------- */ /** * @} */ #endif /* _CAN */ /** * @} */ /* --------------------------------- End Of File ------------------------------ */