]> git.ozlabs.org Git - ppp.git/blob - pppd/sys-str.c
added GetMask
[ppp.git] / pppd / sys-str.c
1 /*
2  * sys-str.c - System-dependent procedures for setting up
3  * PPP interfaces on systems which use the STREAMS ppp interface.
4  *
5  * Copyright (c) 1989 Carnegie Mellon University.
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms are permitted
9  * provided that the above copyright notice and this paragraph are
10  * duplicated in all such forms and that any documentation,
11  * advertising materials, and other materials related to such
12  * distribution and use acknowledge that the software was developed
13  * by Carnegie Mellon University.  The name of the
14  * University may not be used to endorse or promote products derived
15  * from this software without specific prior written permission.
16  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
17  * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
18  * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
19  */
20
21 #ifndef lint
22 static char rcsid[] = "$Id: sys-str.c,v 1.18 1995/04/27 00:35:43 paulus Exp $";
23 #endif
24
25 /*
26  * TODO:
27  */
28
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <errno.h>
32 #include <syslog.h>
33 #include <termios.h>
34 #include <fcntl.h>
35 #include <string.h>
36 #include <time.h>
37 #include <utmp.h>
38 #include <poll.h>
39 #include <sys/types.h>
40 #include <sys/socket.h>
41 #include <sys/sockio.h>
42 #include <sys/time.h>
43 #include <sys/stream.h>
44 #include <sys/stropts.h>
45
46 #include <arpa/inet.h>
47 #include <net/if.h>
48 #include <net/ppp_defs.h>
49 #include <net/ppp_str.h>
50 #include <net/route.h>
51 #include <net/if_arp.h>
52 #include <netinet/in.h>
53
54 #include "pppd.h"
55
56 #ifndef ifr_mtu
57 #define ifr_mtu         ifr_metric
58 #endif
59
60 #define MAXMODULES      10      /* max number of module names to save */
61 static struct   modlist {
62     char        modname[FMNAMESZ+1];
63 } str_modules[MAXMODULES];
64 static int      str_module_count = 0;
65 static int      pushed_ppp;
66 static int      closed_stdio;
67
68 static int      restore_term;   /* 1 => we've munged the terminal */
69 static struct termios inittermios; /* Initial TTY termios */
70
71 int sockfd;                     /* socket for doing interface ioctls */
72 static char *lock_file;         /* lock file created for serial port */
73
74 /*
75  * sys_init - System-dependent initialization.
76  */
77 void
78 sys_init()
79 {
80     openlog("pppd", LOG_PID | LOG_NDELAY, LOG_PPP);
81     setlogmask(LOG_UPTO(LOG_INFO));
82     if (debug)
83         setlogmask(LOG_UPTO(LOG_DEBUG));
84
85     /* Get an internet socket for doing socket ioctl's on. */
86     if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
87         syslog(LOG_ERR, "Couldn't create IP socket: %m");
88         die(1);
89     }
90 }
91
92 /*
93  * note_debug_level - note a change in the debug level.
94  */
95 void
96 note_debug_level()
97 {
98     if (debug) {
99         syslog(LOG_INFO, "Debug turned ON, Level %d", debug);
100         setlogmask(LOG_UPTO(LOG_DEBUG));
101     } else {
102         setlogmask(LOG_UPTO(LOG_WARNING));
103     }
104 }
105
106 /*
107  * daemon - Detach us from the terminal session.
108  */
109 int
110 daemon(nochdir, noclose)
111     int nochdir, noclose;
112 {
113     int pid;
114
115     if ((pid = fork()) < 0)
116         return -1;
117     if (pid != 0)
118         exit(0);                /* parent dies */
119     setsid();
120     if (!nochdir)
121         chdir("/");
122     if (!noclose) {
123         fclose(stdin);          /* don't need stdin, stdout, stderr */
124         fclose(stdout);
125         fclose(stderr);
126     }
127     return 0;
128 }
129
130
131 /*
132  * ppp_available - check if this kernel supports PPP.
133  */
134 int
135 ppp_available()
136 {
137     int fd, ret;
138
139     fd = open("/dev/tty", O_RDONLY, 0);
140     if (fd < 0)
141         return 1;               /* can't find out - assume we have ppp */
142     ret = ioctl(fd, I_FIND, "pppasync") >= 0;
143     close(fd);
144     return ret;
145 }
146
147
148 /*
149  * establish_ppp - Turn the serial port into a ppp interface.
150  */
151 void
152 establish_ppp()
153 {
154     /* go through and save the name of all the modules, then pop em */
155     for (;;) { 
156         if (ioctl(fd, I_LOOK, str_modules[str_module_count].modname) < 0 ||
157             ioctl(fd, I_POP, 0) < 0)
158             break;
159         MAINDEBUG((LOG_DEBUG, "popped stream module : %s",
160                    str_modules[str_module_count].modname));
161         str_module_count++;
162     }
163
164     /* now push the async/fcs module */
165     if (ioctl(fd, I_PUSH, "pppasync") < 0) {
166         syslog(LOG_ERR, "ioctl(I_PUSH, ppp_async): %m");
167         die(1);
168     }
169     /* push the compress module */
170     if (ioctl(fd, I_PUSH, "pppcomp") < 0) {
171         syslog(LOG_WARNING, "ioctl(I_PUSH, ppp_comp): %m");
172     }
173     /* finally, push the ppp_if module that actually handles the */
174     /* network interface */ 
175     if (ioctl(fd, I_PUSH, "pppif") < 0) {
176         syslog(LOG_ERR, "ioctl(I_PUSH, ppp_if): %m");
177         die(1);
178     }
179     pushed_ppp = 1;
180     /* read mode, message non-discard mode */
181     if (ioctl(fd, I_SRDOPT, RMSGN) < 0) {
182         syslog(LOG_ERR, "ioctl(I_SRDOPT, RMSGN): %m");
183         die(1);
184     }
185     /*
186      * Find out which interface we were given.
187      * (ppp_if handles this ioctl)
188      */
189     if (ioctl(fd, SIOCGETU, &ifunit) < 0) {
190         syslog(LOG_ERR, "ioctl(SIOCGETU): %m");
191         die(1);
192     }
193
194     /* Set debug flags in driver */
195     if (ioctl(fd, SIOCSIFDEBUG, &kdebugflag) < 0) {
196         syslog(LOG_ERR, "ioctl(SIOCSIFDEBUG): %m");
197     }
198
199     /* close stdin, stdout, stderr if they might refer to the device */
200     if (default_device && !closed_stdio) {
201         int i;
202
203         for (i = 0; i <= 2; ++i)
204             if (i != fd && i != sockfd)
205                 close(i);
206         closed_stdio = 1;
207     }
208 }
209
210 /*
211  * disestablish_ppp - Restore the serial port to normal operation.
212  * It attempts to reconstruct the stream with the previously popped
213  * modules.  This shouldn't call die() because it's called from die().
214  */
215 void
216 disestablish_ppp()
217 {
218     int flags;
219     char *s;
220
221     if (hungup) {
222         /* we can't push or pop modules after the stream has hung up */
223         str_module_count = 0;
224         restore_term = 0;       /* nor can we fix up terminal settings */
225         return;
226     }
227
228     if (pushed_ppp) {
229         /*
230          * Check whether the link seems not to be 8-bit clean.
231          */
232         if (ioctl(fd, SIOCGIFDEBUG, (caddr_t) &flags) == 0) {
233             s = NULL;
234             switch (~flags & PAI_FLAGS_HIBITS) {
235             case PAI_FLAGS_B7_0:
236                 s = "bit 7 set to 1";
237                 break;
238             case PAI_FLAGS_B7_1:
239                 s = "bit 7 set to 0";
240                 break;
241             case PAI_FLAGS_PAR_EVEN:
242                 s = "odd parity";
243                 break;
244             case PAI_FLAGS_PAR_ODD:
245                 s = "even parity";
246                 break;
247             }
248             if (s != NULL) {
249                 syslog(LOG_WARNING, "Serial link is not 8-bit clean:");
250                 syslog(LOG_WARNING, "All received characters had %s", s);
251             }
252         }
253
254         while (ioctl(fd, I_POP, 0) == 0)        /* pop any we pushed */
255             ;
256   
257         for (; str_module_count > 0; str_module_count--) {
258             if (ioctl(fd, I_PUSH, str_modules[str_module_count-1].modname)) {
259                 if (errno != ENXIO)
260                     syslog(LOG_WARNING, "Couldn't restore module %s: %m",
261                            str_modules[str_module_count-1].modname);
262             } else {
263                 MAINDEBUG((LOG_INFO, "str_restore: pushed module %s",
264                            str_modules[str_module_count-1].modname));
265             }
266         }
267         pushed_ppp = 0;
268     }
269 }
270
271
272 /*
273  * List of valid speeds.
274  */
275 struct speed {
276     int speed_int, speed_val;
277 } speeds[] = {
278 #ifdef B50
279     { 50, B50 },
280 #endif
281 #ifdef B75
282     { 75, B75 },
283 #endif
284 #ifdef B110
285     { 110, B110 },
286 #endif
287 #ifdef B134
288     { 134, B134 },
289 #endif
290 #ifdef B150
291     { 150, B150 },
292 #endif
293 #ifdef B200
294     { 200, B200 },
295 #endif
296 #ifdef B300
297     { 300, B300 },
298 #endif
299 #ifdef B600
300     { 600, B600 },
301 #endif
302 #ifdef B1200
303     { 1200, B1200 },
304 #endif
305 #ifdef B1800
306     { 1800, B1800 },
307 #endif
308 #ifdef B2000
309     { 2000, B2000 },
310 #endif
311 #ifdef B2400
312     { 2400, B2400 },
313 #endif
314 #ifdef B3600
315     { 3600, B3600 },
316 #endif
317 #ifdef B4800
318     { 4800, B4800 },
319 #endif
320 #ifdef B7200
321     { 7200, B7200 },
322 #endif
323 #ifdef B9600
324     { 9600, B9600 },
325 #endif
326 #ifdef B19200
327     { 19200, B19200 },
328 #endif
329 #ifdef B38400
330     { 38400, B38400 },
331 #endif
332 #ifdef EXTA
333     { 19200, EXTA },
334 #endif
335 #ifdef EXTB
336     { 38400, EXTB },
337 #endif
338 #ifdef B57600
339     { 57600, B57600 },
340 #endif
341 #ifdef B115200
342     { 115200, B115200 },
343 #endif
344     { 0, 0 }
345 };
346
347 /*
348  * Translate from bits/second to a speed_t.
349  */
350 int
351 translate_speed(bps)
352     int bps;
353 {
354     struct speed *speedp;
355
356     if (bps == 0)
357         return 0;
358     for (speedp = speeds; speedp->speed_int; speedp++)
359         if (bps == speedp->speed_int)
360             return speedp->speed_val;
361     syslog(LOG_WARNING, "speed %d not supported", bps);
362     return 0;
363 }
364
365 /*
366  * Translate from a speed_t to bits/second.
367  */
368 int
369 baud_rate_of(speed)
370     int speed;
371 {
372     struct speed *speedp;
373
374     if (speed == 0)
375         return 0;
376     for (speedp = speeds; speedp->speed_int; speedp++)
377         if (speed == speedp->speed_val)
378             return speedp->speed_int;
379     return 0;
380 }
381
382 /*
383  * set_up_tty: Set up the serial port on `fd' for 8 bits, no parity,
384  * at the requested speed, etc.  If `local' is true, set CLOCAL
385  * regardless of whether the modem option was specified.
386  */
387 set_up_tty(fd, local)
388     int fd, local;
389 {
390     int speed;
391     struct termios tios;
392
393     if (tcgetattr(fd, &tios) < 0) {
394         syslog(LOG_ERR, "tcgetattr: %m");
395         die(1);
396     }
397
398     if (!restore_term)
399         inittermios = tios;
400
401     tios.c_cflag &= ~(CSIZE | CSTOPB | PARENB | CLOCAL);
402     if (crtscts > 0)
403         tios.c_cflag |= CRTSCTS;
404     else if (crtscts < 0)
405         tios.c_cflag &= ~CRTSCTS;
406
407     tios.c_cflag |= CS8 | CREAD | HUPCL;
408     if (local || !modem)
409         tios.c_cflag |= CLOCAL;
410     tios.c_iflag = IGNBRK | IGNPAR;
411     tios.c_oflag = 0;
412     tios.c_lflag = 0;
413     tios.c_cc[VMIN] = 1;
414     tios.c_cc[VTIME] = 0;
415
416     if (crtscts == 2) {
417         tios.c_iflag |= IXOFF;
418         tios.c_cc[VSTOP] = 0x13;        /* DC3 = XOFF = ^S */
419         tios.c_cc[VSTART] = 0x11;       /* DC1 = XON  = ^Q */
420     }
421
422     speed = translate_speed(inspeed);
423     if (speed) {
424         cfsetospeed(&tios, speed);
425         cfsetispeed(&tios, speed);
426     } else {
427         speed = cfgetospeed(&tios);
428         /*
429          * We can't proceed if the serial port speed is B0,
430          * since that implies that the serial port is disabled.
431          */
432         if (speed == B0) {
433             syslog(LOG_ERR, "Baud rate for %s is 0; need explicit baud rate",
434                    devnam);
435             die(1);
436         }
437     }
438
439     if (tcsetattr(fd, TCSAFLUSH, &tios) < 0) {
440         syslog(LOG_ERR, "tcsetattr: %m");
441         die(1);
442     }
443
444     baud_rate = inspeed = baud_rate_of(speed);
445     restore_term = 1;
446 }
447
448 /*
449  * restore_tty - restore the terminal to the saved settings.
450  */
451 void
452 restore_tty()
453 {
454     if (restore_term) {
455         if (tcsetattr(fd, TCSAFLUSH, &inittermios) < 0)
456             if (errno != ENXIO)
457                 syslog(LOG_WARNING, "tcsetattr: %m");
458         restore_term = 0;
459     }
460 }
461
462 /*
463  * setdtr - control the DTR line on the serial port.
464  * This is called from die(), so it shouldn't call die().
465  */
466 setdtr(fd, on)
467 int fd, on;
468 {
469     int modembits = TIOCM_DTR;
470
471     ioctl(fd, (on? TIOCMBIS: TIOCMBIC), &modembits);
472 }
473
474
475 /*
476  * output - Output PPP packet.
477  */
478 void
479 output(unit, p, len)
480     int unit;
481     u_char *p;
482     int len;
483 {
484     struct strbuf       str;
485
486     if (unit != 0)
487         MAINDEBUG((LOG_WARNING, "output: unit != 0!"));
488     if (debug)
489         log_packet(p, len, "sent ");
490
491     str.len = len;
492     str.buf = (caddr_t) p;
493     if (putmsg(fd, NULL, &str, 0) < 0) {
494         if (errno != ENXIO) {
495             syslog(LOG_ERR, "putmsg: %m");
496             die(1);
497         }
498     }
499 }
500
501 /*
502  * wait_input - wait for input, for a length of time specified in *timo.
503  */
504 wait_input(timo)
505     struct timeval *timo;
506 {
507     int t;
508     struct pollfd pfd;
509
510     t = timo == NULL? -1: timo->tv_sec * 1000 + timo->tv_usec / 1000;
511     pfd.fd = fd;
512     pfd.events = POLLIN | POLLPRI | POLLHUP;
513     if (poll(&pfd, 1, t) < 0 && errno != EINTR) {
514         syslog(LOG_ERR, "poll: %m");
515         die(1);
516     }
517 }
518
519 /*
520  * read_packet - get a PPP packet from the serial device.
521  */
522 int
523 read_packet(buf)
524     u_char *buf;
525 {
526     struct strbuf str, ctl;
527     int len, i;
528     unsigned char ctlbuf[16];
529
530     str.maxlen = PPP_MTU + PPP_HDRLEN;
531     str.buf = (caddr_t) buf;
532     ctl.maxlen = sizeof(ctlbuf);
533     ctl.buf = (caddr_t) ctlbuf;
534     i = 0;
535     len = getmsg(fd, &ctl, &str, &i);
536     if (len < 0) {
537         if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR) {
538             return -1;
539         }
540         syslog(LOG_ERR, "getmsg(fd) %m");
541         die(1);
542     }
543     if (len) 
544         MAINDEBUG((LOG_DEBUG, "getmsg returned 0x%x",len));
545     if (ctl.len > 0)
546         syslog(LOG_NOTICE, "got ctrl msg len %d %x %x\n", ctl.len,
547                ctlbuf[0], ctlbuf[1]);
548
549     if (str.len < 0) {
550         MAINDEBUG((LOG_DEBUG, "getmsg short return length %d", str.len));
551         return -1;
552     }
553
554     return str.len;
555 }
556
557
558 /*
559  * ppp_send_config - configure the transmit characteristics of
560  * the ppp interface.
561  */
562 void
563 ppp_send_config(unit, mtu, asyncmap, pcomp, accomp)
564     int unit, mtu;
565     u_int32_t asyncmap;
566     int pcomp, accomp;
567 {
568     char c;
569     struct ifreq ifr;
570
571     strncpy(ifr.ifr_name, ifname, sizeof (ifr.ifr_name));
572     ifr.ifr_mtu = mtu;
573     if (ioctl(sockfd, SIOCSIFMTU, (caddr_t) &ifr) < 0) {
574         syslog(LOG_ERR, "ioctl(SIOCSIFMTU): %m");
575         quit();
576     }
577
578     if(ioctl(fd, SIOCSIFASYNCMAP, (caddr_t) &asyncmap) < 0) {
579         syslog(LOG_ERR, "ioctl(SIOCSIFASYNCMAP): %m");
580         quit();
581     }
582
583     c = (pcomp? 1: 0);
584     if(ioctl(fd, SIOCSIFCOMPPROT, &c) < 0) {
585         syslog(LOG_ERR, "ioctl(SIOCSIFCOMPPROT): %m");
586         quit();
587     }
588
589     c = (accomp? 1: 0);
590     if(ioctl(fd, SIOCSIFCOMPAC, &c) < 0) {
591         syslog(LOG_ERR, "ioctl(SIOCSIFCOMPAC): %m");
592         quit();
593     }
594 }
595
596
597 /*
598  * ppp_set_xaccm - set the extended transmit ACCM for the interface.
599  */
600 void
601 ppp_set_xaccm(unit, accm)
602     int unit;
603     ext_accm accm;
604 {
605     if (ioctl(fd, SIOCSIFXASYNCMAP, accm) < 0 && errno != ENOTTY)
606         syslog(LOG_WARNING, "ioctl(set extended ACCM): %m");
607 }
608
609
610 /*
611  * ppp_recv_config - configure the receive-side characteristics of
612  * the ppp interface.
613  */
614 void
615 ppp_recv_config(unit, mru, asyncmap, pcomp, accomp)
616     int unit, mru;
617     u_int32_t asyncmap;
618     int pcomp, accomp;
619 {
620     char c;
621
622     if (ioctl(fd, SIOCSIFMRU, &mru) < 0) {
623         syslog(LOG_ERR, "ioctl(SIOCSIFMRU): %m");
624     }
625
626     if (ioctl(fd, SIOCSIFRASYNCMAP, (caddr_t) &asyncmap) < 0) {
627         syslog(LOG_ERR, "ioctl(SIOCSIFRASYNCMAP): %m");
628     }
629
630     c = 2 + (pcomp? 1: 0);
631     if(ioctl(fd, SIOCSIFCOMPPROT, &c) < 0) {
632         syslog(LOG_ERR, "ioctl(SIOCSIFCOMPPROT): %m");
633     }
634
635     c = 2 + (accomp? 1: 0);
636     if (ioctl(fd, SIOCSIFCOMPAC, &c) < 0) {
637         syslog(LOG_ERR, "ioctl(SIOCSIFCOMPAC): %m");
638     }
639 }
640
641 /*
642  * ccp_test - ask kernel whether a given compression method
643  * is acceptable for use.
644  */
645 ccp_test(unit, opt_ptr, opt_len, for_transmit)
646     int unit, opt_len, for_transmit;
647     u_char *opt_ptr;
648 {
649     struct ppp_option_data data;
650
651     if ((unsigned) opt_len > MAX_PPP_OPTION)
652         opt_len = MAX_PPP_OPTION;
653     data.length = opt_len;
654     data.transmit = for_transmit;
655     BCOPY(opt_ptr, data.opt_data, opt_len);
656     return ioctl(fd, SIOCSCOMPRESS, (caddr_t) &data) >= 0;
657 }
658
659 /*
660  * ccp_flags_set - inform kernel about the current state of CCP.
661  */
662 void
663 ccp_flags_set(unit, isopen, isup)
664     int unit, isopen, isup;
665 {
666     int x;
667
668     x = (isopen? 1: 0) + (isup? 2: 0);
669     if (ioctl(fd, SIOCSIFCOMP, (caddr_t) &x) < 0 && errno != ENOTTY)
670         syslog(LOG_ERR, "ioctl (SIOCSIFCOMP): %m");
671 }
672
673 /*
674  * ccp_fatal_error - returns 1 if decompression was disabled as a
675  * result of an error detected after decompression of a packet,
676  * 0 otherwise.  This is necessary because of patent nonsense.
677  */
678 int
679 ccp_fatal_error(unit)
680     int unit;
681 {
682     int x;
683
684     if (ioctl(fd, SIOCGIFCOMP, (caddr_t) &x) < 0) {
685         syslog(LOG_ERR, "ioctl(SIOCGIFCOMP): %m");
686         return 0;
687     }
688     return x & CCP_FATALERROR;
689 }
690
691 /*
692  * sifvjcomp - config tcp header compression
693  */
694 int
695 sifvjcomp(u, vjcomp, cidcomp, maxcid)
696     int u, vjcomp, cidcomp, maxcid;
697 {
698     char x;
699
700     x = (vjcomp? 1: 0) + (cidcomp? 0: 2) + (maxcid << 4);
701     if (ioctl(fd, SIOCSIFVJCOMP, (caddr_t) &x) < 0) {
702         syslog(LOG_ERR, "ioctl(SIOCSIFVJCOMP): %m");
703         return 0;
704     }
705     return 1;
706 }
707
708 /*
709  * sifup - Config the interface up and enable IP packets to pass.
710  */
711 int
712 sifup(u)
713     int u;
714 {
715     struct ifreq ifr;
716     struct npioctl npi;
717
718     strncpy(ifr.ifr_name, ifname, sizeof (ifr.ifr_name));
719     if (ioctl(sockfd, SIOCGIFFLAGS, (caddr_t) &ifr) < 0) {
720         syslog(LOG_ERR, "ioctl (SIOCGIFFLAGS): %m");
721         return 0;
722     }
723     ifr.ifr_flags |= IFF_UP;
724     if (ioctl(sockfd, SIOCSIFFLAGS, (caddr_t) &ifr) < 0) {
725         syslog(LOG_ERR, "ioctl(SIOCSIFFLAGS): %m");
726         return 0;
727     }
728     npi.protocol = PPP_IP;
729     npi.mode = NPMODE_PASS;
730     if (ioctl(fd, SIOCSETNPMODE, &npi) < 0) {
731         if (errno != ENOTTY) {
732             syslog(LOG_ERR, "ioctl(SIOCSETNPMODE): %m");
733             return 0;
734         }
735     }
736     return 1;
737 }
738
739 /*
740  * sifdown - Config the interface down.
741  */
742 int
743 sifdown(u)
744     int u;
745 {
746     struct ifreq ifr;
747     int rv;
748     struct npioctl npi;
749
750     rv = 1;
751     npi.protocol = PPP_IP;
752     npi.mode = NPMODE_ERROR;
753     if (ioctl(fd, SIOCSETNPMODE, (caddr_t) &npi) < 0) {
754         if (errno != ENOTTY) {
755             syslog(LOG_ERR, "ioctl(SIOCSETNPMODE): %m");
756             rv = 0;
757         }
758     }
759
760     strncpy(ifr.ifr_name, ifname, sizeof (ifr.ifr_name));
761     if (ioctl(sockfd, SIOCGIFFLAGS, (caddr_t) &ifr) < 0) {
762         syslog(LOG_ERR, "ioctl (SIOCGIFFLAGS): %m");
763         rv = 0;
764     } else {
765         ifr.ifr_flags &= ~IFF_UP;
766         if (ioctl(sockfd, SIOCSIFFLAGS, (caddr_t) &ifr) < 0) {
767             syslog(LOG_ERR, "ioctl(SIOCSIFFLAGS): %m");
768             rv = 0;
769         }
770     }
771     return rv;
772 }
773
774 /*
775  * SET_SA_FAMILY - initialize a struct sockaddr, setting the sa_family field.
776  */
777 #define SET_SA_FAMILY(addr, family)             \
778     BZERO((char *) &(addr), sizeof(addr));      \
779     addr.sa_family = (family);
780
781 /*
782  * sifaddr - Config the interface IP addresses and netmask.
783  */
784 int
785 sifaddr(u, o, h, m)
786     int u;
787     u_int32_t o, h, m;
788 {
789     int ret;
790     struct ifreq ifr;
791
792     ret = 1;
793     strncpy(ifr.ifr_name, ifname, sizeof (ifr.ifr_name));
794     SET_SA_FAMILY(ifr.ifr_addr, AF_INET);
795     ((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr.s_addr = o;
796     if (ioctl(sockfd, SIOCSIFADDR, (caddr_t) &ifr) < 0) {
797         syslog(LOG_ERR, "ioctl(SIOCSIFADDR): %m");
798         ret = 0;
799     }
800     ((struct sockaddr_in *) &ifr.ifr_dstaddr)->sin_addr.s_addr = h;
801     if (ioctl(sockfd, SIOCSIFDSTADDR, (caddr_t) &ifr) < 0) {
802         syslog(LOG_ERR, "ioctl(SIOCSIFDSTADDR): %m");
803         ret = 0;
804     }
805     if (m != 0) {
806         ((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr.s_addr = m;
807         syslog(LOG_INFO, "Setting interface mask to %s\n", ip_ntoa(m));
808         if (ioctl(sockfd, SIOCSIFNETMASK, (caddr_t) &ifr) < 0) {
809             syslog(LOG_ERR, "ioctl(SIOCSIFNETMASK): %m");
810             ret = 0;
811         }
812     }
813     return ret;
814 }
815
816 /*
817  * cifaddr - Clear the interface IP addresses, and delete routes
818  * through the interface if possible.
819  */
820 int
821 cifaddr(u, o, h)
822     int u;
823     u_int32_t o, h;
824 {
825     struct rtentry rt;
826
827     SET_SA_FAMILY(rt.rt_dst, AF_INET);
828     ((struct sockaddr_in *) &rt.rt_dst)->sin_addr.s_addr = h;
829     SET_SA_FAMILY(rt.rt_gateway, AF_INET);
830     ((struct sockaddr_in *) &rt.rt_gateway)->sin_addr.s_addr = o;
831     rt.rt_flags = RTF_HOST;
832     if (ioctl(sockfd, SIOCDELRT, (caddr_t) &rt) < 0) {
833         syslog(LOG_ERR, "ioctl(SIOCDELRT): %m");
834         return 0;
835     }
836     return 1;
837 }
838
839 /*
840  * sifdefaultroute - assign a default route through the address given.
841  */
842 int
843 sifdefaultroute(u, g)
844     int u;
845     u_int32_t g;
846 {
847     struct rtentry rt;
848
849     SET_SA_FAMILY(rt.rt_dst, AF_INET);
850     SET_SA_FAMILY(rt.rt_gateway, AF_INET);
851     ((struct sockaddr_in *) &rt.rt_gateway)->sin_addr.s_addr = g;
852     rt.rt_flags = RTF_GATEWAY;
853     if (ioctl(sockfd, SIOCADDRT, &rt) < 0) {
854         syslog(LOG_ERR, "default route ioctl(SIOCADDRT): %m");
855         return 0;
856     }
857     return 1;
858 }
859
860 /*
861  * cifdefaultroute - delete a default route through the address given.
862  */
863 int
864 cifdefaultroute(u, g)
865     int u;
866     u_int32_t g;
867 {
868     struct rtentry rt;
869
870     SET_SA_FAMILY(rt.rt_dst, AF_INET);
871     SET_SA_FAMILY(rt.rt_gateway, AF_INET);
872     ((struct sockaddr_in *) &rt.rt_gateway)->sin_addr.s_addr = g;
873     rt.rt_flags = RTF_GATEWAY;
874     if (ioctl(sockfd, SIOCDELRT, &rt) < 0) {
875         syslog(LOG_ERR, "default route ioctl(SIOCDELRT): %m");
876         return 0;
877     }
878     return 1;
879 }
880
881 /*
882  * sifproxyarp - Make a proxy ARP entry for the peer.
883  */
884 int
885 sifproxyarp(unit, hisaddr)
886     int unit;
887     u_int32_t hisaddr;
888 {
889     struct arpreq arpreq;
890
891     BZERO(&arpreq, sizeof(arpreq));
892
893     /*
894      * Get the hardware address of an interface on the same subnet
895      * as our local address.
896      */
897     if (!get_ether_addr(hisaddr, &arpreq.arp_ha)) {
898         syslog(LOG_WARNING, "Cannot determine ethernet address for proxy ARP");
899         return 0;
900     }
901
902     SET_SA_FAMILY(arpreq.arp_pa, AF_INET);
903     ((struct sockaddr_in *) &arpreq.arp_pa)->sin_addr.s_addr = hisaddr;
904     arpreq.arp_flags = ATF_PERM | ATF_PUBL;
905     if (ioctl(sockfd, SIOCSARP, (caddr_t)&arpreq) < 0) {
906         syslog(LOG_ERR, "ioctl(SIOCSARP): %m");
907         return 0;
908     }
909
910     return 1;
911 }
912
913 /*
914  * cifproxyarp - Delete the proxy ARP entry for the peer.
915  */
916 int
917 cifproxyarp(unit, hisaddr)
918     int unit;
919     u_int32_t hisaddr;
920 {
921     struct arpreq arpreq;
922
923     BZERO(&arpreq, sizeof(arpreq));
924     SET_SA_FAMILY(arpreq.arp_pa, AF_INET);
925     ((struct sockaddr_in *) &arpreq.arp_pa)->sin_addr.s_addr = hisaddr;
926     if (ioctl(sockfd, SIOCDARP, (caddr_t)&arpreq) < 0) {
927         syslog(LOG_ERR, "ioctl(SIOCDARP): %m");
928         return 0;
929     }
930     return 1;
931 }
932
933 /*
934  * get_ether_addr - get the hardware address of an interface on the
935  * the same subnet as ipaddr.  Code borrowed from myetheraddr.c
936  * in the cslip-2.6 distribution, which is subject to the following
937  * copyright notice (which also applies to logwtmp below):
938  *
939  * Copyright (c) 1990, 1992 The Regents of the University of California.
940  * All rights reserved.
941  *
942  * Redistribution and use in source and binary forms, with or without
943  * modification, are permitted provided that: (1) source code distributions
944  * retain the above copyright notice and this paragraph in its entirety, (2)
945  * distributions including binary code include the above copyright notice and
946  * this paragraph in its entirety in the documentation or other materials
947  * provided with the distribution, and (3) all advertising materials mentioning
948  * features or use of this software display the following acknowledgement:
949  * ``This product includes software developed by the University of California,
950  * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
951  * the University nor the names of its contributors may be used to endorse
952  * or promote products derived from this software without specific prior
953  * written permission.
954  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
955  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
956  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
957  */
958
959 #include <fcntl.h>
960 #include <nlist.h>
961 #include <kvm.h>
962
963 /* XXX SunOS 4.1 defines this and 3.5 doesn't... */
964 #ifdef _nlist_h
965 #define SUNOS4
966 #endif
967
968 #ifdef SUNOS4
969 #include <netinet/in_var.h>
970 #endif
971 #include <netinet/if_ether.h>
972
973 /* Cast a struct sockaddr to a structaddr_in */
974 #define SATOSIN(sa) ((struct sockaddr_in *)(sa))
975
976 /* Determine if "bits" is set in "flag" */
977 #define ALLSET(flag, bits) (((flag) & (bits)) == (bits))
978
979 static struct nlist nl[] = {
980 #define N_IFNET 0
981         { "_ifnet" },
982         { 0 }
983 };
984
985 static void kread();
986
987 int
988 get_ether_addr(ipaddr, hwaddr)
989     u_int32_t ipaddr;
990     struct sockaddr *hwaddr;
991 {
992     register kvm_t *kd;
993     register struct ifnet *ifp;
994     register struct arpcom *ac;
995     struct arpcom arpcom;
996     struct in_addr *inp;
997 #ifdef SUNOS4
998     register struct ifaddr *ifa;
999     register struct in_ifaddr *in;
1000     union {
1001         struct ifaddr ifa;
1002         struct in_ifaddr in;
1003     } ifaddr;
1004 #endif
1005     u_int32_t addr, mask;
1006
1007     /* Open kernel memory for reading */
1008     kd = kvm_open(0, 0, 0, O_RDONLY, NULL);
1009     if (kd == 0) {
1010         syslog(LOG_ERR, "kvm_open: %m");
1011         return 0;
1012     }
1013
1014     /* Fetch namelist */
1015     if (kvm_nlist(kd, nl) != 0) {
1016         syslog(LOG_ERR, "kvm_nlist failed");
1017         return 0;
1018     }
1019
1020     ac = &arpcom;
1021     ifp = &arpcom.ac_if;
1022 #ifdef SUNOS4
1023     ifa = &ifaddr.ifa;
1024     in = &ifaddr.in;
1025 #endif
1026
1027     if (kvm_read(kd, nl[N_IFNET].n_value, (char *)&addr, sizeof(addr))
1028         != sizeof(addr)) {
1029         syslog(LOG_ERR, "error reading ifnet addr");
1030         return 0;
1031     }
1032     for ( ; addr; addr = (u_int32_t)ifp->if_next) {
1033         if (kvm_read(kd, addr, (char *)ac, sizeof(*ac)) != sizeof(*ac)) {
1034             syslog(LOG_ERR, "error reading ifnet");
1035             return 0;
1036         }
1037
1038         /* Only look at configured, broadcast interfaces */
1039         if (!ALLSET(ifp->if_flags, IFF_UP | IFF_BROADCAST))
1040             continue;
1041 #ifdef SUNOS4
1042         /* This probably can't happen... */
1043         if (ifp->if_addrlist == 0)
1044             continue;
1045 #endif
1046
1047         /* Get interface ip address */
1048 #ifdef SUNOS4
1049         if (kvm_read(kd, (u_int32_t)ifp->if_addrlist, (char *)&ifaddr,
1050                      sizeof(ifaddr)) != sizeof(ifaddr)) {
1051             syslog(LOG_ERR, "error reading ifaddr");
1052             return 0;
1053         }
1054         inp = &SATOSIN(&ifa->ifa_addr)->sin_addr;
1055 #else
1056         inp = &SATOSIN(&ifp->if_addr)->sin_addr;
1057 #endif
1058
1059         /* Check if this interface on the right subnet */
1060 #ifdef SUNOS4
1061         mask = in->ia_subnetmask;
1062 #else
1063         mask = ifp->if_subnetmask;
1064 #endif
1065         if ((ipaddr & mask) != (inp->s_addr & mask))
1066             continue;
1067
1068         /* Copy out the local ethernet address */
1069         hwaddr->sa_family = AF_UNSPEC;
1070         BCOPY((caddr_t) &arpcom.ac_enaddr, hwaddr->sa_data,
1071               sizeof(arpcom.ac_enaddr));
1072         return 1;               /* success! */
1073     }
1074
1075     /* couldn't find one */
1076     return 0;
1077 }
1078
1079 #define WTMPFILE        "/usr/adm/wtmp"
1080
1081 int
1082 logwtmp(line, name, host)
1083     char *line, *name, *host;
1084 {
1085     int fd;
1086     struct stat buf;
1087     struct utmp ut;
1088
1089     if ((fd = open(WTMPFILE, O_WRONLY|O_APPEND, 0)) < 0)
1090         return;
1091     if (!fstat(fd, &buf)) {
1092         (void)strncpy(ut.ut_line, line, sizeof(ut.ut_line));
1093         (void)strncpy(ut.ut_name, name, sizeof(ut.ut_name));
1094         (void)strncpy(ut.ut_host, host, sizeof(ut.ut_host));
1095         (void)time(&ut.ut_time);
1096         if (write(fd, (char *)&ut, sizeof(struct utmp)) != sizeof(struct utmp))
1097             (void)ftruncate(fd, buf.st_size);
1098     }
1099     close(fd);
1100 }
1101
1102 /*
1103  * Return user specified netmask, modified by any mask we might determine
1104  * for address `addr' (in network byte order).
1105  * Here we scan through the system's list of interfaces, looking for
1106  * any non-point-to-point interfaces which might appear to be on the same
1107  * network as `addr'.  If we find any, we OR in their netmask to the
1108  * user-specified netmask.
1109  */
1110 u_int32_t
1111 GetMask(addr)
1112     u_int32_t addr;
1113 {
1114     u_int32_t mask, nmask, ina;
1115     struct ifreq *ifr, *ifend, ifreq;
1116     struct ifconf ifc;
1117     struct ifreq ifs[MAX_IFS];
1118
1119     addr = ntohl(addr);
1120     if (IN_CLASSA(addr))        /* determine network mask for address class */
1121         nmask = IN_CLASSA_NET;
1122     else if (IN_CLASSB(addr))
1123         nmask = IN_CLASSB_NET;
1124     else
1125         nmask = IN_CLASSC_NET;
1126     /* class D nets are disallowed by bad_ip_adrs */
1127     mask = netmask | htonl(nmask);
1128
1129     /*
1130      * Scan through the system's network interfaces.
1131      */
1132     ifc.ifc_len = sizeof(ifs);
1133     ifc.ifc_req = ifs;
1134     if (ioctl(sockfd, SIOCGIFCONF, &ifc) < 0) {
1135         syslog(LOG_WARNING, "ioctl(SIOCGIFCONF): %m");
1136         return mask;
1137     }
1138     ifend = (struct ifreq *) (ifc.ifc_buf + ifc.ifc_len);
1139     for (ifr = ifc.ifc_req; ifr < ifend; ifr = (struct ifreq *)
1140                 ((char *)&ifr->ifr_addr + ifr->ifr_addr.sa_len)) {
1141         /*
1142          * Check the interface's internet address.
1143          */
1144         if (ifr->ifr_addr.sa_family != AF_INET)
1145             continue;
1146         ina = ((struct sockaddr_in *) &ifr->ifr_addr)->sin_addr.s_addr;
1147         if ((ntohl(ina) & nmask) != (addr & nmask))
1148             continue;
1149         /*
1150          * Check that the interface is up, and not point-to-point or loopback.
1151          */
1152         strncpy(ifreq.ifr_name, ifr->ifr_name, sizeof(ifreq.ifr_name));
1153         if (ioctl(sockfd, SIOCGIFFLAGS, &ifreq) < 0)
1154             continue;
1155         if ((ifreq.ifr_flags & (IFF_UP|IFF_POINTOPOINT|IFF_LOOPBACK))
1156             != IFF_UP)
1157             continue;
1158         /*
1159          * Get its netmask and OR it into our mask.
1160          */
1161         if (ioctl(sockfd, SIOCGIFNETMASK, &ifreq) < 0)
1162             continue;
1163         mask |= ((struct sockaddr_in *)&ifreq.ifr_addr)->sin_addr.s_addr;
1164     }
1165
1166     return mask;
1167 }
1168
1169 /*
1170  * Code for locking/unlocking the serial device.
1171  * This code is derived from chat.c.
1172  */
1173
1174 #if defined(SUNOS) && SUNOS >= 41 && !defined(HDB)
1175 #define HDB     1
1176 #endif
1177
1178 #ifndef LOCK_DIR
1179 # ifdef HDB
1180 #  define       PIDSTRING
1181 #  define       LOCK_PREFIX     "/usr/spool/locks/LCK.."
1182 # else /* HDB */
1183 #  define       LOCK_PREFIX     "/usr/spool/uucp/LCK.."
1184 # endif /* HDB */
1185 #endif /* LOCK_DIR */
1186
1187 /*
1188  * lock - create a lock file for the named device.
1189  */
1190 int
1191 lock(dev)
1192     char *dev;
1193 {
1194     char hdb_lock_buffer[12];
1195     int fd, pid, n;
1196     char *p;
1197
1198     if ((p = strrchr(dev, '/')) != NULL)
1199         dev = p + 1;
1200     lock_file = malloc(strlen(LOCK_PREFIX) + strlen(dev) + 1);
1201     if (lock_file == NULL)
1202         novm("lock file name");
1203     strcat(strcpy(lock_file, LOCK_PREFIX), dev);
1204
1205     while ((fd = open(lock_file, O_EXCL | O_CREAT | O_RDWR, 0644)) < 0) {
1206         if (errno == EEXIST
1207             && (fd = open(lock_file, O_RDONLY, 0)) >= 0) {
1208             /* Read the lock file to find out who has the device locked */
1209 #ifdef PIDSTRING
1210             n = read(fd, hdb_lock_buffer, 11);
1211             if (n > 0) {
1212                 hdb_lock_buffer[n] = 0;
1213                 pid = atoi(hdb_lock_buffer);
1214             }
1215 #else
1216             n = read(fd, &pid, sizeof(pid));
1217 #endif
1218             if (n <= 0) {
1219                 syslog(LOG_ERR, "Can't read pid from lock file %s", lock_file);
1220                 close(fd);
1221             } else {
1222                 if (kill(pid, 0) == -1 && errno == ESRCH) {
1223                     /* pid no longer exists - remove the lock file */
1224                     if (unlink(lock_file) == 0) {
1225                         close(fd);
1226                         syslog(LOG_NOTICE, "Removed stale lock on %s (pid %d)",
1227                                dev, pid);
1228                         continue;
1229                     } else
1230                         syslog(LOG_WARNING, "Couldn't remove stale lock on %s",
1231                                dev);
1232                 } else
1233                     syslog(LOG_NOTICE, "Device %s is locked by pid %d",
1234                            dev, pid);
1235             }
1236             close(fd);
1237         } else
1238             syslog(LOG_ERR, "Can't create lock file %s: %m", lock_file);
1239         free(lock_file);
1240         lock_file = NULL;
1241         return -1;
1242     }
1243
1244 #ifdef PIDSTRING
1245     sprintf(hdb_lock_buffer, "%10d\n", getpid());
1246     write(fd, hdb_lock_buffer, 11);
1247 #else
1248     pid = getpid();
1249     write(fd, &pid, sizeof pid);
1250 #endif
1251
1252     close(fd);
1253     return 0;
1254 }
1255
1256 /*
1257  * unlock - remove our lockfile
1258  */
1259 void
1260 unlock()
1261 {
1262     if (lock_file) {
1263         unlink(lock_file);
1264         free(lock_file);
1265         lock_file = NULL;
1266     }
1267 }