Annotation of 43BSD/etc/shutdown.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1983,1986 Regents of the University of California.
        !             3:  * All rights reserved.  The Berkeley software License Agreement
        !             4:  * specifies the terms and conditions for redistribution.
        !             5:  */
        !             6: 
        !             7: #ifndef lint
        !             8: char copyright[] =
        !             9: "@(#) Copyright (c) 1983,1986 Regents of the University of California.\n\
        !            10:  All rights reserved.\n";
        !            11: #endif not lint
        !            12: 
        !            13: #ifndef lint
        !            14: static char sccsid[] = "@(#)shutdown.c 5.6 (Berkeley) 5/26/86";
        !            15: #endif not lint
        !            16: 
        !            17: #include <stdio.h>
        !            18: #include <ctype.h>
        !            19: #include <signal.h>
        !            20: #include <setjmp.h>
        !            21: #include <utmp.h>
        !            22: #include <pwd.h>
        !            23: #include <sys/time.h>
        !            24: #include <sys/resource.h>
        !            25: #include <sys/param.h>
        !            26: #include <sys/syslog.h>
        !            27: 
        !            28: /*
        !            29:  *     /etc/shutdown when [messages]
        !            30:  *
        !            31:  *     allow super users to tell users and remind users
        !            32:  *     of iminent shutdown of unix
        !            33:  *     and shut it down automatically
        !            34:  *     and even reboot or halt the machine if they desire
        !            35:  */
        !            36: 
        !            37: #define        REBOOT  "/etc/reboot"
        !            38: #define        HALT    "/etc/halt"
        !            39: #define MAXINTS 20
        !            40: #define        HOURS   *3600
        !            41: #define MINUTES        *60
        !            42: #define SECONDS
        !            43: #define NLOG           600             /* no of bytes possible for message */
        !            44: #define        NOLOGTIME       5 MINUTES
        !            45: #define IGNOREUSER     "sleeper"
        !            46: 
        !            47: char   hostname[MAXHOSTNAMELEN];
        !            48: 
        !            49: int    timeout();
        !            50: time_t getsdt();
        !            51: 
        !            52: extern char *ctime();
        !            53: extern struct tm *localtime();
        !            54: extern long time();
        !            55: 
        !            56: extern char *strcpy();
        !            57: extern char *strncat();
        !            58: extern off_t lseek();
        !            59: 
        !            60: struct utmp utmp;
        !            61: int    sint;
        !            62: int    stogo;
        !            63: char   tpath[] =       "/dev/";
        !            64: int    nlflag = 1;             /* nolog yet to be done */
        !            65: int    killflg = 1;
        !            66: int    doreboot = 0;
        !            67: int    halt = 0;
        !            68: int     fast = 0;
        !            69: char    *nosync = NULL;
        !            70: char    nosyncflag[] = "-n";
        !            71: char   term[sizeof tpath + sizeof utmp.ut_line];
        !            72: char   tbuf[BUFSIZ];
        !            73: char   nolog1[] = "\n\nNO LOGINS: System going down at %5.5s\n\n";
        !            74: char   nolog2[NLOG+1];
        !            75: #ifdef DEBUG
        !            76: char   nologin[] = "nologin";
        !            77: char    fastboot[] = "fastboot";
        !            78: #else
        !            79: char   nologin[] = "/etc/nologin";
        !            80: char   fastboot[] = "/fastboot";
        !            81: #endif
        !            82: time_t nowtime;
        !            83: jmp_buf        alarmbuf;
        !            84: 
        !            85: struct interval {
        !            86:        int stogo;
        !            87:        int sint;
        !            88: } interval[] = {
        !            89:        4 HOURS,        1 HOURS,
        !            90:        2 HOURS,        30 MINUTES,
        !            91:        1 HOURS,        15 MINUTES,
        !            92:        30 MINUTES,     10 MINUTES,
        !            93:        15 MINUTES,     5 MINUTES,
        !            94:        10 MINUTES,     5 MINUTES,
        !            95:        5 MINUTES,      3 MINUTES,
        !            96:        2 MINUTES,      1 MINUTES,
        !            97:        1 MINUTES,      30 SECONDS,
        !            98:        0 SECONDS,      0 SECONDS
        !            99: };
        !           100: 
        !           101: char *shutter, *getlogin();
        !           102: 
        !           103: main(argc,argv)
        !           104:        int argc;
        !           105:        char **argv;
        !           106: {
        !           107:        register i, ufd;
        !           108:        register char *f;
        !           109:        char *ts;
        !           110:        time_t sdt;
        !           111:        int h, m;
        !           112:        int first;
        !           113:        FILE *termf;
        !           114:        struct passwd *pw, *getpwuid();
        !           115:        extern char *strcat();
        !           116:        extern uid_t geteuid();
        !           117: 
        !           118:        shutter = getlogin();
        !           119:        if (shutter == 0 && (pw = getpwuid(getuid())))
        !           120:                shutter = pw->pw_name;
        !           121:        if (shutter == 0)
        !           122:                shutter = "???";
        !           123:        (void) gethostname(hostname, sizeof (hostname));
        !           124:        openlog("shutdown", 0, LOG_AUTH);
        !           125:        argc--, argv++;
        !           126:        while (argc > 0 && (f = argv[0], *f++ == '-')) {
        !           127:                while (i = *f++) switch (i) {
        !           128:                case 'k':
        !           129:                        killflg = 0;
        !           130:                        continue;
        !           131:                case 'n':
        !           132:                        nosync = nosyncflag;
        !           133:                        continue;
        !           134:                case 'f':
        !           135:                        fast = 1;
        !           136:                        continue;
        !           137:                case 'r':
        !           138:                        doreboot = 1;
        !           139:                        continue;
        !           140:                case 'h':
        !           141:                        halt = 1;
        !           142:                        continue;
        !           143:                default:
        !           144:                        fprintf(stderr, "shutdown: '%c' - unknown flag\n", i);
        !           145:                        exit(1);
        !           146:                }
        !           147:                argc--, argv++;
        !           148:        }
        !           149:        if (argc < 1) {
        !           150:                /* argv[0] is not available after the argument handling. */
        !           151:                printf("Usage: shutdown [ -krhfn ] shutdowntime [ message ]\n");
        !           152:                finish();
        !           153:        }
        !           154:        if (fast && (nosync == nosyncflag)) {
        !           155:                printf ("shutdown: Incompatible switches 'fast' & 'nosync'\n");
        !           156:                finish();
        !           157:        }
        !           158:        if (geteuid()) {
        !           159:                fprintf(stderr, "NOT super-user\n");
        !           160:                finish();
        !           161:        }
        !           162:        nowtime = time((long *)0);
        !           163:        sdt = getsdt(argv[0]);
        !           164:        argc--, argv++;
        !           165:        nolog2[0] = '\0';
        !           166:        while (argc-- > 0) {
        !           167:                (void) strcat(nolog2, " ");
        !           168:                (void) strcat(nolog2, *argv++);
        !           169:        }
        !           170:        m = ((stogo = sdt - nowtime) + 30)/60;
        !           171:        h = m/60; 
        !           172:        m %= 60;
        !           173:        ts = ctime(&sdt);
        !           174:        printf("Shutdown at %5.5s (in ", ts+11);
        !           175:        if (h > 0)
        !           176:                printf("%d hour%s ", h, h != 1 ? "s" : "");
        !           177:        printf("%d minute%s) ", m, m != 1 ? "s" : "");
        !           178: #ifndef DEBUG
        !           179:        (void) signal(SIGHUP, SIG_IGN);
        !           180:        (void) signal(SIGQUIT, SIG_IGN);
        !           181:        (void) signal(SIGINT, SIG_IGN);
        !           182: #endif
        !           183:        (void) signal(SIGTTOU, SIG_IGN);
        !           184:        (void) signal(SIGTERM, finish);
        !           185:        (void) signal(SIGALRM, timeout);
        !           186:        (void) setpriority(PRIO_PROCESS, 0, PRIO_MIN);
        !           187:        (void) fflush(stdout);
        !           188: #ifndef DEBUG
        !           189:        if (i = fork()) {
        !           190:                printf("[pid %d]\n", i);
        !           191:                exit(0);
        !           192:        }
        !           193: #else
        !           194:        (void) putc('\n', stdout);
        !           195: #endif
        !           196:        sint = 1 HOURS;
        !           197:        f = "";
        !           198:        ufd = open("/etc/utmp",0);
        !           199:        if (ufd < 0) {
        !           200:                perror("shutdown: /etc/utmp");
        !           201:                exit(1);
        !           202:        }
        !           203:        first = 1;
        !           204:        for (;;) {
        !           205:                for (i = 0; stogo <= interval[i].stogo && interval[i].sint; i++)
        !           206:                        sint = interval[i].sint;
        !           207:                if (stogo > 0 && (stogo-sint) < interval[i].stogo)
        !           208:                        sint = stogo - interval[i].stogo;
        !           209:                if (stogo <= NOLOGTIME && nlflag) {
        !           210:                        nlflag = 0;
        !           211:                        nolog(sdt);
        !           212:                }
        !           213:                if (sint >= stogo || sint == 0)
        !           214:                        f = "FINAL ";
        !           215:                nowtime = time((long *)0);
        !           216:                (void) lseek(ufd, 0L, 0);
        !           217:                while (read(ufd,(char *)&utmp,sizeof utmp)==sizeof utmp)
        !           218:                if (utmp.ut_name[0] &&
        !           219:                    strncmp(utmp.ut_name, IGNOREUSER, sizeof(utmp.ut_name))) {
        !           220:                        if (setjmp(alarmbuf))
        !           221:                                continue;
        !           222:                        (void) strcpy(term, tpath);
        !           223:                        (void) strncat(term, utmp.ut_line, sizeof utmp.ut_line);
        !           224:                        (void) alarm(3);
        !           225: #ifdef DEBUG
        !           226:                        if ((termf = stdout) != NULL)
        !           227: #else
        !           228:                        if ((termf = fopen(term, "w")) != NULL)
        !           229: #endif
        !           230:                        {
        !           231:                                (void) alarm(0);
        !           232:                                setbuf(termf, tbuf);
        !           233:                                fprintf(termf, "\n\r\n");
        !           234:                                warn(termf, sdt, nowtime, f);
        !           235:                                if (first || sdt - nowtime > 1 MINUTES) {
        !           236:                                        if (*nolog2)
        !           237:                                                fprintf(termf, "\t...%s", nolog2);
        !           238:                                }
        !           239:                                (void) fputc('\r', termf);
        !           240:                                (void) fputc('\n', termf);
        !           241:                                (void) alarm(5);
        !           242: #ifdef DEBUG
        !           243:                                (void) fflush(termf);
        !           244: #else
        !           245:                                (void) fclose(termf);
        !           246: #endif
        !           247:                                (void) alarm(0);
        !           248:                        }
        !           249:                }
        !           250:                if (stogo <= 0) {
        !           251:                        printf("\n\007\007System shutdown time has arrived\007\007\n");
        !           252:                        syslog(LOG_CRIT, "%s by %s: %s",
        !           253:                            doreboot ? "reboot" : halt ? "halt" : "shutdown",
        !           254:                            shutter, nolog2);
        !           255:                        sleep(2);
        !           256:                        (void) unlink(nologin);
        !           257:                        if (!killflg) {
        !           258:                                printf("but you'll have to do it yourself\n");
        !           259:                                finish();
        !           260:                        }
        !           261:                        if (fast)
        !           262:                                doitfast();
        !           263: #ifndef DEBUG
        !           264:                        if (doreboot)
        !           265:                                execle(REBOOT, "reboot", "-l", nosync, 0, 0);
        !           266:                        if (halt)
        !           267:                                execle(HALT, "halt", "-l", nosync, 0, 0);
        !           268:                        (void) kill(1, SIGTERM);        /* to single user */
        !           269: #else
        !           270:                        if (doreboot)
        !           271:                                printf("REBOOT");
        !           272:                        if (halt)
        !           273:                                printf(" HALT");
        !           274:                        if (fast)
        !           275:                                printf(" -l %s (without fsck's)\n", nosync);
        !           276:                        else
        !           277:                                printf(" -l %s\n", nosync);
        !           278:                        else
        !           279:                                printf("kill -HUP 1\n");
        !           280: 
        !           281: #endif
        !           282:                        finish();
        !           283:                }
        !           284:                stogo = sdt - time((long *) 0);
        !           285:                if (stogo > 0 && sint > 0)
        !           286:                        sleep((unsigned)(sint<stogo ? sint : stogo));
        !           287:                stogo -= sint;
        !           288:                first = 0;
        !           289:        }
        !           290: }
        !           291: 
        !           292: time_t
        !           293: getsdt(s)
        !           294:        register char *s;
        !           295: {
        !           296:        time_t t, t1, tim;
        !           297:        register char c;
        !           298:        struct tm *lt;
        !           299: 
        !           300:        if (strcmp(s, "now") == 0)
        !           301:                return(nowtime);
        !           302:        if (*s == '+') {
        !           303:                ++s; 
        !           304:                t = 0;
        !           305:                for (;;) {
        !           306:                        c = *s++;
        !           307:                        if (!isdigit(c))
        !           308:                                break;
        !           309:                        t = t * 10 + c - '0';
        !           310:                }
        !           311:                if (t <= 0)
        !           312:                        t = 5;
        !           313:                t *= 60;
        !           314:                tim = time((long *) 0) + t;
        !           315:                return(tim);
        !           316:        }
        !           317:        t = 0;
        !           318:        while (strlen(s) > 2 && isdigit(*s))
        !           319:                t = t * 10 + *s++ - '0';
        !           320:        if (*s == ':')
        !           321:                s++;
        !           322:        if (t > 23)
        !           323:                goto badform;
        !           324:        tim = t*60;
        !           325:        t = 0;
        !           326:        while (isdigit(*s))
        !           327:                t = t * 10 + *s++ - '0';
        !           328:        if (t > 59)
        !           329:                goto badform;
        !           330:        tim += t; 
        !           331:        tim *= 60;
        !           332:        t1 = time((long *) 0);
        !           333:        lt = localtime(&t1);
        !           334:        t = lt->tm_sec + lt->tm_min*60 + lt->tm_hour*3600;
        !           335:        if (tim < t || tim >= (24*3600)) {
        !           336:                /* before now or after midnight */
        !           337:                printf("That must be tomorrow\nCan't you wait till then?\n");
        !           338:                finish();
        !           339:        }
        !           340:        return (t1 + tim - t);
        !           341: badform:
        !           342:        printf("Bad time format\n");
        !           343:        finish();
        !           344:        /*NOTREACHED*/
        !           345: }
        !           346: 
        !           347: warn(term, sdt, now, type)
        !           348:        FILE *term;
        !           349:        time_t sdt, now;
        !           350:        char *type;
        !           351: {
        !           352:        char *ts;
        !           353:        register delay = sdt - now;
        !           354: 
        !           355:        if (delay > 8)
        !           356:                while (delay % 5)
        !           357:                        delay++;
        !           358: 
        !           359:        fprintf(term,
        !           360:            "\007\007\t*** %sSystem shutdown message from %s@%s ***\r\n\n",
        !           361:                    type, shutter, hostname);
        !           362: 
        !           363:        ts = ctime(&sdt);
        !           364:        if (delay > 10 MINUTES)
        !           365:                fprintf(term, "System going down at %5.5s\r\n", ts+11);
        !           366:        else if (delay > 95 SECONDS) {
        !           367:                fprintf(term, "System going down in %d minute%s\r\n",
        !           368:                    (delay+30)/60, (delay+30)/60 != 1 ? "s" : "");
        !           369:        } else if (delay > 0) {
        !           370:                fprintf(term, "System going down in %d second%s\r\n",
        !           371:                    delay, delay != 1 ? "s" : "");
        !           372:        } else
        !           373:                fprintf(term, "System going down IMMEDIATELY\r\n");
        !           374: }
        !           375: 
        !           376: doitfast()
        !           377: {
        !           378:        FILE *fastd;
        !           379: 
        !           380:        if ((fastd = fopen(fastboot, "w")) != NULL) {
        !           381:                putc('\n', fastd);
        !           382:                (void) fclose(fastd);
        !           383:        }
        !           384: }
        !           385: 
        !           386: nolog(sdt)
        !           387:        time_t sdt;
        !           388: {
        !           389:        FILE *nologf;
        !           390: 
        !           391:        (void) unlink(nologin);                 /* in case linked to std file */
        !           392:        if ((nologf = fopen(nologin, "w")) != NULL) {
        !           393:                fprintf(nologf, nolog1, (ctime(&sdt)) + 11);
        !           394:                if (*nolog2)
        !           395:                        fprintf(nologf, "\t%s\n", nolog2 + 1);
        !           396:                (void) fclose(nologf);
        !           397:        }
        !           398: }
        !           399: 
        !           400: finish()
        !           401: {
        !           402:        (void) signal(SIGTERM, SIG_IGN);
        !           403:        (void) unlink(nologin);
        !           404:        exit(0);
        !           405: }
        !           406: 
        !           407: timeout()
        !           408: {
        !           409:        longjmp(alarmbuf, 1);
        !           410: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.