Annotation of 43BSDReno/domestic/src/rcp/rcp.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1983, 1990 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * Redistribution and use in source and binary forms are permitted provided
        !             6:  * that: (1) source distributions retain this entire copyright notice and
        !             7:  * comment, and (2) distributions including binaries display the following
        !             8:  * acknowledgement:  ``This product includes software developed by the
        !             9:  * University of California, Berkeley and its contributors'' in the
        !            10:  * documentation or other materials provided with the distribution and in
        !            11:  * all advertising materials mentioning features or use of this software.
        !            12:  * Neither the name of the University nor the names of its contributors may
        !            13:  * be used to endorse or promote products derived from this software without
        !            14:  * specific prior written permission.
        !            15:  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
        !            16:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
        !            17:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
        !            18:  */
        !            19: 
        !            20: #ifndef lint
        !            21: char copyright[] =
        !            22: "@(#) Copyright (c) 1983, 1990 The Regents of the University of California.\n\
        !            23:  All rights reserved.\n";
        !            24: #endif /* not lint */
        !            25: 
        !            26: #ifndef lint
        !            27: static char sccsid[] = "@(#)rcp.c      5.29 (Berkeley) 6/29/90";
        !            28: #endif /* not lint */
        !            29: 
        !            30: /*
        !            31:  * rcp
        !            32:  */
        !            33: #include <sys/param.h>
        !            34: #include <sys/file.h>
        !            35: #include <sys/stat.h>
        !            36: #include <sys/time.h>
        !            37: #include <sys/ioctl.h>
        !            38: #include <sys/dir.h>
        !            39: #include <sys/signal.h>
        !            40: #include <netinet/in.h>
        !            41: #include <netinet/in_systm.h>
        !            42: #include <netinet/ip.h>
        !            43: #include <pwd.h>
        !            44: #include <netdb.h>
        !            45: #include <errno.h>
        !            46: #include <string.h>
        !            47: #include <stdio.h>
        !            48: #include <ctype.h>
        !            49: #include "pathnames.h"
        !            50: 
        !            51: #ifdef KERBEROS
        !            52: #include <kerberosIV/des.h>
        !            53: #include <kerberosIV/krb.h>
        !            54: char   dst_realm_buf[REALM_SZ];
        !            55: char   *dest_realm = NULL;
        !            56: int    use_kerberos = 1, encrypt = 0;
        !            57: CREDENTIALS    cred;
        !            58: Key_schedule   schedule;
        !            59: extern char    *krb_realmofhost();
        !            60: #define        OPTIONS "dfkprtx"
        !            61: #else
        !            62: #define        OPTIONS "dfprt"
        !            63: #endif
        !            64: 
        !            65: extern int errno;
        !            66: struct passwd *pwd;
        !            67: u_short        port;
        !            68: uid_t  userid;
        !            69: int errs, rem;
        !            70: int pflag, iamremote, iamrecursive, targetshouldbedirectory;
        !            71: 
        !            72: #define        CMDNEEDS        64
        !            73: char cmd[CMDNEEDS];            /* must hold "rcp -r -p -d\0" */
        !            74: 
        !            75: typedef struct _buf {
        !            76:        int     cnt;
        !            77:        char    *buf;
        !            78: } BUF;
        !            79: 
        !            80: main(argc, argv)
        !            81:        int argc;
        !            82:        char **argv;
        !            83: {
        !            84:        extern int optind;
        !            85:        struct servent *sp;
        !            86:        int ch, fflag, tflag;
        !            87:        char *targ, *colon();
        !            88:        struct passwd *getpwuid();
        !            89:        int lostconn();
        !            90: 
        !            91:        fflag = tflag = 0;
        !            92:        while ((ch = getopt(argc, argv, OPTIONS)) != EOF)
        !            93:                switch(ch) {
        !            94:                /* user-visible flags */
        !            95:                case 'p':                       /* preserve access/mod times */
        !            96:                        ++pflag;
        !            97:                        break;
        !            98:                case 'r':
        !            99:                        ++iamrecursive;
        !           100:                        break;
        !           101: #ifdef KERBEROS
        !           102:                case 'k':
        !           103:                        strncpy(dst_realm_buf, ++argv, REALM_SZ);
        !           104:                        dest_realm = dst_realm_buf;
        !           105:                        break;
        !           106:                case 'x':
        !           107:                        encrypt = 1;
        !           108:                        /* des_set_key(cred.session, schedule); */
        !           109:                        break;
        !           110: #endif
        !           111:                /* rshd-invoked options (server) */
        !           112:                case 'd':
        !           113:                        targetshouldbedirectory = 1;
        !           114:                        break;
        !           115:                case 'f':                       /* "from" */
        !           116:                        iamremote = 1;
        !           117:                        fflag = 1;
        !           118:                        break;
        !           119:                case 't':                       /* "to" */
        !           120:                        iamremote = 1;
        !           121:                        tflag = 1;
        !           122:                        break;
        !           123: 
        !           124:                case '?':
        !           125:                default:
        !           126:                        usage();
        !           127:                }
        !           128:        argc -= optind;
        !           129:        argv += optind;
        !           130: 
        !           131: #ifdef KERBEROS
        !           132:        sp = getservbyname((encrypt ? "ekshell" : "kshell"), "tcp");
        !           133:        if (sp == NULL) {
        !           134:                char    msgbuf[64];
        !           135:                use_kerberos = 0;
        !           136:                (void) sprintf(msgbuf, "can't get entry for %s/tcp service",
        !           137:                        (encrypt ? "ekshell" : "kshell"));
        !           138:                old_warning(msgbuf);
        !           139:                sp = getservbyname("shell", "tcp");
        !           140:        }
        !           141: #else
        !           142:        sp = getservbyname("shell", "tcp");
        !           143: #endif
        !           144:        if (sp == NULL) {
        !           145:                (void)fprintf(stderr, "rcp: shell/tcp: unknown service\n");
        !           146:                exit(1);
        !           147:        }
        !           148:        port = sp->s_port;
        !           149: 
        !           150:        if (!(pwd = getpwuid(userid = getuid()))) {
        !           151:                (void)fprintf(stderr, "rcp: unknown user %d.\n", userid);
        !           152:                exit(1);
        !           153:        }
        !           154: 
        !           155:        if (fflag) {
        !           156:                /* follow "protocol", send data */
        !           157:                (void)response();
        !           158:                (void)setuid(userid);
        !           159:                source(argc, argv);
        !           160:                exit(errs);
        !           161:        }
        !           162: 
        !           163:        if (tflag) {
        !           164:                /* receive data */
        !           165:                (void)setuid(userid);
        !           166:                sink(argc, argv);
        !           167:                exit(errs);
        !           168:        }
        !           169: 
        !           170:        if (argc < 2)
        !           171:                usage();
        !           172:        if (argc > 2)
        !           173:                targetshouldbedirectory = 1;
        !           174: 
        !           175:        rem = -1;
        !           176:        /* command to be executed on remote system using "rsh" */
        !           177: #ifdef KERBEROS
        !           178:        (void)sprintf(cmd, "rcp%s%s%s%s", iamrecursive ? " -r" : "",
        !           179:            ((encrypt && use_kerberos) ? " -x" : ""),
        !           180:            pflag ? " -p" : "", targetshouldbedirectory ? " -d" : "");
        !           181: #else
        !           182:        (void)sprintf(cmd, "rcp%s%s%s", iamrecursive ? " -r" : "",
        !           183:            pflag ? " -p" : "", targetshouldbedirectory ? " -d" : "");
        !           184: #endif
        !           185: 
        !           186:        (void)signal(SIGPIPE, lostconn);
        !           187: 
        !           188:        if (targ = colon(argv[argc - 1]))
        !           189:                toremote(targ, argc, argv);     /* destination is remote host */
        !           190:        else {
        !           191:                tolocal(argc, argv);            /* destination is local host */
        !           192:                if (targetshouldbedirectory)
        !           193:                        verifydir(argv[argc - 1]);
        !           194:        }
        !           195:        exit(errs);
        !           196: }
        !           197: 
        !           198: toremote(targ, argc, argv)
        !           199:        char *targ;
        !           200:        int argc;
        !           201:        char **argv;
        !           202: {
        !           203:        int i, tos;
        !           204:        char *bp, *host, *src, *suser, *thost, *tuser;
        !           205:        char *colon(), *malloc();
        !           206: 
        !           207:        *targ++ = 0;
        !           208:        if (*targ == 0)
        !           209:                targ = ".";
        !           210: 
        !           211:        if (thost = index(argv[argc - 1], '@')) {
        !           212:                /* user@host */
        !           213:                *thost++ = 0;
        !           214:                tuser = argv[argc - 1];
        !           215:                if (*tuser == '\0')
        !           216:                        tuser = NULL;
        !           217:                else if (!okname(tuser))
        !           218:                        exit(1);
        !           219:        } else {
        !           220:                thost = argv[argc - 1];
        !           221:                tuser = NULL;
        !           222:        }
        !           223: 
        !           224:        for (i = 0; i < argc - 1; i++) {
        !           225:                src = colon(argv[i]);
        !           226:                if (src) {                      /* remote to remote */
        !           227:                        *src++ = 0;
        !           228:                        if (*src == 0)
        !           229:                                src = ".";
        !           230:                        host = index(argv[i], '@');
        !           231:                        if (!(bp = malloc((u_int)(strlen(_PATH_RSH) +
        !           232:                                    strlen(argv[i]) + strlen(src) +
        !           233:                                    tuser ? strlen(tuser) : 0 + strlen(thost) +
        !           234:                                    strlen(targ)) + CMDNEEDS + 20)))
        !           235:                                        nospace();
        !           236:                        if (host) {
        !           237:                                *host++ = 0;
        !           238:                                suser = argv[i];
        !           239:                                if (*suser == '\0')
        !           240:                                        suser = pwd->pw_name;
        !           241:                                else if (!okname(suser))
        !           242:                                        continue;
        !           243:                                (void)sprintf(bp,
        !           244:                                    "%s %s -l %s -n %s %s '%s%s%s:%s'",
        !           245:                                    _PATH_RSH, host, suser, cmd, src,
        !           246:                                    tuser ? tuser : "", tuser ? "@" : "",
        !           247:                                    thost, targ);
        !           248:                        } else
        !           249:                                (void)sprintf(bp, "%s %s -n %s %s '%s%s%s:%s'",
        !           250:                                    _PATH_RSH, argv[i], cmd, src,
        !           251:                                    tuser ? tuser : "", tuser ? "@" : "",
        !           252:                                    thost, targ);
        !           253:                        (void)susystem(bp);
        !           254:                        (void)free(bp);
        !           255:                } else {                        /* local to remote */
        !           256:                        if (rem == -1) {
        !           257:                                if (!(bp = malloc((u_int)strlen(targ) +
        !           258:                                    CMDNEEDS + 20)))
        !           259:                                        nospace();
        !           260:                                (void)sprintf(bp, "%s -t %s", cmd, targ);
        !           261:                                host = thost;
        !           262: #ifdef KERBEROS
        !           263:                                if (use_kerberos)
        !           264:                                        rem = kerberos(&host, bp,
        !           265:                                            pwd->pw_name,
        !           266:                                            tuser ? tuser : pwd->pw_name);
        !           267:                                else
        !           268: #endif
        !           269:                                        rem = rcmd(&host, port, pwd->pw_name,
        !           270:                                            tuser ? tuser : pwd->pw_name,
        !           271:                                            bp, 0);
        !           272:                                if (rem < 0)
        !           273:                                        exit(1);
        !           274:                                tos = IPTOS_THROUGHPUT;
        !           275:                                if (setsockopt(rem, IPPROTO_IP, IP_TOS,
        !           276:                                    (char *)&tos, sizeof(int)) < 0)
        !           277:                                        perror("rcp: setsockopt TOS (ignored)");
        !           278:                                if (response() < 0)
        !           279:                                        exit(1);
        !           280:                                (void)free(bp);
        !           281:                                (void)setuid(userid);
        !           282:                        }
        !           283:                        source(1, argv+i);
        !           284:                }
        !           285:        }
        !           286: }
        !           287: 
        !           288: tolocal(argc, argv)
        !           289:        int argc;
        !           290:        char **argv;
        !           291: {
        !           292:        int i, tos;
        !           293:        char *bp, *host, *src, *suser;
        !           294:        char *colon(), *malloc();
        !           295: 
        !           296:        for (i = 0; i < argc - 1; i++) {
        !           297:                if (!(src = colon(argv[i]))) {  /* local to local */
        !           298:                        if (!(bp = malloc((u_int)(strlen(_PATH_CP) +
        !           299:                            strlen(argv[i]) + strlen(argv[argc - 1])) + 20)))
        !           300:                                nospace();
        !           301:                        (void)sprintf(bp, "%s%s%s %s %s", _PATH_CP,
        !           302:                            iamrecursive ? " -r" : "", pflag ? " -p" : "",
        !           303:                            argv[i], argv[argc - 1]);
        !           304:                        (void)susystem(bp);
        !           305:                        (void)free(bp);
        !           306:                        continue;
        !           307:                }
        !           308:                *src++ = 0;
        !           309:                if (*src == 0)
        !           310:                        src = ".";
        !           311:                host = index(argv[i], '@');
        !           312:                if (host) {
        !           313:                        *host++ = 0;
        !           314:                        suser = argv[i];
        !           315:                        if (*suser == '\0')
        !           316:                                suser = pwd->pw_name;
        !           317:                        else if (!okname(suser))
        !           318:                                continue;
        !           319:                } else {
        !           320:                        host = argv[i];
        !           321:                        suser = pwd->pw_name;
        !           322:                }
        !           323:                if (!(bp = malloc((u_int)(strlen(src)) + CMDNEEDS + 20)))
        !           324:                        nospace();
        !           325:                (void)sprintf(bp, "%s -f %s", cmd, src);
        !           326: #ifdef KERBEROS
        !           327:                if (use_kerberos)
        !           328:                        rem = kerberos(&host, bp, pwd->pw_name, suser);
        !           329:                else
        !           330: #endif
        !           331:                        rem = rcmd(&host, port, pwd->pw_name, suser, bp, 0);
        !           332:                (void)free(bp);
        !           333:                if (rem < 0)
        !           334:                        continue;
        !           335:                (void)seteuid(userid);
        !           336:                tos = IPTOS_THROUGHPUT;
        !           337:                if (setsockopt(rem, IPPROTO_IP, IP_TOS,
        !           338:                    (char *)&tos, sizeof(int)) < 0)
        !           339:                        perror("rcp: setsockopt TOS (ignored)");
        !           340:                sink(1, argv + argc - 1);
        !           341:                (void)seteuid(0);
        !           342:                (void)close(rem);
        !           343:                rem = -1;
        !           344:        }
        !           345: }
        !           346: 
        !           347: verifydir(cp)
        !           348:        char *cp;
        !           349: {
        !           350:        struct stat stb;
        !           351: 
        !           352:        if (stat(cp, &stb) >= 0) {
        !           353:                if ((stb.st_mode & S_IFMT) == S_IFDIR)
        !           354:                        return;
        !           355:                errno = ENOTDIR;
        !           356:        }
        !           357:        error("rcp: %s: %s.\n", cp, strerror(errno));
        !           358:        exit(1);
        !           359: }
        !           360: 
        !           361: char *
        !           362: colon(cp)
        !           363:        register char *cp;
        !           364: {
        !           365:        for (; *cp; ++cp) {
        !           366:                if (*cp == ':')
        !           367:                        return(cp);
        !           368:                if (*cp == '/')
        !           369:                        return(0);
        !           370:        }
        !           371:        return(0);
        !           372: }
        !           373: 
        !           374: okname(cp0)
        !           375:        char *cp0;
        !           376: {
        !           377:        register char *cp = cp0;
        !           378:        register int c;
        !           379: 
        !           380:        do {
        !           381:                c = *cp;
        !           382:                if (c & 0200)
        !           383:                        goto bad;
        !           384:                if (!isalpha(c) && !isdigit(c) && c != '_' && c != '-')
        !           385:                        goto bad;
        !           386:        } while (*++cp);
        !           387:        return(1);
        !           388: bad:
        !           389:        (void)fprintf(stderr, "rcp: invalid user name %s\n", cp0);
        !           390:        return(0);
        !           391: }
        !           392: 
        !           393: susystem(s)
        !           394:        char *s;
        !           395: {
        !           396:        int status, pid, w;
        !           397:        register sig_t istat, qstat;
        !           398: 
        !           399:        if ((pid = vfork()) == 0) {
        !           400:                (void)setuid(userid);
        !           401:                execl(_PATH_BSHELL, "sh", "-c", s, (char *)0);
        !           402:                _exit(127);
        !           403:        }
        !           404:        istat = signal(SIGINT, SIG_IGN);
        !           405:        qstat = signal(SIGQUIT, SIG_IGN);
        !           406:        while ((w = wait(&status)) != pid && w != -1)
        !           407:                ;
        !           408:        if (w == -1)
        !           409:                status = -1;
        !           410:        (void)signal(SIGINT, istat);
        !           411:        (void)signal(SIGQUIT, qstat);
        !           412:        return(status);
        !           413: }
        !           414: 
        !           415: source(argc, argv)
        !           416:        int argc;
        !           417:        char **argv;
        !           418: {
        !           419:        struct stat stb;
        !           420:        static BUF buffer;
        !           421:        BUF *bp;
        !           422:        off_t i;
        !           423:        int x, readerr, f, amt;
        !           424:        char *last, *name, buf[BUFSIZ];
        !           425:        BUF *allocbuf();
        !           426: 
        !           427:        for (x = 0; x < argc; x++) {
        !           428:                name = argv[x];
        !           429:                if ((f = open(name, O_RDONLY, 0)) < 0) {
        !           430:                        error("rcp: %s: %s\n", name, strerror(errno));
        !           431:                        continue;
        !           432:                }
        !           433:                if (fstat(f, &stb) < 0)
        !           434:                        goto notreg;
        !           435:                switch (stb.st_mode&S_IFMT) {
        !           436: 
        !           437:                case S_IFREG:
        !           438:                        break;
        !           439: 
        !           440:                case S_IFDIR:
        !           441:                        if (iamrecursive) {
        !           442:                                (void)close(f);
        !           443:                                rsource(name, &stb);
        !           444:                                continue;
        !           445:                        }
        !           446:                        /* FALLTHROUGH */
        !           447:                default:
        !           448: notreg:                        (void)close(f);
        !           449:                        error("rcp: %s: not a plain file\n", name);
        !           450:                        continue;
        !           451:                }
        !           452:                last = rindex(name, '/');
        !           453:                if (last == 0)
        !           454:                        last = name;
        !           455:                else
        !           456:                        last++;
        !           457:                if (pflag) {
        !           458:                        /*
        !           459:                         * Make it compatible with possible future
        !           460:                         * versions expecting microseconds.
        !           461:                         */
        !           462:                        (void)sprintf(buf, "T%ld 0 %ld 0\n", stb.st_mtime,
        !           463:                            stb.st_atime);
        !           464:                        (void)write(rem, buf, strlen(buf));
        !           465:                        if (response() < 0) {
        !           466:                                (void)close(f);
        !           467:                                continue;
        !           468:                        }
        !           469:                }
        !           470:                (void)sprintf(buf, "C%04o %ld %s\n", stb.st_mode&07777,
        !           471:                    stb.st_size, last);
        !           472:                (void)write(rem, buf, strlen(buf));
        !           473:                if (response() < 0) {
        !           474:                        (void)close(f);
        !           475:                        continue;
        !           476:                }
        !           477:                if ((bp = allocbuf(&buffer, f, BUFSIZ)) == 0) {
        !           478:                        (void)close(f);
        !           479:                        continue;
        !           480:                }
        !           481:                readerr = 0;
        !           482:                for (i = 0; i < stb.st_size; i += bp->cnt) {
        !           483:                        amt = bp->cnt;
        !           484:                        if (i + amt > stb.st_size)
        !           485:                                amt = stb.st_size - i;
        !           486:                        if (readerr == 0 && read(f, bp->buf, amt) != amt)
        !           487:                                readerr = errno;
        !           488:                        (void)write(rem, bp->buf, amt);
        !           489:                }
        !           490:                (void)close(f);
        !           491:                if (readerr == 0)
        !           492:                        (void)write(rem, "", 1);
        !           493:                else
        !           494:                        error("rcp: %s: %s\n", name, strerror(readerr));
        !           495:                (void)response();
        !           496:        }
        !           497: }
        !           498: 
        !           499: rsource(name, statp)
        !           500:        char *name;
        !           501:        struct stat *statp;
        !           502: {
        !           503:        DIR *d;
        !           504:        struct direct *dp;
        !           505:        char *last, *vect[1], path[MAXPATHLEN];
        !           506: 
        !           507:        if (!(d = opendir(name))) {
        !           508:                error("rcp: %s: %s\n", name, strerror(errno));
        !           509:                return;
        !           510:        }
        !           511:        last = rindex(name, '/');
        !           512:        if (last == 0)
        !           513:                last = name;
        !           514:        else
        !           515:                last++;
        !           516:        if (pflag) {
        !           517:                (void)sprintf(path, "T%ld 0 %ld 0\n", statp->st_mtime,
        !           518:                    statp->st_atime);
        !           519:                (void)write(rem, path, strlen(path));
        !           520:                if (response() < 0) {
        !           521:                        closedir(d);
        !           522:                        return;
        !           523:                }
        !           524:        }
        !           525:        (void)sprintf(path, "D%04o %d %s\n", statp->st_mode&07777, 0, last);
        !           526:        (void)write(rem, path, strlen(path));
        !           527:        if (response() < 0) {
        !           528:                closedir(d);
        !           529:                return;
        !           530:        }
        !           531:        while (dp = readdir(d)) {
        !           532:                if (dp->d_ino == 0)
        !           533:                        continue;
        !           534:                if (!strcmp(dp->d_name, ".") || !strcmp(dp->d_name, ".."))
        !           535:                        continue;
        !           536:                if (strlen(name) + 1 + strlen(dp->d_name) >= MAXPATHLEN - 1) {
        !           537:                        error("%s/%s: name too long.\n", name, dp->d_name);
        !           538:                        continue;
        !           539:                }
        !           540:                (void)sprintf(path, "%s/%s", name, dp->d_name);
        !           541:                vect[0] = path;
        !           542:                source(1, vect);
        !           543:        }
        !           544:        closedir(d);
        !           545:        (void)write(rem, "E\n", 2);
        !           546:        (void)response();
        !           547: }
        !           548: 
        !           549: response()
        !           550: {
        !           551:        register char *cp;
        !           552:        char ch, resp, rbuf[BUFSIZ];
        !           553: 
        !           554:        if (read(rem, &resp, sizeof(resp)) != sizeof(resp))
        !           555:                lostconn();
        !           556: 
        !           557:        cp = rbuf;
        !           558:        switch(resp) {
        !           559:        case 0:                         /* ok */
        !           560:                return(0);
        !           561:        default:
        !           562:                *cp++ = resp;
        !           563:                /* FALLTHROUGH */
        !           564:        case 1:                         /* error, followed by err msg */
        !           565:        case 2:                         /* fatal error, "" */
        !           566:                do {
        !           567:                        if (read(rem, &ch, sizeof(ch)) != sizeof(ch))
        !           568:                                lostconn();
        !           569:                        *cp++ = ch;
        !           570:                } while (cp < &rbuf[BUFSIZ] && ch != '\n');
        !           571: 
        !           572:                if (!iamremote)
        !           573:                        (void)write(2, rbuf, cp - rbuf);
        !           574:                ++errs;
        !           575:                if (resp == 1)
        !           576:                        return(-1);
        !           577:                exit(1);
        !           578:        }
        !           579:        /*NOTREACHED*/
        !           580: }
        !           581: 
        !           582: lostconn()
        !           583: {
        !           584:        if (!iamremote)
        !           585:                (void)fprintf(stderr, "rcp: lost connection\n");
        !           586:        exit(1);
        !           587: }
        !           588: 
        !           589: sink(argc, argv)
        !           590:        int argc;
        !           591:        char **argv;
        !           592: {
        !           593:        register char *cp;
        !           594:        static BUF buffer;
        !           595:        struct stat stb;
        !           596:        struct timeval tv[2];
        !           597:        enum { YES, NO, DISPLAYED } wrerr;
        !           598:        BUF *bp, *allocbuf();
        !           599:        off_t i, j;
        !           600:        char ch, *targ, *why;
        !           601:        int amt, count, exists, first, mask, mode;
        !           602:        int ofd, setimes, size, targisdir;
        !           603:        char *np, *vect[1], buf[BUFSIZ], *malloc();
        !           604: 
        !           605: #define        atime   tv[0]
        !           606: #define        mtime   tv[1]
        !           607: #define        SCREWUP(str)    { why = str; goto screwup; }
        !           608: 
        !           609:        setimes = targisdir = 0;
        !           610:        mask = umask(0);
        !           611:        if (!pflag)
        !           612:                (void)umask(mask);
        !           613:        if (argc != 1) {
        !           614:                error("rcp: ambiguous target\n");
        !           615:                exit(1);
        !           616:        }
        !           617:        targ = *argv;
        !           618:        if (targetshouldbedirectory)
        !           619:                verifydir(targ);
        !           620:        (void)write(rem, "", 1);
        !           621:        if (stat(targ, &stb) == 0 && (stb.st_mode & S_IFMT) == S_IFDIR)
        !           622:                targisdir = 1;
        !           623:        for (first = 1;; first = 0) {
        !           624:                cp = buf;
        !           625:                if (read(rem, cp, 1) <= 0)
        !           626:                        return;
        !           627:                if (*cp++ == '\n')
        !           628:                        SCREWUP("unexpected <newline>");
        !           629:                do {
        !           630:                        if (read(rem, &ch, sizeof(ch)) != sizeof(ch))
        !           631:                                SCREWUP("lost connection");
        !           632:                        *cp++ = ch;
        !           633:                } while (cp < &buf[BUFSIZ - 1] && ch != '\n');
        !           634:                *cp = 0;
        !           635: 
        !           636:                if (buf[0] == '\01' || buf[0] == '\02') {
        !           637:                        if (iamremote == 0)
        !           638:                                (void)write(2, buf + 1, strlen(buf + 1));
        !           639:                        if (buf[0] == '\02')
        !           640:                                exit(1);
        !           641:                        errs++;
        !           642:                        continue;
        !           643:                }
        !           644:                if (buf[0] == 'E') {
        !           645:                        (void)write(rem, "", 1);
        !           646:                        return;
        !           647:                }
        !           648: 
        !           649:                if (ch == '\n')
        !           650:                        *--cp = 0;
        !           651: 
        !           652: #define getnum(t) (t) = 0; while (isdigit(*cp)) (t) = (t) * 10 + (*cp++ - '0');
        !           653:                cp = buf;
        !           654:                if (*cp == 'T') {
        !           655:                        setimes++;
        !           656:                        cp++;
        !           657:                        getnum(mtime.tv_sec);
        !           658:                        if (*cp++ != ' ')
        !           659:                                SCREWUP("mtime.sec not delimited");
        !           660:                        getnum(mtime.tv_usec);
        !           661:                        if (*cp++ != ' ')
        !           662:                                SCREWUP("mtime.usec not delimited");
        !           663:                        getnum(atime.tv_sec);
        !           664:                        if (*cp++ != ' ')
        !           665:                                SCREWUP("atime.sec not delimited");
        !           666:                        getnum(atime.tv_usec);
        !           667:                        if (*cp++ != '\0')
        !           668:                                SCREWUP("atime.usec not delimited");
        !           669:                        (void)write(rem, "", 1);
        !           670:                        continue;
        !           671:                }
        !           672:                if (*cp != 'C' && *cp != 'D') {
        !           673:                        /*
        !           674:                         * Check for the case "rcp remote:foo\* local:bar".
        !           675:                         * In this case, the line "No match." can be returned
        !           676:                         * by the shell before the rcp command on the remote is
        !           677:                         * executed so the ^Aerror_message convention isn't
        !           678:                         * followed.
        !           679:                         */
        !           680:                        if (first) {
        !           681:                                error("%s\n", cp);
        !           682:                                exit(1);
        !           683:                        }
        !           684:                        SCREWUP("expected control record");
        !           685:                }
        !           686:                mode = 0;
        !           687:                for (++cp; cp < buf + 5; cp++) {
        !           688:                        if (*cp < '0' || *cp > '7')
        !           689:                                SCREWUP("bad mode");
        !           690:                        mode = (mode << 3) | (*cp - '0');
        !           691:                }
        !           692:                if (*cp++ != ' ')
        !           693:                        SCREWUP("mode not delimited");
        !           694:                size = 0;
        !           695:                while (isdigit(*cp))
        !           696:                        size = size * 10 + (*cp++ - '0');
        !           697:                if (*cp++ != ' ')
        !           698:                        SCREWUP("size not delimited");
        !           699:                if (targisdir) {
        !           700:                        static char *namebuf;
        !           701:                        static int cursize;
        !           702:                        int need;
        !           703: 
        !           704:                        need = strlen(targ) + strlen(cp) + 250;
        !           705:                        if (need > cursize) {
        !           706:                                if (!(namebuf = malloc((u_int)need)))
        !           707:                                        error("out of memory\n");
        !           708:                        }
        !           709:                        (void)sprintf(namebuf, "%s%s%s", targ,
        !           710:                            *targ ? "/" : "", cp);
        !           711:                        np = namebuf;
        !           712:                }
        !           713:                else
        !           714:                        np = targ;
        !           715:                exists = stat(np, &stb) == 0;
        !           716:                if (buf[0] == 'D') {
        !           717:                        if (exists) {
        !           718:                                if ((stb.st_mode&S_IFMT) != S_IFDIR) {
        !           719:                                        errno = ENOTDIR;
        !           720:                                        goto bad;
        !           721:                                }
        !           722:                                if (pflag)
        !           723:                                        (void)chmod(np, mode);
        !           724:                        } else if (mkdir(np, mode) < 0)
        !           725:                                goto bad;
        !           726:                        vect[0] = np;
        !           727:                        sink(1, vect);
        !           728:                        if (setimes) {
        !           729:                                setimes = 0;
        !           730:                                if (utimes(np, tv) < 0)
        !           731:                                    error("rcp: can't set times on %s: %s\n",
        !           732:                                        np, strerror(errno));
        !           733:                        }
        !           734:                        continue;
        !           735:                }
        !           736:                if ((ofd = open(np, O_WRONLY|O_CREAT, mode)) < 0) {
        !           737: bad:                   error("rcp: %s: %s\n", np, strerror(errno));
        !           738:                        continue;
        !           739:                }
        !           740:                if (exists && pflag)
        !           741:                        (void)fchmod(ofd, mode);
        !           742:                (void)write(rem, "", 1);
        !           743:                if ((bp = allocbuf(&buffer, ofd, BUFSIZ)) == 0) {
        !           744:                        (void)close(ofd);
        !           745:                        continue;
        !           746:                }
        !           747:                cp = bp->buf;
        !           748:                count = 0;
        !           749:                wrerr = NO;
        !           750:                for (i = 0; i < size; i += BUFSIZ) {
        !           751:                        amt = BUFSIZ;
        !           752:                        if (i + amt > size)
        !           753:                                amt = size - i;
        !           754:                        count += amt;
        !           755:                        do {
        !           756:                                j = read(rem, cp, amt);
        !           757:                                if (j <= 0) {
        !           758:                                        error("rcp: %s\n",
        !           759:                                            j ? strerror(errno) :
        !           760:                                            "dropped connection");
        !           761:                                        exit(1);
        !           762:                                }
        !           763:                                amt -= j;
        !           764:                                cp += j;
        !           765:                        } while (amt > 0);
        !           766:                        if (count == bp->cnt) {
        !           767:                                if (wrerr == NO &&
        !           768:                                    write(ofd, bp->buf, count) != count)
        !           769:                                        wrerr = YES;
        !           770:                                count = 0;
        !           771:                                cp = bp->buf;
        !           772:                        }
        !           773:                }
        !           774:                if (count != 0 && wrerr == NO &&
        !           775:                    write(ofd, bp->buf, count) != count)
        !           776:                        wrerr = YES;
        !           777:                if (ftruncate(ofd, size)) {
        !           778:                        error("rcp: can't truncate %s: %s\n", np,
        !           779:                            strerror(errno));
        !           780:                        wrerr = DISPLAYED;
        !           781:                }
        !           782:                (void)close(ofd);
        !           783:                (void)response();
        !           784:                if (setimes && wrerr == NO) {
        !           785:                        setimes = 0;
        !           786:                        if (utimes(np, tv) < 0) {
        !           787:                                error("rcp: can't set times on %s: %s\n",
        !           788:                                    np, strerror(errno));
        !           789:                                wrerr = DISPLAYED;
        !           790:                        }
        !           791:                }
        !           792:                switch(wrerr) {
        !           793:                case YES:
        !           794:                        error("rcp: %s: %s\n", np, strerror(errno));
        !           795:                        break;
        !           796:                case NO:
        !           797:                        (void)write(rem, "", 1);
        !           798:                        break;
        !           799:                case DISPLAYED:
        !           800:                        break;
        !           801:                }
        !           802:        }
        !           803: screwup:
        !           804:        error("rcp: protocol screwup: %s\n", why);
        !           805:        exit(1);
        !           806: }
        !           807: 
        !           808: BUF *
        !           809: allocbuf(bp, fd, blksize)
        !           810:        BUF *bp;
        !           811:        int fd, blksize;
        !           812: {
        !           813:        struct stat stb;
        !           814:        int size;
        !           815:        char *malloc();
        !           816: 
        !           817:        if (fstat(fd, &stb) < 0) {
        !           818:                error("rcp: fstat: %s\n", strerror(errno));
        !           819:                return(0);
        !           820:        }
        !           821:        size = roundup(stb.st_blksize, blksize);
        !           822:        if (size == 0)
        !           823:                size = blksize;
        !           824:        if (bp->cnt < size) {
        !           825:                if (bp->buf != 0)
        !           826:                        free(bp->buf);
        !           827:                bp->buf = (char *)malloc((u_int)size);
        !           828:                if (!bp->buf) {
        !           829:                        error("rcp: malloc: out of memory\n");
        !           830:                        return(0);
        !           831:                }
        !           832:        }
        !           833:        bp->cnt = size;
        !           834:        return(bp);
        !           835: }
        !           836: 
        !           837: /* VARARGS1 */
        !           838: error(fmt, a1, a2, a3)
        !           839:        char *fmt;
        !           840:        int a1, a2, a3;
        !           841: {
        !           842:        static FILE *fp;
        !           843: 
        !           844:        ++errs;
        !           845:        if (!fp && !(fp = fdopen(rem, "w")))
        !           846:                return;
        !           847:        (void)fprintf(fp, "%c", 0x01);
        !           848:        (void)fprintf(fp, fmt, a1, a2, a3);
        !           849:        (void)fflush(fp);
        !           850:        if (!iamremote)
        !           851:                (void)fprintf(stderr, fmt, a1, a2, a3);
        !           852: }
        !           853: 
        !           854: nospace()
        !           855: {
        !           856:        (void)fprintf(stderr, "rcp: out of memory.\n");
        !           857:        exit(1);
        !           858: }
        !           859: 
        !           860: 
        !           861: usage()
        !           862: {
        !           863: #ifdef KERBEROS
        !           864:        (void)fprintf(stderr, "%s\n\t%s\n",
        !           865:            "usage: rcp [-k realm] [-px] f1 f2",
        !           866:            "or: rcp [-k realm] [-rpx] f1 ... fn directory");
        !           867: #else
        !           868:        (void)fprintf(stderr,
        !           869:            "usage: rcp [-p] f1 f2; or: rcp [-rp] f1 ... fn directory\n");
        !           870: #endif
        !           871:        exit(1);
        !           872: }
        !           873: 
        !           874: #ifdef KERBEROS
        !           875: old_warning(str)
        !           876:        char *str;
        !           877: {
        !           878:        (void)fprintf(stderr, "rcp: warning: %s, using standard rcp\n", str);
        !           879: }
        !           880: 
        !           881: int
        !           882: kerberos(host, bp, locuser, user)
        !           883: 
        !           884:        char **host, *bp, *locuser, *user;
        !           885: {
        !           886:        struct servent *sp;
        !           887: 
        !           888: again:
        !           889:        if (use_kerberos) {
        !           890:                rem = KSUCCESS;
        !           891:                errno = 0;
        !           892:                if (dest_realm == NULL)
        !           893:                        dest_realm = krb_realmofhost(*host);
        !           894: 
        !           895:                if (encrypt)
        !           896:                        rem = krcmd_mutual(
        !           897:                                host, port,
        !           898:                                user, bp, 0,
        !           899:                                dest_realm,
        !           900:                                &cred, schedule);
        !           901:                else
        !           902:                        rem = krcmd(
        !           903:                                host, port,
        !           904:                                user, bp, 0, dest_realm);
        !           905: 
        !           906:                if (rem < 0) {
        !           907:                        use_kerberos = 0;
        !           908:                        sp = getservbyname("shell", "tcp");
        !           909:                        if (sp == NULL) {
        !           910:                                (void)fprintf(stderr,
        !           911:                                    "rcp: unknown service shell/tcp\n");
        !           912:                                exit(1);
        !           913:                        }
        !           914:                        if (errno == ECONNREFUSED)
        !           915:                                old_warning(
        !           916:                                    "remote host doesn't support Kerberos");
        !           917: 
        !           918:                        if (errno == ENOENT)
        !           919:                                old_warning(
        !           920:                                    "Can't provide Kerberos auth data");
        !           921:                        port = sp->s_port;
        !           922:                        goto again;
        !           923:                }
        !           924:        } else {
        !           925:                if (encrypt) {
        !           926:                        fprintf(stderr,
        !           927:                            "The -x option requires Kerberos authentication\n");
        !           928:                        exit(1);
        !           929:                }
        !           930:                rem = rcmd(host, sp->s_port, locuser, user, bp, 0);
        !           931:        }
        !           932:        return(rem);
        !           933: }
        !           934: #endif /* KERBEROS */

unix.superglobalmegacorp.com

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