xref: /freebsd/usr.sbin/ppp/main.c (revision b6dec9f07f112fcc0ffbfaf32cfb0b817239f331)
11ae349f5Scvs2svn /*
21ae349f5Scvs2svn  *			User Process PPP
31ae349f5Scvs2svn  *
41ae349f5Scvs2svn  *	    Written by Toshiharu OHNO (tony-o@iij.ad.jp)
51ae349f5Scvs2svn  *
61ae349f5Scvs2svn  *   Copyright (C) 1993, Internet Initiative Japan, Inc. All rights reserverd.
71ae349f5Scvs2svn  *
81ae349f5Scvs2svn  * Redistribution and use in source and binary forms are permitted
91ae349f5Scvs2svn  * provided that the above copyright notice and this paragraph are
101ae349f5Scvs2svn  * duplicated in all such forms and that any documentation,
111ae349f5Scvs2svn  * advertising materials, and other materials related to such
121ae349f5Scvs2svn  * distribution and use acknowledge that the software was developed
131ae349f5Scvs2svn  * by the Internet Initiative Japan, Inc.  The name of the
141ae349f5Scvs2svn  * IIJ may not be used to endorse or promote products derived
151ae349f5Scvs2svn  * from this software without specific prior written permission.
161ae349f5Scvs2svn  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
171ae349f5Scvs2svn  * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
181ae349f5Scvs2svn  * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
191ae349f5Scvs2svn  *
20b6dec9f0SBrian Somers  * $Id: main.c,v 1.121.2.20 1998/02/10 22:28:51 brian Exp $
211ae349f5Scvs2svn  *
221ae349f5Scvs2svn  *	TODO:
231ae349f5Scvs2svn  *		o Add commands for traffic summary, version display, etc.
241ae349f5Scvs2svn  *		o Add signal handler for misc controls.
251ae349f5Scvs2svn  */
261ae349f5Scvs2svn #include <sys/param.h>
271ae349f5Scvs2svn #include <sys/time.h>
281ae349f5Scvs2svn #include <sys/select.h>
291ae349f5Scvs2svn #include <sys/socket.h>
301ae349f5Scvs2svn #include <net/if.h>
311ae349f5Scvs2svn #include <net/if_tun.h>
321ae349f5Scvs2svn #include <netinet/in.h>
331ae349f5Scvs2svn #include <netinet/in_systm.h>
341ae349f5Scvs2svn #include <netinet/ip.h>
351ae349f5Scvs2svn #include <arpa/inet.h>
361ae349f5Scvs2svn 
371ae349f5Scvs2svn #include <errno.h>
381ae349f5Scvs2svn #include <fcntl.h>
391ae349f5Scvs2svn #include <paths.h>
401ae349f5Scvs2svn #include <signal.h>
411ae349f5Scvs2svn #include <stdio.h>
421ae349f5Scvs2svn #include <stdlib.h>
431ae349f5Scvs2svn #include <string.h>
441ae349f5Scvs2svn #include <sys/time.h>
451ae349f5Scvs2svn #include <sys/wait.h>
461ae349f5Scvs2svn #include <termios.h>
471ae349f5Scvs2svn #include <unistd.h>
481ae349f5Scvs2svn 
491ae349f5Scvs2svn #include "command.h"
501ae349f5Scvs2svn #include "mbuf.h"
511ae349f5Scvs2svn #include "log.h"
521ae349f5Scvs2svn #include "defs.h"
531ae349f5Scvs2svn #include "id.h"
541ae349f5Scvs2svn #include "timer.h"
551ae349f5Scvs2svn #include "fsm.h"
561ae349f5Scvs2svn #include "modem.h"
577a6f8720SBrian Somers #include "bundle.h"
581ae349f5Scvs2svn #include "hdlc.h"
591ae349f5Scvs2svn #include "lcp.h"
601ae349f5Scvs2svn #include "ccp.h"
6129e275ceSBrian Somers #include "iplist.h"
6229e275ceSBrian Somers #include "throughput.h"
631ae349f5Scvs2svn #include "ipcp.h"
641ae349f5Scvs2svn #include "loadalias.h"
651ae349f5Scvs2svn #include "vars.h"
661ae349f5Scvs2svn #include "auth.h"
671ae349f5Scvs2svn #include "filter.h"
681ae349f5Scvs2svn #include "systems.h"
691ae349f5Scvs2svn #include "ip.h"
701ae349f5Scvs2svn #include "sig.h"
711ae349f5Scvs2svn #include "main.h"
721ae349f5Scvs2svn #include "vjcomp.h"
731ae349f5Scvs2svn #include "async.h"
741ae349f5Scvs2svn #include "pathnames.h"
751ae349f5Scvs2svn #include "tun.h"
761ae349f5Scvs2svn #include "route.h"
778c07a7b2SBrian Somers #include "link.h"
7842d4d396SBrian Somers #include "descriptor.h"
7963b73463SBrian Somers #include "physical.h"
8077ff88adSBrian Somers #include "server.h"
8185b542cfSBrian Somers #include "prompt.h"
82b6dec9f0SBrian Somers #include "chat.h"
831ae349f5Scvs2svn 
841ae349f5Scvs2svn #ifndef O_NONBLOCK
851ae349f5Scvs2svn #ifdef O_NDELAY
861ae349f5Scvs2svn #define	O_NONBLOCK O_NDELAY
871ae349f5Scvs2svn #endif
881ae349f5Scvs2svn #endif
891ae349f5Scvs2svn 
901ae349f5Scvs2svn static pid_t BGPid = 0;
911ae349f5Scvs2svn static char pid_filename[MAXPATHLEN];
92b6dec9f0SBrian Somers int dial_up;
93b6dec9f0SBrian Somers int dialing;
941ae349f5Scvs2svn 
957a6f8720SBrian Somers static void DoLoop(struct bundle *);
961ae349f5Scvs2svn static void TerminalStop(int);
971ae349f5Scvs2svn static const char *ex_desc(int);
981ae349f5Scvs2svn 
9983d1af55SBrian Somers static struct bundle *SignalBundle;
100455aabc3SBrian Somers int CleaningUp;
1017a6f8720SBrian Somers 
1021ae349f5Scvs2svn void
1031ae349f5Scvs2svn Cleanup(int excode)
1041ae349f5Scvs2svn {
105455aabc3SBrian Somers   CleaningUp = 1;
106455aabc3SBrian Somers   reconnect(RECON_FALSE);
107455aabc3SBrian Somers   if (bundle_Phase(SignalBundle) != PHASE_DEAD) {
108455aabc3SBrian Somers     bundle_Close(SignalBundle, NULL);
109455aabc3SBrian Somers     return;
110455aabc3SBrian Somers   }
1111afedc4bSBrian Somers   AbortProgram(excode);
1121afedc4bSBrian Somers }
113455aabc3SBrian Somers 
1141afedc4bSBrian Somers void
1151afedc4bSBrian Somers AbortProgram(int excode)
1161afedc4bSBrian Somers {
11785b542cfSBrian Somers   prompt_Drop(&prompt, 1);
1181ae349f5Scvs2svn   ServerClose();
1191ae349f5Scvs2svn   ID0unlink(pid_filename);
1201ae349f5Scvs2svn   if (mode & MODE_BACKGROUND && BGFiledes[1] != -1) {
1211ae349f5Scvs2svn     char c = EX_ERRDEAD;
1221ae349f5Scvs2svn 
1231ae349f5Scvs2svn     if (write(BGFiledes[1], &c, 1) == 1)
1241ae349f5Scvs2svn       LogPrintf(LogPHASE, "Parent notified of failure.\n");
1251ae349f5Scvs2svn     else
1261ae349f5Scvs2svn       LogPrintf(LogPHASE, "Failed to notify parent of failure.\n");
1271ae349f5Scvs2svn     close(BGFiledes[1]);
1281ae349f5Scvs2svn   }
1291ae349f5Scvs2svn   LogPrintf(LogPHASE, "PPP Terminated (%s).\n", ex_desc(excode));
13085b542cfSBrian Somers   prompt_TtyOldMode(&prompt);
13183d1af55SBrian Somers   link_Destroy(physical2link(SignalBundle->physical));
1321ae349f5Scvs2svn   LogClose();
13368a0f0ccSBrian Somers   bundle_Destroy(SignalBundle);
1341ae349f5Scvs2svn   exit(excode);
1351ae349f5Scvs2svn }
1361ae349f5Scvs2svn 
1371ae349f5Scvs2svn static void
1381ae349f5Scvs2svn CloseConnection(int signo)
1391ae349f5Scvs2svn {
1401ae349f5Scvs2svn   /* NOTE, these are manual, we've done a setsid() */
1411ae349f5Scvs2svn   pending_signal(SIGINT, SIG_IGN);
1421ae349f5Scvs2svn   LogPrintf(LogPHASE, "Caught signal %d, abort connection\n", signo);
1431ae349f5Scvs2svn   reconnectState = RECON_FALSE;
1441ae349f5Scvs2svn   reconnectCount = 0;
145455aabc3SBrian Somers   link_Close(&SignalBundle->physical->link, SignalBundle, 0);
1461ae349f5Scvs2svn   dial_up = 0;
1471ae349f5Scvs2svn   pending_signal(SIGINT, CloseConnection);
1481ae349f5Scvs2svn }
1491ae349f5Scvs2svn 
1501ae349f5Scvs2svn static void
1511ae349f5Scvs2svn CloseSession(int signo)
1521ae349f5Scvs2svn {
1531ae349f5Scvs2svn   if (BGPid) {
1541ae349f5Scvs2svn     kill(BGPid, SIGINT);
1551ae349f5Scvs2svn     exit(EX_TERM);
1561ae349f5Scvs2svn   }
1571ae349f5Scvs2svn   LogPrintf(LogPHASE, "Signal %d, terminate.\n", signo);
1581ae349f5Scvs2svn   Cleanup(EX_TERM);
1591ae349f5Scvs2svn }
1601ae349f5Scvs2svn 
1611ae349f5Scvs2svn static void
1621ae349f5Scvs2svn TerminalCont(int signo)
1631ae349f5Scvs2svn {
1641ae349f5Scvs2svn   pending_signal(SIGCONT, SIG_DFL);
1651ae349f5Scvs2svn   pending_signal(SIGTSTP, TerminalStop);
16685b542cfSBrian Somers   prompt_TtyCommandMode(&prompt);
16785b542cfSBrian Somers   if (getpgrp() == prompt_pgrp(&prompt))
16885b542cfSBrian Somers     prompt_Display(&prompt, SignalBundle);
1691ae349f5Scvs2svn }
1701ae349f5Scvs2svn 
1711ae349f5Scvs2svn static void
1721ae349f5Scvs2svn TerminalStop(int signo)
1731ae349f5Scvs2svn {
1741ae349f5Scvs2svn   pending_signal(SIGCONT, TerminalCont);
17585b542cfSBrian Somers   prompt_TtyOldMode(&prompt);
1761ae349f5Scvs2svn   pending_signal(SIGTSTP, SIG_DFL);
1771ae349f5Scvs2svn   kill(getpid(), signo);
1781ae349f5Scvs2svn }
1791ae349f5Scvs2svn 
1801ae349f5Scvs2svn static void
1811ae349f5Scvs2svn SetUpServer(int signo)
1821ae349f5Scvs2svn {
1831ae349f5Scvs2svn   int res;
1841ae349f5Scvs2svn 
1851ae349f5Scvs2svn   VarHaveLocalAuthKey = 0;
1861ae349f5Scvs2svn   LocalAuthInit();
187455aabc3SBrian Somers   if ((res = ServerTcpOpen(SERVER_PORT + SignalBundle->unit)) != 0)
1881ae349f5Scvs2svn     LogPrintf(LogERROR, "SIGUSR1: Failed %d to open port %d\n",
189455aabc3SBrian Somers 	      res, SERVER_PORT + SignalBundle->unit);
1901ae349f5Scvs2svn }
1911ae349f5Scvs2svn 
1921ae349f5Scvs2svn static void
1931ae349f5Scvs2svn BringDownServer(int signo)
1941ae349f5Scvs2svn {
1951ae349f5Scvs2svn   VarHaveLocalAuthKey = 0;
1961ae349f5Scvs2svn   LocalAuthInit();
1971ae349f5Scvs2svn   ServerClose();
1981ae349f5Scvs2svn }
1991ae349f5Scvs2svn 
2001ae349f5Scvs2svn static const char *
2011ae349f5Scvs2svn ex_desc(int ex)
2021ae349f5Scvs2svn {
2031ae349f5Scvs2svn   static char num[12];
2041ae349f5Scvs2svn   static const char *desc[] = {
2051ae349f5Scvs2svn     "normal", "start", "sock", "modem", "dial", "dead", "done",
2061ae349f5Scvs2svn     "reboot", "errdead", "hangup", "term", "nodial", "nologin"
2071ae349f5Scvs2svn   };
2081ae349f5Scvs2svn 
2091ae349f5Scvs2svn   if (ex >= 0 && ex < sizeof desc / sizeof *desc)
2101ae349f5Scvs2svn     return desc[ex];
2111ae349f5Scvs2svn   snprintf(num, sizeof num, "%d", ex);
2121ae349f5Scvs2svn   return num;
2131ae349f5Scvs2svn }
2141ae349f5Scvs2svn 
2151ae349f5Scvs2svn static void
2161ae349f5Scvs2svn Usage(void)
2171ae349f5Scvs2svn {
2181ae349f5Scvs2svn   fprintf(stderr,
2191ae349f5Scvs2svn 	  "Usage: ppp [-auto | -background | -direct | -dedicated | -ddial ]"
2201ae349f5Scvs2svn #ifndef NOALIAS
2211ae349f5Scvs2svn           " [ -alias ]"
2221ae349f5Scvs2svn #endif
2231ae349f5Scvs2svn           " [system]\n");
2241ae349f5Scvs2svn   exit(EX_START);
2251ae349f5Scvs2svn }
2261ae349f5Scvs2svn 
2271ae349f5Scvs2svn static char *
2281ae349f5Scvs2svn ProcessArgs(int argc, char **argv)
2291ae349f5Scvs2svn {
2301ae349f5Scvs2svn   int optc;
2311ae349f5Scvs2svn   char *cp;
2321ae349f5Scvs2svn 
2331ae349f5Scvs2svn   optc = 0;
2341ae349f5Scvs2svn   mode = MODE_INTER;
2351ae349f5Scvs2svn   while (argc > 0 && **argv == '-') {
2361ae349f5Scvs2svn     cp = *argv + 1;
2371ae349f5Scvs2svn     if (strcmp(cp, "auto") == 0) {
2381ae349f5Scvs2svn       mode |= MODE_AUTO;
2391ae349f5Scvs2svn       mode &= ~MODE_INTER;
2401ae349f5Scvs2svn     } else if (strcmp(cp, "background") == 0) {
2411ae349f5Scvs2svn       mode |= MODE_BACKGROUND;
2421ae349f5Scvs2svn       mode &= ~MODE_INTER;
2431ae349f5Scvs2svn     } else if (strcmp(cp, "direct") == 0) {
2441ae349f5Scvs2svn       mode |= MODE_DIRECT;
2451ae349f5Scvs2svn       mode &= ~MODE_INTER;
2461ae349f5Scvs2svn     } else if (strcmp(cp, "dedicated") == 0) {
2471ae349f5Scvs2svn       mode |= MODE_DEDICATED;
2481ae349f5Scvs2svn       mode &= ~MODE_INTER;
2491ae349f5Scvs2svn     } else if (strcmp(cp, "ddial") == 0) {
2501ae349f5Scvs2svn       mode |= MODE_DDIAL;
2511ae349f5Scvs2svn       mode &= ~MODE_INTER;
2521ae349f5Scvs2svn #ifndef NOALIAS
2531ae349f5Scvs2svn     } else if (strcmp(cp, "alias") == 0) {
2541ae349f5Scvs2svn       if (loadAliasHandlers(&VarAliasHandlers) == 0)
2551ae349f5Scvs2svn 	mode |= MODE_ALIAS;
2561ae349f5Scvs2svn       else
2571ae349f5Scvs2svn 	LogPrintf(LogWARN, "Cannot load alias library\n");
2581ae349f5Scvs2svn       optc--;			/* this option isn't exclusive */
2591ae349f5Scvs2svn #endif
2601ae349f5Scvs2svn     } else
2611ae349f5Scvs2svn       Usage();
2621ae349f5Scvs2svn     optc++;
2631ae349f5Scvs2svn     argv++;
2641ae349f5Scvs2svn     argc--;
2651ae349f5Scvs2svn   }
2661ae349f5Scvs2svn   if (argc > 1) {
2671ae349f5Scvs2svn     fprintf(stderr, "specify only one system label.\n");
2681ae349f5Scvs2svn     exit(EX_START);
2691ae349f5Scvs2svn   }
2701ae349f5Scvs2svn 
2711ae349f5Scvs2svn   if (optc > 1) {
2721ae349f5Scvs2svn     fprintf(stderr, "specify only one mode.\n");
2731ae349f5Scvs2svn     exit(EX_START);
2741ae349f5Scvs2svn   }
2751ae349f5Scvs2svn 
2761ae349f5Scvs2svn   return argc == 1 ? *argv : NULL;	/* Don't SetLabel yet ! */
2771ae349f5Scvs2svn }
2781ae349f5Scvs2svn 
2791ae349f5Scvs2svn int
2801ae349f5Scvs2svn main(int argc, char **argv)
2811ae349f5Scvs2svn {
2821ae349f5Scvs2svn   FILE *lockfile;
2831ae349f5Scvs2svn   char *name, *label;
2841ae349f5Scvs2svn   int nfds;
2857a6f8720SBrian Somers   struct bundle *bundle;
2861ae349f5Scvs2svn 
2871ae349f5Scvs2svn   nfds = getdtablesize();
2881ae349f5Scvs2svn   if (nfds >= FD_SETSIZE)
2891ae349f5Scvs2svn     /*
2901ae349f5Scvs2svn      * If we've got loads of file descriptors, make sure they're all
2911ae349f5Scvs2svn      * closed.  If they aren't, we may end up with a seg fault when our
2921ae349f5Scvs2svn      * `fd_set's get too big when select()ing !
2931ae349f5Scvs2svn      */
2941ae349f5Scvs2svn     while (--nfds > 2)
2951ae349f5Scvs2svn       close(nfds);
2961ae349f5Scvs2svn 
2971ae349f5Scvs2svn   name = strrchr(argv[0], '/');
2981ae349f5Scvs2svn   LogOpen(name ? name + 1 : argv[0]);
2991ae349f5Scvs2svn 
3001ae349f5Scvs2svn   argc--;
3011ae349f5Scvs2svn   argv++;
3021ae349f5Scvs2svn   label = ProcessArgs(argc, argv);
30385b542cfSBrian Somers 
30485b542cfSBrian Somers #ifdef __FreeBSD__
30585b542cfSBrian Somers   /*
30685b542cfSBrian Somers    * A FreeBSD hack to dodge a bug in the tty driver that drops output
30785b542cfSBrian Somers    * occasionally.... I must find the real reason some time.  To display
30885b542cfSBrian Somers    * the dodgy behaviour, comment out this bit, make yourself a large
30985b542cfSBrian Somers    * routing table and then run ppp in interactive mode.  The `show route'
31085b542cfSBrian Somers    * command will drop chunks of data !!!
31185b542cfSBrian Somers    */
31285b542cfSBrian Somers   if (mode & MODE_INTER) {
31385b542cfSBrian Somers     close(STDIN_FILENO);
31485b542cfSBrian Somers     if (open(_PATH_TTY, O_RDONLY) != STDIN_FILENO) {
31585b542cfSBrian Somers       fprintf(stderr, "Cannot open %s for input !\n", _PATH_TTY);
31685b542cfSBrian Somers       return 2;
31785b542cfSBrian Somers     }
31885b542cfSBrian Somers   }
31985b542cfSBrian Somers #endif
32085b542cfSBrian Somers 
32185b542cfSBrian Somers   prompt_Init(&prompt, (mode & MODE_DIRECT) ? PROMPT_NONE : PROMPT_STD);
3221ae349f5Scvs2svn 
3231ae349f5Scvs2svn   ID0init();
3241ae349f5Scvs2svn   if (ID0realuid() != 0) {
3251ae349f5Scvs2svn     char conf[200], *ptr;
3261ae349f5Scvs2svn 
3271ae349f5Scvs2svn     snprintf(conf, sizeof conf, "%s/%s", _PATH_PPP, CONFFILE);
3281ae349f5Scvs2svn     do {
3291ae349f5Scvs2svn       if (!access(conf, W_OK)) {
3301ae349f5Scvs2svn         LogPrintf(LogALERT, "ppp: Access violation: Please protect %s\n", conf);
3311ae349f5Scvs2svn         return -1;
3321ae349f5Scvs2svn       }
3331ae349f5Scvs2svn       ptr = conf + strlen(conf)-2;
3341ae349f5Scvs2svn       while (ptr > conf && *ptr != '/')
3351ae349f5Scvs2svn         *ptr-- = '\0';
3361ae349f5Scvs2svn     } while (ptr >= conf);
3371ae349f5Scvs2svn   }
3381ae349f5Scvs2svn 
3391ae349f5Scvs2svn   if (!ValidSystem(label)) {
3401ae349f5Scvs2svn     fprintf(stderr, "You may not use ppp in this mode with this label\n");
3411ae349f5Scvs2svn     if (mode & MODE_DIRECT) {
3421ae349f5Scvs2svn       const char *l;
3431ae349f5Scvs2svn       l = label ? label : "default";
3441ae349f5Scvs2svn       LogPrintf(LogWARN, "Label %s rejected -direct connection\n", l);
3451ae349f5Scvs2svn     }
3461ae349f5Scvs2svn     LogClose();
3471ae349f5Scvs2svn     return 1;
3481ae349f5Scvs2svn   }
3491ae349f5Scvs2svn 
3501ae349f5Scvs2svn   if (mode & MODE_INTER)
3511ae349f5Scvs2svn     VarLocalAuth = LOCAL_AUTH;
3521ae349f5Scvs2svn 
3537a6f8720SBrian Somers   if ((bundle = bundle_Create("/dev/tun")) == NULL) {
3547a6f8720SBrian Somers     LogPrintf(LogWARN, "bundle_Create: %s\n", strerror(errno));
3551ae349f5Scvs2svn     return EX_START;
3561ae349f5Scvs2svn   }
3577a6f8720SBrian Somers 
3582289f246SBrian Somers   if (!GetShortHost())
3592289f246SBrian Somers     return 1;
3602289f246SBrian Somers   IsInteractive(1);
3612289f246SBrian Somers 
36283d1af55SBrian Somers   SignalBundle = bundle;
3637a6f8720SBrian Somers 
36485b542cfSBrian Somers   if (SelectSystem(bundle, "default", CONFFILE) < 0)
36585b542cfSBrian Somers     prompt_Printf(&prompt,
36685b542cfSBrian Somers                   "Warning: No default entry is given in config file.\n");
3677a6f8720SBrian Somers 
3681ae349f5Scvs2svn   if ((mode & MODE_OUTGOING_DAEMON) && !(mode & MODE_DEDICATED))
3691ae349f5Scvs2svn     if (label == NULL) {
37085b542cfSBrian Somers       prompt_Printf(&prompt, "Destination system must be specified in"
3711ae349f5Scvs2svn 		    " auto, background or ddial mode.\n");
3721ae349f5Scvs2svn       return EX_START;
3731ae349f5Scvs2svn     }
3741ae349f5Scvs2svn 
3751ae349f5Scvs2svn   pending_signal(SIGHUP, CloseSession);
3761ae349f5Scvs2svn   pending_signal(SIGTERM, CloseSession);
3771ae349f5Scvs2svn   pending_signal(SIGINT, CloseConnection);
3781ae349f5Scvs2svn   pending_signal(SIGQUIT, CloseSession);
3791ae349f5Scvs2svn #ifdef SIGPIPE
3801ae349f5Scvs2svn   signal(SIGPIPE, SIG_IGN);
3811ae349f5Scvs2svn #endif
3821ae349f5Scvs2svn #ifdef SIGALRM
3831ae349f5Scvs2svn   pending_signal(SIGALRM, SIG_IGN);
3841ae349f5Scvs2svn #endif
3851ae349f5Scvs2svn   if (mode & MODE_INTER) {
3861ae349f5Scvs2svn #ifdef SIGTSTP
3871ae349f5Scvs2svn     pending_signal(SIGTSTP, TerminalStop);
3881ae349f5Scvs2svn #endif
3891ae349f5Scvs2svn #ifdef SIGTTIN
3901ae349f5Scvs2svn     pending_signal(SIGTTIN, TerminalStop);
3911ae349f5Scvs2svn #endif
3921ae349f5Scvs2svn #ifdef SIGTTOU
3931ae349f5Scvs2svn     pending_signal(SIGTTOU, SIG_IGN);
3941ae349f5Scvs2svn #endif
3951ae349f5Scvs2svn   }
3961ae349f5Scvs2svn   if (!(mode & MODE_INTER)) {
3971ae349f5Scvs2svn #ifdef SIGUSR1
3981ae349f5Scvs2svn     pending_signal(SIGUSR1, SetUpServer);
3991ae349f5Scvs2svn #endif
4001ae349f5Scvs2svn #ifdef SIGUSR2
4011ae349f5Scvs2svn     pending_signal(SIGUSR2, BringDownServer);
4021ae349f5Scvs2svn #endif
4031ae349f5Scvs2svn   }
4041ae349f5Scvs2svn 
4051ae349f5Scvs2svn   if (label) {
4067a6f8720SBrian Somers     if (SelectSystem(bundle, label, CONFFILE) < 0) {
4071ae349f5Scvs2svn       LogPrintf(LogWARN, "Destination system %s not found in conf file.\n",
4081ae349f5Scvs2svn                 GetLabel());
4091ae349f5Scvs2svn       Cleanup(EX_START);
4101ae349f5Scvs2svn     }
4111ae349f5Scvs2svn     /*
4121ae349f5Scvs2svn      * We don't SetLabel() 'till now in case SelectSystem() has an
4131ae349f5Scvs2svn      * embeded load "otherlabel" command.
4141ae349f5Scvs2svn      */
4151ae349f5Scvs2svn     SetLabel(label);
416aad81d1eSBrian Somers     if (mode & MODE_AUTO &&
41729e275ceSBrian Somers 	IpcpInfo.DefHisAddress.ipaddr.s_addr == INADDR_ANY) {
418aad81d1eSBrian Somers       LogPrintf(LogWARN, "You must \"set ifaddr\" in label %s for auto mode.\n",
419aad81d1eSBrian Somers 		label);
4201ae349f5Scvs2svn       Cleanup(EX_START);
4211ae349f5Scvs2svn     }
4221ae349f5Scvs2svn   }
4231ae349f5Scvs2svn 
4241ae349f5Scvs2svn   if (mode & MODE_DAEMON) {
4251ae349f5Scvs2svn     if (mode & MODE_BACKGROUND) {
4261ae349f5Scvs2svn       if (pipe(BGFiledes)) {
4271ae349f5Scvs2svn 	LogPrintf(LogERROR, "pipe: %s\n", strerror(errno));
4281ae349f5Scvs2svn 	Cleanup(EX_SOCK);
4291ae349f5Scvs2svn       }
4301ae349f5Scvs2svn     }
4311ae349f5Scvs2svn 
4321ae349f5Scvs2svn     if (!(mode & MODE_DIRECT)) {
4331ae349f5Scvs2svn       pid_t bgpid;
4341ae349f5Scvs2svn 
4351ae349f5Scvs2svn       bgpid = fork();
4361ae349f5Scvs2svn       if (bgpid == -1) {
4371ae349f5Scvs2svn 	LogPrintf(LogERROR, "fork: %s\n", strerror(errno));
4381ae349f5Scvs2svn 	Cleanup(EX_SOCK);
4391ae349f5Scvs2svn       }
4401ae349f5Scvs2svn       if (bgpid) {
4411ae349f5Scvs2svn 	char c = EX_NORMAL;
4421ae349f5Scvs2svn 
4431ae349f5Scvs2svn 	if (mode & MODE_BACKGROUND) {
4441ae349f5Scvs2svn 	  /* Wait for our child to close its pipe before we exit. */
4451ae349f5Scvs2svn 	  BGPid = bgpid;
4461ae349f5Scvs2svn 	  close(BGFiledes[1]);
4471ae349f5Scvs2svn 	  if (read(BGFiledes[0], &c, 1) != 1) {
44885b542cfSBrian Somers 	    prompt_Printf(&prompt, "Child exit, no status.\n");
4491ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: Child exit, no status.\n");
4501ae349f5Scvs2svn 	  } else if (c == EX_NORMAL) {
45185b542cfSBrian Somers 	    prompt_Printf(&prompt, "PPP enabled.\n");
4521ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: PPP enabled.\n");
4531ae349f5Scvs2svn 	  } else {
45485b542cfSBrian Somers 	    prompt_Printf(&prompt, "Child failed (%s).\n", ex_desc((int) c));
4551ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: Child failed (%s).\n",
4561ae349f5Scvs2svn 		      ex_desc((int) c));
4571ae349f5Scvs2svn 	  }
4581ae349f5Scvs2svn 	  close(BGFiledes[0]);
4591ae349f5Scvs2svn 	}
4601ae349f5Scvs2svn 	return c;
4611ae349f5Scvs2svn       } else if (mode & MODE_BACKGROUND)
4621ae349f5Scvs2svn 	close(BGFiledes[0]);
4631ae349f5Scvs2svn     }
4641ae349f5Scvs2svn 
46585b542cfSBrian Somers     prompt_Init(&prompt, PROMPT_NONE);
4661ae349f5Scvs2svn     close(STDOUT_FILENO);
4671ae349f5Scvs2svn     close(STDERR_FILENO);
4681ae349f5Scvs2svn 
4691ae349f5Scvs2svn     if (mode & MODE_DIRECT)
470ecd5172aSBrian Somers       /* STDIN_FILENO gets used by modem_Open in DIRECT mode */
47185b542cfSBrian Somers       prompt_TtyInit(&prompt, PROMPT_DONT_WANT_INT);
4721ae349f5Scvs2svn     else if (mode & MODE_DAEMON) {
4731ae349f5Scvs2svn       setsid();
4741ae349f5Scvs2svn       close(STDIN_FILENO);
4751ae349f5Scvs2svn     }
4761ae349f5Scvs2svn   } else {
4771ae349f5Scvs2svn     close(STDERR_FILENO);
47885b542cfSBrian Somers     prompt_TtyInit(&prompt, PROMPT_WANT_INT);
47985b542cfSBrian Somers     prompt_TtyCommandMode(&prompt);
48085b542cfSBrian Somers     prompt_Display(&prompt, bundle);
4811ae349f5Scvs2svn   }
4821ae349f5Scvs2svn 
4831ae349f5Scvs2svn   snprintf(pid_filename, sizeof pid_filename, "%stun%d.pid",
484455aabc3SBrian Somers            _PATH_VARRUN, bundle->unit);
4851ae349f5Scvs2svn   lockfile = ID0fopen(pid_filename, "w");
4861ae349f5Scvs2svn   if (lockfile != NULL) {
4871ae349f5Scvs2svn     fprintf(lockfile, "%d\n", (int) getpid());
4881ae349f5Scvs2svn     fclose(lockfile);
4891ae349f5Scvs2svn   }
4901ae349f5Scvs2svn #ifndef RELEASE_CRUNCH
4911ae349f5Scvs2svn   else
4921ae349f5Scvs2svn     LogPrintf(LogALERT, "Warning: Can't create %s: %s\n",
4931ae349f5Scvs2svn               pid_filename, strerror(errno));
4941ae349f5Scvs2svn #endif
4951ae349f5Scvs2svn 
4961ae349f5Scvs2svn   LogPrintf(LogPHASE, "PPP Started.\n");
4971ae349f5Scvs2svn 
4981ae349f5Scvs2svn 
4991ae349f5Scvs2svn   do
5007a6f8720SBrian Somers     DoLoop(bundle);
5011ae349f5Scvs2svn   while (mode & MODE_DEDICATED);
5021ae349f5Scvs2svn 
5031afedc4bSBrian Somers   AbortProgram(EX_NORMAL);
5041afedc4bSBrian Somers 
5051afedc4bSBrian Somers   return EX_NORMAL;
5061ae349f5Scvs2svn }
5071ae349f5Scvs2svn 
5081ae349f5Scvs2svn /*
5091ae349f5Scvs2svn  *  Turn into packet mode, where we speak PPP.
5101ae349f5Scvs2svn  */
5111ae349f5Scvs2svn void
5127a6f8720SBrian Somers PacketMode(struct bundle *bundle, int delay)
5131ae349f5Scvs2svn {
5142289f246SBrian Somers   if (modem_Raw(bundle->physical) < 0) {
5151ae349f5Scvs2svn     LogPrintf(LogWARN, "PacketMode: Not connected.\n");
5161ae349f5Scvs2svn     return;
5171ae349f5Scvs2svn   }
5182289f246SBrian Somers   LcpInit(bundle, bundle->physical);
5192289f246SBrian Somers   IpcpInit(bundle, physical2link(bundle->physical));
5202289f246SBrian Somers   CcpInit(bundle, physical2link(bundle->physical));
5211ae349f5Scvs2svn   LcpUp();
5221ae349f5Scvs2svn 
5231ae349f5Scvs2svn   LcpOpen(delay);
52485b542cfSBrian Somers   prompt_TtyCommandMode(&prompt);
52585b542cfSBrian Somers   prompt_Printf(&prompt, "Packet mode.\n");
5261ae349f5Scvs2svn }
5271ae349f5Scvs2svn 
5281ae349f5Scvs2svn static struct pppTimer RedialTimer;
5291ae349f5Scvs2svn 
5301ae349f5Scvs2svn static void
5311ae349f5Scvs2svn RedialTimeout(void *v)
5321ae349f5Scvs2svn {
5331ae349f5Scvs2svn   StopTimer(&RedialTimer);
5341ae349f5Scvs2svn   LogPrintf(LogPHASE, "Redialing timer expired.\n");
5351ae349f5Scvs2svn }
5361ae349f5Scvs2svn 
5371ae349f5Scvs2svn static void
5381ae349f5Scvs2svn StartRedialTimer(int Timeout)
5391ae349f5Scvs2svn {
5401ae349f5Scvs2svn   StopTimer(&RedialTimer);
5411ae349f5Scvs2svn 
5421ae349f5Scvs2svn   if (Timeout) {
5431ae349f5Scvs2svn     RedialTimer.state = TIMER_STOPPED;
5441ae349f5Scvs2svn 
5451ae349f5Scvs2svn     if (Timeout > 0)
5461ae349f5Scvs2svn       RedialTimer.load = Timeout * SECTICKS;
5471ae349f5Scvs2svn     else
5481ae349f5Scvs2svn       RedialTimer.load = (random() % REDIAL_PERIOD) * SECTICKS;
5491ae349f5Scvs2svn 
5501ae349f5Scvs2svn     LogPrintf(LogPHASE, "Enter pause (%d) for redialing.\n",
5511ae349f5Scvs2svn 	      RedialTimer.load / SECTICKS);
5521ae349f5Scvs2svn 
5531ae349f5Scvs2svn     RedialTimer.func = RedialTimeout;
5541ae349f5Scvs2svn     StartTimer(&RedialTimer);
5551ae349f5Scvs2svn   }
5561ae349f5Scvs2svn }
5571ae349f5Scvs2svn 
5581ae349f5Scvs2svn static void
5597a6f8720SBrian Somers DoLoop(struct bundle *bundle)
5601ae349f5Scvs2svn {
5611ae349f5Scvs2svn   fd_set rfds, wfds, efds;
5621ae349f5Scvs2svn   int pri, i, n, wfd, nfds;
5631ae349f5Scvs2svn   struct timeval timeout, *tp;
5641ae349f5Scvs2svn   const u_char *cp;
5651ae349f5Scvs2svn   int tries;
5661ae349f5Scvs2svn   int qlen;
5671ae349f5Scvs2svn   int res;
5681ae349f5Scvs2svn   struct tun_data tun;
5691ae349f5Scvs2svn #define rbuff tun.data
5701ae349f5Scvs2svn 
5711ae349f5Scvs2svn   if (mode & MODE_DIRECT) {
5721ae349f5Scvs2svn     LogPrintf(LogDEBUG, "Opening modem\n");
5732289f246SBrian Somers     if (modem_Open(bundle->physical, bundle) < 0)
5741ae349f5Scvs2svn       return;
5751ae349f5Scvs2svn     LogPrintf(LogPHASE, "Packet mode enabled\n");
5767a6f8720SBrian Somers     PacketMode(bundle, VarOpenMode);
5771ae349f5Scvs2svn   } else if (mode & MODE_DEDICATED) {
5782289f246SBrian Somers     if (!link_IsActive(physical2link(bundle->physical)))
5792289f246SBrian Somers       while (modem_Open(bundle->physical, bundle) < 0)
5801ae349f5Scvs2svn 	nointr_sleep(VarReconnectTimer);
5811ae349f5Scvs2svn   }
5821ae349f5Scvs2svn 
5831ae349f5Scvs2svn   timeout.tv_sec = 0;
5841ae349f5Scvs2svn   timeout.tv_usec = 0;
5851ae349f5Scvs2svn   reconnectState = RECON_UNKNOWN;
5861ae349f5Scvs2svn 
5871ae349f5Scvs2svn   if (mode & MODE_BACKGROUND)
5881ae349f5Scvs2svn     dial_up = 1;		/* Bring the line up */
5891ae349f5Scvs2svn   else
5901ae349f5Scvs2svn     dial_up = 0;		/* XXXX */
5911ae349f5Scvs2svn   tries = 0;
5921ae349f5Scvs2svn   for (;;) {
5931ae349f5Scvs2svn     nfds = 0;
5941ae349f5Scvs2svn     FD_ZERO(&rfds);
5951ae349f5Scvs2svn     FD_ZERO(&wfds);
5961ae349f5Scvs2svn     FD_ZERO(&efds);
5971ae349f5Scvs2svn 
5981ae349f5Scvs2svn     /*
5991ae349f5Scvs2svn      * If the link is down and we're in DDIAL mode, bring it back up.
6001ae349f5Scvs2svn      */
6017308ec68SBrian Somers     if (mode & MODE_DDIAL && LcpInfo.fsm.state <= ST_CLOSED)
6021ae349f5Scvs2svn       dial_up = 1;
6031ae349f5Scvs2svn 
6041ae349f5Scvs2svn     /*
6051ae349f5Scvs2svn      * If we lost carrier and want to re-establish the connection due to the
6061ae349f5Scvs2svn      * "set reconnect" value, we'd better bring the line back up.
6071ae349f5Scvs2svn      */
608b6dec9f0SBrian Somers     if (!dialing && LcpInfo.fsm.state <= ST_CLOSED) {
6091ae349f5Scvs2svn       if (!dial_up && reconnectState == RECON_TRUE) {
6101ae349f5Scvs2svn 	if (++reconnectCount <= VarReconnectTries) {
6111ae349f5Scvs2svn 	  LogPrintf(LogPHASE, "Connection lost, re-establish (%d/%d)\n",
6121ae349f5Scvs2svn 		    reconnectCount, VarReconnectTries);
6131ae349f5Scvs2svn 	  StartRedialTimer(VarReconnectTimer);
6141ae349f5Scvs2svn 	  dial_up = 1;
6151ae349f5Scvs2svn 	} else {
6161ae349f5Scvs2svn 	  if (VarReconnectTries)
6171ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Connection lost, maximum (%d) times\n",
6181ae349f5Scvs2svn 		      VarReconnectTries);
6191ae349f5Scvs2svn 	  reconnectCount = 0;
6201ae349f5Scvs2svn 	  if (mode & MODE_BACKGROUND)
6211ae349f5Scvs2svn 	    Cleanup(EX_DEAD);
6221ae349f5Scvs2svn 	}
6231ae349f5Scvs2svn 	reconnectState = RECON_ENVOKED;
6241ae349f5Scvs2svn       } else if (mode & MODE_DEDICATED)
6257a6f8720SBrian Somers         PacketMode(bundle, VarOpenMode);
6261ae349f5Scvs2svn     }
6271ae349f5Scvs2svn 
6281ae349f5Scvs2svn     /*
6291ae349f5Scvs2svn      * If Ip packet for output is enqueued and require dial up, Just do it!
6301ae349f5Scvs2svn      */
631b6dec9f0SBrian Somers     if (dial_up && !dialing && RedialTimer.state != TIMER_RUNNING) {
63263b73463SBrian Somers       LogPrintf(LogDEBUG, "going to dial: modem = %d\n",
6332289f246SBrian Somers 		Physical_GetFD(bundle->physical));
6342289f246SBrian Somers       if (modem_Open(bundle->physical, bundle) < 0) {
6351ae349f5Scvs2svn 	tries++;
6361ae349f5Scvs2svn 	if (!(mode & MODE_DDIAL) && VarDialTries)
6371ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Failed to open modem (attempt %u of %d)\n",
6381ae349f5Scvs2svn 		    tries, VarDialTries);
6391ae349f5Scvs2svn 	else
6401ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Failed to open modem (attempt %u)\n", tries);
6411ae349f5Scvs2svn 
6421ae349f5Scvs2svn 	if (!(mode & MODE_DDIAL) && VarDialTries && tries >= VarDialTries) {
6431ae349f5Scvs2svn 	  if (mode & MODE_BACKGROUND)
6441ae349f5Scvs2svn 	    Cleanup(EX_DIAL);	/* Can't get the modem */
6451ae349f5Scvs2svn 	  dial_up = 0;
6461ae349f5Scvs2svn 	  reconnectState = RECON_UNKNOWN;
6471ae349f5Scvs2svn 	  reconnectCount = 0;
6481ae349f5Scvs2svn 	  tries = 0;
6491ae349f5Scvs2svn 	} else
6501ae349f5Scvs2svn 	  StartRedialTimer(VarRedialTimeout);
6511ae349f5Scvs2svn       } else {
6521ae349f5Scvs2svn 	tries++;		/* Tries are per number, not per list of
6531ae349f5Scvs2svn 				 * numbers. */
6541ae349f5Scvs2svn 	if (!(mode & MODE_DDIAL) && VarDialTries)
6551ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Dial attempt %u of %d\n", tries, VarDialTries);
6561ae349f5Scvs2svn 	else
6571ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Dial attempt %u\n", tries);
6581ae349f5Scvs2svn 
659b6dec9f0SBrian Somers         chat_Init(&chat, bundle->physical, VarDialScript, 1);
660b6dec9f0SBrian Somers         dialing = 1;
6611ae349f5Scvs2svn         dial_up = 0;
662b6dec9f0SBrian Somers       }
663b6dec9f0SBrian Somers     }
664b6dec9f0SBrian Somers 
665b6dec9f0SBrian Somers     qlen = link_QueueLen(physical2link(bundle->physical));
666b6dec9f0SBrian Somers     if (qlen == 0) {
667b6dec9f0SBrian Somers       IpStartOutput(physical2link(bundle->physical));
668b6dec9f0SBrian Somers       qlen = link_QueueLen(physical2link(bundle->physical));
669b6dec9f0SBrian Somers     }
670b6dec9f0SBrian Somers 
671b6dec9f0SBrian Somers     handle_signals();
672b6dec9f0SBrian Somers 
673b6dec9f0SBrian Somers     if (dialing) {
674b6dec9f0SBrian Somers       descriptor_UpdateSet(&chat.desc, &rfds, &wfds, &efds, &nfds);
675b6dec9f0SBrian Somers       if (dialing == -1) {
676b6dec9f0SBrian Somers         if (chat.state == CHAT_DONE || chat.state == CHAT_FAILED) {
677b6dec9f0SBrian Somers           dialing = 0;
678b6dec9f0SBrian Somers           modem_Close(bundle->physical);
6791ae349f5Scvs2svn           if (mode & MODE_BACKGROUND) {
6801ae349f5Scvs2svn             if (VarNextPhone == NULL || res == EX_SIG)
6811ae349f5Scvs2svn               Cleanup(EX_DIAL);	/* Tried all numbers - no luck */
6821ae349f5Scvs2svn             else
6831ae349f5Scvs2svn               /* Try all numbers in background mode */
6841ae349f5Scvs2svn               StartRedialTimer(VarRedialNextTimeout);
6851ae349f5Scvs2svn           } else if (!(mode & MODE_DDIAL) &&
6861ae349f5Scvs2svn       	             ((VarDialTries && tries >= VarDialTries) ||
6871ae349f5Scvs2svn       	              res == EX_SIG)) {
6881ae349f5Scvs2svn             /* I give up !  Can't get through :( */
6891ae349f5Scvs2svn             StartRedialTimer(VarRedialTimeout);
6901ae349f5Scvs2svn             dial_up = 0;
6911ae349f5Scvs2svn             reconnectState = RECON_UNKNOWN;
6921ae349f5Scvs2svn             reconnectCount = 0;
6931ae349f5Scvs2svn             tries = 0;
6941ae349f5Scvs2svn           } else if (VarNextPhone == NULL)
6951ae349f5Scvs2svn             /* Dial failed. Keep quite during redial wait period. */
6961ae349f5Scvs2svn             StartRedialTimer(VarRedialTimeout);
6971ae349f5Scvs2svn           else
6981ae349f5Scvs2svn             StartRedialTimer(VarRedialNextTimeout);
699b6dec9f0SBrian Somers           continue;
7001ae349f5Scvs2svn         }
701b6dec9f0SBrian Somers       } else if (chat.state == CHAT_DONE) {
702b6dec9f0SBrian Somers         if (dialing == 1) {
703b6dec9f0SBrian Somers           chat_Init(&chat, bundle->physical, VarLoginScript, 0);
704b6dec9f0SBrian Somers           dialing++;
705b6dec9f0SBrian Somers           continue;
706b6dec9f0SBrian Somers         } else {
707b6dec9f0SBrian Somers           PacketMode(bundle, VarOpenMode);
708b6dec9f0SBrian Somers           reconnectState = RECON_UNKNOWN;
709b6dec9f0SBrian Somers           tries = 0;
710b6dec9f0SBrian Somers           dialing = 0;
711b6dec9f0SBrian Somers         }
712b6dec9f0SBrian Somers       } else if (chat.state == CHAT_FAILED) {
713b6dec9f0SBrian Somers         chat_Init(&chat, bundle->physical, VarHangupScript, 0);
714b6dec9f0SBrian Somers         dialing = -1;
715b6dec9f0SBrian Somers         continue;
7161ae349f5Scvs2svn       }
7171ae349f5Scvs2svn     }
7181ae349f5Scvs2svn 
719b6dec9f0SBrian Somers     if (!dialing)
72042d4d396SBrian Somers       descriptor_UpdateSet(&bundle->physical->desc, &rfds, &wfds, &efds, &nfds);
72177ff88adSBrian Somers     descriptor_UpdateSet(&server.desc, &rfds, &wfds, &efds, &nfds);
7221ae349f5Scvs2svn 
7231ae349f5Scvs2svn #ifndef SIGALRM
7241ae349f5Scvs2svn     /*
7251ae349f5Scvs2svn      * *** IMPORTANT ***
7261ae349f5Scvs2svn      * CPU is serviced every TICKUNIT micro seconds. This value must be chosen
7271ae349f5Scvs2svn      * with great care. If this values is too big, it results in loss of
7281ae349f5Scvs2svn      * characters from the modem and poor response.  If this value is too
7291ae349f5Scvs2svn      * small, ppp eats too much CPU time.
7301ae349f5Scvs2svn      */
7311ae349f5Scvs2svn     usleep(TICKUNIT);
7321ae349f5Scvs2svn     TimerService();
7331ae349f5Scvs2svn #endif
7341ae349f5Scvs2svn 
7351ae349f5Scvs2svn     /* If there are aren't many packets queued, look for some more. */
7367a6f8720SBrian Somers     if (qlen < 20 && bundle->tun_fd >= 0) {
7377a6f8720SBrian Somers       if (bundle->tun_fd + 1 > nfds)
7387a6f8720SBrian Somers 	nfds = bundle->tun_fd + 1;
7397a6f8720SBrian Somers       FD_SET(bundle->tun_fd, &rfds);
7401ae349f5Scvs2svn     }
74177ff88adSBrian Somers 
74285b542cfSBrian Somers     descriptor_UpdateSet(&prompt.desc, &rfds, &wfds, &efds, &nfds);
74377ff88adSBrian Somers 
7441ae349f5Scvs2svn #ifndef SIGALRM
7451ae349f5Scvs2svn 
7461ae349f5Scvs2svn     /*
7471ae349f5Scvs2svn      * Normally, select() will not block because modem is writable. In AUTO
7481ae349f5Scvs2svn      * mode, select will block until we find packet from tun
7491ae349f5Scvs2svn      */
7501ae349f5Scvs2svn     tp = (RedialTimer.state == TIMER_RUNNING) ? &timeout : NULL;
7511ae349f5Scvs2svn     i = select(nfds, &rfds, &wfds, &efds, tp);
7521ae349f5Scvs2svn #else
7531ae349f5Scvs2svn 
7541ae349f5Scvs2svn     /*
75577ff88adSBrian Somers      * When SIGALRM timer is running, the select function will return -1 and
75677ff88adSBrian Somers      * EINTR after the Time Service signal handler is done.  If the redial
7571ae349f5Scvs2svn      * timer is not running and we are trying to dial, poll with a 0 value
7581ae349f5Scvs2svn      * timer.
7591ae349f5Scvs2svn      */
7601ae349f5Scvs2svn     tp = (dial_up && RedialTimer.state != TIMER_RUNNING) ? &timeout : NULL;
7611ae349f5Scvs2svn     i = select(nfds, &rfds, &wfds, &efds, tp);
7621ae349f5Scvs2svn #endif
7631ae349f5Scvs2svn 
7641ae349f5Scvs2svn     if (i == 0) {
7651ae349f5Scvs2svn       continue;
7661ae349f5Scvs2svn     }
76777ff88adSBrian Somers 
7681ae349f5Scvs2svn     if (i < 0) {
7691ae349f5Scvs2svn       if (errno == EINTR) {
7701ae349f5Scvs2svn 	handle_signals();
7711ae349f5Scvs2svn 	continue;
7721ae349f5Scvs2svn       }
7731ae349f5Scvs2svn       LogPrintf(LogERROR, "DoLoop: select(): %s\n", strerror(errno));
7741ae349f5Scvs2svn       break;
7751ae349f5Scvs2svn     }
77677ff88adSBrian Somers 
77785b542cfSBrian Somers     if (descriptor_IsSet(&prompt.desc, &efds) ||
77842d4d396SBrian Somers         descriptor_IsSet(&bundle->physical->desc, &efds)) {
7791ae349f5Scvs2svn       LogPrintf(LogALERT, "Exception detected.\n");
7801ae349f5Scvs2svn       break;
7811ae349f5Scvs2svn     }
78277ff88adSBrian Somers 
78377ff88adSBrian Somers     if (descriptor_IsSet(&server.desc, &rfds))
784b77776a7SBrian Somers       descriptor_Read(&server.desc, bundle, &rfds);
78542d4d396SBrian Somers 
78685b542cfSBrian Somers     if (descriptor_IsSet(&prompt.desc, &rfds))
78785b542cfSBrian Somers       descriptor_Read(&prompt.desc, bundle, &rfds);
78842d4d396SBrian Somers 
789b6dec9f0SBrian Somers     if (dialing) {
790b6dec9f0SBrian Somers       if (descriptor_IsSet(&chat.desc, &wfds))
791b6dec9f0SBrian Somers         descriptor_Write(&chat.desc, &wfds);
792b6dec9f0SBrian Somers       if (descriptor_IsSet(&chat.desc, &rfds))
793b6dec9f0SBrian Somers         descriptor_Read(&chat.desc, bundle, &rfds);
794b6dec9f0SBrian Somers     } else {
79542d4d396SBrian Somers       if (descriptor_IsSet(&bundle->physical->desc, &wfds)) {
7961ae349f5Scvs2svn         /* ready to write into modem */
797b77776a7SBrian Somers         descriptor_Write(&bundle->physical->desc, &wfds);
7982289f246SBrian Somers         if (!link_IsActive(physical2link(bundle->physical)))
7991ae349f5Scvs2svn           dial_up = 1;
8001ae349f5Scvs2svn       }
8011ae349f5Scvs2svn 
80242d4d396SBrian Somers       if (descriptor_IsSet(&bundle->physical->desc, &rfds))
803b77776a7SBrian Somers         descriptor_Read(&bundle->physical->desc, bundle, &rfds);
804b6dec9f0SBrian Somers     }
80542d4d396SBrian Somers 
8067a6f8720SBrian Somers     if (bundle->tun_fd >= 0 && FD_ISSET(bundle->tun_fd, &rfds)) {
8077a6f8720SBrian Somers       /* something to read from tun */
8087a6f8720SBrian Somers       n = read(bundle->tun_fd, &tun, sizeof tun);
8091ae349f5Scvs2svn       if (n < 0) {
8101ae349f5Scvs2svn 	LogPrintf(LogERROR, "read from tun: %s\n", strerror(errno));
8111ae349f5Scvs2svn 	continue;
8121ae349f5Scvs2svn       }
8131ae349f5Scvs2svn       n -= sizeof tun - sizeof tun.data;
8141ae349f5Scvs2svn       if (n <= 0) {
8151ae349f5Scvs2svn 	LogPrintf(LogERROR, "read from tun: Only %d bytes read\n", n);
8161ae349f5Scvs2svn 	continue;
8171ae349f5Scvs2svn       }
8181ae349f5Scvs2svn       if (!tun_check_header(tun, AF_INET))
8191ae349f5Scvs2svn           continue;
8201ae349f5Scvs2svn       if (((struct ip *) rbuff)->ip_dst.s_addr == IpcpInfo.want_ipaddr.s_addr) {
8211ae349f5Scvs2svn 	/* we've been asked to send something addressed *to* us :( */
8221ae349f5Scvs2svn 	if (VarLoopback) {
8231ae349f5Scvs2svn 	  pri = PacketCheck(rbuff, n, FL_IN);
8241ae349f5Scvs2svn 	  if (pri >= 0) {
8251ae349f5Scvs2svn 	    struct mbuf *bp;
8261ae349f5Scvs2svn 
8271ae349f5Scvs2svn #ifndef NOALIAS
8281ae349f5Scvs2svn 	    if (mode & MODE_ALIAS) {
8291ae349f5Scvs2svn 	      VarPacketAliasIn(rbuff, sizeof rbuff);
8301ae349f5Scvs2svn 	      n = ntohs(((struct ip *) rbuff)->ip_len);
8311ae349f5Scvs2svn 	    }
8321ae349f5Scvs2svn #endif
8331ae349f5Scvs2svn 	    bp = mballoc(n, MB_IPIN);
8341ae349f5Scvs2svn 	    memcpy(MBUF_CTOP(bp), rbuff, n);
8357a6f8720SBrian Somers 	    IpInput(bundle, bp);
8361ae349f5Scvs2svn 	    LogPrintf(LogDEBUG, "Looped back packet addressed to myself\n");
8371ae349f5Scvs2svn 	  }
8381ae349f5Scvs2svn 	  continue;
8391ae349f5Scvs2svn 	} else
8401ae349f5Scvs2svn 	  LogPrintf(LogDEBUG, "Oops - forwarding packet addressed to myself\n");
8411ae349f5Scvs2svn       }
8421ae349f5Scvs2svn 
8431ae349f5Scvs2svn       /*
8441ae349f5Scvs2svn        * Process on-demand dialup. Output packets are queued within tunnel
8451ae349f5Scvs2svn        * device until IPCP is opened.
8461ae349f5Scvs2svn        */
8477308ec68SBrian Somers       if (LcpInfo.fsm.state <= ST_CLOSED && (mode & MODE_AUTO) &&
8481ae349f5Scvs2svn 	  (pri = PacketCheck(rbuff, n, FL_DIAL)) >= 0)
8491ae349f5Scvs2svn         dial_up = 1;
8501ae349f5Scvs2svn 
8511ae349f5Scvs2svn       pri = PacketCheck(rbuff, n, FL_OUT);
8521ae349f5Scvs2svn       if (pri >= 0) {
8531ae349f5Scvs2svn #ifndef NOALIAS
8541ae349f5Scvs2svn 	if (mode & MODE_ALIAS) {
8551ae349f5Scvs2svn 	  VarPacketAliasOut(rbuff, sizeof rbuff);
8561ae349f5Scvs2svn 	  n = ntohs(((struct ip *) rbuff)->ip_len);
8571ae349f5Scvs2svn 	}
8581ae349f5Scvs2svn #endif
8591ae349f5Scvs2svn 	IpEnqueue(pri, rbuff, n);
8601ae349f5Scvs2svn       }
8611ae349f5Scvs2svn     }
8621ae349f5Scvs2svn   }
8631ae349f5Scvs2svn   LogPrintf(LogDEBUG, "Job (DoLoop) done.\n");
8641ae349f5Scvs2svn }
865