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 }