blob: 1c5d9d407d26f8be9e5c70990b82cec82bddecb8 [file] [log] [blame]
/****************************************************************************
(c) SYSTEC electronic GmbH, D-07973 Greiz, August-Bebel-Str. 29
www.systec-electronic.com
Project: openPOWERLINK
Description: source file for NMT-MN-Module
License:
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of SYSTEC electronic GmbH nor the names of its
contributors may be used to endorse or promote products derived
from this software without prior written permission. For written
permission, please contact info@systec-electronic.com.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE.
Severability Clause:
If a provision of this License is or becomes illegal, invalid or
unenforceable in any jurisdiction, that shall not affect:
1. the validity or enforceability in that jurisdiction of any other
provision of this License; or
2. the validity or enforceability in other jurisdictions of that or
any other provision of this License.
-------------------------------------------------------------------------
$RCSfile: EplNmtMnu.c,v $
$Author: D.Krueger $
$Revision: 1.18 $ $Date: 2008/11/19 09:52:24 $
$State: Exp $
Build Environment:
GCC V3.4
-------------------------------------------------------------------------
Revision History:
2006/06/09 k.t.: start of the implementation
****************************************************************************/
#include "user/EplNmtMnu.h"
#include "user/EplTimeru.h"
#include "user/EplIdentu.h"
#include "user/EplStatusu.h"
#include "user/EplObdu.h"
#include "user/EplDlluCal.h"
#include "Benchmark.h"
#if (((EPL_MODULE_INTEGRATION) & (EPL_MODULE_NMT_MN)) != 0)
#if (((EPL_MODULE_INTEGRATION) & (EPL_MODULE_OBDU)) == 0) && (EPL_OBD_USE_KERNEL == FALSE)
#error "EPL NmtMnu module needs EPL module OBDU or OBDK!"
#endif
//=========================================================================//
// //
// P R I V A T E D E F I N I T I O N S //
// //
//=========================================================================//
//---------------------------------------------------------------------------
// const defines
//---------------------------------------------------------------------------
// TracePoint support for realtime-debugging
#ifdef _DBG_TRACE_POINTS_
void TgtDbgSignalTracePoint(u8 bTracePointNumber_p);
void TgtDbgPostTraceValue(DWORD dwTraceValue_p);
#define TGT_DBG_SIGNAL_TRACE_POINT(p) TgtDbgSignalTracePoint(p)
#define TGT_DBG_POST_TRACE_VALUE(v) TgtDbgPostTraceValue(v)
#else
#define TGT_DBG_SIGNAL_TRACE_POINT(p)
#define TGT_DBG_POST_TRACE_VALUE(v)
#endif
#define EPL_NMTMNU_DBG_POST_TRACE_VALUE(Event_p, uiNodeId_p, wErrorCode_p) \
TGT_DBG_POST_TRACE_VALUE((kEplEventSinkNmtMnu << 28) | (Event_p << 24) \
| (uiNodeId_p << 16) | wErrorCode_p)
// defines for flags in node info structure
#define EPL_NMTMNU_NODE_FLAG_ISOCHRON 0x0001 // CN is being accessed isochronously
#define EPL_NMTMNU_NODE_FLAG_NOT_SCANNED 0x0002 // CN was not scanned once -> decrement SignalCounter and reset flag
#define EPL_NMTMNU_NODE_FLAG_HALTED 0x0004 // boot process for this CN is halted
#define EPL_NMTMNU_NODE_FLAG_NMT_CMD_ISSUED 0x0008 // NMT command was just issued, wrong NMT states will be tolerated
#define EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ 0x0300 // counter for StatusRequest timer handle
#define EPL_NMTMNU_NODE_FLAG_COUNT_LONGER 0x0C00 // counter for longer timeouts timer handle
#define EPL_NMTMNU_NODE_FLAG_INC_STATREQ 0x0100 // increment for StatusRequest timer handle
#define EPL_NMTMNU_NODE_FLAG_INC_LONGER 0x0400 // increment for longer timeouts timer handle
// These counters will be incremented at every timer start
// and copied to timerarg. When the timer event occures
// both will be compared and if unequal the timer event
// will be discarded, because it is an old one.
// defines for timer arguments to draw a distinction between serveral events
#define EPL_NMTMNU_TIMERARG_NODE_MASK 0x000000FFL // mask that contains the node-ID
#define EPL_NMTMNU_TIMERARG_IDENTREQ 0x00010000L // timer event is for IdentRequest
#define EPL_NMTMNU_TIMERARG_STATREQ 0x00020000L // timer event is for StatusRequest
#define EPL_NMTMNU_TIMERARG_LONGER 0x00040000L // timer event is for longer timeouts
#define EPL_NMTMNU_TIMERARG_STATE_MON 0x00080000L // timer event for StatusRequest to monitor execution of NMT state changes
#define EPL_NMTMNU_TIMERARG_COUNT_SR 0x00000300L // counter for StatusRequest
#define EPL_NMTMNU_TIMERARG_COUNT_LO 0x00000C00L // counter for longer timeouts
// The counters must have the same position as in the node flags above.
#define EPL_NMTMNU_SET_FLAGS_TIMERARG_STATREQ(pNodeInfo_p, uiNodeId_p, TimerArg_p) \
pNodeInfo_p->m_wFlags = \
((pNodeInfo_p->m_wFlags + EPL_NMTMNU_NODE_FLAG_INC_STATREQ) \
& EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) \
| (pNodeInfo_p->m_wFlags & ~EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ); \
TimerArg_p.m_ulArg = EPL_NMTMNU_TIMERARG_STATREQ | uiNodeId_p | \
(pNodeInfo_p->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ); \
TimerArg_p.m_EventSink = kEplEventSinkNmtMnu;
#define EPL_NMTMNU_SET_FLAGS_TIMERARG_IDENTREQ(pNodeInfo_p, uiNodeId_p, TimerArg_p) \
pNodeInfo_p->m_wFlags = \
((pNodeInfo_p->m_wFlags + EPL_NMTMNU_NODE_FLAG_INC_STATREQ) \
& EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) \
| (pNodeInfo_p->m_wFlags & ~EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ); \
TimerArg_p.m_ulArg = EPL_NMTMNU_TIMERARG_IDENTREQ | uiNodeId_p | \
(pNodeInfo_p->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ); \
TimerArg_p.m_EventSink = kEplEventSinkNmtMnu;
#define EPL_NMTMNU_SET_FLAGS_TIMERARG_LONGER(pNodeInfo_p, uiNodeId_p, TimerArg_p) \
pNodeInfo_p->m_wFlags = \
((pNodeInfo_p->m_wFlags + EPL_NMTMNU_NODE_FLAG_INC_LONGER) \
& EPL_NMTMNU_NODE_FLAG_COUNT_LONGER) \
| (pNodeInfo_p->m_wFlags & ~EPL_NMTMNU_NODE_FLAG_COUNT_LONGER); \
TimerArg_p.m_ulArg = EPL_NMTMNU_TIMERARG_LONGER | uiNodeId_p | \
(pNodeInfo_p->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_LONGER); \
TimerArg_p.m_EventSink = kEplEventSinkNmtMnu;
#define EPL_NMTMNU_SET_FLAGS_TIMERARG_STATE_MON(pNodeInfo_p, uiNodeId_p, TimerArg_p) \
pNodeInfo_p->m_wFlags = \
((pNodeInfo_p->m_wFlags + EPL_NMTMNU_NODE_FLAG_INC_STATREQ) \
& EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) \
| (pNodeInfo_p->m_wFlags & ~EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ); \
TimerArg_p.m_ulArg = EPL_NMTMNU_TIMERARG_STATE_MON | uiNodeId_p | \
(pNodeInfo_p->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ); \
TimerArg_p.m_EventSink = kEplEventSinkNmtMnu;
// defines for global flags
#define EPL_NMTMNU_FLAG_HALTED 0x0001 // boot process is halted
#define EPL_NMTMNU_FLAG_APP_INFORMED 0x0002 // application was informed about possible NMT state change
// return pointer to node info structure for specified node ID
// d.k. may be replaced by special (hash) function if node ID array is smaller than 254
#define EPL_NMTMNU_GET_NODEINFO(uiNodeId_p) (&EplNmtMnuInstance_g.m_aNodeInfo[uiNodeId_p - 1])
//---------------------------------------------------------------------------
// local types
//---------------------------------------------------------------------------
typedef enum {
kEplNmtMnuIntNodeEventNoIdentResponse = 0x00,
kEplNmtMnuIntNodeEventIdentResponse = 0x01,
kEplNmtMnuIntNodeEventBoot = 0x02,
kEplNmtMnuIntNodeEventExecReset = 0x03,
kEplNmtMnuIntNodeEventConfigured = 0x04,
kEplNmtMnuIntNodeEventNoStatusResponse = 0x05,
kEplNmtMnuIntNodeEventStatusResponse = 0x06,
kEplNmtMnuIntNodeEventHeartbeat = 0x07,
kEplNmtMnuIntNodeEventNmtCmdSent = 0x08,
kEplNmtMnuIntNodeEventTimerIdentReq = 0x09,
kEplNmtMnuIntNodeEventTimerStatReq = 0x0A,
kEplNmtMnuIntNodeEventTimerStateMon = 0x0B,
kEplNmtMnuIntNodeEventTimerLonger = 0x0C,
kEplNmtMnuIntNodeEventError = 0x0D,
} tEplNmtMnuIntNodeEvent;
typedef enum {
kEplNmtMnuNodeStateUnknown = 0x00,
kEplNmtMnuNodeStateIdentified = 0x01,
kEplNmtMnuNodeStateResetConf = 0x02, // CN reset after configuration update
kEplNmtMnuNodeStateConfigured = 0x03, // BootStep1 completed
kEplNmtMnuNodeStateReadyToOp = 0x04, // BootStep2 completed
kEplNmtMnuNodeStateComChecked = 0x05, // Communication checked successfully
kEplNmtMnuNodeStateOperational = 0x06, // CN is in NMT state OPERATIONAL
} tEplNmtMnuNodeState;
typedef struct {
tEplTimerHdl m_TimerHdlStatReq; // timer to delay StatusRequests and IdentRequests
tEplTimerHdl m_TimerHdlLonger; // 2nd timer for NMT command EnableReadyToOp and CheckCommunication
tEplNmtMnuNodeState m_NodeState; // internal node state (kind of sub state of NMT state)
DWORD m_dwNodeCfg; // subindex from 0x1F81
WORD m_wFlags; // flags: CN is being accessed isochronously
} tEplNmtMnuNodeInfo;
typedef struct {
tEplNmtMnuNodeInfo m_aNodeInfo[EPL_NMT_MAX_NODE_ID];
tEplTimerHdl m_TimerHdlNmtState; // timeout for stay in NMT state
unsigned int m_uiMandatorySlaveCount;
unsigned int m_uiSignalSlaveCount;
unsigned long m_ulStatusRequestDelay; // in [ms] (object 0x1006 * EPL_C_NMT_STATREQ_CYCLE)
unsigned long m_ulTimeoutReadyToOp; // in [ms] (object 0x1F89/5)
unsigned long m_ulTimeoutCheckCom; // in [ms] (object 0x1006 * MultiplexedCycleCount)
WORD m_wFlags; // global flags
DWORD m_dwNmtStartup; // object 0x1F80 NMT_StartUp_U32
tEplNmtMnuCbNodeEvent m_pfnCbNodeEvent;
tEplNmtMnuCbBootEvent m_pfnCbBootEvent;
} tEplNmtMnuInstance;
//---------------------------------------------------------------------------
// local vars
//---------------------------------------------------------------------------
static tEplNmtMnuInstance EplNmtMnuInstance_g;
//---------------------------------------------------------------------------
// local function prototypes
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuCbNmtRequest(tEplFrameInfo *pFrameInfo_p);
static tEplKernel EplNmtMnuCbIdentResponse(unsigned int uiNodeId_p,
tEplIdentResponse *pIdentResponse_p);
static tEplKernel EplNmtMnuCbStatusResponse(unsigned int uiNodeId_p,
tEplStatusResponse *pStatusResponse_p);
static tEplKernel EplNmtMnuCheckNmtState(unsigned int uiNodeId_p,
tEplNmtMnuNodeInfo * pNodeInfo_p,
tEplNmtState NodeNmtState_p,
WORD wErrorCode_p,
tEplNmtState LocalNmtState_p);
static tEplKernel EplNmtMnuStartBootStep1(void);
static tEplKernel EplNmtMnuStartBootStep2(void);
static tEplKernel EplNmtMnuStartCheckCom(void);
static tEplKernel EplNmtMnuNodeBootStep2(unsigned int uiNodeId_p,
tEplNmtMnuNodeInfo * pNodeInfo_p);
static tEplKernel EplNmtMnuNodeCheckCom(unsigned int uiNodeId_p,
tEplNmtMnuNodeInfo * pNodeInfo_p);
static tEplKernel EplNmtMnuStartNodes(void);
static tEplKernel EplNmtMnuProcessInternalEvent(unsigned int uiNodeId_p,
tEplNmtState NodeNmtState_p,
WORD wErrorCode_p,
tEplNmtMnuIntNodeEvent
NodeEvent_p);
static tEplKernel EplNmtMnuReset(void);
//=========================================================================//
// //
// P U B L I C F U N C T I O N S //
// //
//=========================================================================//
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuInit
//
// Description: init first instance of the module
//
//
//
// Parameters:
//
//
// Returns: tEplKernel = errorcode
//
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuInit(tEplNmtMnuCbNodeEvent pfnCbNodeEvent_p,
tEplNmtMnuCbBootEvent pfnCbBootEvent_p)
{
tEplKernel Ret;
Ret = EplNmtMnuAddInstance(pfnCbNodeEvent_p, pfnCbBootEvent_p);
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuAddInstance
//
// Description: init other instances of the module
//
//
//
// Parameters:
//
//
// Returns: tEplKernel = errorcode
//
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuAddInstance(tEplNmtMnuCbNodeEvent pfnCbNodeEvent_p,
tEplNmtMnuCbBootEvent pfnCbBootEvent_p)
{
tEplKernel Ret;
Ret = kEplSuccessful;
// reset instance structure
EPL_MEMSET(&EplNmtMnuInstance_g, 0, sizeof(EplNmtMnuInstance_g));
if ((pfnCbNodeEvent_p == NULL) || (pfnCbBootEvent_p == NULL)) {
Ret = kEplNmtInvalidParam;
goto Exit;
}
EplNmtMnuInstance_g.m_pfnCbNodeEvent = pfnCbNodeEvent_p;
EplNmtMnuInstance_g.m_pfnCbBootEvent = pfnCbBootEvent_p;
// initialize StatusRequest delay
EplNmtMnuInstance_g.m_ulStatusRequestDelay = 5000L;
// register NmtMnResponse callback function
Ret =
EplDlluCalRegAsndService(kEplDllAsndNmtRequest,
EplNmtMnuCbNmtRequest,
kEplDllAsndFilterLocal);
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuDelInstance
//
// Description: delete instance
//
//
//
// Parameters:
//
//
// Returns: tEplKernel = errorcode
//
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuDelInstance()
{
tEplKernel Ret;
Ret = kEplSuccessful;
// deregister NmtMnResponse callback function
Ret =
EplDlluCalRegAsndService(kEplDllAsndNmtRequest, NULL,
kEplDllAsndFilterNone);
Ret = EplNmtMnuReset();
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuSendNmtCommandEx
//
// Description: sends the specified NMT command to the specified node.
//
// Parameters: uiNodeId_p = node ID to which the NMT command will be sent
// NmtCommand_p = NMT command
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuSendNmtCommandEx(unsigned int uiNodeId_p,
tEplNmtCommand NmtCommand_p,
void *pNmtCommandData_p,
unsigned int uiDataSize_p)
{
tEplKernel Ret = kEplSuccessful;
tEplFrameInfo FrameInfo;
u8 abBuffer[EPL_C_DLL_MINSIZE_NMTCMDEXT];
tEplFrame *pFrame = (tEplFrame *) abBuffer;
BOOL fSoftDeleteNode = FALSE;
if ((uiNodeId_p == 0) || (uiNodeId_p > EPL_C_ADR_BROADCAST)) { // invalid node ID specified
Ret = kEplInvalidNodeId;
goto Exit;
}
if ((pNmtCommandData_p != NULL)
&& (uiDataSize_p >
(EPL_C_DLL_MINSIZE_NMTCMDEXT - EPL_C_DLL_MINSIZE_NMTCMD))) {
Ret = kEplNmtInvalidParam;
goto Exit;
}
// $$$ d.k. may be check in future versions if the caller wants to perform prohibited state transitions
// the CN should not perform these transitions, but the expected NMT state will be changed and never fullfilled.
// build frame
EPL_MEMSET(pFrame, 0x00, sizeof(abBuffer));
AmiSetByteToLe(&pFrame->m_le_bDstNodeId, (u8) uiNodeId_p);
AmiSetByteToLe(&pFrame->m_Data.m_Asnd.m_le_bServiceId,
(u8) kEplDllAsndNmtCommand);
AmiSetByteToLe(&pFrame->m_Data.m_Asnd.m_Payload.m_NmtCommandService.
m_le_bNmtCommandId, (u8) NmtCommand_p);
if ((pNmtCommandData_p != NULL) && (uiDataSize_p > 0)) { // copy command data to frame
EPL_MEMCPY(&pFrame->m_Data.m_Asnd.m_Payload.m_NmtCommandService.
m_le_abNmtCommandData[0], pNmtCommandData_p,
uiDataSize_p);
}
// build info structure
FrameInfo.m_NetTime.m_dwNanoSec = 0;
FrameInfo.m_NetTime.m_dwSec = 0;
FrameInfo.m_pFrame = pFrame;
FrameInfo.m_uiFrameSize = sizeof(abBuffer);
// send NMT-Request
#if(((EPL_MODULE_INTEGRATION) & (EPL_MODULE_DLLU)) != 0)
Ret = EplDlluCalAsyncSend(&FrameInfo, // pointer to frameinfo
kEplDllAsyncReqPrioNmt); // priority
#endif
if (Ret != kEplSuccessful) {
goto Exit;
}
EPL_DBGLVL_NMTMN_TRACE2("NMTCmd(%02X->%02X)\n", NmtCommand_p,
uiNodeId_p);
switch (NmtCommand_p) {
case kEplNmtCmdStartNode:
case kEplNmtCmdEnterPreOperational2:
case kEplNmtCmdEnableReadyToOperate:
{
// nothing left to do,
// because any further processing is done
// when the NMT command is actually sent
goto Exit;
}
case kEplNmtCmdStopNode:
{
fSoftDeleteNode = TRUE;
break;
}
case kEplNmtCmdResetNode:
case kEplNmtCmdResetCommunication:
case kEplNmtCmdResetConfiguration:
case kEplNmtCmdSwReset:
{
break;
}
default:
goto Exit;
}
// remove CN from isochronous phase;
// This must be done here and not when NMT command is actually sent
// because it will be too late and may cause unwanted errors
if (uiNodeId_p != EPL_C_ADR_BROADCAST) {
if (fSoftDeleteNode == FALSE) { // remove CN immediately from isochronous phase
Ret = EplDlluCalDeleteNode(uiNodeId_p);
} else { // remove CN from isochronous phase softly
Ret = EplDlluCalSoftDeleteNode(uiNodeId_p);
}
} else { // do it for all active CNs
for (uiNodeId_p = 1;
uiNodeId_p <= tabentries(EplNmtMnuInstance_g.m_aNodeInfo);
uiNodeId_p++) {
if ((EPL_NMTMNU_GET_NODEINFO(uiNodeId_p)->
m_dwNodeCfg & (EPL_NODEASSIGN_NODE_IS_CN |
EPL_NODEASSIGN_NODE_EXISTS)) != 0) {
if (fSoftDeleteNode == FALSE) { // remove CN immediately from isochronous phase
Ret = EplDlluCalDeleteNode(uiNodeId_p);
} else { // remove CN from isochronous phase softly
Ret =
EplDlluCalSoftDeleteNode
(uiNodeId_p);
}
}
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuSendNmtCommand
//
// Description: sends the specified NMT command to the specified node.
//
// Parameters: uiNodeId_p = node ID to which the NMT command will be sent
// NmtCommand_p = NMT command
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuSendNmtCommand(unsigned int uiNodeId_p,
tEplNmtCommand NmtCommand_p)
{
tEplKernel Ret = kEplSuccessful;
Ret = EplNmtMnuSendNmtCommandEx(uiNodeId_p, NmtCommand_p, NULL, 0);
//Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuTriggerStateChange
//
// Description: triggers the specified node command for the specified node.
//
// Parameters: uiNodeId_p = node ID for which the node command will be executed
// NodeCommand_p = node command
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuTriggerStateChange(unsigned int uiNodeId_p,
tEplNmtNodeCommand NodeCommand_p)
{
tEplKernel Ret = kEplSuccessful;
tEplNmtMnuIntNodeEvent NodeEvent;
tEplObdSize ObdSize;
u8 bNmtState;
WORD wErrorCode = EPL_E_NO_ERROR;
if ((uiNodeId_p == 0) || (uiNodeId_p >= EPL_C_ADR_BROADCAST)) {
Ret = kEplInvalidNodeId;
goto Exit;
}
switch (NodeCommand_p) {
case kEplNmtNodeCommandBoot:
{
NodeEvent = kEplNmtMnuIntNodeEventBoot;
break;
}
case kEplNmtNodeCommandConfOk:
{
NodeEvent = kEplNmtMnuIntNodeEventConfigured;
break;
}
case kEplNmtNodeCommandConfErr:
{
NodeEvent = kEplNmtMnuIntNodeEventError;
wErrorCode = EPL_E_NMT_BPO1_CF_VERIFY;
break;
}
case kEplNmtNodeCommandConfReset:
{
NodeEvent = kEplNmtMnuIntNodeEventExecReset;
break;
}
default:
{ // invalid node command
goto Exit;
}
}
// fetch current NMT state
ObdSize = 1;
Ret = EplObduReadEntry(0x1F8E, uiNodeId_p, &bNmtState, &ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
Ret = EplNmtMnuProcessInternalEvent(uiNodeId_p,
(tEplNmtState) (bNmtState |
EPL_NMT_TYPE_CS),
wErrorCode, NodeEvent);
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuCbNmtStateChange
//
// Description: callback function for NMT state changes
//
// Parameters: NmtStateChange_p = NMT state change event
//
// Returns: tEplKernel = error code
//
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuCbNmtStateChange(tEplEventNmtStateChange NmtStateChange_p)
{
tEplKernel Ret = kEplSuccessful;
// do work which must be done in that state
switch (NmtStateChange_p.m_NewNmtState) {
// EPL stack is not running
/* case kEplNmtGsOff:
break;
// first init of the hardware
case kEplNmtGsInitialising:
break;
// init of the manufacturer-specific profile area and the
// standardised device profile area
case kEplNmtGsResetApplication:
{
break;
}
// init of the communication profile area
case kEplNmtGsResetCommunication:
{
break;
}
*/
// build the configuration with infos from OD
case kEplNmtGsResetConfiguration:
{
DWORD dwTimeout;
tEplObdSize ObdSize;
// read object 0x1F80 NMT_StartUp_U32
ObdSize = 4;
Ret =
EplObduReadEntry(0x1F80, 0,
&EplNmtMnuInstance_g.
m_dwNmtStartup, &ObdSize);
if (Ret != kEplSuccessful) {
break;
}
// compute StatusReqDelay = object 0x1006 * EPL_C_NMT_STATREQ_CYCLE
ObdSize = sizeof(dwTimeout);
Ret = EplObduReadEntry(0x1006, 0, &dwTimeout, &ObdSize);
if (Ret != kEplSuccessful) {
break;
}
if (dwTimeout != 0L) {
EplNmtMnuInstance_g.m_ulStatusRequestDelay =
dwTimeout * EPL_C_NMT_STATREQ_CYCLE / 1000L;
if (EplNmtMnuInstance_g.
m_ulStatusRequestDelay == 0L) {
EplNmtMnuInstance_g.m_ulStatusRequestDelay = 1L; // at least 1 ms
}
// $$$ fetch and use MultiplexedCycleCount from OD
EplNmtMnuInstance_g.m_ulTimeoutCheckCom =
dwTimeout * EPL_C_NMT_STATREQ_CYCLE / 1000L;
if (EplNmtMnuInstance_g.m_ulTimeoutCheckCom ==
0L) {
EplNmtMnuInstance_g.m_ulTimeoutCheckCom = 1L; // at least 1 ms
}
}
// fetch ReadyToOp Timeout from OD
ObdSize = sizeof(dwTimeout);
Ret = EplObduReadEntry(0x1F89, 5, &dwTimeout, &ObdSize);
if (Ret != kEplSuccessful) {
break;
}
if (dwTimeout != 0L) {
// convert [us] to [ms]
dwTimeout /= 1000L;
if (dwTimeout == 0L) {
dwTimeout = 1L; // at least 1 ms
}
EplNmtMnuInstance_g.m_ulTimeoutReadyToOp =
dwTimeout;
} else {
EplNmtMnuInstance_g.m_ulTimeoutReadyToOp = 0L;
}
break;
}
/*
//-----------------------------------------------------------
// CN part of the state machine
// node liste for EPL-Frames and check timeout
case kEplNmtCsNotActive:
{
break;
}
// node process only async frames
case kEplNmtCsPreOperational1:
{
break;
}
// node process isochronus and asynchronus frames
case kEplNmtCsPreOperational2:
{
break;
}
// node should be configured und application is ready
case kEplNmtCsReadyToOperate:
{
break;
}
// normal work state
case kEplNmtCsOperational:
{
break;
}
// node stopped by MN
// -> only process asynchronus frames
case kEplNmtCsStopped:
{
break;
}
// no EPL cycle
// -> normal ethernet communication
case kEplNmtCsBasicEthernet:
{
break;
}
*/
//-----------------------------------------------------------
// MN part of the state machine
// node listens for EPL-Frames and check timeout
case kEplNmtMsNotActive:
{
break;
}
// node processes only async frames
case kEplNmtMsPreOperational1:
{
DWORD dwTimeout;
tEplTimerArg TimerArg;
tEplObdSize ObdSize;
tEplEvent Event;
// clear global flags, e.g. reenable boot process
EplNmtMnuInstance_g.m_wFlags = 0;
// reset IdentResponses and running IdentRequests and StatusRequests
Ret = EplIdentuReset();
Ret = EplStatusuReset();
// reset timers
Ret = EplNmtMnuReset();
// 2008/11/18 d.k. reset internal node info is not necessary,
// because timer flags are important and other
// things are reset by EplNmtMnuStartBootStep1().
/*
EPL_MEMSET(EplNmtMnuInstance_g.m_aNodeInfo,
0,
sizeof (EplNmtMnuInstance_g.m_aNodeInfo));
*/
// inform DLL about NMT state change,
// so that it can clear the asynchonous queues and start the reduced cycle
Event.m_EventSink = kEplEventSinkDllk;
Event.m_EventType = kEplEventTypeDllkStartReducedCycle;
EPL_MEMSET(&Event.m_NetTime, 0x00,
sizeof(Event.m_NetTime));
Event.m_pArg = NULL;
Event.m_uiSize = 0;
Ret = EplEventuPost(&Event);
if (Ret != kEplSuccessful) {
break;
}
// reset all nodes
// d.k.: skip this step if was just done before, e.g. because of a ResetNode command from a diagnostic node
if (NmtStateChange_p.m_NmtEvent ==
kEplNmtEventTimerMsPreOp1) {
BENCHMARK_MOD_07_TOGGLE(9);
EPL_NMTMNU_DBG_POST_TRACE_VALUE(0,
EPL_C_ADR_BROADCAST,
kEplNmtCmdResetNode);
Ret =
EplNmtMnuSendNmtCommand(EPL_C_ADR_BROADCAST,
kEplNmtCmdResetNode);
if (Ret != kEplSuccessful) {
break;
}
}
// start network scan
Ret = EplNmtMnuStartBootStep1();
// start timer for 0x1F89/2 MNTimeoutPreOp1_U32
ObdSize = sizeof(dwTimeout);
Ret = EplObduReadEntry(0x1F89, 2, &dwTimeout, &ObdSize);
if (Ret != kEplSuccessful) {
break;
}
if (dwTimeout != 0L) {
dwTimeout /= 1000L;
if (dwTimeout == 0L) {
dwTimeout = 1L; // at least 1 ms
}
TimerArg.m_EventSink = kEplEventSinkNmtMnu;
TimerArg.m_ulArg = 0;
Ret =
EplTimeruModifyTimerMs(&EplNmtMnuInstance_g.
m_TimerHdlNmtState,
dwTimeout, TimerArg);
}
break;
}
// node processes isochronous and asynchronous frames
case kEplNmtMsPreOperational2:
{
// add identified CNs to isochronous phase
// send EnableReadyToOp to all identified CNs
Ret = EplNmtMnuStartBootStep2();
// wait for NMT state change of CNs
break;
}
// node should be configured und application is ready
case kEplNmtMsReadyToOperate:
{
// check if PRes of CNs are OK
// d.k. that means wait CycleLength * MultiplexCycleCount (i.e. start timer)
// because Dllk checks PRes of CNs automatically in ReadyToOp
Ret = EplNmtMnuStartCheckCom();
break;
}
// normal work state
case kEplNmtMsOperational:
{
// send StartNode to CNs
// wait for NMT state change of CNs
Ret = EplNmtMnuStartNodes();
break;
}
// no EPL cycle
// -> normal ethernet communication
case kEplNmtMsBasicEthernet:
{
break;
}
default:
{
// TRACE0("EplNmtMnuCbNmtStateChange(): unhandled NMT state\n");
}
}
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuCbCheckEvent
//
// Description: callback funktion for NMT events before they are actually executed.
// The EPL API layer must forward NMT events from NmtCnu module.
// This module will reject some NMT commands while MN.
//
// Parameters: NmtEvent_p = outstanding NMT event for approval
//
// Returns: tEplKernel = error code
// kEplReject = reject the NMT event
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuCbCheckEvent(tEplNmtEvent NmtEvent_p)
{
tEplKernel Ret = kEplSuccessful;
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtuProcessEvent
//
// Description: processes events from event queue
//
// Parameters: pEvent_p = pointer to event
//
// Returns: tEplKernel = errorcode
//
// State:
//
//---------------------------------------------------------------------------
EPLDLLEXPORT tEplKernel EplNmtMnuProcessEvent(tEplEvent *pEvent_p)
{
tEplKernel Ret;
Ret = kEplSuccessful;
// process event
switch (pEvent_p->m_EventType) {
// timer event
case kEplEventTypeTimer:
{
tEplTimerEventArg *pTimerEventArg =
(tEplTimerEventArg *) pEvent_p->m_pArg;
unsigned int uiNodeId;
uiNodeId =
(unsigned int)(pTimerEventArg->
m_ulArg &
EPL_NMTMNU_TIMERARG_NODE_MASK);
if (uiNodeId != 0) {
tEplObdSize ObdSize;
u8 bNmtState;
tEplNmtMnuNodeInfo *pNodeInfo;
pNodeInfo = EPL_NMTMNU_GET_NODEINFO(uiNodeId);
ObdSize = 1;
Ret =
EplObduReadEntry(0x1F8E, uiNodeId,
&bNmtState, &ObdSize);
if (Ret != kEplSuccessful) {
break;
}
if ((pTimerEventArg->
m_ulArg & EPL_NMTMNU_TIMERARG_IDENTREQ) !=
0L) {
if ((pNodeInfo->
m_wFlags &
EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ)
!= (pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR)) { // this is an old (already deleted or modified) timer
// but not the current timer
// so discard it
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(kEplNmtMnuIntNodeEventTimerIdentReq,
uiNodeId,
((pNodeInfo->
m_NodeState << 8)
| 0xFF));
break;
}
/*
EPL_NMTMNU_DBG_POST_TRACE_VALUE(kEplNmtMnuIntNodeEventTimerIdentReq,
uiNodeId,
((pNodeInfo->m_NodeState << 8)
| 0x80
| ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) >> 6)
| ((pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR) >> 8)));
*/
Ret =
EplNmtMnuProcessInternalEvent
(uiNodeId,
(tEplNmtState) (bNmtState |
EPL_NMT_TYPE_CS),
EPL_E_NO_ERROR,
kEplNmtMnuIntNodeEventTimerIdentReq);
}
else if ((pTimerEventArg->
m_ulArg & EPL_NMTMNU_TIMERARG_STATREQ)
!= 0L) {
if ((pNodeInfo->
m_wFlags &
EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ)
!= (pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR)) { // this is an old (already deleted or modified) timer
// but not the current timer
// so discard it
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(kEplNmtMnuIntNodeEventTimerStatReq,
uiNodeId,
((pNodeInfo->
m_NodeState << 8)
| 0xFF));
break;
}
/*
EPL_NMTMNU_DBG_POST_TRACE_VALUE(kEplNmtMnuIntNodeEventTimerStatReq,
uiNodeId,
((pNodeInfo->m_NodeState << 8)
| 0x80
| ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) >> 6)
| ((pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR) >> 8)));
*/
Ret =
EplNmtMnuProcessInternalEvent
(uiNodeId,
(tEplNmtState) (bNmtState |
EPL_NMT_TYPE_CS),
EPL_E_NO_ERROR,
kEplNmtMnuIntNodeEventTimerStatReq);
}
else if ((pTimerEventArg->
m_ulArg &
EPL_NMTMNU_TIMERARG_STATE_MON) !=
0L) {
if ((pNodeInfo->
m_wFlags &
EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ)
!= (pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR)) { // this is an old (already deleted or modified) timer
// but not the current timer
// so discard it
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(kEplNmtMnuIntNodeEventTimerStateMon,
uiNodeId,
((pNodeInfo->
m_NodeState << 8)
| 0xFF));
break;
}
/*
EPL_NMTMNU_DBG_POST_TRACE_VALUE(kEplNmtMnuIntNodeEventTimerStatReq,
uiNodeId,
((pNodeInfo->m_NodeState << 8)
| 0x80
| ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) >> 6)
| ((pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR) >> 8)));
*/
Ret =
EplNmtMnuProcessInternalEvent
(uiNodeId,
(tEplNmtState) (bNmtState |
EPL_NMT_TYPE_CS),
EPL_E_NO_ERROR,
kEplNmtMnuIntNodeEventTimerStateMon);
}
else if ((pTimerEventArg->
m_ulArg & EPL_NMTMNU_TIMERARG_LONGER)
!= 0L) {
if ((pNodeInfo->
m_wFlags &
EPL_NMTMNU_NODE_FLAG_COUNT_LONGER)
!= (pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_LO)) { // this is an old (already deleted or modified) timer
// but not the current timer
// so discard it
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(kEplNmtMnuIntNodeEventTimerLonger,
uiNodeId,
((pNodeInfo->
m_NodeState << 8)
| 0xFF));
break;
}
/*
EPL_NMTMNU_DBG_POST_TRACE_VALUE(kEplNmtMnuIntNodeEventTimerLonger,
uiNodeId,
((pNodeInfo->m_NodeState << 8)
| 0x80
| ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_LONGER) >> 6)
| ((pTimerEventArg->m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_LO) >> 8)));
*/
Ret =
EplNmtMnuProcessInternalEvent
(uiNodeId,
(tEplNmtState) (bNmtState |
EPL_NMT_TYPE_CS),
EPL_E_NO_ERROR,
kEplNmtMnuIntNodeEventTimerLonger);
}
} else { // global timer event
}
break;
}
case kEplEventTypeHeartbeat:
{
tEplHeartbeatEvent *pHeartbeatEvent =
(tEplHeartbeatEvent *) pEvent_p->m_pArg;
Ret =
EplNmtMnuProcessInternalEvent(pHeartbeatEvent->
m_uiNodeId,
pHeartbeatEvent->
m_NmtState,
pHeartbeatEvent->
m_wErrorCode,
kEplNmtMnuIntNodeEventHeartbeat);
break;
}
case kEplEventTypeNmtMnuNmtCmdSent:
{
tEplFrame *pFrame = (tEplFrame *) pEvent_p->m_pArg;
unsigned int uiNodeId;
tEplNmtCommand NmtCommand;
u8 bNmtState;
uiNodeId = AmiGetByteFromLe(&pFrame->m_le_bDstNodeId);
NmtCommand =
(tEplNmtCommand) AmiGetByteFromLe(&pFrame->m_Data.
m_Asnd.m_Payload.
m_NmtCommandService.
m_le_bNmtCommandId);
switch (NmtCommand) {
case kEplNmtCmdStartNode:
bNmtState =
(u8) (kEplNmtCsOperational & 0xFF);
break;
case kEplNmtCmdStopNode:
bNmtState = (u8) (kEplNmtCsStopped & 0xFF);
break;
case kEplNmtCmdEnterPreOperational2:
bNmtState =
(u8) (kEplNmtCsPreOperational2 & 0xFF);
break;
case kEplNmtCmdEnableReadyToOperate:
// d.k. do not change expected node state, because of DS 1.0.0 7.3.1.2.1 Plain NMT State Command
// and because node may not change NMT state within EPL_C_NMT_STATE_TOLERANCE
bNmtState =
(u8) (kEplNmtCsPreOperational2 & 0xFF);
break;
case kEplNmtCmdResetNode:
case kEplNmtCmdResetCommunication:
case kEplNmtCmdResetConfiguration:
case kEplNmtCmdSwReset:
bNmtState = (u8) (kEplNmtCsNotActive & 0xFF);
// EplNmtMnuProcessInternalEvent() sets internal node state to kEplNmtMnuNodeStateUnknown
// after next unresponded IdentRequest/StatusRequest
break;
default:
goto Exit;
}
// process as internal event which update expected NMT state in OD
if (uiNodeId != EPL_C_ADR_BROADCAST) {
Ret = EplNmtMnuProcessInternalEvent(uiNodeId,
(tEplNmtState)
(bNmtState |
EPL_NMT_TYPE_CS),
0,
kEplNmtMnuIntNodeEventNmtCmdSent);
} else { // process internal event for all active nodes (except myself)
for (uiNodeId = 1;
uiNodeId <=
tabentries(EplNmtMnuInstance_g.
m_aNodeInfo); uiNodeId++) {
if ((EPL_NMTMNU_GET_NODEINFO(uiNodeId)->
m_dwNodeCfg &
(EPL_NODEASSIGN_NODE_IS_CN |
EPL_NODEASSIGN_NODE_EXISTS)) !=
0) {
Ret =
EplNmtMnuProcessInternalEvent
(uiNodeId,
(tEplNmtState) (bNmtState |
EPL_NMT_TYPE_CS),
0,
kEplNmtMnuIntNodeEventNmtCmdSent);
if (Ret != kEplSuccessful) {
goto Exit;
}
}
}
}
break;
}
default:
{
Ret = kEplNmtInvalidEvent;
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuGetRunningTimerStatReq
//
// Description: returns a bit field with running StatReq timers
// just for debugging purposes
//
// Parameters: (none)
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
tEplKernel EplNmtMnuGetDiagnosticInfo(unsigned int *puiMandatorySlaveCount_p,
unsigned int *puiSignalSlaveCount_p,
WORD *pwFlags_p)
{
tEplKernel Ret = kEplSuccessful;
if ((puiMandatorySlaveCount_p == NULL)
|| (puiSignalSlaveCount_p == NULL)
|| (pwFlags_p == NULL)) {
Ret = kEplNmtInvalidParam;
goto Exit;
}
*puiMandatorySlaveCount_p = EplNmtMnuInstance_g.m_uiMandatorySlaveCount;
*puiSignalSlaveCount_p = EplNmtMnuInstance_g.m_uiSignalSlaveCount;
*pwFlags_p = EplNmtMnuInstance_g.m_wFlags;
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuGetRunningTimerStatReq
//
// Description: returns a bit field with running StatReq timers
// just for debugging purposes
//
// Parameters: (none)
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
/*
DWORD EplNmtMnuGetRunningTimerStatReq(void)
{
tEplKernel Ret = kEplSuccessful;
unsigned int uiIndex;
tEplNmtMnuNodeInfo* pNodeInfo;
pNodeInfo = EplNmtMnuInstance_g.m_aNodeInfo;
for (uiIndex = 1; uiIndex <= tabentries(EplNmtMnuInstance_g.m_aNodeInfo); uiIndex++, pNodeInfo++)
{
if (pNodeInfo->m_NodeState == kEplNmtMnuNodeStateConfigured)
{
// reset flag "scanned once"
pNodeInfo->m_wFlags &= ~EPL_NMTMNU_NODE_FLAG_SCANNED;
Ret = EplNmtMnuNodeBootStep2(uiIndex, pNodeInfo);
if (Ret != kEplSuccessful)
{
goto Exit;
}
EplNmtMnuInstance_g.m_uiSignalSlaveCount++;
// signal slave counter shall be decremented if StatusRequest was sent once to a CN
// mandatory slave counter shall be decremented if mandatory CN is ReadyToOp
}
}
Exit:
return Ret;
}
*/
//=========================================================================//
// //
// P R I V A T E F U N C T I O N S //
// //
//=========================================================================//
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuCbNmtRequest
//
// Description: callback funktion for NmtRequest
//
// Parameters: pFrameInfo_p = Frame with the NmtRequest
//
// Returns: tEplKernel = error code
//
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuCbNmtRequest(tEplFrameInfo *pFrameInfo_p)
{
tEplKernel Ret = kEplSuccessful;
// $$$ perform NMTRequest
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuCbIdentResponse
//
// Description: callback funktion for IdentResponse
//
// Parameters: uiNodeId_p = node ID for which IdentReponse was received
// pIdentResponse_p = pointer to IdentResponse
// is NULL if node did not answer
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuCbIdentResponse(unsigned int uiNodeId_p,
tEplIdentResponse *pIdentResponse_p)
{
tEplKernel Ret = kEplSuccessful;
if (pIdentResponse_p == NULL) { // node did not answer
Ret = EplNmtMnuProcessInternalEvent(uiNodeId_p, kEplNmtCsNotActive, EPL_E_NMT_NO_IDENT_RES, // was EPL_E_NO_ERROR
kEplNmtMnuIntNodeEventNoIdentResponse);
} else { // node answered IdentRequest
tEplObdSize ObdSize;
DWORD dwDevType;
WORD wErrorCode = EPL_E_NO_ERROR;
tEplNmtState NmtState =
(tEplNmtState) (AmiGetByteFromLe
(&pIdentResponse_p->
m_le_bNmtStatus) | EPL_NMT_TYPE_CS);
// check IdentResponse $$$ move to ProcessIntern, because this function may be called also if CN
// check DeviceType (0x1F84)
ObdSize = 4;
Ret =
EplObduReadEntry(0x1F84, uiNodeId_p, &dwDevType, &ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
if (dwDevType != 0L) { // actually compare it with DeviceType from IdentResponse
if (AmiGetDwordFromLe(&pIdentResponse_p->m_le_dwDeviceType) != dwDevType) { // wrong DeviceType
NmtState = kEplNmtCsNotActive;
wErrorCode = EPL_E_NMT_BPO1_DEVICE_TYPE;
}
}
Ret = EplNmtMnuProcessInternalEvent(uiNodeId_p,
NmtState,
wErrorCode,
kEplNmtMnuIntNodeEventIdentResponse);
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuCbStatusResponse
//
// Description: callback funktion for StatusResponse
//
// Parameters: uiNodeId_p = node ID for which IdentReponse was received
// pIdentResponse_p = pointer to IdentResponse
// is NULL if node did not answer
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuCbStatusResponse(unsigned int uiNodeId_p,
tEplStatusResponse *pStatusResponse_p)
{
tEplKernel Ret = kEplSuccessful;
if (pStatusResponse_p == NULL) { // node did not answer
Ret = EplNmtMnuProcessInternalEvent(uiNodeId_p, kEplNmtCsNotActive, EPL_E_NMT_NO_STATUS_RES, // was EPL_E_NO_ERROR
kEplNmtMnuIntNodeEventNoStatusResponse);
} else { // node answered StatusRequest
Ret = EplNmtMnuProcessInternalEvent(uiNodeId_p,
(tEplNmtState)
(AmiGetByteFromLe
(&pStatusResponse_p->
m_le_bNmtStatus) |
EPL_NMT_TYPE_CS),
EPL_E_NO_ERROR,
kEplNmtMnuIntNodeEventStatusResponse);
}
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuStartBootStep1
//
// Description: starts BootStep1
//
// Parameters: (none)
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuStartBootStep1(void)
{
tEplKernel Ret = kEplSuccessful;
unsigned int uiSubIndex;
unsigned int uiLocalNodeId;
DWORD dwNodeCfg;
tEplObdSize ObdSize;
// $$$ d.k.: save current time for 0x1F89/2 MNTimeoutPreOp1_U32
// start network scan
EplNmtMnuInstance_g.m_uiMandatorySlaveCount = 0;
EplNmtMnuInstance_g.m_uiSignalSlaveCount = 0;
// check 0x1F81
uiLocalNodeId = EplObduGetNodeId();
for (uiSubIndex = 1; uiSubIndex <= 254; uiSubIndex++) {
ObdSize = 4;
Ret =
EplObduReadEntry(0x1F81, uiSubIndex, &dwNodeCfg, &ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
if (uiSubIndex != uiLocalNodeId) {
// reset flags "not scanned" and "isochronous"
EPL_NMTMNU_GET_NODEINFO(uiSubIndex)->m_wFlags &=
~(EPL_NMTMNU_NODE_FLAG_ISOCHRON |
EPL_NMTMNU_NODE_FLAG_NOT_SCANNED);
if (uiSubIndex == EPL_C_ADR_DIAG_DEF_NODE_ID) { // diagnostic node must be scanned by MN in any case
dwNodeCfg |=
(EPL_NODEASSIGN_NODE_IS_CN |
EPL_NODEASSIGN_NODE_EXISTS);
// and it must be isochronously accessed
dwNodeCfg &= ~EPL_NODEASSIGN_ASYNCONLY_NODE;
}
// save node config in local node info structure
EPL_NMTMNU_GET_NODEINFO(uiSubIndex)->m_dwNodeCfg =
dwNodeCfg;
EPL_NMTMNU_GET_NODEINFO(uiSubIndex)->m_NodeState =
kEplNmtMnuNodeStateUnknown;
if ((dwNodeCfg & (EPL_NODEASSIGN_NODE_IS_CN | EPL_NODEASSIGN_NODE_EXISTS)) != 0) { // node is configured as CN
// identify the node
Ret =
EplIdentuRequestIdentResponse(uiSubIndex,
EplNmtMnuCbIdentResponse);
if (Ret != kEplSuccessful) {
goto Exit;
}
// set flag "not scanned"
EPL_NMTMNU_GET_NODEINFO(uiSubIndex)->m_wFlags |=
EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
EplNmtMnuInstance_g.m_uiSignalSlaveCount++;
// signal slave counter shall be decremented if IdentRequest was sent once to a CN
if ((dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // node is a mandatory CN
EplNmtMnuInstance_g.
m_uiMandatorySlaveCount++;
// mandatory slave counter shall be decremented if mandatory CN was configured successfully
}
}
} else { // subindex of MN
if ((dwNodeCfg & (EPL_NODEASSIGN_MN_PRES | EPL_NODEASSIGN_NODE_EXISTS)) != 0) { // MN shall send PRes
tEplDllNodeInfo DllNodeInfo;
EPL_MEMSET(&DllNodeInfo, 0,
sizeof(DllNodeInfo));
DllNodeInfo.m_uiNodeId = uiLocalNodeId;
Ret = EplDlluCalAddNode(&DllNodeInfo);
}
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuStartBootStep2
//
// Description: starts BootStep2.
// That means add nodes to isochronous phase and send
// NMT EnableReadyToOp.
//
// Parameters: (none)
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuStartBootStep2(void)
{
tEplKernel Ret = kEplSuccessful;
unsigned int uiIndex;
tEplNmtMnuNodeInfo *pNodeInfo;
if ((EplNmtMnuInstance_g.m_wFlags & EPL_NMTMNU_FLAG_HALTED) == 0) { // boot process is not halted
// add nodes to isochronous phase and send NMT EnableReadyToOp
EplNmtMnuInstance_g.m_uiMandatorySlaveCount = 0;
EplNmtMnuInstance_g.m_uiSignalSlaveCount = 0;
// reset flag that application was informed about possible state change
EplNmtMnuInstance_g.m_wFlags &= ~EPL_NMTMNU_FLAG_APP_INFORMED;
pNodeInfo = EplNmtMnuInstance_g.m_aNodeInfo;
for (uiIndex = 1;
uiIndex <= tabentries(EplNmtMnuInstance_g.m_aNodeInfo);
uiIndex++, pNodeInfo++) {
if (pNodeInfo->m_NodeState ==
kEplNmtMnuNodeStateConfigured) {
Ret =
EplNmtMnuNodeBootStep2(uiIndex, pNodeInfo);
if (Ret != kEplSuccessful) {
goto Exit;
}
// set flag "not scanned"
pNodeInfo->m_wFlags |=
EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
EplNmtMnuInstance_g.m_uiSignalSlaveCount++;
// signal slave counter shall be decremented if StatusRequest was sent once to a CN
if ((pNodeInfo->m_dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // node is a mandatory CN
EplNmtMnuInstance_g.
m_uiMandatorySlaveCount++;
}
// mandatory slave counter shall be decremented if mandatory CN is ReadyToOp
}
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuNodeBootStep2
//
// Description: starts BootStep2 for the specified node.
// This means the CN is added to isochronous phase if not
// async-only and it gets the NMT command EnableReadyToOp.
// The CN must be in node state Configured, when it enters
// BootStep2. When BootStep2 finishes, the CN is in node state
// ReadyToOp.
// If TimeoutReadyToOp in object 0x1F89/5 is configured,
// TimerHdlLonger will be started with this timeout.
//
// Parameters: uiNodeId_p = node ID
// pNodeInfo_p = pointer to internal node info structure
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuNodeBootStep2(unsigned int uiNodeId_p,
tEplNmtMnuNodeInfo * pNodeInfo_p)
{
tEplKernel Ret = kEplSuccessful;
tEplDllNodeInfo DllNodeInfo;
DWORD dwNodeCfg;
tEplObdSize ObdSize;
tEplTimerArg TimerArg;
dwNodeCfg = pNodeInfo_p->m_dwNodeCfg;
if ((dwNodeCfg & EPL_NODEASSIGN_ASYNCONLY_NODE) == 0) { // add node to isochronous phase
DllNodeInfo.m_uiNodeId = uiNodeId_p;
ObdSize = 4;
Ret =
EplObduReadEntry(0x1F92, uiNodeId_p,
&DllNodeInfo.m_dwPresTimeout, &ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
ObdSize = 2;
Ret =
EplObduReadEntry(0x1F8B, uiNodeId_p,
&DllNodeInfo.m_wPreqPayloadLimit,
&ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
ObdSize = 2;
Ret =
EplObduReadEntry(0x1F8D, uiNodeId_p,
&DllNodeInfo.m_wPresPayloadLimit,
&ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
pNodeInfo_p->m_wFlags |= EPL_NMTMNU_NODE_FLAG_ISOCHRON;
Ret = EplDlluCalAddNode(&DllNodeInfo);
if (Ret != kEplSuccessful) {
goto Exit;
}
}
EPL_NMTMNU_DBG_POST_TRACE_VALUE(0,
uiNodeId_p,
kEplNmtCmdEnableReadyToOperate);
Ret =
EplNmtMnuSendNmtCommand(uiNodeId_p, kEplNmtCmdEnableReadyToOperate);
if (Ret != kEplSuccessful) {
goto Exit;
}
if (EplNmtMnuInstance_g.m_ulTimeoutReadyToOp != 0L) { // start timer
// when the timer expires the CN must be ReadyToOp
EPL_NMTMNU_SET_FLAGS_TIMERARG_LONGER(pNodeInfo_p, uiNodeId_p,
TimerArg);
// TimerArg.m_EventSink = kEplEventSinkNmtMnu;
// TimerArg.m_ulArg = EPL_NMTMNU_TIMERARG_LONGER | uiNodeId_p;
Ret =
EplTimeruModifyTimerMs(&pNodeInfo_p->m_TimerHdlLonger,
EplNmtMnuInstance_g.
m_ulTimeoutReadyToOp, TimerArg);
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuStartCheckCom
//
// Description: starts CheckCommunication
//
// Parameters: (none)
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuStartCheckCom(void)
{
tEplKernel Ret = kEplSuccessful;
unsigned int uiIndex;
tEplNmtMnuNodeInfo *pNodeInfo;
if ((EplNmtMnuInstance_g.m_wFlags & EPL_NMTMNU_FLAG_HALTED) == 0) { // boot process is not halted
// wait some time and check that no communication error occurs
EplNmtMnuInstance_g.m_uiMandatorySlaveCount = 0;
EplNmtMnuInstance_g.m_uiSignalSlaveCount = 0;
// reset flag that application was informed about possible state change
EplNmtMnuInstance_g.m_wFlags &= ~EPL_NMTMNU_FLAG_APP_INFORMED;
pNodeInfo = EplNmtMnuInstance_g.m_aNodeInfo;
for (uiIndex = 1;
uiIndex <= tabentries(EplNmtMnuInstance_g.m_aNodeInfo);
uiIndex++, pNodeInfo++) {
if (pNodeInfo->m_NodeState ==
kEplNmtMnuNodeStateReadyToOp) {
Ret = EplNmtMnuNodeCheckCom(uiIndex, pNodeInfo);
if (Ret == kEplReject) { // timer was started
// wait until it expires
if ((pNodeInfo->m_dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // node is a mandatory CN
EplNmtMnuInstance_g.
m_uiMandatorySlaveCount++;
}
} else if (Ret != kEplSuccessful) {
goto Exit;
}
// set flag "not scanned"
pNodeInfo->m_wFlags |=
EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
EplNmtMnuInstance_g.m_uiSignalSlaveCount++;
// signal slave counter shall be decremented if timeout elapsed and regardless of an error
// mandatory slave counter shall be decremented if timeout elapsed and no error occured
}
}
}
Ret = kEplSuccessful;
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuNodeCheckCom
//
// Description: checks communication of the specified node.
// That means wait some time and if no error occured everything
// is OK.
//
// Parameters: uiNodeId_p = node ID
// pNodeInfo_p = pointer to internal node info structure
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuNodeCheckCom(unsigned int uiNodeId_p,
tEplNmtMnuNodeInfo * pNodeInfo_p)
{
tEplKernel Ret = kEplSuccessful;
DWORD dwNodeCfg;
tEplTimerArg TimerArg;
dwNodeCfg = pNodeInfo_p->m_dwNodeCfg;
if (((dwNodeCfg & EPL_NODEASSIGN_ASYNCONLY_NODE) == 0)
&& (EplNmtMnuInstance_g.m_ulTimeoutCheckCom != 0L)) { // CN is not async-only and timeout for CheckCom was set
// check communication,
// that means wait some time and if no error occured everything is OK;
// start timer (when the timer expires the CN must be still ReadyToOp)
EPL_NMTMNU_SET_FLAGS_TIMERARG_LONGER(pNodeInfo_p, uiNodeId_p,
TimerArg);
// TimerArg.m_EventSink = kEplEventSinkNmtMnu;
// TimerArg.m_ulArg = EPL_NMTMNU_TIMERARG_LONGER | uiNodeId_p;
Ret =
EplTimeruModifyTimerMs(&pNodeInfo_p->m_TimerHdlLonger,
EplNmtMnuInstance_g.
m_ulTimeoutCheckCom, TimerArg);
// update mandatory slave counter, because timer was started
if (Ret == kEplSuccessful) {
Ret = kEplReject;
}
} else { // timer was not started
// assume everything is OK
pNodeInfo_p->m_NodeState = kEplNmtMnuNodeStateComChecked;
}
//Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuStartNodes
//
// Description: really starts all nodes which are ReadyToOp and CheckCom did not fail
//
// Parameters: (none)
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuStartNodes(void)
{
tEplKernel Ret = kEplSuccessful;
unsigned int uiIndex;
tEplNmtMnuNodeInfo *pNodeInfo;
if ((EplNmtMnuInstance_g.m_wFlags & EPL_NMTMNU_FLAG_HALTED) == 0) { // boot process is not halted
// send NMT command Start Node
EplNmtMnuInstance_g.m_uiMandatorySlaveCount = 0;
EplNmtMnuInstance_g.m_uiSignalSlaveCount = 0;
// reset flag that application was informed about possible state change
EplNmtMnuInstance_g.m_wFlags &= ~EPL_NMTMNU_FLAG_APP_INFORMED;
pNodeInfo = EplNmtMnuInstance_g.m_aNodeInfo;
for (uiIndex = 1;
uiIndex <= tabentries(EplNmtMnuInstance_g.m_aNodeInfo);
uiIndex++, pNodeInfo++) {
if (pNodeInfo->m_NodeState ==
kEplNmtMnuNodeStateComChecked) {
if ((EplNmtMnuInstance_g.
m_dwNmtStartup & EPL_NMTST_STARTALLNODES)
== 0) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE(0,
uiIndex,
kEplNmtCmdStartNode);
Ret =
EplNmtMnuSendNmtCommand(uiIndex,
kEplNmtCmdStartNode);
if (Ret != kEplSuccessful) {
goto Exit;
}
}
if ((pNodeInfo->m_dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // node is a mandatory CN
EplNmtMnuInstance_g.
m_uiMandatorySlaveCount++;
}
// set flag "not scanned"
pNodeInfo->m_wFlags |=
EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
EplNmtMnuInstance_g.m_uiSignalSlaveCount++;
// signal slave counter shall be decremented if StatusRequest was sent once to a CN
// mandatory slave counter shall be decremented if mandatory CN is OPERATIONAL
}
}
// $$$ inform application if EPL_NMTST_NO_STARTNODE is set
if ((EplNmtMnuInstance_g.
m_dwNmtStartup & EPL_NMTST_STARTALLNODES) != 0) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE(0, EPL_C_ADR_BROADCAST,
kEplNmtCmdStartNode);
Ret =
EplNmtMnuSendNmtCommand(EPL_C_ADR_BROADCAST,
kEplNmtCmdStartNode);
if (Ret != kEplSuccessful) {
goto Exit;
}
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuProcessInternalEvent
//
// Description: processes internal node events
//
// Parameters: uiNodeId_p = node ID
// NodeNmtState_p = NMT state of CN
// NodeEvent_p = occured events
//
// Returns: tEplKernel = error code
//
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuProcessInternalEvent(unsigned int uiNodeId_p,
tEplNmtState NodeNmtState_p,
WORD wErrorCode_p,
tEplNmtMnuIntNodeEvent
NodeEvent_p)
{
tEplKernel Ret = kEplSuccessful;
tEplNmtState NmtState;
tEplNmtMnuNodeInfo *pNodeInfo;
tEplTimerArg TimerArg;
pNodeInfo = EPL_NMTMNU_GET_NODEINFO(uiNodeId_p);
NmtState = EplNmtuGetNmtState();
if (NmtState <= kEplNmtMsNotActive) { // MN is not active
goto Exit;
}
switch (NodeEvent_p) {
case kEplNmtMnuIntNodeEventIdentResponse:
{
u8 bNmtState;
EPL_NMTMNU_DBG_POST_TRACE_VALUE(NodeEvent_p,
uiNodeId_p,
pNodeInfo->m_NodeState);
if (pNodeInfo->m_NodeState !=
kEplNmtMnuNodeStateResetConf) {
pNodeInfo->m_NodeState =
kEplNmtMnuNodeStateIdentified;
}
// reset flags ISOCHRON and NMT_CMD_ISSUED
pNodeInfo->m_wFlags &= ~(EPL_NMTMNU_NODE_FLAG_ISOCHRON
|
EPL_NMTMNU_NODE_FLAG_NMT_CMD_ISSUED);
if ((NmtState == kEplNmtMsPreOperational1)
&&
((pNodeInfo->
m_wFlags & EPL_NMTMNU_NODE_FLAG_NOT_SCANNED) !=
0)) {
// decrement only signal slave count
EplNmtMnuInstance_g.m_uiSignalSlaveCount--;
pNodeInfo->m_wFlags &=
~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
// update object 0x1F8F NMT_MNNodeExpState_AU8 to PreOp1 (even if local state >= PreOp2)
bNmtState = (u8) (kEplNmtCsPreOperational1 & 0xFF);
Ret =
EplObduWriteEntry(0x1F8F, uiNodeId_p, &bNmtState,
1);
// check NMT state of CN
Ret =
EplNmtMnuCheckNmtState(uiNodeId_p, pNodeInfo,
NodeNmtState_p, wErrorCode_p,
NmtState);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
Ret = kEplSuccessful;
}
break;
}
// request StatusResponse immediately,
// because we want a fast boot-up of CNs
Ret =
EplStatusuRequestStatusResponse(uiNodeId_p,
EplNmtMnuCbStatusResponse);
if (Ret != kEplSuccessful) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE(NodeEvent_p,
uiNodeId_p,
Ret);
if (Ret == kEplInvalidOperation) { // the only situation when this should happen is, when
// StatusResponse was already requested from within
// the StatReq timer event.
// so ignore this error.
Ret = kEplSuccessful;
} else {
break;
}
}
if (pNodeInfo->m_NodeState !=
kEplNmtMnuNodeStateResetConf) {
// inform application
Ret =
EplNmtMnuInstance_g.
m_pfnCbNodeEvent(uiNodeId_p,
kEplNmtNodeEventFound,
NodeNmtState_p,
EPL_E_NO_ERROR,
(pNodeInfo->
m_dwNodeCfg &
EPL_NODEASSIGN_MANDATORY_CN)
!= 0);
if (Ret == kEplReject) { // interrupt boot process on user request
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(NodeEvent_p, uiNodeId_p,
((pNodeInfo->m_NodeState << 8)
| Ret));
Ret = kEplSuccessful;
break;
} else if (Ret != kEplSuccessful) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(NodeEvent_p, uiNodeId_p,
((pNodeInfo->m_NodeState << 8)
| Ret));
break;
}
}
// continue BootStep1
}
case kEplNmtMnuIntNodeEventBoot:
{
// $$$ check identification (vendor ID, product code, revision no, serial no)
if (pNodeInfo->m_NodeState ==
kEplNmtMnuNodeStateIdentified) {
// $$$ check software
// check/start configuration
// inform application
Ret =
EplNmtMnuInstance_g.
m_pfnCbNodeEvent(uiNodeId_p,
kEplNmtNodeEventCheckConf,
NodeNmtState_p,
EPL_E_NO_ERROR,
(pNodeInfo->
m_dwNodeCfg &
EPL_NODEASSIGN_MANDATORY_CN)
!= 0);
if (Ret == kEplReject) { // interrupt boot process on user request
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(kEplNmtMnuIntNodeEventBoot,
uiNodeId_p,
((pNodeInfo->m_NodeState << 8)
| Ret));
Ret = kEplSuccessful;
break;
} else if (Ret != kEplSuccessful) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(kEplNmtMnuIntNodeEventBoot,
uiNodeId_p,
((pNodeInfo->m_NodeState << 8)
| Ret));
break;
}
} else if (pNodeInfo->m_NodeState != kEplNmtMnuNodeStateResetConf) { // wrong CN state
// ignore event
break;
}
// $$$ d.k.: currently we assume configuration is OK
// continue BootStep1
}
case kEplNmtMnuIntNodeEventConfigured:
{
if ((pNodeInfo->m_NodeState !=
kEplNmtMnuNodeStateIdentified)
&& (pNodeInfo->m_NodeState != kEplNmtMnuNodeStateResetConf)) { // wrong CN state
// ignore event
break;
}
pNodeInfo->m_NodeState = kEplNmtMnuNodeStateConfigured;
if (NmtState == kEplNmtMsPreOperational1) {
if ((pNodeInfo->m_dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // decrement mandatory CN counter
EplNmtMnuInstance_g.
m_uiMandatorySlaveCount--;
}
} else {
// put optional node to next step (BootStep2)
Ret =
EplNmtMnuNodeBootStep2(uiNodeId_p,
pNodeInfo);
}
break;
}
case kEplNmtMnuIntNodeEventNoIdentResponse:
{
if ((NmtState == kEplNmtMsPreOperational1)
&&
((pNodeInfo->
m_wFlags & EPL_NMTMNU_NODE_FLAG_NOT_SCANNED) !=
0)) {
// decrement only signal slave count
EplNmtMnuInstance_g.m_uiSignalSlaveCount--;
pNodeInfo->m_wFlags &=
~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
if (pNodeInfo->m_NodeState !=
kEplNmtMnuNodeStateResetConf) {
pNodeInfo->m_NodeState =
kEplNmtMnuNodeStateUnknown;
}
// $$$ d.k. check start time for 0x1F89/2 MNTimeoutPreOp1_U32
// $$$ d.k. check individual timeout 0x1F89/6 MNIdentificationTimeout_U32
// if mandatory node and timeout elapsed -> halt boot procedure
// trigger IdentRequest again (if >= PreOp2, after delay)
if (NmtState >= kEplNmtMsPreOperational2) { // start timer
EPL_NMTMNU_SET_FLAGS_TIMERARG_IDENTREQ
(pNodeInfo, uiNodeId_p, TimerArg);
// TimerArg.m_EventSink = kEplEventSinkNmtMnu;
// TimerArg.m_ulArg = EPL_NMTMNU_TIMERARG_IDENTREQ | uiNodeId_p;
/*
EPL_NMTMNU_DBG_POST_TRACE_VALUE(kEplNmtMnuIntNodeEventNoIdentResponse,
uiNodeId_p,
((pNodeInfo->m_NodeState << 8)
| 0x80
| ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) >> 6)
| ((TimerArg.m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR) >> 8)));
*/
Ret =
EplTimeruModifyTimerMs(&pNodeInfo->
m_TimerHdlStatReq,
EplNmtMnuInstance_g.
m_ulStatusRequestDelay,
TimerArg);
} else { // trigger IdentRequest immediately
Ret =
EplIdentuRequestIdentResponse(uiNodeId_p,
EplNmtMnuCbIdentResponse);
}
break;
}
case kEplNmtMnuIntNodeEventStatusResponse:
{
if ((NmtState >= kEplNmtMsPreOperational2)
&&
((pNodeInfo->
m_wFlags & EPL_NMTMNU_NODE_FLAG_NOT_SCANNED) !=
0)) {
// decrement only signal slave count if checked once for ReadyToOp, CheckCom, Operational
EplNmtMnuInstance_g.m_uiSignalSlaveCount--;
pNodeInfo->m_wFlags &=
~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
// check NMT state of CN
Ret =
EplNmtMnuCheckNmtState(uiNodeId_p, pNodeInfo,
NodeNmtState_p, wErrorCode_p,
NmtState);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
Ret = kEplSuccessful;
}
break;
}
if (NmtState == kEplNmtMsPreOperational1) {
// request next StatusResponse immediately
Ret =
EplStatusuRequestStatusResponse(uiNodeId_p,
EplNmtMnuCbStatusResponse);
if (Ret != kEplSuccessful) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(NodeEvent_p, uiNodeId_p, Ret);
}
} else if ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_ISOCHRON) == 0) { // start timer
// not isochronously accessed CN (e.g. async-only or stopped CN)
EPL_NMTMNU_SET_FLAGS_TIMERARG_STATREQ(pNodeInfo,
uiNodeId_p,
TimerArg);
// TimerArg.m_EventSink = kEplEventSinkNmtMnu;
// TimerArg.m_ulArg = EPL_NMTMNU_TIMERARG_STATREQ | uiNodeId_p;
/*
EPL_NMTMNU_DBG_POST_TRACE_VALUE(kEplNmtMnuIntNodeEventStatusResponse,
uiNodeId_p,
((pNodeInfo->m_NodeState << 8)
| 0x80
| ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_COUNT_STATREQ) >> 6)
| ((TimerArg.m_ulArg & EPL_NMTMNU_TIMERARG_COUNT_SR) >> 8)));
*/
Ret =
EplTimeruModifyTimerMs(&pNodeInfo->
m_TimerHdlStatReq,
EplNmtMnuInstance_g.
m_ulStatusRequestDelay,
TimerArg);
}
break;
}
case kEplNmtMnuIntNodeEventNoStatusResponse:
{
// function CheckNmtState sets node state to unknown if necessary
/*
if ((NmtState >= kEplNmtMsPreOperational2)
&& ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_NOT_SCANNED) != 0))
{
// decrement only signal slave count if checked once for ReadyToOp, CheckCom, Operational
EplNmtMnuInstance_g.m_uiSignalSlaveCount--;
pNodeInfo->m_wFlags &= ~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
*/
// check NMT state of CN
Ret =
EplNmtMnuCheckNmtState(uiNodeId_p, pNodeInfo,
NodeNmtState_p, wErrorCode_p,
NmtState);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
Ret = kEplSuccessful;
}
break;
}
break;
}
case kEplNmtMnuIntNodeEventError:
{ // currently only issued on kEplNmtNodeCommandConfErr
if (pNodeInfo->m_NodeState != kEplNmtMnuNodeStateIdentified) { // wrong CN state
// ignore event
break;
}
// check NMT state of CN
Ret =
EplNmtMnuCheckNmtState(uiNodeId_p, pNodeInfo,
kEplNmtCsNotActive,
wErrorCode_p, NmtState);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
Ret = kEplSuccessful;
}
break;
}
break;
}
case kEplNmtMnuIntNodeEventExecReset:
{
if (pNodeInfo->m_NodeState != kEplNmtMnuNodeStateIdentified) { // wrong CN state
// ignore event
break;
}
pNodeInfo->m_NodeState = kEplNmtMnuNodeStateResetConf;
EPL_NMTMNU_DBG_POST_TRACE_VALUE(NodeEvent_p,
uiNodeId_p,
(((NodeNmtState_p &
0xFF) << 8)
|
kEplNmtCmdResetConfiguration));
// send NMT reset configuration to CN for activation of configuration
Ret =
EplNmtMnuSendNmtCommand(uiNodeId_p,
kEplNmtCmdResetConfiguration);
break;
}
case kEplNmtMnuIntNodeEventHeartbeat:
{
/*
if ((NmtState >= kEplNmtMsPreOperational2)
&& ((pNodeInfo->m_wFlags & EPL_NMTMNU_NODE_FLAG_NOT_SCANNED) != 0))
{
// decrement only signal slave count if checked once for ReadyToOp, CheckCom, Operational
EplNmtMnuInstance_g.m_uiSignalSlaveCount--;
pNodeInfo->m_wFlags &= ~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
*/
// check NMT state of CN
Ret =
EplNmtMnuCheckNmtState(uiNodeId_p, pNodeInfo,
NodeNmtState_p, wErrorCode_p,
NmtState);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
Ret = kEplSuccessful;
}
break;
}
break;
}
case kEplNmtMnuIntNodeEventTimerIdentReq:
{
EPL_DBGLVL_NMTMN_TRACE1
("TimerStatReq->IdentReq(%02X)\n", uiNodeId_p);
// trigger IdentRequest again
Ret =
EplIdentuRequestIdentResponse(uiNodeId_p,
EplNmtMnuCbIdentResponse);
if (Ret != kEplSuccessful) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE(NodeEvent_p,
uiNodeId_p,
(((NodeNmtState_p & 0xFF) << 8)
| Ret));
if (Ret == kEplInvalidOperation) { // this can happen because of a bug in EplTimeruLinuxKernel.c
// so ignore this error.
Ret = kEplSuccessful;
}
}
break;
}
case kEplNmtMnuIntNodeEventTimerStateMon:
{
// reset NMT state change flag
// because from now on the CN must have the correct NMT state
pNodeInfo->m_wFlags &=
~EPL_NMTMNU_NODE_FLAG_NMT_CMD_ISSUED;
// continue with normal StatReq processing
}
case kEplNmtMnuIntNodeEventTimerStatReq:
{
EPL_DBGLVL_NMTMN_TRACE1("TimerStatReq->StatReq(%02X)\n",
uiNodeId_p);
// request next StatusResponse
Ret =
EplStatusuRequestStatusResponse(uiNodeId_p,
EplNmtMnuCbStatusResponse);
if (Ret != kEplSuccessful) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE(NodeEvent_p,
uiNodeId_p,
(((NodeNmtState_p & 0xFF) << 8)
| Ret));
if (Ret == kEplInvalidOperation) { // the only situation when this should happen is, when
// StatusResponse was already requested while processing
// event IdentResponse.
// so ignore this error.
Ret = kEplSuccessful;
}
}
break;
}
case kEplNmtMnuIntNodeEventTimerLonger:
{
switch (pNodeInfo->m_NodeState) {
case kEplNmtMnuNodeStateConfigured:
{ // node should be ReadyToOp but it is not
// check NMT state which shall be intentionally wrong, so that ERROR_TREATMENT will be started
Ret =
EplNmtMnuCheckNmtState(uiNodeId_p,
pNodeInfo,
kEplNmtCsNotActive,
EPL_E_NMT_BPO2,
NmtState);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
Ret = kEplSuccessful;
}
break;
}
break;
}
case kEplNmtMnuNodeStateReadyToOp:
{ // CheckCom finished successfully
pNodeInfo->m_NodeState =
kEplNmtMnuNodeStateComChecked;
if ((pNodeInfo->
m_wFlags &
EPL_NMTMNU_NODE_FLAG_NOT_SCANNED)
!= 0) {
// decrement only signal slave count if checked once for ReadyToOp, CheckCom, Operational
EplNmtMnuInstance_g.
m_uiSignalSlaveCount--;
pNodeInfo->m_wFlags &=
~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
if ((pNodeInfo->
m_dwNodeCfg &
EPL_NODEASSIGN_MANDATORY_CN) !=
0) {
// decrement mandatory slave counter
EplNmtMnuInstance_g.
m_uiMandatorySlaveCount--;
}
if (NmtState != kEplNmtMsReadyToOperate) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE
(NodeEvent_p, uiNodeId_p,
(((NodeNmtState_p & 0xFF)
<< 8)
| kEplNmtCmdStartNode));
// start optional CN
Ret =
EplNmtMnuSendNmtCommand
(uiNodeId_p,
kEplNmtCmdStartNode);
}
break;
}
default:
{
break;
}
}
break;
}
case kEplNmtMnuIntNodeEventNmtCmdSent:
{
u8 bNmtState;
// update expected NMT state with the one that results
// from the sent NMT command
bNmtState = (u8) (NodeNmtState_p & 0xFF);
// write object 0x1F8F NMT_MNNodeExpState_AU8
Ret =
EplObduWriteEntry(0x1F8F, uiNodeId_p, &bNmtState,
1);
if (Ret != kEplSuccessful) {
goto Exit;
}
if (NodeNmtState_p == kEplNmtCsNotActive) { // restart processing with IdentRequest
EPL_NMTMNU_SET_FLAGS_TIMERARG_IDENTREQ
(pNodeInfo, uiNodeId_p, TimerArg);
} else { // monitor NMT state change with StatusRequest after
// the corresponding delay;
// until then wrong NMT states will be ignored
EPL_NMTMNU_SET_FLAGS_TIMERARG_STATE_MON
(pNodeInfo, uiNodeId_p, TimerArg);
// set NMT state change flag
pNodeInfo->m_wFlags |=
EPL_NMTMNU_NODE_FLAG_NMT_CMD_ISSUED;
}
Ret =
EplTimeruModifyTimerMs(&pNodeInfo->
m_TimerHdlStatReq,
EplNmtMnuInstance_g.
m_ulStatusRequestDelay,
TimerArg);
// finish processing, because NmtState_p is the expected and not the current state
goto Exit;
}
default:
{
break;
}
}
// check if network is ready to change local NMT state and this was not done before
if ((EplNmtMnuInstance_g.m_wFlags & (EPL_NMTMNU_FLAG_HALTED | EPL_NMTMNU_FLAG_APP_INFORMED)) == 0) { // boot process is not halted
switch (NmtState) {
case kEplNmtMsPreOperational1:
{
if ((EplNmtMnuInstance_g.m_uiSignalSlaveCount ==
0)
&& (EplNmtMnuInstance_g.m_uiMandatorySlaveCount == 0)) { // all optional CNs scanned once and all mandatory CNs configured successfully
EplNmtMnuInstance_g.m_wFlags |=
EPL_NMTMNU_FLAG_APP_INFORMED;
// inform application
Ret =
EplNmtMnuInstance_g.
m_pfnCbBootEvent
(kEplNmtBootEventBootStep1Finish,
NmtState, EPL_E_NO_ERROR);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
// wait for application
Ret = kEplSuccessful;
}
break;
}
// enter PreOp2
Ret =
EplNmtuNmtEvent
(kEplNmtEventAllMandatoryCNIdent);
}
break;
}
case kEplNmtMsPreOperational2:
{
if ((EplNmtMnuInstance_g.m_uiSignalSlaveCount ==
0)
&& (EplNmtMnuInstance_g.m_uiMandatorySlaveCount == 0)) { // all optional CNs checked once for ReadyToOp and all mandatory CNs are ReadyToOp
EplNmtMnuInstance_g.m_wFlags |=
EPL_NMTMNU_FLAG_APP_INFORMED;
// inform application
Ret =
EplNmtMnuInstance_g.
m_pfnCbBootEvent
(kEplNmtBootEventBootStep2Finish,
NmtState, EPL_E_NO_ERROR);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
// wait for application
Ret = kEplSuccessful;
}
break;
}
// enter ReadyToOp
Ret =
EplNmtuNmtEvent
(kEplNmtEventEnterReadyToOperate);
}
break;
}
case kEplNmtMsReadyToOperate:
{
if ((EplNmtMnuInstance_g.m_uiSignalSlaveCount ==
0)
&& (EplNmtMnuInstance_g.m_uiMandatorySlaveCount == 0)) { // all CNs checked for errorless communication
EplNmtMnuInstance_g.m_wFlags |=
EPL_NMTMNU_FLAG_APP_INFORMED;
// inform application
Ret =
EplNmtMnuInstance_g.
m_pfnCbBootEvent
(kEplNmtBootEventCheckComFinish,
NmtState, EPL_E_NO_ERROR);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
// wait for application
Ret = kEplSuccessful;
}
break;
}
// enter Operational
Ret =
EplNmtuNmtEvent
(kEplNmtEventEnterMsOperational);
}
break;
}
case kEplNmtMsOperational:
{
if ((EplNmtMnuInstance_g.m_uiSignalSlaveCount ==
0)
&& (EplNmtMnuInstance_g.m_uiMandatorySlaveCount == 0)) { // all optional CNs scanned once and all mandatory CNs are OPERATIONAL
EplNmtMnuInstance_g.m_wFlags |=
EPL_NMTMNU_FLAG_APP_INFORMED;
// inform application
Ret =
EplNmtMnuInstance_g.
m_pfnCbBootEvent
(kEplNmtBootEventOperational,
NmtState, EPL_E_NO_ERROR);
if (Ret != kEplSuccessful) {
if (Ret == kEplReject) {
// ignore error code
Ret = kEplSuccessful;
}
break;
}
}
break;
}
default:
{
break;
}
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuCheckNmtState
//
// Description: checks the NMT state, i.e. evaluates it with object 0x1F8F
// NMT_MNNodeExpState_AU8 and updates object 0x1F8E
// NMT_MNNodeCurrState_AU8.
// It manipulates m_NodeState in internal node info structure.
//
// Parameters: uiNodeId_p = node ID
// NodeNmtState_p = NMT state of CN
//
// Returns: tEplKernel = error code
// kEplReject = CN was in wrong state and has been reset
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuCheckNmtState(unsigned int uiNodeId_p,
tEplNmtMnuNodeInfo * pNodeInfo_p,
tEplNmtState NodeNmtState_p,
WORD wErrorCode_p,
tEplNmtState LocalNmtState_p)
{
tEplKernel Ret = kEplSuccessful;
tEplObdSize ObdSize;
u8 bNmtState;
u8 bNmtStatePrev;
tEplNmtState ExpNmtState;
ObdSize = 1;
// read object 0x1F8F NMT_MNNodeExpState_AU8
Ret = EplObduReadEntry(0x1F8F, uiNodeId_p, &bNmtState, &ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
// compute expected NMT state
ExpNmtState = (tEplNmtState) (bNmtState | EPL_NMT_TYPE_CS);
// compute u8 of current NMT state
bNmtState = ((u8) NodeNmtState_p & 0xFF);
if (ExpNmtState == kEplNmtCsNotActive) { // ignore the current state, because the CN shall be not active
Ret = kEplReject;
goto Exit;
} else if ((ExpNmtState == kEplNmtCsPreOperational2)
&& (NodeNmtState_p == kEplNmtCsReadyToOperate)) { // CN switched to ReadyToOp
// delete timer for timeout handling
Ret = EplTimeruDeleteTimer(&pNodeInfo_p->m_TimerHdlLonger);
if (Ret != kEplSuccessful) {
goto Exit;
}
pNodeInfo_p->m_NodeState = kEplNmtMnuNodeStateReadyToOp;
// update object 0x1F8F NMT_MNNodeExpState_AU8 to ReadyToOp
Ret = EplObduWriteEntry(0x1F8F, uiNodeId_p, &bNmtState, 1);
if (Ret != kEplSuccessful) {
goto Exit;
}
if ((pNodeInfo_p->m_dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // node is a mandatory CN -> decrement counter
EplNmtMnuInstance_g.m_uiMandatorySlaveCount--;
}
if (LocalNmtState_p >= kEplNmtMsReadyToOperate) { // start procedure CheckCommunication for this node
Ret = EplNmtMnuNodeCheckCom(uiNodeId_p, pNodeInfo_p);
if (Ret != kEplSuccessful) {
goto Exit;
}
if ((LocalNmtState_p == kEplNmtMsOperational)
&& (pNodeInfo_p->m_NodeState ==
kEplNmtMnuNodeStateComChecked)) {
EPL_NMTMNU_DBG_POST_TRACE_VALUE(0, uiNodeId_p,
(((NodeNmtState_p & 0xFF) << 8)
|
kEplNmtCmdStartNode));
// immediately start optional CN, because communication is always OK (e.g. async-only CN)
Ret =
EplNmtMnuSendNmtCommand(uiNodeId_p,
kEplNmtCmdStartNode);
if (Ret != kEplSuccessful) {
goto Exit;
}
}
}
} else if ((ExpNmtState == kEplNmtCsReadyToOperate)
&& (NodeNmtState_p == kEplNmtCsOperational)) { // CN switched to OPERATIONAL
pNodeInfo_p->m_NodeState = kEplNmtMnuNodeStateOperational;
if ((pNodeInfo_p->m_dwNodeCfg & EPL_NODEASSIGN_MANDATORY_CN) != 0) { // node is a mandatory CN -> decrement counter
EplNmtMnuInstance_g.m_uiMandatorySlaveCount--;
}
} else if ((ExpNmtState != NodeNmtState_p)
&& !((ExpNmtState == kEplNmtCsPreOperational1)
&& (NodeNmtState_p == kEplNmtCsPreOperational2))) { // CN is not in expected NMT state (without the exceptions above)
WORD wbeErrorCode;
if ((pNodeInfo_p->
m_wFlags & EPL_NMTMNU_NODE_FLAG_NOT_SCANNED) != 0) {
// decrement only signal slave count if checked once
EplNmtMnuInstance_g.m_uiSignalSlaveCount--;
pNodeInfo_p->m_wFlags &=
~EPL_NMTMNU_NODE_FLAG_NOT_SCANNED;
}
if (pNodeInfo_p->m_NodeState == kEplNmtMnuNodeStateUnknown) { // CN is already in state unknown, which means that it got
// NMT reset command earlier
goto Exit;
}
// -> CN is in wrong NMT state
pNodeInfo_p->m_NodeState = kEplNmtMnuNodeStateUnknown;
if (wErrorCode_p == 0) { // assume wrong NMT state error
if ((pNodeInfo_p->m_wFlags & EPL_NMTMNU_NODE_FLAG_NMT_CMD_ISSUED) != 0) { // NMT command has been just issued;
// ignore wrong NMT state until timer expires;
// other errors like LOSS_PRES_TH are still processed
goto Exit;
}
wErrorCode_p = EPL_E_NMT_WRONG_STATE;
}
BENCHMARK_MOD_07_TOGGLE(9);
// $$$ start ERROR_TREATMENT and inform application
Ret = EplNmtMnuInstance_g.m_pfnCbNodeEvent(uiNodeId_p,
kEplNmtNodeEventError,
NodeNmtState_p,
wErrorCode_p,
(pNodeInfo_p->
m_dwNodeCfg &
EPL_NODEASSIGN_MANDATORY_CN)
!= 0);
if (Ret != kEplSuccessful) {
goto Exit;
}
EPL_NMTMNU_DBG_POST_TRACE_VALUE(0,
uiNodeId_p,
(((NodeNmtState_p & 0xFF) << 8)
| kEplNmtCmdResetNode));
// reset CN
// store error code in NMT command data for diagnostic purpose
AmiSetWordToLe(&wbeErrorCode, wErrorCode_p);
Ret =
EplNmtMnuSendNmtCommandEx(uiNodeId_p, kEplNmtCmdResetNode,
&wbeErrorCode,
sizeof(wbeErrorCode));
if (Ret == kEplSuccessful) {
Ret = kEplReject;
}
goto Exit;
}
// check if NMT_MNNodeCurrState_AU8 has to be changed
ObdSize = 1;
Ret = EplObduReadEntry(0x1F8E, uiNodeId_p, &bNmtStatePrev, &ObdSize);
if (Ret != kEplSuccessful) {
goto Exit;
}
if (bNmtState != bNmtStatePrev) {
// update object 0x1F8E NMT_MNNodeCurrState_AU8
Ret = EplObduWriteEntry(0x1F8E, uiNodeId_p, &bNmtState, 1);
if (Ret != kEplSuccessful) {
goto Exit;
}
Ret = EplNmtMnuInstance_g.m_pfnCbNodeEvent(uiNodeId_p,
kEplNmtNodeEventNmtState,
NodeNmtState_p,
wErrorCode_p,
(pNodeInfo_p->
m_dwNodeCfg &
EPL_NODEASSIGN_MANDATORY_CN)
!= 0);
if (Ret != kEplSuccessful) {
goto Exit;
}
}
Exit:
return Ret;
}
//---------------------------------------------------------------------------
//
// Function: EplNmtMnuReset
//
// Description: reset internal structures, e.g. timers
//
// Parameters: void
//
// Returns: tEplKernel = error code
//
// State:
//
//---------------------------------------------------------------------------
static tEplKernel EplNmtMnuReset(void)
{
tEplKernel Ret;
int iIndex;
Ret = EplTimeruDeleteTimer(&EplNmtMnuInstance_g.m_TimerHdlNmtState);
for (iIndex = 1; iIndex <= tabentries(EplNmtMnuInstance_g.m_aNodeInfo);
iIndex++) {
// delete timer handles
Ret =
EplTimeruDeleteTimer(&EPL_NMTMNU_GET_NODEINFO(iIndex)->
m_TimerHdlStatReq);
Ret =
EplTimeruDeleteTimer(&EPL_NMTMNU_GET_NODEINFO(iIndex)->
m_TimerHdlLonger);
}
return Ret;
}
#endif // #if(((EPL_MODULE_INTEGRATION) & (EPL_MODULE_NMT_MN)) != 0)
// EOF