Files
linear-Slide/Middleware/CANopenNode/301/CO_PDO.c

1473 lines
54 KiB
C

/*
* CANopen Receive Process Data Object protocol.
*
* @file CO_PDO.c
* @ingroup CO_PDO
* @author Janez Paternoster
* @copyright 2021 Janez Paternoster
*
* This file is part of <https://github.com/CANopenNode/CANopenNode>, a CANopen Stack.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
* file except in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is
* distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and limitations under the License.
*/
#include <string.h>
#include "301/CO_PDO.h"
#if ((CO_CONFIG_PDO) & (CO_CONFIG_RPDO_ENABLE | CO_CONFIG_TPDO_ENABLE)) != 0
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_OD_IO_ACCESS) == 0
#error Dynamic PDO mapping is not possible without CO_CONFIG_PDO_OD_IO_ACCESS
#endif
#endif
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_OD_IO_ACCESS) == 0
#error Bitwise PDO mapping is not possible without CO_CONFIG_PDO_OD_IO_ACCESS
#endif
#endif
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_OD_IO_ACCESS) != 0
/*
* Custom function for write dummy OD object. Will be used only from RPDO.
*
* For more information see file CO_ODinterface.h, OD_IO_t.
*/
static ODR_t
OD_write_dummy(OD_stream_t* stream, const void* buf, OD_size_t count, OD_size_t* countWritten) {
(void)stream;
(void)buf;
if (countWritten != NULL) {
*countWritten = count;
}
return ODR_OK;
}
/*
* Custom function for read dummy OD object. Will be used only from TPDO.
*
* For more information see file CO_ODinterface.h, OD_IO_t.
*/
static ODR_t
OD_read_dummy(OD_stream_t* stream, void* buf, OD_size_t count, OD_size_t* countRead) {
if ((buf == NULL) || (stream == NULL) || (countRead == NULL)) {
return ODR_DEV_INCOMPAT;
}
if (count > stream->dataLength) {
count = stream->dataLength;
}
(void)memset(buf, 0, count);
*countRead = count;
return ODR_OK;
}
/*
* Find mapped variable in Object Dictionary and configure entry in RPDO or TPDO
*
* @param PDO This object will be configured. If map is erroneous, then it will stay unchanged.
* @param map PDO mapping parameter.
* @param mapIndex from 0 to CO_PDO_MAX_MAPPED_ENTRIES
* @param isRPDO True for RPDO and false for TPDO.
* @param OD Object Dictionary.
*
* @return ODR_OK on success, otherwise error reason.
*/
static ODR_t
PDOconfigMap(CO_PDO_common_t* PDO, uint32_t map, uint8_t mapIndex, bool_t isRPDO, OD_t* OD) {
uint16_t index = (uint16_t)(map >> 16);
uint8_t subIndex = (uint8_t)(map >> 8);
uint8_t mappedLengthBits = (uint8_t)map;
uint8_t mappedLength = mappedLengthBits >> 3;
OD_IO_t* OD_IO = &PDO->OD_IO[mapIndex];
/* total PDO length can not be more than CO_PDO_MAX_SIZE bytes */
if (mappedLength > CO_PDO_MAX_SIZE) {
return ODR_MAP_LEN; /* PDO length exceeded */
}
/* is there a reference to the dummy entry */
if ((index < 0x20U) && (subIndex == 0U)) {
OD_stream_t* stream = &OD_IO->stream;
(void)memset(stream, 0, sizeof(OD_stream_t));
stream->dataLength = mappedLength;
stream->dataOffset = mappedLength;
OD_IO->read = OD_read_dummy;
OD_IO->write = OD_write_dummy;
return ODR_OK;
}
/* find entry in the Object Dictionary */
OD_IO_t OD_IOcopy;
OD_entry_t* entry = OD_find(OD, index);
ODR_t odRet = OD_getSub(entry, subIndex, &OD_IOcopy, false);
if (odRet != ODR_OK) {
return odRet;
}
/* verify access attributes, byte alignment and length */
OD_attr_t testAttribute = isRPDO ? (OD_attr_t)(ODA_RPDO) : (OD_attr_t)(ODA_TPDO);
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
/* If the bitwise mapping is enabled, check only length in bits */
if (((OD_IOcopy.stream.attribute & testAttribute) == 0U)
|| ((OD_IOcopy.stream.dataLength * 8) < mappedLengthBits)) {
#else
/* If the bitwise mapping is disabled, theck the alignment and length in bytes*/
if (((OD_IOcopy.stream.attribute & testAttribute) == 0U) || ((mappedLengthBits & 0x07U) != 0U)
|| (OD_IOcopy.stream.dataLength < mappedLength)) {
#endif /* (CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING) */
return ODR_NO_MAP; /* Object cannot be mapped to the PDO. */
}
/* Copy values and store mappedLength temporary. */
*OD_IO = OD_IOcopy;
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
OD_IO->stream.dataOffset = mappedLengthBits;
#else
OD_IO->stream.dataOffset = mappedLength;
#endif
/* get TPDO request flag byte from extension */
#if OD_FLAGS_PDO_SIZE > 0
if (!isRPDO) {
if ((subIndex < (OD_FLAGS_PDO_SIZE * 8U)) && (entry->extension != NULL)) {
PDO->flagPDObyte[mapIndex] = &entry->extension->flagsPDO[subIndex >> 3];
PDO->flagPDObitmask[mapIndex] = 1U << (subIndex & 0x07U);
} else {
PDO->flagPDObyte[mapIndex] = NULL;
}
}
#endif
return ODR_OK;
}
/*
* Initialize PDO mapping parameters
*
* @param PDO This object.
* @param OD Object Dictionary.
* @param OD_PDOMapPar OD entry for "PDO mapping parameter".
* @param isRPDO True for RPDO and false for TPDO.
* @param [out] errInfo Additional information in case of error, may be NULL.
* @param [out] erroneousMap Additional information about erroneous map.
*
* @return #CO_ReturnError_t CO_ERROR_NO on success.
*/
static CO_ReturnError_t
PDO_initMapping(CO_PDO_common_t* PDO, OD_t* OD, OD_entry_t* OD_PDOMapPar, bool_t isRPDO, uint32_t* errInfo,
uint32_t* erroneousMap) {
ODR_t odRet;
size_t pdoDataLength = 0;
uint8_t mappedObjectsCount = 0;
/* number of mapped application objects in PDO */
odRet = OD_get_u8(OD_PDOMapPar, 0, &mappedObjectsCount, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = ((uint32_t)OD_getIndex(OD_PDOMapPar)) << 8;
}
return CO_ERROR_OD_PARAMETERS;
}
for (uint8_t i = 0; i < CO_PDO_MAX_MAPPED_ENTRIES; i++) {
OD_IO_t* OD_IO = &PDO->OD_IO[i];
uint32_t map = 0;
odRet = OD_get_u32(OD_PDOMapPar, i + 1U, &map, true);
if (odRet == ODR_SUB_NOT_EXIST) {
continue;
}
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = (((uint32_t)OD_getIndex(OD_PDOMapPar)) << 8) | i;
}
return CO_ERROR_OD_PARAMETERS;
}
odRet = PDOconfigMap(PDO, map, i, isRPDO, OD);
if (odRet != ODR_OK) {
/* indicate erroneous mapping in initialization phase */
OD_IO->stream.dataLength = 0;
OD_IO->stream.dataOffset = 0xFF;
if (*erroneousMap == 0U) {
*erroneousMap = map;
}
}
if (i < mappedObjectsCount) {
pdoDataLength += OD_IO->stream.dataOffset;
}
}
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
if ((pdoDataLength > CO_PDO_MAX_SIZE * 8) || ((pdoDataLength == 0U) && (mappedObjectsCount > 0U))) {
#else
if ((pdoDataLength > CO_PDO_MAX_SIZE) || ((pdoDataLength == 0U) && (mappedObjectsCount > 0U))) {
#endif
if (*erroneousMap == 0U) {
*erroneousMap = 1;
}
}
if (*erroneousMap == 0U) {
PDO->dataLength = (CO_PDO_size_t)pdoDataLength;
PDO->mappedObjectsCount = mappedObjectsCount;
}
return CO_ERROR_NO;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
/*
* Custom function for writing OD object "PDO mapping parameter"
*
* For more information see file CO_ODinterface.h, OD_IO_t.
*/
static ODR_t
OD_write_PDO_mapping(OD_stream_t* stream, const void* buf, OD_size_t count, OD_size_t* countWritten) {
/* "count" is already verified in *_init() function */
if ((stream == NULL) || (buf == NULL) || (countWritten == NULL) || (stream->subIndex > CO_PDO_MAX_MAPPED_ENTRIES)) {
return ODR_DEV_INCOMPAT;
}
/* Only common part of the CO_RPDO_t or CO_TPDO_t will be used */
CO_PDO_common_t* PDO = stream->object;
/* PDO must be disabled before mapping configuration */
if ((PDO->valid) || ((PDO->mappedObjectsCount != 0U) && (stream->subIndex > 0U))) {
return ODR_UNSUPP_ACCESS;
}
if (stream->subIndex == 0U) {
uint8_t mappedObjectsCount = CO_getUint8(buf);
size_t pdoDataLength = 0;
if (mappedObjectsCount > CO_PDO_MAX_MAPPED_ENTRIES) {
return ODR_MAP_LEN;
}
/* validate enabled mapping parameters */
for (uint8_t i = 0; i < mappedObjectsCount; i++) {
OD_IO_t* OD_IO = &PDO->OD_IO[i];
size_t dataLength = (size_t)OD_IO->stream.dataLength;
size_t mappedLength = (size_t)OD_IO->stream.dataOffset;
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
if (mappedLength > dataLength * 8) {
#else
if (mappedLength > dataLength) {
#endif
/* erroneous map since device initial values */
return ODR_NO_MAP;
}
pdoDataLength += mappedLength;
}
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
if (pdoDataLength > CO_PDO_MAX_SIZE * 8) {
#else
if (pdoDataLength > CO_PDO_MAX_SIZE) {
#endif
return ODR_MAP_LEN;
}
if ((pdoDataLength == 0U) && (mappedObjectsCount > 0U)) {
return ODR_INVALID_VALUE;
}
/* success, update PDO */
PDO->dataLength = (CO_PDO_size_t)pdoDataLength;
PDO->mappedObjectsCount = mappedObjectsCount;
} else {
uint32_t val = CO_getUint32(buf);
ODR_t odRet = PDOconfigMap(PDO, val, stream->subIndex - 1U, PDO->isRPDO, PDO->OD);
if (odRet != ODR_OK) {
return odRet;
}
}
/* write value to the original location in the Object Dictionary */
return OD_writeOriginal(stream, buf, count, countWritten);
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_FLAG_OD_DYNAMIC */
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_PDO_OD_IO_ACCESS */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_OD_IO_ACCESS) == 0
static CO_ReturnError_t
PDO_initMapping(CO_PDO_common_t* PDO, OD_t* OD, OD_entry_t* OD_PDOMapPar, bool_t isRPDO, uint32_t* errInfo,
uint32_t* erroneousMap) {
ODR_t odRet;
size_t pdoDataLength = 0;
/* number of mapped application objects in PDO */
uint8_t mappedObjectsCount = 0;
odRet = OD_get_u8(OD_PDOMapPar, 0, &mappedObjectsCount, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = ((uint32_t)OD_getIndex(OD_PDOMapPar)) << 8;
}
return CO_ERROR_OD_PARAMETERS;
}
if (mappedObjectsCount > CO_PDO_MAX_MAPPED_ENTRIES) {
*erroneousMap = 1;
return CO_ERROR_NO;
}
/* iterate mapped OD variables */
for (uint8_t i = 0; i < mappedObjectsCount; i++) {
uint32_t map = 0;
odRet = OD_get_u32(OD_PDOMapPar, i + 1, &map, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = (((uint32_t)OD_getIndex(OD_PDOMapPar)) << 8) | i;
}
return CO_ERROR_OD_PARAMETERS;
}
uint16_t index = (uint16_t)(map >> 16);
uint8_t subIndex = (uint8_t)(map >> 8);
uint8_t mappedLengthBits = (uint8_t)map;
uint8_t mappedLength = mappedLengthBits >> 3;
uint8_t pdoDataStart = pdoDataLength;
pdoDataLength += mappedLength;
if ((mappedLengthBits & 0x07) != 0 || pdoDataLength > CO_PDO_MAX_SIZE) {
*erroneousMap = map;
return CO_ERROR_NO;
}
/* is there a reference to the dummy entry */
if (index < 0x20 && subIndex == 0) {
for (uint8_t j = pdoDataStart; j < pdoDataLength; j++) {
static uint8_t dummyTX = 0;
static uint8_t dummyRX;
PDO->mapPointer[j] = isRPDO ? &dummyRX : &dummyTX;
}
continue;
}
/* find entry in the Object Dictionary, original location */
OD_IO_t OD_IO;
OD_entry_t* entry = OD_find(OD, index);
OD_attr_t testAttribute = isRPDO ? ODA_RPDO : ODA_TPDO;
ODR_t odRet = OD_getSub(entry, subIndex, &OD_IO, true);
if (odRet != ODR_OK || (OD_IO.stream.attribute & testAttribute) == 0 || OD_IO.stream.dataLength < mappedLength
|| OD_IO.stream.dataOrig == NULL) {
*erroneousMap = map;
return CO_ERROR_NO;
}
/* write locations to OD variable data bytes into PDO map pointers */
#ifdef CO_BIG_ENDIAN
if ((OD_IO.stream.attribute & ODA_MB) != 0) {
uint8_t* odDataPointer = OD_IO.stream.dataOrig + OD_IO.stream.dataLength - 1;
for (uint8_t j = pdoDataStart; j < pdoDataLength; j++) {
PDO->mapPointer[j] = odDataPointer--;
}
} else
#endif
{
uint8_t* odDataPointer = OD_IO.stream.dataOrig;
for (uint8_t j = pdoDataStart; j < pdoDataLength; j++) {
PDO->mapPointer[j] = odDataPointer++;
}
}
/* get TPDO request flag byte from extension */
#if OD_FLAGS_PDO_SIZE > 0
if (!isRPDO && subIndex < (OD_FLAGS_PDO_SIZE * 8) && entry->extension != NULL) {
PDO->flagPDObyte[pdoDataStart] = &entry->extension->flagsPDO[subIndex >> 3];
PDO->flagPDObitmask[pdoDataStart] = 1 << (subIndex & 0x07);
}
#endif
}
PDO->dataLength = PDO->mappedObjectsCount = pdoDataLength;
return CO_ERROR_NO;
}
#endif /* ((CO_CONFIG_PDO) & CO_CONFIG_PDO_OD_IO_ACCESS) == 0 */
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
/*
* Custom function for reading OD object "PDO communication parameter"
*
* For more information see file CO_ODinterface.h, OD_IO_t.
*/
static ODR_t
OD_read_PDO_commParam(OD_stream_t* stream, void* buf, OD_size_t count, OD_size_t* countRead) {
ODR_t returnCode = OD_readOriginal(stream, buf, count, countRead);
/* When reading COB_ID, add Node-Id to the read value, if necessary */
if ((returnCode == ODR_OK) && (stream->subIndex == 1U) && (*countRead == 4U)) {
/* Only common part of the CO_RPDO_t or CO_TPDO_t will be used */
CO_PDO_common_t* PDO = stream->object;
uint32_t COB_ID = CO_getUint32(buf);
uint16_t CAN_ID = (uint16_t)(COB_ID & 0x7FFU);
/* If default CAN-ID is stored in OD (without Node-ID), add Node-ID */
if ((CAN_ID != 0U) && (CAN_ID == (PDO->preDefinedCanId & 0xFF80U))) {
COB_ID = (COB_ID & 0xFFFF0000U) | PDO->preDefinedCanId;
}
/* If PDO is not valid, set bit 31 */
if (!PDO->valid) {
COB_ID |= 0x80000000U;
}
(void)CO_setUint32(buf, COB_ID);
}
return returnCode;
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_FLAG_OD_DYNAMIC */
/*******************************************************************************
* R P D O
******************************************************************************/
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_ENABLE) != 0
/*
* @defgroup CO_PDO_receiveErrors_t States for RPDO->receiveError indicates received RPDOs with wrong length.
* @{
*
*/
#define CO_RPDO_RX_ACK_NO_ERROR 0U /* No error */
#define CO_RPDO_RX_ACK_ERROR 1U /* Error is acknowledged */
#define CO_RPDO_RX_ACK 10U /* Auxiliary value */
#define CO_RPDO_RX_OK 11U /* Correct RPDO received, not acknowledged */
#define CO_RPDO_RX_SHORT 12U /* Too short RPDO received, not acknowledged */
#define CO_RPDO_RX_LONG 13U /* Too long RPDO received, not acknowledged */
/* @} */ /* CO_PDO_receiveErrors_t */
/*
* Read received message from CAN module.
*
* Function will be called (by CAN receive interrupt) every time, when CAN message with correct identifier
* will be received. For more information and description of parameters see file CO_driver.h.
* If new message arrives and previous message wasn't processed yet, then
* previous message will be lost and overwritten by the new message.
*/
static void
CO_PDO_receive(void* object, void* msg) {
CO_RPDO_t* RPDO = object;
CO_PDO_common_t* PDO = &RPDO->PDO_common;
uint8_t DLC = CO_CANrxMsg_readDLC(msg);
const uint8_t* data = CO_CANrxMsg_readData(msg);
uint8_t err = RPDO->receiveError;
if (PDO->valid) {
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
if (DLC >= (PDO->dataLength + 7) / 8) {
#else
if (DLC >= PDO->dataLength) {
#endif
/* indicate errors in PDO length */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
if (DLC == (PDO->dataLength + 7) / 8) {
#else
if (DLC == PDO->dataLength) {
#endif
if (err == CO_RPDO_RX_ACK_ERROR) {
err = CO_RPDO_RX_OK;
}
} else {
if (err == CO_RPDO_RX_ACK_NO_ERROR) {
err = CO_RPDO_RX_LONG;
}
}
/* Determine, to which of the two rx buffers copy the message. */
uint8_t bufNo = 0;
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
if (RPDO->synchronous && (RPDO->SYNC != NULL) && RPDO->SYNC->CANrxToggle) {
bufNo = 1;
}
#endif
/* copy data into appropriate buffer and set 'new message' flag */
(void)memcpy(RPDO->CANrxData[bufNo], data, CO_PDO_MAX_SIZE);
CO_FLAG_SET(RPDO->CANrxNew[bufNo]);
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_CALLBACK_PRE) != 0
/* Optional signal to RTOS, which can resume task, which handles the RPDO. */
if (RPDO->pFunctSignalPre != NULL) {
RPDO->pFunctSignalPre(RPDO->functSignalObjectPre);
}
#endif
} else if (err == CO_RPDO_RX_ACK_NO_ERROR) {
err = CO_RPDO_RX_SHORT;
} else { /* MISRA C 2004 14.10 */
}
}
RPDO->receiveError = err;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
/*
* Custom function for writing OD object "RPDO communication parameter"
*
* For more information see file CO_ODinterface.h, OD_IO_t.
*/
static ODR_t
OD_write_14xx(OD_stream_t* stream, const void* buf, OD_size_t count, OD_size_t* countWritten) {
/* "count" is also verified in *_init() function */
if ((stream == NULL) || (buf == NULL) || (countWritten == NULL) || (count > 4U)) {
return ODR_DEV_INCOMPAT;
}
CO_RPDO_t* RPDO = stream->object;
CO_PDO_common_t* PDO = &RPDO->PDO_common;
uint8_t bufCopy[4];
(void)memcpy((void*)bufCopy, (const void*)buf, count);
switch (stream->subIndex) {
case 1: { /* COB-ID used by PDO */
uint32_t COB_ID = CO_getUint32(buf);
uint16_t CAN_ID = (uint16_t)(COB_ID & 0x7FFU);
bool_t valid = (COB_ID & 0x80000000U) == 0U;
/* bits 11...29 must be zero, PDO must be disabled on change, CAN_ID == 0 is
* not allowed, mapping must be configured before enabling the PDO */
if (((COB_ID & 0x3FFFF800U) != 0U) || (valid && PDO->valid && (CAN_ID != PDO->configuredCanId))
|| (valid && CO_IS_RESTRICTED_CAN_ID(CAN_ID)) || (valid && (PDO->mappedObjectsCount == 0U))) {
return ODR_INVALID_VALUE;
}
/* parameter changed? */
if ((valid != PDO->valid) || (CAN_ID != PDO->configuredCanId)) {
/* if default CAN-ID is written, store to OD without Node-ID */
if (CAN_ID == PDO->preDefinedCanId) {
(void)CO_setUint32(bufCopy, COB_ID & 0xFFFFFF80U);
}
if (!valid) {
CAN_ID = 0;
}
CO_ReturnError_t ret = CO_CANrxBufferInit(PDO->CANdev, PDO->CANdevIdx, CAN_ID, 0x7FF, false,
(void*)RPDO, CO_PDO_receive);
if (valid && (ret == CO_ERROR_NO)) {
PDO->valid = true;
PDO->configuredCanId = CAN_ID;
} else {
PDO->valid = false;
CO_FLAG_CLEAR(RPDO->CANrxNew[0]);
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
CO_FLAG_CLEAR(RPDO->CANrxNew[1]);
#endif
if (ret != CO_ERROR_NO) {
return ODR_DEV_INCOMPAT;
}
}
}
break;
}
case 2: { /* transmission type */ uint8_t transmissionType = CO_getUint8(buf);
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
if ((transmissionType > (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240)
&& (transmissionType < (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO)) {
return ODR_INVALID_VALUE;
}
bool_t synchronous = transmissionType <= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240;
/* Remove old message from the second buffer. */
if (RPDO->synchronous != synchronous) {
CO_FLAG_CLEAR(RPDO->CANrxNew[1]);
}
RPDO->synchronous = synchronous;
#else
if (transmissionType < CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO) {
return ODR_INVALID_VALUE;
}
#endif
break;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
case 5: { /* event-timer */
uint32_t eventTime = CO_getUint16(buf);
RPDO->timeoutTime_us = eventTime * 1000U;
RPDO->timeoutTimer = 0;
break;
}
#endif
default:
/* MISRA C 2004 15.3 */
break;
}
/* write value to the original location in the Object Dictionary */
return OD_writeOriginal(stream, bufCopy, count, countWritten);
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_FLAG_OD_DYNAMIC */
CO_ReturnError_t
CO_RPDO_init(CO_RPDO_t* RPDO, OD_t* OD, CO_EM_t* em,
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
CO_SYNC_t* SYNC,
#endif
uint16_t preDefinedCanId, OD_entry_t* OD_14xx_RPDOCommPar, OD_entry_t* OD_16xx_RPDOMapPar,
CO_CANmodule_t* CANdevRx, uint16_t CANdevRxIdx, uint32_t* errInfo) {
CO_PDO_common_t* PDO = &RPDO->PDO_common;
CO_ReturnError_t ret;
ODR_t odRet;
/* verify arguments */
if ((RPDO == NULL) || (OD == NULL) || (em == NULL) || (OD_14xx_RPDOCommPar == NULL) || (OD_16xx_RPDOMapPar == NULL)
|| (CANdevRx == NULL)) {
return CO_ERROR_ILLEGAL_ARGUMENT;
}
/* clear object */
(void)memset(RPDO, 0, sizeof(CO_RPDO_t));
/* Configure object variables */
PDO->em = em;
PDO->CANdev = CANdevRx;
/* Configure mapping parameters */
uint32_t erroneousMap = 0;
ret = PDO_initMapping(PDO, OD, OD_16xx_RPDOMapPar, true, errInfo, &erroneousMap);
if (ret != CO_ERROR_NO) {
return ret;
}
/* Configure communication parameter - COB-ID */
uint32_t COB_ID = 0;
odRet = OD_get_u32(OD_14xx_RPDOCommPar, 1, &COB_ID, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = (((uint32_t)OD_getIndex(OD_14xx_RPDOCommPar)) << 8) | 1U;
}
return CO_ERROR_OD_PARAMETERS;
}
bool_t valid = (COB_ID & 0x80000000U) == 0U;
uint16_t CAN_ID = (uint16_t)(COB_ID & 0x7FFU);
if (valid && ((PDO->mappedObjectsCount == 0U) || (CAN_ID == 0U))) {
valid = false;
if (erroneousMap == 0U) {
erroneousMap = 1;
}
}
if (erroneousMap != 0U) {
CO_errorReport(PDO->em, CO_EM_PDO_WRONG_MAPPING, CO_EMC_PROTOCOL_ERROR,
(erroneousMap != 1U) ? erroneousMap : COB_ID);
}
if (!valid) {
CAN_ID = 0;
}
/* If default CAN-ID is stored in OD (without Node-ID), add Node-ID */
if ((CAN_ID != 0U) && (CAN_ID == (preDefinedCanId & 0xFF80U))) {
CAN_ID = preDefinedCanId;
}
ret = CO_CANrxBufferInit(CANdevRx, CANdevRxIdx, CAN_ID, 0x7FF, false, (void*)RPDO, CO_PDO_receive);
if (ret != CO_ERROR_NO) {
return ret;
}
PDO->valid = valid;
/* Configure communication parameter - transmission type */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
uint8_t transmissionType = (uint8_t)(CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO);
odRet = OD_get_u8(OD_14xx_RPDOCommPar, 2, &transmissionType, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = (((uint32_t)OD_getIndex(OD_14xx_RPDOCommPar)) << 8) | 2U;
}
return CO_ERROR_OD_PARAMETERS;
}
RPDO->SYNC = SYNC;
RPDO->synchronous = transmissionType <= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240;
#endif
/* Configure communication parameter - event-timer (optional) */
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
uint16_t eventTime = 0;
(void)OD_get_u16(OD_14xx_RPDOCommPar, 5, &eventTime, true);
RPDO->timeoutTime_us = (uint32_t)eventTime * 1000U;
#endif
/* Configure OD extensions */
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
PDO->isRPDO = true;
PDO->OD = OD;
PDO->CANdevIdx = CANdevRxIdx;
PDO->preDefinedCanId = preDefinedCanId;
PDO->configuredCanId = CAN_ID;
PDO->OD_communicationParam_ext.object = RPDO;
PDO->OD_communicationParam_ext.read = OD_read_PDO_commParam;
PDO->OD_communicationParam_ext.write = OD_write_14xx;
PDO->OD_mappingParam_extension.object = RPDO;
PDO->OD_mappingParam_extension.read = OD_readOriginal;
PDO->OD_mappingParam_extension.write = OD_write_PDO_mapping;
(void)OD_extension_init(OD_14xx_RPDOCommPar, &PDO->OD_communicationParam_ext);
(void)OD_extension_init(OD_16xx_RPDOMapPar, &PDO->OD_mappingParam_extension);
#endif
return CO_ERROR_NO;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_CALLBACK_PRE) != 0
void
CO_RPDO_initCallbackPre(CO_RPDO_t* RPDO, void* object, void (*pFunctSignalPre)(void* object)) {
if (RPDO != NULL) {
RPDO->functSignalObjectPre = object;
RPDO->pFunctSignalPre = pFunctSignalPre;
}
}
#endif
void
CO_RPDO_process(CO_RPDO_t* RPDO,
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
uint32_t timeDifference_us, uint32_t* timerNext_us,
#endif
bool_t NMTisOperational, bool_t syncWas) {
(void)syncWas;
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
(void)timerNext_us;
#endif
CO_PDO_common_t* PDO = &RPDO->PDO_common;
if (PDO->valid && NMTisOperational
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
&& (syncWas || !RPDO->synchronous)
#endif
) {
/* Verify errors in length of received RPDO CAN message */
if (RPDO->receiveError > CO_RPDO_RX_ACK) {
bool_t setError = RPDO->receiveError != CO_RPDO_RX_OK;
uint16_t code = (RPDO->receiveError == CO_RPDO_RX_SHORT) ? CO_EMC_PDO_LENGTH : CO_EMC_PDO_LENGTH_EXC;
CO_error(PDO->em, setError, CO_EM_RPDO_WRONG_LENGTH, code, PDO->dataLength);
RPDO->receiveError = setError ? CO_RPDO_RX_ACK_ERROR : CO_RPDO_RX_ACK_NO_ERROR;
}
/* Determine, which of the two rx buffers contains relevant message. */
uint8_t bufNo = 0;
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
if (RPDO->synchronous && (RPDO->SYNC != NULL) && !RPDO->SYNC->CANrxToggle) {
bufNo = 1;
}
#endif
/* copy RPDO into OD variables according to mappings */
bool_t rpdoReceived = false;
while (CO_FLAG_READ(RPDO->CANrxNew[bufNo])) {
rpdoReceived = true;
uint8_t* dataRPDO = RPDO->CANrxData[bufNo];
OD_size_t verifyLength = 0U;
/* Clear the flag. If between the copy operation CANrxNew is set
* by receive thread, then copy the latest data again. */
CO_FLAG_CLEAR(RPDO->CANrxNew[bufNo]);
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
/* Copy everything regardless of the actual PDO size to UINT64 to simplify bit shifting */
uint64_t buf64 = 0;
(void)memcpy(&buf64, dataRPDO, CO_PDO_MAX_SIZE);
#ifdef CO_BIG_ENDIAN
/* For big endian we need to swap all bytes */
buf64 = CO_SWAP_64(buf64);
#endif /* CO_BIG_ENDIAN */
#endif /* ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_OD_IO_ACCESS) != 0
for (uint8_t i = 0; i < PDO->mappedObjectsCount; i++) {
OD_IO_t* OD_IO = &PDO->OD_IO[i];
/* get mappedLength from temporary storage */
OD_size_t* dataOffset = &OD_IO->stream.dataOffset;
uint8_t mappedLength = (uint8_t)(*dataOffset);
/* additional safety check. */
verifyLength += (OD_size_t)mappedLength;
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
if (verifyLength > CO_PDO_MAX_SIZE * 8) {
#else
if (verifyLength > CO_PDO_MAX_SIZE) {
#endif
break;
}
/* length of OD variable may be larger than mappedLength */
OD_size_t ODdataLength = OD_IO->stream.dataLength;
if (ODdataLength > CO_PDO_MAX_SIZE) {
ODdataLength = CO_PDO_MAX_SIZE;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
/* Prepare data for writing into OD variable. If mappedLength
* is smaller than ODdataLength, then use auxiliary buffer */
uint8_t* dataOD;
/* Apply the bitmask */
uint64_t shiftedData = buf64 & (UINT64_MAX >> (64 - mappedLength));
/* Shift the original buffer to get ready for the next mapping */
buf64 >>= mappedLength;
#ifdef CO_BIG_ENDIAN
/* Adjust the pointer ignoring unused most significant bytes */
dataOD = (uint8_t *)&shiftedData + 8 - ODdataLength;
#else
/* For little-endian we have a pointer to LSB */
dataOD = (uint8_t *)&shiftedData;
#endif
#else
/* Prepare data for writing into OD variable. If mappedLength
* is smaller than ODdataLength, then use auxiliary buffer */
uint8_t buf[CO_PDO_MAX_SIZE];
uint8_t* dataOD;
if (ODdataLength > mappedLength) {
(void)memset(buf, 0, sizeof(buf));
(void)memcpy(buf, dataRPDO, mappedLength);
dataOD = buf;
} else {
dataOD = dataRPDO;
}
/* swap multibyte data if big-endian */
#ifdef CO_BIG_ENDIAN
if ((OD_IO->stream.attribute & ODA_MB) != 0) {
uint8_t* lo = dataOD;
uint8_t* hi = dataOD + ODdataLength - 1;
while (lo < hi) {
uint8_t swap = *lo;
*lo++ = *hi;
*hi-- = swap;
}
}
#endif
#endif /* (CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING */
/* Set stream.dataOffset to zero, perform OD_IO.write()
* and store mappedLength back to stream.dataOffset */
*dataOffset = 0;
OD_size_t countWritten;
OD_IO->write(&OD_IO->stream, dataOD, ODdataLength, &countWritten);
*dataOffset = mappedLength;
dataRPDO += mappedLength;
}
#else
verifyLength = (OD_size_t)PDO->dataLength;
for (uint8_t i = 0; i < PDO->dataLength; i++) {
*PDO->mapPointer[i] = dataRPDO[i];
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_PDO_OD_IO_ACCESS */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
if ((verifyLength > (CO_PDO_MAX_SIZE * 8)) || (verifyLength != (OD_size_t)PDO->dataLength)) {
#else
if ((verifyLength > CO_PDO_MAX_SIZE) || (verifyLength != (OD_size_t)PDO->dataLength)) {
#endif
/* bug in software, should not happen */
CO_errorReport(PDO->em, CO_EM_GENERIC_SOFTWARE_ERROR, CO_EMC_SOFTWARE_INTERNAL,
(0x100000U | verifyLength));
}
} /* while (CO_FLAG_READ(RPDO->CANrxNew[bufNo])) */
/* verify RPDO timeout */
(void)rpdoReceived;
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
if (RPDO->timeoutTime_us > 0U) {
if (rpdoReceived) {
if (RPDO->timeoutTimer > RPDO->timeoutTime_us) {
CO_errorReset(PDO->em, CO_EM_RPDO_TIME_OUT, RPDO->timeoutTimer);
}
/* enable monitoring */
RPDO->timeoutTimer = 1;
} else if ((RPDO->timeoutTimer > 0U) && (RPDO->timeoutTimer < RPDO->timeoutTime_us)) {
RPDO->timeoutTimer += timeDifference_us;
if (RPDO->timeoutTimer > RPDO->timeoutTime_us) {
CO_errorReport(PDO->em, CO_EM_RPDO_TIME_OUT, CO_EMC_RPDO_TIMEOUT, RPDO->timeoutTimer);
}
} else { /* MISRA C 2004 14.10 */
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_TIMERNEXT) != 0
if ((timerNext_us != NULL) && (RPDO->timeoutTimer < RPDO->timeoutTime_us)) {
uint32_t diff = RPDO->timeoutTime_us - RPDO->timeoutTimer;
if (*timerNext_us > diff) {
*timerNext_us = diff;
}
}
#endif
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_RPDO_TIMERS_ENABLE */
} /* if (PDO->valid && NMTisOperational) */
else {
/* not valid and operational, clear CAN receive flags and timeoutTimer */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
if (!PDO->valid || !NMTisOperational) {
CO_FLAG_CLEAR(RPDO->CANrxNew[0]);
CO_FLAG_CLEAR(RPDO->CANrxNew[1]);
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
RPDO->timeoutTimer = 0;
#endif
}
#else
CO_FLAG_CLEAR(RPDO->CANrxNew[0]);
#if ((CO_CONFIG_PDO)&CO_CONFIG_RPDO_TIMERS_ENABLE) != 0
RPDO->timeoutTimer = 0;
#endif
#endif
}
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_RPDO_ENABLE */
/*******************************************************************************
* T P D O
******************************************************************************/
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_ENABLE) != 0
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
/*
* Custom function for writing OD object "TPDO communication parameter"
*
* For more information see file CO_ODinterface.h, OD_IO_t.
*/
static ODR_t
OD_write_18xx(OD_stream_t* stream, const void* buf, OD_size_t count, OD_size_t* countWritten) {
/* "count" is also verified in *_init() function */
if ((stream == NULL) || (buf == NULL) || (countWritten == NULL) || (count > 4U)) {
return ODR_DEV_INCOMPAT;
}
CO_TPDO_t* TPDO = stream->object;
CO_PDO_common_t* PDO = &TPDO->PDO_common;
uint8_t bufCopy[4];
(void)memcpy((void*)bufCopy, (const void*)buf, count);
switch (stream->subIndex) {
case 1: { /* COB-ID used by PDO */
uint32_t COB_ID = CO_getUint32(buf);
uint16_t CAN_ID = (uint16_t)(COB_ID & 0x7FFU);
bool_t valid = (COB_ID & 0x80000000U) == 0U;
/* bits 11...29 must be zero, PDO must be disabled on change, CAN_ID == 0 is
* not allowed, mapping must be configured before enabling the PDO */
if (((COB_ID & 0x3FFFF800U) != 0U) || (valid && (PDO->valid && (CAN_ID != PDO->configuredCanId)))
|| (valid && CO_IS_RESTRICTED_CAN_ID(CAN_ID)) || (valid && (PDO->mappedObjectsCount == 0U))) {
return ODR_INVALID_VALUE;
}
/* parameter changed? */
if ((valid != PDO->valid) || (CAN_ID != PDO->configuredCanId)) {
/* if default CAN-ID is written, store to OD without Node-ID */
if (CAN_ID == PDO->preDefinedCanId) {
(void)CO_setUint32(bufCopy, COB_ID & 0xFFFFFF80U);
}
if (!valid) {
CAN_ID = 0;
}
CO_CANtx_t* CANtxBuff = CO_CANtxBufferInit(
PDO->CANdev, PDO->CANdevIdx, CAN_ID, false, PDO->dataLength,
TPDO->transmissionType <= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240);
if (CANtxBuff == NULL) {
return ODR_DEV_INCOMPAT;
}
TPDO->CANtxBuff = CANtxBuff;
PDO->valid = valid;
PDO->configuredCanId = CAN_ID;
}
break;
}
case 2: { /* transmission type */ uint8_t transmissionType = CO_getUint8(buf);
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
if ((transmissionType > (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240)
&& (transmissionType < (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO)) {
return ODR_INVALID_VALUE;
}
TPDO->CANtxBuff->syncFlag = transmissionType <= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240;
TPDO->syncCounter = 255;
#else
if (transmissionType < CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO) {
return ODR_INVALID_VALUE;
}
#endif
TPDO->transmissionType = transmissionType;
TPDO->sendRequest = true;
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
TPDO->inhibitTimer = 0;
TPDO->eventTimer = 0;
#endif
break;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
case 3: { /* inhibit time */
if (PDO->valid) {
return ODR_INVALID_VALUE;
}
uint32_t inhibitTime = CO_getUint16(buf);
TPDO->inhibitTime_us = inhibitTime * 100U;
TPDO->inhibitTimer = 0;
break;
}
case 5: { /* event-timer */
uint32_t eventTime = CO_getUint16(buf);
TPDO->eventTime_us = eventTime * 1000U;
TPDO->eventTimer = 0;
break;
}
#endif
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
case 6: { /* SYNC start value */
uint8_t syncStartValue = CO_getUint8(buf);
if (PDO->valid || (syncStartValue > 240U)) {
return ODR_INVALID_VALUE;
}
TPDO->syncStartValue = syncStartValue;
break;
}
#endif
default:
/* MISRA C 2004 15.3 */
break;
}
/* write value to the original location in the Object Dictionary */
return OD_writeOriginal(stream, bufCopy, count, countWritten);
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_FLAG_OD_DYNAMIC */
CO_ReturnError_t
CO_TPDO_init(CO_TPDO_t* TPDO, OD_t* OD, CO_EM_t* em,
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
CO_SYNC_t* SYNC,
#endif
uint16_t preDefinedCanId, OD_entry_t* OD_18xx_TPDOCommPar, OD_entry_t* OD_1Axx_TPDOMapPar,
CO_CANmodule_t* CANdevTx, uint16_t CANdevTxIdx, uint32_t* errInfo) {
CO_PDO_common_t* PDO = &TPDO->PDO_common;
ODR_t odRet;
/* verify arguments */
if ((TPDO == NULL) || (OD == NULL) || (em == NULL) || (OD_18xx_TPDOCommPar == NULL) || (OD_1Axx_TPDOMapPar == NULL)
|| (CANdevTx == NULL)) {
return CO_ERROR_ILLEGAL_ARGUMENT;
}
/* clear object */
(void)memset(TPDO, 0, sizeof(CO_TPDO_t));
/* Configure object variables */
PDO->em = em;
PDO->CANdev = CANdevTx;
/* Configure mapping parameters */
uint32_t erroneousMap = 0;
CO_ReturnError_t ret = PDO_initMapping(PDO, OD, OD_1Axx_TPDOMapPar, false, errInfo, &erroneousMap);
if (ret != CO_ERROR_NO) {
return ret;
}
/* Configure communication parameter - transmission type */
uint8_t transmissionType = (uint8_t)(CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO);
odRet = OD_get_u8(OD_18xx_TPDOCommPar, 2, &transmissionType, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = (((uint32_t)OD_getIndex(OD_18xx_TPDOCommPar)) << 8) | 2U;
}
return CO_ERROR_OD_PARAMETERS;
}
if ((transmissionType < (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO)
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
&& (transmissionType > (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240)
#endif
) {
transmissionType = (uint8_t)(CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO);
}
TPDO->transmissionType = transmissionType;
TPDO->sendRequest = true;
/* Configure communication parameter - COB-ID */
uint32_t COB_ID = 0;
odRet = OD_get_u32(OD_18xx_TPDOCommPar, 1, &COB_ID, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) {
*errInfo = (((uint32_t)OD_getIndex(OD_18xx_TPDOCommPar)) << 8) | 1U;
}
return CO_ERROR_OD_PARAMETERS;
}
bool_t valid = (COB_ID & 0x80000000U) == 0U;
uint16_t CAN_ID = (uint16_t)(COB_ID & 0x7FFU);
if (valid && ((PDO->mappedObjectsCount == 0U) || (CAN_ID == 0U))) {
valid = false;
if (erroneousMap == 0U) {
erroneousMap = 1;
}
}
if (erroneousMap != 0U) {
CO_errorReport(PDO->em, CO_EM_PDO_WRONG_MAPPING, CO_EMC_PROTOCOL_ERROR,
(erroneousMap != 1U) ? erroneousMap : COB_ID);
}
if (!valid) {
CAN_ID = 0;
}
/* If default CAN-ID is stored in OD (without Node-ID), add Node-ID */
if ((CAN_ID != 0U) && (CAN_ID == (preDefinedCanId & 0xFF80U))) {
CAN_ID = preDefinedCanId;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_BITWISE_MAPPING) != 0
TPDO->CANtxBuff = CO_CANtxBufferInit(CANdevTx, CANdevTxIdx, CAN_ID, false, (PDO->dataLength + 7) / 8,
TPDO->transmissionType <= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240);
#else
TPDO->CANtxBuff = CO_CANtxBufferInit(CANdevTx, CANdevTxIdx, CAN_ID, false, PDO->dataLength,
TPDO->transmissionType <= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_240);
#endif
if (TPDO->CANtxBuff == NULL) {
return CO_ERROR_ILLEGAL_ARGUMENT;
}
PDO->valid = valid;
/* Configure communication parameter - inhibit time and event-timer (opt) */
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
uint16_t inhibitTime = 0;
uint16_t eventTime = 0;
(void)OD_get_u16(OD_18xx_TPDOCommPar, 3, &inhibitTime, true);
(void)OD_get_u16(OD_18xx_TPDOCommPar, 5, &eventTime, true);
TPDO->inhibitTime_us = (uint32_t)inhibitTime * 100U;
TPDO->eventTime_us = (uint32_t)eventTime * 1000U;
#endif
/* Configure communication parameter - SYNC start value (optional) */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
TPDO->syncStartValue = 0;
(void)OD_get_u8(OD_18xx_TPDOCommPar, 6, &TPDO->syncStartValue, true);
TPDO->SYNC = SYNC;
TPDO->syncCounter = 255;
#endif
/* Configure OD extensions */
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_OD_DYNAMIC) != 0
PDO->isRPDO = false;
PDO->OD = OD;
PDO->CANdevIdx = CANdevTxIdx;
PDO->preDefinedCanId = preDefinedCanId;
PDO->configuredCanId = CAN_ID;
PDO->OD_communicationParam_ext.object = TPDO;
PDO->OD_communicationParam_ext.read = OD_read_PDO_commParam;
PDO->OD_communicationParam_ext.write = OD_write_18xx;
PDO->OD_mappingParam_extension.object = TPDO;
PDO->OD_mappingParam_extension.read = OD_readOriginal;
PDO->OD_mappingParam_extension.write = OD_write_PDO_mapping;
(void)OD_extension_init(OD_18xx_TPDOCommPar, &PDO->OD_communicationParam_ext);
(void)OD_extension_init(OD_1Axx_TPDOMapPar, &PDO->OD_mappingParam_extension);
#endif
return CO_ERROR_NO;
}
/*
* Send TPDO message.
*
* Function prepares TPDO data from Object Dictionary variables. It is called
* from CO_TPDO_process() according to TPDO communication parameters.
*
* @param TPDO TPDO object.
*
* @return Same as CO_CANsend().
*/
static CO_ReturnError_t
CO_TPDOsend(CO_TPDO_t* TPDO) {
CO_PDO_common_t* PDO = &TPDO->PDO_common;
uint8_t* dataTPDO = &TPDO->CANtxBuff->data[0];
OD_size_t verifyLength = 0U;
#if OD_FLAGS_PDO_SIZE > 0
bool_t eventDriven = ((TPDO->transmissionType == (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_ACYCLIC)
|| (TPDO->transmissionType >= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO));
#endif
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_OD_IO_ACCESS) != 0
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
uint64_t buf64 = 0;
#endif
for (uint8_t i = 0; i < PDO->mappedObjectsCount; i++) {
OD_IO_t* OD_IO = &PDO->OD_IO[i];
OD_stream_t* stream = &OD_IO->stream;
/* get mappedLength from temporary storage */
uint8_t mappedLength = (uint8_t)stream->dataOffset;
/* additional safety check */
verifyLength += (OD_size_t)mappedLength;
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
if (verifyLength > CO_PDO_MAX_SIZE * 8) {
#else
if (verifyLength > CO_PDO_MAX_SIZE) {
#endif
break;
}
/* length of OD variable may be larger than mappedLength */
OD_size_t ODdataLength = stream->dataLength;
if (ODdataLength > CO_PDO_MAX_SIZE) {
ODdataLength = CO_PDO_MAX_SIZE;
}
/* If mappedLength is smaller than ODdataLength, use auxiliary buffer */
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
uint64_t buf = 0;
#else
uint8_t buf[CO_PDO_MAX_SIZE];
uint8_t* dataTPDOCopy;
if (ODdataLength > mappedLength) {
(void)memset(buf, 0, sizeof(buf));
dataTPDOCopy = buf;
} else {
dataTPDOCopy = dataTPDO;
}
#endif /* (CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING) */
/* Set stream.dataOffset to zero, perform OD_IO.read() and store mappedLength back to stream.dataOffset */
stream->dataOffset = 0;
OD_size_t countRd;
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
OD_IO->read(stream, &buf, ODdataLength, &countRd);
#ifdef CO_BIG_ENDIAN
/* Shift right according to the OD data length in bytes to compensate the difference in byte length
* For LE not needed, low bytes from the OD entry are copied to low bytes of u64 */
buf >>= 64 - 8 * ODdataLength;
#endif /* CO_BIG_ENDIAN */
/* Apply the mask and merge with the rest */
buf &= (UINT64_MAX >> (64 - mappedLength));
buf <<= (verifyLength - mappedLength);
buf64 |= buf;
#else
OD_IO->read(stream, dataTPDOCopy, ODdataLength, &countRd);
#endif /* (CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING) */
stream->dataOffset = mappedLength;
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) == 0
/* swap multibyte data if big-endian */
#ifdef CO_BIG_ENDIAN
if ((stream->attribute & ODA_MB) != 0) {
uint8_t* lo = dataTPDOCopy;
uint8_t* hi = dataTPDOCopy + ODdataLength - 1;
while (lo < hi) {
uint8_t swap = *lo;
*lo++ = *hi;
*hi-- = swap;
}
}
#endif
/* If auxiliary buffer, copy it to the TPDO */
if (ODdataLength > mappedLength) {
(void)memcpy(dataTPDO, buf, mappedLength);
}
#endif /* (CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING) == 0*/
/* In event driven TPDO indicate transmission of OD variable */
#if OD_FLAGS_PDO_SIZE > 0
uint8_t* flagPDObyte = PDO->flagPDObyte[i];
if ((flagPDObyte != NULL) && eventDriven) {
*flagPDObyte |= PDO->flagPDObitmask[i];
}
#endif
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) == 0
dataTPDO += mappedLength;
#endif
}
#else
verifyLength = (OD_size_t)PDO->dataLength;
for (uint8_t i = 0; i < PDO->dataLength; i++) {
dataTPDO[i] = *PDO->mapPointer[i];
/* In event driven TPDO indicate transmission of OD variable */
#if OD_FLAGS_PDO_SIZE > 0
uint8_t* flagPDObyte = PDO->flagPDObyte[i];
if (flagPDObyte != NULL && eventDriven) {
*flagPDObyte |= PDO->flagPDObitmask[i];
}
#endif
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_PDO_OD_IO_ACCESS */
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
if ((verifyLength > (CO_PDO_MAX_SIZE * 8)) || (verifyLength != (OD_size_t)PDO->dataLength)) {
#else
if ((verifyLength > CO_PDO_MAX_SIZE) || (verifyLength != (OD_size_t)PDO->dataLength)) {
#endif
/* bug in software, should not happen */
CO_errorReport(PDO->em, CO_EM_GENERIC_SOFTWARE_ERROR, CO_EMC_SOFTWARE_INTERNAL, (0x200000U | verifyLength));
return CO_ERROR_DATA_CORRUPT;
}
#if ((CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING)) != 0
#ifdef CO_BIG_ENDIAN
buf64 = CO_SWAP_64(buf64);
#endif
/* Copy the calculated uint64 data to the PDO buffer */
(void)memcpy(dataTPDO, &buf64, CO_PDO_MAX_SIZE);
#endif /* (CO_CONFIG_PDO) & (CO_CONFIG_PDO_BITWISE_MAPPING) */
TPDO->sendRequest = false;
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
TPDO->eventTimer = TPDO->eventTime_us;
TPDO->inhibitTimer = TPDO->inhibitTime_us;
#endif
return CO_CANsend(PDO->CANdev, TPDO->CANtxBuff);
}
void
CO_TPDO_process(CO_TPDO_t* TPDO,
#if (((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0) || defined CO_DOXYGEN
uint32_t timeDifference_us, uint32_t* timerNext_us,
#endif
bool_t NMTisOperational, bool_t syncWas) {
CO_PDO_common_t* PDO = &TPDO->PDO_common;
#if (((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE)) != 0
(void)timerNext_us;
#endif
(void)syncWas;
if (PDO->valid && NMTisOperational) {
/* check for event timer or application event */
#if (((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0) || (OD_FLAGS_PDO_SIZE > 0)
if ((TPDO->transmissionType == (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_ACYCLIC)
|| (TPDO->transmissionType >= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO)) {
/* event timer */
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
if (TPDO->eventTime_us != 0U) {
TPDO->eventTimer = (TPDO->eventTimer > timeDifference_us) ? (TPDO->eventTimer - timeDifference_us) : 0U;
if (TPDO->eventTimer == 0U) {
TPDO->sendRequest = true;
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_TIMERNEXT) != 0
if ((timerNext_us != NULL) && (*timerNext_us > TPDO->eventTimer)) {
/* Schedule for next event time */
*timerNext_us = TPDO->eventTimer;
}
#endif
}
#endif
/* check for any OD_requestTPDO() */
#if OD_FLAGS_PDO_SIZE > 0
if (!TPDO->sendRequest) {
for (uint8_t i = 0; i < PDO->mappedObjectsCount; i++) {
uint8_t* flagPDObyte = PDO->flagPDObyte[i];
if (flagPDObyte != NULL) {
if ((*flagPDObyte & PDO->flagPDObitmask[i]) == 0U) {
TPDO->sendRequest = true;
break;
}
}
}
}
#endif
}
#endif /* ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE)||(OD_FLAGS_PDO_SIZE>0) */
/* Send PDO by application request or by Event timer */
if (TPDO->transmissionType >= (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO) {
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
TPDO->inhibitTimer = (TPDO->inhibitTimer > timeDifference_us) ? (TPDO->inhibitTimer - timeDifference_us)
: 0U;
/* send TPDO */
if (TPDO->sendRequest && (TPDO->inhibitTimer == 0U)) {
(void)CO_TPDOsend(TPDO);
}
#if ((CO_CONFIG_PDO)&CO_CONFIG_FLAG_TIMERNEXT) != 0
if (TPDO->sendRequest && (timerNext_us != NULL) && (*timerNext_us > TPDO->inhibitTimer)) {
/* Schedule for just beyond inhibit window */
*timerNext_us = TPDO->inhibitTimer;
}
#endif
#else
if (TPDO->sendRequest) {
(void)CO_TPDOsend(TPDO);
}
#endif
} /* if (TPDO->transmissionType >= CO_PDO_TRANSM_TYPE_SYNC_EVENT_LO) */
/* Synchronous PDOs */
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
else if ((TPDO->SYNC != NULL) && syncWas) {
/* send synchronous acyclic TPDO */
if (TPDO->transmissionType == (uint8_t)CO_PDO_TRANSM_TYPE_SYNC_ACYCLIC) {
if (TPDO->sendRequest) {
(void)CO_TPDOsend(TPDO);
}
}
/* send synchronous cyclic TPDO */
else {
/* is the start of synchronous TPDO transmission */
if (TPDO->syncCounter == 255U) {
if ((TPDO->SYNC->counterOverflowValue != 0U) && (TPDO->syncStartValue != 0U)) {
/* syncStartValue is in use */
TPDO->syncCounter = 254;
} else {
/* Send first TPDO somewhere in the middle */
TPDO->syncCounter = (TPDO->transmissionType / 2U) + 1U;
}
}
/* If the syncStartValue is in use, start first TPDO after SYNC with matched syncStartValue. */
if (TPDO->syncCounter == 254U) {
if (TPDO->SYNC->counter == TPDO->syncStartValue) {
TPDO->syncCounter = TPDO->transmissionType;
(void)CO_TPDOsend(TPDO);
}
}
/* Send TPDO after every N-th Sync */
else if (--TPDO->syncCounter == 0U) {
TPDO->syncCounter = TPDO->transmissionType;
(void)CO_TPDOsend(TPDO);
} else { /* MISRA C 2004 14.10 */
}
}
} /* else if (TPDO->SYNC && syncWas) */
else { /* MISRA C 2004 14.10 */
}
#endif
} else {
/* Not operational or valid, reset triggers */
TPDO->sendRequest = true;
#if ((CO_CONFIG_PDO)&CO_CONFIG_TPDO_TIMERS_ENABLE) != 0
TPDO->inhibitTimer = 0;
TPDO->eventTimer = 0;
#endif
#if ((CO_CONFIG_PDO)&CO_CONFIG_PDO_SYNC_ENABLE) != 0
TPDO->syncCounter = 255;
#endif
}
}
#endif /* (CO_CONFIG_PDO) & CO_CONFIG_TPDO_ENABLE */
#endif /* (CO_CONFIG_PDO) & (CO_CONFIG_RPDO_ENABLE | CO_CONFIG_TPDO_ENABLE) */