xref: /nrf52832-nimble/rt-thread/components/net/freemodbus/modbus/mb.c (revision 104654410c56c573564690304ae786df310c91fc)
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