1 /*
   2  * CDDL HEADER START
   3  *
   4  * The contents of this file are subject to the terms of the
   5  * Common Development and Distribution License (the "License").
   6  * You may not use this file except in compliance with the License.
   7  *
   8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
   9  * or http://www.opensolaris.org/os/licensing.
  10  * See the License for the specific language governing permissions
  11  * and limitations under the License.
  12  *
  13  * When distributing Covered Code, include this CDDL HEADER in each
  14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
  15  * If applicable, add the following below this CDDL HEADER, with the
  16  * fields enclosed by brackets "[]" replaced with your own identifying
  17  * information: Portions Copyright [yyyy] [name of copyright owner]
  18  *
  19  * CDDL HEADER END
  20  */
  21 /*
  22  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
  23  * Use is subject to license terms.
  24  */
  25 
  26 /*      Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T     */
  27 /*        All Rights Reserved   */
  28 
  29 /*      Copyright (c) 1987, 1988 Microsoft Corporation  */
  30 /*        All Rights Reserved   */
  31 
  32 #ifdef lint
  33 /* make lint happy */
  34 #define __EXTENSIONS__
  35 #endif
  36 
  37 #include <sys/contract/process.h>
  38 #include <sys/ctfs.h>
  39 #include <sys/param.h>
  40 #include <sys/resource.h>
  41 #include <sys/stat.h>
  42 #include <sys/task.h>
  43 #include <sys/time.h>
  44 #include <sys/types.h>
  45 #include <sys/utsname.h>
  46 #include <sys/wait.h>
  47 
  48 #include <security/pam_appl.h>
  49 
  50 #include <alloca.h>
  51 #include <ctype.h>
  52 #include <deflt.h>
  53 #include <dirent.h>
  54 #include <errno.h>
  55 #include <fcntl.h>
  56 #include <grp.h>
  57 #include <libcontract.h>
  58 #include <libcontract_priv.h>
  59 #include <limits.h>
  60 #include <locale.h>
  61 #include <poll.h>
  62 #include <project.h>
  63 #include <pwd.h>
  64 #include <signal.h>
  65 #include <stdarg.h>
  66 #include <stdio.h>
  67 #include <stdlib.h>
  68 #include <string.h>
  69 #include <stropts.h>
  70 #include <time.h>
  71 #include <unistd.h>
  72 #include <libzoneinfo.h>
  73 
  74 #include "cron.h"
  75 
  76 /*
  77  * #define      DEBUG
  78  */
  79 
  80 #define MAIL            "/usr/bin/mail" /* mail program to use */
  81 #define CONSOLE         "/dev/console"  /* where messages go when cron dies */
  82 
  83 #define TMPINFILE       "/tmp/crinXXXXXX"  /* file to put stdin in for cmd  */
  84 #define TMPDIR          "/tmp"
  85 #define PFX             "crout"
  86 #define TMPOUTFILE      "/tmp/croutXXXXXX" /* file to place stdout, stderr */
  87 
  88 #define INMODE          00400           /* mode for stdin file  */
  89 #define OUTMODE         00600           /* mode for stdout file */
  90 #define ISUID           S_ISUID         /* mode for verifing at jobs */
  91 
  92 #define INFINITY        2147483647L     /* upper bound on time  */
  93 #define CUSHION         180L
  94 #define ZOMB            100             /* proc slot used for mailing output */
  95 
  96 #define JOBF            'j'
  97 #define NICEF           'n'
  98 #define USERF           'u'
  99 #define WAITF           'w'
 100 
 101 #define BCHAR           '>'
 102 #define ECHAR           '<'
 103 
 104 #define DEFAULT         0
 105 #define LOAD            1
 106 #define QBUFSIZ         80
 107 
 108 /* Defined actions for crabort() routine */
 109 #define NO_ACTION       000
 110 #define REMOVE_FIFO     001
 111 #define CONSOLE_MSG     002
 112 
 113 #define BADCD           "can't change directory to the crontab directory."
 114 #define NOREADDIR       "can't read the crontab directory."
 115 
 116 #define BADJOBOPEN      "unable to read your at job."
 117 #define BADSHELL        "because your login shell \
 118 isn't /usr/bin/sh, you can't use cron."
 119 
 120 #define BADSTAT         "can't access your crontab or at-job file. Resubmit it."
 121 #define BADPROJID       "can't set project id for your job."
 122 #define CANTCDHOME      "can't change directory to %s.\
 123 \nYour commands will not be executed."
 124 #define CANTEXECSH      "unable to exec the shell, %s, for one of your \
 125 commands."
 126 #define CANT_STR_LEN (sizeof (CANTEXECSH) > sizeof (CANTCDHOME) ? \
 127         sizeof (CANTEXECSH) : sizeof (CANTCDHOME))
 128 #define NOREAD          "can't read your crontab file.  Resubmit it."
 129 #define BADTYPE         "crontab or at-job file is not a regular file.\n"
 130 #define NOSTDIN         "unable to create a standard input file for \
 131 one of your crontab commands. \
 132 \nThat command was not executed."
 133 
 134 #define NOTALLOWED      "you are not authorized to use cron.  Sorry."
 135 #define STDERRMSG       "\n\n********************************************\
 136 *****\nCron: The previous message is the \
 137 standard output and standard error \
 138 \nof one of your cron commands.\n"
 139 
 140 #define STDOUTERR       "one of your commands generated output or errors, \
 141 but cron was unable to mail you this output.\
 142 \nRemember to redirect standard output and standard \
 143 error for each of your commands."
 144 
 145 #define CLOCK_DRIFT     "clock time drifted backwards after event!\n"
 146 #define PIDERR          "unexpected pid returned %d (ignored)"
 147 #define CRONTABERR      "Subject: Your crontab file has an error in it\n\n"
 148 #define CRONOUT         "Subject: Output from \"cron\" command\n\n"
 149 #define MALLOCERR       "out of space, cannot create new string\n"
 150 
 151 #define DIDFORK didfork
 152 #define NOFORK !didfork
 153 
 154 #define MAILBUFLEN      (8*1024)
 155 #define LINELIMIT       80
 156 #define MAILBINITFREE   (MAILBUFLEN - (sizeof (cte_intro) - 1) \
 157             - (sizeof (cte_trail1) - 1) - (sizeof (cte_trail2) - 1) - 1)
 158 
 159 #define ERR_CRONTABENT  0       /* error in crontab file entry */
 160 #define ERR_UNIXERR     1       /* error in some system call */
 161 #define ERR_CANTEXECCRON 2      /* error setting up "cron" job environment */
 162 #define ERR_CANTEXECAT  3       /* error setting up "at" job environment */
 163 #define ERR_NOTREG      4       /* error not a regular file */
 164 
 165 #define PROJECT         "project="
 166 
 167 #define MAX_LOST_CONTRACTS      2048    /* reset if this many failed abandons */
 168 
 169 #define FORMAT  "%a %b %e %H:%M:%S %Y"
 170 static char     timebuf[80];
 171 
 172 static struct message msgbuf;
 173 
 174 struct shared {
 175         int count;                      /* usage count */
 176         void (*free)(void *obj);        /* routine that will free obj */
 177         void *obj;                      /* object */
 178 };
 179 
 180 struct event {
 181         time_t time;    /* time of the event    */
 182         short etype;    /* what type of event; 0=cron, 1=at     */
 183         char *cmd;      /* command for cron, job name for at    */
 184         struct usr *u;  /* ptr to the owner (usr) of this event */
 185         struct event *link;     /* ptr to another event for this user */
 186         union {
 187                 struct { /* for crontab events */
 188                         char *minute;   /*  (these      */
 189                         char *hour;     /*   fields     */
 190                         char *daymon;   /*   are        */
 191                         char *month;    /*   from       */
 192                         char *dayweek;  /*   crontab)   */
 193                         char *input;    /* ptr to stdin */
 194                         struct shared *tz;      /* timezone of this event */
 195                         struct shared *home;    /* directory for this event */
 196                         struct shared *shell;   /* shell for this event */
 197                 } ct;
 198                 struct { /* for at events */
 199                         short exists;   /* for revising at events       */
 200                         int eventid;    /* for el_remove-ing at events  */
 201                 } at;
 202         } of;
 203 };
 204 
 205 struct usr {
 206         char *name;     /* name of user (e.g. "root")   */
 207         char *home;     /* home directory for user      */
 208         uid_t uid;      /* user id      */
 209         gid_t gid;      /* group id     */
 210         int aruncnt;    /* counter for running jobs per uid */
 211         int cruncnt;    /* counter for running cron jobs per uid */
 212         int ctid;       /* for el_remove-ing crontab events */
 213         short ctexists; /* for revising crontab events  */
 214         struct event *ctevents; /* list of this usr's crontab events */
 215         struct event *atevents; /* list of this usr's at events */
 216         struct usr *nextusr;
 217 };      /* ptr to next user     */
 218 
 219 static struct   queue
 220 {
 221         int njob;       /* limit */
 222         int nice;       /* nice for execution */
 223         int nwait;      /* wait time to next execution attempt */
 224         int nrun;       /* number running */
 225 }
 226         qd = {100, 2, 60},              /* default values for queue defs */
 227         qt[NQUEUE];
 228 static struct   queue   qq;
 229 
 230 static struct runinfo
 231 {
 232         pid_t   pid;
 233         short   que;
 234         struct  usr *rusr;      /* pointer to usr struct */
 235         char    *outfile;       /* file where stdout & stderr are trapped */
 236         short   jobtype;        /* what type of event: 0=cron, 1=at */
 237         char    *jobname;       /* command for "cron", jobname for "at" */
 238         int     mailwhendone;   /* 1 = send mail even if no ouptut */
 239         struct runinfo *next;
 240 }       *rthead;
 241 
 242 static struct miscpid {
 243         pid_t           pid;
 244         struct miscpid  *next;
 245 }       *miscpid_head;
 246 
 247 static pid_t cron_pid;  /* own pid */
 248 static char didfork = 0; /* flag to see if I'm process group leader */
 249 static int msgfd;       /* file descriptor for fifo queue */
 250 static int ecid = 1;    /* event class id for el_remove(); MUST be set to 1 */
 251 static int delayed;     /* is job being rescheduled or did it run first time */
 252 static int cwd;         /* current working directory */
 253 static struct event *next_event;        /* the next event to execute    */
 254 static struct usr *uhead;               /* ptr to the list of users     */
 255 
 256 /* Variables for error handling at reading crontabs. */
 257 static char cte_intro[] = "Line(s) with errors:\n\n";
 258 static char cte_trail1[] = "\nMax number of errors encountered.";
 259 static char cte_trail2[] = " Evaluation of crontab aborted.\n";
 260 static int cte_free = MAILBINITFREE;    /* Free buffer space */
 261 static char *cte_text = NULL;           /* Text buffer pointer */
 262 static char *cte_lp;                    /* Next free line in cte_text */
 263 static int cte_nvalid;                  /* Valid lines found */
 264 
 265 /* user's default environment for the shell */
 266 #define ROOTPATH        "PATH=/usr/sbin:/usr/bin"
 267 #define NONROOTPATH     "PATH=/usr/bin:"
 268 
 269 static char *Def_supath = NULL;
 270 static char *Def_path           = NULL;
 271 static char path[LINE_MAX]      = "PATH=";
 272 static char supath[LINE_MAX]    = "PATH=";
 273 static char homedir[LINE_MAX]   = ENV_HOME;
 274 static char logname[LINE_MAX]   = "LOGNAME=";
 275 static char tzone[LINE_MAX]     = ENV_TZ;
 276 static char *envinit[] = {
 277         homedir,
 278         logname,
 279         ROOTPATH,
 280         "SHELL=/usr/bin/sh",
 281         tzone,
 282         NULL
 283 };
 284 
 285 extern char **environ;
 286 
 287 #define DEFTZ           "GMT"
 288 static  int     log = 0;
 289 static  char    hzname[10];
 290 
 291 static void cronend(int);
 292 static void thaw_handler(int);
 293 static void child_handler(int);
 294 static void child_sigreset(void);
 295 
 296 static void mod_ctab(char *, time_t);
 297 static void mod_atjob(char *, time_t);
 298 static void add_atevent(struct usr *, char *, time_t, int);
 299 static void rm_ctevents(struct usr *);
 300 static void cleanup(struct runinfo *rn, int r);
 301 static void crabort(char *, int);
 302 static void msg(char *fmt, ...);
 303 static void ignore_msg(char *, char *, struct event *);
 304 static void logit(int, struct runinfo *, int);
 305 static void parsqdef(char *);
 306 static void defaults();
 307 static void initialize(int);
 308 static void quedefs(int);
 309 static int idle(long);
 310 static struct usr *find_usr(char *);
 311 static int ex(struct event *e);
 312 static void read_dirs(int);
 313 static void mail(char *, char *, int);
 314 static char *next_field(int, int);
 315 static void readcron(struct usr *, time_t);
 316 static int next_ge(int, char *);
 317 static void free_if_unused(struct usr *);
 318 static void del_atjob(char *, char *);
 319 static void del_ctab(char *);
 320 static void resched(int);
 321 static int msg_wait(long);
 322 static struct runinfo *rinfo_get(pid_t);
 323 static void rinfo_free(struct runinfo *rp);
 324 static void mail_result(struct usr *p, struct runinfo *pr, size_t filesize);
 325 static time_t next_time(struct event *, time_t);
 326 static time_t get_switching_time(int, time_t);
 327 static time_t xmktime(struct tm *);
 328 static void process_msg(struct message *, time_t);
 329 static void reap_child(void);
 330 static void miscpid_insert(pid_t);
 331 static int miscpid_delete(pid_t);
 332 static void contract_set_template(void);
 333 static void contract_clear_template(void);
 334 static void contract_abandon_latest(pid_t);
 335 
 336 static void cte_init(void);
 337 static void cte_add(int, char *);
 338 static void cte_valid(void);
 339 static int cte_istoomany(void);
 340 static void cte_sendmail(char *);
 341 
 342 static int set_user_cred(const struct usr *, struct project *);
 343 
 344 static struct shared *create_shared_str(char *str);
 345 static struct shared *dup_shared(struct shared *obj);
 346 static void rel_shared(struct shared *obj);
 347 static void *get_obj(struct shared *obj);
 348 /*
 349  * last_time is set immediately prior to exection of an event (via ex())
 350  * to indicate the last time an event was executed.  This was (surely)
 351  * it's original intended use.
 352  */
 353 static time_t last_time, init_time, t_old;
 354 static int reset_needed; /* set to 1 when cron(1M) needs to re-initialize */
 355 
 356 static int              refresh;
 357 static sigset_t         defmask, sigmask;
 358 
 359 /*
 360  * BSM hooks
 361  */
 362 extern int      audit_cron_session(char *, char *, uid_t, gid_t, char *);
 363 extern void     audit_cron_new_job(char *, int, void *);
 364 extern void     audit_cron_bad_user(char *);
 365 extern void     audit_cron_user_acct_expired(char *);
 366 extern int      audit_cron_create_anc_file(char *, char *, char *, uid_t);
 367 extern int      audit_cron_delete_anc_file(char *, char *);
 368 extern int      audit_cron_is_anc_name(char *);
 369 extern int      audit_cron_mode();
 370 
 371 static int cron_conv(int, struct pam_message **,
 372                 struct pam_response **, void *);
 373 
 374 static struct pam_conv pam_conv = {cron_conv, NULL};
 375 static pam_handle_t *pamh;      /* Authentication handle */
 376 
 377 /*
 378  * Function to help check a user's credentials.
 379  */
 380 
 381 static int verify_user_cred(struct usr *u);
 382 
 383 /*
 384  * Values returned by verify_user_cred and set_user_cred:
 385  */
 386 
 387 #define VUC_OK          0
 388 #define VUC_BADUSER     1
 389 #define VUC_NOTINGROUP  2
 390 #define VUC_EXPIRED     3
 391 #define VUC_NEW_AUTH    4
 392 
 393 /*
 394  * Modes of process_anc_files function
 395  */
 396 #define CRON_ANC_DELETE 1
 397 #define CRON_ANC_CREATE 0
 398 
 399 /*
 400  * Functions to remove a user or job completely from the running database.
 401  */
 402 static void clean_out_atjobs(struct usr *u);
 403 static void clean_out_ctab(struct usr *u);
 404 static void clean_out_user(struct usr *u);
 405 static void cron_unlink(char *name);
 406 static void process_anc_files(int);
 407 
 408 /*
 409  * functions in elm.c
 410  */
 411 extern void el_init(int, time_t, time_t, int);
 412 extern int el_add(void *, time_t, int);
 413 extern void el_remove(int, int);
 414 extern int el_empty(void);
 415 extern void *el_first(void);
 416 extern void el_delete(void);
 417 
 418 static int valid_entry(char *, int);
 419 static struct usr *create_ulist(char *, int);
 420 static void init_cronevent(char *, int);
 421 static void init_atevent(char *, time_t, int, int);
 422 static void update_atevent(struct usr *, char *, time_t, int);
 423 
 424 int
 425 main(int argc, char *argv[])
 426 {
 427         time_t t;
 428         time_t ne_time;         /* amt of time until next event execution */
 429         time_t newtime, lastmtime = 0L;
 430         struct usr *u;
 431         struct event *e, *e2, *eprev;
 432         struct stat buf;
 433         pid_t rfork;
 434         struct sigaction act;
 435 
 436         /*
 437          * reset_needed is set to 1 whenever el_add() finds out that a cron
 438          * job is scheduled to be run before the time when cron(1M) daemon
 439          * initialized.
 440          * Other cases where a reset is needed is when ex() finds that the
 441          * event to be executed is being run at the wrong time, or when idle()
 442          * determines that time was reset.
 443          * We immediately return to the top of the while (TRUE) loop in
 444          * main() where the event list is cleared and rebuilt, and reset_needed
 445          * is set back to 0.
 446          */
 447         reset_needed = 0;
 448 
 449         /*
 450          * Only the privileged user can run this command.
 451          */
 452         if (getuid() != 0)
 453                 crabort(NOTALLOWED, 0);
 454 
 455 begin:
 456         (void) setlocale(LC_ALL, "");
 457         /* fork unless 'nofork' is specified */
 458         if ((argc <= 1) || (strcmp(argv[1], "nofork"))) {
 459                 if (rfork = fork()) {
 460                         if (rfork == (pid_t)-1) {
 461                                 (void) sleep(30);
 462                                 goto begin;
 463                         }
 464                         return (0);
 465                 }
 466                 didfork++;
 467                 (void) setpgrp();       /* detach cron from console */
 468         }
 469 
 470         (void) umask(022);
 471         (void) signal(SIGHUP, SIG_IGN);
 472         (void) signal(SIGINT, SIG_IGN);
 473         (void) signal(SIGQUIT, SIG_IGN);
 474         (void) signal(SIGTERM, cronend);
 475 
 476         defaults();
 477         initialize(1);
 478         quedefs(DEFAULT);       /* load default queue definitions */
 479         cron_pid = getpid();
 480         msg("*** cron started ***   pid = %d", cron_pid);
 481 
 482         /* setup THAW handler */
 483         act.sa_handler = thaw_handler;
 484         act.sa_flags = 0;
 485         (void) sigemptyset(&act.sa_mask);
 486         (void) sigaction(SIGTHAW, &act, NULL);
 487 
 488         /* setup CHLD handler */
 489         act.sa_handler = child_handler;
 490         act.sa_flags = 0;
 491         (void) sigemptyset(&act.sa_mask);
 492         (void) sigaddset(&act.sa_mask, SIGCLD);
 493         (void) sigaction(SIGCLD, &act, NULL);
 494 
 495         (void) sigemptyset(&defmask);
 496         (void) sigemptyset(&sigmask);
 497         (void) sigaddset(&sigmask, SIGCLD);
 498         (void) sigaddset(&sigmask, SIGTHAW);
 499         (void) sigprocmask(SIG_BLOCK, &sigmask, NULL);
 500 
 501         t_old = init_time;
 502         last_time = t_old;
 503         for (;;) {              /* MAIN LOOP */
 504                 t = time(NULL);
 505                 if ((t_old > t) || (t-last_time > CUSHION) || reset_needed) {
 506                         reset_needed = 0;
 507                         /*
 508                          * the time was set backwards or forward or
 509                          * refresh is requested.
 510                          */
 511                         if (refresh)
 512                                 msg("re-scheduling jobs");
 513                         else
 514                                 msg("time was reset, re-initializing");
 515                         el_delete();
 516                         u = uhead;
 517                         while (u != NULL) {
 518                                 rm_ctevents(u);
 519                                 e = u->atevents;
 520                                 while (e != NULL) {
 521                                         free(e->cmd);
 522                                         e2 = e->link;
 523                                         free(e);
 524                                         e = e2;
 525                                 }
 526                                 u->atevents = NULL;
 527                                 u = u->nextusr;
 528                         }
 529                         (void) close(msgfd);
 530                         initialize(0);
 531                         t = time(NULL);
 532                         last_time = t;
 533                         /*
 534                          * reset_needed might have been set in the functions
 535                          * call path from initialize()
 536                          */
 537                         if (reset_needed) {
 538                                 continue;
 539                         }
 540                 }
 541                 t_old = t;
 542 
 543                 if (next_event == NULL && !el_empty()) {
 544                         next_event = (struct event *)el_first();
 545                 }
 546                 if (next_event == NULL) {
 547                         ne_time = INFINITY;
 548                 } else {
 549                         ne_time = next_event->time - t;
 550 #ifdef DEBUG
 551                         cftime(timebuf, "%C", &next_event->time);
 552                         (void) fprintf(stderr, "next_time=%ld %s\n",
 553                             next_event->time, timebuf);
 554 #endif
 555                 }
 556                 if (ne_time > 0) {
 557                         /*
 558                          * reset_needed may be set in the functions call path
 559                          * from idle()
 560                          */
 561                         if (idle(ne_time) || reset_needed) {
 562                                 reset_needed = 1;
 563                                 continue;
 564                         }
 565                 }
 566 
 567                 if (stat(QUEDEFS, &buf)) {
 568                         msg("cannot stat QUEDEFS file");
 569                 } else if (lastmtime != buf.st_mtime) {
 570                         quedefs(LOAD);
 571                         lastmtime = buf.st_mtime;
 572                 }
 573 
 574                 last_time = next_event->time; /* save execution time */
 575 
 576                 /*
 577                  * reset_needed may be set in the functions call path
 578                  * from ex()
 579                  */
 580                 if (ex(next_event) || reset_needed) {
 581                         reset_needed = 1;
 582                         continue;
 583                 }
 584 
 585                 switch (next_event->etype) {
 586                 case CRONEVENT:
 587                         /* add cronevent back into the main event list */
 588                         if (delayed) {
 589                                 delayed = 0;
 590                                 break;
 591                         }
 592 
 593                         /*
 594                          * check if time(0)< last_time. if so, then the
 595                          * system clock has gone backwards. to prevent this
 596                          * job from being started twice, we reschedule this
 597                          * job for the >>next time after last_time<<, and
 598                          * then set next_event->time to this. note that
 599                          * crontab's resolution is 1 minute.
 600                          */
 601 
 602                         if (last_time > time(NULL)) {
 603                                 msg(CLOCK_DRIFT);
 604                                 /*
 605                                  * bump up to next 30 second
 606                                  * increment
 607                                  * 1 <= newtime <= 30
 608                                  */
 609                                 newtime = 30 - (last_time % 30);
 610                                 newtime += last_time;
 611 
 612                                 /*
 613                                  * get the next scheduled event,
 614                                  * not the one that we just
 615                                  * kicked off!
 616                                  */
 617                                 next_event->time =
 618                                     next_time(next_event, newtime);
 619                                 t_old = time(NULL);
 620                         } else {
 621                                 next_event->time =
 622                                     next_time(next_event, (time_t)0);
 623                         }
 624 #ifdef DEBUG
 625                         cftime(timebuf, "%C", &next_event->time);
 626                         (void) fprintf(stderr,
 627                             "pushing back cron event %s at %ld (%s)\n",
 628                             next_event->cmd, next_event->time, timebuf);
 629 #endif
 630 
 631                         switch (el_add(next_event, next_event->time,
 632                             (next_event->u)->ctid)) {
 633                         case -1:
 634                                 ignore_msg("main", "cron", next_event);
 635                                 break;
 636                         case -2: /* event time lower than init time */
 637                                 reset_needed = 1;
 638                                 break;
 639                         }
 640                         break;
 641                 default:
 642                         /* remove at or batch job from system */
 643                         if (delayed) {
 644                                 delayed = 0;
 645                                 break;
 646                         }
 647                         eprev = NULL;
 648                         e = (next_event->u)->atevents;
 649                         while (e != NULL) {
 650                                 if (e == next_event) {
 651                                         if (eprev == NULL)
 652                                                 (e->u)->atevents = e->link;
 653                                         else
 654                                                 eprev->link = e->link;
 655                                         free(e->cmd);
 656                                         free(e);
 657                                         break;
 658                                 } else {
 659                                         eprev = e;
 660                                         e = e->link;
 661                                 }
 662                         }
 663                         break;
 664                 }
 665                 next_event = NULL;
 666         }
 667 
 668         /*NOTREACHED*/
 669 }
 670 
 671 static void
 672 initialize(int firstpass)
 673 {
 674 #ifdef DEBUG
 675         (void) fprintf(stderr, "in initialize\n");
 676 #endif
 677         if (firstpass) {
 678                 /* for mail(1), make sure messages come from root */
 679                 if (putenv("LOGNAME=root") != 0) {
 680                         crabort("cannot expand env variable",
 681                             REMOVE_FIFO|CONSOLE_MSG);
 682                 }
 683                 if (access(FIFO, R_OK) == -1) {
 684                         if (errno == ENOENT) {
 685                                 if (mknod(FIFO, S_IFIFO|0600, 0) != 0)
 686                                         crabort("cannot create fifo queue",
 687                                             REMOVE_FIFO|CONSOLE_MSG);
 688                         } else {
 689                                 if (NOFORK) {
 690                                         /* didn't fork... init(1M) is waiting */
 691                                         (void) sleep(60);
 692                                 }
 693                                 perror("FIFO");
 694                                 crabort("cannot access fifo queue",
 695                                     REMOVE_FIFO|CONSOLE_MSG);
 696                         }
 697                 } else {
 698                         if (NOFORK) {
 699                                 /* didn't fork... init(1M) is waiting */
 700                                 (void) sleep(60);
 701                                 /*
 702                                  * the wait is painful, but we don't want
 703                                  * init respawning this quickly
 704                                  */
 705                         }
 706                         crabort("cannot start cron; FIFO exists", CONSOLE_MSG);
 707                 }
 708         }
 709 
 710         if ((msgfd = open(FIFO, O_RDWR)) < 0) {
 711                 perror("! open");
 712                 crabort("cannot open fifo queue", REMOVE_FIFO|CONSOLE_MSG);
 713         }
 714 
 715         init_time = time(NULL);
 716         el_init(8, init_time, (time_t)(60*60*24), 10);
 717 
 718         init_time = time(NULL);
 719         el_init(8, init_time, (time_t)(60*60*24), 10);
 720 
 721         /*
 722          * read directories, create users list, and add events to the
 723          * main event list. Only zero user list on firstpass.
 724          */
 725         if (firstpass)
 726                 uhead = NULL;
 727         read_dirs(firstpass);
 728         next_event = NULL;
 729 
 730         if (!firstpass)
 731                 return;
 732 
 733         /* stdout is log file */
 734         if (freopen(ACCTFILE, "a", stdout) == NULL)
 735                 (void) fprintf(stderr, "cannot open %s\n", ACCTFILE);
 736 
 737         /* log should be root-only */
 738         (void) fchmod(1, S_IRUSR|S_IWUSR);
 739 
 740         /* stderr also goes to ACCTFILE */
 741         (void) close(fileno(stderr));
 742         (void) dup(1);
 743         /* null for stdin */
 744         (void) freopen("/dev/null", "r", stdin);
 745 
 746         contract_set_template();
 747 }
 748 
 749 static void
 750 read_dirs(int first)
 751 {
 752         DIR             *dir;
 753         struct dirent   *dp;
 754         char            *ptr;
 755         int             jobtype;
 756         time_t          tim;
 757 
 758 
 759         if (chdir(CRONDIR) == -1)
 760                 crabort(BADCD, REMOVE_FIFO|CONSOLE_MSG);
 761         cwd = CRON;
 762         if ((dir = opendir(".")) == NULL)
 763                 crabort(NOREADDIR, REMOVE_FIFO|CONSOLE_MSG);
 764         while ((dp = readdir(dir)) != NULL) {
 765                 if (!valid_entry(dp->d_name, CRONEVENT))
 766                         continue;
 767                 init_cronevent(dp->d_name, first);
 768         }
 769         (void) closedir(dir);
 770 
 771         if (chdir(ATDIR) == -1) {
 772                 msg("cannot chdir to at directory");
 773                 return;
 774         }
 775         if ((dir = opendir(".")) == NULL) {
 776                 msg("cannot read at at directory");
 777                 return;
 778         }
 779         cwd = AT;
 780         while ((dp = readdir(dir)) != NULL) {
 781                 if (!valid_entry(dp->d_name, ATEVENT))
 782                         continue;
 783                 ptr = dp->d_name;
 784                 if (((tim = num(&ptr)) == 0) || (*ptr != '.'))
 785                         continue;
 786                 ptr++;
 787                 if (!isalpha(*ptr))
 788                         continue;
 789                 jobtype = *ptr - 'a';
 790                 if (jobtype >= NQUEUE) {
 791                         cron_unlink(dp->d_name);
 792                         continue;
 793                 }
 794                 init_atevent(dp->d_name, tim, jobtype, first);
 795         }
 796         (void) closedir(dir);
 797 }
 798 
 799 static int
 800 valid_entry(char *name, int type)
 801 {
 802         struct stat     buf;
 803 
 804         if (strcmp(name, ".") == 0 ||
 805             strcmp(name, "..") == 0)
 806                 return (0);
 807 
 808         /* skip over ancillary file names */
 809         if (audit_cron_is_anc_name(name))
 810                 return (0);
 811 
 812         if (stat(name, &buf)) {
 813                 mail(name, BADSTAT, ERR_UNIXERR);
 814                 cron_unlink(name);
 815                 return (0);
 816         }
 817         if (!S_ISREG(buf.st_mode)) {
 818                 mail(name, BADTYPE, ERR_NOTREG);
 819                 cron_unlink(name);
 820                 return (0);
 821         }
 822         if (type == ATEVENT) {
 823                 if (!(buf.st_mode & ISUID)) {
 824                         cron_unlink(name);
 825                         return (0);
 826                 }
 827         }
 828         return (1);
 829 }
 830 
 831 struct usr *
 832 create_ulist(char *name, int type)
 833 {
 834         struct usr      *u;
 835 
 836         u = xcalloc(1, sizeof (struct usr));
 837         u->name = xstrdup(name);
 838         if (type == CRONEVENT) {
 839                 u->ctexists = TRUE;
 840                 u->ctid = ecid++;
 841         } else {
 842                 u->ctexists = FALSE;
 843                 u->ctid = 0;
 844         }
 845         u->uid = (uid_t)-1;
 846         u->gid = (uid_t)-1;
 847         u->nextusr = uhead;
 848         uhead = u;
 849         return (u);
 850 }
 851 
 852 void
 853 init_cronevent(char *name, int first)
 854 {
 855         struct usr      *u;
 856 
 857         if (first) {
 858                 u = create_ulist(name, CRONEVENT);
 859                 readcron(u, 0);
 860         } else {
 861                 if ((u = find_usr(name)) == NULL) {
 862                         u = create_ulist(name, CRONEVENT);
 863                         readcron(u, 0);
 864                 } else {
 865                         u->ctexists = TRUE;
 866                         rm_ctevents(u);
 867                         el_remove(u->ctid, 0);
 868                         readcron(u, 0);
 869                 }
 870         }
 871 }
 872 
 873 void
 874 init_atevent(char *name, time_t tim, int jobtype, int first)
 875 {
 876         struct usr      *u;
 877 
 878         if (first) {
 879                 u = create_ulist(name, ATEVENT);
 880                 add_atevent(u, name, tim, jobtype);
 881         } else {
 882                 if ((u = find_usr(name)) == NULL) {
 883                         u = create_ulist(name, ATEVENT);
 884                         add_atevent(u, name, tim, jobtype);
 885                 } else {
 886                         update_atevent(u, name, tim, jobtype);
 887                 }
 888         }
 889 }
 890 
 891 static void
 892 mod_ctab(char *name, time_t reftime)
 893 {
 894         struct  passwd  *pw;
 895         struct  stat    buf;
 896         struct  usr     *u;
 897         char    namebuf[LINE_MAX];
 898         char    *pname;
 899 
 900         /* skip over ancillary file names */
 901         if (audit_cron_is_anc_name(name))
 902                 return;
 903 
 904         if ((pw = getpwnam(name)) == NULL) {
 905                 msg("No such user as %s - cron entries not created", name);
 906                 return;
 907         }
 908         if (cwd != CRON) {
 909                 if (snprintf(namebuf, sizeof (namebuf), "%s/%s",
 910                     CRONDIR, name) >= sizeof (namebuf)) {
 911                         msg("Too long path name %s - cron entries not created",
 912                             namebuf);
 913                         return;
 914                 }
 915                 pname = namebuf;
 916         } else {
 917                 pname = name;
 918         }
 919         /*
 920          * a warning message is given by the crontab command so there is
 921          * no need to give one here......  use this code if you only want
 922          * users with a login shell of /usr/bin/sh to use cron
 923          */
 924 #ifdef BOURNESHELLONLY
 925         if ((strcmp(pw->pw_shell, "") != 0) &&
 926             (strcmp(pw->pw_shell, SHELL) != 0)) {
 927                 mail(name, BADSHELL, ERR_CANTEXECCRON);
 928                 cron_unlink(pname);
 929                 return;
 930         }
 931 #endif
 932         if (stat(pname, &buf)) {
 933                 mail(name, BADSTAT, ERR_UNIXERR);
 934                 cron_unlink(pname);
 935                 return;
 936         }
 937         if (!S_ISREG(buf.st_mode)) {
 938                 mail(name, BADTYPE, ERR_CRONTABENT);
 939                 return;
 940         }
 941         if ((u = find_usr(name)) == NULL) {
 942 #ifdef DEBUG
 943                 (void) fprintf(stderr, "new user (%s) with a crontab\n", name);
 944 #endif
 945                 u = create_ulist(name, CRONEVENT);
 946                 u->home = xmalloc(strlen(pw->pw_dir) + 1);
 947                 (void) strcpy(u->home, pw->pw_dir);
 948                 u->uid = pw->pw_uid;
 949                 u->gid = pw->pw_gid;
 950                 readcron(u, reftime);
 951         } else {
 952                 u->uid = pw->pw_uid;
 953                 u->gid = pw->pw_gid;
 954                 if (u->home != NULL) {
 955                         if (strcmp(u->home, pw->pw_dir) != 0) {
 956                                 free(u->home);
 957                                 u->home = xmalloc(strlen(pw->pw_dir) + 1);
 958                                 (void) strcpy(u->home, pw->pw_dir);
 959                         }
 960                 } else {
 961                         u->home = xmalloc(strlen(pw->pw_dir) + 1);
 962                         (void) strcpy(u->home, pw->pw_dir);
 963                 }
 964                 u->ctexists = TRUE;
 965                 if (u->ctid == 0) {
 966 #ifdef DEBUG
 967                         (void) fprintf(stderr, "%s now has a crontab\n",
 968                             u->name);
 969 #endif
 970                         /* user didnt have a crontab last time */
 971                         u->ctid = ecid++;
 972                         u->ctevents = NULL;
 973                         readcron(u, reftime);
 974                         return;
 975                 }
 976 #ifdef DEBUG
 977                 (void) fprintf(stderr, "%s has revised his crontab\n", u->name);
 978 #endif
 979                 rm_ctevents(u);
 980                 el_remove(u->ctid, 0);
 981                 readcron(u, reftime);
 982         }
 983 }
 984 
 985 /* ARGSUSED */
 986 static void
 987 mod_atjob(char *name, time_t reftime)
 988 {
 989         char    *ptr;
 990         time_t  tim;
 991         struct  passwd  *pw;
 992         struct  stat    buf;
 993         struct  usr     *u;
 994         char    namebuf[PATH_MAX];
 995         char    *pname;
 996         int     jobtype;
 997 
 998         ptr = name;
 999         if (((tim = num(&ptr)) == 0) || (*ptr != '.'))
1000                 return;
1001         ptr++;
1002         if (!isalpha(*ptr))
1003                 return;
1004         jobtype = *ptr - 'a';
1005 
1006         /* check for audit ancillary file */
1007         if (audit_cron_is_anc_name(name))
1008                 return;
1009 
1010         if (cwd != AT) {
1011                 if (snprintf(namebuf, sizeof (namebuf), "%s/%s", ATDIR, name)
1012                     >= sizeof (namebuf)) {
1013                         return;
1014                 }
1015                 pname = namebuf;
1016         } else {
1017                 pname = name;
1018         }
1019         if (stat(pname, &buf) || jobtype >= NQUEUE) {
1020                 cron_unlink(pname);
1021                 return;
1022         }
1023         if (!(buf.st_mode & ISUID) || !S_ISREG(buf.st_mode)) {
1024                 cron_unlink(pname);
1025                 return;
1026         }
1027         if ((pw = getpwuid(buf.st_uid)) == NULL) {
1028                 cron_unlink(pname);
1029                 return;
1030         }
1031         /*
1032          * a warning message is given by the at command so there is no
1033          * need to give one here......use this code if you only want
1034          * users with a login shell of /usr/bin/sh to use cron
1035          */
1036 #ifdef BOURNESHELLONLY
1037         if ((strcmp(pw->pw_shell, "") != 0) &&
1038             (strcmp(pw->pw_shell, SHELL) != 0)) {
1039                 mail(pw->pw_name, BADSHELL, ERR_CANTEXECAT);
1040                 cron_unlink(pname);
1041                 return;
1042         }
1043 #endif
1044         if ((u = find_usr(pw->pw_name)) == NULL) {
1045 #ifdef DEBUG
1046                 (void) fprintf(stderr, "new user (%s) with an at job = %s\n",
1047                     pw->pw_name, name);
1048 #endif
1049                 u = create_ulist(pw->pw_name, ATEVENT);
1050                 u->home = xstrdup(pw->pw_dir);
1051                 u->uid = pw->pw_uid;
1052                 u->gid = pw->pw_gid;
1053                 add_atevent(u, name, tim, jobtype);
1054         } else {
1055                 u->uid = pw->pw_uid;
1056                 u->gid = pw->pw_gid;
1057                 free(u->home);
1058                 u->home = xstrdup(pw->pw_dir);
1059                 update_atevent(u, name, tim, jobtype);
1060         }
1061 }
1062 
1063 static void
1064 add_atevent(struct usr *u, char *job, time_t tim, int jobtype)
1065 {
1066         struct event *e;
1067 
1068         e = xmalloc(sizeof (struct event));
1069         e->etype = jobtype;
1070         e->cmd = xmalloc(strlen(job) + 1);
1071         (void) strcpy(e->cmd, job);
1072         e->u = u;
1073         e->link = u->atevents;
1074         u->atevents = e;
1075         e->of.at.exists = TRUE;
1076         e->of.at.eventid = ecid++;
1077         if (tim < init_time) /* old job */
1078                 e->time = init_time;
1079         else
1080                 e->time = tim;
1081 #ifdef DEBUG
1082         (void) fprintf(stderr, "add_atevent: user=%s, job=%s, time=%ld\n",
1083             u->name, e->cmd, e->time);
1084 #endif
1085         if (el_add(e, e->time, e->of.at.eventid) < 0) {
1086                 ignore_msg("add_atevent", "at", e);
1087         }
1088 }
1089 
1090 void
1091 update_atevent(struct usr *u, char *name, time_t tim, int jobtype)
1092 {
1093         struct event *e;
1094 
1095         e = u->atevents;
1096         while (e != NULL) {
1097                 if (strcmp(e->cmd, name) == 0) {
1098                         e->of.at.exists = TRUE;
1099                         break;
1100                 } else {
1101                         e = e->link;
1102                 }
1103         }
1104         if (e == NULL) {
1105 #ifdef DEBUG
1106                 (void) fprintf(stderr, "%s has a new at job = %s\n",
1107                     u->name, name);
1108 #endif
1109                         add_atevent(u, name, tim, jobtype);
1110         }
1111 }
1112 
1113 static char line[CTLINESIZE];   /* holds a line from a crontab file */
1114 static int cursor;              /* cursor for the above line */
1115 
1116 static void
1117 readcron(struct usr *u, time_t reftime)
1118 {
1119         /*
1120          * readcron reads in a crontab file for a user (u). The list of
1121          * events for user u is built, and u->events is made to point to
1122          * this list. Each event is also entered into the main event
1123          * list.
1124          */
1125         FILE *cf;       /* cf will be a user's crontab file */
1126         struct event *e;
1127         int start;
1128         unsigned int i;
1129         char namebuf[PATH_MAX];
1130         char *pname;
1131         struct shared *tz = NULL;
1132         struct shared *home = NULL;
1133         struct shared *shell = NULL;
1134         int lineno = 0;
1135 
1136         /* read the crontab file */
1137         cte_init();             /* Init error handling */
1138         if (cwd != CRON) {
1139                 if (snprintf(namebuf, sizeof (namebuf), "%s/%s",
1140                     CRONDIR, u->name) >= sizeof (namebuf)) {
1141                         return;
1142                 }
1143                 pname = namebuf;
1144         } else {
1145                 pname = u->name;
1146         }
1147         if ((cf = fopen(pname, "r")) == NULL) {
1148                 mail(u->name, NOREAD, ERR_UNIXERR);
1149                 return;
1150         }
1151         while (fgets(line, CTLINESIZE, cf) != NULL) {
1152                 char *tmp;
1153                 /* process a line of a crontab file */
1154                 lineno++;
1155                 if (cte_istoomany())
1156                         break;
1157                 cursor = 0;
1158                 while (line[cursor] == ' ' || line[cursor] == '\t')
1159                         cursor++;
1160                 if (line[cursor] == '#' || line[cursor] == '\n')
1161                         continue;
1162 
1163                 if (strncmp(&line[cursor], ENV_TZ,
1164                     strlen(ENV_TZ)) == 0) {
1165                         if ((tmp = strchr(&line[cursor], '\n')) != NULL) {
1166                                 *tmp = NULL;
1167                         }
1168 
1169                         if (!isvalid_tz(&line[cursor + strlen(ENV_TZ)], NULL,
1170                             _VTZ_ALL)) {
1171                                 cte_add(lineno, line);
1172                                 break;
1173                         }
1174                         if (tz == NULL || strcmp(&line[cursor], get_obj(tz))) {
1175                                 rel_shared(tz);
1176                                 tz = create_shared_str(&line[cursor]);
1177                         }
1178                         continue;
1179                 }
1180 
1181                 if (strncmp(&line[cursor], ENV_HOME,
1182                     strlen(ENV_HOME)) == 0) {
1183                         if ((tmp = strchr(&line[cursor], '\n')) != NULL) {
1184                                 *tmp = NULL;
1185                         }
1186                         if (home == NULL ||
1187                             strcmp(&line[cursor], get_obj(home))) {
1188                                 rel_shared(home);
1189                                 home = create_shared_str(
1190                                     &line[cursor + strlen(ENV_HOME)]);
1191                         }
1192                         continue;
1193                 }
1194 
1195                 if (strncmp(&line[cursor], ENV_SHELL,
1196                     strlen(ENV_SHELL)) == 0) {
1197                         if ((tmp = strchr(&line[cursor], '\n')) != NULL) {
1198                                 *tmp = NULL;
1199                         }
1200                         if (shell == NULL ||
1201                             strcmp(&line[cursor], get_obj(shell))) {
1202                                 rel_shared(shell);
1203                                 shell = create_shared_str(&line[cursor]);
1204                         }
1205                         continue;
1206                 }
1207 
1208                 e = xmalloc(sizeof (struct event));
1209                 e->etype = CRONEVENT;
1210                 if (!(((e->of.ct.minute = next_field(0, 59)) != NULL) &&
1211                     ((e->of.ct.hour = next_field(0, 23)) != NULL) &&
1212                     ((e->of.ct.daymon = next_field(1, 31)) != NULL) &&
1213                     ((e->of.ct.month = next_field(1, 12)) != NULL) &&
1214                     ((e->of.ct.dayweek = next_field(0, 6)) != NULL))) {
1215                         free(e);
1216                         cte_add(lineno, line);
1217                         continue;
1218                 }
1219                 while (line[cursor] == ' ' || line[cursor] == '\t')
1220                         cursor++;
1221                 if (line[cursor] == '\n' || line[cursor] == '\0')
1222                         continue;
1223                 /* get the command to execute   */
1224                 start = cursor;
1225 again:
1226                 while ((line[cursor] != '%') &&
1227                     (line[cursor] != '\n') &&
1228                     (line[cursor] != '\0') &&
1229                     (line[cursor] != '\\'))
1230                         cursor++;
1231                 if (line[cursor] == '\\') {
1232                         cursor += 2;
1233                         goto again;
1234                 }
1235                 e->cmd = xmalloc(cursor-start + 1);
1236                 (void) strncpy(e->cmd, line + start, cursor-start);
1237                 e->cmd[cursor-start] = '\0';
1238                 /* see if there is any standard input   */
1239                 if (line[cursor] == '%') {
1240                         e->of.ct.input = xmalloc(strlen(line)-cursor + 1);
1241                         (void) strcpy(e->of.ct.input, line + cursor + 1);
1242                         for (i = 0; i < strlen(e->of.ct.input); i++) {
1243                                 if (e->of.ct.input[i] == '%')
1244                                         e->of.ct.input[i] = '\n';
1245                         }
1246                 } else {
1247                         e->of.ct.input = NULL;
1248                 }
1249                 /* set the timezone of this entry */
1250                 e->of.ct.tz = dup_shared(tz);
1251                 /* set the shell of this entry */
1252                 e->of.ct.shell = dup_shared(shell);
1253                 /* set the home of this entry */
1254                 e->of.ct.home = dup_shared(home);
1255                 /* have the event point to it's owner   */
1256                 e->u = u;
1257                 /* insert this event at the front of this user's event list */
1258                 e->link = u->ctevents;
1259                 u->ctevents = e;
1260                 /* set the time for the first occurance of this event   */
1261                 e->time = next_time(e, reftime);
1262                 /* finally, add this event to the main event list       */
1263                 switch (el_add(e, e->time, u->ctid)) {
1264                 case -1:
1265                         ignore_msg("readcron", "cron", e);
1266                         break;
1267                 case -2: /* event time lower than init time */
1268                         reset_needed = 1;
1269                         break;
1270                 }
1271                 cte_valid();
1272 #ifdef DEBUG
1273                 cftime(timebuf, "%C", &e->time);
1274                 (void) fprintf(stderr, "inserting cron event %s at %ld (%s)\n",
1275                     e->cmd, e->time, timebuf);
1276 #endif
1277         }
1278         cte_sendmail(u->name);       /* mail errors if any to user */
1279         (void) fclose(cf);
1280         rel_shared(tz);
1281         rel_shared(shell);
1282         rel_shared(home);
1283 }
1284 
1285 /*
1286  * Below are the functions for handling of errors in crontabs. Concept is to
1287  * collect faulty lines and send one email at the end of the crontab
1288  * evaluation. If there are erroneous lines only ((cte_nvalid == 0), evaluation
1289  * of crontab is aborted. Otherwise reading of crontab is continued to the end
1290  * of the file but no further error logging appears.
1291  */
1292 static void
1293 cte_init()
1294 {
1295         if (cte_text == NULL)
1296                 cte_text = xmalloc(MAILBUFLEN);
1297         (void) strlcpy(cte_text, cte_intro, MAILBUFLEN);
1298         cte_lp = cte_text + sizeof (cte_intro) - 1;
1299         cte_free = MAILBINITFREE;
1300         cte_nvalid = 0;
1301 }
1302 
1303 static void
1304 cte_add(int lineno, char *ctline)
1305 {
1306         int len;
1307         char *p;
1308 
1309         if (cte_free >= LINELIMIT) {
1310                 (void) sprintf(cte_lp, "%4d: ", lineno);
1311                 (void) strlcat(cte_lp, ctline, LINELIMIT - 1);
1312                 len = strlen(cte_lp);
1313                 if (cte_lp[len - 1] != '\n') {
1314                         cte_lp[len++] = '\n';
1315                         cte_lp[len] = '\0';
1316                 }
1317                 for (p = cte_lp; *p; p++) {
1318                         if (isprint(*p) || *p == '\n' || *p == '\t')
1319                                 continue;
1320                         *p = '.';
1321                 }
1322                 cte_lp += len;
1323                 cte_free -= len;
1324                 if (cte_free < LINELIMIT) {
1325                         size_t buflen = MAILBUFLEN - (cte_lp - cte_text);
1326                         (void) strlcpy(cte_lp, cte_trail1, buflen);
1327                         if (cte_nvalid == 0)
1328                                 (void) strlcat(cte_lp, cte_trail2, buflen);
1329                 }
1330         }
1331 }
1332 
1333 static void
1334 cte_valid()
1335 {
1336         cte_nvalid++;
1337 }
1338 
1339 static int
1340 cte_istoomany()
1341 {
1342         /*
1343          * Return TRUE only if all lines are faulty. So evaluation of
1344          * a crontab is not aborted if at least one valid line was found.
1345          */
1346         return (cte_nvalid == 0 && cte_free < LINELIMIT);
1347 }
1348 
1349 static void
1350 cte_sendmail(char *username)
1351 {
1352         if (cte_free < MAILBINITFREE)
1353                 mail(username, cte_text, ERR_CRONTABENT);
1354 }
1355 
1356 /*
1357  * Send mail with error message to a user
1358  */
1359 static void
1360 mail(char *usrname, char *mesg, int format)
1361 {
1362         /* mail mails a user a message. */
1363         FILE *pipe;
1364         char *temp;
1365         struct passwd   *ruser_ids;
1366         pid_t fork_val;
1367         int saveerrno = errno;
1368         struct utsname  name;
1369 
1370 #ifdef TESTING
1371         return;
1372 #endif
1373         (void) uname(&name);
1374         if ((fork_val = fork()) == (pid_t)-1) {
1375                 msg("cron cannot fork\n");
1376                 return;
1377         }
1378         if (fork_val == 0) {
1379                 child_sigreset();
1380                 contract_clear_template();
1381                 if ((ruser_ids = getpwnam(usrname)) == NULL)
1382                         exit(0);
1383                 (void) setuid(ruser_ids->pw_uid);
1384                 temp = xmalloc(strlen(MAIL) + strlen(usrname) + 2);
1385                 (void) sprintf(temp, "%s %s", MAIL, usrname);
1386                 pipe = popen(temp, "w");
1387                 if (pipe != NULL) {
1388                         (void) fprintf(pipe, "To: %s\n", usrname);
1389                         switch (format) {
1390                         case ERR_CRONTABENT:
1391                                 (void) fprintf(pipe, CRONTABERR);
1392                                 (void) fprintf(pipe, "Your \"crontab\" on %s\n",
1393                                     name.nodename);
1394                                 (void) fprintf(pipe, mesg);
1395                                 (void) fprintf(pipe,
1396                                     "\nEntries or crontab have been ignored\n");
1397                                 break;
1398                         case ERR_UNIXERR:
1399                                 (void) fprintf(pipe, "Subject: %s\n\n", mesg);
1400                                 (void) fprintf(pipe,
1401                                     "The error on %s was \"%s\"\n",
1402                                     name.nodename, errmsg(saveerrno));
1403                                 break;
1404 
1405                         case ERR_CANTEXECCRON:
1406                                 (void) fprintf(pipe,
1407                                 "Subject: Couldn't run your \"cron\" job\n\n");
1408                                 (void) fprintf(pipe,
1409                                     "Your \"cron\" job on %s ", name.nodename);
1410                                 (void) fprintf(pipe, "couldn't be run\n");
1411                                 (void) fprintf(pipe, "%s\n", mesg);
1412                                 (void) fprintf(pipe,
1413                                 "The error was \"%s\"\n", errmsg(saveerrno));
1414                                 break;
1415 
1416                         case ERR_CANTEXECAT:
1417                                 (void) fprintf(pipe,
1418                                 "Subject: Couldn't run your \"at\" job\n\n");
1419                                 (void) fprintf(pipe, "Your \"at\" job on %s ",
1420                                     name.nodename);
1421                                 (void) fprintf(pipe, "couldn't be run\n");
1422                                 (void) fprintf(pipe, "%s\n", mesg);
1423                                 (void) fprintf(pipe,
1424                                 "The error was \"%s\"\n", errmsg(saveerrno));
1425                                 break;
1426 
1427                         default:
1428                                 break;
1429                         }
1430                         (void) pclose(pipe);
1431                 }
1432                 free(temp);
1433                 exit(0);
1434         }
1435 
1436         contract_abandon_latest(fork_val);
1437 
1438         if (cron_pid == getpid()) {
1439                 miscpid_insert(fork_val);
1440         }
1441 }
1442 
1443 static char *
1444 next_field(int lower, int upper)
1445 {
1446         /*
1447          * next_field returns a pointer to a string which holds the next
1448          * field of a line of a crontab file.
1449          *   if (numbers in this field are out of range (lower..upper),
1450          *      or there is a syntax error) then
1451          *      NULL is returned, and a mail message is sent to the
1452          *      user telling him which line the error was in.
1453          */
1454 
1455         char *s;
1456         int num, num2, start;
1457 
1458         while ((line[cursor] == ' ') || (line[cursor] == '\t'))
1459                 cursor++;
1460         start = cursor;
1461         if (line[cursor] == '\0') {
1462                 return (NULL);
1463         }
1464         if (line[cursor] == '*') {
1465                 cursor++;
1466                 if ((line[cursor] != ' ') && (line[cursor] != '\t'))
1467                         return (NULL);
1468                 s = xmalloc(2);
1469                 (void) strcpy(s, "*");
1470                 return (s);
1471         }
1472         for (;;) {
1473                 if (!isdigit(line[cursor]))
1474                         return (NULL);
1475                 num = 0;
1476                 do {
1477                         num = num*10 + (line[cursor]-'0');
1478                 } while (isdigit(line[++cursor]));
1479                 if ((num < lower) || (num > upper))
1480                         return (NULL);
1481                 if (line[cursor] == '-') {
1482                         if (!isdigit(line[++cursor]))
1483                                 return (NULL);
1484                         num2 = 0;
1485                         do {
1486                                 num2 = num2*10 + (line[cursor]-'0');
1487                         } while (isdigit(line[++cursor]));
1488                         if ((num2 < lower) || (num2 > upper))
1489                                 return (NULL);
1490                 }
1491                 if ((line[cursor] == ' ') || (line[cursor] == '\t'))
1492                         break;
1493                 if (line[cursor] == '\0')
1494                         return (NULL);
1495                 if (line[cursor++] != ',')
1496                         return (NULL);
1497         }
1498         s = xmalloc(cursor-start + 1);
1499         (void) strncpy(s, line + start, cursor-start);
1500         s[cursor-start] = '\0';
1501         return (s);
1502 }
1503 
1504 #define tm_cmp(t1, t2) (\
1505         (t1)->tm_year == (t2)->tm_year && \
1506         (t1)->tm_mon == (t2)->tm_mon && \
1507         (t1)->tm_mday == (t2)->tm_mday && \
1508         (t1)->tm_hour == (t2)->tm_hour && \
1509         (t1)->tm_min == (t2)->tm_min)
1510 
1511 #define tm_setup(tp, yr, mon, dy, hr, min, dst) \
1512         (tp)->tm_year = yr; \
1513         (tp)->tm_mon = mon; \
1514         (tp)->tm_mday = dy; \
1515         (tp)->tm_hour = hr; \
1516         (tp)->tm_min = min; \
1517         (tp)->tm_isdst = dst; \
1518         (tp)->tm_sec = 0; \
1519         (tp)->tm_wday = 0; \
1520         (tp)->tm_yday = 0;
1521 
1522 /*
1523  * modification for bugid 1104537. the second argument to next_time is
1524  * now the value of time(2) to be used. if this is 0, then use the
1525  * current time. otherwise, the second argument is the time from which to
1526  * calculate things. this is useful to correct situations where you've
1527  * gone backwards in time (I.e. the system's internal clock is correcting
1528  * itself backwards).
1529  */
1530 
1531 
1532 
1533 static time_t
1534 tz_next_time(struct event *e, time_t tflag)
1535 {
1536         /*
1537          * returns the integer time for the next occurance of event e.
1538          * the following fields have ranges as indicated:
1539          * PRGM  | min  hour    day of month    mon     day of week
1540          * ------|-------------------------------------------------------
1541          * cron  | 0-59 0-23        1-31        1-12    0-6 (0=sunday)
1542          * time  | 0-59 0-23        1-31        0-11    0-6 (0=sunday)
1543          * NOTE: this routine is hard to understand.
1544          */
1545 
1546         struct tm *tm, ref_tm, tmp, tmp1, tmp2;
1547         int tm_mon, tm_mday, tm_wday, wday, m, min, h, hr, carry, day, days;
1548         int d1, day1, carry1, d2, day2, carry2, daysahead, mon, yr, db, wd;
1549         int today;
1550         time_t t, ref_t, t1, t2, zone_start;
1551         int fallback;
1552         extern int days_btwn(int, int, int, int, int, int);
1553 
1554         if (tflag == 0) {
1555                 t = time(NULL); /* original way of doing things */
1556         } else {
1557                 t =  tflag;
1558         }
1559 
1560         tm = &ref_tm;       /* use a local variable and call localtime_r() */
1561         ref_t = t;      /* keep a copy of the reference time */
1562 
1563 recalc:
1564         fallback = 0;
1565 
1566         (void) localtime_r(&t, tm);
1567 
1568         if (daylight) {
1569                 tmp = *tm;
1570                 tmp.tm_isdst = (tm->tm_isdst > 0 ? 0 : 1);
1571                 t1 = xmktime(&tmp);
1572                 /*
1573                  * see if we will have timezone switch over, and clock will
1574                  * fall back. zone_start will hold the time when it happens
1575                  * (ie time of PST -> PDT switch over).
1576                  */
1577                 if (tm->tm_isdst != tmp.tm_isdst &&
1578                     (t1 - t) == (timezone - altzone) &&
1579                     tm_cmp(tm, &tmp)) {
1580                         zone_start = get_switching_time(tmp.tm_isdst, t);
1581                         fallback = 1;
1582                 }
1583         }
1584 
1585         tm_mon = next_ge(tm->tm_mon + 1, e->of.ct.month) - 1;     /* 0-11 */
1586         tm_mday = next_ge(tm->tm_mday, e->of.ct.daymon);  /* 1-31 */
1587         tm_wday = next_ge(tm->tm_wday, e->of.ct.dayweek); /* 0-6  */
1588         today = TRUE;
1589         if ((strcmp(e->of.ct.daymon, "*") == 0 && tm->tm_wday != tm_wday) ||
1590             (strcmp(e->of.ct.dayweek, "*") == 0 && tm->tm_mday != tm_mday) ||
1591             (tm->tm_mday != tm_mday && tm->tm_wday != tm_wday) ||
1592             (tm->tm_mon != tm_mon)) {
1593                 today = FALSE;
1594         }
1595         m = tm->tm_min + (t == ref_t ? 1 : 0);
1596         if ((tm->tm_hour + 1) <= next_ge(tm->tm_hour, e->of.ct.hour)) {
1597                 m = 0;
1598         }
1599         min = next_ge(m%60, e->of.ct.minute);
1600         carry = (min < m) ? 1 : 0;
1601         h = tm->tm_hour + carry;
1602         hr = next_ge(h%24, e->of.ct.hour);
1603         carry = (hr < h) ? 1 : 0;
1604 
1605         if (carry == 0 && today) {
1606                 /* this event must occur today */
1607                 tm_setup(&tmp, tm->tm_year, tm->tm_mon, tm->tm_mday,
1608                     hr, min, tm->tm_isdst);
1609                 tmp1 = tmp;
1610                 if ((t1 = xmktime(&tmp1)) == (time_t)-1) {
1611                         return (0);
1612                 }
1613                 if (daylight && tmp.tm_isdst != tmp1.tm_isdst) {
1614                         /* In case we are falling back */
1615                         if (fallback) {
1616                                 /* we may need to run the job once more. */
1617                                 t = zone_start;
1618                                 goto recalc;
1619                         }
1620 
1621                         /*
1622                          * In case we are not in falling back period,
1623                          * calculate the time assuming the DST. If the
1624                          * date/time is not altered by mktime, it is the
1625                          * time to execute the job.
1626                          */
1627                         tmp2 = tmp;
1628                         tmp2.tm_isdst = tmp1.tm_isdst;
1629                         if ((t1 = xmktime(&tmp2)) == (time_t)-1) {
1630                                 return (0);
1631                         }
1632                         if (tmp1.tm_isdst == tmp2.tm_isdst &&
1633                             tm_cmp(&tmp, &tmp2)) {
1634                                 /*
1635                                  * We got a valid time.
1636                                  */
1637                                 return (t1);
1638                         } else {
1639                                 /*
1640                                  * If the date does not match even if
1641                                  * we assume the alternate timezone, then
1642                                  * it must be the invalid time. eg
1643                                  * 2am while switching 1:59am to 3am.
1644                                  * t1 should point the time before the
1645                                  * switching over as we've calculate the
1646                                  * time with assuming alternate zone.
1647                                  */
1648                                 if (tmp1.tm_isdst != tmp2.tm_isdst) {
1649                                         t = get_switching_time(tmp1.tm_isdst,
1650                                             t1);
1651                                 } else {
1652                                         /* does this really happen? */
1653                                         t = get_switching_time(tmp1.tm_isdst,
1654                                             t1 - abs(timezone - altzone));
1655                                 }
1656                                 if (t == (time_t)-1) {
1657                                         return (0);
1658                                 }
1659                         }
1660                         goto recalc;
1661                 }
1662                 if (tm_cmp(&tmp, &tmp1)) {
1663                         /* got valid time */
1664                         return (t1);
1665                 } else {
1666                         /*
1667                          * This should never happen, but just in
1668                          * case, we fall back to the old code.
1669                          */
1670                         if (tm->tm_min > min) {
1671                                 t += (time_t)(hr-tm->tm_hour-1) * HOUR +
1672                                     (time_t)(60-tm->tm_min + min) * MINUTE;
1673                         } else {
1674                                 t += (time_t)(hr-tm->tm_hour) * HOUR +
1675                                     (time_t)(min-tm->tm_min) * MINUTE;
1676                         }
1677                         t1 = t;
1678                         t -= (time_t)tm->tm_sec;
1679                         (void) localtime_r(&t, &tmp);
1680                         if ((tm->tm_isdst == 0) && (tmp.tm_isdst > 0))
1681                                 t -= (timezone - altzone);
1682                         return ((t <= ref_t) ? t1 : t);
1683                 }
1684         }
1685 
1686         /*
1687          * Job won't run today, however if we have a switch over within
1688          * one hour and we will have one hour time drifting back in this
1689          * period, we may need to run the job one more time if the job was
1690          * set to run on this hour of clock.
1691          */
1692         if (fallback) {
1693                 t = zone_start;
1694                 goto recalc;
1695         }
1696 
1697         min = next_ge(0, e->of.ct.minute);
1698         hr = next_ge(0, e->of.ct.hour);
1699 
1700         /*
1701          * calculate the date of the next occurance of this event, which
1702          * will be on a different day than the current
1703          */
1704 
1705         /* check monthly day specification      */
1706         d1 = tm->tm_mday + 1;
1707         day1 = next_ge((d1-1)%days_in_mon(tm->tm_mon, tm->tm_year) + 1,
1708             e->of.ct.daymon);
1709         carry1 = (day1 < d1) ? 1 : 0;
1710 
1711         /* check weekly day specification       */
1712         d2 = tm->tm_wday + 1;
1713         wday = next_ge(d2%7, e->of.ct.dayweek);
1714         if (wday < d2)
1715                 daysahead = 7 - d2 + wday;
1716         else
1717                 daysahead = wday - d2;
1718         day2 = (d1 + daysahead-1)%days_in_mon(tm->tm_mon, tm->tm_year) + 1;
1719         carry2 = (day2 < d1) ? 1 : 0;
1720 
1721         /*
1722          *      based on their respective specifications, day1, and day2 give
1723          *      the day of the month for the next occurance of this event.
1724          */
1725         if ((strcmp(e->of.ct.daymon, "*") == 0) &&
1726             (strcmp(e->of.ct.dayweek, "*") != 0)) {
1727                 day1 = day2;
1728                 carry1 = carry2;
1729         }
1730         if ((strcmp(e->of.ct.daymon, "*") != 0) &&
1731             (strcmp(e->of.ct.dayweek, "*") == 0)) {
1732                 day2 = day1;
1733                 carry2 = carry1;
1734         }
1735 
1736         yr = tm->tm_year;
1737         if ((carry1 && carry2) || (tm->tm_mon != tm_mon)) {
1738                 /* event does not occur in this month   */
1739                 m = tm->tm_mon + 1;
1740                 mon = next_ge(m%12 + 1, e->of.ct.month) - 1; /* 0..11 */
1741                 carry = (mon < m) ? 1 : 0;
1742                 yr += carry;
1743                 /* recompute day1 and day2      */
1744                 day1 = next_ge(1, e->of.ct.daymon);
1745                 db = days_btwn(tm->tm_mon, tm->tm_mday, tm->tm_year, mon,
1746                     1, yr) + 1;
1747                 wd = (tm->tm_wday + db)%7;
1748                 /* wd is the day of the week of the first of month mon  */
1749                 wday = next_ge(wd, e->of.ct.dayweek);
1750                 if (wday < wd)
1751                         day2 = 1 + 7 - wd + wday;
1752                 else
1753                         day2 = 1 + wday - wd;
1754                 if ((strcmp(e->of.ct.daymon, "*") != 0) &&
1755                     (strcmp(e->of.ct.dayweek, "*") == 0))
1756                         day2 = day1;
1757                 if ((strcmp(e->of.ct.daymon, "*") == 0) &&
1758                     (strcmp(e->of.ct.dayweek, "*") != 0))
1759                         day1 = day2;
1760                 day = (day1 < day2) ? day1 : day2;
1761         } else {                        /* event occurs in this month   */
1762                 mon = tm->tm_mon;
1763                 if (!carry1 && !carry2)
1764                         day = (day1 < day2) ? day1 : day2;
1765                 else if (!carry1)
1766                         day = day1;
1767                 else
1768                         day = day2;
1769         }
1770 
1771         /*
1772          * now that we have the min, hr, day, mon, yr of the next event,
1773          * figure out what time that turns out to be.
1774          */
1775         tm_setup(&tmp, yr, mon, day, hr, min, -1);
1776         tmp2 = tmp;
1777         if ((t1 = xmktime(&tmp2)) == (time_t)-1) {
1778                 return (0);
1779         }
1780         if (tm_cmp(&tmp, &tmp2)) {
1781                 /*
1782                  * mktime returns clock for the current time zone. If the
1783                  * target date was in fallback period, it needs to be adjusted
1784                  * to the time comes first.
1785                  * Suppose, we are at Jan and scheduling job at 1:30am10/26/03.
1786                  * mktime returns the time in PST, but 1:30am in PDT comes
1787                  * first. So reverse the tm_isdst, and see if we have such
1788                  * time/date.
1789                  */
1790                 if (daylight) {
1791                         int dst = tmp2.tm_isdst;
1792 
1793                         tmp2 = tmp;
1794                         tmp2.tm_isdst = (dst > 0 ? 0 : 1);
1795                         if ((t2 = xmktime(&tmp2)) == (time_t)-1) {
1796                                 return (0);
1797                         }
1798                         if (tm_cmp(&tmp, &tmp2)) {
1799                                 /*
1800                                  * same time/date found in the opposite zone.
1801                                  * check the clock to see which comes early.
1802                                  */
1803                                 if (t2 > ref_t && t2 < t1) {
1804                                         t1 = t2;
1805                                 }
1806                         }
1807                 }
1808                 return (t1);
1809         } else {
1810                 /*
1811                  * mktime has set different time/date for the given date.
1812                  * This means that the next job is scheduled to be run on the
1813                  * invalid time. There are three possible invalid date/time.
1814                  * 1. Non existing day of the month. such as April 31th.
1815                  * 2. Feb 29th in the non-leap year.
1816                  * 3. Time gap during the DST switch over.
1817                  */
1818                 d1 = days_in_mon(mon, yr);
1819                 if ((mon != 1 && day > d1) || (mon == 1 && day > 29)) {
1820                         /*
1821                          * see if we have got a specific date which
1822                          * is invalid.
1823                          */
1824                         if (strcmp(e->of.ct.dayweek, "*") == 0 &&
1825                             mon == (next_ge((mon + 1)%12 + 1,
1826                             e->of.ct.month) - 1) &&
1827                             day <= next_ge(1, e->of.ct.daymon)) {
1828                                 /* job never run */
1829                                 return (0);
1830                         }
1831                         /*
1832                          * Since the day has gone invalid, we need to go to
1833                          * next month, and recalcuate the first occurrence.
1834                          * eg the cron tab such as:
1835                          * 0 0 1,15,31 1,2,3,4,5 * /usr/bin....
1836                          * 2/31 is invalid, so the next job is 3/1.
1837                          */
1838                         tmp2 = tmp;
1839                         tmp2.tm_min = 0;
1840                         tmp2.tm_hour = 0;
1841                         tmp2.tm_mday = 1; /* 1st day of the month */
1842                         if (mon == 11) {
1843                                 tmp2.tm_mon = 0;
1844                                 tmp2.tm_year = yr + 1;
1845                         } else {
1846                                 tmp2.tm_mon = mon + 1;
1847                         }
1848                         if ((t = xmktime(&tmp2)) == (time_t)-1) {
1849                                 return (0);
1850                         }
1851                 } else if (mon == 1 && day > d1) {
1852                         /*
1853                          * ie 29th in the non-leap year. Forwarding the
1854                          * clock to Feb 29th 00:00 (March 1st), and recalculate
1855                          * the next time.
1856                          */
1857                         tmp2 = tmp;
1858                         tmp2.tm_min = 0;
1859                         tmp2.tm_hour = 0;
1860                         if ((t = xmktime(&tmp2)) == (time_t)-1) {
1861                                 return (0);
1862                         }
1863                 } else if (daylight) {
1864                         /*
1865                          * Non existing time, eg 2am PST during summer time
1866                          * switch.
1867                          * We need to get the correct isdst which we are
1868                          * swithing to, by adding time difference to make sure
1869                          * that t2 is in the zone being switched.
1870                          */
1871                         t2 = t1;
1872                         t2 += abs(timezone - altzone);
1873                         (void) localtime_r(&t2, &tmp2);
1874                         zone_start = get_switching_time(tmp2.tm_isdst,
1875                             t1 - abs(timezone - altzone));
1876                         if (zone_start == (time_t)-1) {
1877                                 return (0);
1878                         }
1879                         t = zone_start;
1880                 } else {
1881                         /*
1882                          * This should never happen, but fall back to the
1883                          * old code.
1884                          */
1885                         days = days_btwn(tm->tm_mon,
1886                             tm->tm_mday, tm->tm_year, mon, day, yr);
1887                         t += (time_t)(23-tm->tm_hour)*HOUR
1888                             + (time_t)(60-tm->tm_min)*MINUTE
1889                             + (time_t)hr*HOUR + (time_t)min*MINUTE
1890                             + (time_t)days*DAY;
1891                         t1 = t;
1892                         t -= (time_t)tm->tm_sec;
1893                         (void) localtime_r(&t, &tmp);
1894                         if ((tm->tm_isdst == 0) && (tmp.tm_isdst > 0))
1895                                 t -= (timezone - altzone);
1896                         return (t <= ref_t ? t1 : t);
1897                 }
1898                 goto recalc;
1899         }
1900         /*NOTREACHED*/
1901 }
1902 
1903 static time_t
1904 next_time(struct event *e, time_t tflag)
1905 {
1906         if (e->of.ct.tz != NULL) {
1907                 time_t ret;
1908 
1909                 (void) putenv((char *)get_obj(e->of.ct.tz));
1910                 tzset();
1911                 ret = tz_next_time(e, tflag);
1912                 (void) putenv(tzone);
1913                 tzset();
1914                 return (ret);
1915         } else {
1916                 return (tz_next_time(e, tflag));
1917         }
1918 }
1919 
1920 /*
1921  * This returns TOD in time_t that zone switch will happen, and this
1922  * will be called when clock fallback is about to happen.
1923  * (ie 30minutes before the time of PST -> PDT switch. 2:00 AM PST
1924  * will fall back to 1:00 PDT. So this function will be called only
1925  * for the time between 1:00 AM PST and 2:00 PST(1:00 PST)).
1926  * First goes through the common time differences to see if zone
1927  * switch happens at those minutes later. If not, check every minutes
1928  * until 6 hours ahead see if it happens(We might have 45minutes
1929  * fallback).
1930  */
1931 static time_t
1932 get_switching_time(int to_dst, time_t t_ref)
1933 {
1934         time_t t, t1;
1935         struct tm tmp, tmp1;
1936         int hints[] = { 60, 120, 30, 90, 0}; /* minutes */
1937         int i;
1938 
1939         (void) localtime_r(&t_ref, &tmp);
1940         tmp1 = tmp;
1941         tmp1.tm_sec = 0;
1942         tmp1.tm_min = 0;
1943         if ((t = xmktime(&tmp1)) == (time_t)-1)
1944                 return ((time_t)-1);
1945 
1946         /* fast path */
1947         for (i = 0; hints[i] != 0; i++) {
1948                 t1 = t + hints[i] * 60;
1949                 (void) localtime_r(&t1, &tmp1);
1950                 if (tmp1.tm_isdst == to_dst) {
1951                         t1--;
1952                         (void) localtime_r(&t1, &tmp1);
1953                         if (tmp1.tm_isdst != to_dst) {
1954                                 return (t1 + 1);
1955                         }
1956                 }
1957         }
1958 
1959         /* ugly, but don't know other than this. */
1960         tmp1 = tmp;
1961         tmp1.tm_sec = 0;
1962         if ((t = xmktime(&tmp1)) == (time_t)-1)
1963                 return ((time_t)-1);
1964         while (t < (t_ref + 6*60*60)) { /* 6 hours should be enough */
1965                 t += 60; /* at least one minute, I assume */
1966                 (void) localtime_r(&t, &tmp);
1967                 if (tmp.tm_isdst == to_dst)
1968                         return (t);
1969         }
1970         return ((time_t)-1);
1971 }
1972 
1973 static time_t
1974 xmktime(struct tm *tmp)
1975 {
1976         time_t ret;
1977 
1978         if ((ret = mktime(tmp)) == (time_t)-1) {
1979                 if (errno == EOVERFLOW) {
1980                         return ((time_t)-1);
1981                 }
1982                 crabort("internal error: mktime failed",
1983                     REMOVE_FIFO|CONSOLE_MSG);
1984         }
1985         return (ret);
1986 }
1987 
1988 #define DUMMY   100
1989 
1990 static int
1991 next_ge(int current, char *list)
1992 {
1993         /*
1994          * list is a character field as in a crontab file;
1995          * for example: "40, 20, 50-10"
1996          * next_ge returns the next number in the list that is
1997          * greater than  or equal to current. if no numbers of list
1998          * are >= current, the smallest element of list is returned.
1999          * NOTE: current must be in the appropriate range.
2000          */
2001 
2002         char *ptr;
2003         int n, n2, min, min_gt;
2004 
2005         if (strcmp(list, "*") == 0)
2006                 return (current);
2007         ptr = list;
2008         min = DUMMY;
2009         min_gt = DUMMY;
2010         for (;;) {
2011                 if ((n = (int)num(&ptr)) == current)
2012                         return (current);
2013                 if (n < min)
2014                         min = n;
2015                 if ((n > current) && (n < min_gt))
2016                         min_gt = n;
2017                 if (*ptr == '-') {
2018                         ptr++;
2019                         if ((n2 = (int)num(&ptr)) > n) {
2020                                 if ((current > n) && (current <= n2))
2021                                         return (current);
2022                         } else {        /* range that wraps around */
2023                                 if (current > n)
2024                                         return (current);
2025                                 if (current <= n2)
2026                                         return (current);
2027                         }
2028                 }
2029                 if (*ptr == '\0')
2030                         break;
2031                 ptr += 1;
2032         }
2033         if (min_gt != DUMMY)
2034                 return (min_gt);
2035         else
2036                 return (min);
2037 }
2038 
2039 static void
2040 free_if_unused(struct usr *u)
2041 {
2042         struct usr *cur, *prev;
2043         /*
2044          *      To make sure a usr structure is idle we must check that
2045          *      there are no at jobs queued for the user; the user does
2046          *      not have a crontab, and also that there are no running at
2047          *      or cron jobs (since the runinfo structure also has a
2048          *      pointer to the usr structure).
2049          */
2050         if (!u->ctexists && u->atevents == NULL &&
2051             u->cruncnt == 0 && u->aruncnt == 0) {
2052 #ifdef DEBUG
2053                 (void) fprintf(stderr, "%s removed from usr list\n", u->name);
2054 #endif
2055                 for (cur = uhead, prev = NULL;
2056                     cur != u;
2057                     prev = cur, cur = cur->nextusr) {
2058                         if (cur == NULL) {
2059                                 return;
2060                         }
2061                 }
2062 
2063                 if (prev == NULL)
2064                         uhead = u->nextusr;
2065                 else
2066                         prev->nextusr = u->nextusr;
2067                 free(u->name);
2068                 free(u->home);
2069                 free(u);
2070         }
2071 }
2072 
2073 static void
2074 del_atjob(char *name, char *usrname)
2075 {
2076 
2077         struct  event   *e, *eprev;
2078         struct  usr     *u;
2079 
2080         if ((u = find_usr(usrname)) == NULL)
2081                 return;
2082         e = u->atevents;
2083         eprev = NULL;
2084         while (e != NULL) {
2085                 if (strcmp(name, e->cmd) == 0) {
2086                         if (next_event == e)
2087                                 next_event = NULL;
2088                         if (eprev == NULL)
2089                                 u->atevents = e->link;
2090                         else
2091                                 eprev->link = e->link;
2092                         el_remove(e->of.at.eventid, 1);
2093                         free(e->cmd);
2094                         free(e);
2095                         break;
2096                 } else {
2097                         eprev = e;
2098                         e = e->link;
2099                 }
2100         }
2101 
2102         free_if_unused(u);
2103 }
2104 
2105 static void
2106 del_ctab(char *name)
2107 {
2108 
2109         struct  usr *u;
2110 
2111         if ((u = find_usr(name)) == NULL)
2112                 return;
2113         rm_ctevents(u);
2114         el_remove(u->ctid, 0);
2115         u->ctid = 0;
2116         u->ctexists = 0;
2117 
2118         free_if_unused(u);
2119 }
2120 
2121 static void
2122 rm_ctevents(struct usr *u)
2123 {
2124         struct event *e2, *e3;
2125 
2126         /*
2127          * see if the next event (to be run by cron) is a cronevent
2128          * owned by this user.
2129          */
2130 
2131         if ((next_event != NULL) &&
2132             (next_event->etype == CRONEVENT) &&
2133             (next_event->u == u)) {
2134                 next_event = NULL;
2135         }
2136         e2 = u->ctevents;
2137         while (e2 != NULL) {
2138                 free(e2->cmd);
2139                 rel_shared(e2->of.ct.tz);
2140                 rel_shared(e2->of.ct.shell);
2141                 rel_shared(e2->of.ct.home);
2142                 free(e2->of.ct.minute);
2143                 free(e2->of.ct.hour);
2144                 free(e2->of.ct.daymon);
2145                 free(e2->of.ct.month);
2146                 free(e2->of.ct.dayweek);
2147                 if (e2->of.ct.input != NULL)
2148                         free(e2->of.ct.input);
2149                 e3 = e2->link;
2150                 free(e2);
2151                 e2 = e3;
2152         }
2153         u->ctevents = NULL;
2154 }
2155 
2156 
2157 static struct usr *
2158 find_usr(char *uname)
2159 {
2160         struct usr *u;
2161 
2162         u = uhead;
2163         while (u != NULL) {
2164                 if (strcmp(u->name, uname) == 0)
2165                         return (u);
2166                 u = u->nextusr;
2167         }
2168         return (NULL);
2169 }
2170 
2171 /*
2172  * Execute cron command or at/batch job.
2173  * If ever a premature return is added to this function pay attention to
2174  * free at_cmdfile and outfile plus jobname buffers of the runinfo structure.
2175  */
2176 static int
2177 ex(struct event *e)
2178 {
2179         int r;
2180         int fd;
2181         pid_t rfork;
2182         FILE *atcmdfp;
2183         char mailvar[4];
2184         char *at_cmdfile = NULL;
2185         struct stat buf;
2186         struct queue *qp;
2187         struct runinfo *rp;
2188         struct project proj, *pproj = NULL;
2189         union {
2190                 struct {
2191                         char buf[PROJECT_BUFSZ];
2192                         char buf2[PROJECT_BUFSZ];
2193                 } p;
2194                 char error[CANT_STR_LEN + PATH_MAX];
2195         } bufs;
2196         char *tmpfile;
2197         FILE *fptr;
2198         time_t dhltime;
2199         projid_t projid;
2200         int projflag = 0;
2201         char *home;
2202         char *sh;
2203 
2204         qp = &qt[e->etype];      /* set pointer to queue defs */
2205         if (qp->nrun >= qp->njob) {
2206                 msg("%c queue max run limit reached", e->etype + 'a');
2207                 resched(qp->nwait);
2208                 return (0);
2209         }
2210 
2211         rp = rinfo_get(0); /* allocating a new runinfo struct */
2212 
2213         /*
2214          * the tempnam() function uses malloc(3C) to allocate space for the
2215          * constructed file name, and returns a pointer to this area, which
2216          * is assigned to rp->outfile. Here rp->outfile is not overwritten.
2217          */
2218 
2219         rp->outfile = tempnam(TMPDIR, PFX);
2220         rp->jobtype = e->etype;
2221         if (e->etype == CRONEVENT) {
2222                 rp->jobname = xmalloc(strlen(e->cmd) + 1);
2223                 (void) strcpy(rp->jobname, e->cmd);
2224                 /* "cron" jobs only produce mail if there's output */
2225                 rp->mailwhendone = 0;
2226         } else {
2227                 at_cmdfile = xmalloc(strlen(ATDIR) + strlen(e->cmd) + 2);
2228                 (void) sprintf(at_cmdfile, "%s/%s", ATDIR, e->cmd);
2229                 if ((atcmdfp = fopen(at_cmdfile, "r")) == NULL) {
2230                         if (errno == ENAMETOOLONG) {
2231                                 if (chdir(ATDIR) == 0)
2232                                         cron_unlink(e->cmd);
2233                         } else {
2234                                 cron_unlink(at_cmdfile);
2235                         }
2236                         mail((e->u)->name, BADJOBOPEN, ERR_CANTEXECAT);
2237                         free(at_cmdfile);
2238                         rinfo_free(rp);
2239                         return (0);
2240                 }
2241                 rp->jobname = xmalloc(strlen(at_cmdfile) + 1);
2242                 (void) strcpy(rp->jobname, at_cmdfile);
2243 
2244                 /*
2245                  * Skip over the first two lines.
2246                  */
2247                 (void) fscanf(atcmdfp, "%*[^\n]\n");
2248                 (void) fscanf(atcmdfp, "%*[^\n]\n");
2249                 if (fscanf(atcmdfp, ": notify by mail: %3s%*[^\n]\n",
2250                     mailvar) == 1) {
2251                         /*
2252                          * Check to see if we should always send mail
2253                          * to the owner.
2254                          */
2255                         rp->mailwhendone = (strcmp(mailvar, "yes") == 0);
2256                 } else {
2257                         rp->mailwhendone = 0;
2258                 }
2259 
2260                 if (fscanf(atcmdfp, "\n: project: %d\n", &projid) == 1) {
2261                         projflag = 1;
2262                 }
2263                 (void) fclose(atcmdfp);
2264         }
2265 
2266         /*
2267          * we make sure that the system time
2268          * hasn't drifted backwards. if it has, el_add() is now
2269          * called, to make sure that the event queue is back in order,
2270          * and we set the delayed flag. cron will pick up the request
2271          * later on at the proper time.
2272          */
2273         dhltime = time(NULL);
2274         if ((dhltime - e->time) < 0) {
2275                 msg("clock time drifted backwards!\n");
2276                 if (next_event->etype == CRONEVENT) {
2277                         msg("correcting cron event\n");
2278                         next_event->time = next_time(next_event, dhltime);
2279                         switch (el_add(next_event, next_event->time,
2280                             (next_event->u)->ctid)) {
2281                         case -1:
2282                                 ignore_msg("ex", "cron", next_event);
2283                                 break;
2284                         case -2: /* event time lower than init time */
2285                                 reset_needed = 1;
2286                                 break;
2287                         }
2288                 } else { /* etype == ATEVENT */
2289                         msg("correcting batch event\n");
2290                         if (el_add(next_event, next_event->time,
2291                             next_event->of.at.eventid) < 0) {
2292                                 ignore_msg("ex", "at", next_event);
2293                         }
2294                 }
2295                 delayed++;
2296                 t_old = time(NULL);
2297                 free(at_cmdfile);
2298                 rinfo_free(rp);
2299                 return (0);
2300         }
2301 
2302         if ((rfork = fork()) == (pid_t)-1) {
2303                 reap_child();
2304                 if ((rfork = fork()) == (pid_t)-1) {
2305                         msg("cannot fork");
2306                         free(at_cmdfile);
2307                         rinfo_free(rp);
2308                         resched(60);
2309                         (void) sleep(30);
2310                         return (0);
2311                 }
2312         }
2313         if (rfork) {            /* parent process */
2314                 contract_abandon_latest(rfork);
2315 
2316                 ++qp->nrun;
2317                 rp->pid = rfork;
2318                 rp->que = e->etype;
2319                 if (e->etype != CRONEVENT)
2320                         (e->u)->aruncnt++;
2321                 else
2322                         (e->u)->cruncnt++;
2323                 rp->rusr = (e->u);
2324                 logit(BCHAR, rp, 0);
2325                 free(at_cmdfile);
2326 
2327                 return (0);
2328         }
2329 
2330         child_sigreset();
2331         contract_clear_template();
2332 
2333         if (e->etype != CRONEVENT) {
2334                 /* open jobfile as stdin to shell */
2335                 if (stat(at_cmdfile, &buf)) {
2336                         if (errno == ENAMETOOLONG) {
2337                                 if (chdir(ATDIR) == 0)
2338                                         cron_unlink(e->cmd);
2339                         } else
2340                                 cron_unlink(at_cmdfile);
2341                         mail((e->u)->name, BADJOBOPEN, ERR_CANTEXECCRON);
2342                         exit(1);
2343                 }
2344                 if (!(buf.st_mode&ISUID)) {
2345                         /*
2346                          * if setuid bit off, original owner has
2347                          * given this file to someone else
2348                          */
2349                         cron_unlink(at_cmdfile);
2350                         exit(1);
2351                 }
2352                 if ((fd = open(at_cmdfile, O_RDONLY)) == -1) {
2353                         mail((e->u)->name, BADJOBOPEN, ERR_CANTEXECCRON);
2354                         cron_unlink(at_cmdfile);
2355                         exit(1);
2356                 }
2357                 if (fd != 0) {
2358                         (void) dup2(fd, 0);
2359                         (void) close(fd);
2360                 }
2361                 /*
2362                  * retrieve the project id of the at job and convert it
2363                  * to a project name.  fail if it's not a valid project
2364                  * or if the user isn't a member of the project.
2365                  */
2366                 if (projflag == 1) {
2367                         if ((pproj = getprojbyid(projid, &proj,
2368                             (void *)&bufs.p.buf,
2369                             sizeof (bufs.p.buf))) == NULL ||
2370                             !inproj(e->u->name, pproj->pj_name,
2371                             bufs.p.buf2, sizeof (bufs.p.buf2))) {
2372                                 cron_unlink(at_cmdfile);
2373                                 mail((e->u)->name, BADPROJID, ERR_CANTEXECAT);
2374                                 exit(1);
2375                         }
2376                 }
2377         }
2378 
2379         /*
2380          * Put process in a new session, and create a new task.
2381          */
2382         if (setsid() < 0) {
2383                 msg("setsid failed with errno = %d. job failed (%s)"
2384                     " for user %s", errno, e->cmd, e->u->name);
2385                 if (e->etype != CRONEVENT)
2386                         cron_unlink(at_cmdfile);
2387                 exit(1);
2388         }
2389 
2390         /*
2391          * set correct user identification and check his account
2392          */
2393         r = set_user_cred(e->u, pproj);
2394         if (r == VUC_EXPIRED) {
2395                 msg("user (%s) account is expired", e->u->name);
2396                 audit_cron_user_acct_expired(e->u->name);
2397                 clean_out_user(e->u);
2398                 exit(1);
2399         }
2400         if (r == VUC_NEW_AUTH) {
2401                 msg("user (%s) password has expired", e->u->name);
2402                 audit_cron_user_acct_expired(e->u->name);
2403                 clean_out_user(e->u);
2404                 exit(1);
2405         }
2406         if (r != VUC_OK) {
2407                 msg("bad user (%s)", e->u->name);
2408                 audit_cron_bad_user(e->u->name);
2409                 clean_out_user(e->u);
2410                 exit(1);
2411         }
2412         /*
2413          * check user and initialize the supplementary group access list.
2414          * bugid 1230784: deleted from parent to avoid cron hang. Now
2415          * only child handles the call.
2416          */
2417 
2418         if (verify_user_cred(e->u) != VUC_OK ||
2419             setgid(e->u->gid) == -1 ||
2420             initgroups(e->u->name, e->u->gid) == -1) {
2421                 msg("bad user (%s) or setgid failed (%s)",
2422                     e->u->name, e->u->name);
2423                 audit_cron_bad_user(e->u->name);
2424                 clean_out_user(e->u);
2425                 exit(1);
2426         }
2427 
2428         if ((e->u)->uid == 0) { /* set default path */
2429                 /* path settable in defaults file */
2430                 envinit[2] = supath;
2431         } else {
2432                 envinit[2] = path;
2433         }
2434 
2435         if (e->etype != CRONEVENT) {
2436                 r = audit_cron_session(e->u->name, NULL,
2437                     e->u->uid, e->u->gid, at_cmdfile);
2438                 cron_unlink(at_cmdfile);
2439         } else {
2440                 r = audit_cron_session(e->u->name, CRONDIR,
2441                     e->u->uid, e->u->gid, NULL);
2442         }
2443         if (r != 0) {
2444                 msg("cron audit problem. job failed (%s) for user %s",
2445                     e->cmd, e->u->name);
2446                 exit(1);
2447         }
2448 
2449         audit_cron_new_job(e->cmd, e->etype, (void *)e);
2450 
2451         if (setuid(e->u->uid) == -1)  {
2452                 msg("setuid failed (%s)", e->u->name);
2453                 clean_out_user(e->u);
2454                 exit(1);
2455         }
2456 
2457         if (e->etype == CRONEVENT) {
2458                 /* check for standard input to command  */
2459                 if (e->of.ct.input != NULL) {
2460                         if ((tmpfile = strdup(TMPINFILE)) == NULL) {
2461                                 mail((e->u)->name, MALLOCERR,
2462                                     ERR_CANTEXECCRON);
2463                                 exit(1);
2464                         }
2465                         if ((fd = mkstemp(tmpfile)) == -1 ||
2466                             (fptr = fdopen(fd, "w")) == NULL) {
2467                                 mail((e->u)->name, NOSTDIN,
2468                                     ERR_CANTEXECCRON);
2469                                 cron_unlink(tmpfile);
2470                                 free(tmpfile);
2471                                 exit(1);
2472                         }
2473                         if ((fwrite(e->of.ct.input, sizeof (char),
2474                             strlen(e->of.ct.input), fptr)) !=
2475                             strlen(e->of.ct.input)) {
2476                                 mail((e->u)->name, NOSTDIN, ERR_CANTEXECCRON);
2477                                 cron_unlink(tmpfile);
2478                                 free(tmpfile);
2479                                 (void) close(fd);
2480                                 (void) fclose(fptr);
2481                                 exit(1);
2482                         }
2483                         if (fseek(fptr, (off_t)0, SEEK_SET) != -1) {
2484                                 if (fd != 0) {
2485                                         (void) dup2(fd, 0);
2486                                         (void) close(fd);
2487                                 }
2488                         }
2489                         cron_unlink(tmpfile);
2490                         free(tmpfile);
2491                         (void) fclose(fptr);
2492                 } else if ((fd = open("/dev/null", O_RDONLY)) > 0) {
2493                         (void) dup2(fd, 0);
2494                         (void) close(fd);
2495                 }
2496         }
2497 
2498         /* redirect stdout and stderr for the shell     */
2499         if ((fd = open(rp->outfile, O_WRONLY|O_CREAT|O_EXCL, OUTMODE)) == 1)
2500                 fd = open("/dev/null", O_WRONLY);
2501 
2502         if (fd >= 0 && fd != 1)
2503                 (void) dup2(fd, 1);
2504 
2505         if (fd >= 0 && fd != 2) {
2506                 (void) dup2(fd, 2);
2507                 if (fd != 1)
2508                         (void) close(fd);
2509         }
2510 
2511         if (e->etype == CRONEVENT && e->of.ct.home != NULL) {
2512                 home = (char *)get_obj(e->of.ct.home);
2513         } else {
2514                 home = (e->u)->home;
2515         }
2516         (void) strlcat(homedir, home, sizeof (homedir));
2517         (void) strlcat(logname, (e->u)->name, sizeof (logname));
2518         environ = envinit;
2519         if (chdir(home) == -1) {
2520                 snprintf(bufs.error, sizeof (bufs.error), CANTCDHOME, home);
2521                 mail((e->u)->name, bufs.error,
2522                     e->etype == CRONEVENT ? ERR_CANTEXECCRON :
2523                     ERR_CANTEXECAT);
2524                 exit(1);
2525         }
2526 #ifdef TESTING
2527         exit(1);
2528 #endif
2529         /*
2530          * make sure that all file descriptors EXCEPT 0, 1 and 2
2531          * will be closed.
2532          */
2533         closefrom(3);
2534 
2535         if ((e->u)->uid != 0)
2536                 (void) nice(qp->nice);
2537         if (e->etype == CRONEVENT) {
2538                 if (e->of.ct.tz) {
2539                         (void) putenv((char *)get_obj(e->of.ct.tz));
2540                 }
2541                 if (e->of.ct.shell) {
2542                         char *name;
2543 
2544                         sh = (char *)get_obj(e->of.ct.shell);
2545                         name = strrchr(sh, '/');
2546                         if (name == NULL)
2547                                 name = sh;
2548                         else
2549                                 name++;
2550 
2551                         (void) putenv(sh);
2552                         sh += strlen(ENV_SHELL);
2553                         (void) execl(sh, name, "-c", e->cmd, 0);
2554                 } else {
2555                         (void) execl(SHELL, "sh", "-c", e->cmd, 0);
2556                         sh = SHELL;
2557                 }
2558         } else {                /* type == ATEVENT */
2559                 (void) execl(SHELL, "sh", 0);
2560                 sh = SHELL;
2561         }
2562         snprintf(bufs.error, sizeof (bufs.error), CANTEXECSH, sh);
2563         mail((e->u)->name, bufs.error,
2564             e->etype == CRONEVENT ? ERR_CANTEXECCRON : ERR_CANTEXECAT);
2565         exit(1);
2566         /*NOTREACHED*/
2567 }
2568 
2569 /*
2570  * Main idle loop.
2571  * When timed out to run the job, return 0.
2572  * If for some reasons we need to reschedule jobs, return 1.
2573  */
2574 static int
2575 idle(long t)
2576 {
2577         time_t  now;
2578 
2579         refresh = 0;
2580 
2581         while (t > 0L) {
2582                 if (msg_wait(t) != 0) {
2583                         /* we need to run next job immediately */
2584                         return (0);
2585                 }
2586 
2587                 reap_child();
2588 
2589                 if (refresh) {
2590                         /* We got THAW or REFRESH message  */
2591                         return (1);
2592                 }
2593 
2594                 now = time(NULL);
2595                 if (last_time > now) {
2596                         /* clock has been reset to backward */
2597                         return (1);
2598                 }
2599 
2600                 if (next_event == NULL && !el_empty()) {
2601                         next_event = (struct event *)el_first();
2602                 }
2603 
2604                 if (next_event == NULL)
2605                         t = INFINITY;
2606                 else
2607                         t = (long)next_event->time - now;
2608         }
2609         return (0);
2610 }
2611 
2612 /*
2613  * This used to be in the idle(), but moved to the separate function.
2614  * This called from various place when cron needs to reap the
2615  * child. It includes the situation that cron hit maxrun, and needs
2616  * to reschedule the job.
2617  */
2618 static void
2619 reap_child()
2620 {
2621         pid_t   pid;
2622         int     prc;
2623         struct  runinfo *rp;
2624 
2625         for (;;) {
2626                 pid = waitpid((pid_t)-1, &prc, WNOHANG);
2627                 if (pid <= 0)
2628                         break;
2629 #ifdef DEBUG
2630                 fprintf(stderr,
2631                     "wait returned %x for process %d\n", prc, pid);
2632 #endif
2633                 if ((rp = rinfo_get(pid)) == NULL) {
2634                         if (miscpid_delete(pid) == 0) {
2635                                 /* not found in anywhere */
2636                                 msg(PIDERR, pid);
2637                         }
2638                 } else if (rp->que == ZOMB) {
2639                         (void) unlink(rp->outfile);
2640                         rinfo_free(rp);
2641                 } else {
2642                         cleanup(rp, prc);
2643                 }
2644         }
2645 }
2646 
2647 static void
2648 cleanup(struct runinfo *pr, int rc)
2649 {
2650         int     nextfork = 1;
2651         struct  usr     *p;
2652         struct  stat    buf;
2653 
2654         logit(ECHAR, pr, rc);
2655         --qt[pr->que].nrun;
2656         p = pr->rusr;
2657         if (pr->que != CRONEVENT)
2658                 --p->aruncnt;
2659         else
2660                 --p->cruncnt;
2661 
2662         if (lstat(pr->outfile, &buf) == 0) {
2663                 if (!S_ISLNK(buf.st_mode) &&
2664                     (buf.st_size > 0 || pr->mailwhendone)) {
2665                         /* mail user stdout and stderr */
2666                         for (;;) {
2667                                 if ((pr->pid = fork()) < 0) {
2668                                         /*
2669                                          * if fork fails try forever in doubling
2670                                          * retry times, up to 16 seconds
2671                                          */
2672                                         (void) sleep(nextfork);
2673                                         if (nextfork < 16)
2674                                                 nextfork += nextfork;
2675                                         continue;
2676                                 } else if (pr->pid == 0) {
2677                                         child_sigreset();
2678                                         contract_clear_template();
2679 
2680                                         mail_result(p, pr, buf.st_size);
2681                                         /* NOTREACHED */
2682                                 } else {
2683                                         contract_abandon_latest(pr->pid);
2684                                         pr->que = ZOMB;
2685                                         break;
2686                                 }
2687                         }
2688                 } else {
2689                         (void) unlink(pr->outfile);
2690                         rinfo_free(pr);
2691                 }
2692         } else {
2693                 rinfo_free(pr);
2694         }
2695 
2696         free_if_unused(p);
2697 }
2698 
2699 /*
2700  * Mail stdout and stderr of a job to user. Get uid for real user and become
2701  * that person. We do this so that mail won't come from root since this
2702  * could be a security hole. If failure, quit - don't send mail as root.
2703  */
2704 static void
2705 mail_result(struct usr *p, struct runinfo *pr, size_t filesize)
2706 {
2707         struct  passwd  *ruser_ids;
2708         FILE    *mailpipe;
2709         FILE    *st;
2710         struct utsname  name;
2711         int     nbytes;
2712         char    iobuf[BUFSIZ];
2713         char    *cmd;
2714 
2715         (void) uname(&name);
2716         if ((ruser_ids = getpwnam(p->name)) == NULL)
2717                 exit(0);
2718         (void) setuid(ruser_ids->pw_uid);
2719 
2720         cmd = xmalloc(strlen(MAIL) + strlen(p->name)+2);
2721         (void) sprintf(cmd, "%s %s", MAIL, p->name);
2722         mailpipe = popen(cmd, "w");
2723         free(cmd);
2724         if (mailpipe == NULL)
2725                 exit(127);
2726         (void) fprintf(mailpipe, "To: %s\n", p->name);
2727         if (pr->jobtype == CRONEVENT) {
2728                 (void) fprintf(mailpipe, CRONOUT);
2729                 (void) fprintf(mailpipe, "Your \"cron\" job on %s\n",
2730                     name.nodename);
2731                 if (pr->jobname != NULL) {
2732                         (void) fprintf(mailpipe, "%s\n\n", pr->jobname);
2733                 }
2734         } else {
2735                 (void) fprintf(mailpipe, "Subject: Output from \"at\" job\n\n");
2736                 (void) fprintf(mailpipe, "Your \"at\" job on %s\n",
2737                     name.nodename);
2738                 if (pr->jobname != NULL) {
2739                         (void) fprintf(mailpipe, "\"%s\"\n\n", pr->jobname);
2740                 }
2741         }
2742         /* Tmp. file is fopen'ed w/ "r",  secure open */
2743         if (filesize > 0 &&
2744             (st = fopen(pr->outfile, "r")) != NULL) {
2745                 (void) fprintf(mailpipe,
2746                     "produced the following output:\n\n");
2747                 while ((nbytes = fread(iobuf, sizeof (char), BUFSIZ, st)) != 0)
2748                         (void) fwrite(iobuf, sizeof (char), nbytes, mailpipe);
2749                 (void) fclose(st);
2750         } else {
2751                 (void) fprintf(mailpipe, "completed.\n");
2752         }
2753         (void) pclose(mailpipe);
2754         exit(0);
2755 }
2756 
2757 static int
2758 msg_wait(long tim)
2759 {
2760         struct  message msg;
2761         int     cnt;
2762         time_t  reftime;
2763         fd_set  fds;
2764         struct timespec tout, *toutp;
2765         static int      pending_msg;
2766         static time_t   pending_reftime;
2767 
2768         if (pending_msg) {
2769                 process_msg(&msgbuf, pending_reftime);
2770                 pending_msg = 0;
2771                 return (0);
2772         }
2773 
2774         FD_ZERO(&fds);
2775         FD_SET(msgfd, &fds);
2776 
2777         toutp = NULL;
2778         if (tim != INFINITY) {
2779 #ifdef CRON_MAXSLEEP
2780                 /*
2781                  * CRON_MAXSLEEP can be defined to have cron periodically wake
2782                  * up, so that cron can detect a change of TOD and adjust the
2783                  * sleep time more frequently.
2784                  */
2785                 tim = (tim > CRON_MAXSLEEP) ? CRON_MAXSLEEP : tim;
2786 #endif
2787                 tout.tv_nsec = 0;
2788                 tout.tv_sec = tim;
2789                 toutp = &tout;
2790         }
2791 
2792         cnt = pselect(msgfd + 1, &fds, NULL, NULL, toutp, &defmask);
2793         if (cnt == -1 && errno != EINTR)
2794                 perror("! pselect");
2795 
2796         /* pselect timeout or interrupted */
2797         if (cnt <= 0)
2798                 return (0);
2799 
2800         errno = 0;
2801         if ((cnt = read(msgfd, &msg, sizeof (msg))) != sizeof (msg)) {
2802                 if (cnt != -1 || errno != EAGAIN)
2803                         perror("! read");
2804                 return (0);
2805         }
2806         reftime = time(NULL);
2807         if (next_event != NULL && reftime >= next_event->time) {
2808                 /*
2809                  * we need to run the job before reloading crontab.
2810                  */
2811                 (void) memcpy(&msgbuf, &msg, sizeof (msg));
2812                 pending_msg = 1;
2813                 pending_reftime = reftime;
2814                 return (1);
2815         }
2816         process_msg(&msg, reftime);
2817         return (0);
2818 }
2819 
2820 /*
2821  * process the message supplied via pipe. This will be called either
2822  * immediately after cron read the message from pipe, or idle time
2823  * if the message was pending due to the job execution.
2824  */
2825 static void
2826 process_msg(struct message *pmsg, time_t reftime)
2827 {
2828         if (pmsg->etype == NULL)
2829                 return;
2830 
2831         switch (pmsg->etype) {
2832         case AT:
2833                 if (pmsg->action == DELETE)
2834                         del_atjob(pmsg->fname, pmsg->logname);
2835                 else
2836                         mod_atjob(pmsg->fname, (time_t)0);
2837                 break;
2838         case CRON:
2839                 if (pmsg->action == DELETE)
2840                         del_ctab(pmsg->fname);
2841                 else
2842                         mod_ctab(pmsg->fname, reftime);
2843                 break;
2844         case REFRESH:
2845                 refresh = 1;
2846                 pmsg->etype = 0;
2847                 return;
2848         default:
2849                 msg("message received - bad format");
2850                 break;
2851         }
2852         if (next_event != NULL) {
2853                 if (next_event->etype == CRONEVENT) {
2854                         switch (el_add(next_event, next_event->time,
2855                             (next_event->u)->ctid)) {
2856                         case -1:
2857                                 ignore_msg("process_msg", "cron", next_event);
2858                                 break;
2859                         case -2: /* event time lower than init time */
2860                                 reset_needed = 1;
2861                                 break;
2862                         }
2863                 } else { /* etype == ATEVENT */
2864                         if (el_add(next_event, next_event->time,
2865                             next_event->of.at.eventid) < 0) {
2866                                 ignore_msg("process_msg", "at", next_event);
2867                         }
2868                 }
2869                 next_event = NULL;
2870         }
2871         (void) fflush(stdout);
2872         pmsg->etype = 0;
2873 }
2874 
2875 /*
2876  * Allocate a new or find an existing runinfo structure
2877  */
2878 static struct runinfo *
2879 rinfo_get(pid_t pid)
2880 {
2881         struct runinfo *rp;
2882 
2883         if (pid == 0) {         /* allocate a new entry */
2884                 rp = xcalloc(1, sizeof (struct runinfo));
2885                 rp->next = rthead;   /* link the entry into the list */
2886                 rthead = rp;
2887                 return (rp);
2888         }
2889         /* search the list for an existing entry */
2890         for (rp = rthead; rp != NULL; rp = rp->next) {
2891                 if (rp->pid == pid)
2892                         break;
2893         }
2894         return (rp);
2895 }
2896 
2897 /*
2898  * Free a runinfo structure and its associated memory
2899  */
2900 static void
2901 rinfo_free(struct runinfo *entry)
2902 {
2903         struct runinfo **rpp;
2904         struct runinfo *rp;
2905 
2906 #ifdef DEBUG
2907         (void) fprintf(stderr, "freeing job %s\n", entry->jobname);
2908 #endif
2909         for (rpp = &rthead; (rp = *rpp) != NULL; rpp = &rp->next) {
2910                 if (rp == entry) {
2911                         *rpp = rp->next;     /* unlink the entry */
2912                         free(rp->outfile);
2913                         free(rp->jobname);
2914                         free(rp);
2915                         break;
2916                 }
2917         }
2918 }
2919 
2920 /* ARGSUSED */
2921 static void
2922 thaw_handler(int sig)
2923 {
2924         refresh = 1;
2925 }
2926 
2927 
2928 /* ARGSUSED */
2929 static void
2930 cronend(int sig)
2931 {
2932         crabort("SIGTERM", REMOVE_FIFO);
2933 }
2934 
2935 /*ARGSUSED*/
2936 static void
2937 child_handler(int sig)
2938 {
2939         ;
2940 }
2941 
2942 static void
2943 child_sigreset(void)
2944 {
2945         (void) signal(SIGCLD, SIG_DFL);
2946         (void) sigprocmask(SIG_SETMASK, &defmask, NULL);
2947 }
2948 
2949 /*
2950  * crabort() - handle exits out of cron
2951  */
2952 static void
2953 crabort(char *mssg, int action)
2954 {
2955         int     c;
2956 
2957         if (action & REMOVE_FIFO) {
2958                 /* FIFO vanishes when cron finishes */
2959                 if (unlink(FIFO) < 0)
2960                         perror("cron could not unlink FIFO");
2961         }
2962 
2963         if (action & CONSOLE_MSG) {
2964                 /* write error msg to console */
2965                 if ((c = open(CONSOLE, O_WRONLY)) >= 0) {
2966                         (void) write(c, "cron aborted: ", 14);
2967                         (void) write(c, mssg, strlen(mssg));
2968                         (void) write(c, "\n", 1);
2969                         (void) close(c);
2970                 }
2971         }
2972 
2973         /* always log the message */
2974         msg(mssg);
2975         msg("******* CRON ABORTED ********");
2976         exit(1);
2977 }
2978 
2979 /*
2980  * msg() - time-stamped error reporting function
2981  */
2982 /*PRINTFLIKE1*/
2983 static void
2984 msg(char *fmt, ...)
2985 {
2986         va_list args;
2987         time_t  t;
2988 
2989         t = time(NULL);
2990 
2991         (void) fflush(stdout);
2992 
2993         (void) fprintf(stderr, "! ");
2994 
2995         va_start(args, fmt);
2996         (void) vfprintf(stderr, fmt, args);
2997         va_end(args);
2998 
2999         (void) strftime(timebuf, sizeof (timebuf), FORMAT, localtime(&t));
3000         (void) fprintf(stderr, " %s\n", timebuf);
3001 
3002         (void) fflush(stderr);
3003 }
3004 
3005 static void
3006 ignore_msg(char *func_name, char *job_type, struct event *event)
3007 {
3008         msg("%s: ignoring %s job (user: %s, cmd: %s, time: %ld)",
3009             func_name, job_type,
3010             event->u->name ? event->u->name : "unknown",
3011             event->cmd ? event->cmd : "unknown",
3012             event->time);
3013 }
3014 
3015 static void
3016 logit(int cc, struct runinfo *rp, int rc)
3017 {
3018         time_t t;
3019         int    ret;
3020 
3021         if (!log)
3022                 return;
3023 
3024         t = time(NULL);
3025         if (cc == BCHAR)
3026                 (void) printf("%c  CMD: %s\n", cc, next_event->cmd);
3027         (void) strftime(timebuf, sizeof (timebuf), FORMAT, localtime(&t));
3028         (void) printf("%c  %s %u %c %s",
3029             cc, (rp->rusr)->name, rp->pid, QUE(rp->que), timebuf);
3030         if ((ret = TSTAT(rc)) != 0)
3031                 (void) printf(" ts=%d", ret);
3032         if ((ret = RCODE(rc)) != 0)
3033                 (void) printf(" rc=%d", ret);
3034         (void) putchar('\n');
3035         (void) fflush(stdout);
3036 }
3037 
3038 static void
3039 resched(int delay)
3040 {
3041         time_t  nt;
3042 
3043         /* run job at a later time */
3044         nt = next_event->time + delay;
3045         if (next_event->etype == CRONEVENT) {
3046                 next_event->time = next_time(next_event, (time_t)0);
3047                 if (nt < next_event->time)
3048                         next_event->time = nt;
3049                 switch (el_add(next_event, next_event->time,
3050                     (next_event->u)->ctid)) {
3051                 case -1:
3052                         ignore_msg("resched", "cron", next_event);
3053                         break;
3054                 case -2: /* event time lower than init time */
3055                         reset_needed = 1;
3056                         break;
3057                 }
3058                 delayed = 1;
3059                 msg("rescheduling a cron job");
3060                 return;
3061         }
3062         add_atevent(next_event->u, next_event->cmd, nt, next_event->etype);
3063         msg("rescheduling at job");
3064 }
3065 
3066 static void
3067 quedefs(int action)
3068 {
3069         int     i;
3070         int     j;
3071         char    qbuf[QBUFSIZ];
3072         FILE    *fd;
3073 
3074         /* set up default queue definitions */
3075         for (i = 0; i < NQUEUE; i++) {
3076                 qt[i].njob = qd.njob;
3077                 qt[i].nice = qd.nice;
3078                 qt[i].nwait = qd.nwait;
3079         }
3080         if (action == DEFAULT)
3081                 return;
3082         if ((fd = fopen(QUEDEFS, "r")) == NULL) {
3083                 msg("cannot open quedefs file");
3084                 msg("using default queue definitions");
3085                 return;
3086         }
3087         while (fgets(qbuf, QBUFSIZ, fd) != NULL) {
3088                 if ((j = qbuf[0]-'a') < 0 || j >= NQUEUE || qbuf[1] != '.')
3089                         continue;
3090                 parsqdef(&qbuf[2]);
3091                 qt[j].njob = qq.njob;
3092                 qt[j].nice = qq.nice;
3093                 qt[j].nwait = qq.nwait;
3094         }
3095         (void) fclose(fd);
3096 }
3097 
3098 static void
3099 parsqdef(char *name)
3100 {
3101         int i;
3102 
3103         qq = qd;
3104         while (*name) {
3105                 i = 0;
3106                 while (isdigit(*name)) {
3107                         i *= 10;
3108                         i += *name++ - '0';
3109                 }
3110                 switch (*name++) {
3111                 case JOBF:
3112                         qq.njob = i;
3113                         break;
3114                 case NICEF:
3115                         qq.nice = i;
3116                         break;
3117                 case WAITF:
3118                         qq.nwait = i;
3119                         break;
3120                 }
3121         }
3122 }
3123 
3124 /*
3125  * defaults - read defaults from /etc/default/cron
3126  */
3127 static void
3128 defaults()
3129 {
3130         int  flags;
3131         char *deflog;
3132         char *hz, *tz;
3133 
3134         /*
3135          * get HZ value for environment
3136          */
3137         if ((hz = getenv("HZ")) == (char *)NULL)
3138                 (void) sprintf(hzname, "HZ=%d", HZ);
3139         else
3140                 (void) snprintf(hzname, sizeof (hzname), "HZ=%s", hz);
3141         /*
3142          * get TZ value for environment
3143          */
3144         (void) snprintf(tzone, sizeof (tzone), "TZ=%s",
3145             ((tz = getenv("TZ")) != NULL) ? tz : DEFTZ);
3146 
3147         if (defopen(DEFFILE) == 0) {
3148                 /* ignore case */
3149                 flags = defcntl(DC_GETFLAGS, 0);
3150                 TURNOFF(flags, DC_CASE);
3151                 (void) defcntl(DC_SETFLAGS, flags);
3152 
3153                 if (((deflog = defread("CRONLOG=")) == NULL) ||
3154                     (*deflog == 'N') || (*deflog == 'n'))
3155                         log = 0;
3156                 else
3157                         log = 1;
3158                 /* fix for 1087611 - allow paths to be set in defaults file */
3159                 if ((Def_path = defread("PATH=")) != NULL) {
3160                         (void) strlcat(path, Def_path, LINE_MAX);
3161                 } else {
3162                         (void) strlcpy(path, NONROOTPATH, LINE_MAX);
3163                 }
3164                 if ((Def_supath = defread("SUPATH=")) != NULL) {
3165                         (void) strlcat(supath, Def_supath, LINE_MAX);
3166                 } else {
3167                         (void) strlcpy(supath, ROOTPATH, LINE_MAX);
3168                 }
3169                 (void) defopen(NULL);
3170         }
3171 }
3172 
3173 /*
3174  * Determine if a user entry for a job is still ok.  The method used here
3175  * is a lot (about 75x) faster than using setgrent() / getgrent()
3176  * endgrent().  It should be safe because we use the sysconf to determine
3177  * the max, and it tolerates the max being 0.
3178  */
3179 
3180 static int
3181 verify_user_cred(struct usr *u)
3182 {
3183         struct passwd *pw;
3184         size_t numUsrGrps = 0;
3185         size_t numOrigGrps = 0;
3186         size_t i;
3187         int retval;
3188 
3189         /*
3190          * Maximum number of groups a user may be in concurrently.  This
3191          * is a value which we obtain at runtime through a sysconf()
3192          * call.
3193          */
3194 
3195         static size_t nGroupsMax = (size_t)-1;
3196 
3197         /*
3198          * Arrays for cron user's group list, constructed at startup to
3199          * be nGroupsMax elements long, used for verifying user
3200          * credentials prior to execution.
3201          */
3202 
3203         static gid_t *UsrGrps;
3204         static gid_t *OrigGrps;
3205 
3206         if ((pw = getpwnam(u->name)) == NULL)
3207                 return (VUC_BADUSER);
3208         if (u->home != NULL) {
3209                 if (strcmp(u->home, pw->pw_dir) != 0) {
3210                         free(u->home);
3211                         u->home = xmalloc(strlen(pw->pw_dir) + 1);
3212                         (void) strcpy(u->home, pw->pw_dir);
3213                 }
3214         } else {
3215                 u->home = xmalloc(strlen(pw->pw_dir) + 1);
3216                 (void) strcpy(u->home, pw->pw_dir);
3217         }
3218         if (u->uid != pw->pw_uid)
3219                 u->uid = pw->pw_uid;
3220         if (u->gid != pw->pw_gid)
3221                 u->gid  = pw->pw_gid;
3222 
3223         /*
3224          * Create the group id lists needed for job credential
3225          * verification.
3226          */
3227 
3228         if (nGroupsMax == (size_t)-1) {
3229                 if ((nGroupsMax = sysconf(_SC_NGROUPS_MAX)) > 0) {
3230                         UsrGrps = xcalloc(nGroupsMax, sizeof (gid_t));
3231                         OrigGrps = xcalloc(nGroupsMax, sizeof (gid_t));
3232                 }
3233 
3234 #ifdef DEBUG
3235                 (void) fprintf(stderr, "nGroupsMax = %ld\n", nGroupsMax);
3236 #endif
3237         }
3238 
3239 #ifdef DEBUG
3240         (void) fprintf(stderr, "verify_user_cred (%s-%d)\n", pw->pw_name,
3241             pw->pw_uid);
3242         (void) fprintf(stderr, "verify_user_cred: pw->pw_gid = %d, "
3243             "u->gid = %d\n", pw->pw_gid, u->gid);
3244 #endif
3245 
3246         retval = (u->gid == pw->pw_gid) ? VUC_OK : VUC_NOTINGROUP;
3247 
3248         if (nGroupsMax > 0) {
3249                 numOrigGrps = getgroups(nGroupsMax, OrigGrps);
3250 
3251                 (void) initgroups(pw->pw_name, pw->pw_gid);
3252                 numUsrGrps = getgroups(nGroupsMax, UsrGrps);
3253 
3254                 for (i = 0; i < numUsrGrps; i++) {
3255                         if (UsrGrps[i] == u->gid) {
3256                                 retval = VUC_OK;
3257                                 break;
3258                         }
3259                 }
3260 
3261                 if (OrigGrps) {
3262                         (void) setgroups(numOrigGrps, OrigGrps);
3263                 }
3264         }
3265 
3266 #ifdef DEBUG
3267         (void) fprintf(stderr, "verify_user_cred: VUC = %d\n", retval);
3268 #endif
3269 
3270         return (retval);
3271 }
3272 
3273 static int
3274 set_user_cred(const struct usr *u, struct project *pproj)
3275 {
3276         static char *progname = "cron";
3277         int r = 0, rval = 0;
3278 
3279         if ((r = pam_start(progname, u->name, &pam_conv, &pamh))
3280             != PAM_SUCCESS) {
3281 #ifdef DEBUG
3282                 msg("pam_start returns %d\n", r);
3283 #endif
3284                 rval = VUC_BADUSER;
3285                 goto set_eser_cred_exit;
3286         }
3287 
3288         r = pam_acct_mgmt(pamh, 0);
3289 #ifdef DEBUG
3290         msg("pam_acc_mgmt returns %d\n", r);
3291 #endif
3292         if (r == PAM_ACCT_EXPIRED) {
3293                 rval = VUC_EXPIRED;
3294                 goto set_eser_cred_exit;
3295         }
3296         if (r == PAM_NEW_AUTHTOK_REQD) {
3297                 rval = VUC_NEW_AUTH;
3298                 goto set_eser_cred_exit;
3299         }
3300         if (r != PAM_SUCCESS) {
3301                 rval = VUC_BADUSER;
3302                 goto set_eser_cred_exit;
3303         }
3304 
3305         if (pproj != NULL) {
3306                 size_t sz = sizeof (PROJECT) + strlen(pproj->pj_name);
3307                 char *buf = alloca(sz);
3308 
3309                 (void) snprintf(buf, sz, PROJECT "%s", pproj->pj_name);
3310                 (void) pam_set_item(pamh, PAM_RESOURCE, buf);
3311         }
3312 
3313         r = pam_setcred(pamh, PAM_ESTABLISH_CRED);
3314         if (r != PAM_SUCCESS)
3315                 rval = VUC_BADUSER;
3316 
3317 set_eser_cred_exit:
3318         (void) pam_end(pamh, r);
3319         return (rval);
3320 }
3321 
3322 static void
3323 clean_out_user(struct usr *u)
3324 {
3325         if (next_event->u == u) {
3326                 next_event = NULL;
3327         }
3328 
3329         clean_out_ctab(u);
3330         clean_out_atjobs(u);
3331         free_if_unused(u);
3332 }
3333 
3334 static void
3335 clean_out_atjobs(struct usr *u)
3336 {
3337         struct event *ev, *pv;
3338 
3339         for (pv = NULL, ev = u->atevents;
3340             ev != NULL;
3341             pv = ev, ev = ev->link, free(pv)) {
3342                 el_remove(ev->of.at.eventid, 1);
3343                 if (cwd == AT)
3344                         cron_unlink(ev->cmd);
3345                 else {
3346                         char buf[PATH_MAX];
3347                         if (strlen(ATDIR) + strlen(ev->cmd) + 2
3348                             < PATH_MAX) {
3349                                 (void) sprintf(buf, "%s/%s", ATDIR, ev->cmd);
3350                                 cron_unlink(buf);
3351                         }
3352                 }
3353                 free(ev->cmd);
3354         }
3355 
3356         u->atevents = NULL;
3357 }
3358 
3359 static void
3360 clean_out_ctab(struct usr *u)
3361 {
3362         rm_ctevents(u);
3363         el_remove(u->ctid, 0);
3364         u->ctid = 0;
3365         u->ctexists = 0;
3366 }
3367 
3368 static void
3369 cron_unlink(char *name)
3370 {
3371         int r;
3372 
3373         r = unlink(name);
3374         if (r == 0 || (r == -1 && errno == ENOENT)) {
3375                 (void) audit_cron_delete_anc_file(name, NULL);
3376         }
3377 }
3378 
3379 static void
3380 create_anc_ctab(struct event *e)
3381 {
3382         if (audit_cron_create_anc_file(e->u->name,
3383             (cwd == CRON) ? NULL:CRONDIR,
3384             e->u->name, e->u->uid) == -1) {
3385                 process_anc_files(CRON_ANC_DELETE);
3386                 crabort("cannot create ancillary files for crontabs",
3387                     REMOVE_FIFO|CONSOLE_MSG);
3388         }
3389 }
3390 
3391 static void
3392 delete_anc_ctab(struct event *e)
3393 {
3394         (void) audit_cron_delete_anc_file(e->u->name,
3395             (cwd == CRON) ? NULL:CRONDIR);
3396 }
3397 
3398 static void
3399 create_anc_atjob(struct event *e)
3400 {
3401         if (!e->of.at.exists)
3402                 return;
3403 
3404         if (audit_cron_create_anc_file(e->cmd,
3405             (cwd == AT) ? NULL:ATDIR,
3406             e->u->name, e->u->uid) == -1) {
3407                 process_anc_files(CRON_ANC_DELETE);
3408                 crabort("cannot create ancillary files for atjobs",
3409                     REMOVE_FIFO|CONSOLE_MSG);
3410         }
3411 }
3412 
3413 static void
3414 delete_anc_atjob(struct event *e)
3415 {
3416         if (!e->of.at.exists)
3417                 return;
3418 
3419         (void) audit_cron_delete_anc_file(e->cmd,
3420             (cwd == AT) ? NULL:ATDIR);
3421 }
3422 
3423 
3424 static void
3425 process_anc_files(int del)
3426 {
3427         struct usr      *u = uhead;
3428         struct event    *e;
3429 
3430         if (!audit_cron_mode())
3431                 return;
3432 
3433         for (;;) {
3434                 if (u->ctexists && u->ctevents != NULL) {
3435                         e = u->ctevents;
3436                         for (;;) {
3437                                 if (del)
3438                                         delete_anc_ctab(e);
3439                                 else
3440                                         create_anc_ctab(e);
3441                                 if ((e = e->link) == NULL)
3442                                         break;
3443                         }
3444                 }
3445 
3446                 if (u->atevents != NULL) {
3447                         e = u->atevents;
3448                         for (;;) {
3449                                 if (del)
3450                                         delete_anc_atjob(e);
3451                                 else
3452                                         create_anc_atjob(e);
3453                                 if ((e = e->link) == NULL)
3454                                         break;
3455                         }
3456                 }
3457 
3458                 if ((u = u->nextusr)  == NULL)
3459                         break;
3460         }
3461 }
3462 
3463 /*ARGSUSED*/
3464 static int
3465 cron_conv(int num_msg, struct pam_message **msgs,
3466     struct pam_response **response, void *appdata_ptr)
3467 {
3468         struct pam_message      **m = msgs;
3469         int i;
3470 
3471         for (i = 0; i < num_msg; i++) {
3472                 switch (m[i]->msg_style) {
3473                 case PAM_ERROR_MSG:
3474                 case PAM_TEXT_INFO:
3475                         if (m[i]->msg != NULL) {
3476                                 (void) msg("%s\n", m[i]->msg);
3477                         }
3478                         break;
3479 
3480                 default:
3481                         break;
3482                 }
3483         }
3484         return (0);
3485 }
3486 
3487 /*
3488  * Cron creates process for other than job. Mail process is the
3489  * one which rinfo does not cover. Therefore, miscpid will keep
3490  * track of the pids executed from cron. Otherwise, we will see
3491  * "unexpected pid returned.." messages appear in the log file.
3492  */
3493 static void
3494 miscpid_insert(pid_t pid)
3495 {
3496         struct miscpid *mp;
3497 
3498         mp = xmalloc(sizeof (*mp));
3499         mp->pid = pid;
3500         mp->next = miscpid_head;
3501         miscpid_head = mp;
3502 }
3503 
3504 static int
3505 miscpid_delete(pid_t pid)
3506 {
3507         struct miscpid *mp, *omp;
3508         int found = 0;
3509 
3510         omp = NULL;
3511         for (mp = miscpid_head; mp != NULL; mp = mp->next) {
3512                 if (mp->pid == pid) {
3513                         found = 1;
3514                         break;
3515                 }
3516                 omp = mp;
3517         }
3518         if (found) {
3519                 if (omp != NULL)
3520                         omp->next = mp->next;
3521                 else
3522                         miscpid_head = NULL;
3523                 free(mp);
3524         }
3525         return (found);
3526 }
3527 
3528 /*
3529  * Establish contract terms such that all children are in abandoned
3530  * process contracts.
3531  */
3532 static void
3533 contract_set_template(void)
3534 {
3535         int fd;
3536 
3537         if ((fd = open64(CTFS_ROOT "/process/template", O_RDWR)) < 0)
3538                 crabort("cannot open process contract template",
3539                     REMOVE_FIFO | CONSOLE_MSG);
3540 
3541         if (ct_pr_tmpl_set_param(fd, 0) ||
3542             ct_tmpl_set_informative(fd, 0) ||
3543             ct_pr_tmpl_set_fatal(fd, CT_PR_EV_HWERR))
3544                 crabort("cannot establish contract template terms",
3545                     REMOVE_FIFO | CONSOLE_MSG);
3546 
3547         if (ct_tmpl_activate(fd))
3548                 crabort("cannot activate contract template",
3549                     REMOVE_FIFO | CONSOLE_MSG);
3550 
3551         (void) close(fd);
3552 }
3553 
3554 /*
3555  * Clear active process contract template.
3556  */
3557 static void
3558 contract_clear_template(void)
3559 {
3560         int fd;
3561 
3562         if ((fd = open64(CTFS_ROOT "/process/template", O_RDWR)) < 0)
3563                 crabort("cannot open process contract template",
3564                     REMOVE_FIFO | CONSOLE_MSG);
3565 
3566         if (ct_tmpl_clear(fd))
3567                 crabort("cannot clear contract template",
3568                     REMOVE_FIFO | CONSOLE_MSG);
3569 
3570         (void) close(fd);
3571 }
3572 
3573 /*
3574  * Abandon latest process contract unconditionally.  If we have leaked [some
3575  * critical amount], exit such that the kernel reaps our contracts.
3576  */
3577 static void
3578 contract_abandon_latest(pid_t pid)
3579 {
3580         int r;
3581         ctid_t id;
3582         static uint_t cts_lost;
3583 
3584         if (cts_lost > MAX_LOST_CONTRACTS)
3585                 crabort("repeated failure to abandon contracts",
3586                     REMOVE_FIFO | CONSOLE_MSG);
3587 
3588         if (r = contract_latest(&id)) {
3589                 msg("could not obtain latest contract for "
3590                     "PID %ld: %s", pid, strerror(r));
3591                 cts_lost++;
3592                 return;
3593         }
3594 
3595         if (r = contract_abandon_id(id)) {
3596                 msg("could not abandon latest contract %ld: %s", id,
3597                     strerror(r));
3598                 cts_lost++;
3599                 return;
3600         }
3601 }
3602 
3603 static struct shared *
3604 create_shared(void *obj, void * (*obj_alloc)(void *obj),
3605         void (*obj_free)(void *))
3606 {
3607         struct shared *out;
3608 
3609         if ((out = xmalloc(sizeof (struct shared))) == NULL) {
3610                 return (NULL);
3611         }
3612         if ((out->obj = obj_alloc(obj)) == NULL) {
3613                 free(out);
3614                 return (NULL);
3615         }
3616         out->count = 1;
3617         out->free = obj_free;
3618 
3619         return (out);
3620 }
3621 
3622 static struct shared *
3623 create_shared_str(char *str)
3624 {
3625         return (create_shared(str, (void *(*)(void *))strdup, free));
3626 }
3627 
3628 static struct shared *
3629 dup_shared(struct shared *obj)
3630 {
3631         if (obj != NULL) {
3632                 obj->count++;
3633         }
3634         return (obj);
3635 }
3636 
3637 static void
3638 rel_shared(struct shared *obj)
3639 {
3640         if (obj && (--obj->count) == 0) {
3641                 obj->free(obj->obj);
3642                 free(obj);
3643         }
3644 }
3645 
3646 static void *
3647 get_obj(struct shared *obj)
3648 {
3649         return (obj->obj);
3650 }