xref: /freebsd/usr.sbin/ppp/main.c (revision 7a6f872047b686aa10864866e8c80b7d591f4f45)
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  *
207a6f8720SBrian Somers  * $Id: main.c,v 1.121.2.4 1998/01/31 02:48:25 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 "server.h"
721ae349f5Scvs2svn #include "main.h"
731ae349f5Scvs2svn #include "vjcomp.h"
741ae349f5Scvs2svn #include "async.h"
751ae349f5Scvs2svn #include "pathnames.h"
761ae349f5Scvs2svn #include "tun.h"
771ae349f5Scvs2svn #include "route.h"
788c07a7b2SBrian Somers #include "link.h"
7963b73463SBrian Somers #include "physical.h"
801ae349f5Scvs2svn 
811ae349f5Scvs2svn #ifndef O_NONBLOCK
821ae349f5Scvs2svn #ifdef O_NDELAY
831ae349f5Scvs2svn #define	O_NONBLOCK O_NDELAY
841ae349f5Scvs2svn #endif
851ae349f5Scvs2svn #endif
861ae349f5Scvs2svn 
871ae349f5Scvs2svn int TermMode = 0;
881ae349f5Scvs2svn int tunno = 0;
891ae349f5Scvs2svn 
901ae349f5Scvs2svn static struct termios oldtio;	/* Original tty mode */
911ae349f5Scvs2svn static struct termios comtio;	/* Command level tty mode */
921ae349f5Scvs2svn static pid_t BGPid = 0;
931ae349f5Scvs2svn static char pid_filename[MAXPATHLEN];
941ae349f5Scvs2svn static int dial_up;
951ae349f5Scvs2svn 
967a6f8720SBrian Somers static void DoLoop(struct bundle *);
971ae349f5Scvs2svn static void TerminalStop(int);
981ae349f5Scvs2svn static const char *ex_desc(int);
991ae349f5Scvs2svn 
1001ae349f5Scvs2svn static void
1011ae349f5Scvs2svn TtyInit(int DontWantInt)
1021ae349f5Scvs2svn {
1031ae349f5Scvs2svn   struct termios newtio;
1041ae349f5Scvs2svn   int stat;
1051ae349f5Scvs2svn 
1061ae349f5Scvs2svn   stat = fcntl(netfd, F_GETFL, 0);
1071ae349f5Scvs2svn   if (stat > 0) {
1081ae349f5Scvs2svn     stat |= O_NONBLOCK;
1091ae349f5Scvs2svn     (void) fcntl(netfd, F_SETFL, stat);
1101ae349f5Scvs2svn   }
1111ae349f5Scvs2svn   newtio = oldtio;
1121ae349f5Scvs2svn   newtio.c_lflag &= ~(ECHO | ISIG | ICANON);
1131ae349f5Scvs2svn   newtio.c_iflag = 0;
1141ae349f5Scvs2svn   newtio.c_oflag &= ~OPOST;
1151ae349f5Scvs2svn   newtio.c_cc[VEOF] = _POSIX_VDISABLE;
1161ae349f5Scvs2svn   if (DontWantInt)
1171ae349f5Scvs2svn     newtio.c_cc[VINTR] = _POSIX_VDISABLE;
1181ae349f5Scvs2svn   newtio.c_cc[VMIN] = 1;
1191ae349f5Scvs2svn   newtio.c_cc[VTIME] = 0;
1201ae349f5Scvs2svn   newtio.c_cflag |= CS8;
1211ae349f5Scvs2svn   tcsetattr(netfd, TCSANOW, &newtio);
1221ae349f5Scvs2svn   comtio = newtio;
1231ae349f5Scvs2svn }
1241ae349f5Scvs2svn 
1251ae349f5Scvs2svn /*
1261ae349f5Scvs2svn  *  Set tty into command mode. We allow canonical input and echo processing.
1271ae349f5Scvs2svn  */
1281ae349f5Scvs2svn void
1291ae349f5Scvs2svn TtyCommandMode(int prompt)
1301ae349f5Scvs2svn {
1311ae349f5Scvs2svn   struct termios newtio;
1321ae349f5Scvs2svn   int stat;
1331ae349f5Scvs2svn 
1341ae349f5Scvs2svn   if (!(mode & MODE_INTER))
1351ae349f5Scvs2svn     return;
1361ae349f5Scvs2svn   tcgetattr(netfd, &newtio);
1371ae349f5Scvs2svn   newtio.c_lflag |= (ECHO | ISIG | ICANON);
1381ae349f5Scvs2svn   newtio.c_iflag = oldtio.c_iflag;
1391ae349f5Scvs2svn   newtio.c_oflag |= OPOST;
1401ae349f5Scvs2svn   tcsetattr(netfd, TCSADRAIN, &newtio);
1411ae349f5Scvs2svn   stat = fcntl(netfd, F_GETFL, 0);
1421ae349f5Scvs2svn   if (stat > 0) {
1431ae349f5Scvs2svn     stat |= O_NONBLOCK;
1441ae349f5Scvs2svn     (void) fcntl(netfd, F_SETFL, stat);
1451ae349f5Scvs2svn   }
1461ae349f5Scvs2svn   TermMode = 0;
1471ae349f5Scvs2svn   if (prompt)
1481ae349f5Scvs2svn     Prompt();
1491ae349f5Scvs2svn }
1501ae349f5Scvs2svn 
1511ae349f5Scvs2svn /*
1521ae349f5Scvs2svn  * Set tty into terminal mode which is used while we invoke term command.
1531ae349f5Scvs2svn  */
1541ae349f5Scvs2svn void
1551ae349f5Scvs2svn TtyTermMode()
1561ae349f5Scvs2svn {
1571ae349f5Scvs2svn   int stat;
1581ae349f5Scvs2svn 
1591ae349f5Scvs2svn   tcsetattr(netfd, TCSADRAIN, &comtio);
1601ae349f5Scvs2svn   stat = fcntl(netfd, F_GETFL, 0);
1611ae349f5Scvs2svn   if (stat > 0) {
1621ae349f5Scvs2svn     stat &= ~O_NONBLOCK;
1631ae349f5Scvs2svn     (void) fcntl(netfd, F_SETFL, stat);
1641ae349f5Scvs2svn   }
1651ae349f5Scvs2svn   TermMode = 1;
1661ae349f5Scvs2svn }
1671ae349f5Scvs2svn 
1681ae349f5Scvs2svn void
1691ae349f5Scvs2svn TtyOldMode()
1701ae349f5Scvs2svn {
1711ae349f5Scvs2svn   int stat;
1721ae349f5Scvs2svn 
1731ae349f5Scvs2svn   stat = fcntl(netfd, F_GETFL, 0);
1741ae349f5Scvs2svn   if (stat > 0) {
1751ae349f5Scvs2svn     stat &= ~O_NONBLOCK;
1761ae349f5Scvs2svn     (void) fcntl(netfd, F_SETFL, stat);
1771ae349f5Scvs2svn   }
1781ae349f5Scvs2svn   tcsetattr(netfd, TCSADRAIN, &oldtio);
1791ae349f5Scvs2svn }
1801ae349f5Scvs2svn 
1817a6f8720SBrian Somers static struct bundle *CleanupBundle;
1827a6f8720SBrian Somers 
1831ae349f5Scvs2svn void
1841ae349f5Scvs2svn Cleanup(int excode)
1851ae349f5Scvs2svn {
1861ae349f5Scvs2svn   DropClient(1);
1871ae349f5Scvs2svn   ServerClose();
1887a6f8720SBrian Somers   bundle_InterfaceDown(CleanupBundle);
1898c07a7b2SBrian Somers   link_Close(physical2link(pppVars.physical), 1); /* XXX gotta get a handle on
1908c07a7b2SBrian Somers                                                    * the logical link */
1911ae349f5Scvs2svn   nointr_sleep(1);
1927a6f8720SBrian Somers   DeleteIfRoutes(CleanupBundle, 1);
1931ae349f5Scvs2svn   ID0unlink(pid_filename);
1941ae349f5Scvs2svn   if (mode & MODE_BACKGROUND && BGFiledes[1] != -1) {
1951ae349f5Scvs2svn     char c = EX_ERRDEAD;
1961ae349f5Scvs2svn 
1971ae349f5Scvs2svn     if (write(BGFiledes[1], &c, 1) == 1)
1981ae349f5Scvs2svn       LogPrintf(LogPHASE, "Parent notified of failure.\n");
1991ae349f5Scvs2svn     else
2001ae349f5Scvs2svn       LogPrintf(LogPHASE, "Failed to notify parent of failure.\n");
2011ae349f5Scvs2svn     close(BGFiledes[1]);
2021ae349f5Scvs2svn   }
2031ae349f5Scvs2svn   LogPrintf(LogPHASE, "PPP Terminated (%s).\n", ex_desc(excode));
2041ae349f5Scvs2svn   TtyOldMode();
2051ae349f5Scvs2svn   LogClose();
2061ae349f5Scvs2svn 
2071ae349f5Scvs2svn   exit(excode);
2081ae349f5Scvs2svn }
2091ae349f5Scvs2svn 
2101ae349f5Scvs2svn static void
2111ae349f5Scvs2svn CloseConnection(int signo)
2121ae349f5Scvs2svn {
2131ae349f5Scvs2svn   /* NOTE, these are manual, we've done a setsid() */
2141ae349f5Scvs2svn   pending_signal(SIGINT, SIG_IGN);
2151ae349f5Scvs2svn   LogPrintf(LogPHASE, "Caught signal %d, abort connection\n", signo);
2161ae349f5Scvs2svn   reconnectState = RECON_FALSE;
2171ae349f5Scvs2svn   reconnectCount = 0;
2181ae349f5Scvs2svn   DownConnection();
2191ae349f5Scvs2svn   dial_up = 0;
2201ae349f5Scvs2svn   pending_signal(SIGINT, CloseConnection);
2211ae349f5Scvs2svn }
2221ae349f5Scvs2svn 
2231ae349f5Scvs2svn static void
2241ae349f5Scvs2svn CloseSession(int signo)
2251ae349f5Scvs2svn {
2261ae349f5Scvs2svn   if (BGPid) {
2271ae349f5Scvs2svn     kill(BGPid, SIGINT);
2281ae349f5Scvs2svn     exit(EX_TERM);
2291ae349f5Scvs2svn   }
2301ae349f5Scvs2svn   LogPrintf(LogPHASE, "Signal %d, terminate.\n", signo);
2311ae349f5Scvs2svn   reconnect(RECON_FALSE);
2327308ec68SBrian Somers   LcpClose(&LcpInfo.fsm);
2331ae349f5Scvs2svn   Cleanup(EX_TERM);
2341ae349f5Scvs2svn }
2351ae349f5Scvs2svn 
2361ae349f5Scvs2svn static void
2371ae349f5Scvs2svn TerminalCont(int signo)
2381ae349f5Scvs2svn {
2391ae349f5Scvs2svn   pending_signal(SIGCONT, SIG_DFL);
2401ae349f5Scvs2svn   pending_signal(SIGTSTP, TerminalStop);
2411ae349f5Scvs2svn   TtyCommandMode(getpgrp() == tcgetpgrp(netfd));
2421ae349f5Scvs2svn }
2431ae349f5Scvs2svn 
2441ae349f5Scvs2svn static void
2451ae349f5Scvs2svn TerminalStop(int signo)
2461ae349f5Scvs2svn {
2471ae349f5Scvs2svn   pending_signal(SIGCONT, TerminalCont);
2481ae349f5Scvs2svn   TtyOldMode();
2491ae349f5Scvs2svn   pending_signal(SIGTSTP, SIG_DFL);
2501ae349f5Scvs2svn   kill(getpid(), signo);
2511ae349f5Scvs2svn }
2521ae349f5Scvs2svn 
2531ae349f5Scvs2svn static void
2541ae349f5Scvs2svn SetUpServer(int signo)
2551ae349f5Scvs2svn {
2561ae349f5Scvs2svn   int res;
2571ae349f5Scvs2svn 
2581ae349f5Scvs2svn   VarHaveLocalAuthKey = 0;
2591ae349f5Scvs2svn   LocalAuthInit();
2601ae349f5Scvs2svn   if ((res = ServerTcpOpen(SERVER_PORT + tunno)) != 0)
2611ae349f5Scvs2svn     LogPrintf(LogERROR, "SIGUSR1: Failed %d to open port %d\n",
2621ae349f5Scvs2svn 	      res, SERVER_PORT + tunno);
2631ae349f5Scvs2svn }
2641ae349f5Scvs2svn 
2651ae349f5Scvs2svn static void
2661ae349f5Scvs2svn BringDownServer(int signo)
2671ae349f5Scvs2svn {
2681ae349f5Scvs2svn   VarHaveLocalAuthKey = 0;
2691ae349f5Scvs2svn   LocalAuthInit();
2701ae349f5Scvs2svn   ServerClose();
2711ae349f5Scvs2svn }
2721ae349f5Scvs2svn 
2731ae349f5Scvs2svn static const char *
2741ae349f5Scvs2svn ex_desc(int ex)
2751ae349f5Scvs2svn {
2761ae349f5Scvs2svn   static char num[12];
2771ae349f5Scvs2svn   static const char *desc[] = {
2781ae349f5Scvs2svn     "normal", "start", "sock", "modem", "dial", "dead", "done",
2791ae349f5Scvs2svn     "reboot", "errdead", "hangup", "term", "nodial", "nologin"
2801ae349f5Scvs2svn   };
2811ae349f5Scvs2svn 
2821ae349f5Scvs2svn   if (ex >= 0 && ex < sizeof desc / sizeof *desc)
2831ae349f5Scvs2svn     return desc[ex];
2841ae349f5Scvs2svn   snprintf(num, sizeof num, "%d", ex);
2851ae349f5Scvs2svn   return num;
2861ae349f5Scvs2svn }
2871ae349f5Scvs2svn 
2881ae349f5Scvs2svn static void
2891ae349f5Scvs2svn Usage(void)
2901ae349f5Scvs2svn {
2911ae349f5Scvs2svn   fprintf(stderr,
2921ae349f5Scvs2svn 	  "Usage: ppp [-auto | -background | -direct | -dedicated | -ddial ]"
2931ae349f5Scvs2svn #ifndef NOALIAS
2941ae349f5Scvs2svn           " [ -alias ]"
2951ae349f5Scvs2svn #endif
2961ae349f5Scvs2svn           " [system]\n");
2971ae349f5Scvs2svn   exit(EX_START);
2981ae349f5Scvs2svn }
2991ae349f5Scvs2svn 
3001ae349f5Scvs2svn static char *
3011ae349f5Scvs2svn ProcessArgs(int argc, char **argv)
3021ae349f5Scvs2svn {
3031ae349f5Scvs2svn   int optc;
3041ae349f5Scvs2svn   char *cp;
3051ae349f5Scvs2svn 
3061ae349f5Scvs2svn   optc = 0;
3071ae349f5Scvs2svn   mode = MODE_INTER;
3081ae349f5Scvs2svn   while (argc > 0 && **argv == '-') {
3091ae349f5Scvs2svn     cp = *argv + 1;
3101ae349f5Scvs2svn     if (strcmp(cp, "auto") == 0) {
3111ae349f5Scvs2svn       mode |= MODE_AUTO;
3121ae349f5Scvs2svn       mode &= ~MODE_INTER;
3131ae349f5Scvs2svn     } else if (strcmp(cp, "background") == 0) {
3141ae349f5Scvs2svn       mode |= MODE_BACKGROUND;
3151ae349f5Scvs2svn       mode &= ~MODE_INTER;
3161ae349f5Scvs2svn     } else if (strcmp(cp, "direct") == 0) {
3171ae349f5Scvs2svn       mode |= MODE_DIRECT;
3181ae349f5Scvs2svn       mode &= ~MODE_INTER;
3191ae349f5Scvs2svn     } else if (strcmp(cp, "dedicated") == 0) {
3201ae349f5Scvs2svn       mode |= MODE_DEDICATED;
3211ae349f5Scvs2svn       mode &= ~MODE_INTER;
3221ae349f5Scvs2svn     } else if (strcmp(cp, "ddial") == 0) {
3231ae349f5Scvs2svn       mode |= MODE_DDIAL;
3241ae349f5Scvs2svn       mode &= ~MODE_INTER;
3251ae349f5Scvs2svn #ifndef NOALIAS
3261ae349f5Scvs2svn     } else if (strcmp(cp, "alias") == 0) {
3271ae349f5Scvs2svn       if (loadAliasHandlers(&VarAliasHandlers) == 0)
3281ae349f5Scvs2svn 	mode |= MODE_ALIAS;
3291ae349f5Scvs2svn       else
3301ae349f5Scvs2svn 	LogPrintf(LogWARN, "Cannot load alias library\n");
3311ae349f5Scvs2svn       optc--;			/* this option isn't exclusive */
3321ae349f5Scvs2svn #endif
3331ae349f5Scvs2svn     } else
3341ae349f5Scvs2svn       Usage();
3351ae349f5Scvs2svn     optc++;
3361ae349f5Scvs2svn     argv++;
3371ae349f5Scvs2svn     argc--;
3381ae349f5Scvs2svn   }
3391ae349f5Scvs2svn   if (argc > 1) {
3401ae349f5Scvs2svn     fprintf(stderr, "specify only one system label.\n");
3411ae349f5Scvs2svn     exit(EX_START);
3421ae349f5Scvs2svn   }
3431ae349f5Scvs2svn 
3441ae349f5Scvs2svn   if (optc > 1) {
3451ae349f5Scvs2svn     fprintf(stderr, "specify only one mode.\n");
3461ae349f5Scvs2svn     exit(EX_START);
3471ae349f5Scvs2svn   }
3481ae349f5Scvs2svn 
3491ae349f5Scvs2svn   return argc == 1 ? *argv : NULL;	/* Don't SetLabel yet ! */
3501ae349f5Scvs2svn }
3511ae349f5Scvs2svn 
3521ae349f5Scvs2svn int
3531ae349f5Scvs2svn main(int argc, char **argv)
3541ae349f5Scvs2svn {
3551ae349f5Scvs2svn   FILE *lockfile;
3561ae349f5Scvs2svn   char *name, *label;
3571ae349f5Scvs2svn   int nfds;
3587a6f8720SBrian Somers   struct bundle *bundle;
3591ae349f5Scvs2svn 
3601ae349f5Scvs2svn   nfds = getdtablesize();
3611ae349f5Scvs2svn   if (nfds >= FD_SETSIZE)
3621ae349f5Scvs2svn     /*
3631ae349f5Scvs2svn      * If we've got loads of file descriptors, make sure they're all
3641ae349f5Scvs2svn      * closed.  If they aren't, we may end up with a seg fault when our
3651ae349f5Scvs2svn      * `fd_set's get too big when select()ing !
3661ae349f5Scvs2svn      */
3671ae349f5Scvs2svn     while (--nfds > 2)
3681ae349f5Scvs2svn       close(nfds);
3691ae349f5Scvs2svn 
3701ae349f5Scvs2svn   VarTerm = 0;
3711ae349f5Scvs2svn   name = strrchr(argv[0], '/');
3721ae349f5Scvs2svn   LogOpen(name ? name + 1 : argv[0]);
3731ae349f5Scvs2svn 
3741ae349f5Scvs2svn   tcgetattr(STDIN_FILENO, &oldtio);	/* Save original tty mode */
3751ae349f5Scvs2svn 
3761ae349f5Scvs2svn   argc--;
3771ae349f5Scvs2svn   argv++;
3781ae349f5Scvs2svn   label = ProcessArgs(argc, argv);
3791ae349f5Scvs2svn   if (!(mode & MODE_DIRECT))
3801ae349f5Scvs2svn     VarTerm = stdout;
3811ae349f5Scvs2svn 
3821ae349f5Scvs2svn   ID0init();
3831ae349f5Scvs2svn   if (ID0realuid() != 0) {
3841ae349f5Scvs2svn     char conf[200], *ptr;
3851ae349f5Scvs2svn 
3861ae349f5Scvs2svn     snprintf(conf, sizeof conf, "%s/%s", _PATH_PPP, CONFFILE);
3871ae349f5Scvs2svn     do {
3881ae349f5Scvs2svn       if (!access(conf, W_OK)) {
3891ae349f5Scvs2svn         LogPrintf(LogALERT, "ppp: Access violation: Please protect %s\n", conf);
3901ae349f5Scvs2svn         return -1;
3911ae349f5Scvs2svn       }
3921ae349f5Scvs2svn       ptr = conf + strlen(conf)-2;
3931ae349f5Scvs2svn       while (ptr > conf && *ptr != '/')
3941ae349f5Scvs2svn         *ptr-- = '\0';
3951ae349f5Scvs2svn     } while (ptr >= conf);
3961ae349f5Scvs2svn   }
3971ae349f5Scvs2svn 
3981ae349f5Scvs2svn   if (!ValidSystem(label)) {
3991ae349f5Scvs2svn     fprintf(stderr, "You may not use ppp in this mode with this label\n");
4001ae349f5Scvs2svn     if (mode & MODE_DIRECT) {
4011ae349f5Scvs2svn       const char *l;
4021ae349f5Scvs2svn       l = label ? label : "default";
4031ae349f5Scvs2svn       LogPrintf(LogWARN, "Label %s rejected -direct connection\n", l);
4041ae349f5Scvs2svn     }
4051ae349f5Scvs2svn     LogClose();
4061ae349f5Scvs2svn     return 1;
4071ae349f5Scvs2svn   }
4081ae349f5Scvs2svn 
4091ae349f5Scvs2svn   if (!GetShortHost())
4101ae349f5Scvs2svn     return 1;
4111ae349f5Scvs2svn   IsInteractive(1);
4121ae349f5Scvs2svn   IpcpDefAddress();
4131ae349f5Scvs2svn 
4141ae349f5Scvs2svn   if (mode & MODE_INTER)
4151ae349f5Scvs2svn     VarLocalAuth = LOCAL_AUTH;
4161ae349f5Scvs2svn 
4177a6f8720SBrian Somers   if ((bundle = bundle_Create("/dev/tun")) == NULL) {
4187a6f8720SBrian Somers     LogPrintf(LogWARN, "bundle_Create: %s\n", strerror(errno));
4191ae349f5Scvs2svn     return EX_START;
4201ae349f5Scvs2svn   }
4217a6f8720SBrian Somers 
4227a6f8720SBrian Somers   CleanupBundle = bundle;
4237a6f8720SBrian Somers 
4247a6f8720SBrian Somers   if (SelectSystem(bundle, "default", CONFFILE) < 0 && VarTerm)
4257a6f8720SBrian Somers     fprintf(VarTerm, "Warning: No default entry is given in config file.\n");
4267a6f8720SBrian Somers 
4271ae349f5Scvs2svn   if ((mode & MODE_OUTGOING_DAEMON) && !(mode & MODE_DEDICATED))
4281ae349f5Scvs2svn     if (label == NULL) {
4291ae349f5Scvs2svn       if (VarTerm)
4301ae349f5Scvs2svn 	fprintf(VarTerm, "Destination system must be specified in"
4311ae349f5Scvs2svn 		" auto, background or ddial mode.\n");
4321ae349f5Scvs2svn       return EX_START;
4331ae349f5Scvs2svn     }
4341ae349f5Scvs2svn 
4351ae349f5Scvs2svn   pending_signal(SIGHUP, CloseSession);
4361ae349f5Scvs2svn   pending_signal(SIGTERM, CloseSession);
4371ae349f5Scvs2svn   pending_signal(SIGINT, CloseConnection);
4381ae349f5Scvs2svn   pending_signal(SIGQUIT, CloseSession);
4391ae349f5Scvs2svn #ifdef SIGPIPE
4401ae349f5Scvs2svn   signal(SIGPIPE, SIG_IGN);
4411ae349f5Scvs2svn #endif
4421ae349f5Scvs2svn #ifdef SIGALRM
4431ae349f5Scvs2svn   pending_signal(SIGALRM, SIG_IGN);
4441ae349f5Scvs2svn #endif
4451ae349f5Scvs2svn   if (mode & MODE_INTER) {
4461ae349f5Scvs2svn #ifdef SIGTSTP
4471ae349f5Scvs2svn     pending_signal(SIGTSTP, TerminalStop);
4481ae349f5Scvs2svn #endif
4491ae349f5Scvs2svn #ifdef SIGTTIN
4501ae349f5Scvs2svn     pending_signal(SIGTTIN, TerminalStop);
4511ae349f5Scvs2svn #endif
4521ae349f5Scvs2svn #ifdef SIGTTOU
4531ae349f5Scvs2svn     pending_signal(SIGTTOU, SIG_IGN);
4541ae349f5Scvs2svn #endif
4551ae349f5Scvs2svn   }
4561ae349f5Scvs2svn   if (!(mode & MODE_INTER)) {
4571ae349f5Scvs2svn #ifdef SIGUSR1
4581ae349f5Scvs2svn     pending_signal(SIGUSR1, SetUpServer);
4591ae349f5Scvs2svn #endif
4601ae349f5Scvs2svn #ifdef SIGUSR2
4611ae349f5Scvs2svn     pending_signal(SIGUSR2, BringDownServer);
4621ae349f5Scvs2svn #endif
4631ae349f5Scvs2svn   }
4641ae349f5Scvs2svn 
4651ae349f5Scvs2svn   if (label) {
4667a6f8720SBrian Somers     if (SelectSystem(bundle, label, CONFFILE) < 0) {
4671ae349f5Scvs2svn       LogPrintf(LogWARN, "Destination system %s not found in conf file.\n",
4681ae349f5Scvs2svn                 GetLabel());
4691ae349f5Scvs2svn       Cleanup(EX_START);
4701ae349f5Scvs2svn     }
4711ae349f5Scvs2svn     /*
4721ae349f5Scvs2svn      * We don't SetLabel() 'till now in case SelectSystem() has an
4731ae349f5Scvs2svn      * embeded load "otherlabel" command.
4741ae349f5Scvs2svn      */
4751ae349f5Scvs2svn     SetLabel(label);
4761ae349f5Scvs2svn     if (mode & MODE_OUTGOING_DAEMON &&
47729e275ceSBrian Somers 	IpcpInfo.DefHisAddress.ipaddr.s_addr == INADDR_ANY) {
4781ae349f5Scvs2svn       LogPrintf(LogWARN, "You must \"set ifaddr\" in label %s for"
4791ae349f5Scvs2svn 		" auto, background or ddial mode.\n", label);
4801ae349f5Scvs2svn       Cleanup(EX_START);
4811ae349f5Scvs2svn     }
4821ae349f5Scvs2svn   }
4831ae349f5Scvs2svn 
4841ae349f5Scvs2svn   if (mode & MODE_DAEMON) {
4851ae349f5Scvs2svn     if (mode & MODE_BACKGROUND) {
4861ae349f5Scvs2svn       if (pipe(BGFiledes)) {
4871ae349f5Scvs2svn 	LogPrintf(LogERROR, "pipe: %s\n", strerror(errno));
4881ae349f5Scvs2svn 	Cleanup(EX_SOCK);
4891ae349f5Scvs2svn       }
4901ae349f5Scvs2svn     }
4911ae349f5Scvs2svn 
4921ae349f5Scvs2svn     if (!(mode & MODE_DIRECT)) {
4931ae349f5Scvs2svn       pid_t bgpid;
4941ae349f5Scvs2svn 
4951ae349f5Scvs2svn       bgpid = fork();
4961ae349f5Scvs2svn       if (bgpid == -1) {
4971ae349f5Scvs2svn 	LogPrintf(LogERROR, "fork: %s\n", strerror(errno));
4981ae349f5Scvs2svn 	Cleanup(EX_SOCK);
4991ae349f5Scvs2svn       }
5001ae349f5Scvs2svn       if (bgpid) {
5011ae349f5Scvs2svn 	char c = EX_NORMAL;
5021ae349f5Scvs2svn 
5031ae349f5Scvs2svn 	if (mode & MODE_BACKGROUND) {
5041ae349f5Scvs2svn 	  /* Wait for our child to close its pipe before we exit. */
5051ae349f5Scvs2svn 	  BGPid = bgpid;
5061ae349f5Scvs2svn 	  close(BGFiledes[1]);
5071ae349f5Scvs2svn 	  if (read(BGFiledes[0], &c, 1) != 1) {
5081ae349f5Scvs2svn 	    fprintf(VarTerm, "Child exit, no status.\n");
5091ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: Child exit, no status.\n");
5101ae349f5Scvs2svn 	  } else if (c == EX_NORMAL) {
5111ae349f5Scvs2svn 	    fprintf(VarTerm, "PPP enabled.\n");
5121ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: PPP enabled.\n");
5131ae349f5Scvs2svn 	  } else {
5141ae349f5Scvs2svn 	    fprintf(VarTerm, "Child failed (%s).\n", ex_desc((int) c));
5151ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: Child failed (%s).\n",
5161ae349f5Scvs2svn 		      ex_desc((int) c));
5171ae349f5Scvs2svn 	  }
5181ae349f5Scvs2svn 	  close(BGFiledes[0]);
5191ae349f5Scvs2svn 	}
5201ae349f5Scvs2svn 	return c;
5211ae349f5Scvs2svn       } else if (mode & MODE_BACKGROUND)
5221ae349f5Scvs2svn 	close(BGFiledes[0]);
5231ae349f5Scvs2svn     }
5241ae349f5Scvs2svn 
5251ae349f5Scvs2svn     VarTerm = 0;		/* We know it's currently stdout */
5261ae349f5Scvs2svn     close(STDOUT_FILENO);
5271ae349f5Scvs2svn     close(STDERR_FILENO);
5281ae349f5Scvs2svn 
5291ae349f5Scvs2svn     if (mode & MODE_DIRECT)
5301ae349f5Scvs2svn       /* STDIN_FILENO gets used by OpenModem in DIRECT mode */
5311ae349f5Scvs2svn       TtyInit(1);
5321ae349f5Scvs2svn     else if (mode & MODE_DAEMON) {
5331ae349f5Scvs2svn       setsid();
5341ae349f5Scvs2svn       close(STDIN_FILENO);
5351ae349f5Scvs2svn     }
5361ae349f5Scvs2svn   } else {
5371ae349f5Scvs2svn     close(STDIN_FILENO);
5381ae349f5Scvs2svn     if ((netfd = open(_PATH_TTY, O_RDONLY)) < 0) {
5391ae349f5Scvs2svn       fprintf(stderr, "Cannot open %s for intput !\n", _PATH_TTY);
5401ae349f5Scvs2svn       return 2;
5411ae349f5Scvs2svn     }
5421ae349f5Scvs2svn     close(STDERR_FILENO);
5431ae349f5Scvs2svn     TtyInit(0);
5441ae349f5Scvs2svn     TtyCommandMode(1);
5451ae349f5Scvs2svn   }
5461ae349f5Scvs2svn 
5471ae349f5Scvs2svn   snprintf(pid_filename, sizeof pid_filename, "%stun%d.pid",
5481ae349f5Scvs2svn            _PATH_VARRUN, tunno);
5491ae349f5Scvs2svn   lockfile = ID0fopen(pid_filename, "w");
5501ae349f5Scvs2svn   if (lockfile != NULL) {
5511ae349f5Scvs2svn     fprintf(lockfile, "%d\n", (int) getpid());
5521ae349f5Scvs2svn     fclose(lockfile);
5531ae349f5Scvs2svn   }
5541ae349f5Scvs2svn #ifndef RELEASE_CRUNCH
5551ae349f5Scvs2svn   else
5561ae349f5Scvs2svn     LogPrintf(LogALERT, "Warning: Can't create %s: %s\n",
5571ae349f5Scvs2svn               pid_filename, strerror(errno));
5581ae349f5Scvs2svn #endif
5591ae349f5Scvs2svn 
5601ae349f5Scvs2svn   LogPrintf(LogPHASE, "PPP Started.\n");
5611ae349f5Scvs2svn 
5621ae349f5Scvs2svn 
5631ae349f5Scvs2svn   do
5647a6f8720SBrian Somers     DoLoop(bundle);
5651ae349f5Scvs2svn   while (mode & MODE_DEDICATED);
5661ae349f5Scvs2svn 
5671ae349f5Scvs2svn   Cleanup(EX_DONE);
5681ae349f5Scvs2svn   return 0;
5691ae349f5Scvs2svn }
5701ae349f5Scvs2svn 
5711ae349f5Scvs2svn /*
5721ae349f5Scvs2svn  *  Turn into packet mode, where we speak PPP.
5731ae349f5Scvs2svn  */
5741ae349f5Scvs2svn void
5757a6f8720SBrian Somers PacketMode(struct bundle *bundle, int delay)
5761ae349f5Scvs2svn {
57763b73463SBrian Somers   if (RawModem(pppVars.physical) < 0) {
5781ae349f5Scvs2svn     LogPrintf(LogWARN, "PacketMode: Not connected.\n");
5791ae349f5Scvs2svn     return;
5801ae349f5Scvs2svn   }
5811ae349f5Scvs2svn   AsyncInit();
5821ae349f5Scvs2svn   VjInit(15);
5837a6f8720SBrian Somers   LcpInit(bundle, pppVars.physical);
5847a6f8720SBrian Somers   IpcpInit(bundle, physical2link(pppVars.physical));
5857a6f8720SBrian Somers   CcpInit(bundle, physical2link(pppVars.physical));
5861ae349f5Scvs2svn   LcpUp();
5871ae349f5Scvs2svn 
5881ae349f5Scvs2svn   LcpOpen(delay);
5891ae349f5Scvs2svn   if (mode & MODE_INTER)
5901ae349f5Scvs2svn     TtyCommandMode(1);
5911ae349f5Scvs2svn   if (VarTerm) {
5921ae349f5Scvs2svn     fprintf(VarTerm, "Packet mode.\n");
5931ae349f5Scvs2svn     aft_cmd = 1;
5941ae349f5Scvs2svn   }
5951ae349f5Scvs2svn }
5961ae349f5Scvs2svn 
5971ae349f5Scvs2svn static void
5981ae349f5Scvs2svn ShowHelp(void)
5991ae349f5Scvs2svn {
6001ae349f5Scvs2svn   fprintf(stderr, "The following commands are available:\r\n");
6011ae349f5Scvs2svn   fprintf(stderr, " ~p\tEnter Packet mode\r\n");
6021ae349f5Scvs2svn   fprintf(stderr, " ~-\tDecrease log level\r\n");
6031ae349f5Scvs2svn   fprintf(stderr, " ~+\tIncrease log level\r\n");
6041ae349f5Scvs2svn   fprintf(stderr, " ~t\tShow timers (only in \"log debug\" mode)\r\n");
6051ae349f5Scvs2svn   fprintf(stderr, " ~m\tShow memory map (only in \"log debug\" mode)\r\n");
6061ae349f5Scvs2svn   fprintf(stderr, " ~.\tTerminate program\r\n");
6071ae349f5Scvs2svn   fprintf(stderr, " ~?\tThis help\r\n");
6081ae349f5Scvs2svn }
6091ae349f5Scvs2svn 
6101ae349f5Scvs2svn static void
6117a6f8720SBrian Somers ReadTty(struct bundle *bundle)
6121ae349f5Scvs2svn {
6131ae349f5Scvs2svn   int n;
6141ae349f5Scvs2svn   char ch;
6151ae349f5Scvs2svn   static int ttystate;
6161ae349f5Scvs2svn   char linebuff[LINE_LEN];
6171ae349f5Scvs2svn 
6181ae349f5Scvs2svn   LogPrintf(LogDEBUG, "termode = %d, netfd = %d, mode = %d\n",
6191ae349f5Scvs2svn 	    TermMode, netfd, mode);
6201ae349f5Scvs2svn   if (!TermMode) {
6211ae349f5Scvs2svn     n = read(netfd, linebuff, sizeof linebuff - 1);
6221ae349f5Scvs2svn     if (n > 0) {
6231ae349f5Scvs2svn       aft_cmd = 1;
6241ae349f5Scvs2svn       if (linebuff[n-1] == '\n')
6251ae349f5Scvs2svn         linebuff[--n] = '\0';
6261ae349f5Scvs2svn       else
6271ae349f5Scvs2svn         linebuff[n] = '\0';
6281ae349f5Scvs2svn       if (n)
6297a6f8720SBrian Somers         DecodeCommand(bundle, linebuff, n, IsInteractive(0) ? NULL : "Client");
6301ae349f5Scvs2svn       Prompt();
6311ae349f5Scvs2svn     } else if (n <= 0) {
6321ae349f5Scvs2svn       LogPrintf(LogPHASE, "Client connection closed.\n");
6331ae349f5Scvs2svn       DropClient(0);
6341ae349f5Scvs2svn     }
6351ae349f5Scvs2svn     return;
6361ae349f5Scvs2svn   }
6371ae349f5Scvs2svn 
6381ae349f5Scvs2svn   /*
6391ae349f5Scvs2svn    * We are in terminal mode, decode special sequences
6401ae349f5Scvs2svn    */
6411ae349f5Scvs2svn   n = read(netfd, &ch, 1);
6421ae349f5Scvs2svn   LogPrintf(LogDEBUG, "Got %d bytes (reading from the terminal)\n", n);
6431ae349f5Scvs2svn 
6441ae349f5Scvs2svn   if (n > 0) {
6451ae349f5Scvs2svn     switch (ttystate) {
6461ae349f5Scvs2svn     case 0:
6471ae349f5Scvs2svn       if (ch == '~')
6481ae349f5Scvs2svn 	ttystate++;
6491ae349f5Scvs2svn       else
65063b73463SBrian Somers 	/* XXX missing return value check */
65163b73463SBrian Somers 	Physical_Write(pppVars.physical, &ch, n);
6521ae349f5Scvs2svn       break;
6531ae349f5Scvs2svn     case 1:
6541ae349f5Scvs2svn       switch (ch) {
6551ae349f5Scvs2svn       case '?':
6561ae349f5Scvs2svn 	ShowHelp();
6571ae349f5Scvs2svn 	break;
6581ae349f5Scvs2svn       case 'p':
6591ae349f5Scvs2svn 
6601ae349f5Scvs2svn 	/*
6611ae349f5Scvs2svn 	 * XXX: Should check carrier.
6621ae349f5Scvs2svn 	 */
6637308ec68SBrian Somers 	if (LcpInfo.fsm.state <= ST_CLOSED)
6647a6f8720SBrian Somers 	  PacketMode(bundle, 0);
6651ae349f5Scvs2svn 	break;
6661ae349f5Scvs2svn       case '.':
6671ae349f5Scvs2svn 	TermMode = 1;
6681ae349f5Scvs2svn 	aft_cmd = 1;
6691ae349f5Scvs2svn 	TtyCommandMode(1);
6701ae349f5Scvs2svn 	break;
6711ae349f5Scvs2svn       case 't':
6721ae349f5Scvs2svn 	if (LogIsKept(LogDEBUG)) {
6731ae349f5Scvs2svn 	  ShowTimers();
6741ae349f5Scvs2svn 	  break;
6751ae349f5Scvs2svn 	}
6761ae349f5Scvs2svn       case 'm':
6771ae349f5Scvs2svn 	if (LogIsKept(LogDEBUG)) {
6781ae349f5Scvs2svn 	  ShowMemMap(NULL);
6791ae349f5Scvs2svn 	  break;
6801ae349f5Scvs2svn 	}
6811ae349f5Scvs2svn       default:
68263b73463SBrian Somers 	if (Physical_Write(pppVars.physical, &ch, n) < 0)
6831ae349f5Scvs2svn 	  LogPrintf(LogERROR, "error writing to modem.\n");
6841ae349f5Scvs2svn 	break;
6851ae349f5Scvs2svn       }
6861ae349f5Scvs2svn       ttystate = 0;
6871ae349f5Scvs2svn       break;
6881ae349f5Scvs2svn     }
6891ae349f5Scvs2svn   }
6901ae349f5Scvs2svn }
6911ae349f5Scvs2svn 
6921ae349f5Scvs2svn 
6931ae349f5Scvs2svn /*
6941ae349f5Scvs2svn  *  Here, we'll try to detect HDLC frame
6951ae349f5Scvs2svn  */
6961ae349f5Scvs2svn 
6971ae349f5Scvs2svn static const char *FrameHeaders[] = {
6981ae349f5Scvs2svn   "\176\377\003\300\041",
6991ae349f5Scvs2svn   "\176\377\175\043\300\041",
7001ae349f5Scvs2svn   "\176\177\175\043\100\041",
7011ae349f5Scvs2svn   "\176\175\337\175\043\300\041",
7021ae349f5Scvs2svn   "\176\175\137\175\043\100\041",
7031ae349f5Scvs2svn   NULL,
7041ae349f5Scvs2svn };
7051ae349f5Scvs2svn 
7061ae349f5Scvs2svn static const u_char *
70763b73463SBrian Somers HdlcDetect(struct physical *physical, u_char * cp, int n)
7081ae349f5Scvs2svn {
7091ae349f5Scvs2svn   const char *ptr, *fp, **hp;
7101ae349f5Scvs2svn 
7111ae349f5Scvs2svn   cp[n] = '\0';			/* be sure to null terminated */
7121ae349f5Scvs2svn   ptr = NULL;
7131ae349f5Scvs2svn   for (hp = FrameHeaders; *hp; hp++) {
7141ae349f5Scvs2svn     fp = *hp;
71563b73463SBrian Somers     if (Physical_IsSync(physical))
7161ae349f5Scvs2svn       fp++;
7171ae349f5Scvs2svn     ptr = strstr((char *) cp, fp);
7181ae349f5Scvs2svn     if (ptr)
7191ae349f5Scvs2svn       break;
7201ae349f5Scvs2svn   }
7211ae349f5Scvs2svn   return ((const u_char *) ptr);
7221ae349f5Scvs2svn }
7231ae349f5Scvs2svn 
7241ae349f5Scvs2svn static struct pppTimer RedialTimer;
7251ae349f5Scvs2svn 
7261ae349f5Scvs2svn static void
7271ae349f5Scvs2svn RedialTimeout(void *v)
7281ae349f5Scvs2svn {
7291ae349f5Scvs2svn   StopTimer(&RedialTimer);
7301ae349f5Scvs2svn   LogPrintf(LogPHASE, "Redialing timer expired.\n");
7311ae349f5Scvs2svn }
7321ae349f5Scvs2svn 
7331ae349f5Scvs2svn static void
7341ae349f5Scvs2svn StartRedialTimer(int Timeout)
7351ae349f5Scvs2svn {
7361ae349f5Scvs2svn   StopTimer(&RedialTimer);
7371ae349f5Scvs2svn 
7381ae349f5Scvs2svn   if (Timeout) {
7391ae349f5Scvs2svn     RedialTimer.state = TIMER_STOPPED;
7401ae349f5Scvs2svn 
7411ae349f5Scvs2svn     if (Timeout > 0)
7421ae349f5Scvs2svn       RedialTimer.load = Timeout * SECTICKS;
7431ae349f5Scvs2svn     else
7441ae349f5Scvs2svn       RedialTimer.load = (random() % REDIAL_PERIOD) * SECTICKS;
7451ae349f5Scvs2svn 
7461ae349f5Scvs2svn     LogPrintf(LogPHASE, "Enter pause (%d) for redialing.\n",
7471ae349f5Scvs2svn 	      RedialTimer.load / SECTICKS);
7481ae349f5Scvs2svn 
7491ae349f5Scvs2svn     RedialTimer.func = RedialTimeout;
7501ae349f5Scvs2svn     StartTimer(&RedialTimer);
7511ae349f5Scvs2svn   }
7521ae349f5Scvs2svn }
7531ae349f5Scvs2svn 
7541ae349f5Scvs2svn #define IN_SIZE sizeof(struct sockaddr_in)
7551ae349f5Scvs2svn #define UN_SIZE sizeof(struct sockaddr_in)
7561ae349f5Scvs2svn #define ADDRSZ (IN_SIZE > UN_SIZE ? IN_SIZE : UN_SIZE)
7571ae349f5Scvs2svn 
7581ae349f5Scvs2svn static void
7597a6f8720SBrian Somers DoLoop(struct bundle *bundle)
7601ae349f5Scvs2svn {
7611ae349f5Scvs2svn   fd_set rfds, wfds, efds;
7621ae349f5Scvs2svn   int pri, i, n, wfd, nfds;
7631ae349f5Scvs2svn   char hisaddr[ADDRSZ];
7641ae349f5Scvs2svn   struct sockaddr *sa = (struct sockaddr *)hisaddr;
7651ae349f5Scvs2svn   struct sockaddr_in *sin = (struct sockaddr_in *)hisaddr;
7661ae349f5Scvs2svn   struct timeval timeout, *tp;
7671ae349f5Scvs2svn   int ssize = ADDRSZ;
7681ae349f5Scvs2svn   const u_char *cp;
7691ae349f5Scvs2svn   int tries;
7701ae349f5Scvs2svn   int qlen;
7711ae349f5Scvs2svn   int res;
7721ae349f5Scvs2svn   struct tun_data tun;
7731ae349f5Scvs2svn #define rbuff tun.data
7741ae349f5Scvs2svn 
7751ae349f5Scvs2svn   if (mode & MODE_DIRECT) {
7761ae349f5Scvs2svn     LogPrintf(LogDEBUG, "Opening modem\n");
7777a6f8720SBrian Somers     if (OpenModem(bundle, pppVars.physical) < 0)
7781ae349f5Scvs2svn       return;
7791ae349f5Scvs2svn     LogPrintf(LogPHASE, "Packet mode enabled\n");
7807a6f8720SBrian Somers     PacketMode(bundle, VarOpenMode);
7811ae349f5Scvs2svn   } else if (mode & MODE_DEDICATED) {
7828c07a7b2SBrian Somers     if (!link_IsActive(physical2link(pppVars.physical)))
7837a6f8720SBrian Somers       while (OpenModem(bundle, pppVars.physical) < 0)
7841ae349f5Scvs2svn 	nointr_sleep(VarReconnectTimer);
7851ae349f5Scvs2svn   }
7861ae349f5Scvs2svn   fflush(VarTerm);
7871ae349f5Scvs2svn 
7881ae349f5Scvs2svn   timeout.tv_sec = 0;
7891ae349f5Scvs2svn   timeout.tv_usec = 0;
7901ae349f5Scvs2svn   reconnectState = RECON_UNKNOWN;
7911ae349f5Scvs2svn 
7921ae349f5Scvs2svn   if (mode & MODE_BACKGROUND)
7931ae349f5Scvs2svn     dial_up = 1;		/* Bring the line up */
7941ae349f5Scvs2svn   else
7951ae349f5Scvs2svn     dial_up = 0;		/* XXXX */
7961ae349f5Scvs2svn   tries = 0;
7971ae349f5Scvs2svn   for (;;) {
7981ae349f5Scvs2svn     nfds = 0;
7991ae349f5Scvs2svn     FD_ZERO(&rfds);
8001ae349f5Scvs2svn     FD_ZERO(&wfds);
8011ae349f5Scvs2svn     FD_ZERO(&efds);
8021ae349f5Scvs2svn 
8031ae349f5Scvs2svn     /*
8041ae349f5Scvs2svn      * If the link is down and we're in DDIAL mode, bring it back up.
8051ae349f5Scvs2svn      */
8067308ec68SBrian Somers     if (mode & MODE_DDIAL && LcpInfo.fsm.state <= ST_CLOSED)
8071ae349f5Scvs2svn       dial_up = 1;
8081ae349f5Scvs2svn 
8091ae349f5Scvs2svn     /*
8101ae349f5Scvs2svn      * If we lost carrier and want to re-establish the connection due to the
8111ae349f5Scvs2svn      * "set reconnect" value, we'd better bring the line back up.
8121ae349f5Scvs2svn      */
8137308ec68SBrian Somers     if (LcpInfo.fsm.state <= ST_CLOSED) {
8141ae349f5Scvs2svn       if (!dial_up && reconnectState == RECON_TRUE) {
8151ae349f5Scvs2svn 	if (++reconnectCount <= VarReconnectTries) {
8161ae349f5Scvs2svn 	  LogPrintf(LogPHASE, "Connection lost, re-establish (%d/%d)\n",
8171ae349f5Scvs2svn 		    reconnectCount, VarReconnectTries);
8181ae349f5Scvs2svn 	  StartRedialTimer(VarReconnectTimer);
8191ae349f5Scvs2svn 	  dial_up = 1;
8201ae349f5Scvs2svn 	} else {
8211ae349f5Scvs2svn 	  if (VarReconnectTries)
8221ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Connection lost, maximum (%d) times\n",
8231ae349f5Scvs2svn 		      VarReconnectTries);
8241ae349f5Scvs2svn 	  reconnectCount = 0;
8251ae349f5Scvs2svn 	  if (mode & MODE_BACKGROUND)
8261ae349f5Scvs2svn 	    Cleanup(EX_DEAD);
8271ae349f5Scvs2svn 	}
8281ae349f5Scvs2svn 	reconnectState = RECON_ENVOKED;
8291ae349f5Scvs2svn       } else if (mode & MODE_DEDICATED)
8307a6f8720SBrian Somers         PacketMode(bundle, VarOpenMode);
8311ae349f5Scvs2svn     }
8321ae349f5Scvs2svn 
8331ae349f5Scvs2svn     /*
8341ae349f5Scvs2svn      * If Ip packet for output is enqueued and require dial up, Just do it!
8351ae349f5Scvs2svn      */
8361ae349f5Scvs2svn     if (dial_up && RedialTimer.state != TIMER_RUNNING) {
83763b73463SBrian Somers       LogPrintf(LogDEBUG, "going to dial: modem = %d\n",
83863b73463SBrian Somers 		Physical_GetFD(pppVars.physical));
8397a6f8720SBrian Somers       if (OpenModem(bundle, pppVars.physical) < 0) {
8401ae349f5Scvs2svn 	tries++;
8411ae349f5Scvs2svn 	if (!(mode & MODE_DDIAL) && VarDialTries)
8421ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Failed to open modem (attempt %u of %d)\n",
8431ae349f5Scvs2svn 		    tries, VarDialTries);
8441ae349f5Scvs2svn 	else
8451ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Failed to open modem (attempt %u)\n", tries);
8461ae349f5Scvs2svn 
8471ae349f5Scvs2svn 	if (!(mode & MODE_DDIAL) && VarDialTries && tries >= VarDialTries) {
8481ae349f5Scvs2svn 	  if (mode & MODE_BACKGROUND)
8491ae349f5Scvs2svn 	    Cleanup(EX_DIAL);	/* Can't get the modem */
8501ae349f5Scvs2svn 	  dial_up = 0;
8511ae349f5Scvs2svn 	  reconnectState = RECON_UNKNOWN;
8521ae349f5Scvs2svn 	  reconnectCount = 0;
8531ae349f5Scvs2svn 	  tries = 0;
8541ae349f5Scvs2svn 	} else
8551ae349f5Scvs2svn 	  StartRedialTimer(VarRedialTimeout);
8561ae349f5Scvs2svn       } else {
8571ae349f5Scvs2svn 	tries++;		/* Tries are per number, not per list of
8581ae349f5Scvs2svn 				 * numbers. */
8591ae349f5Scvs2svn 	if (!(mode & MODE_DDIAL) && VarDialTries)
8601ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Dial attempt %u of %d\n", tries, VarDialTries);
8611ae349f5Scvs2svn 	else
8621ae349f5Scvs2svn 	  LogPrintf(LogCHAT, "Dial attempt %u\n", tries);
8631ae349f5Scvs2svn 
8647a6f8720SBrian Somers 	if ((res = DialModem(bundle, pppVars.physical)) == EX_DONE) {
8657a6f8720SBrian Somers 	  PacketMode(bundle, VarOpenMode);
8661ae349f5Scvs2svn 	  dial_up = 0;
8671ae349f5Scvs2svn 	  reconnectState = RECON_UNKNOWN;
8681ae349f5Scvs2svn 	  tries = 0;
8691ae349f5Scvs2svn 	} else {
8701ae349f5Scvs2svn 	  if (mode & MODE_BACKGROUND) {
8711ae349f5Scvs2svn 	    if (VarNextPhone == NULL || res == EX_SIG)
8721ae349f5Scvs2svn 	      Cleanup(EX_DIAL);	/* Tried all numbers - no luck */
8731ae349f5Scvs2svn 	    else
8741ae349f5Scvs2svn 	      /* Try all numbers in background mode */
8751ae349f5Scvs2svn 	      StartRedialTimer(VarRedialNextTimeout);
8761ae349f5Scvs2svn 	  } else if (!(mode & MODE_DDIAL) &&
8771ae349f5Scvs2svn 		     ((VarDialTries && tries >= VarDialTries) ||
8781ae349f5Scvs2svn 		      res == EX_SIG)) {
8791ae349f5Scvs2svn 	    /* I give up !  Can't get through :( */
8801ae349f5Scvs2svn 	    StartRedialTimer(VarRedialTimeout);
8811ae349f5Scvs2svn 	    dial_up = 0;
8821ae349f5Scvs2svn 	    reconnectState = RECON_UNKNOWN;
8831ae349f5Scvs2svn 	    reconnectCount = 0;
8841ae349f5Scvs2svn 	    tries = 0;
8851ae349f5Scvs2svn 	  } else if (VarNextPhone == NULL)
8861ae349f5Scvs2svn 	    /* Dial failed. Keep quite during redial wait period. */
8871ae349f5Scvs2svn 	    StartRedialTimer(VarRedialTimeout);
8881ae349f5Scvs2svn 	  else
8891ae349f5Scvs2svn 	    StartRedialTimer(VarRedialNextTimeout);
8901ae349f5Scvs2svn 	}
8911ae349f5Scvs2svn       }
8921ae349f5Scvs2svn     }
8931ae349f5Scvs2svn 
8948c07a7b2SBrian Somers     qlen = link_QueueLen(physical2link(pppVars.physical));
8951ae349f5Scvs2svn     if (qlen == 0) {
8968c07a7b2SBrian Somers       IpStartOutput(physical2link(pppVars.physical));
8978c07a7b2SBrian Somers       qlen = link_QueueLen(physical2link(pppVars.physical));
8981ae349f5Scvs2svn     }
8998c07a7b2SBrian Somers 
9008c07a7b2SBrian Somers     if (link_IsActive(physical2link(pppVars.physical))) {
90163b73463SBrian Somers       /* XXX-ML this should probably be abstracted */
90263b73463SBrian Somers       if (Physical_GetFD(pppVars.physical) + 1 > nfds)
90363b73463SBrian Somers 	nfds = Physical_GetFD(pppVars.physical) + 1;
90463b73463SBrian Somers       Physical_FD_SET(pppVars.physical, &rfds);
90563b73463SBrian Somers       Physical_FD_SET(pppVars.physical, &efds);
9061ae349f5Scvs2svn       if (qlen > 0) {
90763b73463SBrian Somers 	Physical_FD_SET(pppVars.physical, &wfds);
9081ae349f5Scvs2svn       }
9091ae349f5Scvs2svn     }
9101ae349f5Scvs2svn     if (server >= 0) {
9111ae349f5Scvs2svn       if (server + 1 > nfds)
9121ae349f5Scvs2svn 	nfds = server + 1;
9131ae349f5Scvs2svn       FD_SET(server, &rfds);
9141ae349f5Scvs2svn     }
9151ae349f5Scvs2svn 
9161ae349f5Scvs2svn #ifndef SIGALRM
9171ae349f5Scvs2svn     /*
9181ae349f5Scvs2svn      * *** IMPORTANT ***
9191ae349f5Scvs2svn      * CPU is serviced every TICKUNIT micro seconds. This value must be chosen
9201ae349f5Scvs2svn      * with great care. If this values is too big, it results in loss of
9211ae349f5Scvs2svn      * characters from the modem and poor response.  If this value is too
9221ae349f5Scvs2svn      * small, ppp eats too much CPU time.
9231ae349f5Scvs2svn      */
9241ae349f5Scvs2svn     usleep(TICKUNIT);
9251ae349f5Scvs2svn     TimerService();
9261ae349f5Scvs2svn #else
9271ae349f5Scvs2svn     handle_signals();
9281ae349f5Scvs2svn #endif
9291ae349f5Scvs2svn 
9301ae349f5Scvs2svn     /* If there are aren't many packets queued, look for some more. */
9317a6f8720SBrian Somers     if (qlen < 20 && bundle->tun_fd >= 0) {
9327a6f8720SBrian Somers       if (bundle->tun_fd + 1 > nfds)
9337a6f8720SBrian Somers 	nfds = bundle->tun_fd + 1;
9347a6f8720SBrian Somers       FD_SET(bundle->tun_fd, &rfds);
9351ae349f5Scvs2svn     }
9361ae349f5Scvs2svn     if (netfd >= 0) {
9371ae349f5Scvs2svn       if (netfd + 1 > nfds)
9381ae349f5Scvs2svn 	nfds = netfd + 1;
9391ae349f5Scvs2svn       FD_SET(netfd, &rfds);
9401ae349f5Scvs2svn       FD_SET(netfd, &efds);
9411ae349f5Scvs2svn     }
9421ae349f5Scvs2svn #ifndef SIGALRM
9431ae349f5Scvs2svn 
9441ae349f5Scvs2svn     /*
9451ae349f5Scvs2svn      * Normally, select() will not block because modem is writable. In AUTO
9461ae349f5Scvs2svn      * mode, select will block until we find packet from tun
9471ae349f5Scvs2svn      */
9481ae349f5Scvs2svn     tp = (RedialTimer.state == TIMER_RUNNING) ? &timeout : NULL;
9491ae349f5Scvs2svn     i = select(nfds, &rfds, &wfds, &efds, tp);
9501ae349f5Scvs2svn #else
9511ae349f5Scvs2svn 
9521ae349f5Scvs2svn     /*
9531ae349f5Scvs2svn      * When SIGALRM timer is running, a select function will be return -1 and
9541ae349f5Scvs2svn      * EINTR after a Time Service signal hundler is done.  If the redial
9551ae349f5Scvs2svn      * timer is not running and we are trying to dial, poll with a 0 value
9561ae349f5Scvs2svn      * timer.
9571ae349f5Scvs2svn      */
9581ae349f5Scvs2svn     tp = (dial_up && RedialTimer.state != TIMER_RUNNING) ? &timeout : NULL;
9591ae349f5Scvs2svn     i = select(nfds, &rfds, &wfds, &efds, tp);
9601ae349f5Scvs2svn #endif
9611ae349f5Scvs2svn 
9621ae349f5Scvs2svn     if (i == 0) {
9631ae349f5Scvs2svn       continue;
9641ae349f5Scvs2svn     }
9651ae349f5Scvs2svn     if (i < 0) {
9661ae349f5Scvs2svn       if (errno == EINTR) {
9671ae349f5Scvs2svn 	handle_signals();
9681ae349f5Scvs2svn 	continue;
9691ae349f5Scvs2svn       }
9701ae349f5Scvs2svn       LogPrintf(LogERROR, "DoLoop: select(): %s\n", strerror(errno));
9711ae349f5Scvs2svn       break;
9721ae349f5Scvs2svn     }
97363b73463SBrian Somers     if ((netfd >= 0 && FD_ISSET(netfd, &efds)) ||
97463b73463SBrian Somers 	(Physical_FD_ISSET(pppVars.physical, &efds))) {
9751ae349f5Scvs2svn       LogPrintf(LogALERT, "Exception detected.\n");
9761ae349f5Scvs2svn       break;
9771ae349f5Scvs2svn     }
9781ae349f5Scvs2svn     if (server >= 0 && FD_ISSET(server, &rfds)) {
9791ae349f5Scvs2svn       wfd = accept(server, sa, &ssize);
9801ae349f5Scvs2svn       if (wfd < 0) {
9811ae349f5Scvs2svn 	LogPrintf(LogERROR, "DoLoop: accept(): %s\n", strerror(errno));
9821ae349f5Scvs2svn 	continue;
9831ae349f5Scvs2svn       }
9841ae349f5Scvs2svn       switch (sa->sa_family) {
9851ae349f5Scvs2svn         case AF_LOCAL:
9861ae349f5Scvs2svn           LogPrintf(LogPHASE, "Connected to local client.\n");
9871ae349f5Scvs2svn           break;
9881ae349f5Scvs2svn         case AF_INET:
9891ae349f5Scvs2svn           if (ntohs(sin->sin_port) < 1024) {
9901ae349f5Scvs2svn             LogPrintf(LogALERT, "Rejected client connection from %s:%u"
9911ae349f5Scvs2svn                       "(invalid port number) !\n",
9921ae349f5Scvs2svn                       inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
9931ae349f5Scvs2svn 	    close(wfd);
9941ae349f5Scvs2svn 	    continue;
9951ae349f5Scvs2svn           }
9961ae349f5Scvs2svn           LogPrintf(LogPHASE, "Connected to client from %s:%u\n",
9971ae349f5Scvs2svn                     inet_ntoa(sin->sin_addr), sin->sin_port);
9981ae349f5Scvs2svn           break;
9991ae349f5Scvs2svn         default:
10001ae349f5Scvs2svn 	  write(wfd, "Unrecognised access !\n", 22);
10011ae349f5Scvs2svn 	  close(wfd);
10021ae349f5Scvs2svn 	  continue;
10031ae349f5Scvs2svn       }
10041ae349f5Scvs2svn       if (netfd >= 0) {
10051ae349f5Scvs2svn 	write(wfd, "Connection already in use.\n", 27);
10061ae349f5Scvs2svn 	close(wfd);
10071ae349f5Scvs2svn 	continue;
10081ae349f5Scvs2svn       }
10091ae349f5Scvs2svn       netfd = wfd;
10101ae349f5Scvs2svn       VarTerm = fdopen(netfd, "a+");
10111ae349f5Scvs2svn       LocalAuthInit();
10121ae349f5Scvs2svn       IsInteractive(1);
10131ae349f5Scvs2svn       Prompt();
10141ae349f5Scvs2svn     }
10151ae349f5Scvs2svn     if (netfd >= 0 && FD_ISSET(netfd, &rfds))
10161ae349f5Scvs2svn       /* something to read from tty */
10177a6f8720SBrian Somers       ReadTty(bundle);
101863b73463SBrian Somers     if (Physical_FD_ISSET(pppVars.physical, &wfds)) {
10191ae349f5Scvs2svn       /* ready to write into modem */
10208c07a7b2SBrian Somers       link_StartOutput(physical2link(pppVars.physical));
10218c07a7b2SBrian Somers       if (!link_IsActive(physical2link(pppVars.physical)))
10221ae349f5Scvs2svn         dial_up = 1;
10231ae349f5Scvs2svn     }
102463b73463SBrian Somers     if (Physical_FD_ISSET(pppVars.physical, &rfds)) {
10251ae349f5Scvs2svn       /* something to read from modem */
10267308ec68SBrian Somers       if (LcpInfo.fsm.state <= ST_CLOSED)
10271ae349f5Scvs2svn 	nointr_usleep(10000);
102863b73463SBrian Somers       n = Physical_Read(pppVars.physical, rbuff, sizeof rbuff);
10291ae349f5Scvs2svn       if ((mode & MODE_DIRECT) && n <= 0) {
10301ae349f5Scvs2svn 	DownConnection();
10311ae349f5Scvs2svn       } else
10321ae349f5Scvs2svn 	LogDumpBuff(LogASYNC, "ReadFromModem", rbuff, n);
10331ae349f5Scvs2svn 
10347308ec68SBrian Somers       if (LcpInfo.fsm.state <= ST_CLOSED) {
10351ae349f5Scvs2svn 	/*
10361ae349f5Scvs2svn 	 * In dedicated mode, we just discard input until LCP is started.
10371ae349f5Scvs2svn 	 */
10381ae349f5Scvs2svn 	if (!(mode & MODE_DEDICATED)) {
103963b73463SBrian Somers 	  cp = HdlcDetect(pppVars.physical, rbuff, n);
10401ae349f5Scvs2svn 	  if (cp) {
10411ae349f5Scvs2svn 	    /*
10421ae349f5Scvs2svn 	     * LCP packet is detected. Turn ourselves into packet mode.
10431ae349f5Scvs2svn 	     */
10441ae349f5Scvs2svn 	    if (cp != rbuff) {
104563b73463SBrian Somers 	      /* XXX missing return value checks */
104663b73463SBrian Somers 	      Physical_Write(pppVars.physical, rbuff, cp - rbuff);
104763b73463SBrian Somers 	      Physical_Write(pppVars.physical, "\r\n", 2);
10481ae349f5Scvs2svn 	    }
10497a6f8720SBrian Somers 	    PacketMode(bundle, 0);
10501ae349f5Scvs2svn 	  } else
10511ae349f5Scvs2svn 	    write(fileno(VarTerm), rbuff, n);
10521ae349f5Scvs2svn 	}
10531ae349f5Scvs2svn       } else {
10541ae349f5Scvs2svn 	if (n > 0)
10557a6f8720SBrian Somers 	  AsyncInput(bundle, rbuff, n, pppVars.physical);
10561ae349f5Scvs2svn       }
10571ae349f5Scvs2svn     }
10587a6f8720SBrian Somers     if (bundle->tun_fd >= 0 && FD_ISSET(bundle->tun_fd, &rfds)) {
10597a6f8720SBrian Somers       /* something to read from tun */
10607a6f8720SBrian Somers       n = read(bundle->tun_fd, &tun, sizeof tun);
10611ae349f5Scvs2svn       if (n < 0) {
10621ae349f5Scvs2svn 	LogPrintf(LogERROR, "read from tun: %s\n", strerror(errno));
10631ae349f5Scvs2svn 	continue;
10641ae349f5Scvs2svn       }
10651ae349f5Scvs2svn       n -= sizeof tun - sizeof tun.data;
10661ae349f5Scvs2svn       if (n <= 0) {
10671ae349f5Scvs2svn 	LogPrintf(LogERROR, "read from tun: Only %d bytes read\n", n);
10681ae349f5Scvs2svn 	continue;
10691ae349f5Scvs2svn       }
10701ae349f5Scvs2svn       if (!tun_check_header(tun, AF_INET))
10711ae349f5Scvs2svn           continue;
10721ae349f5Scvs2svn       if (((struct ip *) rbuff)->ip_dst.s_addr == IpcpInfo.want_ipaddr.s_addr) {
10731ae349f5Scvs2svn 	/* we've been asked to send something addressed *to* us :( */
10741ae349f5Scvs2svn 	if (VarLoopback) {
10751ae349f5Scvs2svn 	  pri = PacketCheck(rbuff, n, FL_IN);
10761ae349f5Scvs2svn 	  if (pri >= 0) {
10771ae349f5Scvs2svn 	    struct mbuf *bp;
10781ae349f5Scvs2svn 
10791ae349f5Scvs2svn #ifndef NOALIAS
10801ae349f5Scvs2svn 	    if (mode & MODE_ALIAS) {
10811ae349f5Scvs2svn 	      VarPacketAliasIn(rbuff, sizeof rbuff);
10821ae349f5Scvs2svn 	      n = ntohs(((struct ip *) rbuff)->ip_len);
10831ae349f5Scvs2svn 	    }
10841ae349f5Scvs2svn #endif
10851ae349f5Scvs2svn 	    bp = mballoc(n, MB_IPIN);
10861ae349f5Scvs2svn 	    memcpy(MBUF_CTOP(bp), rbuff, n);
10877a6f8720SBrian Somers 	    IpInput(bundle, bp);
10881ae349f5Scvs2svn 	    LogPrintf(LogDEBUG, "Looped back packet addressed to myself\n");
10891ae349f5Scvs2svn 	  }
10901ae349f5Scvs2svn 	  continue;
10911ae349f5Scvs2svn 	} else
10921ae349f5Scvs2svn 	  LogPrintf(LogDEBUG, "Oops - forwarding packet addressed to myself\n");
10931ae349f5Scvs2svn       }
10941ae349f5Scvs2svn 
10951ae349f5Scvs2svn       /*
10961ae349f5Scvs2svn        * Process on-demand dialup. Output packets are queued within tunnel
10971ae349f5Scvs2svn        * device until IPCP is opened.
10981ae349f5Scvs2svn        */
10997308ec68SBrian Somers       if (LcpInfo.fsm.state <= ST_CLOSED && (mode & MODE_AUTO) &&
11001ae349f5Scvs2svn 	  (pri = PacketCheck(rbuff, n, FL_DIAL)) >= 0)
11011ae349f5Scvs2svn         dial_up = 1;
11021ae349f5Scvs2svn 
11031ae349f5Scvs2svn       pri = PacketCheck(rbuff, n, FL_OUT);
11041ae349f5Scvs2svn       if (pri >= 0) {
11051ae349f5Scvs2svn #ifndef NOALIAS
11061ae349f5Scvs2svn 	if (mode & MODE_ALIAS) {
11071ae349f5Scvs2svn 	  VarPacketAliasOut(rbuff, sizeof rbuff);
11081ae349f5Scvs2svn 	  n = ntohs(((struct ip *) rbuff)->ip_len);
11091ae349f5Scvs2svn 	}
11101ae349f5Scvs2svn #endif
11111ae349f5Scvs2svn 	IpEnqueue(pri, rbuff, n);
11121ae349f5Scvs2svn       }
11131ae349f5Scvs2svn     }
11141ae349f5Scvs2svn   }
11151ae349f5Scvs2svn   LogPrintf(LogDEBUG, "Job (DoLoop) done.\n");
11161ae349f5Scvs2svn }
1117