MotorControlModuleSDFM_TMS3.../Projects/mem_test/Freemaster/PE_freemaster_appcmd.c

454 lines
14 KiB
C
Raw Normal View History

/******************************************************************************
*
* Freescale Semiconductor Inc.
* (c) Copyright 2004-2007 Freescale Semiconductor, Inc.
* (c) Copyright 2001-2004 Motorola, Inc.
* ALL RIGHTS RESERVED.
*
****************************************************************************//*!
*
* @file PE_freemaster_appcmd.c
*
* @brief FreeMASTER Application Commands implementation
*
* @version 1.0.9.0
*
* @date Oct-19-2007
*
*******************************************************************************/
#include "PE_freemaster.h"
#include "PE_freemaster_private.h"
#include "PE_freemaster_protocol.h"
#if FMSTR_USE_APPCMD
/***********************************
* local variables
***********************************/
static FMSTR_APPCMD_CODE pcm_nAppCmd; /* app.cmd code (to application) */
static FMSTR_APPCMD_DATA pcm_pAppCmdBuff[FMSTR_APPCMD_BUFF_SIZE]; /* app.cmd data buffer */
static FMSTR_SIZE pcm_nAppCmdLen; /* app.cmd data length */
static FMSTR_APPCMD_RESULT pcm_nAppCmdResult; /* app.cmd result code (from application) */
static FMSTR_SIZE8 pcm_nAppCmdResultDataLen;
#if FMSTR_MAX_APPCMD_CALLS > 0
static FMSTR_APPCMD_CODE pcm_pAppCmdCallId[FMSTR_MAX_APPCMD_CALLS]; /* registerred callback commands */
static FMSTR_PAPPCMDFUNC pcm_pAppCmdCallFunc[FMSTR_MAX_APPCMD_CALLS]; /* registerred callback handlers */
#endif
/***********************************
* local functions
***********************************/
static FMSTR_INDEX FMSTR_FindAppCmdCallIndex(FMSTR_APPCMD_CODE nAppcmdCode);
/**************************************************************************//*!
*
* @brief Initialize app.cmds interface
*
******************************************************************************/
void FMSTR_InitAppCmds(void)
{
#if FMSTR_MAX_APPCMD_CALLS
FMSTR_INDEX i;
for(i=0; i<FMSTR_MAX_APPCMD_CALLS; i++)
{
pcm_pAppCmdCallId[i] = FMSTR_APPCMDRESULT_NOCMD;
pcm_pAppCmdCallFunc[i] = NULL;
}
#endif
pcm_nAppCmd = FMSTR_APPCMDRESULT_NOCMD;
pcm_nAppCmdResult = FMSTR_APPCMDRESULT_NOCMD;
}
/**************************************************************************//*!
*
* @brief Handling SANDAPPCMD command
*
* @param pMessageIO - original command (in) and response buffer (out)
*
* @return As all command handlers, the return value should be the buffer
* pointer where the response output finished (except checksum)
*
******************************************************************************/
FMSTR_BPTR FMSTR_StoreAppCmd(FMSTR_BPTR pMessageIO)
{
FMSTR_BPTR pResponse = pMessageIO;
FMSTR_U8 nArgsLen;
FMSTR_U8 nCode;
/* the previous command not yet processed */
if(pcm_nAppCmd != FMSTR_APPCMDRESULT_NOCMD)
{
return FMSTR_ConstToBuffer8(pResponse, FMSTR_STC_SERVBUSY);
}
pMessageIO = FMSTR_SkipInBuffer(pMessageIO, 1U);
pMessageIO = FMSTR_ValueFromBuffer8(&nArgsLen, pMessageIO);
pMessageIO = FMSTR_ValueFromBuffer8(&nCode, pMessageIO);
/* args len is datalen minus one */
nArgsLen--;
/* does the application command fit to buffer ? */
if (nArgsLen > (FMSTR_SIZE8) FMSTR_APPCMD_BUFF_SIZE)
{
return FMSTR_ConstToBuffer8(pResponse, FMSTR_STC_INVBUFF);
}
/* store command data into dedicated buffer */
pcm_nAppCmd = nCode;
pcm_nAppCmdLen = nArgsLen;
/* data copy */
if(nArgsLen)
{
FMSTR_ADDR appCmdBuffAddr;
FMSTR_ARR2ADDR(appCmdBuffAddr, pcm_pAppCmdBuff);
/*lint -e{534} ignoring return value */
FMSTR_CopyFromBuffer(appCmdBuffAddr, pMessageIO, (FMSTR_SIZE8) nArgsLen);
}
/* mark command as "running" (without any response data) */
pcm_nAppCmdResult = FMSTR_APPCMDRESULT_RUNNING;
pcm_nAppCmdResultDataLen = 0U;
return FMSTR_ConstToBuffer8(pResponse, FMSTR_STS_OK);
}
/**************************************************************************//*!
*
* @brief Handling GETAPPCMDSTS command
*
* @param pMessageIO - original command (in) and response buffer (out)
*
* @return As all command handlers, the return value should be the buffer
* pointer where the response output finished (except checksum)
*
* @note The callback-registerred commands are processed at the moment the PC
* tries to get the result for the first time. At this moment, we are
* sure the PC already got the command delivery packet acknowledged.
*
******************************************************************************/
FMSTR_BPTR FMSTR_GetAppCmdStatus(FMSTR_BPTR pMessageIO)
{
#if FMSTR_MAX_APPCMD_CALLS
FMSTR_PAPPCMDFUNC pFunc = NULL;
FMSTR_INDEX nIndex;
/* time to execute the command's callback */
if((nIndex = FMSTR_FindAppCmdCallIndex(pcm_nAppCmd)) >= 0)
{
pFunc = pcm_pAppCmdCallFunc[nIndex];
}
/* valid callback function found? */
if(pFunc)
{
/* do execute callback, return value is app.cmd result code */
pcm_nAppCmdResult = pFunc(pcm_nAppCmd, pcm_pAppCmdBuff, pcm_nAppCmdLen);
/* nothing more to do with this command (i.e. command acknowledged) */
pcm_nAppCmd = FMSTR_APPCMDRESULT_NOCMD;
}
#endif
pMessageIO = FMSTR_ConstToBuffer8(pMessageIO, FMSTR_STS_OK);
return FMSTR_ValueToBuffer8(pMessageIO, (FMSTR_U8) pcm_nAppCmdResult);
}
/**************************************************************************//*!
*
* @brief Handling GETAPPCMDDATA command
*
* @param pMessageIO - original command (in) and response buffer (out)
*
* @return As all command handlers, the return value should be the buffer
* pointer where the response output finished (except checksum)
*
******************************************************************************/
FMSTR_BPTR FMSTR_GetAppCmdRespData(FMSTR_BPTR pMessageIO)
{
FMSTR_BPTR pResponse = pMessageIO;
FMSTR_U8 nDataLen;
FMSTR_U8 nDataOffset;
/* the previous command not yet processed */
if(pcm_nAppCmd != FMSTR_APPCMDRESULT_NOCMD)
{
return FMSTR_ConstToBuffer8(pResponse, FMSTR_STC_SERVBUSY);
}
pMessageIO = FMSTR_SkipInBuffer(pMessageIO, 1U);
pMessageIO = FMSTR_ValueFromBuffer8(&nDataLen, pMessageIO);
pMessageIO = FMSTR_ValueFromBuffer8(&nDataOffset, pMessageIO);
/* the response would not fit into comm buffer */
if(nDataLen > (FMSTR_U16)FMSTR_COMM_BUFFER_SIZE)
{
return FMSTR_ConstToBuffer8(pResponse, FMSTR_STC_RSPBUFFOVF);
}
/* the data would be fetched outside the app.cmd response data */
if((((FMSTR_U16)nDataOffset) + nDataLen) > (FMSTR_SIZE8)pcm_nAppCmdResultDataLen)
{
return FMSTR_ConstToBuffer8(pResponse, FMSTR_STC_INVSIZE);
}
pResponse = FMSTR_ConstToBuffer8(pResponse, FMSTR_STS_OK);
/* copy to buffer */
{
FMSTR_ADDR appCmdBuffAddr;
FMSTR_ARR2ADDR(appCmdBuffAddr, pcm_pAppCmdBuff);
pResponse = FMSTR_CopyToBuffer(pResponse, appCmdBuffAddr, (FMSTR_SIZE8)nDataLen);
}
return pResponse;
}
/**************************************************************************//*!
*
* @brief Find index of registerred app.cmd callback
*
* @param nAppcmdCode - App. command ID
*
* @return Index of function pointer in our local tables
*
******************************************************************************/
static FMSTR_INDEX FMSTR_FindAppCmdCallIndex(FMSTR_APPCMD_CODE nAppcmdCode)
{
#if FMSTR_MAX_APPCMD_CALLS > 0
FMSTR_INDEX i;
for(i=0; i<FMSTR_MAX_APPCMD_CALLS; i++)
{
if(pcm_pAppCmdCallId[i] == nAppcmdCode)
{
return i;
}
}
#else
/*lint -esym(528, FMSTR_FindAppCmdCallIndex) this function is
not referenced when APPCMD_CALLS are not used */
FMSTR_UNUSED(nAppcmdCode);
#endif
return -1;
}
/**************************************************************************//*!
*
* @brief API: Mark the application command is processed by the application
*
* @param nResultCode - the result code which is returned to a host
*
******************************************************************************/
void FMSTR_AppCmdAck(FMSTR_APPCMD_RESULT nResultCode)
{
pcm_nAppCmdResult = nResultCode;
pcm_nAppCmdResultDataLen = 0U;
/* waiting for a new command to come */
pcm_nAppCmd = FMSTR_APPCMDRESULT_NOCMD;
}
/**************************************************************************//*!
*
* @brief API: Mark the application command is processed by the application
*
* @param pResultDataAddr - address of data we want to return to the PC
* @param nResultDataLen - length of return data
*
******************************************************************************/
void FMSTR_AppCmdSetResponseData(FMSTR_ADDR nResultDataAddr, FMSTR_SIZE nResultDataLen)
{
/* any data supplied by user? */
if(nResultDataAddr)
{
/* response data length is trimmed if response data would not fit into buffer */
pcm_nAppCmdResultDataLen = (FMSTR_SIZE8) nResultDataLen;
if(pcm_nAppCmdResultDataLen > (FMSTR_SIZE8) FMSTR_APPCMD_BUFF_SIZE)
{
pcm_nAppCmdResultDataLen = (FMSTR_SIZE8) FMSTR_APPCMD_BUFF_SIZE;
}
if(pcm_nAppCmdResultDataLen > 0U)
{
FMSTR_ADDR appCmdBuffAddr;
FMSTR_ARR2ADDR(appCmdBuffAddr, pcm_pAppCmdBuff);
FMSTR_CopyMemory(appCmdBuffAddr, nResultDataAddr, pcm_nAppCmdResultDataLen);
}
}
else
{
/* no data being returned at all (same effect as pure FMSTR_AppCmdAck) */
pcm_nAppCmdResultDataLen = 0U;
}
}
/**************************************************************************//*!
*
* @brief API: Fetch the application command code if one is ready for processing
*
* @return A command code stored in the application cmd buffer.
* The return value is FMSTR_APPCMDRESULT_NOCMD if there is no
* new command since the last call to FMSTR_AppCmdAck
*
******************************************************************************/
FMSTR_APPCMD_CODE FMSTR_GetAppCmd(void)
{
#if FMSTR_MAX_APPCMD_CALLS
/* the user can never see the callback-registerred commands */
if(FMSTR_FindAppCmdCallIndex(pcm_nAppCmd) >= 0)
{
return FMSTR_APPCMDRESULT_NOCMD;
}
#endif
/* otherwise, return the appcomand pending */
return pcm_nAppCmd;
}
/**************************************************************************//*!
*
* @brief API: Get a pointer to application command data
*
* @param pDataLen - A pointer to variable which receives the data length
*
* @return Pointer to the "application command" data
*
******************************************************************************/
FMSTR_APPCMD_PDATA FMSTR_GetAppCmdData(FMSTR_SIZE* pDataLen)
{
/* no command, no data */
if(pcm_nAppCmd == FMSTR_APPCMDRESULT_NOCMD)
{
return NULL;
}
#if FMSTR_MAX_APPCMD_CALLS
/* the user never sees the callback-registerred commands */
if(FMSTR_FindAppCmdCallIndex(pcm_nAppCmd) >= 0)
{
return NULL;
}
#endif
/* caller want to know the data length */
if(pDataLen)
{
*pDataLen = pcm_nAppCmdLen;
}
/* data are saved in out buffer */
return pcm_nAppCmdLen ? pcm_pAppCmdBuff : (FMSTR_APPCMD_PDATA) NULL;
}
/**************************************************************************//*!
*
* @brief API: Register or unregister app.cmd callback handler
*
* @param nAppCmdCode - App.command ID
* @param pCallbackFunc - Pointer to handler function (NULL to unregister)
*
* @return Non-zero if successfull, zero if maximum callbacks already set
*
******************************************************************************/
FMSTR_BOOL FMSTR_RegisterAppCmdCall(FMSTR_APPCMD_CODE nAppCmdCode, FMSTR_PAPPCMDFUNC pCallbackFunc)
{
#if FMSTR_MAX_APPCMD_CALLS > 0
FMSTR_INDEX nIndex;
/* keep "void" ID as reserved */
if(nAppCmdCode == FMSTR_APPCMDRESULT_NOCMD)
{
return FMSTR_FALSE;
}
/* get index of app.cmd ID (if already set) */
nIndex = FMSTR_FindAppCmdCallIndex(nAppCmdCode);
/* when not found, get a free slot (only if registerring new callback) */
if(nIndex < 0 && pCallbackFunc != NULL)
{
nIndex = FMSTR_FindAppCmdCallIndex(FMSTR_APPCMDRESULT_NOCMD);
}
/* failed? */
if(nIndex < 0)
{
return FMSTR_FALSE;
}
/* register handler */
pcm_pAppCmdCallFunc[nIndex] = pCallbackFunc;
pcm_pAppCmdCallId[nIndex] = (FMSTR_APPCMD_CODE) (pCallbackFunc ?
nAppCmdCode : FMSTR_APPCMDRESULT_NOCMD);
return FMSTR_TRUE;
#else
FMSTR_UNUSED(pCallbackFunc);
FMSTR_UNUSED(nAppCmdCode);
/* app.cmd callback not implemented */
return FMSTR_FALSE;
#endif
}
#else /* FMSTR_USE_APPCMD */
/* void Application Command API functions */
void FMSTR_AppCmdAck(FMSTR_APPCMD_RESULT nResultCode)
{
// FMSTR_UNUSED(nResultCode);
}
void FMSTR_AppCmdSetResponseData(FMSTR_ADDR pResultData, FMSTR_SIZE nResultDataLen)
{
// FMSTR_UNUSED(pResultData);
// FMSTR_UNUSED(nResultDataLen);
}
FMSTR_APPCMD_CODE FMSTR_GetAppCmd(void)
{
return FMSTR_APPCMDRESULT_NOCMD;
}
FMSTR_APPCMD_PDATA FMSTR_GetAppCmdData(FMSTR_SIZE* pDataLen)
{
// FMSTR_UNUSED(pDataLen);
return NULL;
}
FMSTR_BOOL FMSTR_RegisterAppCmdCall(FMSTR_APPCMD_CODE nAppCmdCode, FMSTR_PAPPCMDFUNC pCallbackFunc)
{
// FMSTR_UNUSED(nAppCmdCode);
// FMSTR_UNUSED(pCallbackFunc);
return FMSTR_FALSE;
}
/*lint -efile(766, PE_freemaster_protocol.h) include file is not used in this case */
#endif /* FMSTR_USE_APPCMD */