1*10465441SEvalZero /*
2*10465441SEvalZero * FreeModbus Libary: A portable Modbus implementation for Modbus ASCII/RTU.
3*10465441SEvalZero * Copyright (c) 2006 Christian Walter <[email protected]>
4*10465441SEvalZero * All rights reserved.
5*10465441SEvalZero *
6*10465441SEvalZero * Redistribution and use in source and binary forms, with or without
7*10465441SEvalZero * modification, are permitted provided that the following conditions
8*10465441SEvalZero * are met:
9*10465441SEvalZero * 1. Redistributions of source code must retain the above copyright
10*10465441SEvalZero * notice, this list of conditions and the following disclaimer.
11*10465441SEvalZero * 2. Redistributions in binary form must reproduce the above copyright
12*10465441SEvalZero * notice, this list of conditions and the following disclaimer in the
13*10465441SEvalZero * documentation and/or other materials provided with the distribution.
14*10465441SEvalZero * 3. The name of the author may not be used to endorse or promote products
15*10465441SEvalZero * derived from this software without specific prior written permission.
16*10465441SEvalZero *
17*10465441SEvalZero * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18*10465441SEvalZero * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19*10465441SEvalZero * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20*10465441SEvalZero * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21*10465441SEvalZero * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22*10465441SEvalZero * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23*10465441SEvalZero * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24*10465441SEvalZero * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25*10465441SEvalZero * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26*10465441SEvalZero * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27*10465441SEvalZero *
28*10465441SEvalZero * File: $Id: mb.c,v 1.27 2007/02/18 23:45:41 wolti Exp $
29*10465441SEvalZero */
30*10465441SEvalZero
31*10465441SEvalZero /* ----------------------- System includes ----------------------------------*/
32*10465441SEvalZero #include "stdlib.h"
33*10465441SEvalZero #include "string.h"
34*10465441SEvalZero
35*10465441SEvalZero /* ----------------------- Platform includes --------------------------------*/
36*10465441SEvalZero #include "port.h"
37*10465441SEvalZero
38*10465441SEvalZero /* ----------------------- Modbus includes ----------------------------------*/
39*10465441SEvalZero
40*10465441SEvalZero #include "mb.h"
41*10465441SEvalZero #include "mbconfig.h"
42*10465441SEvalZero #include "mbframe.h"
43*10465441SEvalZero #include "mbproto.h"
44*10465441SEvalZero #include "mbfunc.h"
45*10465441SEvalZero
46*10465441SEvalZero #include "mbport.h"
47*10465441SEvalZero #if MB_SLAVE_RTU_ENABLED == 1
48*10465441SEvalZero #include "mbrtu.h"
49*10465441SEvalZero #endif
50*10465441SEvalZero #if MB_SLAVE_ASCII_ENABLED == 1
51*10465441SEvalZero #include "mbascii.h"
52*10465441SEvalZero #endif
53*10465441SEvalZero #if MB_SLAVE_TCP_ENABLED == 1
54*10465441SEvalZero #include "mbtcp.h"
55*10465441SEvalZero #endif
56*10465441SEvalZero
57*10465441SEvalZero #ifndef MB_PORT_HAS_CLOSE
58*10465441SEvalZero #define MB_PORT_HAS_CLOSE 0
59*10465441SEvalZero #endif
60*10465441SEvalZero
61*10465441SEvalZero /* ----------------------- Static variables ---------------------------------*/
62*10465441SEvalZero
63*10465441SEvalZero static UCHAR ucMBAddress;
64*10465441SEvalZero static eMBMode eMBCurrentMode;
65*10465441SEvalZero
66*10465441SEvalZero static enum
67*10465441SEvalZero {
68*10465441SEvalZero STATE_ENABLED,
69*10465441SEvalZero STATE_DISABLED,
70*10465441SEvalZero STATE_NOT_INITIALIZED
71*10465441SEvalZero } eMBState = STATE_NOT_INITIALIZED;
72*10465441SEvalZero
73*10465441SEvalZero /* Functions pointer which are initialized in eMBInit( ). Depending on the
74*10465441SEvalZero * mode (RTU or ASCII) the are set to the correct implementations.
75*10465441SEvalZero * Using for Modbus Slave
76*10465441SEvalZero */
77*10465441SEvalZero static peMBFrameSend peMBFrameSendCur;
78*10465441SEvalZero static pvMBFrameStart pvMBFrameStartCur;
79*10465441SEvalZero static pvMBFrameStop pvMBFrameStopCur;
80*10465441SEvalZero static peMBFrameReceive peMBFrameReceiveCur;
81*10465441SEvalZero static pvMBFrameClose pvMBFrameCloseCur;
82*10465441SEvalZero
83*10465441SEvalZero /* Callback functions required by the porting layer. They are called when
84*10465441SEvalZero * an external event has happend which includes a timeout or the reception
85*10465441SEvalZero * or transmission of a character.
86*10465441SEvalZero * Using for Modbus Slave
87*10465441SEvalZero */
88*10465441SEvalZero BOOL( *pxMBFrameCBByteReceived ) ( void );
89*10465441SEvalZero BOOL( *pxMBFrameCBTransmitterEmpty ) ( void );
90*10465441SEvalZero BOOL( *pxMBPortCBTimerExpired ) ( void );
91*10465441SEvalZero
92*10465441SEvalZero BOOL( *pxMBFrameCBReceiveFSMCur ) ( void );
93*10465441SEvalZero BOOL( *pxMBFrameCBTransmitFSMCur ) ( void );
94*10465441SEvalZero
95*10465441SEvalZero /* An array of Modbus functions handlers which associates Modbus function
96*10465441SEvalZero * codes with implementing functions.
97*10465441SEvalZero */
98*10465441SEvalZero static xMBFunctionHandler xFuncHandlers[MB_FUNC_HANDLERS_MAX] = {
99*10465441SEvalZero #if MB_FUNC_OTHER_REP_SLAVEID_ENABLED > 0
100*10465441SEvalZero {MB_FUNC_OTHER_REPORT_SLAVEID, eMBFuncReportSlaveID},
101*10465441SEvalZero #endif
102*10465441SEvalZero #if MB_FUNC_READ_INPUT_ENABLED > 0
103*10465441SEvalZero {MB_FUNC_READ_INPUT_REGISTER, eMBFuncReadInputRegister},
104*10465441SEvalZero #endif
105*10465441SEvalZero #if MB_FUNC_READ_HOLDING_ENABLED > 0
106*10465441SEvalZero {MB_FUNC_READ_HOLDING_REGISTER, eMBFuncReadHoldingRegister},
107*10465441SEvalZero #endif
108*10465441SEvalZero #if MB_FUNC_WRITE_MULTIPLE_HOLDING_ENABLED > 0
109*10465441SEvalZero {MB_FUNC_WRITE_MULTIPLE_REGISTERS, eMBFuncWriteMultipleHoldingRegister},
110*10465441SEvalZero #endif
111*10465441SEvalZero #if MB_FUNC_WRITE_HOLDING_ENABLED > 0
112*10465441SEvalZero {MB_FUNC_WRITE_REGISTER, eMBFuncWriteHoldingRegister},
113*10465441SEvalZero #endif
114*10465441SEvalZero #if MB_FUNC_READWRITE_HOLDING_ENABLED > 0
115*10465441SEvalZero {MB_FUNC_READWRITE_MULTIPLE_REGISTERS, eMBFuncReadWriteMultipleHoldingRegister},
116*10465441SEvalZero #endif
117*10465441SEvalZero #if MB_FUNC_READ_COILS_ENABLED > 0
118*10465441SEvalZero {MB_FUNC_READ_COILS, eMBFuncReadCoils},
119*10465441SEvalZero #endif
120*10465441SEvalZero #if MB_FUNC_WRITE_COIL_ENABLED > 0
121*10465441SEvalZero {MB_FUNC_WRITE_SINGLE_COIL, eMBFuncWriteCoil},
122*10465441SEvalZero #endif
123*10465441SEvalZero #if MB_FUNC_WRITE_MULTIPLE_COILS_ENABLED > 0
124*10465441SEvalZero {MB_FUNC_WRITE_MULTIPLE_COILS, eMBFuncWriteMultipleCoils},
125*10465441SEvalZero #endif
126*10465441SEvalZero #if MB_FUNC_READ_DISCRETE_INPUTS_ENABLED > 0
127*10465441SEvalZero {MB_FUNC_READ_DISCRETE_INPUTS, eMBFuncReadDiscreteInputs},
128*10465441SEvalZero #endif
129*10465441SEvalZero };
130*10465441SEvalZero
131*10465441SEvalZero /* ----------------------- Start implementation -----------------------------*/
132*10465441SEvalZero eMBErrorCode
eMBInit(eMBMode eMode,UCHAR ucSlaveAddress,UCHAR ucPort,ULONG ulBaudRate,eMBParity eParity)133*10465441SEvalZero eMBInit( eMBMode eMode, UCHAR ucSlaveAddress, UCHAR ucPort, ULONG ulBaudRate, eMBParity eParity )
134*10465441SEvalZero {
135*10465441SEvalZero eMBErrorCode eStatus = MB_ENOERR;
136*10465441SEvalZero
137*10465441SEvalZero /* check preconditions */
138*10465441SEvalZero if( ( ucSlaveAddress == MB_ADDRESS_BROADCAST ) ||
139*10465441SEvalZero ( ucSlaveAddress < MB_ADDRESS_MIN ) || ( ucSlaveAddress > MB_ADDRESS_MAX ) )
140*10465441SEvalZero {
141*10465441SEvalZero eStatus = MB_EINVAL;
142*10465441SEvalZero }
143*10465441SEvalZero else
144*10465441SEvalZero {
145*10465441SEvalZero ucMBAddress = ucSlaveAddress;
146*10465441SEvalZero
147*10465441SEvalZero switch ( eMode )
148*10465441SEvalZero {
149*10465441SEvalZero #if MB_SLAVE_RTU_ENABLED > 0
150*10465441SEvalZero case MB_RTU:
151*10465441SEvalZero pvMBFrameStartCur = eMBRTUStart;
152*10465441SEvalZero pvMBFrameStopCur = eMBRTUStop;
153*10465441SEvalZero peMBFrameSendCur = eMBRTUSend;
154*10465441SEvalZero peMBFrameReceiveCur = eMBRTUReceive;
155*10465441SEvalZero pvMBFrameCloseCur = MB_PORT_HAS_CLOSE ? vMBPortClose : NULL;
156*10465441SEvalZero pxMBFrameCBByteReceived = xMBRTUReceiveFSM;
157*10465441SEvalZero pxMBFrameCBTransmitterEmpty = xMBRTUTransmitFSM;
158*10465441SEvalZero pxMBPortCBTimerExpired = xMBRTUTimerT35Expired;
159*10465441SEvalZero
160*10465441SEvalZero eStatus = eMBRTUInit( ucMBAddress, ucPort, ulBaudRate, eParity );
161*10465441SEvalZero break;
162*10465441SEvalZero #endif
163*10465441SEvalZero #if MB_SLAVE_ASCII_ENABLED > 0
164*10465441SEvalZero case MB_ASCII:
165*10465441SEvalZero pvMBFrameStartCur = eMBASCIIStart;
166*10465441SEvalZero pvMBFrameStopCur = eMBASCIIStop;
167*10465441SEvalZero peMBFrameSendCur = eMBASCIISend;
168*10465441SEvalZero peMBFrameReceiveCur = eMBASCIIReceive;
169*10465441SEvalZero pvMBFrameCloseCur = MB_PORT_HAS_CLOSE ? vMBPortClose : NULL;
170*10465441SEvalZero pxMBFrameCBByteReceived = xMBASCIIReceiveFSM;
171*10465441SEvalZero pxMBFrameCBTransmitterEmpty = xMBASCIITransmitFSM;
172*10465441SEvalZero pxMBPortCBTimerExpired = xMBASCIITimerT1SExpired;
173*10465441SEvalZero
174*10465441SEvalZero eStatus = eMBASCIIInit( ucMBAddress, ucPort, ulBaudRate, eParity );
175*10465441SEvalZero break;
176*10465441SEvalZero #endif
177*10465441SEvalZero default:
178*10465441SEvalZero eStatus = MB_EINVAL;
179*10465441SEvalZero break;
180*10465441SEvalZero }
181*10465441SEvalZero
182*10465441SEvalZero if( eStatus == MB_ENOERR )
183*10465441SEvalZero {
184*10465441SEvalZero if( !xMBPortEventInit( ) )
185*10465441SEvalZero {
186*10465441SEvalZero /* port dependent event module initalization failed. */
187*10465441SEvalZero eStatus = MB_EPORTERR;
188*10465441SEvalZero }
189*10465441SEvalZero else
190*10465441SEvalZero {
191*10465441SEvalZero eMBCurrentMode = eMode;
192*10465441SEvalZero eMBState = STATE_DISABLED;
193*10465441SEvalZero }
194*10465441SEvalZero }
195*10465441SEvalZero }
196*10465441SEvalZero return eStatus;
197*10465441SEvalZero }
198*10465441SEvalZero
199*10465441SEvalZero #if MB_SLAVE_TCP_ENABLED > 0
200*10465441SEvalZero eMBErrorCode
eMBTCPInit(USHORT ucTCPPort)201*10465441SEvalZero eMBTCPInit( USHORT ucTCPPort )
202*10465441SEvalZero {
203*10465441SEvalZero eMBErrorCode eStatus = MB_ENOERR;
204*10465441SEvalZero
205*10465441SEvalZero if( ( eStatus = eMBTCPDoInit( ucTCPPort ) ) != MB_ENOERR )
206*10465441SEvalZero {
207*10465441SEvalZero eMBState = STATE_DISABLED;
208*10465441SEvalZero }
209*10465441SEvalZero else if( !xMBPortEventInit( ) )
210*10465441SEvalZero {
211*10465441SEvalZero /* Port dependent event module initalization failed. */
212*10465441SEvalZero eStatus = MB_EPORTERR;
213*10465441SEvalZero }
214*10465441SEvalZero else
215*10465441SEvalZero {
216*10465441SEvalZero pvMBFrameStartCur = eMBTCPStart;
217*10465441SEvalZero pvMBFrameStopCur = eMBTCPStop;
218*10465441SEvalZero peMBFrameReceiveCur = eMBTCPReceive;
219*10465441SEvalZero peMBFrameSendCur = eMBTCPSend;
220*10465441SEvalZero pvMBFrameCloseCur = MB_PORT_HAS_CLOSE ? vMBTCPPortClose : NULL;
221*10465441SEvalZero ucMBAddress = MB_TCP_PSEUDO_ADDRESS;
222*10465441SEvalZero eMBCurrentMode = MB_TCP;
223*10465441SEvalZero eMBState = STATE_DISABLED;
224*10465441SEvalZero }
225*10465441SEvalZero return eStatus;
226*10465441SEvalZero }
227*10465441SEvalZero #endif
228*10465441SEvalZero
229*10465441SEvalZero eMBErrorCode
eMBRegisterCB(UCHAR ucFunctionCode,pxMBFunctionHandler pxHandler)230*10465441SEvalZero eMBRegisterCB( UCHAR ucFunctionCode, pxMBFunctionHandler pxHandler )
231*10465441SEvalZero {
232*10465441SEvalZero int i;
233*10465441SEvalZero eMBErrorCode eStatus;
234*10465441SEvalZero
235*10465441SEvalZero if( ( 0 < ucFunctionCode ) && ( ucFunctionCode <= 127 ) )
236*10465441SEvalZero {
237*10465441SEvalZero ENTER_CRITICAL_SECTION( );
238*10465441SEvalZero if( pxHandler != NULL )
239*10465441SEvalZero {
240*10465441SEvalZero for( i = 0; i < MB_FUNC_HANDLERS_MAX; i++ )
241*10465441SEvalZero {
242*10465441SEvalZero if( ( xFuncHandlers[i].pxHandler == NULL ) ||
243*10465441SEvalZero ( xFuncHandlers[i].pxHandler == pxHandler ) )
244*10465441SEvalZero {
245*10465441SEvalZero xFuncHandlers[i].ucFunctionCode = ucFunctionCode;
246*10465441SEvalZero xFuncHandlers[i].pxHandler = pxHandler;
247*10465441SEvalZero break;
248*10465441SEvalZero }
249*10465441SEvalZero }
250*10465441SEvalZero eStatus = ( i != MB_FUNC_HANDLERS_MAX ) ? MB_ENOERR : MB_ENORES;
251*10465441SEvalZero }
252*10465441SEvalZero else
253*10465441SEvalZero {
254*10465441SEvalZero for( i = 0; i < MB_FUNC_HANDLERS_MAX; i++ )
255*10465441SEvalZero {
256*10465441SEvalZero if( xFuncHandlers[i].ucFunctionCode == ucFunctionCode )
257*10465441SEvalZero {
258*10465441SEvalZero xFuncHandlers[i].ucFunctionCode = 0;
259*10465441SEvalZero xFuncHandlers[i].pxHandler = NULL;
260*10465441SEvalZero break;
261*10465441SEvalZero }
262*10465441SEvalZero }
263*10465441SEvalZero /* Remove can't fail. */
264*10465441SEvalZero eStatus = MB_ENOERR;
265*10465441SEvalZero }
266*10465441SEvalZero EXIT_CRITICAL_SECTION( );
267*10465441SEvalZero }
268*10465441SEvalZero else
269*10465441SEvalZero {
270*10465441SEvalZero eStatus = MB_EINVAL;
271*10465441SEvalZero }
272*10465441SEvalZero return eStatus;
273*10465441SEvalZero }
274*10465441SEvalZero
275*10465441SEvalZero
276*10465441SEvalZero eMBErrorCode
eMBClose(void)277*10465441SEvalZero eMBClose( void )
278*10465441SEvalZero {
279*10465441SEvalZero eMBErrorCode eStatus = MB_ENOERR;
280*10465441SEvalZero
281*10465441SEvalZero if( eMBState == STATE_DISABLED )
282*10465441SEvalZero {
283*10465441SEvalZero if( pvMBFrameCloseCur != NULL )
284*10465441SEvalZero {
285*10465441SEvalZero pvMBFrameCloseCur( );
286*10465441SEvalZero }
287*10465441SEvalZero }
288*10465441SEvalZero else
289*10465441SEvalZero {
290*10465441SEvalZero eStatus = MB_EILLSTATE;
291*10465441SEvalZero }
292*10465441SEvalZero return eStatus;
293*10465441SEvalZero }
294*10465441SEvalZero
295*10465441SEvalZero
296*10465441SEvalZero eMBErrorCode
eMBEnable(void)297*10465441SEvalZero eMBEnable( void )
298*10465441SEvalZero {
299*10465441SEvalZero eMBErrorCode eStatus = MB_ENOERR;
300*10465441SEvalZero
301*10465441SEvalZero if( eMBState == STATE_DISABLED )
302*10465441SEvalZero {
303*10465441SEvalZero /* Activate the protocol stack. */
304*10465441SEvalZero pvMBFrameStartCur( );
305*10465441SEvalZero eMBState = STATE_ENABLED;
306*10465441SEvalZero }
307*10465441SEvalZero else
308*10465441SEvalZero {
309*10465441SEvalZero eStatus = MB_EILLSTATE;
310*10465441SEvalZero }
311*10465441SEvalZero return eStatus;
312*10465441SEvalZero }
313*10465441SEvalZero
314*10465441SEvalZero eMBErrorCode
eMBDisable(void)315*10465441SEvalZero eMBDisable( void )
316*10465441SEvalZero {
317*10465441SEvalZero eMBErrorCode eStatus;
318*10465441SEvalZero
319*10465441SEvalZero if( eMBState == STATE_ENABLED )
320*10465441SEvalZero {
321*10465441SEvalZero pvMBFrameStopCur( );
322*10465441SEvalZero eMBState = STATE_DISABLED;
323*10465441SEvalZero eStatus = MB_ENOERR;
324*10465441SEvalZero }
325*10465441SEvalZero else if( eMBState == STATE_DISABLED )
326*10465441SEvalZero {
327*10465441SEvalZero eStatus = MB_ENOERR;
328*10465441SEvalZero }
329*10465441SEvalZero else
330*10465441SEvalZero {
331*10465441SEvalZero eStatus = MB_EILLSTATE;
332*10465441SEvalZero }
333*10465441SEvalZero return eStatus;
334*10465441SEvalZero }
335*10465441SEvalZero
eMBPoll(void)336*10465441SEvalZero eMBErrorCode eMBPoll( void )
337*10465441SEvalZero {
338*10465441SEvalZero static UCHAR *ucMBFrame;
339*10465441SEvalZero static UCHAR ucRcvAddress;
340*10465441SEvalZero static UCHAR ucFunctionCode;
341*10465441SEvalZero static USHORT usLength;
342*10465441SEvalZero static eMBException eException;
343*10465441SEvalZero
344*10465441SEvalZero int i;
345*10465441SEvalZero eMBErrorCode eStatus = MB_ENOERR;
346*10465441SEvalZero eMBEventType eEvent;
347*10465441SEvalZero
348*10465441SEvalZero /* Check if the protocol stack is ready. */
349*10465441SEvalZero if( eMBState != STATE_ENABLED )
350*10465441SEvalZero {
351*10465441SEvalZero return MB_EILLSTATE;
352*10465441SEvalZero }
353*10465441SEvalZero
354*10465441SEvalZero /* Check if there is a event available. If not return control to caller.
355*10465441SEvalZero * Otherwise we will handle the event. */
356*10465441SEvalZero if( xMBPortEventGet( &eEvent ) == TRUE )
357*10465441SEvalZero {
358*10465441SEvalZero switch ( eEvent )
359*10465441SEvalZero {
360*10465441SEvalZero case EV_READY:
361*10465441SEvalZero break;
362*10465441SEvalZero
363*10465441SEvalZero case EV_FRAME_RECEIVED:
364*10465441SEvalZero eStatus = peMBFrameReceiveCur( &ucRcvAddress, &ucMBFrame, &usLength );
365*10465441SEvalZero if( eStatus == MB_ENOERR )
366*10465441SEvalZero {
367*10465441SEvalZero /* Check if the frame is for us. If not ignore the frame. */
368*10465441SEvalZero if( ( ucRcvAddress == ucMBAddress ) || ( ucRcvAddress == MB_ADDRESS_BROADCAST ) )
369*10465441SEvalZero {
370*10465441SEvalZero ( void )xMBPortEventPost( EV_EXECUTE );
371*10465441SEvalZero }
372*10465441SEvalZero }
373*10465441SEvalZero break;
374*10465441SEvalZero
375*10465441SEvalZero case EV_EXECUTE:
376*10465441SEvalZero ucFunctionCode = ucMBFrame[MB_PDU_FUNC_OFF];
377*10465441SEvalZero eException = MB_EX_ILLEGAL_FUNCTION;
378*10465441SEvalZero for( i = 0; i < MB_FUNC_HANDLERS_MAX; i++ )
379*10465441SEvalZero {
380*10465441SEvalZero /* No more function handlers registered. Abort. */
381*10465441SEvalZero if( xFuncHandlers[i].ucFunctionCode == 0 )
382*10465441SEvalZero {
383*10465441SEvalZero break;
384*10465441SEvalZero }
385*10465441SEvalZero else if( xFuncHandlers[i].ucFunctionCode == ucFunctionCode )
386*10465441SEvalZero {
387*10465441SEvalZero eException = xFuncHandlers[i].pxHandler( ucMBFrame, &usLength );
388*10465441SEvalZero break;
389*10465441SEvalZero }
390*10465441SEvalZero }
391*10465441SEvalZero
392*10465441SEvalZero /* If the request was not sent to the broadcast address we
393*10465441SEvalZero * return a reply. */
394*10465441SEvalZero if( ucRcvAddress != MB_ADDRESS_BROADCAST )
395*10465441SEvalZero {
396*10465441SEvalZero if( eException != MB_EX_NONE )
397*10465441SEvalZero {
398*10465441SEvalZero /* An exception occured. Build an error frame. */
399*10465441SEvalZero usLength = 0;
400*10465441SEvalZero ucMBFrame[usLength++] = ( UCHAR )( ucFunctionCode | MB_FUNC_ERROR );
401*10465441SEvalZero ucMBFrame[usLength++] = eException;
402*10465441SEvalZero }
403*10465441SEvalZero eStatus = peMBFrameSendCur( ucMBAddress, ucMBFrame, usLength );
404*10465441SEvalZero }
405*10465441SEvalZero break;
406*10465441SEvalZero
407*10465441SEvalZero case EV_FRAME_SENT:
408*10465441SEvalZero break;
409*10465441SEvalZero }
410*10465441SEvalZero }
411*10465441SEvalZero return MB_ENOERR;
412*10465441SEvalZero }
413