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