xref: /minix3/minix/lib/liblwip/dist/src/netif/ppp/demand.c (revision 5d5fbe79c1b60734f34c69330aec5496644e8651)
1*5d5fbe79SDavid van Moolenbroek /*
2*5d5fbe79SDavid van Moolenbroek  * demand.c - Support routines for demand-dialling.
3*5d5fbe79SDavid van Moolenbroek  *
4*5d5fbe79SDavid van Moolenbroek  * Copyright (c) 1996-2002 Paul Mackerras. All rights reserved.
5*5d5fbe79SDavid van Moolenbroek  *
6*5d5fbe79SDavid van Moolenbroek  * Redistribution and use in source and binary forms, with or without
7*5d5fbe79SDavid van Moolenbroek  * modification, are permitted provided that the following conditions
8*5d5fbe79SDavid van Moolenbroek  * are met:
9*5d5fbe79SDavid van Moolenbroek  *
10*5d5fbe79SDavid van Moolenbroek  * 1. Redistributions of source code must retain the above copyright
11*5d5fbe79SDavid van Moolenbroek  *    notice, this list of conditions and the following disclaimer.
12*5d5fbe79SDavid van Moolenbroek  *
13*5d5fbe79SDavid van Moolenbroek  * 2. The name(s) of the authors of this software must not be used to
14*5d5fbe79SDavid van Moolenbroek  *    endorse or promote products derived from this software without
15*5d5fbe79SDavid van Moolenbroek  *    prior written permission.
16*5d5fbe79SDavid van Moolenbroek  *
17*5d5fbe79SDavid van Moolenbroek  * 3. Redistributions of any form whatsoever must retain the following
18*5d5fbe79SDavid van Moolenbroek  *    acknowledgment:
19*5d5fbe79SDavid van Moolenbroek  *    "This product includes software developed by Paul Mackerras
20*5d5fbe79SDavid van Moolenbroek  *     <paulus@samba.org>".
21*5d5fbe79SDavid van Moolenbroek  *
22*5d5fbe79SDavid van Moolenbroek  * THE AUTHORS OF THIS SOFTWARE DISCLAIM ALL WARRANTIES WITH REGARD TO
23*5d5fbe79SDavid van Moolenbroek  * THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
24*5d5fbe79SDavid van Moolenbroek  * AND FITNESS, IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY
25*5d5fbe79SDavid van Moolenbroek  * SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
26*5d5fbe79SDavid van Moolenbroek  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN
27*5d5fbe79SDavid van Moolenbroek  * AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
28*5d5fbe79SDavid van Moolenbroek  * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
29*5d5fbe79SDavid van Moolenbroek  */
30*5d5fbe79SDavid van Moolenbroek 
31*5d5fbe79SDavid van Moolenbroek #include "netif/ppp/ppp_opts.h"
32*5d5fbe79SDavid van Moolenbroek #if PPP_SUPPORT && DEMAND_SUPPORT  /* don't build if not configured for use in lwipopts.h */
33*5d5fbe79SDavid van Moolenbroek 
34*5d5fbe79SDavid van Moolenbroek #include <stdio.h>
35*5d5fbe79SDavid van Moolenbroek #include <stdlib.h>
36*5d5fbe79SDavid van Moolenbroek #include <string.h>
37*5d5fbe79SDavid van Moolenbroek #include <errno.h>
38*5d5fbe79SDavid van Moolenbroek #include <fcntl.h>
39*5d5fbe79SDavid van Moolenbroek #include <netdb.h>
40*5d5fbe79SDavid van Moolenbroek #include <unistd.h>
41*5d5fbe79SDavid van Moolenbroek #include <syslog.h>
42*5d5fbe79SDavid van Moolenbroek #include <sys/param.h>
43*5d5fbe79SDavid van Moolenbroek #include <sys/types.h>
44*5d5fbe79SDavid van Moolenbroek #include <sys/wait.h>
45*5d5fbe79SDavid van Moolenbroek #include <sys/time.h>
46*5d5fbe79SDavid van Moolenbroek #include <sys/resource.h>
47*5d5fbe79SDavid van Moolenbroek #include <sys/stat.h>
48*5d5fbe79SDavid van Moolenbroek #include <sys/socket.h>
49*5d5fbe79SDavid van Moolenbroek #include <netinet/in.h>
50*5d5fbe79SDavid van Moolenbroek #include <arpa/inet.h>
51*5d5fbe79SDavid van Moolenbroek #ifdef PPP_FILTER
52*5d5fbe79SDavid van Moolenbroek #include <pcap-bpf.h>
53*5d5fbe79SDavid van Moolenbroek #endif
54*5d5fbe79SDavid van Moolenbroek 
55*5d5fbe79SDavid van Moolenbroek #include "netif/ppp/ppp_impl.h"
56*5d5fbe79SDavid van Moolenbroek 
57*5d5fbe79SDavid van Moolenbroek #include "netif/ppp/fsm.h"
58*5d5fbe79SDavid van Moolenbroek #include "netif/ppp/ipcp.h"
59*5d5fbe79SDavid van Moolenbroek #include "netif/ppp/lcp.h"
60*5d5fbe79SDavid van Moolenbroek 
61*5d5fbe79SDavid van Moolenbroek char *frame;
62*5d5fbe79SDavid van Moolenbroek int framelen;
63*5d5fbe79SDavid van Moolenbroek int framemax;
64*5d5fbe79SDavid van Moolenbroek int escape_flag;
65*5d5fbe79SDavid van Moolenbroek int flush_flag;
66*5d5fbe79SDavid van Moolenbroek int fcs;
67*5d5fbe79SDavid van Moolenbroek 
68*5d5fbe79SDavid van Moolenbroek struct packet {
69*5d5fbe79SDavid van Moolenbroek     int length;
70*5d5fbe79SDavid van Moolenbroek     struct packet *next;
71*5d5fbe79SDavid van Moolenbroek     unsigned char data[1];
72*5d5fbe79SDavid van Moolenbroek };
73*5d5fbe79SDavid van Moolenbroek 
74*5d5fbe79SDavid van Moolenbroek struct packet *pend_q;
75*5d5fbe79SDavid van Moolenbroek struct packet *pend_qtail;
76*5d5fbe79SDavid van Moolenbroek 
77*5d5fbe79SDavid van Moolenbroek static int active_packet (unsigned char *, int);
78*5d5fbe79SDavid van Moolenbroek 
79*5d5fbe79SDavid van Moolenbroek /*
80*5d5fbe79SDavid van Moolenbroek  * demand_conf - configure the interface for doing dial-on-demand.
81*5d5fbe79SDavid van Moolenbroek  */
82*5d5fbe79SDavid van Moolenbroek void
demand_conf()83*5d5fbe79SDavid van Moolenbroek demand_conf()
84*5d5fbe79SDavid van Moolenbroek {
85*5d5fbe79SDavid van Moolenbroek     int i;
86*5d5fbe79SDavid van Moolenbroek     const struct protent *protp;
87*5d5fbe79SDavid van Moolenbroek 
88*5d5fbe79SDavid van Moolenbroek /*    framemax = lcp_allowoptions[0].mru;
89*5d5fbe79SDavid van Moolenbroek     if (framemax < PPP_MRU) */
90*5d5fbe79SDavid van Moolenbroek 	framemax = PPP_MRU;
91*5d5fbe79SDavid van Moolenbroek     framemax += PPP_HDRLEN + PPP_FCSLEN;
92*5d5fbe79SDavid van Moolenbroek     frame = malloc(framemax);
93*5d5fbe79SDavid van Moolenbroek     if (frame == NULL)
94*5d5fbe79SDavid van Moolenbroek 	novm("demand frame");
95*5d5fbe79SDavid van Moolenbroek     framelen = 0;
96*5d5fbe79SDavid van Moolenbroek     pend_q = NULL;
97*5d5fbe79SDavid van Moolenbroek     escape_flag = 0;
98*5d5fbe79SDavid van Moolenbroek     flush_flag = 0;
99*5d5fbe79SDavid van Moolenbroek     fcs = PPP_INITFCS;
100*5d5fbe79SDavid van Moolenbroek 
101*5d5fbe79SDavid van Moolenbroek     netif_set_mtu(pcb, LWIP_MIN(lcp_allowoptions[0].mru, PPP_MRU));
102*5d5fbe79SDavid van Moolenbroek     if (ppp_send_config(pcb, PPP_MRU, (u32_t) 0, 0, 0) < 0
103*5d5fbe79SDavid van Moolenbroek 	|| ppp_recv_config(pcb, PPP_MRU, (u32_t) 0, 0, 0) < 0)
104*5d5fbe79SDavid van Moolenbroek 	    fatal("Couldn't set up demand-dialled PPP interface: %m");
105*5d5fbe79SDavid van Moolenbroek 
106*5d5fbe79SDavid van Moolenbroek #ifdef PPP_FILTER
107*5d5fbe79SDavid van Moolenbroek     set_filters(&pass_filter, &active_filter);
108*5d5fbe79SDavid van Moolenbroek #endif
109*5d5fbe79SDavid van Moolenbroek 
110*5d5fbe79SDavid van Moolenbroek     /*
111*5d5fbe79SDavid van Moolenbroek      * Call the demand_conf procedure for each protocol that's got one.
112*5d5fbe79SDavid van Moolenbroek      */
113*5d5fbe79SDavid van Moolenbroek     for (i = 0; (protp = protocols[i]) != NULL; ++i)
114*5d5fbe79SDavid van Moolenbroek 	if (protp->demand_conf != NULL)
115*5d5fbe79SDavid van Moolenbroek 	    ((*protp->demand_conf)(pcb));
116*5d5fbe79SDavid van Moolenbroek /* FIXME: find a way to die() here */
117*5d5fbe79SDavid van Moolenbroek #if 0
118*5d5fbe79SDavid van Moolenbroek 	    if (!((*protp->demand_conf)(pcb)))
119*5d5fbe79SDavid van Moolenbroek 		die(1);
120*5d5fbe79SDavid van Moolenbroek #endif
121*5d5fbe79SDavid van Moolenbroek }
122*5d5fbe79SDavid van Moolenbroek 
123*5d5fbe79SDavid van Moolenbroek 
124*5d5fbe79SDavid van Moolenbroek /*
125*5d5fbe79SDavid van Moolenbroek  * demand_block - set each network protocol to block further packets.
126*5d5fbe79SDavid van Moolenbroek  */
127*5d5fbe79SDavid van Moolenbroek void
demand_block()128*5d5fbe79SDavid van Moolenbroek demand_block()
129*5d5fbe79SDavid van Moolenbroek {
130*5d5fbe79SDavid van Moolenbroek     int i;
131*5d5fbe79SDavid van Moolenbroek     const struct protent *protp;
132*5d5fbe79SDavid van Moolenbroek 
133*5d5fbe79SDavid van Moolenbroek     for (i = 0; (protp = protocols[i]) != NULL; ++i)
134*5d5fbe79SDavid van Moolenbroek 	if (protp->demand_conf != NULL)
135*5d5fbe79SDavid van Moolenbroek 	    sifnpmode(pcb, protp->protocol & ~0x8000, NPMODE_QUEUE);
136*5d5fbe79SDavid van Moolenbroek     get_loop_output();
137*5d5fbe79SDavid van Moolenbroek }
138*5d5fbe79SDavid van Moolenbroek 
139*5d5fbe79SDavid van Moolenbroek /*
140*5d5fbe79SDavid van Moolenbroek  * demand_discard - set each network protocol to discard packets
141*5d5fbe79SDavid van Moolenbroek  * with an error.
142*5d5fbe79SDavid van Moolenbroek  */
143*5d5fbe79SDavid van Moolenbroek void
demand_discard()144*5d5fbe79SDavid van Moolenbroek demand_discard()
145*5d5fbe79SDavid van Moolenbroek {
146*5d5fbe79SDavid van Moolenbroek     struct packet *pkt, *nextpkt;
147*5d5fbe79SDavid van Moolenbroek     int i;
148*5d5fbe79SDavid van Moolenbroek     const struct protent *protp;
149*5d5fbe79SDavid van Moolenbroek 
150*5d5fbe79SDavid van Moolenbroek     for (i = 0; (protp = protocols[i]) != NULL; ++i)
151*5d5fbe79SDavid van Moolenbroek 	if (protp->demand_conf != NULL)
152*5d5fbe79SDavid van Moolenbroek 	    sifnpmode(pcb, protp->protocol & ~0x8000, NPMODE_ERROR);
153*5d5fbe79SDavid van Moolenbroek     get_loop_output();
154*5d5fbe79SDavid van Moolenbroek 
155*5d5fbe79SDavid van Moolenbroek     /* discard all saved packets */
156*5d5fbe79SDavid van Moolenbroek     for (pkt = pend_q; pkt != NULL; pkt = nextpkt) {
157*5d5fbe79SDavid van Moolenbroek 	nextpkt = pkt->next;
158*5d5fbe79SDavid van Moolenbroek 	free(pkt);
159*5d5fbe79SDavid van Moolenbroek     }
160*5d5fbe79SDavid van Moolenbroek     pend_q = NULL;
161*5d5fbe79SDavid van Moolenbroek     framelen = 0;
162*5d5fbe79SDavid van Moolenbroek     flush_flag = 0;
163*5d5fbe79SDavid van Moolenbroek     escape_flag = 0;
164*5d5fbe79SDavid van Moolenbroek     fcs = PPP_INITFCS;
165*5d5fbe79SDavid van Moolenbroek }
166*5d5fbe79SDavid van Moolenbroek 
167*5d5fbe79SDavid van Moolenbroek /*
168*5d5fbe79SDavid van Moolenbroek  * demand_unblock - set each enabled network protocol to pass packets.
169*5d5fbe79SDavid van Moolenbroek  */
170*5d5fbe79SDavid van Moolenbroek void
demand_unblock()171*5d5fbe79SDavid van Moolenbroek demand_unblock()
172*5d5fbe79SDavid van Moolenbroek {
173*5d5fbe79SDavid van Moolenbroek     int i;
174*5d5fbe79SDavid van Moolenbroek     const struct protent *protp;
175*5d5fbe79SDavid van Moolenbroek 
176*5d5fbe79SDavid van Moolenbroek     for (i = 0; (protp = protocols[i]) != NULL; ++i)
177*5d5fbe79SDavid van Moolenbroek 	if (protp->demand_conf != NULL)
178*5d5fbe79SDavid van Moolenbroek 	    sifnpmode(pcb, protp->protocol & ~0x8000, NPMODE_PASS);
179*5d5fbe79SDavid van Moolenbroek }
180*5d5fbe79SDavid van Moolenbroek 
181*5d5fbe79SDavid van Moolenbroek /*
182*5d5fbe79SDavid van Moolenbroek  * FCS lookup table as calculated by genfcstab.
183*5d5fbe79SDavid van Moolenbroek  */
184*5d5fbe79SDavid van Moolenbroek static u_short fcstab[256] = {
185*5d5fbe79SDavid van Moolenbroek 	0x0000,	0x1189,	0x2312,	0x329b,	0x4624,	0x57ad,	0x6536,	0x74bf,
186*5d5fbe79SDavid van Moolenbroek 	0x8c48,	0x9dc1,	0xaf5a,	0xbed3,	0xca6c,	0xdbe5,	0xe97e,	0xf8f7,
187*5d5fbe79SDavid van Moolenbroek 	0x1081,	0x0108,	0x3393,	0x221a,	0x56a5,	0x472c,	0x75b7,	0x643e,
188*5d5fbe79SDavid van Moolenbroek 	0x9cc9,	0x8d40,	0xbfdb,	0xae52,	0xdaed,	0xcb64,	0xf9ff,	0xe876,
189*5d5fbe79SDavid van Moolenbroek 	0x2102,	0x308b,	0x0210,	0x1399,	0x6726,	0x76af,	0x4434,	0x55bd,
190*5d5fbe79SDavid van Moolenbroek 	0xad4a,	0xbcc3,	0x8e58,	0x9fd1,	0xeb6e,	0xfae7,	0xc87c,	0xd9f5,
191*5d5fbe79SDavid van Moolenbroek 	0x3183,	0x200a,	0x1291,	0x0318,	0x77a7,	0x662e,	0x54b5,	0x453c,
192*5d5fbe79SDavid van Moolenbroek 	0xbdcb,	0xac42,	0x9ed9,	0x8f50,	0xfbef,	0xea66,	0xd8fd,	0xc974,
193*5d5fbe79SDavid van Moolenbroek 	0x4204,	0x538d,	0x6116,	0x709f,	0x0420,	0x15a9,	0x2732,	0x36bb,
194*5d5fbe79SDavid van Moolenbroek 	0xce4c,	0xdfc5,	0xed5e,	0xfcd7,	0x8868,	0x99e1,	0xab7a,	0xbaf3,
195*5d5fbe79SDavid van Moolenbroek 	0x5285,	0x430c,	0x7197,	0x601e,	0x14a1,	0x0528,	0x37b3,	0x263a,
196*5d5fbe79SDavid van Moolenbroek 	0xdecd,	0xcf44,	0xfddf,	0xec56,	0x98e9,	0x8960,	0xbbfb,	0xaa72,
197*5d5fbe79SDavid van Moolenbroek 	0x6306,	0x728f,	0x4014,	0x519d,	0x2522,	0x34ab,	0x0630,	0x17b9,
198*5d5fbe79SDavid van Moolenbroek 	0xef4e,	0xfec7,	0xcc5c,	0xddd5,	0xa96a,	0xb8e3,	0x8a78,	0x9bf1,
199*5d5fbe79SDavid van Moolenbroek 	0x7387,	0x620e,	0x5095,	0x411c,	0x35a3,	0x242a,	0x16b1,	0x0738,
200*5d5fbe79SDavid van Moolenbroek 	0xffcf,	0xee46,	0xdcdd,	0xcd54,	0xb9eb,	0xa862,	0x9af9,	0x8b70,
201*5d5fbe79SDavid van Moolenbroek 	0x8408,	0x9581,	0xa71a,	0xb693,	0xc22c,	0xd3a5,	0xe13e,	0xf0b7,
202*5d5fbe79SDavid van Moolenbroek 	0x0840,	0x19c9,	0x2b52,	0x3adb,	0x4e64,	0x5fed,	0x6d76,	0x7cff,
203*5d5fbe79SDavid van Moolenbroek 	0x9489,	0x8500,	0xb79b,	0xa612,	0xd2ad,	0xc324,	0xf1bf,	0xe036,
204*5d5fbe79SDavid van Moolenbroek 	0x18c1,	0x0948,	0x3bd3,	0x2a5a,	0x5ee5,	0x4f6c,	0x7df7,	0x6c7e,
205*5d5fbe79SDavid van Moolenbroek 	0xa50a,	0xb483,	0x8618,	0x9791,	0xe32e,	0xf2a7,	0xc03c,	0xd1b5,
206*5d5fbe79SDavid van Moolenbroek 	0x2942,	0x38cb,	0x0a50,	0x1bd9,	0x6f66,	0x7eef,	0x4c74,	0x5dfd,
207*5d5fbe79SDavid van Moolenbroek 	0xb58b,	0xa402,	0x9699,	0x8710,	0xf3af,	0xe226,	0xd0bd,	0xc134,
208*5d5fbe79SDavid van Moolenbroek 	0x39c3,	0x284a,	0x1ad1,	0x0b58,	0x7fe7,	0x6e6e,	0x5cf5,	0x4d7c,
209*5d5fbe79SDavid van Moolenbroek 	0xc60c,	0xd785,	0xe51e,	0xf497,	0x8028,	0x91a1,	0xa33a,	0xb2b3,
210*5d5fbe79SDavid van Moolenbroek 	0x4a44,	0x5bcd,	0x6956,	0x78df,	0x0c60,	0x1de9,	0x2f72,	0x3efb,
211*5d5fbe79SDavid van Moolenbroek 	0xd68d,	0xc704,	0xf59f,	0xe416,	0x90a9,	0x8120,	0xb3bb,	0xa232,
212*5d5fbe79SDavid van Moolenbroek 	0x5ac5,	0x4b4c,	0x79d7,	0x685e,	0x1ce1,	0x0d68,	0x3ff3,	0x2e7a,
213*5d5fbe79SDavid van Moolenbroek 	0xe70e,	0xf687,	0xc41c,	0xd595,	0xa12a,	0xb0a3,	0x8238,	0x93b1,
214*5d5fbe79SDavid van Moolenbroek 	0x6b46,	0x7acf,	0x4854,	0x59dd,	0x2d62,	0x3ceb,	0x0e70,	0x1ff9,
215*5d5fbe79SDavid van Moolenbroek 	0xf78f,	0xe606,	0xd49d,	0xc514,	0xb1ab,	0xa022,	0x92b9,	0x8330,
216*5d5fbe79SDavid van Moolenbroek 	0x7bc7,	0x6a4e,	0x58d5,	0x495c,	0x3de3,	0x2c6a,	0x1ef1,	0x0f78
217*5d5fbe79SDavid van Moolenbroek };
218*5d5fbe79SDavid van Moolenbroek 
219*5d5fbe79SDavid van Moolenbroek /*
220*5d5fbe79SDavid van Moolenbroek  * loop_chars - process characters received from the loopback.
221*5d5fbe79SDavid van Moolenbroek  * Calls loop_frame when a complete frame has been accumulated.
222*5d5fbe79SDavid van Moolenbroek  * Return value is 1 if we need to bring up the link, 0 otherwise.
223*5d5fbe79SDavid van Moolenbroek  */
224*5d5fbe79SDavid van Moolenbroek int
loop_chars(p,n)225*5d5fbe79SDavid van Moolenbroek loop_chars(p, n)
226*5d5fbe79SDavid van Moolenbroek     unsigned char *p;
227*5d5fbe79SDavid van Moolenbroek     int n;
228*5d5fbe79SDavid van Moolenbroek {
229*5d5fbe79SDavid van Moolenbroek     int c, rv;
230*5d5fbe79SDavid van Moolenbroek 
231*5d5fbe79SDavid van Moolenbroek     rv = 0;
232*5d5fbe79SDavid van Moolenbroek 
233*5d5fbe79SDavid van Moolenbroek /* check for synchronous connection... */
234*5d5fbe79SDavid van Moolenbroek 
235*5d5fbe79SDavid van Moolenbroek     if ( (p[0] == 0xFF) && (p[1] == 0x03) ) {
236*5d5fbe79SDavid van Moolenbroek         rv = loop_frame(p,n);
237*5d5fbe79SDavid van Moolenbroek         return rv;
238*5d5fbe79SDavid van Moolenbroek     }
239*5d5fbe79SDavid van Moolenbroek 
240*5d5fbe79SDavid van Moolenbroek     for (; n > 0; --n) {
241*5d5fbe79SDavid van Moolenbroek 	c = *p++;
242*5d5fbe79SDavid van Moolenbroek 	if (c == PPP_FLAG) {
243*5d5fbe79SDavid van Moolenbroek 	    if (!escape_flag && !flush_flag
244*5d5fbe79SDavid van Moolenbroek 		&& framelen > 2 && fcs == PPP_GOODFCS) {
245*5d5fbe79SDavid van Moolenbroek 		framelen -= 2;
246*5d5fbe79SDavid van Moolenbroek 		if (loop_frame((unsigned char *)frame, framelen))
247*5d5fbe79SDavid van Moolenbroek 		    rv = 1;
248*5d5fbe79SDavid van Moolenbroek 	    }
249*5d5fbe79SDavid van Moolenbroek 	    framelen = 0;
250*5d5fbe79SDavid van Moolenbroek 	    flush_flag = 0;
251*5d5fbe79SDavid van Moolenbroek 	    escape_flag = 0;
252*5d5fbe79SDavid van Moolenbroek 	    fcs = PPP_INITFCS;
253*5d5fbe79SDavid van Moolenbroek 	    continue;
254*5d5fbe79SDavid van Moolenbroek 	}
255*5d5fbe79SDavid van Moolenbroek 	if (flush_flag)
256*5d5fbe79SDavid van Moolenbroek 	    continue;
257*5d5fbe79SDavid van Moolenbroek 	if (escape_flag) {
258*5d5fbe79SDavid van Moolenbroek 	    c ^= PPP_TRANS;
259*5d5fbe79SDavid van Moolenbroek 	    escape_flag = 0;
260*5d5fbe79SDavid van Moolenbroek 	} else if (c == PPP_ESCAPE) {
261*5d5fbe79SDavid van Moolenbroek 	    escape_flag = 1;
262*5d5fbe79SDavid van Moolenbroek 	    continue;
263*5d5fbe79SDavid van Moolenbroek 	}
264*5d5fbe79SDavid van Moolenbroek 	if (framelen >= framemax) {
265*5d5fbe79SDavid van Moolenbroek 	    flush_flag = 1;
266*5d5fbe79SDavid van Moolenbroek 	    continue;
267*5d5fbe79SDavid van Moolenbroek 	}
268*5d5fbe79SDavid van Moolenbroek 	frame[framelen++] = c;
269*5d5fbe79SDavid van Moolenbroek 	fcs = PPP_FCS(fcs, c);
270*5d5fbe79SDavid van Moolenbroek     }
271*5d5fbe79SDavid van Moolenbroek     return rv;
272*5d5fbe79SDavid van Moolenbroek }
273*5d5fbe79SDavid van Moolenbroek 
274*5d5fbe79SDavid van Moolenbroek /*
275*5d5fbe79SDavid van Moolenbroek  * loop_frame - given a frame obtained from the loopback,
276*5d5fbe79SDavid van Moolenbroek  * decide whether to bring up the link or not, and, if we want
277*5d5fbe79SDavid van Moolenbroek  * to transmit this frame later, put it on the pending queue.
278*5d5fbe79SDavid van Moolenbroek  * Return value is 1 if we need to bring up the link, 0 otherwise.
279*5d5fbe79SDavid van Moolenbroek  * We assume that the kernel driver has already applied the
280*5d5fbe79SDavid van Moolenbroek  * pass_filter, so we won't get packets it rejected.
281*5d5fbe79SDavid van Moolenbroek  * We apply the active_filter to see if we want this packet to
282*5d5fbe79SDavid van Moolenbroek  * bring up the link.
283*5d5fbe79SDavid van Moolenbroek  */
284*5d5fbe79SDavid van Moolenbroek int
loop_frame(frame,len)285*5d5fbe79SDavid van Moolenbroek loop_frame(frame, len)
286*5d5fbe79SDavid van Moolenbroek     unsigned char *frame;
287*5d5fbe79SDavid van Moolenbroek     int len;
288*5d5fbe79SDavid van Moolenbroek {
289*5d5fbe79SDavid van Moolenbroek     struct packet *pkt;
290*5d5fbe79SDavid van Moolenbroek 
291*5d5fbe79SDavid van Moolenbroek     /* dbglog("from loop: %P", frame, len); */
292*5d5fbe79SDavid van Moolenbroek     if (len < PPP_HDRLEN)
293*5d5fbe79SDavid van Moolenbroek 	return 0;
294*5d5fbe79SDavid van Moolenbroek     if ((PPP_PROTOCOL(frame) & 0x8000) != 0)
295*5d5fbe79SDavid van Moolenbroek 	return 0;		/* shouldn't get any of these anyway */
296*5d5fbe79SDavid van Moolenbroek     if (!active_packet(frame, len))
297*5d5fbe79SDavid van Moolenbroek 	return 0;
298*5d5fbe79SDavid van Moolenbroek 
299*5d5fbe79SDavid van Moolenbroek     pkt = (struct packet *) malloc(sizeof(struct packet) + len);
300*5d5fbe79SDavid van Moolenbroek     if (pkt != NULL) {
301*5d5fbe79SDavid van Moolenbroek 	pkt->length = len;
302*5d5fbe79SDavid van Moolenbroek 	pkt->next = NULL;
303*5d5fbe79SDavid van Moolenbroek 	memcpy(pkt->data, frame, len);
304*5d5fbe79SDavid van Moolenbroek 	if (pend_q == NULL)
305*5d5fbe79SDavid van Moolenbroek 	    pend_q = pkt;
306*5d5fbe79SDavid van Moolenbroek 	else
307*5d5fbe79SDavid van Moolenbroek 	    pend_qtail->next = pkt;
308*5d5fbe79SDavid van Moolenbroek 	pend_qtail = pkt;
309*5d5fbe79SDavid van Moolenbroek     }
310*5d5fbe79SDavid van Moolenbroek     return 1;
311*5d5fbe79SDavid van Moolenbroek }
312*5d5fbe79SDavid van Moolenbroek 
313*5d5fbe79SDavid van Moolenbroek /*
314*5d5fbe79SDavid van Moolenbroek  * demand_rexmit - Resend all those frames which we got via the
315*5d5fbe79SDavid van Moolenbroek  * loopback, now that the real serial link is up.
316*5d5fbe79SDavid van Moolenbroek  */
317*5d5fbe79SDavid van Moolenbroek void
demand_rexmit(proto,newip)318*5d5fbe79SDavid van Moolenbroek demand_rexmit(proto, newip)
319*5d5fbe79SDavid van Moolenbroek     int proto;
320*5d5fbe79SDavid van Moolenbroek     u32_t newip;
321*5d5fbe79SDavid van Moolenbroek {
322*5d5fbe79SDavid van Moolenbroek     struct packet *pkt, *prev, *nextpkt;
323*5d5fbe79SDavid van Moolenbroek     unsigned short checksum;
324*5d5fbe79SDavid van Moolenbroek     unsigned short pkt_checksum = 0;
325*5d5fbe79SDavid van Moolenbroek     unsigned iphdr;
326*5d5fbe79SDavid van Moolenbroek     struct timeval tv;
327*5d5fbe79SDavid van Moolenbroek     char cv = 0;
328*5d5fbe79SDavid van Moolenbroek     char ipstr[16];
329*5d5fbe79SDavid van Moolenbroek 
330*5d5fbe79SDavid van Moolenbroek     prev = NULL;
331*5d5fbe79SDavid van Moolenbroek     pkt = pend_q;
332*5d5fbe79SDavid van Moolenbroek     pend_q = NULL;
333*5d5fbe79SDavid van Moolenbroek     tv.tv_sec = 1;
334*5d5fbe79SDavid van Moolenbroek     tv.tv_usec = 0;
335*5d5fbe79SDavid van Moolenbroek     select(0,NULL,NULL,NULL,&tv);	/* Sleep for 1 Seconds */
336*5d5fbe79SDavid van Moolenbroek     for (; pkt != NULL; pkt = nextpkt) {
337*5d5fbe79SDavid van Moolenbroek 	nextpkt = pkt->next;
338*5d5fbe79SDavid van Moolenbroek 	if (PPP_PROTOCOL(pkt->data) == proto) {
339*5d5fbe79SDavid van Moolenbroek             if ( (proto == PPP_IP) && newip ) {
340*5d5fbe79SDavid van Moolenbroek 		/* Get old checksum */
341*5d5fbe79SDavid van Moolenbroek 
342*5d5fbe79SDavid van Moolenbroek 		iphdr = (pkt->data[4] & 15) << 2;
343*5d5fbe79SDavid van Moolenbroek 		checksum = *((unsigned short *) (pkt->data+14));
344*5d5fbe79SDavid van Moolenbroek                 if (checksum == 0xFFFF) {
345*5d5fbe79SDavid van Moolenbroek                     checksum = 0;
346*5d5fbe79SDavid van Moolenbroek                 }
347*5d5fbe79SDavid van Moolenbroek 
348*5d5fbe79SDavid van Moolenbroek 
349*5d5fbe79SDavid van Moolenbroek                 if (pkt->data[13] == 17) {
350*5d5fbe79SDavid van Moolenbroek                     pkt_checksum =  *((unsigned short *) (pkt->data+10+iphdr));
351*5d5fbe79SDavid van Moolenbroek 		    if (pkt_checksum) {
352*5d5fbe79SDavid van Moolenbroek                         cv = 1;
353*5d5fbe79SDavid van Moolenbroek                         if (pkt_checksum == 0xFFFF) {
354*5d5fbe79SDavid van Moolenbroek                             pkt_checksum = 0;
355*5d5fbe79SDavid van Moolenbroek                         }
356*5d5fbe79SDavid van Moolenbroek                     }
357*5d5fbe79SDavid van Moolenbroek                     else {
358*5d5fbe79SDavid van Moolenbroek                        cv = 0;
359*5d5fbe79SDavid van Moolenbroek                     }
360*5d5fbe79SDavid van Moolenbroek                 }
361*5d5fbe79SDavid van Moolenbroek 
362*5d5fbe79SDavid van Moolenbroek 		if (pkt->data[13] == 6) {
363*5d5fbe79SDavid van Moolenbroek 		    pkt_checksum = *((unsigned short *) (pkt->data+20+iphdr));
364*5d5fbe79SDavid van Moolenbroek 		    cv = 1;
365*5d5fbe79SDavid van Moolenbroek                     if (pkt_checksum == 0xFFFF) {
366*5d5fbe79SDavid van Moolenbroek                         pkt_checksum = 0;
367*5d5fbe79SDavid van Moolenbroek                     }
368*5d5fbe79SDavid van Moolenbroek 		}
369*5d5fbe79SDavid van Moolenbroek 
370*5d5fbe79SDavid van Moolenbroek 		/* Delete old Source-IP-Address */
371*5d5fbe79SDavid van Moolenbroek                 checksum -= *((unsigned short *) (pkt->data+16)) ^ 0xFFFF;
372*5d5fbe79SDavid van Moolenbroek                 checksum -= *((unsigned short *) (pkt->data+18)) ^ 0xFFFF;
373*5d5fbe79SDavid van Moolenbroek 
374*5d5fbe79SDavid van Moolenbroek 		pkt_checksum -= *((unsigned short *) (pkt->data+16)) ^ 0xFFFF;
375*5d5fbe79SDavid van Moolenbroek 		pkt_checksum -= *((unsigned short *) (pkt->data+18)) ^ 0xFFFF;
376*5d5fbe79SDavid van Moolenbroek 
377*5d5fbe79SDavid van Moolenbroek 		/* Change Source-IP-Address */
378*5d5fbe79SDavid van Moolenbroek                 * ((u32_t *) (pkt->data + 16)) = newip;
379*5d5fbe79SDavid van Moolenbroek 
380*5d5fbe79SDavid van Moolenbroek 		/* Add new Source-IP-Address */
381*5d5fbe79SDavid van Moolenbroek                 checksum += *((unsigned short *) (pkt->data+16)) ^ 0xFFFF;
382*5d5fbe79SDavid van Moolenbroek                 checksum += *((unsigned short *) (pkt->data+18)) ^ 0xFFFF;
383*5d5fbe79SDavid van Moolenbroek 
384*5d5fbe79SDavid van Moolenbroek                 pkt_checksum += *((unsigned short *) (pkt->data+16)) ^ 0xFFFF;
385*5d5fbe79SDavid van Moolenbroek                 pkt_checksum += *((unsigned short *) (pkt->data+18)) ^ 0xFFFF;
386*5d5fbe79SDavid van Moolenbroek 
387*5d5fbe79SDavid van Moolenbroek 		/* Write new checksum */
388*5d5fbe79SDavid van Moolenbroek                 if (!checksum) {
389*5d5fbe79SDavid van Moolenbroek                     checksum = 0xFFFF;
390*5d5fbe79SDavid van Moolenbroek                 }
391*5d5fbe79SDavid van Moolenbroek                 *((unsigned short *) (pkt->data+14)) = checksum;
392*5d5fbe79SDavid van Moolenbroek 		if (pkt->data[13] == 6) {
393*5d5fbe79SDavid van Moolenbroek 		    *((unsigned short *) (pkt->data+20+iphdr)) = pkt_checksum;
394*5d5fbe79SDavid van Moolenbroek 		}
395*5d5fbe79SDavid van Moolenbroek 		if (cv && (pkt->data[13] == 17) ) {
396*5d5fbe79SDavid van Moolenbroek 		    *((unsigned short *) (pkt->data+10+iphdr)) = pkt_checksum;
397*5d5fbe79SDavid van Moolenbroek 		}
398*5d5fbe79SDavid van Moolenbroek 
399*5d5fbe79SDavid van Moolenbroek 		/* Log Packet */
400*5d5fbe79SDavid van Moolenbroek 		strcpy(ipstr,inet_ntoa(*( (struct in_addr *) (pkt->data+16))));
401*5d5fbe79SDavid van Moolenbroek 		if (pkt->data[13] == 1) {
402*5d5fbe79SDavid van Moolenbroek 		    syslog(LOG_INFO,"Open ICMP %s -> %s\n",
403*5d5fbe79SDavid van Moolenbroek 			ipstr,
404*5d5fbe79SDavid van Moolenbroek 			inet_ntoa(*( (struct in_addr *) (pkt->data+20))));
405*5d5fbe79SDavid van Moolenbroek 		} else {
406*5d5fbe79SDavid van Moolenbroek 		    syslog(LOG_INFO,"Open %s %s:%d -> %s:%d\n",
407*5d5fbe79SDavid van Moolenbroek 			pkt->data[13] == 6 ? "TCP" : "UDP",
408*5d5fbe79SDavid van Moolenbroek 			ipstr,
409*5d5fbe79SDavid van Moolenbroek 			ntohs(*( (short *) (pkt->data+iphdr+4))),
410*5d5fbe79SDavid van Moolenbroek 			inet_ntoa(*( (struct in_addr *) (pkt->data+20))),
411*5d5fbe79SDavid van Moolenbroek 			ntohs(*( (short *) (pkt->data+iphdr+6))));
412*5d5fbe79SDavid van Moolenbroek                 }
413*5d5fbe79SDavid van Moolenbroek             }
414*5d5fbe79SDavid van Moolenbroek 	    output(pcb, pkt->data, pkt->length);
415*5d5fbe79SDavid van Moolenbroek 	    free(pkt);
416*5d5fbe79SDavid van Moolenbroek 	} else {
417*5d5fbe79SDavid van Moolenbroek 	    if (prev == NULL)
418*5d5fbe79SDavid van Moolenbroek 		pend_q = pkt;
419*5d5fbe79SDavid van Moolenbroek 	    else
420*5d5fbe79SDavid van Moolenbroek 		prev->next = pkt;
421*5d5fbe79SDavid van Moolenbroek 	    prev = pkt;
422*5d5fbe79SDavid van Moolenbroek 	}
423*5d5fbe79SDavid van Moolenbroek     }
424*5d5fbe79SDavid van Moolenbroek     pend_qtail = prev;
425*5d5fbe79SDavid van Moolenbroek     if (prev != NULL)
426*5d5fbe79SDavid van Moolenbroek 	prev->next = NULL;
427*5d5fbe79SDavid van Moolenbroek }
428*5d5fbe79SDavid van Moolenbroek 
429*5d5fbe79SDavid van Moolenbroek /*
430*5d5fbe79SDavid van Moolenbroek  * Scan a packet to decide whether it is an "active" packet,
431*5d5fbe79SDavid van Moolenbroek  * that is, whether it is worth bringing up the link for.
432*5d5fbe79SDavid van Moolenbroek  */
433*5d5fbe79SDavid van Moolenbroek static int
active_packet(p,len)434*5d5fbe79SDavid van Moolenbroek active_packet(p, len)
435*5d5fbe79SDavid van Moolenbroek     unsigned char *p;
436*5d5fbe79SDavid van Moolenbroek     int len;
437*5d5fbe79SDavid van Moolenbroek {
438*5d5fbe79SDavid van Moolenbroek     int proto, i;
439*5d5fbe79SDavid van Moolenbroek     const struct protent *protp;
440*5d5fbe79SDavid van Moolenbroek 
441*5d5fbe79SDavid van Moolenbroek     if (len < PPP_HDRLEN)
442*5d5fbe79SDavid van Moolenbroek 	return 0;
443*5d5fbe79SDavid van Moolenbroek     proto = PPP_PROTOCOL(p);
444*5d5fbe79SDavid van Moolenbroek #ifdef PPP_FILTER
445*5d5fbe79SDavid van Moolenbroek     p[0] = 1;		/* outbound packet indicator */
446*5d5fbe79SDavid van Moolenbroek     if ((pass_filter.bf_len != 0
447*5d5fbe79SDavid van Moolenbroek 	 && bpf_filter(pass_filter.bf_insns, p, len, len) == 0)
448*5d5fbe79SDavid van Moolenbroek 	|| (active_filter.bf_len != 0
449*5d5fbe79SDavid van Moolenbroek 	    && bpf_filter(active_filter.bf_insns, p, len, len) == 0)) {
450*5d5fbe79SDavid van Moolenbroek 	p[0] = 0xff;
451*5d5fbe79SDavid van Moolenbroek 	return 0;
452*5d5fbe79SDavid van Moolenbroek     }
453*5d5fbe79SDavid van Moolenbroek     p[0] = 0xff;
454*5d5fbe79SDavid van Moolenbroek #endif
455*5d5fbe79SDavid van Moolenbroek     for (i = 0; (protp = protocols[i]) != NULL; ++i) {
456*5d5fbe79SDavid van Moolenbroek 	if (protp->protocol < 0xC000 && (protp->protocol & ~0x8000) == proto) {
457*5d5fbe79SDavid van Moolenbroek 	    if (protp->active_pkt == NULL)
458*5d5fbe79SDavid van Moolenbroek 		return 1;
459*5d5fbe79SDavid van Moolenbroek 	    return (*protp->active_pkt)(p, len);
460*5d5fbe79SDavid van Moolenbroek 	}
461*5d5fbe79SDavid van Moolenbroek     }
462*5d5fbe79SDavid van Moolenbroek     return 0;			/* not a supported protocol !!?? */
463*5d5fbe79SDavid van Moolenbroek }
464*5d5fbe79SDavid van Moolenbroek 
465*5d5fbe79SDavid van Moolenbroek #endif /* PPP_SUPPORT && DEMAND_SUPPORT */
466