xref: /freebsd/usr.sbin/ppp/main.c (revision 565e35e50e2cdac423588a3d18742544bde128b0)
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  *
20565e35e5SBrian Somers  * $Id: main.c,v 1.121.2.47 1998/04/07 23:46:02 brian Exp $
211ae349f5Scvs2svn  *
221ae349f5Scvs2svn  *	TODO:
231ae349f5Scvs2svn  */
242764b86aSBrian Somers 
251ae349f5Scvs2svn #include <sys/param.h>
261ae349f5Scvs2svn #include <netinet/in.h>
271ae349f5Scvs2svn #include <netinet/in_systm.h>
281ae349f5Scvs2svn #include <netinet/ip.h>
29565e35e5SBrian Somers #include <sys/un.h>
301ae349f5Scvs2svn 
311ae349f5Scvs2svn #include <errno.h>
321ae349f5Scvs2svn #include <fcntl.h>
331ae349f5Scvs2svn #include <paths.h>
341ae349f5Scvs2svn #include <signal.h>
351ae349f5Scvs2svn #include <stdio.h>
361ae349f5Scvs2svn #include <string.h>
371ae349f5Scvs2svn #include <termios.h>
381ae349f5Scvs2svn #include <unistd.h>
391ae349f5Scvs2svn 
401ae349f5Scvs2svn #include "mbuf.h"
411ae349f5Scvs2svn #include "log.h"
421ae349f5Scvs2svn #include "defs.h"
431ae349f5Scvs2svn #include "id.h"
441ae349f5Scvs2svn #include "timer.h"
451ae349f5Scvs2svn #include "fsm.h"
46879ed6faSBrian Somers #include "lqr.h"
471ae349f5Scvs2svn #include "hdlc.h"
481ae349f5Scvs2svn #include "lcp.h"
491ae349f5Scvs2svn #include "ccp.h"
5029e275ceSBrian Somers #include "iplist.h"
5129e275ceSBrian Somers #include "throughput.h"
52eaa4df37SBrian Somers #include "slcompress.h"
531ae349f5Scvs2svn #include "ipcp.h"
545ca5389aSBrian Somers #include "filter.h"
552f786681SBrian Somers #include "descriptor.h"
563b0f8d2eSBrian Somers #include "link.h"
573b0f8d2eSBrian Somers #include "mp.h"
585828db6dSBrian Somers #include "bundle.h"
591ae349f5Scvs2svn #include "loadalias.h"
601ae349f5Scvs2svn #include "vars.h"
611ae349f5Scvs2svn #include "auth.h"
621ae349f5Scvs2svn #include "systems.h"
631ae349f5Scvs2svn #include "ip.h"
641ae349f5Scvs2svn #include "sig.h"
651ae349f5Scvs2svn #include "main.h"
661ae349f5Scvs2svn #include "pathnames.h"
671ae349f5Scvs2svn #include "tun.h"
6877ff88adSBrian Somers #include "server.h"
6985b542cfSBrian Somers #include "prompt.h"
70b6dec9f0SBrian Somers #include "chat.h"
71e2ebb036SBrian Somers #include "chap.h"
723006ec67SBrian Somers #include "datalink.h"
731ae349f5Scvs2svn 
741ae349f5Scvs2svn #ifndef O_NONBLOCK
751ae349f5Scvs2svn #ifdef O_NDELAY
761ae349f5Scvs2svn #define	O_NONBLOCK O_NDELAY
771ae349f5Scvs2svn #endif
781ae349f5Scvs2svn #endif
791ae349f5Scvs2svn 
801ae349f5Scvs2svn static char pid_filename[MAXPATHLEN];
811ae349f5Scvs2svn 
82b6217683SBrian Somers static void DoLoop(struct bundle *, struct prompt *);
831ae349f5Scvs2svn static void TerminalStop(int);
841ae349f5Scvs2svn static const char *ex_desc(int);
851ae349f5Scvs2svn 
8683d1af55SBrian Somers static struct bundle *SignalBundle;
87b6217683SBrian Somers static struct prompt *SignalPrompt;
887a6f8720SBrian Somers 
891ae349f5Scvs2svn void
901ae349f5Scvs2svn Cleanup(int excode)
911ae349f5Scvs2svn {
92a0cbd833SBrian Somers   SignalBundle->CleaningUp = 1;
933006ec67SBrian Somers   if (bundle_Phase(SignalBundle) != PHASE_DEAD)
943b0f8d2eSBrian Somers     bundle_Close(SignalBundle, NULL, 1);
951afedc4bSBrian Somers }
96455aabc3SBrian Somers 
971afedc4bSBrian Somers void
981afedc4bSBrian Somers AbortProgram(int excode)
991afedc4bSBrian Somers {
100b6217683SBrian Somers   ServerClose(SignalBundle);
1011ae349f5Scvs2svn   ID0unlink(pid_filename);
1021ae349f5Scvs2svn   LogPrintf(LogPHASE, "PPP Terminated (%s).\n", ex_desc(excode));
103d345321bSBrian Somers   bundle_Close(SignalBundle, NULL, 1);
10468a0f0ccSBrian Somers   bundle_Destroy(SignalBundle);
1053006ec67SBrian Somers   LogClose();
1061ae349f5Scvs2svn   exit(excode);
1071ae349f5Scvs2svn }
1081ae349f5Scvs2svn 
1091ae349f5Scvs2svn static void
1101ae349f5Scvs2svn CloseConnection(int signo)
1111ae349f5Scvs2svn {
1121ae349f5Scvs2svn   /* NOTE, these are manual, we've done a setsid() */
1133b0f8d2eSBrian Somers   struct datalink *dl;
1143b0f8d2eSBrian Somers 
1151ae349f5Scvs2svn   pending_signal(SIGINT, SIG_IGN);
1163b0f8d2eSBrian Somers   LogPrintf(LogPHASE, "Caught signal %d, abort connection(s)\n", signo);
1173b0f8d2eSBrian Somers   for (dl = SignalBundle->links; dl; dl = dl->next)
1183b0f8d2eSBrian Somers     datalink_Down(dl, 1);
1191ae349f5Scvs2svn   pending_signal(SIGINT, CloseConnection);
1201ae349f5Scvs2svn }
1211ae349f5Scvs2svn 
1221ae349f5Scvs2svn static void
1231ae349f5Scvs2svn CloseSession(int signo)
1241ae349f5Scvs2svn {
1251ae349f5Scvs2svn   LogPrintf(LogPHASE, "Signal %d, terminate.\n", signo);
1261ae349f5Scvs2svn   Cleanup(EX_TERM);
1271ae349f5Scvs2svn }
1281ae349f5Scvs2svn 
12975b8d283SBrian Somers static pid_t BGPid = 0;
13075b8d283SBrian Somers 
13175b8d283SBrian Somers static void
13275b8d283SBrian Somers KillChild(int signo)
13375b8d283SBrian Somers {
13475b8d283SBrian Somers   LogPrintf(LogPHASE, "Parent: Signal %d\n", signo);
13575b8d283SBrian Somers   kill(BGPid, SIGINT);
13675b8d283SBrian Somers }
13775b8d283SBrian Somers 
1381ae349f5Scvs2svn static void
1391ae349f5Scvs2svn TerminalCont(int signo)
1401ae349f5Scvs2svn {
141f91ad6b0SBrian Somers   signal(SIGCONT, SIG_DFL);
142f91ad6b0SBrian Somers   prompt_Continue(SignalPrompt);
1431ae349f5Scvs2svn }
1441ae349f5Scvs2svn 
1451ae349f5Scvs2svn static void
1461ae349f5Scvs2svn TerminalStop(int signo)
1471ae349f5Scvs2svn {
148f91ad6b0SBrian Somers   prompt_Suspend(SignalPrompt);
149f91ad6b0SBrian Somers   signal(SIGCONT, TerminalCont);
150f91ad6b0SBrian Somers   raise(SIGSTOP);
1511ae349f5Scvs2svn }
1521ae349f5Scvs2svn 
153b6217683SBrian Somers #if 0 /* What's our passwd :-O */
1541ae349f5Scvs2svn static void
1551ae349f5Scvs2svn SetUpServer(int signo)
1561ae349f5Scvs2svn {
1571ae349f5Scvs2svn   int res;
1581ae349f5Scvs2svn 
1591ae349f5Scvs2svn   VarHaveLocalAuthKey = 0;
1601ae349f5Scvs2svn   LocalAuthInit();
161455aabc3SBrian Somers   if ((res = ServerTcpOpen(SERVER_PORT + SignalBundle->unit)) != 0)
1621ae349f5Scvs2svn     LogPrintf(LogERROR, "SIGUSR1: Failed %d to open port %d\n",
163455aabc3SBrian Somers 	      res, SERVER_PORT + SignalBundle->unit);
1641ae349f5Scvs2svn }
165b6217683SBrian Somers #endif
1661ae349f5Scvs2svn 
1671ae349f5Scvs2svn static void
1681ae349f5Scvs2svn BringDownServer(int signo)
1691ae349f5Scvs2svn {
170b6217683SBrian Somers   /* Drops all child prompts too ! */
171b6217683SBrian Somers   ServerClose(SignalBundle);
1721ae349f5Scvs2svn }
1731ae349f5Scvs2svn 
1741ae349f5Scvs2svn static const char *
1751ae349f5Scvs2svn ex_desc(int ex)
1761ae349f5Scvs2svn {
1771ae349f5Scvs2svn   static char num[12];
1781ae349f5Scvs2svn   static const char *desc[] = {
1791ae349f5Scvs2svn     "normal", "start", "sock", "modem", "dial", "dead", "done",
1801ae349f5Scvs2svn     "reboot", "errdead", "hangup", "term", "nodial", "nologin"
1811ae349f5Scvs2svn   };
1821ae349f5Scvs2svn 
1831ae349f5Scvs2svn   if (ex >= 0 && ex < sizeof desc / sizeof *desc)
1841ae349f5Scvs2svn     return desc[ex];
1851ae349f5Scvs2svn   snprintf(num, sizeof num, "%d", ex);
1861ae349f5Scvs2svn   return num;
1871ae349f5Scvs2svn }
1881ae349f5Scvs2svn 
1891ae349f5Scvs2svn static void
1901ae349f5Scvs2svn Usage(void)
1911ae349f5Scvs2svn {
1921ae349f5Scvs2svn   fprintf(stderr,
1931ae349f5Scvs2svn 	  "Usage: ppp [-auto | -background | -direct | -dedicated | -ddial ]"
1941ae349f5Scvs2svn #ifndef NOALIAS
1951ae349f5Scvs2svn           " [ -alias ]"
1961ae349f5Scvs2svn #endif
1971ae349f5Scvs2svn           " [system]\n");
1981ae349f5Scvs2svn   exit(EX_START);
1991ae349f5Scvs2svn }
2001ae349f5Scvs2svn 
2011ae349f5Scvs2svn static char *
202565e35e5SBrian Somers ProcessArgs(int argc, char **argv, int *mode)
2031ae349f5Scvs2svn {
204565e35e5SBrian Somers   int optc, labelrequired;
2051ae349f5Scvs2svn   char *cp;
2061ae349f5Scvs2svn 
207565e35e5SBrian Somers   optc = labelrequired = 0;
208565e35e5SBrian Somers   *mode = PHYS_MANUAL;
2091ae349f5Scvs2svn   while (argc > 0 && **argv == '-') {
2101ae349f5Scvs2svn     cp = *argv + 1;
2111ae349f5Scvs2svn     if (strcmp(cp, "auto") == 0) {
212565e35e5SBrian Somers       *mode = PHYS_DEMAND;
213565e35e5SBrian Somers       labelrequired = 1;
2141ae349f5Scvs2svn     } else if (strcmp(cp, "background") == 0) {
215565e35e5SBrian Somers       *mode = PHYS_1OFF;
216565e35e5SBrian Somers       labelrequired = 1;
217565e35e5SBrian Somers     } else if (strcmp(cp, "direct") == 0)
218565e35e5SBrian Somers       *mode = PHYS_STDIN;
219565e35e5SBrian Somers     else if (strcmp(cp, "dedicated") == 0)
220565e35e5SBrian Somers       *mode = PHYS_DEDICATED;
221565e35e5SBrian Somers     else if (strcmp(cp, "ddial") == 0) {
222565e35e5SBrian Somers       *mode = PHYS_PERM;
223565e35e5SBrian Somers       labelrequired = 1;
2241ae349f5Scvs2svn     } else if (strcmp(cp, "alias") == 0) {
225565e35e5SBrian Somers #ifndef NOALIAS
22685602e52SBrian Somers       if (loadAliasHandlers() != 0)
227565e35e5SBrian Somers #endif
2281ae349f5Scvs2svn 	LogPrintf(LogWARN, "Cannot load alias library\n");
2291ae349f5Scvs2svn       optc--;			/* this option isn't exclusive */
2301ae349f5Scvs2svn     } else
2311ae349f5Scvs2svn       Usage();
2321ae349f5Scvs2svn     optc++;
2331ae349f5Scvs2svn     argv++;
2341ae349f5Scvs2svn     argc--;
2351ae349f5Scvs2svn   }
2361ae349f5Scvs2svn   if (argc > 1) {
23785602e52SBrian Somers     fprintf(stderr, "You may specify only one system label.\n");
2381ae349f5Scvs2svn     exit(EX_START);
2391ae349f5Scvs2svn   }
2401ae349f5Scvs2svn 
2411ae349f5Scvs2svn   if (optc > 1) {
24285602e52SBrian Somers     fprintf(stderr, "You may specify only one mode.\n");
2431ae349f5Scvs2svn     exit(EX_START);
2441ae349f5Scvs2svn   }
2451ae349f5Scvs2svn 
246565e35e5SBrian Somers   if (labelrequired && argc != 1) {
247565e35e5SBrian Somers     fprintf(stderr, "Destination system must be specified in"
248565e35e5SBrian Somers             " auto, background or ddial mode.\n");
249565e35e5SBrian Somers     exit(EX_START);
250565e35e5SBrian Somers   }
251565e35e5SBrian Somers 
2521ae349f5Scvs2svn   return argc == 1 ? *argv : NULL;	/* Don't SetLabel yet ! */
2531ae349f5Scvs2svn }
2541ae349f5Scvs2svn 
2551ae349f5Scvs2svn int
2561ae349f5Scvs2svn main(int argc, char **argv)
2571ae349f5Scvs2svn {
2581ae349f5Scvs2svn   FILE *lockfile;
2591ae349f5Scvs2svn   char *name, *label;
260565e35e5SBrian Somers   int nfds, mode;
2617a6f8720SBrian Somers   struct bundle *bundle;
262b6217683SBrian Somers   struct prompt *prompt;
2631ae349f5Scvs2svn 
2641ae349f5Scvs2svn   nfds = getdtablesize();
2651ae349f5Scvs2svn   if (nfds >= FD_SETSIZE)
2661ae349f5Scvs2svn     /*
2671ae349f5Scvs2svn      * If we've got loads of file descriptors, make sure they're all
2681ae349f5Scvs2svn      * closed.  If they aren't, we may end up with a seg fault when our
2691ae349f5Scvs2svn      * `fd_set's get too big when select()ing !
2701ae349f5Scvs2svn      */
2711ae349f5Scvs2svn     while (--nfds > 2)
2721ae349f5Scvs2svn       close(nfds);
2731ae349f5Scvs2svn 
2741ae349f5Scvs2svn   name = strrchr(argv[0], '/');
2751ae349f5Scvs2svn   LogOpen(name ? name + 1 : argv[0]);
2761ae349f5Scvs2svn 
2771ae349f5Scvs2svn   argc--;
2781ae349f5Scvs2svn   argv++;
279565e35e5SBrian Somers   label = ProcessArgs(argc, argv, &mode);
28085b542cfSBrian Somers 
28185b542cfSBrian Somers #ifdef __FreeBSD__
28285b542cfSBrian Somers   /*
28385b542cfSBrian Somers    * A FreeBSD hack to dodge a bug in the tty driver that drops output
28485b542cfSBrian Somers    * occasionally.... I must find the real reason some time.  To display
28585b542cfSBrian Somers    * the dodgy behaviour, comment out this bit, make yourself a large
28685b542cfSBrian Somers    * routing table and then run ppp in interactive mode.  The `show route'
28785b542cfSBrian Somers    * command will drop chunks of data !!!
28885b542cfSBrian Somers    */
289565e35e5SBrian Somers   if (mode == PHYS_MANUAL) {
29085b542cfSBrian Somers     close(STDIN_FILENO);
29185b542cfSBrian Somers     if (open(_PATH_TTY, O_RDONLY) != STDIN_FILENO) {
29285b542cfSBrian Somers       fprintf(stderr, "Cannot open %s for input !\n", _PATH_TTY);
29385b542cfSBrian Somers       return 2;
29485b542cfSBrian Somers     }
29585b542cfSBrian Somers   }
29685b542cfSBrian Somers #endif
29785b542cfSBrian Somers 
298b6217683SBrian Somers   /* Allow output for the moment (except in direct mode) */
299565e35e5SBrian Somers   if (mode == PHYS_STDIN)
300b6217683SBrian Somers     prompt = NULL;
30185602e52SBrian Somers   else {
30285602e52SBrian Somers     const char *m;
30385602e52SBrian Somers 
304b6217683SBrian Somers     SignalPrompt = prompt = prompt_Create(NULL, NULL, PROMPT_STD);
305565e35e5SBrian Somers     if (mode == PHYS_PERM)
30685602e52SBrian Somers       m = "direct dial";
307565e35e5SBrian Somers     else if (mode & PHYS_1OFF)
30885602e52SBrian Somers       m = "background";
309565e35e5SBrian Somers     else if (mode & PHYS_DEMAND)
31085602e52SBrian Somers       m = "auto";
311565e35e5SBrian Somers     else if (mode & PHYS_DEDICATED)
31285602e52SBrian Somers       m = "dedicated";
313565e35e5SBrian Somers     else if (mode & PHYS_MANUAL)
31485602e52SBrian Somers       m = "interactive";
31585602e52SBrian Somers     else
31685602e52SBrian Somers       m = NULL;
31785602e52SBrian Somers 
31885602e52SBrian Somers     if (m)
31985602e52SBrian Somers       prompt_Printf(prompt, "Working in %s mode\n", m);
32085602e52SBrian Somers   }
3211ae349f5Scvs2svn 
3221ae349f5Scvs2svn   ID0init();
3231ae349f5Scvs2svn   if (ID0realuid() != 0) {
3241ae349f5Scvs2svn     char conf[200], *ptr;
3251ae349f5Scvs2svn 
3261ae349f5Scvs2svn     snprintf(conf, sizeof conf, "%s/%s", _PATH_PPP, CONFFILE);
3271ae349f5Scvs2svn     do {
3281ae349f5Scvs2svn       if (!access(conf, W_OK)) {
3291ae349f5Scvs2svn         LogPrintf(LogALERT, "ppp: Access violation: Please protect %s\n", conf);
3301ae349f5Scvs2svn         return -1;
3311ae349f5Scvs2svn       }
3321ae349f5Scvs2svn       ptr = conf + strlen(conf)-2;
3331ae349f5Scvs2svn       while (ptr > conf && *ptr != '/')
3341ae349f5Scvs2svn         *ptr-- = '\0';
3351ae349f5Scvs2svn     } while (ptr >= conf);
3361ae349f5Scvs2svn   }
3371ae349f5Scvs2svn 
338565e35e5SBrian Somers   if (!ValidSystem(label, prompt, mode)) {
3391ae349f5Scvs2svn     fprintf(stderr, "You may not use ppp in this mode with this label\n");
340565e35e5SBrian Somers     if (mode == PHYS_STDIN) {
3411ae349f5Scvs2svn       const char *l;
3421ae349f5Scvs2svn       l = label ? label : "default";
3431ae349f5Scvs2svn       LogPrintf(LogWARN, "Label %s rejected -direct connection\n", l);
3441ae349f5Scvs2svn     }
3451ae349f5Scvs2svn     LogClose();
3461ae349f5Scvs2svn     return 1;
3471ae349f5Scvs2svn   }
3481ae349f5Scvs2svn 
349565e35e5SBrian Somers   if ((bundle = bundle_Create(TUN_PREFIX, prompt, mode)) == NULL) {
3507a6f8720SBrian Somers     LogPrintf(LogWARN, "bundle_Create: %s\n", strerror(errno));
3511ae349f5Scvs2svn     return EX_START;
3521ae349f5Scvs2svn   }
35383d1af55SBrian Somers   SignalBundle = bundle;
3547a6f8720SBrian Somers 
355b6217683SBrian Somers   if (SelectSystem(bundle, "default", CONFFILE, prompt) < 0)
356565e35e5SBrian Somers     prompt_Printf(prompt, "Warning: No default entry found in config file.\n");
3571ae349f5Scvs2svn 
3581ae349f5Scvs2svn   pending_signal(SIGHUP, CloseSession);
3591ae349f5Scvs2svn   pending_signal(SIGTERM, CloseSession);
3601ae349f5Scvs2svn   pending_signal(SIGINT, CloseConnection);
3611ae349f5Scvs2svn   pending_signal(SIGQUIT, CloseSession);
3623b0f8d2eSBrian Somers   pending_signal(SIGALRM, SIG_IGN);
3631ae349f5Scvs2svn   signal(SIGPIPE, SIG_IGN);
364565e35e5SBrian Somers 
365565e35e5SBrian Somers   if (mode == PHYS_MANUAL)
3661ae349f5Scvs2svn     pending_signal(SIGTSTP, TerminalStop);
367f91ad6b0SBrian Somers 
368b6217683SBrian Somers #if 0 /* What's our passwd :-O */
3691ae349f5Scvs2svn   pending_signal(SIGUSR1, SetUpServer);
3701ae349f5Scvs2svn #endif
3711ae349f5Scvs2svn   pending_signal(SIGUSR2, BringDownServer);
3721ae349f5Scvs2svn 
3731ae349f5Scvs2svn   if (label) {
374b6217683SBrian Somers     if (SelectSystem(bundle, label, CONFFILE, prompt) < 0) {
375b6217683SBrian Somers       prompt_Printf(prompt, "Destination system (%s) not found.\n", label);
3768390b576SBrian Somers       AbortProgram(EX_START);
3771ae349f5Scvs2svn     }
3781ae349f5Scvs2svn     /*
3791ae349f5Scvs2svn      * We don't SetLabel() 'till now in case SelectSystem() has an
3801ae349f5Scvs2svn      * embeded load "otherlabel" command.
3811ae349f5Scvs2svn      */
3821ae349f5Scvs2svn     SetLabel(label);
383565e35e5SBrian Somers     if (mode == PHYS_DEMAND &&
3845828db6dSBrian Somers 	bundle->ncp.ipcp.cfg.peer_range.ipaddr.s_addr == INADDR_ANY) {
385565e35e5SBrian Somers       prompt_Printf(prompt, "You must \"set ifaddr\" with a peer address "
386565e35e5SBrian Somers                     "in label %s for auto mode.\n", label);
3878390b576SBrian Somers       AbortProgram(EX_START);
3881ae349f5Scvs2svn     }
3891ae349f5Scvs2svn   }
3901ae349f5Scvs2svn 
391565e35e5SBrian Somers   if (mode != PHYS_MANUAL) {
392565e35e5SBrian Somers     if (mode != PHYS_STDIN) {
3935cf4388bSBrian Somers       int bgpipe[2];
3945cf4388bSBrian Somers       pid_t bgpid;
3955cf4388bSBrian Somers 
396565e35e5SBrian Somers       if (mode == PHYS_1OFF && pipe(bgpipe)) {
3971ae349f5Scvs2svn         LogPrintf(LogERROR, "pipe: %s\n", strerror(errno));
3988390b576SBrian Somers 	AbortProgram(EX_SOCK);
3991ae349f5Scvs2svn       }
4001ae349f5Scvs2svn 
4011ae349f5Scvs2svn       bgpid = fork();
4021ae349f5Scvs2svn       if (bgpid == -1) {
4031ae349f5Scvs2svn 	LogPrintf(LogERROR, "fork: %s\n", strerror(errno));
4048390b576SBrian Somers 	AbortProgram(EX_SOCK);
4051ae349f5Scvs2svn       }
4065cf4388bSBrian Somers 
4071ae349f5Scvs2svn       if (bgpid) {
4081ae349f5Scvs2svn 	char c = EX_NORMAL;
4091ae349f5Scvs2svn 
410565e35e5SBrian Somers 	if (mode == PHYS_1OFF) {
4115cf4388bSBrian Somers 	  close(bgpipe[1]);
41275b8d283SBrian Somers 	  BGPid = bgpid;
41375b8d283SBrian Somers           /* If we get a signal, kill the child */
41475b8d283SBrian Somers           signal(SIGHUP, KillChild);
41575b8d283SBrian Somers           signal(SIGTERM, KillChild);
41675b8d283SBrian Somers           signal(SIGINT, KillChild);
41775b8d283SBrian Somers           signal(SIGQUIT, KillChild);
41875b8d283SBrian Somers 
41975b8d283SBrian Somers 	  /* Wait for our child to close its pipe before we exit */
4205cf4388bSBrian Somers 	  if (read(bgpipe[0], &c, 1) != 1) {
421b6217683SBrian Somers 	    prompt_Printf(prompt, "Child exit, no status.\n");
4221ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: Child exit, no status.\n");
4231ae349f5Scvs2svn 	  } else if (c == EX_NORMAL) {
424b6217683SBrian Somers 	    prompt_Printf(prompt, "PPP enabled.\n");
4251ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: PPP enabled.\n");
4261ae349f5Scvs2svn 	  } else {
427b6217683SBrian Somers 	    prompt_Printf(prompt, "Child failed (%s).\n", ex_desc((int) c));
4281ae349f5Scvs2svn 	    LogPrintf(LogPHASE, "Parent: Child failed (%s).\n",
4291ae349f5Scvs2svn 		      ex_desc((int) c));
4301ae349f5Scvs2svn 	  }
4315cf4388bSBrian Somers 	  close(bgpipe[0]);
4321ae349f5Scvs2svn 	}
4331ae349f5Scvs2svn 	return c;
434565e35e5SBrian Somers       } else if (mode == PHYS_1OFF) {
4355cf4388bSBrian Somers 	close(bgpipe[0]);
4365cf4388bSBrian Somers         bundle->notify.fd = bgpipe[1];
4375cf4388bSBrian Somers       }
4381ae349f5Scvs2svn 
439565e35e5SBrian Somers       /* -auto, -dedicated, -ddial & -background */
440b6217683SBrian Somers       prompt_Destroy(prompt, 0);
441b6217683SBrian Somers       close(STDOUT_FILENO);
442b6217683SBrian Somers       close(STDERR_FILENO);
4431ae349f5Scvs2svn       close(STDIN_FILENO);
444b6217683SBrian Somers       setsid();
445565e35e5SBrian Somers     } else {
446565e35e5SBrian Somers       /* -direct: STDIN_FILENO gets used by modem_Open */
447565e35e5SBrian Somers       prompt_TtyInit(NULL);
448565e35e5SBrian Somers       close(STDOUT_FILENO);
449565e35e5SBrian Somers       close(STDERR_FILENO);
4501ae349f5Scvs2svn     }
4511ae349f5Scvs2svn   } else {
452565e35e5SBrian Somers     /* Interactive mode */
4531ae349f5Scvs2svn     close(STDERR_FILENO);
454565e35e5SBrian Somers     prompt_TtyInit(prompt);
455b6217683SBrian Somers     prompt_TtyCommandMode(prompt);
456b6217683SBrian Somers     prompt_Required(prompt);
4571ae349f5Scvs2svn   }
4581ae349f5Scvs2svn 
4591ae349f5Scvs2svn   snprintf(pid_filename, sizeof pid_filename, "%stun%d.pid",
460455aabc3SBrian Somers            _PATH_VARRUN, bundle->unit);
4611ae349f5Scvs2svn   lockfile = ID0fopen(pid_filename, "w");
4621ae349f5Scvs2svn   if (lockfile != NULL) {
4631ae349f5Scvs2svn     fprintf(lockfile, "%d\n", (int) getpid());
4641ae349f5Scvs2svn     fclose(lockfile);
4651ae349f5Scvs2svn   }
4661ae349f5Scvs2svn #ifndef RELEASE_CRUNCH
4671ae349f5Scvs2svn   else
4681ae349f5Scvs2svn     LogPrintf(LogALERT, "Warning: Can't create %s: %s\n",
4691ae349f5Scvs2svn               pid_filename, strerror(errno));
4701ae349f5Scvs2svn #endif
4711ae349f5Scvs2svn 
472565e35e5SBrian Somers   LogPrintf(LogPHASE, "PPP Started (%s mode).\n", mode2Nam(mode));
473b6217683SBrian Somers   DoLoop(bundle, prompt);
4741afedc4bSBrian Somers   AbortProgram(EX_NORMAL);
4751afedc4bSBrian Somers 
4761afedc4bSBrian Somers   return EX_NORMAL;
4771ae349f5Scvs2svn }
4781ae349f5Scvs2svn 
4791ae349f5Scvs2svn static void
480b6217683SBrian Somers DoLoop(struct bundle *bundle, struct prompt *prompt)
4811ae349f5Scvs2svn {
4821ae349f5Scvs2svn   fd_set rfds, wfds, efds;
4833006ec67SBrian Somers   int pri, i, n, nfds;
4841ae349f5Scvs2svn   int qlen;
4851ae349f5Scvs2svn   struct tun_data tun;
4861ae349f5Scvs2svn 
487565e35e5SBrian Somers   do {
4881ae349f5Scvs2svn     nfds = 0;
4891ae349f5Scvs2svn     FD_ZERO(&rfds);
4901ae349f5Scvs2svn     FD_ZERO(&wfds);
4911ae349f5Scvs2svn     FD_ZERO(&efds);
4921ae349f5Scvs2svn 
4933006ec67SBrian Somers     qlen = bundle_FillQueues(bundle);
494b6dec9f0SBrian Somers 
495b6dec9f0SBrian Somers     handle_signals();
496b6dec9f0SBrian Somers 
4972f786681SBrian Somers     descriptor_UpdateSet(&bundle->desc, &rfds, &wfds, &efds, &nfds);
49877ff88adSBrian Somers     descriptor_UpdateSet(&server.desc, &rfds, &wfds, &efds, &nfds);
4991ae349f5Scvs2svn 
5001ae349f5Scvs2svn     /* If there are aren't many packets queued, look for some more. */
5017a6f8720SBrian Somers     if (qlen < 20 && bundle->tun_fd >= 0) {
5027a6f8720SBrian Somers       if (bundle->tun_fd + 1 > nfds)
5037a6f8720SBrian Somers 	nfds = bundle->tun_fd + 1;
5047a6f8720SBrian Somers       FD_SET(bundle->tun_fd, &rfds);
5051ae349f5Scvs2svn     }
50677ff88adSBrian Somers 
5073b0f8d2eSBrian Somers     if (bundle_IsDead(bundle))
508f4768038SBrian Somers       /* Don't select - we'll be here forever */
509f4768038SBrian Somers       break;
510f4768038SBrian Somers 
5113006ec67SBrian Somers     i = select(nfds, &rfds, &wfds, &efds, NULL);
5121ae349f5Scvs2svn 
5133006ec67SBrian Somers     if (i == 0)
5141ae349f5Scvs2svn       continue;
51577ff88adSBrian Somers 
5161ae349f5Scvs2svn     if (i < 0) {
5171ae349f5Scvs2svn       if (errno == EINTR) {
5181ae349f5Scvs2svn 	handle_signals();
5191ae349f5Scvs2svn 	continue;
5201ae349f5Scvs2svn       }
5211ae349f5Scvs2svn       LogPrintf(LogERROR, "DoLoop: select(): %s\n", strerror(errno));
5221ae349f5Scvs2svn       break;
5231ae349f5Scvs2svn     }
52477ff88adSBrian Somers 
5253006ec67SBrian Somers     for (i = 0; i <= nfds; i++)
5263006ec67SBrian Somers       if (FD_ISSET(i, &efds)) {
5273006ec67SBrian Somers         LogPrintf(LogALERT, "Exception detected on descriptor %d\n", i);
5281ae349f5Scvs2svn         break;
5291ae349f5Scvs2svn       }
53077ff88adSBrian Somers 
53177ff88adSBrian Somers     if (descriptor_IsSet(&server.desc, &rfds))
532b77776a7SBrian Somers       descriptor_Read(&server.desc, bundle, &rfds);
53342d4d396SBrian Somers 
5342f786681SBrian Somers     if (descriptor_IsSet(&bundle->desc, &wfds))
5352f786681SBrian Somers       descriptor_Write(&bundle->desc, bundle, &wfds);
5362f786681SBrian Somers 
5372f786681SBrian Somers     if (descriptor_IsSet(&bundle->desc, &rfds))
5382f786681SBrian Somers       descriptor_Read(&bundle->desc, bundle, &rfds);
53942d4d396SBrian Somers 
5407a6f8720SBrian Somers     if (bundle->tun_fd >= 0 && FD_ISSET(bundle->tun_fd, &rfds)) {
5417a6f8720SBrian Somers       /* something to read from tun */
5427a6f8720SBrian Somers       n = read(bundle->tun_fd, &tun, sizeof tun);
5431ae349f5Scvs2svn       if (n < 0) {
5441ae349f5Scvs2svn 	LogPrintf(LogERROR, "read from tun: %s\n", strerror(errno));
5451ae349f5Scvs2svn 	continue;
5461ae349f5Scvs2svn       }
5471ae349f5Scvs2svn       n -= sizeof tun - sizeof tun.data;
5481ae349f5Scvs2svn       if (n <= 0) {
5491ae349f5Scvs2svn 	LogPrintf(LogERROR, "read from tun: Only %d bytes read\n", n);
5501ae349f5Scvs2svn 	continue;
5511ae349f5Scvs2svn       }
5521ae349f5Scvs2svn       if (!tun_check_header(tun, AF_INET))
5531ae349f5Scvs2svn           continue;
5545828db6dSBrian Somers       if (((struct ip *)tun.data)->ip_dst.s_addr ==
5555828db6dSBrian Somers           bundle->ncp.ipcp.my_ip.s_addr) {
5561ae349f5Scvs2svn 	/* we've been asked to send something addressed *to* us :( */
5570b3acc6eSBrian Somers 	if (Enabled(ConfLoopback)) {
5585ca5389aSBrian Somers 	  pri = PacketCheck(bundle, tun.data, n, &bundle->filter.in);
5591ae349f5Scvs2svn 	  if (pri >= 0) {
5601ae349f5Scvs2svn 	    struct mbuf *bp;
5611ae349f5Scvs2svn 
5621ae349f5Scvs2svn #ifndef NOALIAS
56385602e52SBrian Somers             if (AliasEnabled()) {
5642764b86aSBrian Somers 	      (*PacketAlias.In)(tun.data, sizeof tun.data);
565f4768038SBrian Somers 	      n = ntohs(((struct ip *)tun.data)->ip_len);
5661ae349f5Scvs2svn 	    }
5671ae349f5Scvs2svn #endif
5681ae349f5Scvs2svn 	    bp = mballoc(n, MB_IPIN);
569f4768038SBrian Somers 	    memcpy(MBUF_CTOP(bp), tun.data, n);
5707a6f8720SBrian Somers 	    IpInput(bundle, bp);
5711ae349f5Scvs2svn 	    LogPrintf(LogDEBUG, "Looped back packet addressed to myself\n");
5721ae349f5Scvs2svn 	  }
5731ae349f5Scvs2svn 	  continue;
5741ae349f5Scvs2svn 	} else
5751ae349f5Scvs2svn 	  LogPrintf(LogDEBUG, "Oops - forwarding packet addressed to myself\n");
5761ae349f5Scvs2svn       }
5771ae349f5Scvs2svn 
5781ae349f5Scvs2svn       /*
5791ae349f5Scvs2svn        * Process on-demand dialup. Output packets are queued within tunnel
5801ae349f5Scvs2svn        * device until IPCP is opened.
5811ae349f5Scvs2svn        */
5828390b576SBrian Somers       if (bundle_Phase(bundle) == PHASE_DEAD)
5838390b576SBrian Somers         /*
5848390b576SBrian Somers          * Note, we must be in AUTO mode :-/ otherwise our interface should
5858390b576SBrian Somers          * *not* be UP and we can't receive data
5868390b576SBrian Somers          */
587565e35e5SBrian Somers         if ((pri = PacketCheck(bundle, tun.data, n, &bundle->filter.dial)) >= 0)
588565e35e5SBrian Somers           bundle_Open(bundle, NULL, PHYS_DEMAND);
5898390b576SBrian Somers         else
5908390b576SBrian Somers           /*
5918390b576SBrian Somers            * Drop the packet.  If we were to queue it, we'd just end up with
5928390b576SBrian Somers            * a pile of timed-out data in our output queue by the time we get
5938390b576SBrian Somers            * around to actually dialing.  We'd also prematurely reach the
5948390b576SBrian Somers            * threshold at which we stop select()ing to read() the tun
5958390b576SBrian Somers            * device - breaking auto-dial.
5968390b576SBrian Somers            */
5978390b576SBrian Somers           continue;
5981ae349f5Scvs2svn 
5995ca5389aSBrian Somers       pri = PacketCheck(bundle, tun.data, n, &bundle->filter.out);
6001ae349f5Scvs2svn       if (pri >= 0) {
6011ae349f5Scvs2svn #ifndef NOALIAS
60285602e52SBrian Somers         if (AliasEnabled()) {
6032764b86aSBrian Somers 	  (*PacketAlias.Out)(tun.data, sizeof tun.data);
604f4768038SBrian Somers 	  n = ntohs(((struct ip *)tun.data)->ip_len);
6051ae349f5Scvs2svn 	}
6061ae349f5Scvs2svn #endif
607f4768038SBrian Somers 	IpEnqueue(pri, tun.data, n);
6081ae349f5Scvs2svn       }
6091ae349f5Scvs2svn     }
610565e35e5SBrian Somers   } while (bundle_CleanDatalinks(bundle), !bundle_IsDead(bundle));
611565e35e5SBrian Somers 
612565e35e5SBrian Somers   LogPrintf(LogDEBUG, "DoLoop done.\n");
6131ae349f5Scvs2svn }
614