Annotation of 40BSD/cmd/at.c, revision 1.1.1.1

1.1       root        1: static char *sccsid = "@(#)at.c        4.2 (Berkeley) 10/21/80";
                      2: /*
                      3:  * at time mon day
                      4:  * at time wday
                      5:  * at time wday 'week'
                      6:  *
                      7:  */
                      8: #include <stdio.h>
                      9: #include <ctype.h>
                     10: #include <time.h>
                     11: #include <signal.h>
                     12: 
                     13: #define HOUR 100
                     14: #define HALFDAY        (12*HOUR)
                     15: #define DAY    (24*HOUR)
                     16: #define THISDAY "/usr/spool/at"
                     17: 
                     18: char *days[] = {
                     19:        "sunday",
                     20:        "monday",
                     21:        "tuesday",
                     22:        "wednesday",
                     23:        "thursday",
                     24:        "friday",
                     25:        "saturday",
                     26: };
                     27: 
                     28: struct monstr {
                     29:        char *mname; 
                     30:        int mlen;
                     31: } months[] = {
                     32:        { "january", 31 },
                     33:        { "february", 28 },
                     34:        { "march", 31 },
                     35:        { "april", 30 },
                     36:        { "may", 31 },
                     37:        { "june", 30 },
                     38:        { "july", 31 },
                     39:        { "august", 31 },
                     40:        { "september", 30 },
                     41:        { "october", 31 },
                     42:        { "november", 30 },
                     43:        { "december", 31 },
                     44:        { 0, 0 },
                     45: };
                     46: 
                     47: char   fname[100];
                     48: int    utime;  /* requested time in grains */
                     49: int    now;    /* when is it */
                     50: int    uday; /* day of year to be done */
                     51: int    uyear; /* year */
                     52: int    today; /* day of year today */
                     53: FILE   *file;
                     54: FILE   *ifile;
                     55: char   **environ;
                     56: char   *prefix();
                     57: char    *getenv();
                     58: FILE   *popen();
                     59: 
                     60: main(argc, argv)
                     61: char **argv;
                     62: {
                     63:        extern onintr();
                     64:        register c;
                     65:        char pwbuf[100];
                     66:        FILE *pwfil;
                     67:        int larg;
                     68:        char *tmp;
                     69: 
                     70:        /* argv[1] is the user's time: e.g.,  3AM */
                     71:        /* argv[2] is a month name or day of week */
                     72:        /* argv[3] is day of month or 'week' */
                     73:        /* another argument might be an input file */
                     74:        if (argc < 2) {
                     75:                fprintf(stderr, "at: arg count\n");
                     76:                exit(1);
                     77:        }
                     78:        makeutime(argv[1]);
                     79:        larg = makeuday(argc,argv)+1;
                     80:        if (uday==today && larg<=2 && utime<=now)
                     81:                uday++;
                     82:        c = uyear%4==0? 366: 365;
                     83:        if (uday >= c) {
                     84:                uday -= c;
                     85:                uyear++;
                     86:        }
                     87:        filename(THISDAY, uyear, uday, utime);
                     88:        /* Create file, then change UIDS */
                     89:        close(creat(fname,0644));
                     90:        chown(fname,getuid(),getgid());
                     91:        setuid(getuid());
                     92:        ifile = stdin;
                     93:        if (argc > larg)
                     94:                ifile = fopen(argv[larg], "r");
                     95:        if (ifile == NULL) {
                     96:                fprintf(stderr, "at: cannot open input: %s\n", argv[larg]);
                     97:                exit(1);
                     98:        }
                     99:        if (signal(SIGINT, SIG_IGN) != SIG_IGN)
                    100:                signal(SIGINT, onintr);
                    101:        file = fopen(fname, "w");
                    102:        if (file == NULL) {
                    103:                fprintf(stderr, "at: cannot open memo file\n");
                    104:                exit(1);
                    105:        }
                    106:        if ((pwfil = popen("pwd", "r")) == NULL) {
                    107:                fprintf(stderr, "at: can't execute pwd\n");
                    108:                exit(1);
                    109:        }
                    110:        fgets(pwbuf, 100, pwfil);
                    111:        pclose(pwfil);
                    112:        fprintf(file, "cd %s", pwbuf);
                    113:        c = umask(0);
                    114:        umask(c);
                    115:        fprintf(file, "umask %.1o\n", c);
                    116:        if (environ) {
                    117:                char **ep = environ;
                    118:                while(*ep)
                    119:                {
                    120:                        char *cp;
                    121:                        for (tmp = *ep, cp = "TERMCAP"; *tmp==*cp; tmp++,cp++);
                    122:                        if (*cp == 0 && *tmp== '=') {
                    123:                                ep++;
                    124:                                continue;
                    125:                        }
                    126:                        for(tmp = *ep ; *tmp != '=' ; tmp++) putc(*tmp,file);
                    127:                        putc('=', file);
                    128:                        putc('\'', file);
                    129:                        for (tmp++; *tmp; tmp++) {
                    130:                                if (*tmp == '\'')
                    131:                                        putc('\\', file);
                    132:                                putc(*tmp, file);
                    133:                        }
                    134:                        putc('\'', file);
                    135:                        fprintf(file, "\nexport ");
                    136:                        for(tmp = *ep ; *tmp != '=' ; tmp++) putc(*tmp,file);
                    137:                        putc('\n',file);
                    138:                        ep++;
                    139:                }
                    140:        }
                    141:        /*
                    142:         * see if the SHELL variable in the current enviroment is /bin/csh
                    143:         * and in that case, use the csh as the shell
                    144:         */
                    145:        tmp = getenv("SHELL");
                    146:        if (strcmp(tmp+strlen(tmp)-3, "csh") == 0)
                    147:                fprintf(file, "%s %s\n", tmp, "<< 'xxFUNNYxx'");
                    148:        while((c = getc(ifile)) != EOF) {
                    149:                putc(c, file);
                    150:        }
                    151:        if (strcmp(tmp+strlen(tmp)-3, "csh") == 0)
                    152:                fprintf(file, "%s\n", "xxFUNNYxx");
                    153:        exit(0);
                    154: }
                    155: 
                    156: makeutime(pp)
                    157: char *pp; 
                    158: {
                    159:        register val;
                    160:        register char *p;
                    161: 
                    162:        /* p points to a user time */
                    163:        p = pp;
                    164:        val = 0;
                    165:        while(isdigit(*p)) {
                    166:                val = val*10+(*p++ -'0');
                    167:        }
                    168:        if (p-pp < 3)
                    169:                val *= HOUR;
                    170: 
                    171:        for (;;) {
                    172:                switch(*p) {
                    173: 
                    174:                case ':':
                    175:                        ++p;
                    176:                        if (isdigit(*p)) {
                    177:                                if (isdigit(p[1])) {
                    178:                                        val +=(10* *p + p[1] - 11*'0');
                    179:                                        p += 2;
                    180:                                        continue;
                    181:                                }
                    182:                        }
                    183:                        fprintf(stderr, "at: bad time format:\n");
                    184:                        exit(1);
                    185: 
                    186:                case 'A':
                    187:                case 'a':
                    188:                        if (val >= HALFDAY+HOUR)
                    189:                                val = DAY+1;  /* illegal */
                    190:                        if (val >= HALFDAY && val <(HALFDAY+HOUR))
                    191:                                val -= HALFDAY;
                    192:                        break;
                    193: 
                    194:                case 'P':
                    195:                case 'p':
                    196:                        if (val >= HALFDAY+HOUR)
                    197:                                val = DAY+1;  /* illegal */
                    198:                        if (val < HALFDAY)
                    199:                                val += HALFDAY;
                    200:                        break;
                    201: 
                    202:                case 'n':
                    203:                case 'N':
                    204:                        val = HALFDAY;
                    205:                        break;
                    206: 
                    207:                case 'M':
                    208:                case 'm':
                    209:                        val = 0;
                    210:                        break;
                    211: 
                    212: 
                    213:                case '\0':
                    214:                case ' ':
                    215:                        /* 24 hour time */
                    216:                        if (val == DAY)
                    217:                                val -= DAY;
                    218:                        break;
                    219: 
                    220:                default:
                    221:                        fprintf(stderr, "at: bad time format\n");
                    222:                        exit(1);
                    223: 
                    224:                }
                    225:                break;
                    226:        }
                    227:        if (val < 0 || val >= DAY) {
                    228:                fprintf(stderr, "at: time out of range\n");
                    229:                exit(1);
                    230:        }
                    231:        if (val%HOUR >= 60) {
                    232:                fprintf(stderr, "at: illegal minute field\n");
                    233:                exit(1);
                    234:        }
                    235:        utime = val;
                    236: }
                    237: 
                    238: 
                    239: makeuday(argc,argv)
                    240: char **argv;
                    241: {
                    242:        /* the presumption is that argv[2], argv[3] are either
                    243:           month day OR weekday [week].  Returns either 2 or 3 as last
                    244:           argument used */
                    245:        /* first of all, what's today */
                    246:        long tm;
                    247:        int found = -1;
                    248:        char **ps;
                    249:        struct tm *detail, *localtime();
                    250:        struct monstr *pt;
                    251: 
                    252:        time(&tm);
                    253:        detail = localtime(&tm);
                    254:        uday = today = detail->tm_yday;
                    255:        uyear = detail->tm_year;
                    256:        now = detail->tm_hour*100+detail->tm_min;
                    257:        if (argc<=2)
                    258:                return(1);
                    259:        /* is the next argument a month name ? */
                    260:        for (pt=months; pt->mname; pt++) {
                    261:                if (prefix(argv[2], pt->mname)) {
                    262:                        if (found<0)
                    263:                                found = pt-months;
                    264:                        else {
                    265:                                fprintf(stderr, "at: ambiguous month\n");
                    266:                                exit(1);
                    267:                        }
                    268:                }
                    269:        }
                    270:        if (found>=0) {
                    271:                if (argc<=3)
                    272:                        return(2);
                    273:                uday = atoi(argv[3]) - 1;
                    274:                if (uday<0) {
                    275:                        fprintf(stderr, "at: illegal day\n");
                    276:                        exit(1);
                    277:                }
                    278:                while(--found>=0)
                    279:                        uday += months[found].mlen;
                    280:                if (detail->tm_year%4==0 && uday>59)
                    281:                        uday += 1;
                    282:                return(3);
                    283:        }
                    284:        /* not a month, try day of week */
                    285:        found = -1;
                    286:        for (ps=days; ps<days+7; ps++) {
                    287:                if (prefix(argv[2], *ps)) {
                    288:                        if (found<0)
                    289:                                found = ps-days;
                    290:                        else {
                    291:                                fprintf(stderr, "at: ambiguous day of week\n");
                    292:                                exit(1);
                    293:                        }
                    294:                }
                    295:        }
                    296:        if (found<0)
                    297:                return(1);
                    298:        /* find next day of this sort */
                    299:        uday = found - detail->tm_wday;
                    300:        if (uday<=0)
                    301:                uday += 7;
                    302:        uday += today;
                    303:        if (argc>3 && strcmp("week", argv[3])==0) {
                    304:                uday += 7;
                    305:                return(3);
                    306:        }
                    307:        return(2);
                    308: }
                    309: 
                    310: char *
                    311: prefix(begin, full)
                    312: char *begin, *full;
                    313: {
                    314:        int c;
                    315:        while (c = *begin++) {
                    316:                if (isupper(c))
                    317:                        c = tolower(c);
                    318:                if (*full != c)
                    319:                        return(0);
                    320:                else
                    321:                        full++;
                    322:        }
                    323:        return(full);
                    324: }
                    325: 
                    326: filename(dir, y, d, t)
                    327: char *dir;
                    328: {
                    329:        register i;
                    330: 
                    331:        for (i=0; ; i += 53) {
                    332:                sprintf(fname, "%s/%02d.%03d.%04d.%02d", dir, y, d, t,
                    333:                   (getpid()+i)%100);
                    334:                if (access(fname, 0) == -1)
                    335:                        return;
                    336:        }
                    337: }
                    338: 
                    339: onintr()
                    340: {
                    341:        unlink(fname);
                    342:        exit(1);
                    343: }

unix.superglobalmegacorp.com

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