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 ******************************************************************************
 
  28 * 03-01-01 Marc Boucher <marc@mbsi.ca>
 
  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.
 
  54  * Randomize fsm id on link/init.
 
  55  * Deal with variable outgoing MTU.
 
  60 #if PPP_SUPPORT /* don't build if not configured for use in lwipopts.h */
 
  68 /*************************/
 
  69 /*** LOCAL DEFINITIONS ***/
 
  70 /*************************/
 
  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.
 
 132   f->state = LS_INITIAL;
 
 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.
 
 149   int oldState = f->state;
 
 151   LWIP_UNUSED_ARG(oldState);
 
 155       f->state = LS_CLOSED;
 
 159       if( f->flags & OPT_SILENT ) {
 
 160         f->state = LS_STOPPED;
 
 162         /* Send an initial configure-request */
 
 164         f->state = LS_REQSENT;
 
 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.
 
 184 fsm_lowerdown(fsm *f)
 
 186   int oldState = f->state;
 
 188   LWIP_UNUSED_ARG(oldState);
 
 192       f->state = LS_INITIAL;
 
 196       f->state = LS_STARTING;
 
 197       if( f->callbacks->starting ) {
 
 198         (*f->callbacks->starting)(f);
 
 203       f->state = LS_INITIAL;
 
 204       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 211       f->state = LS_STARTING;
 
 212       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 216       if( f->callbacks->down ) {
 
 217         (*f->callbacks->down)(f);
 
 219       f->state = LS_STARTING;
 
 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.
 
 238   int oldState = f->state;
 
 240   LWIP_UNUSED_ARG(oldState);
 
 244       f->state = LS_STARTING;
 
 245       if( f->callbacks->starting ) {
 
 246         (*f->callbacks->starting)(f);
 
 251       if( f->flags & OPT_SILENT ) {
 
 252         f->state = LS_STOPPED;
 
 254         /* Send an initial configure-request */
 
 256         f->state = LS_REQSENT;
 
 261       f->state = LS_STOPPING;
 
 265       if( f->flags & OPT_RESTART ) {
 
 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.
 
 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));
 
 294       f->state = LS_INITIAL;
 
 297       f->state = LS_CLOSED;
 
 300       f->state = LS_CLOSING;
 
 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);
 
 319       f->state = LS_CLOSING;
 
 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.
 
 334 fsm_sdata( fsm *f, u_char code, u_char id, u_char *data, int datalen)
 
 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);
 
 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.
 
 362 fsm_input(fsm *f, u_char *inpacket, int l)
 
 364   u_char *inp = inpacket;
 
 369   * Parse header (code, id and length).
 
 370   * If packet too short, drop it.
 
 373     FSMDEBUG((LOG_WARNING, "fsm_input(%x): Rcvd short header.\n",
 
 380   if (len < HEADERLEN) {
 
 381     FSMDEBUG((LOG_INFO, "fsm_input(%x): Rcvd illegal length.\n",
 
 386     FSMDEBUG((LOG_INFO, "fsm_input(%x): Rcvd short packet.\n",
 
 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]));
 
 397   FSMDEBUG((LOG_INFO, "fsm_input(%s):%d,%d,%d\n", PROTO_NAME(f), code, id, l));
 
 399    * Action depends on code.
 
 403       fsm_rconfreq(f, id, inp, len);
 
 407       fsm_rconfack(f, id, inp, len);
 
 412       fsm_rconfnakrej(f, code, id, inp, len);
 
 416       fsm_rtermreq(f, id, inp, len);
 
 424       fsm_rcoderej(f, inp, len);
 
 428       if( !f->callbacks->extcode ||
 
 429           !(*f->callbacks->extcode)(f, code, id, inp, len) ) {
 
 430         fsm_sdata(f, CODEREJ, ++f->id, inpacket, len + HEADERLEN);
 
 438  * fsm_protreject - Peer doesn't speak this protocol.
 
 440  * Treat this as a catastrophic error (RXJ-).
 
 443 fsm_protreject(fsm *f)
 
 447       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 450       f->state = LS_CLOSED;
 
 451       if( f->callbacks->finished ) {
 
 452         (*f->callbacks->finished)(f);
 
 460       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 463       f->state = LS_STOPPED;
 
 464       if( f->callbacks->finished ) {
 
 465         (*f->callbacks->finished)(f);
 
 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);
 
 480       f->state = LS_STOPPING;
 
 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.
 
 501 fsm_timeout(void *arg)
 
 503   fsm *f = (fsm *) arg;
 
 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);
 
 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);
 
 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);
 
 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;
 
 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.
 
 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]));
 
 572       /* Go away, we're closed */
 
 573       fsm_sdata(f, TERMACK, id, NULL, 0);
 
 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 */
 
 588       /* Negotiation started by our peer */
 
 589       fsm_sconfreq(f, 0);    /* Send initial Configure-Request */
 
 590       f->state = LS_REQSENT;
 
 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);
 
 602     code = CONFREJ;      /* Reject all CI */
 
 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 */
 
 618       f->state = LS_ACKSENT;
 
 622     /* we sent CONFACK or CONFREJ */
 
 623     if (f->state != LS_ACKRCVD) {
 
 624       f->state = LS_REQSENT;
 
 626     if( code == CONFNAK ) {
 
 634  * fsm_rconfack - Receive Configure-Ack.
 
 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));
 
 656       fsm_sdata(f, TERMACK, (u_char)id, NULL, 0);
 
 660       f->state = LS_ACKRCVD;
 
 661       f->retransmits = f->maxconfreqtransmits;
 
 665       /* Huh? an extra valid Ack? oh well... */
 
 666       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 668       f->state = LS_REQSENT;
 
 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 */
 
 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;
 
 693  * fsm_rconfnakrej - Receive Configure-Nak or Configure-Reject.
 
 696 fsm_rconfnakrej(fsm *f, int code, int id, u_char *inp, int len)
 
 698   int (*proc) (fsm *, u_char *, int);
 
 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));
 
 719       fsm_sdata(f, TERMACK, (u_char)id, NULL, 0);
 
 724       /* They didn't agree to what we wanted - try another request */
 
 725       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 727         f->state = LS_STOPPED;    /* kludge for stopping CCP */
 
 729         fsm_sconfreq(f, 0);    /* Send Configure-Request */
 
 734       /* Got a Nak/reject when we had already had an Ack?? oh well... */
 
 735       UNTIMEOUT(fsm_timeout, f);  /* Cancel timeout */
 
 737       f->state = LS_REQSENT;
 
 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;
 
 753  * fsm_rtermreq - Receive Terminate-Req.
 
 756 fsm_rtermreq(fsm *f, int id, u_char *p, int len)
 
 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]));
 
 766       f->state = LS_REQSENT;    /* Start over but keep trying */
 
 771         FSMDEBUG((LOG_INFO, "%s terminated by peer (%x)\n", PROTO_NAME(f), p));
 
 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 */
 
 779       f->state = LS_STOPPING;
 
 780       TIMEOUT(fsm_timeout, f, f->timeouttime);
 
 784   fsm_sdata(f, TERMACK, (u_char)id, NULL, 0);
 
 789  * fsm_rtermack - Receive Terminate-Ack.
 
 794   FSMDEBUG((LOG_INFO, "fsm_rtermack(%s): state=%d (%s)\n", 
 
 795         PROTO_NAME(f), f->state, ppperr_strerr[f->state]));
 
 799       UNTIMEOUT(fsm_timeout, f);
 
 800       f->state = LS_CLOSED;
 
 801       if( f->callbacks->finished ) {
 
 802         (*f->callbacks->finished)(f);
 
 807       UNTIMEOUT(fsm_timeout, f);
 
 808       f->state = LS_STOPPED;
 
 809       if( f->callbacks->finished ) {
 
 810         (*f->callbacks->finished)(f);
 
 815       f->state = LS_REQSENT;
 
 819       if (f->callbacks->down) {
 
 820         (*f->callbacks->down)(f);  /* Inform upper layers */
 
 829  * fsm_rcoderej - Receive an Code-Reject.
 
 832 fsm_rcoderej(fsm *f, u_char *inp, int len)
 
 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"));
 
 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.
 
 858 fsm_sconfreq(fsm *f, int retransmit)
 
 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);
 
 872     /* New request - reset retransmission counter, use new ID */
 
 873     f->retransmits = f->maxconfreqtransmits;
 
 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);
 
 895   /* send the request to our peer */
 
 896   fsm_sdata(f, CONFREQ, f->reqid, outp, cilen);
 
 898   /* start the retransmit timer */
 
 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 */