/************************************************************************************//** * \file Source\xcp.c * \brief XCP 1.0 protocol core source file. * \ingroup Core * \internal *---------------------------------------------------------------------------------------- * C O P Y R I G H T *---------------------------------------------------------------------------------------- * Copyright (c) 2011 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 "boot.h" /* bootloader generic header */ /**************************************************************************************** * Defines ****************************************************************************************/ /** \brief XCP protocol layer version number (16-bit). */ #define XCP_VERSION_PROTOCOL_LAYER (0x0100) /** \brief XCP transport layer version number (16-bit). */ #define XCP_VERSION_TRANSPORT_LAYER (0x0100) /* XCP packet identifiers */ /** \brief Command response packet identifier. */ #define XCP_PID_RES (0xff) /** \brief Error packet identifier. */ #define XCP_PID_ERR (0xfe) /* XCP error codes */ /** \brief Cmd processor synchronization error code. */ #define XCP_ERR_CMD_SYNCH (0x00) /** \brief Command was not executed error code. */ #define XCP_ERR_CMD_BUSY (0x10) /** \brief Unknown or unsupported command error code. */ #define XCP_ERR_CMD_UNKNOWN (0x20) /** \brief Parameter out of range error code. */ #define XCP_ERR_OUT_OF_RANGE (0x22) /** \brief Protected error code. Seed/key required. */ #define XCP_ERR_ACCESS_LOCKED (0x25) /** \brief Cal page not valid error code. */ #define XCP_ERR_PAGE_NOT_VALID (0x26) /** \brief Sequence error code. */ #define XCP_ERR_SEQUENCE (0x29) /** \brief Generic error code. */ #define XCP_ERR_GENERIC (0x31) /* XCP command codes */ /** \brief CONNECT command code. */ #define XCP_CMD_CONNECT (0xff) /** \brief DISCONNECT command code. */ #define XCP_CMD_DISCONNECT (0xfe) /** \brief GET_STATUS command code. */ #define XCP_CMD_GET_STATUS (0xfd) /** \brief SYNCH command code. */ #define XCP_CMD_SYNCH (0xfc) /** \brief GET_ID command code. */ #define XCP_CMD_GET_ID (0xfa) /** \brief GET_SEED command code. */ #define XCP_CMD_GET_SEED (0xf8) /** \brief UNLOCK command code. */ #define XCP_CMD_UNLOCK (0xf7) /** \brief SET_MTA command code. */ #define XCP_CMD_SET_MTA (0xf6) /** \brief UPLOAD command code. */ #define XCP_CMD_UPLOAD (0xf5) /** \brief SHORT_UPLOAD command code. */ #define XCP_CMD_SHORT_UPLOAD (0xf4) /** \brief BUILD_CHECKSUM command code. */ #define XCP_CMD_BUILD_CHECKSUM (0xf3) /** \brief DOWNLOAD command code. */ #define XCP_CMD_DOWNLOAD (0xf0) /** \brief DOWNLOAD_MAX command code. */ #define XCP_CMD_DOWLOAD_MAX (0xee) /** \brief SET_CALPAGE command code. */ #define XCP_CMD_SET_CAL_PAGE (0xeb) /** \brief GET_CALPAGE command code. */ #define XCP_CMD_GET_CAL_PAGE (0xea) /** \brief PROGRAM_START command code. */ #define XCP_CMD_PROGRAM_START (0xd2) /** \brief PROGRAM_CLEAR command code. */ #define XCP_CMD_PROGRAM_CLEAR (0xd1) /** \brief PROGRAM command code. */ #define XCP_CMD_PROGRAM (0xd0) /** \brief PROGRAM_RESET command code. */ #define XCP_CMD_PROGRAM_RESET (0xcf) /** \brief PROGRAM_PREPARE command code. */ #define XCP_CMD_PROGRAM_PREPARE (0xcc) /** \brief PROGRAM_MAX command code. */ #define XCP_CMD_PROGRAM_MAX (0xc9) /**************************************************************************************** * Type definitions ****************************************************************************************/ /** \brief Struture type for grouping XCP internal module information. */ typedef struct { blt_int8u connected; /**< connection established */ blt_int8u protection; /**< protection state */ blt_int8u s_n_k_resource; /**< for seed/key sequence */ blt_int8u ctoData[BOOT_COM_RX_MAX_DATA]; /**< cto packet data buffer */ blt_int8u ctoPending; /**< cto transmission pending flag */ blt_int16s ctoLen; /**< cto current packet length */ blt_int32u mta; /**< memory transfer address */ } tXcpInfo; /**************************************************************************************** * Function prototypes ****************************************************************************************/ /* transport layer specific functions */ static void XcpTransmitPacket(blt_int8u *data, blt_int16s len); /* application specific functions */ static blt_int8u XcpComputeChecksum(blt_int32u address, blt_int32u length, blt_int32u *checksum); #if (XCP_SEED_KEY_PROTECTION_EN == 1) static blt_int8u XcpGetSeed(blt_int8u resource, blt_int8u *seed); static blt_int8u XcpVerifyKey(blt_int8u resource, blt_int8u *key, blt_int8u len); #endif /* general utility functions */ static void XcpProtectResources(void); static void XcpSetCtoError(blt_int8u error); /* XCP command processors */ static void XcpCmdConnect(blt_int8u *data); static void XcpCmdDisconnect(blt_int8u *data); static void XcpCmdGetStatus(blt_int8u *data); static void XcpCmdSynch(blt_int8u *data); static void XcpCmdGetId(blt_int8u *data); static void XcpCmdSetMta(blt_int8u *data); static void XcpCmdUpload(blt_int8u *data); static void XcpCmdShortUpload(blt_int8u *data); static void XcpCmdBuildCheckSum(blt_int8u *data); #if (XCP_SEED_KEY_PROTECTION_EN == 1) static void XcpCmdGetSeed(blt_int8u *data); static void XcpCmdUnlock(blt_int8u *data); #endif #if (XCP_RES_CALIBRATION_EN == 1) static void XcpCmdDownload(blt_int8u *data); static void XcpCmdDownloadMax(blt_int8u *data); #endif #if (XCP_RES_PAGING_EN == 1) static void XcpCmdSetCalPage(blt_int8u *data); static void XcpCmdGetCalPage(blt_int8u *data); #endif #if (XCP_RES_PROGRAMMING_EN == 1) static void XcpCmdProgramMax(blt_int8u *data); static void XcpCmdProgram(blt_int8u *data); static void XcpCmdProgramStart(blt_int8u *data); static void XcpCmdProgramClear(blt_int8u *data); static void XcpCmdProgramReset(blt_int8u *data); static void XcpCmdProgramPrepare(blt_int8u *data); #endif /**************************************************************************************** * Hook functions ****************************************************************************************/ #if (XCP_RES_PAGING_EN == 1) extern blt_int8u XcpCalSetPageHook(blt_int8u segment, blt_int8u page); extern blt_int8u XcpCalGetPageHook(blt_int8u segment); #endif #if (XCP_CONNECT_MODE_HOOK_EN == 1) extern blt_bool XcpConnectModeHook(blt_int8u mode); #endif #if (XCP_SEED_KEY_PROTECTION_EN == 1) extern blt_int8u XcpGetSeedHook(blt_int8u resource, blt_int8u *seed); extern blt_int8u XcpVerifyKeyHook(blt_int8u resource, blt_int8u *key, blt_int8u len); #endif /**************************************************************************************** * External functions ****************************************************************************************/ #if (BOOT_COM_ENABLE == 0) /* in case no internally supported communication interface is used, a custom * communication module can be added. In order to use the XCP protocol in the custom * communication module, this hook function needs to be implemented. In the XCP protocol * is not needed, then simply remove the xcp.c source from the project. */ extern void XcpTransmitPacketHook(blt_int8u *data, blt_int16u len); #endif /**************************************************************************************** * Local constants ****************************************************************************************/ /** \brief String buffer with station id. */ static const blt_int8s xcpStationId[] = XCP_STATION_ID_STRING; /**************************************************************************************** * Local data definitions ****************************************************************************************/ /** \brief Local variable for storing XCP internal module info. */ static tXcpInfo xcpInfo; /************************************************************************************//** ** \brief Initializes the XCP driver. Should be called once upon system startup. ** \return none ** ****************************************************************************************/ void XcpInit(void) { /* reset xcp module info */ xcpInfo.connected = 0; xcpInfo.mta = 0; xcpInfo.ctoPending = 0; xcpInfo.ctoLen = 0; xcpInfo.s_n_k_resource = 0; xcpInfo.protection = 0; } /*** end of XcpInit ***/ /************************************************************************************//** ** \brief Obtains information about the XCP connection state. ** \return BLT_TRUE is an XCP connection is established, BLT_FALSE otherwise. ** ****************************************************************************************/ blt_bool XcpIsConnected(void) { if (xcpInfo.connected == 0) { return BLT_FALSE; } return BLT_TRUE; } /*** end of XcpIsConnected ***/ /************************************************************************************//** ** \brief Informs the core that a pending packet transmission was completed by ** the transport layer. ** \return none ** ****************************************************************************************/ void XcpPacketTransmitted(void) { /* reset packet transmission pending flag */ xcpInfo.ctoPending = 0; } /*** end of XcpPacketTransmitted ***/ /************************************************************************************//** ** \brief Informs the core that a new packet was received by the transport layer. ** \param data Pointer to byte buffer with packet data. ** \return none ** ****************************************************************************************/ void XcpPacketReceived(blt_int8u *data) { /* was this a connect command? */ if (data[0] == XCP_CMD_CONNECT) { /* process the connect command */ XcpCmdConnect(data); } /* only continue if connected */ else if (xcpInfo.connected == 1) { switch (data[0]) { case XCP_CMD_UPLOAD: XcpCmdUpload(data); break; case XCP_CMD_SHORT_UPLOAD: XcpCmdShortUpload(data); break; case XCP_CMD_SET_MTA: XcpCmdSetMta(data); break; case XCP_CMD_BUILD_CHECKSUM: XcpCmdBuildCheckSum(data); break; case XCP_CMD_GET_ID: XcpCmdGetId(data); break; case XCP_CMD_SYNCH: XcpCmdSynch(data); break; case XCP_CMD_GET_STATUS: XcpCmdGetStatus(data); break; case XCP_CMD_DISCONNECT: XcpCmdDisconnect(data); break; #if (XCP_RES_CALIBRATION_EN == 1) case XCP_CMD_DOWNLOAD: XcpCmdDownload(data); break; case XCP_CMD_DOWLOAD_MAX: XcpCmdDownloadMax(data); break; #endif #if (XCP_RES_PROGRAMMING_EN == 1) case XCP_CMD_PROGRAM_MAX: XcpCmdProgramMax(data); break; case XCP_CMD_PROGRAM: XcpCmdProgram(data); break; case XCP_CMD_PROGRAM_START: XcpCmdProgramStart(data); break; case XCP_CMD_PROGRAM_CLEAR: XcpCmdProgramClear(data); break; case XCP_CMD_PROGRAM_RESET: XcpCmdProgramReset(data); break; case XCP_CMD_PROGRAM_PREPARE: XcpCmdProgramPrepare(data); break; #endif #if (XCP_SEED_KEY_PROTECTION_EN == 1) case XCP_CMD_GET_SEED: XcpCmdGetSeed(data); break; case XCP_CMD_UNLOCK: XcpCmdUnlock(data); break; #endif #if (XCP_RES_PAGING_EN == 1) case XCP_CMD_SET_CAL_PAGE: XcpCmdSetCalPage(data); break; case XCP_CMD_GET_CAL_PAGE: XcpCmdGetCalPage(data); break; #endif default: XcpSetCtoError(XCP_ERR_CMD_UNKNOWN); break; } } else { /* return to make sure response packet is not send because we are not connected */ return; } /* make sure the previous command was completed */ if (xcpInfo.ctoPending == 1) { /* command overrun occurred */ XcpSetCtoError(XCP_ERR_CMD_BUSY); } /* send the response if it contains something */ if (xcpInfo.ctoLen > 0) { /* set cto packet transmission pending flag */ xcpInfo.ctoPending = 1; /* transmit the cto response packet */ XcpTransmitPacket(xcpInfo.ctoData, xcpInfo.ctoLen); } } /*** end of XcpPacketReceived ***/ /************************************************************************************//** ** \brief Transmits the packet using the xcp transport layer. ** \param data Pointer to the byte buffer with packet data. ** \param len Number of data bytes that need to be transmitted. ** \return none ** ****************************************************************************************/ static void XcpTransmitPacket(blt_int8u *data, blt_int16s len) { /* submit packet to the communication interface for transmission */ #if (BOOT_COM_ENABLE == 0) XcpTransmitPacketHook(data, len); #else ComTransmitPacket(data, len); #endif } /*** end of XcpTransmitPacket ***/ /************************************************************************************//** ** \brief Called by the BUILD_CHECKSUM command to perform a checksum calculation ** over the specified memory region. ** \param address The start address of the memory region. ** \param length Length of the memory region in bytes. ** \param checksum Pointer to where the calculated checksum is to be stored. ** \return Checksum type that was used during the checksum calculation. ** ****************************************************************************************/ static blt_int8u XcpComputeChecksum(blt_int32u address, blt_int32u length, blt_int32u *checksum) { blt_int8u cs = 0; /* this example computes the checksum using the add byte to byte algorithm */ while (length-- > 0) { cs += *((blt_int8u *)(blt_addr)address); address++; } *checksum = cs; return XCP_CS_ADD11; } /*** end of XcpComputeChecksum ***/ #if (XCP_SEED_KEY_PROTECTION_EN == 1) /************************************************************************************//** ** \brief Provides a seed to the XCP master that will be used for the key ** generation when the master attempts to unlock the specified resource. ** Called by the GET_SEED command. ** \param resource Resource that the seed if requested for (XCP_RES_XXX). ** \param seed Pointer to byte buffer wher the seed will be stored. ** \return Length of the seed in bytes. ** ****************************************************************************************/ static blt_int8u XcpGetSeed(blt_int8u resource, blt_int8u *seed) { /* pass request on to the application through a hook function */ return XcpGetSeedHook(resource, seed); } /*** end of XcpGetSeed ***/ /************************************************************************************//** ** \brief Called by the UNLOCK command and checks if the key to unlock the ** specified resource was correct. If so, then the resource protection ** will be removed. ** \param resource resource to unlock (XCP_RES_XXX). ** \param key pointer to the byte buffer holding the key. ** \param len length of the key in bytes. ** \return 1 if the key was correct, 0 otherwise. ** ****************************************************************************************/ static blt_int8u XcpVerifyKey(blt_int8u resource, blt_int8u *key, blt_int8u len) { /* pass request on to the application through a hook function */ return XcpVerifyKeyHook(resource, key, len); } /*** end of XcpVerifyKey ***/ #endif /* XCP_SEED_KEY_PROTECTION_EN == 1 */ /************************************************************************************//** ** \brief Utility function to protects all the available resources. ** \return none ** ****************************************************************************************/ static void XcpProtectResources(void) { xcpInfo.protection = 0; #if (XCP_SEED_KEY_PROTECTION_EN == 1) #if (XCP_RES_CALIBRATION_EN == 1) xcpInfo.protection |= XCP_RES_CALPAG; #endif #if (XCP_RES_PAGING_EN == 1) xcpInfo.protection |= XCP_RES_CALPAG; #endif #if (XCP_RES_PROGRAMMING_EN == 1) xcpInfo.protection |= XCP_RES_PGM; #endif #if (XCP_RES_DATA_ACQUISITION_EN == 1) xcpInfo.protection |= XCP_RES_DAQ; #endif #if (XCP_RES_DATA_STIMULATION_EN == 1) xcpInfo.protection |= XCP_RES_STIM; #endif #endif /* XCP_SEED_KEY_PROTECTION_EN == 1 */ } /*** end of XcpProtectResources ***/ /************************************************************************************//** ** \brief Prepares the cto packet data for the specified error. ** \param error XCP error code (XCP_ERR_XXX). ** \return none ** ****************************************************************************************/ static void XcpSetCtoError(blt_int8u error) { /* prepare the error packet */ xcpInfo.ctoData[0] = XCP_PID_ERR; xcpInfo.ctoData[1] = error; xcpInfo.ctoLen = 2; } /*** end of XcpSetCtoError ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the CONNECT command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdConnect(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; #if (BOOT_FILE_SYS_ENABLE > 0) /* reject the connection if the file module is not idle. this means that a firmware * update from the locally attached storage is in progress */ if (FileIsIdle() == BLT_FALSE) { /* command not processed because we are busy */ XcpSetCtoError(XCP_ERR_CMD_BUSY); return; } #endif #if (XCP_CONNECT_MODE_HOOK_EN == 1) /* pass on the mode to a application specific hook function. This function can determine * is the mode is supported or not. A return value of BLT_FALSE causes the CONNECT command * to be ignored. Note that this mode could potentially be used to specify a node ID in a * multi XCP slave system. */ if (XcpConnectModeHook(data[1]) == BLT_FALSE) { /* set the response length to 0 to suppress it */ xcpInfo.ctoLen = 0; return; } #endif /* enable resource protection */ XcpProtectResources(); /* indicate that the connection is established */ xcpInfo.connected = 1; /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* report available resources */ xcpInfo.ctoData[1] = 0; #if (XCP_RES_CALIBRATION_EN == 1) xcpInfo.ctoData[1] |= XCP_RES_CALPAG; #endif #if (XCP_RES_PAGING_EN == 1) xcpInfo.ctoData[1] |= XCP_RES_CALPAG; #endif #if (XCP_RES_PROGRAMMING_EN == 1) xcpInfo.ctoData[1] |= XCP_RES_PGM; #endif #if (XCP_RES_DATA_ACQUISITION_EN == 1) xcpInfo.ctoData[1] |= XCP_RES_DAQ; #endif #if (XCP_RES_DATA_STIMULATION_EN == 1) xcpInfo.ctoData[1] |= XCP_RES_STIM; #endif /* report communication mode info. only byte granularity is supported */ xcpInfo.ctoData[2] = 0; /* configure for motorola or intel byte ordering */ xcpInfo.ctoData[2] |= XCP_MOTOROLA_FORMAT; /* report max cto data length */ xcpInfo.ctoData[3] = (blt_int8u)XCP_CTO_PACKET_LEN; /* report max dto data length */ #if (XCP_MOTOROLA_FORMAT == 0) xcpInfo.ctoData[4] = (blt_int8u)XCP_DTO_PACKET_LEN; xcpInfo.ctoData[5] = (blt_int8u)(XCP_DTO_PACKET_LEN >> 8); #else xcpInfo.ctoData[4] = (blt_int8u)(XCP_DTO_PACKET_LEN >> 8); xcpInfo.ctoData[5] = (blt_int8u)XCP_DTO_PACKET_LEN; #endif /* report msb of protocol layer version number */ xcpInfo.ctoData[6] = XCP_VERSION_PROTOCOL_LAYER >> 8; /* report msb of transport layer version number */ xcpInfo.ctoData[7] = XCP_VERSION_TRANSPORT_LAYER >> 8; /* set packet length */ xcpInfo.ctoLen = 8; } /*** end of XcpCmdConnect ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the DISCONNECT command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdDisconnect(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; /* indicate that the xcp connection is disconnected */ xcpInfo.connected = 0; /* enable resource protection */ XcpProtectResources(); /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdDisconnect ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the GET_STATUS command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdGetStatus(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* report session status */ xcpInfo.ctoData[1] = 0; /* report current resource protection status */ xcpInfo.ctoData[2] = xcpInfo.protection; /* reset reserved and session configuration id values */ xcpInfo.ctoData[3] = 0; xcpInfo.ctoData[4] = 0; xcpInfo.ctoData[5] = 0; /* set packet length */ xcpInfo.ctoLen = 6; } /*** end of XcpCmdGetStatus ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the SYNCH command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdSynch(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; /* synch requires a negative response */ XcpSetCtoError(XCP_ERR_CMD_SYNCH); } /*** end of XcpCmdSynch ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the GET_ID command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdGetId(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* point mta to start of station id string */ xcpInfo.mta = (blt_int32u)&xcpStationId[0]; /* set station id mode to 0 */ xcpInfo.ctoData[1] = 0; /* reset reserved values */ xcpInfo.ctoData[2] = 0; xcpInfo.ctoData[3] = 0; /* store station id length (excl. null termination) for response packet */ *(blt_int32u *)&xcpInfo.ctoData[4] = (sizeof(xcpStationId)/sizeof(xcpStationId[0])) - 1; /* set packet length */ xcpInfo.ctoLen = 8; } /*** end of XcpCmdGetId ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the SET_MTA command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdSetMta(blt_int8u *data) { /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* update mta. current implementation ignores address extension */ xcpInfo.mta = *(blt_int32u *)&data[4]; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdSetMta ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the UPLOAD command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdUpload(blt_int8u *data) { /* validate length of upload request */ if (data[1] > (XCP_CTO_PACKET_LEN-1)) { /* requested data length is too long */ XcpSetCtoError(XCP_ERR_OUT_OF_RANGE); return; } /* copy the data from memory to the data packet */ CpuMemCopy(((blt_addr)(blt_int32u)&xcpInfo.ctoData[1]),(blt_addr)xcpInfo.mta, data[1]); /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* post increment the mta */ xcpInfo.mta += data[1]; /* set packet length */ xcpInfo.ctoLen = data[1]+1; } /*** end of XcpCmdUpload ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the SHORT_UPLOAD command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdShortUpload(blt_int8u *data) { /* validate length of upload request */ if (data[1] > (XCP_CTO_PACKET_LEN-1)) { /* requested data length is too long */ XcpSetCtoError(XCP_ERR_OUT_OF_RANGE); return; } /* update mta. current implementation ignores address extension */ xcpInfo.mta = *(blt_int32u *)&data[4]; /* copy the data from memory to the data packet */ CpuMemCopy((blt_addr)((blt_int32u)&xcpInfo.ctoData[1]),(blt_addr)xcpInfo.mta, data[1]); /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* post increment the mta */ xcpInfo.mta += data[1]; /* set packet length */ xcpInfo.ctoLen = data[1]+1; } /*** end of XcpCmdShortUpload ***/ #if (XCP_RES_CALIBRATION_EN == 1) /************************************************************************************//** ** \brief XCP command processor function which handles the DOWNLOAD command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdDownload(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if CAL_PAG resource is unlocked */ if ((xcpInfo.protection & XCP_RES_CALPAG) != 0) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* validate length of download request */ if (data[1] > (XCP_CTO_PACKET_LEN-2)) { /* requested data length is too long */ XcpSetCtoError(XCP_ERR_OUT_OF_RANGE); return; } /* copy the data from the data packet to memory */ CpuMemCopy((blt_addr)xcpInfo.mta, (blt_addr)((blt_int32u)&data[2]), data[1]); /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* post increment the mta */ xcpInfo.mta += data[1]; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdDownload ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the DOWNLOAD_MAX command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdDownloadMax(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if CAL_PAG resource is unlocked */ if ((xcpInfo.protection & XCP_RES_CALPAG) != 0) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* copy the data from the data packet to memory */ CpuMemCopy((blt_addr)xcpInfo.mta, (blt_addr)((blt_int32u)&data[1]), \ XCP_CTO_PACKET_LEN-1); /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* post increment the mta */ xcpInfo.mta += XCP_CTO_PACKET_LEN-1; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdDownloadMax ***/ #endif /* XCP_RES_CALIBRATION_EN == 1 */ /************************************************************************************//** ** \brief XCP command processor function which handles the BUILD_CHECKSUM command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdBuildCheckSum(blt_int8u *data) { /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* obtain checksum and checksum type */ xcpInfo.ctoData[1] = XcpComputeChecksum(xcpInfo.mta, *(blt_int32u *)&data[4], (blt_int32u *)&xcpInfo.ctoData[4]); /* initialize reserved parameters */ xcpInfo.ctoData[2] = 0; xcpInfo.ctoData[3] = 0; /* set packet length */ xcpInfo.ctoLen = 8; } /*** end of XcpCmdBuildCheckSum ***/ #if (XCP_SEED_KEY_PROTECTION_EN == 1) /************************************************************************************//** ** \brief XCP command processor function which handles the GET_SEED command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdGetSeed(blt_int8u *data) { blt_int8u resourceOK; /* init resource check variable as if an illegal resource is requested */ resourceOK = 0; /* check if calibration/paging resource is requested for seed/key and make * sure this is the only requested resource */ if (((data[2] & XCP_RES_CALPAG) > 0) && ((data[2] & ~XCP_RES_CALPAG) == 0)) { resourceOK = 1; } /* check if programming resource is requested for seed/key and make * sure this is the only requested resource */ if (((data[2] & XCP_RES_PGM) > 0) && ((data[2] & ~XCP_RES_PGM) == 0)) { resourceOK = 1; } /* check if data acquisition resource is requested for seed/key and make * sure this is the only requested resource */ if (((data[2] & XCP_RES_DAQ) > 0) && ((data[2] & ~XCP_RES_DAQ) == 0)) { resourceOK = 1; } /* check if data stimulation resource is requested for seed/key and make * sure this is the only requested resource */ if (((data[2] & XCP_RES_STIM) > 0) && ((data[2] & ~XCP_RES_STIM) == 0)) { resourceOK = 1; } /* now process the resource validation */ if (resourceOK == 0) { XcpSetCtoError(XCP_ERR_OUT_OF_RANGE); return; } /* store resource for which the seed/key sequence is started */ xcpInfo.s_n_k_resource = data[2]; /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* request the seed from the application */ xcpInfo.ctoData[1] = XcpGetSeed(xcpInfo.s_n_k_resource, &xcpInfo.ctoData[2]); /* seed cannot be longer than XCP_CTO_PACKET_LEN-2 */ if (xcpInfo.ctoData[1] > (XCP_CTO_PACKET_LEN-2)) { /* seed length length is too long */ XcpSetCtoError(XCP_ERR_OUT_OF_RANGE); return; } /* set packet length */ xcpInfo.ctoLen = xcpInfo.ctoData[1] + 2; } /*** end of XcpCmdGetSeed ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the UNLOCK command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdUnlock(blt_int8u *data) { /* key cannot be longer than XCP_CTO_PACKET_LEN-2 */ if (data[1] > (XCP_CTO_PACKET_LEN-2)) { /* key is too long incorrect */ XcpSetCtoError(XCP_ERR_SEQUENCE); return; } /* verify the key */ if (XcpVerifyKey(xcpInfo.s_n_k_resource, &data[2], data[1]) == 0) { /* invalid key so inform the master and do a disconnect */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); /* indicate that the xcp connection is disconnected */ xcpInfo.connected = 0; /* enable resource protection */ XcpProtectResources(); return; } /* key correct so unlock the resource */ xcpInfo.protection &= ~xcpInfo.s_n_k_resource; /* reset seed/key resource variable for possible next unlock */ xcpInfo.s_n_k_resource = 0; /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* report the current resource protection */ xcpInfo.ctoData[1] = xcpInfo.protection; /* set packet length */ xcpInfo.ctoLen = 2; } /*** end of XcpCmdUnlock ***/ #endif /* XCP_SEED_KEY_PROTECTION_EN == 1 */ #if (XCP_RES_PAGING_EN == 1) /************************************************************************************//** ** \brief XCP command processor function which handles the SET_CAL_PAGE command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdSetCalPage(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if CAL_PAG resource is unlocked */ if ((xcpInfo.protection & XCP_RES_CALPAG) == XCP_RES_CALPAG) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* select the page. note that the mode parameter is ignored */ if (XcpCalSetPageHook(data[2], data[3]) == 0) { /* calibration page could not be selected */ XcpSetCtoError(XCP_ERR_PAGE_NOT_VALID); return; } /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdSetCalPage ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the GET_CAL_PAGE command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdGetCalPage(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if CAL_PAG resource is unlocked */ if ((xcpInfo.protection & XCP_RES_CALPAG) == XCP_RES_CALPAG) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* initialize reserved parameters */ xcpInfo.ctoData[1] = 0; xcpInfo.ctoData[2] = 0; /* store the calibration page */ xcpInfo.ctoData[3] = XcpCalGetPageHook(data[2]); /* set packet length */ xcpInfo.ctoLen = 4; } /*** end of XcpCmdGetCalPage ***/ #endif /* XCP_RES_PAGING_EN == 1 */ #if (XCP_RES_PROGRAMMING_EN == 1) /************************************************************************************//** ** \brief XCP command processor function which handles the PROGRAM_START command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdProgramStart(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if PGM resource is unlocked */ if ((xcpInfo.protection & XCP_RES_PGM) == XCP_RES_PGM) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* initialize reserved parameter */ xcpInfo.ctoData[1] = 0; /* no special communication mode supported during programming */ xcpInfo.ctoData[2] = 0; /* cto packet length stays the same during programming */ xcpInfo.ctoData[3] = (blt_int8u)XCP_CTO_PACKET_LEN; /* no block size, st-min time, or queue size supported */ xcpInfo.ctoData[4] = 0; xcpInfo.ctoData[5] = 0; xcpInfo.ctoData[6] = 0; /* set packet length */ xcpInfo.ctoLen = 7; } /*** end of XcpCmdProgramStart ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the PROGRAM_MAX command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdProgramMax(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if PGM resource is unlocked */ if ((xcpInfo.protection & XCP_RES_PGM) == XCP_RES_PGM) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* program the data */ if (NvmWrite((blt_addr)xcpInfo.mta, XCP_CTO_PACKET_LEN-1, &data[1]) == 0) { /* error occurred during programming */ XcpSetCtoError(XCP_ERR_GENERIC); return; } /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* post increment the mta */ xcpInfo.mta += XCP_CTO_PACKET_LEN-1; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdProgramMax ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the PROGRAM command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdProgram(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if PGM resource is unlocked */ if ((xcpInfo.protection & XCP_RES_PGM) == XCP_RES_PGM) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* validate length of download request */ if (data[1] > (XCP_CTO_PACKET_LEN-2)) { /* requested data length is too long */ XcpSetCtoError(XCP_ERR_OUT_OF_RANGE); return; } /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* set packet length */ xcpInfo.ctoLen = 1; /* end of programming sequence (datasize is 0)? */ if (data[1] == 0) { /* call erase/programming cleanup routine */ if (NvmDone() == BLT_FALSE) { /* error occurred while finishing up programming */ XcpSetCtoError(XCP_ERR_GENERIC); } return; } /* program the data */ if (NvmWrite((blt_addr)xcpInfo.mta, data[1], &data[2]) == 0) { /* error occurred during programming */ XcpSetCtoError(XCP_ERR_GENERIC); return; } /* post increment the mta */ xcpInfo.mta += data[1]; } /*** end of XcpCmdProgram ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the PROGRAM_CLEAR command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdProgramClear(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if PGM resource is unlocked */ if ((xcpInfo.protection & XCP_RES_PGM) == XCP_RES_PGM) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* erase the memory */ if (NvmErase((blt_addr)xcpInfo.mta, *(blt_int32u *)&data[4]) == 0) { /* error occurred during erasure */ XcpSetCtoError(XCP_ERR_GENERIC); return; } /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdProgramClear ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the PROGRAM_RESET command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdProgramReset(blt_int8u *data) { /* suppress compiler warning for unused parameter */ data = data; #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if PGM resource is unlocked */ if ((xcpInfo.protection & XCP_RES_PGM) == XCP_RES_PGM) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* reset the ecu after programming is done. so basically, just start the newly programmed * firmware. it is okay if the code does not return here. if CpuReset() is used here, then * the bootloader is first activated again, including the backdoor timer which is not * desired. */ CpuStartUserProgram(); /* set packet id to command response packet */ xcpInfo.ctoData[0] = XCP_PID_RES; /* set packet length */ xcpInfo.ctoLen = 1; } /*** end of XcpCmdProgramReset ***/ /************************************************************************************//** ** \brief XCP command processor function which handles the PROGRAM_PREPARE command as ** defined by the protocol. ** \param data Pointer to a byte buffer with the packet data. ** \return none ** ****************************************************************************************/ static void XcpCmdProgramPrepare(blt_int8u *data) { #if (XCP_SEED_KEY_PROTECTION_EN == 1) /* check if PGM resource is unlocked */ if ((xcpInfo.protection & XCP_RES_PGM) == XCP_RES_PGM) { /* resource is locked. use seed/key sequence to unlock */ XcpSetCtoError(XCP_ERR_ACCESS_LOCKED); return; } #endif /* programming with kernel currently not needed and therefore not supported */ XcpSetCtoError(XCP_ERR_GENERIC); return; } /*** end of XcpCmdProgramPrepare ***/ #endif /* XCP_RES_PROGRAMMING_EN == 1 */ /******************************** end of xcp.c *****************************************/