Change UID to ORIG_UID because bash defines UID.
[ppp.git] / pppd / main.c
index 3273cdf9aa32412a282bafdafb54758b6dd1f9b9..7072d0a871fabcf9b615e3df1b179ff41ae2aa1d 100644 (file)
@@ -18,7 +18,7 @@
  */
 
 #ifndef lint
  */
 
 #ifndef lint
-static char rcsid[] = "$Id: main.c,v 1.44 1998/02/04 01:40:59 paulus Exp $";
+static char rcsid[] = "$Id: main.c,v 1.50 1998/09/13 23:38:49 paulus Exp $";
 #endif
 
 #include <stdio.h>
 #endif
 
 #include <stdio.h>
@@ -85,11 +85,14 @@ int baud_rate;                      /* Actual bits/second for serial device */
 int hungup;                    /* terminal has been hung up */
 int privileged;                        /* we're running as real uid root */
 int need_holdoff;              /* need holdoff period before restarting */
 int hungup;                    /* terminal has been hung up */
 int privileged;                        /* we're running as real uid root */
 int need_holdoff;              /* need holdoff period before restarting */
+int detached;                  /* have detached from terminal */
 
 int phase;                     /* where the link is at */
 int kill_link;
 int open_ccp_flag;
 
 int phase;                     /* where the link is at */
 int kill_link;
 int open_ccp_flag;
-int redirect_stderr;           /* Connector's stderr should go to file */
+
+char **script_env;             /* Env. variable values for scripts */
+int s_env_nalloc;              /* # words avail at script_env */
 
 u_char outpacket_buf[PPP_MRU+PPP_HDRLEN]; /* buffer for outgoing packet */
 u_char inpacket_buf[PPP_MRU+PPP_HDRLEN]; /* buffer for incoming packet */
 
 u_char outpacket_buf[PPP_MRU+PPP_HDRLEN]; /* buffer for outgoing packet */
 u_char inpacket_buf[PPP_MRU+PPP_HDRLEN]; /* buffer for incoming packet */
@@ -102,6 +105,7 @@ char *no_ppp_msg = "Sorry - this system lacks PPP kernel support\n";
 
 /* Prototypes for procedures local to this file. */
 
 
 /* Prototypes for procedures local to this file. */
 
+static void create_pidfile __P((void));
 static void cleanup __P((void));
 static void close_tty __P((void));
 static void get_input __P((void));
 static void cleanup __P((void));
 static void close_tty __P((void));
 static void get_input __P((void));
@@ -162,13 +166,13 @@ main(argc, argv)
 {
     int i, fdflags;
     struct sigaction sa;
 {
     int i, fdflags;
     struct sigaction sa;
-    FILE *pidfile;
     char *p;
     struct passwd *pw;
     struct timeval timo;
     sigset_t mask;
     struct protent *protp;
     struct stat statbuf;
     char *p;
     struct passwd *pw;
     struct timeval timo;
     sigset_t mask;
     struct protent *protp;
     struct stat statbuf;
+    char numbuf[16];
 
     phase = PHASE_INITIALIZE;
     p = ttyname(0);
 
     phase = PHASE_INITIALIZE;
     p = ttyname(0);
@@ -176,6 +180,8 @@ main(argc, argv)
        strcpy(devnam, p);
     strcpy(default_devnam, devnam);
 
        strcpy(devnam, p);
     strcpy(default_devnam, devnam);
 
+    script_env = NULL;
+
     /* Initialize syslog facilities */
 #ifdef ULTRIX
     openlog("pppd", LOG_PID);
     /* Initialize syslog facilities */
 #ifdef ULTRIX
     openlog("pppd", LOG_PID);
@@ -192,6 +198,8 @@ main(argc, argv)
 
     uid = getuid();
     privileged = uid == 0;
 
     uid = getuid();
     privileged = uid == 0;
+    sprintf(numbuf, "%d", uid);
+    script_setenv("ORIG_UID", numbuf);
 
     /*
      * Initialize to the standard option set, then parse, in order,
 
     /*
      * Initialize to the standard option set, then parse, in order,
@@ -200,7 +208,7 @@ main(argc, argv)
      */
     for (i = 0; (protp = protocols[i]) != NULL; ++i)
         (*protp->init)(0);
      */
     for (i = 0; (protp = protocols[i]) != NULL; ++i)
         (*protp->init)(0);
-  
+
     progname = *argv;
 
     if (!options_from_file(_PATH_SYSOPTIONS, !privileged, 0, 1)
     progname = *argv;
 
     if (!options_from_file(_PATH_SYSOPTIONS, !privileged, 0, 1)
@@ -238,13 +246,18 @@ main(argc, argv)
        exit(1);
     }
 
        exit(1);
     }
 
+    script_setenv("DEVICE", devnam);
+    sprintf(numbuf, "%d", baud_rate);
+    script_setenv("SPEED", numbuf);
+
     /*
      * If the user has specified the default device name explicitly,
      * pretend they hadn't.
      */
     if (!default_device && strcmp(devnam, default_devnam) == 0)
        default_device = 1;
     /*
      * If the user has specified the default device name explicitly,
      * pretend they hadn't.
      */
     if (!default_device && strcmp(devnam, default_devnam) == 0)
        default_device = 1;
-    redirect_stderr = !nodetach || default_device;
+    if (default_device)
+       nodetach = 1;
 
     /*
      * Initialize system-dependent stuff and magic number package.
 
     /*
      * Initialize system-dependent stuff and magic number package.
@@ -258,10 +271,8 @@ main(argc, argv)
      * Detach ourselves from the terminal, if required,
      * and identify who is running us.
      */
      * Detach ourselves from the terminal, if required,
      * and identify who is running us.
      */
-    if (!default_device && !nodetach && daemon(0, 0) < 0) {
-       perror("Couldn't detach from controlling terminal");
-       exit(1);
-    }
+    if (nodetach == 0)
+       detach();
     pid = getpid();
     p = getlogin();
     if (p == NULL) {
     pid = getpid();
     p = getlogin();
     if (p == NULL) {
@@ -360,16 +371,9 @@ main(argc, argv)
 
        syslog(LOG_INFO, "Using interface ppp%d", ifunit);
        (void) sprintf(ifname, "ppp%d", ifunit);
 
        syslog(LOG_INFO, "Using interface ppp%d", ifunit);
        (void) sprintf(ifname, "ppp%d", ifunit);
+       script_setenv("IFNAME", ifname);
 
 
-       /* write pid to file */
-       (void) sprintf(pidfilename, "%s%s.pid", _PATH_VARRUN, ifname);
-       if ((pidfile = fopen(pidfilename, "w")) != NULL) {
-           fprintf(pidfile, "%d\n", pid);
-           (void) fclose(pidfile);
-       } else {
-           syslog(LOG_ERR, "Failed to create pid file %s: %m", pidfilename);
-           pidfilename[0] = 0;
-       }
+       create_pidfile();       /* write pid to file */
 
        /*
         * Configure the interface and mark it up, etc.
 
        /*
         * Configure the interface and mark it up, etc.
@@ -508,17 +512,9 @@ main(argc, argv)
            
            syslog(LOG_INFO, "Using interface ppp%d", ifunit);
            (void) sprintf(ifname, "ppp%d", ifunit);
            
            syslog(LOG_INFO, "Using interface ppp%d", ifunit);
            (void) sprintf(ifname, "ppp%d", ifunit);
-           
-           /* write pid to file */
-           (void) sprintf(pidfilename, "%s%s.pid", _PATH_VARRUN, ifname);
-           if ((pidfile = fopen(pidfilename, "w")) != NULL) {
-               fprintf(pidfile, "%d\n", pid);
-               (void) fclose(pidfile);
-           } else {
-               syslog(LOG_ERR, "Failed to create pid file %s: %m",
-                      pidfilename);
-               pidfilename[0] = 0;
-           }
+           script_setenv("IFNAME", ifname);
+
+           create_pidfile();   /* write pid to file */
        }
 
        /*
        }
 
        /*
@@ -610,6 +606,43 @@ main(argc, argv)
     return 0;
 }
 
     return 0;
 }
 
+/*
+ * detach - detach us from the controlling terminal.
+ */
+void
+detach()
+{
+    if (detached)
+       return;
+    if (daemon(0, 0) < 0) {
+       perror("Couldn't detach from controlling terminal");
+       die(1);
+    }
+    detached = 1;
+    pid = getpid();
+    /* update pid file if it has been written already */
+    if (pidfilename[0])
+       create_pidfile();
+}
+
+/*
+ * Create a file containing our process ID.
+ */
+static void
+create_pidfile()
+{
+    FILE *pidfile;
+
+    (void) sprintf(pidfilename, "%s%s.pid", _PATH_VARRUN, ifname);
+    if ((pidfile = fopen(pidfilename, "w")) != NULL) {
+       fprintf(pidfile, "%d\n", pid);
+       (void) fclose(pidfile);
+    } else {
+       syslog(LOG_ERR, "Failed to create pid file %s: %m", pidfilename);
+       pidfilename[0] = 0;
+    }
+}
+
 /*
  * holdoff_end - called via a timeout when the holdoff period ends.
  */
 /*
  * holdoff_end - called via a timeout when the holdoff period ends.
  */
@@ -1004,6 +1037,11 @@ static void
 bad_signal(sig)
     int sig;
 {
 bad_signal(sig)
     int sig;
 {
+    static int crashed = 0;
+
+    if (crashed)
+       _exit(127);
+    crashed = 1;
     syslog(LOG_ERR, "Fatal signal %d", sig);
     if (conn_running)
        kill_my_pg(SIGTERM);
     syslog(LOG_ERR, "Fatal signal %d", sig);
     if (conn_running)
        kill_my_pg(SIGTERM);
@@ -1054,9 +1092,9 @@ device_script(program, in, out)
                close(out);
            }
        }
                close(out);
            }
        }
-       if (redirect_stderr) {
+       if (nodetach == 0) {
            close(2);
            close(2);
-           errfd = open(_PATH_CONNERRS, O_WRONLY | O_APPEND | O_CREAT, 0644);
+           errfd = open(_PATH_CONNERRS, O_WRONLY | O_APPEND | O_CREAT, 0600);
            if (errfd >= 0 && errfd != 2) {
                dup2(errfd, 2);
                close(errfd);
            if (errfd >= 0 && errfd != 2) {
                dup2(errfd, 2);
                close(errfd);
@@ -1095,7 +1133,6 @@ run_program(prog, args, must_exist)
     int must_exist;
 {
     int pid;
     int must_exist;
 {
     int pid;
-    char *nullenv[1];
 
     pid = fork();
     if (pid == -1) {
 
     pid = fork();
     if (pid == -1) {
@@ -1140,8 +1177,7 @@ run_program(prog, args, must_exist)
        /* SysV recommends a second fork at this point. */
 
        /* run the program; give it a null environment */
        /* SysV recommends a second fork at this point. */
 
        /* run the program; give it a null environment */
-       nullenv[0] = NULL;
-       execve(prog, args, nullenv);
+       execve(prog, args, script_env);
        if (must_exist || errno != ENOENT)
            syslog(LOG_WARNING, "Can't execute %s: %m", prog);
        _exit(-1);
        if (must_exist || errno != ENOENT)
            syslog(LOG_WARNING, "Can't execute %s: %m", prog);
        _exit(-1);
@@ -1256,10 +1292,9 @@ pr_log __V((void *arg, char *fmt, ...))
     fmt = va_arg(pvar, char *);
 #endif
 
     fmt = va_arg(pvar, char *);
 #endif
 
-    vsprintf(buf, fmt, pvar);
+    n = vfmtmsg(buf, sizeof(buf), fmt, pvar);
     va_end(pvar);
 
     va_end(pvar);
 
-    n = strlen(buf);
     if (linep + n + 1 > line + sizeof(line)) {
        syslog(LOG_DEBUG, "%s", line);
        linep = line;
     if (linep + n + 1 > line + sizeof(line)) {
        syslog(LOG_DEBUG, "%s", line);
        linep = line;
@@ -1566,3 +1601,78 @@ vfmtmsg(buf, buflen, fmt, args)
     *buf = 0;
     return buf - buf0;
 }
     *buf = 0;
     return buf - buf0;
 }
+
+/*
+ * script_setenv - set an environment variable value to be used
+ * for scripts that we run (e.g. ip-up, auth-up, etc.)
+ */
+void
+script_setenv(var, value)
+    char *var, *value;
+{
+    int vl = strlen(var);
+    int i;
+    char *p, *newstring;
+
+    newstring = (char *) malloc(vl + strlen(value) + 2);
+    if (newstring == 0)
+       return;
+    strcpy(newstring, var);
+    newstring[vl] = '=';
+    strcpy(newstring+vl+1, value);
+
+    /* check if this variable is already set */
+    if (script_env != 0) {
+       for (i = 0; (p = script_env[i]) != 0; ++i) {
+           if (strncmp(p, var, vl) == 0 && p[vl] == '=') {
+               free(p);
+               script_env[i] = newstring;
+               return;
+           }
+       }
+    } else {
+       i = 0;
+       script_env = (char **) malloc(16 * sizeof(char *));
+       if (script_env == 0)
+           return;
+       s_env_nalloc = 16;
+    }
+
+    /* reallocate script_env with more space if needed */
+    if (i + 1 >= s_env_nalloc) {
+       int new_n = i + 17;
+       char **newenv = (char **) realloc((void *)script_env,
+                                         new_n * sizeof(char *));
+       if (newenv == 0)
+           return;
+       script_env = newenv;
+       s_env_nalloc = new_n;
+    }
+
+    script_env[i] = newstring;
+    script_env[i+1] = 0;
+}
+
+/*
+ * script_unsetenv - remove a variable from the environment
+ * for scripts.
+ */
+void
+script_unsetenv(var)
+    char *var;
+{
+    int vl = strlen(var);
+    int i;
+    char *p;
+
+    if (script_env == 0)
+       return;
+    for (i = 0; (p = script_env[i]) != 0; ++i) {
+       if (strncmp(p, var, vl) == 0 && p[vl] == '=') {
+           free(p);
+           while ((script_env[i] = script_env[i+1]) != 0)
+               ++i;
+           break;
+       }
+    }
+}