From ba70bb8ef478b9b68215538e02054c3ff03def79 Mon Sep 17 00:00:00 2001 From: Frank Voorburg Date: Mon, 31 Jul 2017 17:02:09 +0000 Subject: [PATCH] Refs #316. Implemented SocketCAN interface for the CAN driver module under Linux. git-svn-id: https://svn.code.sf.net/p/openblt/code/trunk@320 5dc33758-31d5-4daf-9ae8-b24bf3d40d73 --- Host/Source/LibOpenBLT/lint/gnu/co-gcc.lnt | 2 +- .../port/linux/canif/socketcan/socketcan.c | 634 ++++++++++++++++++ Host/Source/LibOpenBLT/port/linux/critutil.c | 6 - .../port/windows/canif/peak/pcanusb.c | 2 +- Host/Source/LibOpenBLT/xcptpcan.c | 12 + 5 files changed, 648 insertions(+), 8 deletions(-) create mode 100644 Host/Source/LibOpenBLT/port/linux/canif/socketcan/socketcan.c diff --git a/Host/Source/LibOpenBLT/lint/gnu/co-gcc.lnt b/Host/Source/LibOpenBLT/lint/gnu/co-gcc.lnt index a98c43d2..2132dc7b 100644 --- a/Host/Source/LibOpenBLT/lint/gnu/co-gcc.lnt +++ b/Host/Source/LibOpenBLT/lint/gnu/co-gcc.lnt @@ -207,5 +207,5 @@ size-options.lnt // This .lnt file should be generated (preferrably -esym(1055,*__builtin*) -esym(718,*__builtin*) // The compiler does not need these ... -esym(746,*__builtin*) // declared and it knows their prototypes. - +-esym(793, pthread_mutexattr_getprioceiling, pthread_mutexattr_setprioceiling) diff --git a/Host/Source/LibOpenBLT/port/linux/canif/socketcan/socketcan.c b/Host/Source/LibOpenBLT/port/linux/canif/socketcan/socketcan.c new file mode 100644 index 00000000..4f2a91a6 --- /dev/null +++ b/Host/Source/LibOpenBLT/port/linux/canif/socketcan/socketcan.c @@ -0,0 +1,634 @@ +/************************************************************************************//** +* \file socketcan.c +* \brief Linux SocketCAN interface source file. +* \ingroup SocketCan +* \internal +*---------------------------------------------------------------------------------------- +* C O P Y R I G H T +*---------------------------------------------------------------------------------------- +* Copyright (c) 2017 by Feaser http://www.feaser.com All rights reserved +* +*---------------------------------------------------------------------------------------- +* L I C E N S E +*---------------------------------------------------------------------------------------- +* This file is part of OpenBLT. OpenBLT is free software: you can redistribute it and/or +* modify it under the terms of the GNU General Public License as published by the Free +* Software Foundation, either version 3 of the License, or (at your option) any later +* version. +* +* OpenBLT is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; +* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR +* PURPOSE. See the GNU General Public License for more details. +* +* You have received a copy of the GNU General Public License along with OpenBLT. It +* should be located in ".\Doc\license.html". If not, contact Feaser to obtain a copy. +* +* \endinternal +****************************************************************************************/ + +/**************************************************************************************** +* Include files +****************************************************************************************/ +#include /* for assertions */ +#include /* for standard integer types */ +#include /* for NULL declaration */ +#include /* for boolean type */ +#include /* for standard library */ +#include /* for string library */ +#include /* for posix threads */ +#include /* UNIX standard functions */ +#include /* file control operations */ +#include /* I/O control operations */ +#include /* network interfaces */ +#include /* CAN kernel definitions */ +#include /* CAN raw sockets */ +#include /* CAN errors */ +#include "util.h" /* Utility module */ +#include "candriver.h" /* Generic CAN driver module */ +#include "socketcan.h" /* SocketCAN interface */ + + +/**************************************************************************************** +* Type definitions +****************************************************************************************/ +/** \brief Groups data for thread control. */ +typedef struct t_socket_can_thread_ctrl +{ + bool terminate; /**< flag to request thread termination. */ + bool terminated; /**< handshake flag. */ +} tSocketCanThreadCtrl; + + +/*************************************************************************************** +* Function prototypes +****************************************************************************************/ +/* CAN interface functions. */ +static void SocketCanInit(tCanSettings const * settings); +static void SocketCanTerminate(void); +static bool SocketCanConnect(void); +static void SocketCanDisconnect(void); +static bool SocketCanTransmit(tCanMsg const * msg); +static bool SocketCanIsBusError(void); +static void SocketCanRegisterEvents(tCanEvents const * events); +/* Event thread functions. */ +static bool SocketCanStartEventThread(void); +static void SocketCanStopEventThread(void); +static void *SocketCanEventThread(void *param); + + +/**************************************************************************************** +* Local constant declarations +****************************************************************************************/ +/** \brief CAN interface structure filled with SocketCAN specifics. */ +static const tCanInterface socketCanInterface = +{ + SocketCanInit, + SocketCanTerminate, + SocketCanConnect, + SocketCanDisconnect, + SocketCanTransmit, + SocketCanIsBusError, + SocketCanRegisterEvents +}; + + +/**************************************************************************************** +* Local data declarations +****************************************************************************************/ +/** \brief The settings to use in this CAN interface. */ +static tCanSettings socketCanSettings; + +/** \brief List with callback functions that this driver should use. */ +static volatile tCanEvents * socketCanEventsList; + +/** \brief Total number of event entries into the \ref socketCanEventsList list. */ +static volatile uint32_t socketCanEventsEntries; + +/** \brief Flag to set in the event thread when either a bus off or bus heavy situation + * + */ +static volatile bool socketCanErrorDetected; + +/** \brief Event thread control. */ +static volatile tSocketCanThreadCtrl eventThreadCtrl; + +/** \brief The ID of the event thread. */ +static pthread_t eventThreadId; + +/** \brief CAN raw socket. */ +static volatile int32_t canSocket; + + +/***********************************************************************************//** +** \brief Obtains a pointer to the CAN interface structure, so that it can be linked +** to the generic CAN driver module. +** \return Pointer to CAN interface structure. +** +****************************************************************************************/ +tCanInterface const * SocketCanGetInterface(void) +{ + return &socketCanInterface; +} /*** end of SocketCanGetInterface ***/ + + +/************************************************************************************//** +** \brief Initializes the CAN interface. Note that this module assumes that the CAN +** device was already properly configured and brought online on the Linux +** system. Terminal command "ip addr" can be used to verify this. +** \param settings Pointer to the CAN interface settings. +** +****************************************************************************************/ +static void SocketCanInit(tCanSettings const * settings) +{ + char * canDeviceName; + + /* Initialize locals. */ + socketCanEventsList = NULL; + socketCanEventsEntries = 0; + socketCanErrorDetected = false; + /* Reset CAN interface settings. */ + socketCanSettings.devicename = ""; + socketCanSettings.channel = 0; + socketCanSettings.baudrate = CAN_BR500K; + socketCanSettings.code = 0x00000000u; + socketCanSettings.mask = 0x00000000u; + + /* Initialize the critical section module. */ + UtilCriticalSectionInit(); + + /* Check parameters. */ + assert(settings != NULL); + + /* Only continue with valid parameters. */ + if (settings != NULL) /*lint !e774 */ + { + /* Shallow copy the CAN interface settings for later usage. */ + socketCanSettings = *settings; + /* The devicename is a pointer and it is not gauranteed that it stays valid so we + * need to deep copy this one. note the +1 for '\0' in malloc. + */ + assert(settings->devicename != NULL); + if (settings->devicename != NULL) /*lint !e774 */ + { + canDeviceName = malloc(strlen(settings->devicename) + 1); + assert(canDeviceName != NULL); + if (canDeviceName != NULL) /*lint !e774 */ + { + strcpy(canDeviceName, settings->devicename); + socketCanSettings.devicename = canDeviceName; + } + } + } +} /*** end of SocketCanInit ***/ + + +/************************************************************************************//** +** \brief Terminates the CAN interface. +** +****************************************************************************************/ +static void SocketCanTerminate(void) +{ + /* Release memory that was allocated for storing the device name. */ + if (socketCanSettings.devicename != NULL) + { + free((char *)socketCanSettings.devicename); + } + /* Reset CAN interface settings. */ + socketCanSettings.devicename = ""; + socketCanSettings.channel = 0; + socketCanSettings.baudrate = CAN_BR500K; + socketCanSettings.code = 0x00000000u; + socketCanSettings.mask = 0x00000000u; + /* Release memory that was allocated for CAN events and reset the entry count. */ + if ( (socketCanEventsList != NULL) && (socketCanEventsEntries != 0) ) + { + free((void *)socketCanEventsList); + socketCanEventsEntries = 0; + } + /* Terminate the critical section module. */ + UtilCriticalSectionTerminate(); +} /*** end of SocketCanTerminate ***/ + + +/************************************************************************************//** +** \brief Connects the CAN interface. Note that the channel and baudrate settings +** are ignored for the SocketCAN, because these are expected to be configured +** when the CAN device was brought online on the Linux system. +** \return True if connected, false otherwise. +** +****************************************************************************************/ +static bool SocketCanConnect(void) +{ + bool result = false; + struct sockaddr_can addr; + struct ifreq ifr; + int32_t flags; + struct can_filter rxFilter; + can_err_mask_t errMask; + + /* Reset the error flag. */ + socketCanErrorDetected = false; + + /* Check settings. */ + assert(socketCanSettings.devicename != NULL); + + /* Only continue with valid settings. */ + if (socketCanSettings.devicename != NULL) /*lint !e774 */ + { + /* Set positive result at this point and negate upon error detected. */ + result = true; + + /* Create an ifreq structure for passing data in and out of ioctl. */ + strncpy(ifr.ifr_name, socketCanSettings.devicename, IFNAMSIZ - 1); + ifr.ifr_name[IFNAMSIZ - 1] = '\0'; + + /* Get open socket descriptor */ + if ((canSocket = socket(PF_CAN, (int)SOCK_RAW, CAN_RAW)) < 0) + { + result = false; + } + + if (result) + { + /* Obtain interface index. */ + if (ioctl(canSocket, SIOCGIFINDEX, &ifr) < 0) + { + close(canSocket); + result = false; + } + } + + if (result) + { + /* Configure socket to work in non-blocking mode. */ + flags = fcntl(canSocket, F_GETFL, 0); + if (flags == -1) + { + flags = 0; + } + if (fcntl(canSocket, F_SETFL, flags | O_NONBLOCK) == -1) + { + close(canSocket); + result = false; + } + } + + if (result) + { + /* Set the address info. */ + addr.can_family = AF_CAN; + addr.can_ifindex = ifr.ifr_ifindex; + + /* Bind the socket. */ + if (bind(canSocket, (struct sockaddr *)&addr, sizeof(addr)) < 0) /*lint !e740 */ + { + close(canSocket); + result = false; + } + } + + /* Configure reception acceptance filter, if it is not supposed to be fully open. */ + if ( (result) && (socketCanSettings.mask != 0x00000000u) ) + { + /* Configure reception acceptance filter. The CAN_MSG_EXT_ID_MASK-bit from + * the CAN driver is currently the same as the CAN_EFF_FLAG-bit from SocketCAN. + * However, we don't know if this gets changed in the future in SocketCAN so + * process this bit manually. + */ + rxFilter.can_id = socketCanSettings.code; + if ((rxFilter.can_id & CAN_MSG_EXT_ID_MASK) != 0) + { + rxFilter.can_id &= ~CAN_MSG_EXT_ID_MASK; + rxFilter.can_id |= CAN_EFF_FLAG; + } + rxFilter.can_mask = socketCanSettings.mask; + if ((rxFilter.can_mask & CAN_MSG_EXT_ID_MASK) != 0) + { + rxFilter.can_mask &= ~CAN_MSG_EXT_ID_MASK; + rxFilter.can_mask |= CAN_EFF_FLAG; + } + if (setsockopt(canSocket, SOL_CAN_RAW, CAN_RAW_FILTER, &rxFilter, sizeof(rxFilter)) != 0) + { + close(canSocket); + result = false; + } + } + + /* Configure reception of bus off and other CAN controller status related events. */ + if (result) + { + errMask = (CAN_ERR_BUSOFF | CAN_ERR_CRTL); + if (setsockopt(canSocket, SOL_CAN_RAW, CAN_RAW_ERR_FILTER, &errMask, sizeof(errMask)) != 0) + { + close(canSocket); + result = false; + } + } + + if (result) + { + /* Start the event thread. */ + if (!SocketCanStartEventThread()) + { + close(canSocket); + result = false; + } + } + } + /* Give the result back to the caller. */ + return result; +} /*** end of SocketCanConnect ***/ + + +/************************************************************************************//** +** \brief Disconnects the CAN interface. +** +****************************************************************************************/ +static void SocketCanDisconnect(void) +{ + /* Stop the event thread. */ + SocketCanStopEventThread(); + /* Close the socket. */ + close(canSocket); + /* Reset the error flag. */ + socketCanErrorDetected = false; +} /*** end of SocketCanDisconnect ***/ + + +/************************************************************************************//** +** \brief Submits a message for transmission on the CAN bus. +** \param msg Pointer to CAN message structure. +** \return True if successful, false otherwise. +** +****************************************************************************************/ +static bool SocketCanTransmit(tCanMsg const * msg) +{ + bool result = false; + struct can_frame canTxFrame; + tCanEvents volatile const * pEvents; + + /* Check parameters. */ + assert(msg != NULL); + + /* Only continue with valid parameters. */ + if (msg != NULL) /*lint !e774 */ + { + /* Construct the message frame. */ + canTxFrame.can_id = msg->id; + if ((msg->id & CAN_MSG_EXT_ID_MASK) != 0) + { + canTxFrame.can_id &= ~CAN_MSG_EXT_ID_MASK; + canTxFrame.can_id |= CAN_EFF_FLAG; + } + canTxFrame.can_dlc = ((msg->dlc <= CAN_MSG_MAX_LEN) ? msg->dlc : CAN_MSG_MAX_LEN); + for (uint8_t idx = 0; idx < canTxFrame.can_dlc; idx++) + { + canTxFrame.data[idx] = msg->data[idx]; + } + + /* Submit the frame for transmission. */ + if (write(canSocket, &canTxFrame, sizeof(struct can_frame)) == (ssize_t)sizeof(struct can_frame)) + { + /* Update result value to success. */ + result = true; + /* Trigger transmit complete event(s). */ + pEvents = socketCanEventsList; + for (uint32_t idx = 0; idx < socketCanEventsEntries; idx++) + { + if (pEvents != NULL) + { + if (pEvents->MsgTxed != NULL) + { + pEvents->MsgTxed(msg); + } + /* Move on to the next entry in the list. */ + pEvents++; + } + } + } + } + /* Give the result back to the caller. */ + return result; +} /*** end of SocketCanTransmit ***/ + + +/************************************************************************************//** +** \brief Checks if a bus off or bus heavy situation occurred. +** \return True if a bus error situation was detected, false otherwise. +** +****************************************************************************************/ +static bool SocketCanIsBusError(void) +{ + bool result= false; + + /* Read flag to detect bus off or bus heavy. */ + UtilCriticalSectionEnter(); + if (socketCanErrorDetected) + { + /* Update result. */ + result = true; + /* Reset the error flag. */ + socketCanErrorDetected = false; + } + UtilCriticalSectionExit(); + + /* Give the result back to the caller. */ + return result; +} /*** end of SocketCanIsBusError ***/ + + +/************************************************************************************//** +** \brief Registers the event callback functions that should be called by the CAN +** interface. +** \param events Pointer to structure with event callback function pointers. +** +****************************************************************************************/ +static void SocketCanRegisterEvents(tCanEvents const * events) +{ + /* Check parameters. */ + assert(events != NULL); + + /* Only continue with valid parameters. */ + if (events != NULL) /*lint !e774 */ + { + /* Increase length of the list to make space for one more event entry. Note that + * it is okay to call realloc with a NULL pointer. In this case it simply behaves + * as malloc. + */ + socketCanEventsList = realloc((void *)socketCanEventsList, + (sizeof(tCanEvents) * (socketCanEventsEntries + 1))); + /* Assert reallocation. */ + assert(socketCanEventsList != NULL); + /* Only continue if reallocation was successful. */ + if (socketCanEventsList != NULL) + { + /* Increment events entry count. */ + socketCanEventsEntries++; + /* Store the events in the new entry. */ + socketCanEventsList[socketCanEventsEntries - 1] = *events; + } + /* Reallocation failed. */ + else + { + /* Reset events entry count. */ + socketCanEventsEntries = 0; + } + } +} /*** end of SocketCanRegisterEvents ***/ + + +/************************************************************************************//** +** \brief Starts the event thread. +** \return True if the thread was successfully started, false otherwise. +** +****************************************************************************************/ +static bool SocketCanStartEventThread(void) +{ + bool result = false; + + /* Prepare thread parameters. */ + UtilCriticalSectionEnter(); + eventThreadCtrl.terminate = false; + eventThreadCtrl.terminated = false; + UtilCriticalSectionExit(); + /* Create and start the event thread */ + if (pthread_create(&eventThreadId, NULL, &SocketCanEventThread, NULL) == 0) /*lint !e546 */ + { + result = true; + } + /* Give the result back to the caller. */ + return result; +} /*** end of SocketCanStartEventThread ***/ + + +/************************************************************************************//** +** \brief Stops the event thread. It sets the termination request and then waits for +** the termination handshake. +** \return None. +** +****************************************************************************************/ +static void SocketCanStopEventThread(void) +{ + bool terminatedCpy = false; + + /* Set the terminate request. */ + UtilCriticalSectionEnter(); + eventThreadCtrl.terminate = true; + UtilCriticalSectionExit(); + /* Wait for thread to terminate. */ + while (!terminatedCpy) + { + UtilCriticalSectionEnter(); + terminatedCpy = eventThreadCtrl.terminated; + UtilCriticalSectionExit(); + /* Wait a little without starving the CPU. */ + UtilTimeDelayMs(10); + } +} /*** end of SocketCanStopEventThread ***/ + + +/************************************************************************************//** +** \brief Event thread that handles the asynchronous reception of data from the CAN +** interface. +** \param arg Pointer to thread parameters. +** \return Thread return value. Not used in this case, so always set to NULL. +** +****************************************************************************************/ +static void *SocketCanEventThread(void *param) +{ + bool terminateRequest = false; + struct can_frame canRxFrame; + tCanMsg rxMsg; + tCanEvents volatile const * pEvents; + + /* Unused parameter. */ + (void)param; + + /* Run thread as long as it is not requested to terminate. */ + while (!terminateRequest) + { + /* Update termination request. */ + UtilCriticalSectionEnter(); + terminateRequest = eventThreadCtrl.terminate; + UtilCriticalSectionExit(); + + /* Check if CAN frames were received. */ + while (read(canSocket, &canRxFrame, sizeof(struct can_frame)) == (ssize_t)sizeof(struct can_frame)) + { + /* Ignore remote frames */ + if (canRxFrame.can_id & CAN_RTR_FLAG) + { + continue; + } + /* Does the message contain error information? */ + else if (canRxFrame.can_id & CAN_ERR_FLAG) + { + /* Was it a bus off event? */ + if ((canRxFrame.can_id & CAN_ERR_BUSOFF) != 0) + { + /* Set the error flag. */ + UtilCriticalSectionEnter(); + socketCanErrorDetected = true; + UtilCriticalSectionExit(); + } + /* Was it a CAN controller event? */ + else if ((canRxFrame.can_id & CAN_ERR_CRTL) != 0) + { + /* Is the controller in error passive mode (bus heavy)? */ + if ((canRxFrame.data[1] & (CAN_ERR_CRTL_RX_PASSIVE | CAN_ERR_CRTL_TX_PASSIVE)) != 0) + { + /* Set the error flag. */ + UtilCriticalSectionEnter(); + socketCanErrorDetected = true; + UtilCriticalSectionExit(); + } + } + } + /* It was a regular CAN message with either 11- or 29-bit identifier. */ + else + { + /* Copy it to the CAN message object */ + rxMsg.id = canRxFrame.can_id; + if (rxMsg.id & CAN_EFF_FLAG) + { + rxMsg.id &= ~CAN_EFF_FLAG; + rxMsg.id |= CAN_MSG_EXT_ID_MASK; + } + rxMsg.dlc = canRxFrame.can_dlc; + for (uint8_t idx = 0; idx < rxMsg.dlc; idx++) + { + rxMsg.data[idx] = canRxFrame.data[idx]; + } + + /* Trigger message reception event(s). */ + pEvents = socketCanEventsList; + for (uint32_t idx = 0; idx < socketCanEventsEntries; idx++) + { + if (pEvents != NULL) + { + if (pEvents->MsgRxed != NULL) + { + pEvents->MsgRxed(&rxMsg); + } + /* Move on to the next entry in the list. */ + pEvents++; + } + } + } + } + /* wait a little to not starve the CPU */ + UtilTimeDelayMs(1); + } + + /* handshake */ + UtilCriticalSectionEnter(); + eventThreadCtrl.terminated = true; + UtilCriticalSectionExit(); + + /* exit the thread */ + return NULL; +} /*** end of SocketCanEventThread ***/ + + +/*********************************** end of socketcan.c ********************************/ + diff --git a/Host/Source/LibOpenBLT/port/linux/critutil.c b/Host/Source/LibOpenBLT/port/linux/critutil.c index 4c0c7ed8..5b2d88b5 100644 --- a/Host/Source/LibOpenBLT/port/linux/critutil.c +++ b/Host/Source/LibOpenBLT/port/linux/critutil.c @@ -147,11 +147,5 @@ void UtilCriticalSectionExit(void) } /*** end of UtilCriticalSectionExit ***/ -/*lint -esym(793, pthread_mutexattr_getprioceiling, pthread_mutexattr_setprioceiling) - * suppress info message regarding 31 significant character limit for these library - * functions. - */ - - /*********************************** end of critutil.c *********************************/ diff --git a/Host/Source/LibOpenBLT/port/windows/canif/peak/pcanusb.c b/Host/Source/LibOpenBLT/port/windows/canif/peak/pcanusb.c index a41acce4..2d0fc303 100644 --- a/Host/Source/LibOpenBLT/port/windows/canif/peak/pcanusb.c +++ b/Host/Source/LibOpenBLT/port/windows/canif/peak/pcanusb.c @@ -535,7 +535,7 @@ static bool PCanUsbTransmit(tCanMsg const * msg) msgBuf.ID = msg->id & 0x1fffffffu; msgBuf.MSGTYPE = PCAN_MESSAGE_EXTENDED; } - msgBuf.LEN = msg->dlc; + msgBuf.LEN = ((msg->dlc <= CAN_MSG_MAX_LEN) ? msg->dlc : CAN_MSG_MAX_LEN); for (uint8_t idx = 0; idx < msgBuf.LEN; idx++) { msgBuf.DATA[idx] = msg->data[idx]; diff --git a/Host/Source/LibOpenBLT/xcptpcan.c b/Host/Source/LibOpenBLT/xcptpcan.c index 69947371..f97c8f43 100644 --- a/Host/Source/LibOpenBLT/xcptpcan.c +++ b/Host/Source/LibOpenBLT/xcptpcan.c @@ -43,6 +43,9 @@ #if defined(PLATFORM_WIN32) #include "pcanusb.h" /* Peak PCAN-USB interface */ #endif +#if defined(PLATFORM_LINUX) +#include "socketcan.h" /* SocketCAN interface */ +#endif /**************************************************************************************** @@ -156,6 +159,9 @@ static void XcpTpCanInit(void const * settings) strcpy(canDeviceName, ((tXcpTpCanSettings *)settings)->device); tpCanSettings.device = canDeviceName; + /* ##Vg TODO Refactor such that the CAN driver does this interface linking + * automatically. + */ /* Determine the pointer to the correct CAN interface, based on the specified * device name. */ @@ -164,6 +170,12 @@ static void XcpTpCanInit(void const * settings) { canInterface = PCanUsbGetInterface(); } +#endif +#if defined(PLATFORM_LINUX) + /* On Linux, the device name is the name of the SocketCAN link, so always link + * the SocketCAN interface to the CAN driver. + */ + canInterface = SocketCanGetInterface(); #endif } }