FreeRTOS
[armadillo_firmware.git] / FreeRTOS / Common / ethernet / lwIP_130 / src / netif / ppp / fsm.c
blob8c2fab4a57c246de404e7f7037f869196d416d6b
1 /*****************************************************************************
2 * fsm.c - Network Control Protocol Finite State Machine program file.
4 * Copyright (c) 2003 by Marc Boucher, Services Informatiques (MBSI) inc.
5 * portions Copyright (c) 1997 by Global Election Systems Inc.
7 * The authors hereby grant permission to use, copy, modify, distribute,
8 * and license this software and its documentation for any purpose, provided
9 * that existing copyright notices are retained in all copies and that this
10 * notice and the following disclaimer are included verbatim in any
11 * distributions. No written agreement, license, or royalty fee is required
12 * for any of the authorized uses.
14 * THIS SOFTWARE IS PROVIDED BY THE CONTRIBUTORS *AS IS* AND ANY EXPRESS OR
15 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
16 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
17 * IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
18 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
19 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
20 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
21 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
23 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25 ******************************************************************************
26 * REVISION HISTORY
28 * 03-01-01 Marc Boucher <marc@mbsi.ca>
29 * Ported to lwIP.
30 * 97-12-01 Guy Lancaster <lancasterg@acm.org>, Global Election Systems Inc.
31 * Original based on BSD fsm.c.
32 *****************************************************************************/
34 * fsm.c - {Link, IP} Control Protocol Finite State Machine.
36 * Copyright (c) 1989 Carnegie Mellon University.
37 * All rights reserved.
39 * Redistribution and use in source and binary forms are permitted
40 * provided that the above copyright notice and this paragraph are
41 * duplicated in all such forms and that any documentation,
42 * advertising materials, and other materials related to such
43 * distribution and use acknowledge that the software was developed
44 * by Carnegie Mellon University. The name of the
45 * University may not be used to endorse or promote products derived
46 * from this software without specific prior written permission.
47 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
48 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
49 * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
53 * TODO:
54 * Randomize fsm id on link/init.
55 * Deal with variable outgoing MTU.
58 #include "lwip/opt.h"
60 #if PPP_SUPPORT /* don't build if not configured for use in lwipopts.h */
62 #include "ppp.h"
63 #include "pppdebug.h"
65 #include "fsm.h"
68 /*************************/
69 /*** LOCAL DEFINITIONS ***/
70 /*************************/
72 #if PPP_DEBUG
74 static const char *ppperr_strerr[] = {
75 "LS_INITIAL", /* LS_INITIAL 0 */
76 "LS_STARTING", /* LS_STARTING 1 */
77 "LS_CLOSED", /* LS_CLOSED 2 */
78 "LS_STOPPED", /* LS_STOPPED 3 */
79 "LS_CLOSING", /* LS_CLOSING 4 */
80 "LS_STOPPING", /* LS_STOPPING 5 */
81 "LS_REQSENT", /* LS_REQSENT 6 */
82 "LS_ACKRCVD", /* LS_ACKRCVD 7 */
83 "LS_ACKSENT", /* LS_ACKSENT 8 */
84 "LS_OPENED" /* LS_OPENED 9 */
87 #endif /* PPP_DEBUG */
89 /************************/
90 /*** LOCAL DATA TYPES ***/
91 /************************/
94 /***********************************/
95 /*** LOCAL FUNCTION DECLARATIONS ***/
96 /***********************************/
97 static void fsm_timeout (void *);
98 static void fsm_rconfreq (fsm *, u_char, u_char *, int);
99 static void fsm_rconfack (fsm *, int, u_char *, int);
100 static void fsm_rconfnakrej (fsm *, int, int, u_char *, int);
101 static void fsm_rtermreq (fsm *, int, u_char *, int);
102 static void fsm_rtermack (fsm *);
103 static void fsm_rcoderej (fsm *, u_char *, int);
104 static void fsm_sconfreq (fsm *, int);
106 #define PROTO_NAME(f) ((f)->callbacks->proto_name)
109 /******************************/
110 /*** PUBLIC DATA STRUCTURES ***/
111 /******************************/
114 /*****************************/
115 /*** LOCAL DATA STRUCTURES ***/
116 /*****************************/
117 int peer_mru[NUM_PPP];
120 /***********************************/
121 /*** PUBLIC FUNCTION DEFINITIONS ***/
122 /***********************************/
125 * fsm_init - Initialize fsm.
127 * Initialize fsm state.
129 void
130 fsm_init(fsm *f)
132 f->state = LS_INITIAL;
133 f->flags = 0;
134 f->id = 0; /* XXX Start with random id? */
135 f->timeouttime = FSM_DEFTIMEOUT;
136 f->maxconfreqtransmits = FSM_DEFMAXCONFREQS;
137 f->maxtermtransmits = FSM_DEFMAXTERMREQS;
138 f->maxnakloops = FSM_DEFMAXNAKLOOPS;
139 f->term_reason_len = 0;
144 * fsm_lowerup - The lower layer is up.
146 void
147 fsm_lowerup(fsm *f)
149 int oldState = f->state;
151 LWIP_UNUSED_ARG(oldState);
153 switch( f->state ) {
154 case LS_INITIAL:
155 f->state = LS_CLOSED;
156 break;
158 case LS_STARTING:
159 if( f->flags & OPT_SILENT ) {
160 f->state = LS_STOPPED;
161 } else {
162 /* Send an initial configure-request */
163 fsm_sconfreq(f, 0);
164 f->state = LS_REQSENT;
166 break;
168 default:
169 FSMDEBUG((LOG_INFO, "%s: Up event in state %d (%s)!\n",
170 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
173 FSMDEBUG((LOG_INFO, "%s: lowerup state %d (%s) -> %d (%s)\n",
174 PROTO_NAME(f), oldState, ppperr_strerr[oldState], f->state, ppperr_strerr[f->state]));
179 * fsm_lowerdown - The lower layer is down.
181 * Cancel all timeouts and inform upper layers.
183 void
184 fsm_lowerdown(fsm *f)
186 int oldState = f->state;
188 LWIP_UNUSED_ARG(oldState);
190 switch( f->state ) {
191 case LS_CLOSED:
192 f->state = LS_INITIAL;
193 break;
195 case LS_STOPPED:
196 f->state = LS_STARTING;
197 if( f->callbacks->starting ) {
198 (*f->callbacks->starting)(f);
200 break;
202 case LS_CLOSING:
203 f->state = LS_INITIAL;
204 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
205 break;
207 case LS_STOPPING:
208 case LS_REQSENT:
209 case LS_ACKRCVD:
210 case LS_ACKSENT:
211 f->state = LS_STARTING;
212 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
213 break;
215 case LS_OPENED:
216 if( f->callbacks->down ) {
217 (*f->callbacks->down)(f);
219 f->state = LS_STARTING;
220 break;
222 default:
223 FSMDEBUG((LOG_INFO, "%s: Down event in state %d (%s)!\n",
224 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
227 FSMDEBUG((LOG_INFO, "%s: lowerdown state %d (%s) -> %d (%s)\n",
228 PROTO_NAME(f), oldState, ppperr_strerr[oldState], f->state, ppperr_strerr[f->state]));
233 * fsm_open - Link is allowed to come up.
235 void
236 fsm_open(fsm *f)
238 int oldState = f->state;
240 LWIP_UNUSED_ARG(oldState);
242 switch( f->state ) {
243 case LS_INITIAL:
244 f->state = LS_STARTING;
245 if( f->callbacks->starting ) {
246 (*f->callbacks->starting)(f);
248 break;
250 case LS_CLOSED:
251 if( f->flags & OPT_SILENT ) {
252 f->state = LS_STOPPED;
253 } else {
254 /* Send an initial configure-request */
255 fsm_sconfreq(f, 0);
256 f->state = LS_REQSENT;
258 break;
260 case LS_CLOSING:
261 f->state = LS_STOPPING;
262 /* fall through */
263 case LS_STOPPED:
264 case LS_OPENED:
265 if( f->flags & OPT_RESTART ) {
266 fsm_lowerdown(f);
267 fsm_lowerup(f);
269 break;
272 FSMDEBUG((LOG_INFO, "%s: open state %d (%s) -> %d (%s)\n",
273 PROTO_NAME(f), oldState, ppperr_strerr[oldState], f->state, ppperr_strerr[f->state]));
278 * fsm_close - Start closing connection.
280 * Cancel timeouts and either initiate close or possibly go directly to
281 * the LS_CLOSED state.
283 void
284 fsm_close(fsm *f, char *reason)
286 int oldState = f->state;
288 LWIP_UNUSED_ARG(oldState);
290 f->term_reason = reason;
291 f->term_reason_len = (reason == NULL? 0: strlen(reason));
292 switch( f->state ) {
293 case LS_STARTING:
294 f->state = LS_INITIAL;
295 break;
296 case LS_STOPPED:
297 f->state = LS_CLOSED;
298 break;
299 case LS_STOPPING:
300 f->state = LS_CLOSING;
301 break;
303 case LS_REQSENT:
304 case LS_ACKRCVD:
305 case LS_ACKSENT:
306 case LS_OPENED:
307 if( f->state != LS_OPENED ) {
308 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
309 } else if( f->callbacks->down ) {
310 (*f->callbacks->down)(f); /* Inform upper layers we're down */
312 /* Init restart counter, send Terminate-Request */
313 f->retransmits = f->maxtermtransmits;
314 fsm_sdata(f, TERMREQ, f->reqid = ++f->id,
315 (u_char *) f->term_reason, f->term_reason_len);
316 TIMEOUT(fsm_timeout, f, f->timeouttime);
317 --f->retransmits;
319 f->state = LS_CLOSING;
320 break;
323 FSMDEBUG((LOG_INFO, "%s: close reason=%s state %d (%s) -> %d (%s)\n",
324 PROTO_NAME(f), reason, oldState, ppperr_strerr[oldState], f->state, ppperr_strerr[f->state]));
329 * fsm_sdata - Send some data.
331 * Used for all packets sent to our peer by this module.
333 void
334 fsm_sdata( fsm *f, u_char code, u_char id, u_char *data, int datalen)
336 u_char *outp;
337 int outlen;
339 /* Adjust length to be smaller than MTU */
340 outp = outpacket_buf[f->unit];
341 if (datalen > peer_mru[f->unit] - (int)HEADERLEN) {
342 datalen = peer_mru[f->unit] - HEADERLEN;
344 if (datalen && data != outp + PPP_HDRLEN + HEADERLEN) {
345 BCOPY(data, outp + PPP_HDRLEN + HEADERLEN, datalen);
347 outlen = datalen + HEADERLEN;
348 MAKEHEADER(outp, f->protocol);
349 PUTCHAR(code, outp);
350 PUTCHAR(id, outp);
351 PUTSHORT(outlen, outp);
352 pppWrite(f->unit, outpacket_buf[f->unit], outlen + PPP_HDRLEN);
353 FSMDEBUG((LOG_INFO, "fsm_sdata(%s): Sent code %d,%d,%d.\n",
354 PROTO_NAME(f), code, id, outlen));
359 * fsm_input - Input packet.
361 void
362 fsm_input(fsm *f, u_char *inpacket, int l)
364 u_char *inp = inpacket;
365 u_char code, id;
366 int len;
369 * Parse header (code, id and length).
370 * If packet too short, drop it.
372 if (l < HEADERLEN) {
373 FSMDEBUG((LOG_WARNING, "fsm_input(%x): Rcvd short header.\n",
374 f->protocol));
375 return;
377 GETCHAR(code, inp);
378 GETCHAR(id, inp);
379 GETSHORT(len, inp);
380 if (len < HEADERLEN) {
381 FSMDEBUG((LOG_INFO, "fsm_input(%x): Rcvd illegal length.\n",
382 f->protocol));
383 return;
385 if (len > l) {
386 FSMDEBUG((LOG_INFO, "fsm_input(%x): Rcvd short packet.\n",
387 f->protocol));
388 return;
390 len -= HEADERLEN; /* subtract header length */
392 if( f->state == LS_INITIAL || f->state == LS_STARTING ) {
393 FSMDEBUG((LOG_INFO, "fsm_input(%x): Rcvd packet in state %d (%s).\n",
394 f->protocol, f->state, ppperr_strerr[f->state]));
395 return;
397 FSMDEBUG((LOG_INFO, "fsm_input(%s):%d,%d,%d\n", PROTO_NAME(f), code, id, l));
399 * Action depends on code.
401 switch (code) {
402 case CONFREQ:
403 fsm_rconfreq(f, id, inp, len);
404 break;
406 case CONFACK:
407 fsm_rconfack(f, id, inp, len);
408 break;
410 case CONFNAK:
411 case CONFREJ:
412 fsm_rconfnakrej(f, code, id, inp, len);
413 break;
415 case TERMREQ:
416 fsm_rtermreq(f, id, inp, len);
417 break;
419 case TERMACK:
420 fsm_rtermack(f);
421 break;
423 case CODEREJ:
424 fsm_rcoderej(f, inp, len);
425 break;
427 default:
428 if( !f->callbacks->extcode ||
429 !(*f->callbacks->extcode)(f, code, id, inp, len) ) {
430 fsm_sdata(f, CODEREJ, ++f->id, inpacket, len + HEADERLEN);
432 break;
438 * fsm_protreject - Peer doesn't speak this protocol.
440 * Treat this as a catastrophic error (RXJ-).
442 void
443 fsm_protreject(fsm *f)
445 switch( f->state ) {
446 case LS_CLOSING:
447 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
448 /* fall through */
449 case LS_CLOSED:
450 f->state = LS_CLOSED;
451 if( f->callbacks->finished ) {
452 (*f->callbacks->finished)(f);
454 break;
456 case LS_STOPPING:
457 case LS_REQSENT:
458 case LS_ACKRCVD:
459 case LS_ACKSENT:
460 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
461 /* fall through */
462 case LS_STOPPED:
463 f->state = LS_STOPPED;
464 if( f->callbacks->finished ) {
465 (*f->callbacks->finished)(f);
467 break;
469 case LS_OPENED:
470 if( f->callbacks->down ) {
471 (*f->callbacks->down)(f);
473 /* Init restart counter, send Terminate-Request */
474 f->retransmits = f->maxtermtransmits;
475 fsm_sdata(f, TERMREQ, f->reqid = ++f->id,
476 (u_char *) f->term_reason, f->term_reason_len);
477 TIMEOUT(fsm_timeout, f, f->timeouttime);
478 --f->retransmits;
480 f->state = LS_STOPPING;
481 break;
483 default:
484 FSMDEBUG((LOG_INFO, "%s: Protocol-reject event in state %d (%s)!\n",
485 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
493 /**********************************/
494 /*** LOCAL FUNCTION DEFINITIONS ***/
495 /**********************************/
498 * fsm_timeout - Timeout expired.
500 static void
501 fsm_timeout(void *arg)
503 fsm *f = (fsm *) arg;
505 switch (f->state) {
506 case LS_CLOSING:
507 case LS_STOPPING:
508 if( f->retransmits <= 0 ) {
509 FSMDEBUG((LOG_WARNING, "%s: timeout sending Terminate-Request state=%d (%s)\n",
510 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
512 * We've waited for an ack long enough. Peer probably heard us.
514 f->state = (f->state == LS_CLOSING)? LS_CLOSED: LS_STOPPED;
515 if( f->callbacks->finished ) {
516 (*f->callbacks->finished)(f);
518 } else {
519 FSMDEBUG((LOG_WARNING, "%s: timeout resending Terminate-Requests state=%d (%s)\n",
520 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
521 /* Send Terminate-Request */
522 fsm_sdata(f, TERMREQ, f->reqid = ++f->id,
523 (u_char *) f->term_reason, f->term_reason_len);
524 TIMEOUT(fsm_timeout, f, f->timeouttime);
525 --f->retransmits;
527 break;
529 case LS_REQSENT:
530 case LS_ACKRCVD:
531 case LS_ACKSENT:
532 if (f->retransmits <= 0) {
533 FSMDEBUG((LOG_WARNING, "%s: timeout sending Config-Requests state=%d (%s)\n",
534 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
535 f->state = LS_STOPPED;
536 if( (f->flags & OPT_PASSIVE) == 0 && f->callbacks->finished ) {
537 (*f->callbacks->finished)(f);
539 } else {
540 FSMDEBUG((LOG_WARNING, "%s: timeout resending Config-Request state=%d (%s)\n",
541 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
542 /* Retransmit the configure-request */
543 if (f->callbacks->retransmit) {
544 (*f->callbacks->retransmit)(f);
546 fsm_sconfreq(f, 1); /* Re-send Configure-Request */
547 if( f->state == LS_ACKRCVD ) {
548 f->state = LS_REQSENT;
551 break;
553 default:
554 FSMDEBUG((LOG_INFO, "%s: Timeout event in state %d (%s)!\n",
555 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
561 * fsm_rconfreq - Receive Configure-Request.
563 static void
564 fsm_rconfreq(fsm *f, u_char id, u_char *inp, int len)
566 int code, reject_if_disagree;
568 FSMDEBUG((LOG_INFO, "fsm_rconfreq(%s): Rcvd id %d state=%d (%s)\n",
569 PROTO_NAME(f), id, f->state, ppperr_strerr[f->state]));
570 switch( f->state ) {
571 case LS_CLOSED:
572 /* Go away, we're closed */
573 fsm_sdata(f, TERMACK, id, NULL, 0);
574 return;
575 case LS_CLOSING:
576 case LS_STOPPING:
577 return;
579 case LS_OPENED:
580 /* Go down and restart negotiation */
581 if( f->callbacks->down ) {
582 (*f->callbacks->down)(f); /* Inform upper layers */
584 fsm_sconfreq(f, 0); /* Send initial Configure-Request */
585 break;
587 case LS_STOPPED:
588 /* Negotiation started by our peer */
589 fsm_sconfreq(f, 0); /* Send initial Configure-Request */
590 f->state = LS_REQSENT;
591 break;
595 * Pass the requested configuration options
596 * to protocol-specific code for checking.
598 if (f->callbacks->reqci) { /* Check CI */
599 reject_if_disagree = (f->nakloops >= f->maxnakloops);
600 code = (*f->callbacks->reqci)(f, inp, &len, reject_if_disagree);
601 } else if (len) {
602 code = CONFREJ; /* Reject all CI */
603 } else {
604 code = CONFACK;
607 /* send the Ack, Nak or Rej to the peer */
608 fsm_sdata(f, (u_char)code, id, inp, len);
610 if (code == CONFACK) {
611 if (f->state == LS_ACKRCVD) {
612 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
613 f->state = LS_OPENED;
614 if (f->callbacks->up) {
615 (*f->callbacks->up)(f); /* Inform upper layers */
617 } else {
618 f->state = LS_ACKSENT;
620 f->nakloops = 0;
621 } else {
622 /* we sent CONFACK or CONFREJ */
623 if (f->state != LS_ACKRCVD) {
624 f->state = LS_REQSENT;
626 if( code == CONFNAK ) {
627 ++f->nakloops;
634 * fsm_rconfack - Receive Configure-Ack.
636 static void
637 fsm_rconfack(fsm *f, int id, u_char *inp, int len)
639 FSMDEBUG((LOG_INFO, "fsm_rconfack(%s): Rcvd id %d state=%d (%s)\n",
640 PROTO_NAME(f), id, f->state, ppperr_strerr[f->state]));
642 if (id != f->reqid || f->seen_ack) { /* Expected id? */
643 return; /* Nope, toss... */
645 if( !(f->callbacks->ackci? (*f->callbacks->ackci)(f, inp, len): (len == 0)) ) {
646 /* Ack is bad - ignore it */
647 FSMDEBUG((LOG_INFO, "%s: received bad Ack (length %d)\n",
648 PROTO_NAME(f), len));
649 return;
651 f->seen_ack = 1;
653 switch (f->state) {
654 case LS_CLOSED:
655 case LS_STOPPED:
656 fsm_sdata(f, TERMACK, (u_char)id, NULL, 0);
657 break;
659 case LS_REQSENT:
660 f->state = LS_ACKRCVD;
661 f->retransmits = f->maxconfreqtransmits;
662 break;
664 case LS_ACKRCVD:
665 /* Huh? an extra valid Ack? oh well... */
666 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
667 fsm_sconfreq(f, 0);
668 f->state = LS_REQSENT;
669 break;
671 case LS_ACKSENT:
672 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
673 f->state = LS_OPENED;
674 f->retransmits = f->maxconfreqtransmits;
675 if (f->callbacks->up) {
676 (*f->callbacks->up)(f); /* Inform upper layers */
678 break;
680 case LS_OPENED:
681 /* Go down and restart negotiation */
682 if (f->callbacks->down) {
683 (*f->callbacks->down)(f); /* Inform upper layers */
685 fsm_sconfreq(f, 0); /* Send initial Configure-Request */
686 f->state = LS_REQSENT;
687 break;
693 * fsm_rconfnakrej - Receive Configure-Nak or Configure-Reject.
695 static void
696 fsm_rconfnakrej(fsm *f, int code, int id, u_char *inp, int len)
698 int (*proc) (fsm *, u_char *, int);
699 int ret;
701 FSMDEBUG((LOG_INFO, "fsm_rconfnakrej(%s): Rcvd id %d state=%d (%s)\n",
702 PROTO_NAME(f), id, f->state, ppperr_strerr[f->state]));
704 if (id != f->reqid || f->seen_ack) { /* Expected id? */
705 return; /* Nope, toss... */
707 proc = (code == CONFNAK)? f->callbacks->nakci: f->callbacks->rejci;
708 if (!proc || !((ret = proc(f, inp, len)))) {
709 /* Nak/reject is bad - ignore it */
710 FSMDEBUG((LOG_INFO, "%s: received bad %s (length %d)\n",
711 PROTO_NAME(f), (code==CONFNAK? "Nak": "reject"), len));
712 return;
714 f->seen_ack = 1;
716 switch (f->state) {
717 case LS_CLOSED:
718 case LS_STOPPED:
719 fsm_sdata(f, TERMACK, (u_char)id, NULL, 0);
720 break;
722 case LS_REQSENT:
723 case LS_ACKSENT:
724 /* They didn't agree to what we wanted - try another request */
725 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
726 if (ret < 0) {
727 f->state = LS_STOPPED; /* kludge for stopping CCP */
728 } else {
729 fsm_sconfreq(f, 0); /* Send Configure-Request */
731 break;
733 case LS_ACKRCVD:
734 /* Got a Nak/reject when we had already had an Ack?? oh well... */
735 UNTIMEOUT(fsm_timeout, f); /* Cancel timeout */
736 fsm_sconfreq(f, 0);
737 f->state = LS_REQSENT;
738 break;
740 case LS_OPENED:
741 /* Go down and restart negotiation */
742 if (f->callbacks->down) {
743 (*f->callbacks->down)(f); /* Inform upper layers */
745 fsm_sconfreq(f, 0); /* Send initial Configure-Request */
746 f->state = LS_REQSENT;
747 break;
753 * fsm_rtermreq - Receive Terminate-Req.
755 static void
756 fsm_rtermreq(fsm *f, int id, u_char *p, int len)
758 LWIP_UNUSED_ARG(p);
760 FSMDEBUG((LOG_INFO, "fsm_rtermreq(%s): Rcvd id %d state=%d (%s)\n",
761 PROTO_NAME(f), id, f->state, ppperr_strerr[f->state]));
763 switch (f->state) {
764 case LS_ACKRCVD:
765 case LS_ACKSENT:
766 f->state = LS_REQSENT; /* Start over but keep trying */
767 break;
769 case LS_OPENED:
770 if (len > 0) {
771 FSMDEBUG((LOG_INFO, "%s terminated by peer (%x)\n", PROTO_NAME(f), p));
772 } else {
773 FSMDEBUG((LOG_INFO, "%s terminated by peer\n", PROTO_NAME(f)));
775 if (f->callbacks->down) {
776 (*f->callbacks->down)(f); /* Inform upper layers */
778 f->retransmits = 0;
779 f->state = LS_STOPPING;
780 TIMEOUT(fsm_timeout, f, f->timeouttime);
781 break;
784 fsm_sdata(f, TERMACK, (u_char)id, NULL, 0);
789 * fsm_rtermack - Receive Terminate-Ack.
791 static void
792 fsm_rtermack(fsm *f)
794 FSMDEBUG((LOG_INFO, "fsm_rtermack(%s): state=%d (%s)\n",
795 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
797 switch (f->state) {
798 case LS_CLOSING:
799 UNTIMEOUT(fsm_timeout, f);
800 f->state = LS_CLOSED;
801 if( f->callbacks->finished ) {
802 (*f->callbacks->finished)(f);
804 break;
806 case LS_STOPPING:
807 UNTIMEOUT(fsm_timeout, f);
808 f->state = LS_STOPPED;
809 if( f->callbacks->finished ) {
810 (*f->callbacks->finished)(f);
812 break;
814 case LS_ACKRCVD:
815 f->state = LS_REQSENT;
816 break;
818 case LS_OPENED:
819 if (f->callbacks->down) {
820 (*f->callbacks->down)(f); /* Inform upper layers */
822 fsm_sconfreq(f, 0);
823 break;
829 * fsm_rcoderej - Receive an Code-Reject.
831 static void
832 fsm_rcoderej(fsm *f, u_char *inp, int len)
834 u_char code, id;
836 FSMDEBUG((LOG_INFO, "fsm_rcoderej(%s): state=%d (%s)\n",
837 PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
839 if (len < HEADERLEN) {
840 FSMDEBUG((LOG_INFO, "fsm_rcoderej: Rcvd short Code-Reject packet!\n"));
841 return;
843 GETCHAR(code, inp);
844 GETCHAR(id, inp);
845 FSMDEBUG((LOG_WARNING, "%s: Rcvd Code-Reject for code %d, id %d\n",
846 PROTO_NAME(f), code, id));
848 if( f->state == LS_ACKRCVD ) {
849 f->state = LS_REQSENT;
855 * fsm_sconfreq - Send a Configure-Request.
857 static void
858 fsm_sconfreq(fsm *f, int retransmit)
860 u_char *outp;
861 int cilen;
863 if( f->state != LS_REQSENT && f->state != LS_ACKRCVD && f->state != LS_ACKSENT ) {
864 /* Not currently negotiating - reset options */
865 if( f->callbacks->resetci ) {
866 (*f->callbacks->resetci)(f);
868 f->nakloops = 0;
871 if( !retransmit ) {
872 /* New request - reset retransmission counter, use new ID */
873 f->retransmits = f->maxconfreqtransmits;
874 f->reqid = ++f->id;
877 f->seen_ack = 0;
880 * Make up the request packet
882 outp = outpacket_buf[f->unit] + PPP_HDRLEN + HEADERLEN;
883 if( f->callbacks->cilen && f->callbacks->addci ) {
884 cilen = (*f->callbacks->cilen)(f);
885 if( cilen > peer_mru[f->unit] - (int)HEADERLEN ) {
886 cilen = peer_mru[f->unit] - HEADERLEN;
888 if (f->callbacks->addci) {
889 (*f->callbacks->addci)(f, outp, &cilen);
891 } else {
892 cilen = 0;
895 /* send the request to our peer */
896 fsm_sdata(f, CONFREQ, f->reqid, outp, cilen);
898 /* start the retransmit timer */
899 --f->retransmits;
900 TIMEOUT(fsm_timeout, f, f->timeouttime);
902 FSMDEBUG((LOG_INFO, "%s: sending Configure-Request, id %d\n",
903 PROTO_NAME(f), f->reqid));
906 #endif /* PPP_SUPPORT */