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