Annotation of 42BSD/bin/login.c, revision 1.1.1.1

1.1       root        1: #ifndef lint
                      2: static char *sccsid = "@(#)login.c     4.33 (Berkeley) 83/09/02";
                      3: #endif
                      4: 
                      5: /*
                      6:  * login [ name ]
                      7:  * login -r hostname (for rlogind)
                      8:  * login -h hostname (for telnetd, etc.)
                      9:  */
                     10: 
                     11: #include <sys/param.h>
                     12: #include <sys/quota.h>
                     13: #include <sys/stat.h>
                     14: #include <sys/time.h>
                     15: #include <sys/resource.h>
                     16: 
                     17: #include <sgtty.h>
                     18: #include <utmp.h>
                     19: #include <signal.h>
                     20: #include <pwd.h>
                     21: #include <stdio.h>
                     22: #include <lastlog.h>
                     23: #include <errno.h>
                     24: 
                     25: #define        SCPYN(a, b)     strncpy(a, b, sizeof(a))
                     26: 
                     27: #define NMAX   sizeof(utmp.ut_name)
                     28: 
                     29: #define        FALSE   0
                     30: #define        TRUE    -1
                     31: 
                     32: char   nolog[] =       "/etc/nologin";
                     33: char   qlog[]  =       ".hushlogin";
                     34: char   securetty[] =   "/etc/securetty";
                     35: char   maildir[30] =   "/usr/spool/mail/";
                     36: char   lastlog[] =     "/usr/adm/lastlog";
                     37: struct passwd nouser = {"", "nope", -1, -1, -1, "", "", "", "" };
                     38: struct sgttyb ttyb;
                     39: struct utmp utmp;
                     40: char   minusnam[16] = "-";
                     41: /*
                     42:  * This bounds the time given to login.  We initialize it here
                     43:  * so it can be patched on machines where it's too small.
                     44:  */
                     45: int    timeout = 60;
                     46: 
                     47: char   homedir[64] = "HOME=";
                     48: char   shell[64] = "SHELL=";
                     49: char   term[64] = "TERM=";
                     50: char   user[20] = "USER=";
                     51: 
                     52: char   *envinit[] =
                     53:     { homedir, shell, "PATH=:/usr/ucb:/bin:/usr/bin", term, user, 0 };
                     54: 
                     55: struct passwd *pwd;
                     56: struct passwd *getpwnam();
                     57: char   *strcat(), *rindex(), *index();
                     58: int    setpwent();
                     59: int    timedout();
                     60: char   *ttyname();
                     61: char   *crypt();
                     62: char   *getpass();
                     63: char   *stypeof();
                     64: extern char **environ;
                     65: extern int errno;
                     66: 
                     67: struct tchars tc = {
                     68:        CINTR, CQUIT, CSTART, CSTOP, CEOT, CBRK
                     69: };
                     70: struct ltchars ltc = {
                     71:        CSUSP, CDSUSP, CRPRNT, CFLUSH, CWERASE, CLNEXT
                     72: };
                     73: 
                     74: int    rflag;
                     75: char   rusername[NMAX+1], lusername[NMAX+1];
                     76: char   rpassword[NMAX+1];
                     77: char   name[NMAX+1];
                     78: char   *rhost;
                     79: 
                     80: main(argc, argv)
                     81:        char *argv[];
                     82: {
                     83:        register char *namep;
                     84:        int t, f, c, i;
                     85:        int invalid, quietlog;
                     86:        FILE *nlfd;
                     87:        char *ttyn;
                     88:        int ldisc = 0, zero = 0;
                     89: 
                     90:        signal(SIGALRM, timedout);
                     91:        alarm(timeout);
                     92:        signal(SIGQUIT, SIG_IGN);
                     93:        signal(SIGINT, SIG_IGN);
                     94:        setpriority(PRIO_PROCESS, 0, 0);
                     95:        quota(Q_SETUID, 0, 0, 0);
                     96:        /*
                     97:         * -r is used by rlogind to cause the autologin protocol;
                     98:         * -h is used by other servers to pass the name of the
                     99:         * remote host to login so that it may be placed in utmp and wtmp
                    100:         */
                    101:        if (argc > 1) {
                    102:                if (strcmp(argv[1], "-r") == 0) {
                    103:                        rflag = doremotelogin(argv[2]);
                    104:                        SCPYN(utmp.ut_host, argv[2]);
                    105:                        argc = 0;
                    106:                }
                    107:                if (strcmp(argv[1], "-h") == 0 && getuid() == 0) {
                    108:                        SCPYN(utmp.ut_host, argv[2]);
                    109:                        argc = 0;
                    110:                }
                    111:        }
                    112:        ioctl(0, TIOCLSET, &zero);
                    113:        ioctl(0, TIOCNXCL, 0);
                    114:        ioctl(0, FIONBIO, &zero);
                    115:        ioctl(0, FIOASYNC, &zero);
                    116:        ioctl(0, TIOCGETP, &ttyb);
                    117:        /*
                    118:         * If talking to an rlogin process,
                    119:         * propagate the terminal type and
                    120:         * baud rate across the network.
                    121:         */
                    122:        if (rflag)
                    123:                doremoteterm(term, &ttyb);
                    124:        ioctl(0, TIOCSLTC, &ltc);
                    125:        ioctl(0, TIOCSETC, &tc);
                    126:        ioctl(0, TIOCSETP, &ttyb);
                    127:        for (t = getdtablesize(); t > 3; t--)
                    128:                close(t);
                    129:        ttyn = ttyname(0);
                    130:        if (ttyn==(char *)0)
                    131:                ttyn = "/dev/tty??";
                    132:        do {
                    133:                ldisc = 0;
                    134:                ioctl(0, TIOCSETD, &ldisc);
                    135:                invalid = FALSE;
                    136:                SCPYN(utmp.ut_name, "");
                    137:                /*
                    138:                 * Name specified, take it.
                    139:                 */
                    140:                if (argc > 1) {
                    141:                        SCPYN(utmp.ut_name, argv[1]);
                    142:                        argc = 0;
                    143:                }
                    144:                /*
                    145:                 * If remote login take given name,
                    146:                 * otherwise prompt user for something.
                    147:                 */
                    148:                if (rflag) {
                    149:                        SCPYN(utmp.ut_name, lusername);
                    150:                        /* autologin failed, prompt for passwd */
                    151:                        if (rflag == -1)
                    152:                                rflag = 0;
                    153:                } else
                    154:                        getloginname(&utmp);
                    155:                if (!strcmp(pwd->pw_shell, "/bin/csh")) {
                    156:                        ldisc = NTTYDISC;
                    157:                        ioctl(0, TIOCSETD, &ldisc);
                    158:                }
                    159:                /*
                    160:                 * If no remote login authentication and
                    161:                 * a password exists for this user, prompt
                    162:                 * for one and verify it.
                    163:                 */
                    164:                if (!rflag && *pwd->pw_passwd != '\0') {
                    165:                        char *pp;
                    166: 
                    167:                        setpriority(PRIO_PROCESS, 0, -4);
                    168:                        pp = getpass("Password:");
                    169:                        namep = crypt(pp, pwd->pw_passwd);
                    170:                        setpriority(PRIO_PROCESS, 0, 0);
                    171:                        if (strcmp(namep, pwd->pw_passwd))
                    172:                                invalid = TRUE;
                    173:                }
                    174:                /*
                    175:                 * If user not super-user, check for logins disabled.
                    176:                 */
                    177:                if (pwd->pw_uid != 0 && (nlfd = fopen(nolog, "r")) > 0) {
                    178:                        while ((c = getc(nlfd)) != EOF)
                    179:                                putchar(c);
                    180:                        fflush(stdout);
                    181:                        sleep(5);
                    182:                        exit(0);
                    183:                }
                    184:                /*
                    185:                 * If valid so far and root is logging in,
                    186:                 * see if root logins on this terminal are permitted.
                    187:                 */
                    188:                if (!invalid && pwd->pw_uid == 0 &&
                    189:                    !rootterm(ttyn+sizeof("/dev/")-1)) {
                    190:                        logerr("ROOT LOGIN REFUSED %s",
                    191:                            ttyn+sizeof("/dev/")-1);
                    192:                        invalid = TRUE;
                    193:                }
                    194:                if (invalid) {
                    195:                        printf("Login incorrect\n");
                    196:                        if (ttyn[sizeof("/dev/tty")-1] == 'd')
                    197:                                logerr("BADDIALUP %s %s",
                    198:                                    ttyn+sizeof("/dev/")-1, utmp.ut_name);
                    199:                }
                    200:                if (*pwd->pw_shell == '\0')
                    201:                        pwd->pw_shell = "/bin/sh";
                    202:                i = strlen(pwd->pw_shell);
                    203:                if (chdir(pwd->pw_dir) < 0 && !invalid ) {
                    204:                        if (chdir("/") < 0) {
                    205:                                printf("No directory!\n");
                    206:                                invalid = TRUE;
                    207:                        } else {
                    208:                                printf("No directory! %s\n",
                    209:                                   "Logging in with home=/");
                    210:                                pwd->pw_dir = "/";
                    211:                        }
                    212:                }
                    213:                /*
                    214:                 * Remote login invalid must have been because
                    215:                 * of a restriction of some sort, no extra chances.
                    216:                 */
                    217:                if (rflag && invalid)
                    218:                        exit(1);
                    219:        } while (invalid);
                    220: /* committed to login turn off timeout */
                    221:        alarm(0);
                    222: 
                    223:        if (quota(Q_SETUID, pwd->pw_uid, 0, 0) < 0) {
                    224:                if (errno == EUSERS)
                    225:                        printf("%s.\n%s.\n",
                    226:                           "Too many users logged on already",
                    227:                           "Try again later");
                    228:                else if (errno == EPROCLIM)
                    229:                        printf("You have too many processes running.\n");
                    230:                else
                    231:                        perror("setuid");
                    232:                sleep(5);
                    233:                exit(0);
                    234:        }
                    235:        time(&utmp.ut_time);
                    236:        t = ttyslot();
                    237:        if (t > 0 && (f = open("/etc/utmp", 1)) >= 0) {
                    238:                lseek(f, (long)(t*sizeof(utmp)), 0);
                    239:                SCPYN(utmp.ut_line, rindex(ttyn, '/')+1);
                    240:                write(f, (char *)&utmp, sizeof(utmp));
                    241:                close(f);
                    242:        }
                    243:        if (t > 0 && (f = open("/usr/adm/wtmp", 1)) >= 0) {
                    244:                lseek(f, 0L, 2);
                    245:                write(f, (char *)&utmp, sizeof(utmp));
                    246:                close(f);
                    247:        }
                    248:        quietlog = access(qlog, 0) == 0;
                    249:        if ((f = open(lastlog, 2)) >= 0) {
                    250:                struct lastlog ll;
                    251: 
                    252:                lseek(f, (long)pwd->pw_uid * sizeof (struct lastlog), 0);
                    253:                if (read(f, (char *) &ll, sizeof ll) == sizeof ll &&
                    254:                    ll.ll_time != 0 && !quietlog) {
                    255:                        printf("Last login: %.*s ",
                    256:                            24-5, (char *)ctime(&ll.ll_time));
                    257:                        if (*ll.ll_host != '\0')
                    258:                                printf("from %.*s\n",
                    259:                                    sizeof (ll.ll_host), ll.ll_host);
                    260:                        else
                    261:                                printf("on %.*s\n",
                    262:                                    sizeof (ll.ll_line), ll.ll_line);
                    263:                }
                    264:                lseek(f, (long)pwd->pw_uid * sizeof (struct lastlog), 0);
                    265:                time(&ll.ll_time);
                    266:                SCPYN(ll.ll_line, rindex(ttyn, '/')+1);
                    267:                SCPYN(ll.ll_host, utmp.ut_host);
                    268:                write(f, (char *) &ll, sizeof ll);
                    269:                close(f);
                    270:        }
                    271:        chown(ttyn, pwd->pw_uid, pwd->pw_gid);
                    272:        chmod(ttyn, 0622);
                    273:        setgid(pwd->pw_gid);
                    274:        strncpy(name, utmp.ut_name, NMAX);
                    275:        name[NMAX] = '\0';
                    276:        initgroups(name, pwd->pw_gid);
                    277:        quota(Q_DOWARN, pwd->pw_uid, (dev_t)-1, 0);
                    278:        setuid(pwd->pw_uid);
                    279:        environ = envinit;
                    280:        strncat(homedir, pwd->pw_dir, sizeof(homedir)-6);
                    281:        strncat(shell, pwd->pw_shell, sizeof(shell)-7);
                    282:        if (term[strlen("TERM=")] == 0)
                    283:                strncat(term, stypeof(ttyn), sizeof(term)-6);
                    284:        strncat(user, pwd->pw_name, sizeof(user)-6);
                    285:        if ((namep = rindex(pwd->pw_shell, '/')) == NULL)
                    286:                namep = pwd->pw_shell;
                    287:        else
                    288:                namep++;
                    289:        strcat(minusnam, namep);
                    290:        umask(022);
                    291:        if (ttyn[sizeof("/dev/tty")-1] == 'd')
                    292:                logerr("DIALUP %s %s",
                    293:                    ttyn+sizeof("/dev/")-1, pwd->pw_name);
                    294:        if (!quietlog) {
                    295:                showmotd();
                    296:                strcat(maildir, pwd->pw_name);
                    297:                if (access(maildir,4)==0) {
                    298:                        struct stat statb;
                    299:                        stat(maildir, &statb);
                    300:                        if (statb.st_size)
                    301:                                printf("You have mail.\n");
                    302:                }
                    303:        }
                    304:        signal(SIGALRM, SIG_DFL);
                    305:        signal(SIGQUIT, SIG_DFL);
                    306:        signal(SIGINT, SIG_DFL);
                    307:        signal(SIGTSTP, SIG_IGN);
                    308:        execlp(pwd->pw_shell, minusnam, 0);
                    309:        perror(pwd->pw_shell);
                    310:        printf("No shell\n");
                    311:        exit(0);
                    312: }
                    313: 
                    314: getloginname(up)
                    315:        register struct utmp *up;
                    316: {
                    317:        register char *namep;
                    318:        char c;
                    319: 
                    320:        while (up->ut_name[0] == '\0') {
                    321:                namep = up->ut_name;
                    322:                printf("login: ");
                    323:                while ((c = getchar()) != '\n') {
                    324:                        if (c == ' ')
                    325:                                c = '_';
                    326:                        if (c == EOF)
                    327:                                exit(0);
                    328:                        if (namep < up->ut_name+NMAX)
                    329:                                *namep++ = c;
                    330:                }
                    331:        }
                    332:        strncpy(lusername, up->ut_name, NMAX);
                    333:        lusername[NMAX] = 0;
                    334:        setpwent();
                    335:        if ((pwd = getpwnam(lusername)) == NULL)
                    336:                pwd = &nouser;
                    337:        endpwent();
                    338: }
                    339: 
                    340: timedout()
                    341: {
                    342: 
                    343:        printf("Login timed out after %d seconds\n", timeout);
                    344:        exit(0);
                    345: }
                    346: 
                    347: int    stopmotd;
                    348: catch()
                    349: {
                    350: 
                    351:        signal(SIGINT, SIG_IGN);
                    352:        stopmotd++;
                    353: }
                    354: 
                    355: rootterm(tty)
                    356:        char *tty;
                    357: {
                    358:        register FILE *fd;
                    359:        char buf[100];
                    360: 
                    361:        if ((fd = fopen(securetty, "r")) == NULL)
                    362:                return(1);
                    363:        while (fgets(buf, sizeof buf, fd) != NULL) {
                    364:                buf[strlen(buf)-1] = '\0';
                    365:                if (strcmp(tty, buf) == 0) {
                    366:                        fclose(fd);
                    367:                        return(1);
                    368:                }
                    369:        }
                    370:        fclose(fd);
                    371:        return(0);
                    372: }
                    373: 
                    374: showmotd()
                    375: {
                    376:        FILE *mf;
                    377:        register c;
                    378: 
                    379:        signal(SIGINT, catch);
                    380:        if ((mf = fopen("/etc/motd","r")) != NULL) {
                    381:                while ((c = getc(mf)) != EOF && stopmotd == 0)
                    382:                        putchar(c);
                    383:                fclose(mf);
                    384:        }
                    385:        signal(SIGINT, SIG_IGN);
                    386: }
                    387: 
                    388: #undef UNKNOWN
                    389: #define UNKNOWN "su"
                    390: 
                    391: char *
                    392: stypeof(ttyid)
                    393:        char *ttyid;
                    394: {
                    395:        static char typebuf[16];
                    396:        char buf[50];
                    397:        register FILE *f;
                    398:        register char *p, *t, *q;
                    399: 
                    400:        if (ttyid == NULL)
                    401:                return (UNKNOWN);
                    402:        f = fopen("/etc/ttytype", "r");
                    403:        if (f == NULL)
                    404:                return (UNKNOWN);
                    405:        /* split off end of name */
                    406:        for (p = q = ttyid; *p != 0; p++)
                    407:                if (*p == '/')
                    408:                        q = p + 1;
                    409: 
                    410:        /* scan the file */
                    411:        while (fgets(buf, sizeof buf, f) != NULL) {
                    412:                for (t = buf; *t != ' ' && *t != '\t'; t++)
                    413:                        ;
                    414:                *t++ = 0;
                    415:                while (*t == ' ' || *t == '\t')
                    416:                        t++;
                    417:                for (p = t; *p > ' '; p++)
                    418:                        ;
                    419:                *p = 0;
                    420:                if (strcmp(q,t) == 0) {
                    421:                        strcpy(typebuf, buf);
                    422:                        fclose(f);
                    423:                        return (typebuf);
                    424:                }
                    425:        }
                    426:        fclose (f);
                    427:        return (UNKNOWN);
                    428: }
                    429: 
                    430: doremotelogin(host)
                    431:        char *host;
                    432: {
                    433:        FILE *hostf;
                    434:        int first = 1;
                    435: 
                    436:        getstr(rusername, sizeof (rusername), "remuser");
                    437:        getstr(lusername, sizeof (lusername), "locuser");
                    438:        getstr(term+5, sizeof(term)-5, "Terminal type");
                    439:        if (getuid()) {
                    440:                pwd = &nouser;
                    441:                goto bad;
                    442:        }
                    443:        setpwent();
                    444:        pwd = getpwnam(lusername);
                    445:        endpwent();
                    446:        if (pwd == NULL) {
                    447:                pwd = &nouser;
                    448:                goto bad;
                    449:        }
                    450:        hostf = pwd->pw_uid ? fopen("/etc/hosts.equiv", "r") : 0;
                    451: again:
                    452:        if (hostf) {
                    453:                char ahost[32];
                    454: 
                    455:                while (fgets(ahost, sizeof (ahost), hostf)) {
                    456:                        char *user;
                    457: 
                    458:                        if ((user = index(ahost, '\n')) != 0)
                    459:                                *user++ = '\0';
                    460:                        if ((user = index(ahost, ' ')) != 0)
                    461:                                *user++ = '\0';
                    462:                        if (!strcmp(host, ahost) &&
                    463:                            !strcmp(rusername, user ? user : lusername)) {
                    464:                                fclose(hostf);
                    465:                                return (1);
                    466:                        }
                    467:                }
                    468:                fclose(hostf);
                    469:        }
                    470:        if (first == 1) {
                    471:                char *rhosts = ".rhosts";
                    472:                struct stat sbuf;
                    473: 
                    474:                first = 0;
                    475:                if (chdir(pwd->pw_dir) < 0)
                    476:                        goto again;
                    477:                if (lstat(rhosts, &sbuf) < 0)
                    478:                        goto again;
                    479:                if ((sbuf.st_mode & S_IFMT) == S_IFLNK) {
                    480:                        printf("login: .rhosts is a soft link.\r\n");
                    481:                        goto bad;
                    482:                }
                    483:                hostf = fopen(rhosts, "r");
                    484:                fstat(fileno(hostf), &sbuf);
                    485:                if (sbuf.st_uid && sbuf.st_uid != pwd->pw_uid) {
                    486:                        printf("login: Bad .rhosts ownership.\r\n");
                    487:                        fclose(hostf);
                    488:                        goto bad;
                    489:                }
                    490:                goto again;
                    491:        }
                    492: bad:
                    493:        return (-1);
                    494: }
                    495: 
                    496: getstr(buf, cnt, err)
                    497:        char *buf;
                    498:        int cnt;
                    499:        char *err;
                    500: {
                    501:        char c;
                    502: 
                    503:        do {
                    504:                if (read(0, &c, 1) != 1)
                    505:                        exit(1);
                    506:                if (--cnt < 0) {
                    507:                        printf("%s too long\r\n", err);
                    508:                        exit(1);
                    509:                }
                    510:                *buf++ = c;
                    511:        } while (c != 0);
                    512: }
                    513: 
                    514: char   *speeds[] =
                    515:     { "0", "50", "75", "110", "134", "150", "200", "300",
                    516:       "600", "1200", "1800", "2400", "4800", "9600", "19200", "38400" };
                    517: #define        NSPEEDS (sizeof (speeds) / sizeof (speeds[0]))
                    518: 
                    519: doremoteterm(term, tp)
                    520:        char *term;
                    521:        struct sgttyb *tp;
                    522: {
                    523:        char *cp = index(term, '/');
                    524:        register int i;
                    525: 
                    526:        if (cp) {
                    527:                *cp++ = 0;
                    528:                for (i = 0; i < NSPEEDS; i++)
                    529:                        if (!strcmp(speeds[i], cp)) {
                    530:                                tp->sg_ispeed = tp->sg_ospeed = i;
                    531:                                break;
                    532:                        }
                    533:        }
                    534:        tp->sg_flags = ECHO|CRMOD|ANYP|XTABS;
                    535: }
                    536: 
                    537: logerr(fmt, a1, a2, a3)
                    538:        char *fmt, *a1, *a2, *a3;
                    539: {
                    540: #ifdef LOGERR
                    541:        FILE *cons = fopen("/dev/console", "w");
                    542: 
                    543:        if (cons != NULL) {
                    544:                fprintf(cons, fmt, a1, a2, a3);
                    545:                fprintf(cons, "\n\r");
                    546:                fclose(cons);
                    547:        }
                    548: #endif
                    549: }

unix.superglobalmegacorp.com

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