From ec46aee19b44150d8ab727c126cf5587c62d4bd2 Mon Sep 17 00:00:00 2001 From: "Aubr.Cool" Date: Sat, 4 Jul 2015 08:06:02 +0800 Subject: [PATCH] add lpc408x can driver --- bsp/lpc408x/applications/startup.c | 2 +- bsp/lpc408x/drivers/drv_lpccan.c | 973 ++++++++++++++++++ bsp/lpc408x/drivers/drv_lpccan.h | 29 + .../{project.uvproj => project.uvprojx} | 347 +++---- bsp/lpc408x/rtconfig.h | 5 +- 5 files changed, 1136 insertions(+), 220 deletions(-) create mode 100644 bsp/lpc408x/drivers/drv_lpccan.c create mode 100644 bsp/lpc408x/drivers/drv_lpccan.h rename bsp/lpc408x/{project.uvproj => project.uvprojx} (84%) diff --git a/bsp/lpc408x/applications/startup.c b/bsp/lpc408x/applications/startup.c index 67df608698..e1e9c2d4b5 100644 --- a/bsp/lpc408x/applications/startup.c +++ b/bsp/lpc408x/applications/startup.c @@ -39,7 +39,7 @@ void rtthread_startup(void) rt_system_heap_init((void *)HEAP_BEGIN, (void *)HEAP_END); #endif #endif - + rt_components_board_init(); /* initialize scheduler system */ rt_system_scheduler_init(); /* initialize system timer*/ diff --git a/bsp/lpc408x/drivers/drv_lpccan.c b/bsp/lpc408x/drivers/drv_lpccan.c new file mode 100644 index 0000000000..368949371e --- /dev/null +++ b/bsp/lpc408x/drivers/drv_lpccan.c @@ -0,0 +1,973 @@ +/* + * File : drv_lpccan.c + * This file is part of RT-Thread RTOS + * COPYRIGHT (C) 2015, RT-Thread Development Team + * + * The license and distribution terms for this file may be + * found in the file LICENSE in this distribution or at + * http://www.rt-thread.org/license/LICENSE + * + * Change Logs: + * Date Author Notes + * 2015-06-30 aubrcool@qq.com first version + */ +#include +#include +#include +#include +#ifdef RT_USING_COMPONENTS_INIT +#include +#endif + +#ifdef RT_USING_CAN + +#include "lpc_types.h" +#include "lpc_can.h" +#include "lpc_pinsel.h" +#include "lpc_exti.h" +#include "lpc_clkpwr.h" + +struct lpccandata { + en_CAN_unitId id; +}; +static const rt_uint32_t LPCBAUDTAB[] = { + 1000000, + 800000, + 500000, + 250000, + 125000, + 100000, + 50000, + 20000, + 10000, +}; +static LPC_CAN_TypeDef* lcpcan_get_reg_base(rt_uint32_t id) +{ + LPC_CAN_TypeDef* pCan; + + switch (id) { + case CAN_ID_1: + pCan = LPC_CAN1; + break; + + case CAN_ID_2: + pCan = LPC_CAN2; + break; + default: + pCan = NULL; + } + return pCan; +} +static void lpccan_irqstate_init(rt_uint32_t id) +{ + LPC_CAN_TypeDef* pCan = lcpcan_get_reg_base(id); + + pCan->MOD = 1; // Enter Reset Mode + pCan->IER = 0; // Disable All CAN Interrupts + pCan->GSR = 0; + + /* Request command to release Rx, Tx buffer and clear data overrun */ + //pCan->CMR = CAN_CMR_AT | CAN_CMR_RRB | CAN_CMR_CDO; + pCan->CMR = (1 << 1) | (1 << 2) | (1 << 3); + + /* Read to clear interrupt pending in interrupt capture register */ + rt_int32_t i = pCan->ICR; + pCan->MOD = 0;// Return Normal operating +} +static void lpccan_baud_set(rt_uint32_t id, rt_uint32_t baud) +{ + uint32_t result = 0; + uint8_t NT, TSEG1, TSEG2; + uint32_t CANPclk = 0; + uint32_t BRP; + LPC_CAN_TypeDef* pCan = lcpcan_get_reg_base(id); + + CANPclk = CLKPWR_GetCLK(CLKPWR_CLKTYPE_PER); + result = CANPclk / LPCBAUDTAB[baud]; + /* Calculate suitable nominal time value + * NT (nominal time) = (TSEG1 + TSEG2 + 3) + * NT <= 24 + * TSEG1 >= 2*TSEG2 + */ + 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; + break; + } + } + /* Enter reset mode */ + pCan->MOD = 0x01; + /* Set bit timing + * Default: SAM = 0x00; + * SJW = 0x03; + */ + pCan->BTR = (TSEG2 << 20) | (TSEG1 << 16) | (3 << 14) | BRP; + /* Return to normal operating */ + pCan->MOD = 0; +} +static void lpccan_init_alut_ram(void) +{ + //Reset CANAF value + LPC_CANAF->AFMR = 0x01; + //clear ALUT RAM + rt_memset((void *)LPC_CANAF_RAM->mask, 0, 2048); + LPC_CANAF->SFF_sa = 0; + LPC_CANAF->SFF_GRP_sa = 0; + LPC_CANAF->EFF_sa = 0; + LPC_CANAF->EFF_GRP_sa = 0; + LPC_CANAF->ENDofTable = 0; + LPC_CANAF->AFMR = 0x00; + // Set AF Mode + CAN_SetAFMode(CAN_NORMAL); +} +#ifdef RT_USING_LPCCAN1 +static void lpccan1_turnon_clk(void) +{ + CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCAN1, ENABLE); +} +static void lpccan1_filter_init(struct rt_can_device *can) +{ +} +static void lpccan1_hw_init(enum CANBAUD baud, CAN_MODE_Type mode) +{ + if(mode != CAN_SELFTEST_MODE) { +#ifndef LPCCAN1_USEING_GPIO_SECOND + PINSEL_ConfigPin (0, 0, 1); + PINSEL_ConfigPin (0, 1, 1); +#else + PINSEL_ConfigPin (0, 21, 4); + PINSEL_ConfigPin (0, 22, 4); +#endif + } + lpccan1_turnon_clk(); + lpccan_irqstate_init(CAN_1); + lpccan_init_alut_ram(); + lpccan1_turnon_clk(); + lpccan_baud_set(CAN_1, baud); + CAN_ModeConfig(CAN_1, mode, ENABLE); + if(mode == CAN_SELFTEST_MODE) { + //CAN_ModeConfig(CAN_1, CAN_TEST_MODE, ENABLE); + CAN_SetAFMode(CAN_ACC_BP); + } +} +#endif /*RT_USING_LPCCAN1*/ +#ifdef RT_USING_LPCCAN2 +static void lpccan2_turnon_clk(void) +{ + CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCAN2, ENABLE); +} +static void lpccan2_filter_init(struct rt_can_device *can) +{ +} +static void lpccan2_hw_init(enum CANBAUD baud, CAN_MODE_Type mode) +{ + if(mode != CAN_SELFTEST_MODE) { +#ifndef LPCCAN2_USEING_GPIO_SECOND + PINSEL_ConfigPin (0, 4, 2); + PINSEL_ConfigPin (0, 5, 2); +#else + PINSEL_ConfigPin (2, 7, 1); + PINSEL_ConfigPin (2, 8, 1); +#endif + } + lpccan2_turnon_clk(); + lpccan_irqstate_init(CAN_2); +#ifndef RT_USING_LPCCAN1 + lpccan_init_alut_ram(); +#endif /*RT_USING_LPCCAN1*/ + lpccan_baud_set(CAN_2, baud); + CAN_ModeConfig(CAN_2, mode, ENABLE); + if(mode == CAN_SELFTEST_MODE) { + CAN_SetAFMode(CAN_ACC_BP); + } +} +#endif /*RT_USING_LPCCAN2*/ +static rt_err_t configure(struct rt_can_device *can, struct can_configure *cfg) +{ + CAN_MODE_Type mode; + switch(cfg->mode) { + case RT_CAN_MODE_NORMAL: + mode = CAN_OPERATING_MODE; + break; + case RT_CAN_MODE_LISEN: + mode = CAN_LISTENONLY_MODE; + break; + case RT_CAN_MODE_LOOPBACKANLISEN: + mode = CAN_SELFTEST_MODE; + break; + default: + return RT_EIO; + } + rt_uint32_t canid; + canid = ((struct lpccandata *) can->parent.user_data)->id; +#ifdef RT_USING_LPCCAN1 + if(canid == CAN_1) + { + lpccan1_hw_init(cfg->baud_rate, mode); + lpccan1_filter_init(can); + } +#endif /*RT_USING_LPCCAN1*/ +#ifdef RT_USING_LPCCAN2 +#ifdef RT_USING_LPCCAN1 + else +#endif /*RT_USING_LPCCAN1*/ + { + lpccan2_hw_init(cfg->baud_rate, mode); + lpccan2_filter_init(can); + } +#endif /*RT_USING_LPCCAN2*/ + return RT_EOK; +} +static CAN_ERROR findfilter(struct lpccandata* plpccan, struct rt_can_filter_item* pitem, rt_int32_t* pos) +{ + extern uint16_t CANAF_FullCAN_cnt; + extern uint16_t CANAF_std_cnt; + extern uint16_t CANAF_gstd_cnt; + extern uint16_t CANAF_ext_cnt; + extern uint16_t CANAF_gext_cnt; + + rt_uint32_t buf0 = 0, buf1 = 0; + rt_int16_t cnt1 = 0, cnt2 = 0, bound1 = 0; + CAN_ID_FORMAT_Type format; + *pos = -1; + if(pitem->ide) { + format = EXT_ID_FORMAT; + } else { + format = STD_ID_FORMAT; + } + if(pitem->mode) { + rt_uint32_t id = pitem->id; + if(format == STD_ID_FORMAT) + { + id &= 0x07FF; + + id |= plpccan->id << 13;/* Add controller number */ + + if (CANAF_std_cnt == 0) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else if (CANAF_std_cnt == 1) + { + cnt2 = (CANAF_FullCAN_cnt + 1) >> 1; + if(id != LPC_CANAF_RAM->mask[cnt2] >> 16) { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + } + else + { + cnt1 = (CANAF_FullCAN_cnt+1)>>1; + + 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) + { + *pos = cnt1 * 2; + return CAN_OK; + } + + if ((LPC_CANAF_RAM->mask[cnt1] & 0x0000E7FF) == id) + { + *pos = cnt1 * 2 + 1; + return CAN_OK; + } + if (((LPC_CANAF_RAM->mask[cnt1] >> 16) & 0xE7FF) > id) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + + if ((LPC_CANAF_RAM->mask[cnt1] & 0x0000E7FF) > id) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + cnt1++; + } + return CAN_ENTRY_NOT_EXIT_ERROR; + } + } + /*********** Add Explicit Extended Identifier Frame Format entry *********/ + else + { + /* Add controller number */ + id |= plpccan->id << 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) + { + *pos = cnt2; + return CAN_OK; + } + if (LPC_CANAF_RAM->mask[cnt1] > id) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + cnt1++; + cnt2++; + } + } + } else { + rt_uint32_t lowerID = pitem->id; + rt_uint32_t upperID = pitem->mask; + rt_uint32_t LID,UID; + if(lowerID > upperID) + return CAN_CONFLICT_ID_ERROR; + if(format == STD_ID_FORMAT) + { + lowerID &=0x7FF; //mask ID + upperID &=0x7FF; + cnt1 = ((CANAF_FullCAN_cnt+1)>>1) + ((CANAF_std_cnt + 1) >> 1); + if(CANAF_gstd_cnt == 0) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else + { + 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) < (plpccan->id))//increase until meet greater or equal controller + cnt1++; + buf0 = LPC_CANAF_RAM->mask[cnt1]; + if((LPC_CANAF_RAM->mask[cnt1] >> 29) > (plpccan->id)) //meet greater controller + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else //meet equal controller + { + LID = (buf0 >> 16)&0x7FF; + UID = buf0 & 0x7FF; + if (upperID == LID && lowerID == UID) + { + *pos = cnt1; + return CAN_OK; + } + if (upperID < LID) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else if (lowerID >= UID) + { + cnt1 ++; + } + else + return CAN_CONFLICT_ID_ERROR; + } + } + if(cnt1 >= bound1) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + } + } + /*********Add Group of Extended Identifier Frame Format************/ + else + { + lowerID &= 0x1FFFFFFF; //mask ID + upperID &= 0x1FFFFFFF; + 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) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else + { + 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)< plpccan->id ) //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) > plpccan->id ) //meet greater controller + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else //meet equal controller + { + LID = buf0 & 0x1FFFFFFF; //mask ID + UID = buf1 & 0x1FFFFFFF; + if (upperID == LID && lowerID == UID) + { + *pos = cnt1; + return CAN_OK; + } + if (upperID < LID) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + else if (lowerID >= UID) + { + //load next entry to compare + cnt1 +=2; + } + else + return CAN_CONFLICT_ID_ERROR; + } + } + if(cnt1 >= bound1) + { + return CAN_ENTRY_NOT_EXIT_ERROR; + } + } + } + } + return CAN_ENTRY_NOT_EXIT_ERROR; +} +static rt_err_t setfilter(struct lpccandata* plpccan,struct rt_can_filter_config *pconfig) +{ + struct rt_can_filter_item* pitem = pconfig->items; + rt_uint32_t count = pconfig->count; + rt_int32_t pos; + CAN_ID_FORMAT_Type format; + CAN_ERROR lpccanres; + while(count) { + if(pitem->ide) { + format = EXT_ID_FORMAT; + } else { + format = STD_ID_FORMAT; + } + lpccanres = findfilter(plpccan, pitem, &pos); + if(pconfig->actived && lpccanres != CAN_OK) { + if(pitem->mode) { + lpccanres = CAN_LoadGroupEntry(plpccan->id, pitem->id, pitem->mask, format); + } else { + lpccanres = CAN_LoadExplicitEntry(plpccan->id, pitem->id, format); + } + } else if(!pconfig->actived && lpccanres == CAN_OK) { + AFLUT_ENTRY_Type type; + if(pitem->mode) { + if(format == EXT_ID_FORMAT) { + type = GROUP_EXTEND_ENTRY; + } else { + type = GROUP_STANDARD_ENTRY; + } + } else { + if(format == EXT_ID_FORMAT) { + type = EXPLICIT_EXTEND_ENTRY; + } else { + type = EXPLICIT_STANDARD_ENTRY; + } + } + lpccanres = CAN_RemoveEntry(type, (rt_uint16_t)(pos)); + } else if(!pconfig->actived && lpccanres != CAN_OK) { + lpccanres = CAN_OK; + } + if(lpccanres != CAN_OK) { + return RT_EIO; + } + pitem++; + count--; + } + return RT_EOK; +} +static rt_err_t control(struct rt_can_device *can, int cmd, void *arg) +{ + struct lpccandata* plpccan; + rt_uint32_t argval; + CAN_MODE_Type mode; + plpccan = (struct lpccandata* ) can->parent.user_data; + RT_ASSERT(plpccan != RT_NULL); + switch (cmd) + { + case RT_DEVICE_CTRL_CLR_INT: + argval = (rt_uint32_t) arg; + if(argval == RT_DEVICE_FLAG_INT_RX) + { + CAN_IRQCmd(plpccan->id, CANINT_RIE, DISABLE); + CAN_IRQCmd(plpccan->id, CANINT_DOIE, DISABLE); + } else if(argval == RT_DEVICE_FLAG_INT_TX) { + CAN_IRQCmd(plpccan->id, CANINT_TIE1, DISABLE); + CAN_IRQCmd(plpccan->id, CANINT_TIE2, DISABLE); + CAN_IRQCmd(plpccan->id, CANINT_TIE3, DISABLE); + } else if(argval == RT_DEVICE_CAN_INT_ERR) { + CAN_IRQCmd(plpccan->id, CANINT_EIE, DISABLE); + } + break; + case RT_DEVICE_CTRL_SET_INT: + argval = (rt_uint32_t) arg; + if(argval == RT_DEVICE_FLAG_INT_RX) + { + CAN_IRQCmd(plpccan->id, CANINT_RIE, ENABLE); + CAN_IRQCmd(plpccan->id, CANINT_DOIE, ENABLE); + } else if(argval == RT_DEVICE_FLAG_INT_TX) { + CAN_IRQCmd(plpccan->id, CANINT_TIE1, ENABLE); + CAN_IRQCmd(plpccan->id, CANINT_TIE2, ENABLE); + CAN_IRQCmd(plpccan->id, CANINT_TIE3, ENABLE); + } else if(argval == RT_DEVICE_CAN_INT_ERR) { + CAN_IRQCmd(plpccan->id, CANINT_EIE, ENABLE); + } + break; + case RT_CAN_CMD_SET_FILTER: + return setfilter(plpccan, (struct rt_can_filter_config*) arg); + case RT_CAN_CMD_SET_MODE: + argval = (rt_uint32_t) arg; + if(argval != RT_CAN_MODE_NORMAL || + argval != RT_CAN_MODE_LISEN) { + return RT_ERROR; + } + if(argval != can->config.mode) + { + can->config.mode = argval; + switch(argval) { + case RT_CAN_MODE_NORMAL: + mode = CAN_OPERATING_MODE; + break; + case RT_CAN_MODE_LISEN: + mode = CAN_LISTENONLY_MODE; + break; + case RT_CAN_MODE_LOOPBACKANLISEN: + mode = CAN_SELFTEST_MODE; + break; + default: + return RT_EIO; + } + CAN_ModeConfig(plpccan->id, mode, ENABLE); + if(mode == CAN_SELFTEST_MODE) { + //CAN_ModeConfig(CAN_1, CAN_TEST_MODE, ENABLE); + CAN_SetAFMode(CAN_ACC_BP); + } + } + break; + case RT_CAN_CMD_SET_BAUD: + argval = (rt_uint32_t) arg; + if(argval != CAN1MBaud && + argval != CAN800kBaud && + argval != CAN500kBaud && + argval != CAN250kBaud && + argval != CAN125kBaud && + argval != CAN100kBaud && + argval != CAN50kBaud && + argval != CAN20kBaud && + argval != CAN10kBaud ) { + return RT_ERROR; + } + if(argval != can->config.baud_rate) + { + can->config.baud_rate = argval; + lpccan_baud_set(plpccan->id, (rt_uint32_t) arg); + } + break; + case RT_CAN_CMD_SET_PRIV: + argval = (rt_uint32_t) arg; + if(argval != RT_CAN_MODE_PRIV || + argval != RT_CAN_MODE_NOPRIV) { + return RT_ERROR; + } + if(argval != can->config.privmode) + { + can->config.privmode = argval; + CAN_ModeConfig(plpccan->id, CAN_TXPRIORITY_MODE, ENABLE); + } + break; + case RT_CAN_CMD_GET_STATUS: + { + rt_uint32_t errtype; + can->status.rcverrcnt = 0; + can->status.snderrcnt = 0; + can->status.errcode = 0; + if(arg != &can->status) { + rt_memcpy(arg,&can->status,sizeof(can->status)); + } + } + break; + } + return RT_EOK; +} +static int sendmsg(struct rt_can_device *can, const void* buf, rt_uint32_t boxno) +{ + struct lpccandata* plpccan; + plpccan = (struct lpccandata* ) can->parent.user_data; + RT_ASSERT(plpccan != RT_NULL); + LPC_CAN_TypeDef* pCan = lcpcan_get_reg_base(plpccan->id); + RT_ASSERT(pCan != RT_NULL); + struct rt_can_msg* pmsg = (struct rt_can_msg*) buf; + rt_uint32_t SR_Mask; + if(boxno > 2) { + return RT_ERROR; + } + rt_uint32_t CMRMsk = 0x01 | (0x01 << (boxno + 5)); + SR_Mask = 0x01 <<(boxno * 8 + 2); + volatile unsigned int *pTFI = (&pCan->TFI1 + 0 + 4 * boxno); + volatile unsigned int *pTID = (&pCan->TFI1 + 1 + 4 * boxno); + volatile unsigned int *pTDA = (&pCan->TFI1 + 2 + 4 * boxno); + volatile unsigned int *pTDB = (&pCan->TFI1 + 3 + 4 * boxno); + rt_uint32_t data; + if(pCan->SR & SR_Mask) { + /* Transmit Channel 1 is available */ + /* Write frame informations and frame data into its CANxTFI1, + * CANxTID1, CANxTDA1, CANxTDB1 register */ + *pTFI &= ~ 0x000F0000; + *pTFI |= (pmsg->len) << 16; + if(pmsg->rtr == REMOTE_FRAME) + { + *pTFI |= (1 << 30); //set bit RTR + } + else + { + *pTFI &= ~(1 << 30); + } + + if(pmsg->ide == EXT_ID_FORMAT) + { + *pTFI |= (((uint32_t)1) << 31); //set bit FF + } + else + { + *pTFI &= ~(((uint32_t)1) << 31); + } + if(can->config.privmode) { + *pTFI &= ~0x000000FF; + *pTFI |= pmsg->priv; + } + /* Write CAN ID*/ + *pTID = pmsg->id; + /*Write first 4 data bytes*/ + data = (pmsg->data[0]) | (((pmsg->data[1]))<< 8) | ((pmsg->data[2]) << 16) | ((pmsg->data[3]) << 24); + *pTDA = data; + /*Write second 4 data bytes*/ + data = (pmsg->data[4]) | (((pmsg->data[5])) << 8) | ((pmsg->data[6]) << 16) | ((pmsg->data[7]) << 24); + *pTDB = data; + /*Write transmission request*/ + pCan->CMR = CMRMsk; + return RT_EOK; + } else { + return RT_ERROR; + } +} +static int recvmsg(struct rt_can_device *can, void* buf, rt_uint32_t boxno) +{ + struct lpccandata* plpccan; + plpccan = (struct lpccandata* ) can->parent.user_data; + RT_ASSERT(plpccan != RT_NULL); + LPC_CAN_TypeDef* pCan = lcpcan_get_reg_base(plpccan->id); + RT_ASSERT(pCan != RT_NULL); + //CAN_ReceiveMsg + uint32_t data; + struct rt_can_msg* pmsg = (struct rt_can_msg*) buf; + //check status of Receive Buffer + if((pCan->SR &0x00000001)) + { + /* Receive message is available */ + /* Read frame informations */ + pmsg->ide = (uint8_t)(((pCan->RFS) & 0x80000000) >> 31); + pmsg->rtr = (uint8_t)(((pCan->RFS) & 0x40000000) >> 30); + pmsg->len = (uint8_t)(((pCan->RFS) & 0x000F0000) >> 16); + /* Read CAN message identifier */ + pmsg->id = pCan->RID; + /* Read the data if received message was DATA FRAME */ + if (!pmsg->rtr) + { + /* Read first 4 data bytes */ + data = pCan->RDA; + pmsg->data[0] = data & 0x000000FF; + pmsg->data[1] = (data & 0x0000FF00) >> 8; + pmsg->data[2] = (data & 0x00FF0000) >> 16; + pmsg->data[3] = (data & 0xFF000000) >> 24; + /* Read second 4 data bytes */ + if(pmsg->len > 4) { + data = pCan->RDB; + pmsg->data[4] = data & 0x000000FF; + pmsg->data[5] = (data & 0x0000FF00) >> 8; + pmsg->data[6] = (data & 0x00FF0000) >> 16; + pmsg->data[7] = (data & 0xFF000000) >> 24; + } + pmsg->hdr = 0; + /*release receive buffer*/ + pCan->CMR = 0x04; + } + else + { + /* Received Frame is a Remote Frame, not have data, we just receive + * message information only */ + pCan->CMR = 0x04; /*release receive buffer*/ + return SUCCESS; + } + } + else + { + // no receive message available + return ERROR; + } + return RT_EOK; +} + +static const struct rt_can_ops canops = +{ + configure, + control, + sendmsg, + recvmsg, +}; +#ifdef RT_USING_LPCCAN1 +#ifdef RT_CAN_USING_LED +#endif +static struct lpccandata lpccandata1 = { + CAN_1, +}; +static struct rt_can_device lpccan1; +#endif /*RT_USINGLPCCAN1*/ + +#ifdef RT_USING_LPCCAN2 +#ifdef RT_CAN_USING_LED +#endif +static struct lpccandata lpccandata2 = { + CAN_2, +}; +static struct rt_can_device lpccan2; +#endif /*RT_USINGLPCCAN2*/ +/*----------------- INTERRUPT SERVICE ROUTINES --------------------------*/ +/*********************************************************************//** + * @brief Event Router IRQ Handler + * @param[in] None + * @return None + **********************************************************************/ +void CAN_IRQHandler(void) +{ + rt_uint32_t IntStatus; +#ifdef RT_USING_LPCCAN1 + IntStatus = CAN_IntGetStatus(CAN_1); + //check receive interrupt + if((IntStatus >> 0) & 0x01) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_RX_IND | 0<<8); + } + //check Transmit Interrupt interrupt1 + if((IntStatus >> 1) & 0x01) + { + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_1, CANCTRL_STS); + if(state & (0x01 << 3)) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_DONE | 0<<8); + } else { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_FAIL | 0<<8); + } + } + //check Error Warning Interrupt + if((IntStatus >> 2) & 0x01) + { + rt_uint32_t errtype; + errtype = (IntStatus >> 16); + if(errtype & 0x1F && lpccan1.status.lasterrtype == (errtype & 0x1F)) { + switch((errtype & 0x1F)) { + case 00011: // Start of Frame + case 00010: // ID28 ... ID21 + case 00110: //ID20 ... ID18 + case 00100: // SRTR Bit + case 00101: // IDE bit + case 00111: // ID17 ... 13 + case 01111: // ID12 ... ID5 + case 01110: // ID4 ... ID0 + case 01100: // RTR Bit + case 01011: // Data Length Code + case 01010: // Data Field + lpccan1.status.formaterrcnt++; + break; + case 01101: // Reserved Bit 1 + case 01001: // Reserved Bit 0 + lpccan1.status.bitpaderrcnt++; + break; + case 01000: // CRC Sequence + case 11000: // CRC Delimiter + lpccan1.status.crcerrcnt++; + break; + case 11001: // Acknowledge Slot + case 11011: // Acknowledge Delimiter + lpccan1.status.ackerrcnt++; + break; + case 11010: // End of Frame + case 10010: // Intermission + lpccan1.status.formaterrcnt++; + break; + } + lpccan1.status.lasterrtype = errtype & 0x1F; + } + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_1, CANCTRL_GLOBAL_STS); + lpccan1.status.rcverrcnt = (state >> 16) & 0xFF; + lpccan1.status.snderrcnt = (state >> 24) & 0xFF; + lpccan1.status.errcode = (state >> 5) & 0x06; + } + //check Data Overrun Interrupt Interrupt + if((IntStatus >> 3) & 0x01) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_RXOF_IND | 0<<8); + } + //check Transmit Interrupt interrupt2 + if((IntStatus >> 9) & 0x01) + { + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_1, CANCTRL_STS); + if(state & (0x01 << 11)) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_DONE | 1<<8); + } else { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_FAIL | 1<<8); + } + } + //check Transmit Interrupt interrupt3 + if((IntStatus >> 10) & 0x01) + { + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_1, CANCTRL_STS); + if(state & (0x01 << 19)) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_DONE | 2<<8); + } else { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_FAIL | 2<<8); + } + } +#endif /*RT_USING_LPCCAN1*/ +#ifdef RT_USING_LPCCAN2 + IntStatus = CAN_IntGetStatus(CAN_2); + //check receive interrupt + if((IntStatus >> 0) & 0x01) + { + rt_hw_can_isr(&lpccan2,RT_CAN_EVENT_RX_IND | 0<<8); + } + //check Transmit Interrupt interrupt1 + if((IntStatus >> 1) & 0x01) + { + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_2, CANCTRL_STS); + if(state & (0x01 << 3)) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_DONE | 0<<8); + } else { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_FAIL | 0<<8); + } + } + //check Error Warning Interrupt + if((IntStatus >> 2) & 0x01) + { + rt_uint32_t errtype; + errtype = (IntStatus >> 16); + if(errtype & 0x1F && lpccan2.status.lasterrtype == (errtype & 0x1F)) { + switch((errtype & 0x1F)) { + case 00011: // Start of Frame + case 00010: // ID28 ... ID21 + case 00110: //ID20 ... ID18 + case 00100: // SRTR Bit + case 00101: // IDE bit + case 00111: // ID17 ... 13 + case 01111: // ID12 ... ID5 + case 01110: // ID4 ... ID0 + case 01100: // RTR Bit + case 01011: // Data Length Code + case 01010: // Data Field + lpccan2.status.formaterrcnt++; + break; + case 01101: // Reserved Bit 1 + case 01001: // Reserved Bit 0 + lpccan2.status.bitpaderrcnt++; + break; + case 01000: // CRC Sequence + case 11000: // CRC Delimiter + lpccan2.status.crcerrcnt++; + break; + case 11001: // Acknowledge Slot + case 11011: // Acknowledge Delimiter + lpccan2.status.ackerrcnt++; + break; + case 11010: // End of Frame + case 10010: // Intermission + lpccan2.status.formaterrcnt++; + break; + } + lpccan2.status.lasterrtype = errtype & 0x1F; + } + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_2, CANCTRL_GLOBAL_STS); + lpccan2.status.rcverrcnt = (state >> 16) & 0xFF; + lpccan2.status.snderrcnt = (state >> 24) & 0xFF; + lpccan2.status.errcode = (state >> 5) & 0x06; + } + //check Data Overrun Interrupt Interrupt + if((IntStatus >> 3) & 0x01) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_RXOF_IND | 0<<8); + } + //check Transmit Interrupt interrupt2 + if((IntStatus >> 9) & 0x01) + { + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_2, CANCTRL_STS); + if(state & (0x01 << 11)) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_DONE | 1<<8); + } else { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_FAIL | 1<<8); + } + } + //check Transmit Interrupt interrupt3 + if((IntStatus >> 10) & 0x01) + { + rt_uint32_t state = 0; + state = CAN_GetCTRLStatus(CAN_2, CANCTRL_STS); + if(state & (0x01 << 19)) + { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_DONE | 2<<8); + } else { + rt_hw_can_isr(&lpccan1,RT_CAN_EVENT_TX_FAIL | 2<<8); + } + } +#endif /*RT_USING_LPCCAN2*/ +} +int lpc_can_init(void) +{ +#ifdef RT_USING_LPCCAN1 + lpccan1.config.baud_rate=CAN1MBaud; + lpccan1.config.msgboxsz=16; + lpccan1.config.sndboxnumber=3; + lpccan1.config.mode=RT_CAN_MODE_NORMAL; + lpccan1.config.privmode=0; +#ifdef RT_CAN_USING_LED +#endif + lpccan1.config.ticks = 50; +#ifdef RT_CAN_USING_HDR +#endif + //Enable CAN Interrupt + NVIC_EnableIRQ(CAN_IRQn); + rt_hw_can_register(&lpccan1, "lpccan1", &canops, &lpccandata1); +#endif + +#ifdef RT_USING_LPCCAN2 + lpccan2.config.baud_rate=CAN1MBaud; + lpccan2.config.msgboxsz=16; + lpccan2.config.sndboxnumber=3; + lpccan2.config.mode=RT_CAN_MODE_NORMAL; + lpccan2.config.privmode=0; +#ifdef RT_CAN_USING_LED +#endif + lpccan2.config.ticks = 50; +#ifdef RT_CAN_USING_HDR +#endif +#ifndef RT_USING_LPCCAN1 + //Enable CAN Interrupt + NVIC_EnableIRQ(CAN_IRQn); +#endif +#ifdef RT_CAN_USING_HDR +#endif + rt_hw_can_register(&lpccan2, "lpccan2", &canops, &lpccandata2); +#endif + return RT_EOK; +} +INIT_BOARD_EXPORT(lpc_can_init); + +#endif /*RT_USING_CAN*/ diff --git a/bsp/lpc408x/drivers/drv_lpccan.h b/bsp/lpc408x/drivers/drv_lpccan.h new file mode 100644 index 0000000000..1d81912dc9 --- /dev/null +++ b/bsp/lpc408x/drivers/drv_lpccan.h @@ -0,0 +1,29 @@ +/* + * File : drv_lpccan.h + * This file is part of RT-Thread RTOS + * COPYRIGHT (C) 2015, RT-Thread Development Team + * + * The license and distribution terms for this file may be + * found in the file LICENSE in this distribution or at + * http://www.rt-thread.org/license/LICENSE + * + * Change Logs: + * Date Author Notes + * 2015-06-30 aubrcool@qq.com first version + */ + +#ifndef DRV_LPCCAN_H_ +#define DRV_LPCCAN_H_ +#include +#include + +#define LPC_CAN_AF_STD_INIT(id) \ + RT_CAN_FILTER_ITEM_INIT(id,0,0,0,0xFFFFFFFF) +#define LPC_CAN_AF_EXT_INIT(id) \ + RT_CAN_FILTER_ITEM_INIT(id,1,0,0,0xFFFFFFFF) +#define LPC_CAN_AF_STD_GRP_INIT(id1,id2) \ + RT_CAN_FILTER_ITEM_INIT(id1,0,0,1,id2) +#define LPC_CAN_AF_EXT_GRP_INIT(id1,id2) \ + RT_CAN_FILTER_ITEM_INIT(id1,1,0,1,id2) + +#endif /*DRV_LPCCAN_H_*/ diff --git a/bsp/lpc408x/project.uvproj b/bsp/lpc408x/project.uvprojx similarity index 84% rename from bsp/lpc408x/project.uvproj rename to bsp/lpc408x/project.uvprojx index 47403d9620..c5f481a937 100644 --- a/bsp/lpc408x/project.uvproj +++ b/bsp/lpc408x/project.uvprojx @@ -1,7 +1,10 @@ - - 1.1 + + + 2.1 +
### uVision Project, (C) Keil Software
+ RT-Thread LPC408x @@ -9,30 +12,33 @@ ARM-ADS - LPC4088 - NXP (founded by Philips) - IRAM(0x10000000-0x1000FFFF) IRAM2(0x20000000-0x20007FFF) IROM(0-0x7FFFF) CLOCK(12000000) CPUTYPE("Cortex-M4") FPU2 - - "STARTUP\NXP\LPC407x_8x_177x_8x\startup_LPC407x_8x_177x_8x.s" ("NXP LPC407x_8x_177x_8x Startup Code") - UL2CM3(-O4303 -S0 -C0 -FO7 -FD10000000 -FC800 -FN1 -FF0LPC_IAP_512 -FS00 -FL080000) - 6493 - LPC407x_8x_177x_8x.h - - - - - - - - - - SFD\NXP\LPC407x_8x_177x_8x\LPC408x_7x.SFR + LPC4088FBD208 + NXP + Keil.LPC4000_DFP.1.1.0 + http://www.keil.com/pack/ + IROM(0x00000000,0x80000) IRAM(0x10000000,0x10000) IRAM2(0x20000000,0x8000) CPUTYPE("Cortex-M4") FPU2 CLOCK(12000000) ELITTLE + + + UL2CM3(-S0 -C0 -P0 -FD10000000 -FCFE0 -FN1 -FF0LPC_IAP_512 -FS00 -FL080000 -FP0($$Device:LPC4088FBD208$Flash\LPC_IAP_512.FLM)) + 0 + $$Device:LPC4088FBD208$Device\Include\LPC407x_8x_177x_8x.h + + + + + + + + + + $$Device:LPC4088FBD208$SVD\LPC408x_7x.svd + 0 0 - - - - NXP\LPC407x_8x_177x_8x\ - NXP\LPC407x_8x_177x_8x\ + + + + + 0 0 @@ -54,8 +60,8 @@ 0 0 - - + + 0 0 0 @@ -64,21 +70,25 @@ 0 0 - - + + 0 0 + 0 + 0 0 0 - - + + 0 0 + 0 + 0 0 - + 0 @@ -92,16 +102,17 @@ 0 0 3 - - + + + 1 SARMCM3.DLL - -MPU + -MPU DCM.DLL -pCM4 SARMCM3.DLL - -MPU + -MPU TCM.DLL -pCM4 @@ -123,33 +134,35 @@ 1 1 0 + 1 1 1 - 0 + 1 1 1 1 0 1 0 + 1 0 - 7 + 6 - - - - - + + + + + - - - - - + + + + + Segger\JL2CM3.dll @@ -158,14 +171,18 @@ 1 0 0 - 0 + 1 1 4099 - 0 + 1 Segger\JL2CM3.dll "" () - + + + + + 0 @@ -197,7 +214,7 @@ 0 0 "Cortex-M4" - + 0 0 0 @@ -328,7 +345,7 @@ 0x8000 - + 1 @@ -344,10 +361,14 @@ 0 0 0 + 1 + 0 + 0 + 0 - + CORE_M4, RT_USING_ARM_LIBC - + Libraries/Device/NXP/LPC407x_8x_177x_8x/Include;Libraries/CMSIS/Include;Libraries/Drivers/include;applications;.;drivers;../../include;../../libcpu/arm/cortex-m4;../../libcpu/arm/common;../../components/pthreads;../../components/libc/armlibc;../../components/drivers/include;../../components/drivers/include;../../components/finsh @@ -360,11 +381,12 @@ 0 0 0 + 0 - - - - + + + + @@ -376,12 +398,13 @@ 0 0x00000000 0x10000000 - - - + + + + --keep *.o(.rti_fn.*) --keep *.o(FSymTab) --keep *.o(VSymTab) - - + + @@ -394,8 +417,6 @@ 1 Libraries/Device/NXP/LPC407x_8x_177x_8x/Source/Templates/system_LPC407x_8x_177x_8x.c - - startup_LPC407x_8x_177x_8x.s 2 @@ -411,176 +432,126 @@ 1 Libraries/Drivers/source/lpc_adc.c - - lpc_bod.c 1 Libraries/Drivers/source/lpc_bod.c - - lpc_can.c 1 Libraries/Drivers/source/lpc_can.c - - lpc_clkpwr.c 1 Libraries/Drivers/source/lpc_clkpwr.c - - lpc_crc.c 1 Libraries/Drivers/source/lpc_crc.c - - lpc_dac.c 1 Libraries/Drivers/source/lpc_dac.c - - lpc_eeprom.c 1 Libraries/Drivers/source/lpc_eeprom.c - - lpc_emc.c 1 Libraries/Drivers/source/lpc_emc.c - - lpc_exti.c 1 Libraries/Drivers/source/lpc_exti.c - - lpc_gpdma.c 1 Libraries/Drivers/source/lpc_gpdma.c - - lpc_gpio.c 1 Libraries/Drivers/source/lpc_gpio.c - - lpc_i2c.c 1 Libraries/Drivers/source/lpc_i2c.c - - lpc_i2s.c 1 Libraries/Drivers/source/lpc_i2s.c - - lpc_iap.c 1 Libraries/Drivers/source/lpc_iap.c - - lpc_lcd.c 1 Libraries/Drivers/source/lpc_lcd.c - - lpc_mcpwm.c 1 Libraries/Drivers/source/lpc_mcpwm.c - - lpc_nvic.c 1 Libraries/Drivers/source/lpc_nvic.c - - lpc_pinsel.c 1 Libraries/Drivers/source/lpc_pinsel.c - - lpc_pwm.c 1 Libraries/Drivers/source/lpc_pwm.c - - lpc_qei.c 1 Libraries/Drivers/source/lpc_qei.c - - lpc_rtc.c 1 Libraries/Drivers/source/lpc_rtc.c - - lpc_ssp.c 1 Libraries/Drivers/source/lpc_ssp.c - - lpc_systick.c 1 Libraries/Drivers/source/lpc_systick.c - - lpc_timer.c 1 Libraries/Drivers/source/lpc_timer.c - - lpc_uart.c 1 Libraries/Drivers/source/lpc_uart.c - - lpc_wwdt.c 1 @@ -596,27 +567,26 @@ 1 applications/application.c - - board.c 1 applications/board.c - - sram.c 1 applications/sram.c - - startup.c 1 applications/startup.c + + canapp.c + 1 + .\applications\canapp.c + @@ -627,13 +597,16 @@ 1 drivers/drv_led.c - - drv_uart.c 1 drivers/drv_uart.c + + drv_lpccan.c + 1 + .\drivers\drv_lpccan.c + @@ -644,92 +617,66 @@ 1 ../../src/clock.c - - components.c 1 ../../src/components.c - - device.c 1 ../../src/device.c - - idle.c 1 ../../src/idle.c - - ipc.c 1 ../../src/ipc.c - - irq.c 1 ../../src/irq.c - - kservice.c 1 ../../src/kservice.c - - mem.c 1 ../../src/mem.c - - memheap.c 1 ../../src/memheap.c - - mempool.c 1 ../../src/mempool.c - - object.c 1 ../../src/object.c - - scheduler.c 1 ../../src/scheduler.c - - thread.c 1 ../../src/thread.c - - timer.c 1 @@ -745,29 +692,21 @@ 1 ../../libcpu/arm/cortex-m4/cpuport.c - - context_rvds.S 2 ../../libcpu/arm/cortex-m4/context_rvds.S - - backtrace.c 1 ../../libcpu/arm/common/backtrace.c - - div0.c 1 ../../libcpu/arm/common/div0.c - - showmem.c 1 @@ -783,78 +722,56 @@ 1 ../../components/pthreads/clock_time.c - - mqueue.c 1 ../../components/pthreads/mqueue.c - - pthread.c 1 ../../components/pthreads/pthread.c - - pthread_attr.c 1 ../../components/pthreads/pthread_attr.c - - pthread_barrier.c 1 ../../components/pthreads/pthread_barrier.c - - pthread_cond.c 1 ../../components/pthreads/pthread_cond.c - - pthread_mutex.c 1 ../../components/pthreads/pthread_mutex.c - - pthread_rwlock.c 1 ../../components/pthreads/pthread_rwlock.c - - pthread_spin.c 1 ../../components/pthreads/pthread_spin.c - - pthread_tls.c 1 ../../components/pthreads/pthread_tls.c - - sched.c 1 ../../components/pthreads/sched.c - - semaphore.c 1 @@ -870,8 +787,6 @@ 1 ../../components/libc/armlibc/mem_std.c - - stubs.c 1 @@ -887,48 +802,41 @@ 1 ../../components/drivers/serial/serial.c - - completion.c 1 ../../components/drivers/src/completion.c - - dataqueue.c 1 ../../components/drivers/src/dataqueue.c - - pipe.c 1 ../../components/drivers/src/pipe.c - - portal.c 1 ../../components/drivers/src/portal.c - - ringbuffer.c 1 ../../components/drivers/src/ringbuffer.c - - workqueue.c 1 ../../components/drivers/src/workqueue.c + + can.c + 1 + ..\..\components\drivers\can\can.c + @@ -939,93 +847,96 @@ 1 ../../components/finsh/shell.c - - symbol.c 1 ../../components/finsh/symbol.c - - cmd.c 1 ../../components/finsh/cmd.c - - finsh_compiler.c 1 ../../components/finsh/finsh_compiler.c - - finsh_error.c 1 ../../components/finsh/finsh_error.c - - finsh_heap.c 1 ../../components/finsh/finsh_heap.c - - finsh_init.c 1 ../../components/finsh/finsh_init.c - - finsh_node.c 1 ../../components/finsh/finsh_node.c - - finsh_ops.c 1 ../../components/finsh/finsh_ops.c - - finsh_parser.c 1 ../../components/finsh/finsh_parser.c - - finsh_var.c 1 ../../components/finsh/finsh_var.c - - finsh_vm.c 1 ../../components/finsh/finsh_vm.c - - finsh_token.c 1 ../../components/finsh/finsh_token.c + + msh.c + 1 + ..\..\components\finsh\msh.c + + + msh_cmd.c + 1 + ..\..\components\finsh\msh_cmd.c + + + ::CMSIS + + + + + + + + + + + + + + +
diff --git a/bsp/lpc408x/rtconfig.h b/bsp/lpc408x/rtconfig.h index 27903b6dd8..ddb85fe8f9 100644 --- a/bsp/lpc408x/rtconfig.h +++ b/bsp/lpc408x/rtconfig.h @@ -72,6 +72,7 @@ #define RT_USING_DEVICE_IPC // #define RT_USING_SERIAL +#define RT_USING_CAN // #define RT_UART_RX_BUFFER_SIZE 64 // @@ -95,6 +96,7 @@ #define RT_USING_COMPONENTS_INIT //
#define RT_USING_FINSH +#define FINSH_USING_MSH // #define FINSH_USING_SYMTAB // @@ -236,5 +238,6 @@ //
// - +#define RT_USING_LPCCAN1 +#define RT_USING_CPU_FFS #endif