/* $NetBSD: supfilesrv.c,v 1.20 2001/01/16 02:50:32 cgd Exp $ */ /* * Copyright (c) 1992 Carnegie Mellon University * All Rights Reserved. * * Permission to use, copy, modify and distribute this software and its * documentation is hereby granted, provided that both the copyright * notice and this permission notice appear in all copies of the * software, derivative works or modified versions, and any portions * thereof, and that both notices appear in supporting documentation. * * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS" * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE. * * Carnegie Mellon requests users of this software to return to * * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU * School of Computer Science * Carnegie Mellon University * Pittsburgh PA 15213-3890 * * any improvements or extensions that they make and grant Carnegie Mellon * the rights to redistribute these changes. * */ /* * supfilesrv -- SUP File Server * * Usage: supfilesrv [-d] [-l] [-P] [-N] [-R] [-S] * -d "debug" -- don't fork daemon * -l "log" -- print successull connects (when compiled with libwrap) * -P "debug ports" -- use debugging network ports * -N "debug network" -- print debugging messages for network i/o * -R "RCS mode" -- if file is an rcs file, use co to get contents * -S "Operate silently" -- Only print error messages * ********************************************************************** * HISTORY * 2-Aug-99 Manuel Bouyer at LIP6 * Added libwrap support * * 13-Sep-92 Mary Thompson (mrt) at Carnegie-Mellon University * Changed name of sup program in xpatch from /usr/cs/bin/sup to * /usr/bin/sup for exported version of sup. * * 7-July-93 Nate Williams at Montana State University * Modified SUP to use gzip based compression when sending files * across the network to save BandWidth * * Revision 1.20 92/09/09 22:05:00 mrt * Added Brad's change to make sendfile take a va_list. * Added support in login to accept an non-encrypted login * message if no user or password is being sent. This supports * a non-crypting version of sup. Also fixed to skip leading * white space from crypts in host files. * [92/09/01 mrt] * * Revision 1.19 92/08/11 12:07:59 mrt * Made maxchildren a patchable variable, which can be set by the * command line switch -C or else defaults to the MAXCHILDREN * defined in sup.h. Added most of Brad's STUMP changes. * Increased PGMVERSION to 12 to reflect substantial changes. * [92/07/28 mrt] * * Revision 1.18 90/12/25 15:15:39 ern * Yet another rewrite of the logging code. Make up the text we will write * and then get in, write it and get out. * Also set error on write-to-full-disk if the logging is for recording * server is busy. * [90/12/25 15:15:15 ern] * * Revision 1.17 90/05/07 09:31:13 dlc * Sigh, some more fixes to the new "crypt" file handling code. First, * just because the "crypt" file is in a local file system does not mean * it can be trusted. We have to check for hard links to root owned * files whose contents could be interpretted as a crypt key. For * checking this fact, the new routine stat_info_ok() was added. This * routine also makes other sanity checks, such as owner only permission, * the file is a regular file, etc. Also, even if the uid/gid of th * "crypt" file is not going to be used, still use its contents in order * to cause fewer surprises to people supping out of a shared file system * such as AFS. * [90/05/07 dlc] * * Revision 1.16 90/04/29 04:21:08 dlc * Fixed logic bug in docrypt() which would not get the stat information * from the crypt file if the crypt key had already been set from a * "host" file. * [90/04/29 dlc] * * Revision 1.15 90/04/18 19:51:27 dlc * Added the new routines local_file(), link_nofollow() for use in * dectecting whether a file is located in a local file system. These * routines probably should have been in another module, but only * supfilesrv needs to do the check and none of its other modules seemed * appropriate. Note, the implementation should be changed once we have * direct kernel support, for example the fstatfs(2) system call, for * detecting the type of file system a file resides. Also, I changed * the routines which read the crosspatch crypt file or collection crypt * file to save the uid and gid from the stat information obtained via * the local_file() call (when the file is local) at the same time the * crypt key is read. This change disallows non-local files for the * crypt key to plug a security hole involving the usage of the uid/gid * of the crypt file to define who the the file server should run as. If * the saved uid/gid are both valid, then the server will set its uid/gid * to these values. * [90/04/18 dlc] * * Revision 1.14 89/08/23 14:56:15 gm0w * Changed msgf routines to msg routines. * [89/08/23 gm0w] * * Revision 1.13 89/08/03 19:57:33 mja * Remove setaid() call. * * Revision 1.12 89/08/03 19:49:24 mja * Updated to use v*printf() in place of _doprnt(). * [89/04/19 mja] * * 11-Sep-88 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added code to record release name in logfile. * * 18-Mar-88 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added host= support to releases file. [V7.12] * * 27-Dec-87 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added crosspatch support. Created docrypt() routine for crypt * test message. * * 09-Sep-87 Glenn Marcy (gm0w) at Carnegie-Mellon University * Removed common information logging code, the quiet switch, and * moved samehost() check to after device/inode check. * * 28-Jun-87 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added code for "release" support. [V5.11] * * 26-May-87 Doug Philips (dwp) at Carnegie-Mellon University * Added code to record final status of client in logfile. [V5.10] * * 22-May-87 Chriss Stephens (chriss) at Carnegie Mellon University * Mergered divergent CS and ECE versions. [V5.9a] * * 20-May-87 Glenn Marcy (gm0w) at Carnegie-Mellon University * Removed support for version 3 of SUP protocol. Added changes * to make lint happy. Added calls to new logging routines. [V5.9] * * 31-Mar-87 Dan Nydick (dan) at Carnegie-Mellon University * Fixed so no password check is done when crypts are used. * * 25-Nov-86 Rudy Nedved (ern) at Carnegie-Mellon University * Set F_APPEND fcntl in logging to increase the chance * that the log entry from this incarnation of the file * server will not be lost by another incarnation. [V5.8] * * 20-Oct-86 Dan Nydick (dan) at Carnegie-Mellon University * Changed not to call okmumbles when not compiled with CMUCS. * * 04-Aug-86 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added code to increment scmdebug as more -N flags are * added. [V5.7] * * 25-May-86 Jonathan J. Chew (jjc) at Carnegie-Mellon University * Renamed local variable in main program from "sigmask" to * "signalmask" to avoid name conflict with 4.3BSD identifier. * Conditionally compile in calls to CMU routines, "setaid" and * "logaccess". [V5.6] * * 21-Jan-86 Glenn Marcy (gm0w) at Carnegie-Mellon University * Changed supfilesrv to use the crypt file owner and group for * access purposes, rather than the directory containing the crypt * file. [V5.5] * * 07-Jan-86 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added code to keep logfiles in repository collection directory. * Added code for locking collections. [V5.4] * * 05-Jan-86 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added code to support new FSETUPBUSY return. Now accepts all * connections and tells any clients after the 8th that the * fileserver is busy. New clients will retry again later. [V5.3] * * 29-Dec-85 Glenn Marcy (gm0w) at Carnegie-Mellon University * Major rewrite for protocol version 4. [V4.2] * * 12-Dec-85 Glenn Marcy (gm0w) at Carnegie-Mellon University * Fixed close of crypt file to use file pointer as argument * instead of string pointer. * * 24-Nov-85 Glenn Marcy (gm0w) at Carnegie-Mellon University * Allow "!hostname" lines and comments in collection "host" file. * * 13-Nov-85 Glenn Marcy (gm0w) at Carnegie-Mellon University * Don't use access() on symbolic links since they may not point to * an existing file. * * 22-Oct-85 Glenn Marcy (gm0w) at Carnegie-Mellon University * Added code to restrict file server availability to when it has * less than or equal to eight children. * * 22-Sep-85 Glenn Marcy (gm0w) at Carnegie-Mellon University * Merged 4.1 and 4.2 versions together. * * 04-Jun-85 Steven Shafer (sas) at Carnegie-Mellon University * Created for 4.2 BSD. * ********************************************************************** */ #include #ifdef AFS #include #undef MAXNAMLEN #endif #include #include #include #include #include #include #include #include #ifdef __STDC__ #include #else #include #endif #include #include #include #include #include #include #ifndef HAS_POSIX_DIR #include #else #include #endif #if MACH #include #endif #if CMUCS #include #include #include #include #else /* CMUCS */ #define ACCESS_CODE_OK 0 #define ACCESS_CODE_BADPASSWORD (-2) #endif /* CMUCS */ #ifdef __SVR4 # include # include #endif #ifdef LIBWRAP #include #endif #include "supcdefs.h" #include "supextern.h" #define MSGFILE #include "supmsg.h" int maxchildren; /* * These are used to save the stat information from the crosspatch crypt * file or collection crypt file at the time it is opened for the crypt * key and it is verified to be a local file. */ int runas_uid = -1; int runas_gid = -1; #define PGMVERSION 13 /************************* *** M A C R O S *** *************************/ #define HASHBITS 8 #define HASHSIZE (1< 0) nchildren++; (void) sigprocmask(SIG_SETMASK, &oset, NULL); } } /* * Child status signal handler */ void chldsig(snum) int snum; { int w; while (wait3((int *) &w, WNOHANG, (struct rusage *)0) > 0) { if (nchildren) nchildren--; } } /***************************************** *** I N I T I A L I Z A T I O N *** *****************************************/ void usage () { #ifdef LIBWRAP quit (1,"Usage: supfilesrv [ -l | -d | -P | -N | -C | -H ]\n"); #else quit (1,"Usage: supfilesrv [ -d | -P | -N | -C | -H ]\n"); #endif } void init (argc,argv) int argc; char **argv; { register int i; register int x; char *clienthost,*clientuser; char *p,*q; char buf[STRINGLENGTH]; int maxsleep; register FILE *f; #ifdef RCS candorcs = FALSE; #endif live = FALSE; #ifdef LIBWRAP clog = FALSE; #endif dbgportsq = FALSE; scmdebug = 0; clienthost = NULL; clientuser = NULL; maxsleep = 5; if (--argc < 0) usage (); argv++; while (clienthost == NULL && argc > 0 && argv[0][0] == '-') { switch (argv[0][1]) { case 'S': silent = TRUE; break; #ifdef LIBWRAP case 'l': clog = TRUE; break; #endif case 'd': live = TRUE; break; case 'P': dbgportsq = TRUE; break; case 'N': scmdebug++; break; case 'C': if (--argc < 1) quit (1,"Missing arg to -C\n"); argv++; maxchildren = atoi(argv[0]); break; case 'H': if (--argc < 3) quit (1,"Missing args to -H\n"); argv++; clienthost = argv[0]; clientuser = argv[1]; cryptkey = argv[2]; argc -= 2; argv += 2; break; #ifdef RCS case 'R': candorcs = TRUE; break; #endif default: fprintf (stderr,"Unknown flag %s ignored\n",argv[0]); break; } --argc; argv++; } if (clienthost == NULL) { if (argc != 0) usage (); x = servicesetup (dbgportsq ? DEBUGFPORT : FILEPORT); if (x != SCMOK) quit (1,"Error in network setup"); for (i = 0; i < HASHSIZE; i++) uidH[i] = gidH[i] = inodeH[i] = NULL; return; } server = FALSE; if (argc < 1) usage (); f = fopen (cryptkey,"r"); if (f == NULL) quit (1,"Unable to open cryptfile %s\n",cryptkey); if ((p = fgets (buf,STRINGLENGTH,f)) != NULL) { if ((q = index (p,'\n')) != NULL) *q = '\0'; if (*p == '\0') quit (1,"No cryptkey found in %s\n",cryptkey); cryptkey = salloc (buf); } (void) fclose (f); x = request (dbgportsq ? DEBUGFPORT : FILEPORT,clienthost,&maxsleep); if (x != SCMOK) quit (1,"Unable to connect to host %s\n",clienthost); x = msgsignon (); if (x != SCMOK) quit (1,"Error sending signon request to fileserver\n"); x = msgsignonack (); if (x != SCMOK) quit (1,"Error reading signon reply from fileserver\n"); printf ("SUP Fileserver %d.%d (%s) %d on %s\n", protver,pgmver,scmver,fspid,remotehost()); free (scmver); scmver = NULL; if (protver < 7) quit (1,"Remote fileserver does not implement reverse sup\n"); xpatch = TRUE; xuser = clientuser; x = msgsetup (); if (x != SCMOK) quit (1,"Error sending setup request to fileserver\n"); x = msgsetupack (); if (x != SCMOK) quit (1,"Error reading setup reply from fileserver\n"); switch (setupack) { case FSETUPOK: break; case FSETUPSAME: quit (1,"User %s not found on remote client\n",xuser); case FSETUPHOST: quit (1,"This host has no permission to reverse sup\n"); default: quit (1,"Unrecognized file server setup status %d\n",setupack); } if (netcrypt (cryptkey) != SCMOK ) quit (1,"Running non-crypting fileserver\n"); crypttest = CRYPTTEST; x = msgcrypt (); if (x != SCMOK) quit (1,"Error sending encryption test request\n"); x = msgcryptok (); if (x == SCMEOF) quit (1,"Data encryption test failed\n"); if (x != SCMOK) quit (1,"Error reading encryption test reply\n"); logcrypt = CRYPTTEST; loguser = NULL; logpswd = NULL; if (netcrypt (PSWDCRYPT) != SCMOK) /* encrypt password data */ quit (1,"Running non-crypting fileserver\n"); x = msglogin (); (void) netcrypt ((char *)NULL); /* turn off encryption */ if (x != SCMOK) quit (1,"Error sending login request to file server\n"); x = msglogack (); if (x != SCMOK) quit (1,"Error reading login reply from file server\n"); if (logack == FLOGNG) quit (1,"%s\nImproper login to %s account\n",logerror,xuser); xargc = argc; xargv = argv; x = msgxpatch (); if (x != SCMOK) quit (1,"Error sending crosspatch request\n"); crosspatch (); exit (0); } /***************************************** *** A N S W E R R E Q U E S T *** *****************************************/ void answer () { time_t starttime; register int x; progpid = fspid = getpid (); collname = NULL; basedir = NULL; prefix = NULL; release = NULL; rcs_branch = NULL; #ifdef CVS cvs_root = NULL; #endif goawayreason = NULL; donereason = NULL; lockfd = -1; starttime = time ((time_t *)NULL); if (!setjmp (sjbuf)) { srvsignon (); srvsetup (); docrypt (); srvlogin (); if (xpatch) { int fd; x = msgxpatch (); if (x != SCMOK) exit (0); xargv[0] = "sup"; xargv[1] = "-X"; xargv[xargc] = (char *)NULL; (void) dup2 (netfile,0); (void) dup2 (netfile,1); (void) dup2 (netfile,2); fd = getdtablesize (); while (--fd > 2) (void) close (fd); execvp (xargv[0],xargv); exit (0); } listfiles (); sendfiles (); } srvfinishup (starttime); if (collname) free (collname); if (basedir) free (basedir); if (prefix) free (prefix); if (release) free (release); if (rcs_branch) free (rcs_branch); #ifdef CVS if (cvs_root) free (cvs_root); #endif if (goawayreason) { if (donereason == goawayreason) donereason = NULL; free (goawayreason); } if (donereason) free (donereason); if (lockfd >= 0) (void) close (lockfd); endpwent (); (void) endgrent (); #if CMUCS endacent (); #endif /* CMUCS */ Hfree (uidH); Hfree (gidH); Hfree (inodeH); } /***************************************** *** S I G N O N C L I E N T *** *****************************************/ void srvsignon () { register int x; xpatch = FALSE; x = msgsignon (); if (x != SCMOK) goaway ("Error reading signon request from client"); x = msgsignonack (); if (x != SCMOK) goaway ("Error sending signon reply to client"); free (scmver); scmver = NULL; } /***************************************************************** *** E X C H A N G E S E T U P I N F O R M A T I O N *** *****************************************************************/ void srvsetup () { register int x; char *p,*q; char buf[STRINGLENGTH]; register FILE *f; struct stat sbuf; register TREELIST *tl; if (protver > 7) { cancompress = TRUE; } x = msgsetup (); if (x != SCMOK) goaway ("Error reading setup request from client"); if (protver < 4) { setupack = FSETUPOLD; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Sup client using obsolete version of protocol"); } if (xpatch) { register struct passwd *pw; if ((pw = getpwnam (xuser)) == NULL) { setupack = FSETUPSAME; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("User `%s' not found", xuser); } (void) free (xuser); xuser = salloc (pw->pw_dir); /* check crosspatch host access file */ cryptkey = NULL; (void) sprintf (buf,FILEXPATCH,xuser); /* Turn off link following */ if (link_nofollow(1) != -1) { int hostok = FALSE; /* get stat info before open */ if (stat(buf, &sbuf) == -1) (void) bzero((char *)&sbuf, sizeof(sbuf)); if ((f = fopen (buf,"r")) != NULL) { struct stat fsbuf; while ((p = fgets (buf,STRINGLENGTH,f)) != NULL) { q = index (p,'\n'); if (q) *q = 0; if (index ("#;:",*p)) continue; q = nxtarg (&p," \t"); if (*p == '\0') continue; if (!matchhost(q)) continue; cryptkey = salloc (p); hostok = TRUE; if (local_file(fileno(f), &fsbuf) > 0 && stat_info_ok(&sbuf, &fsbuf)) { runas_uid = sbuf.st_uid; runas_gid = sbuf.st_gid; } break; } (void) fclose (f); } /* Restore link following */ if (link_nofollow(0) == -1) goaway ("Restore link following"); if (!hostok) { setupack = FSETUPHOST; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Host not on access list"); } } setupack = FSETUPOK; x = msgsetupack (); if (x != SCMOK) goaway ("Error sending setup reply to client"); return; } #ifdef RCS if (candorcs && release != NULL && (strncmp(release, "RCS.", 4) == 0)) { rcs_branch = salloc(&release[4]); free(release); release = salloc("RCS"); dorcs = TRUE; } #endif if (release == NULL) release = salloc (DEFRELEASE); if (basedir == NULL || *basedir == '\0') { basedir = NULL; (void) sprintf (buf,FILEDIRS,DEFDIR); f = fopen (buf,"r"); if (f) { while ((p = fgets (buf,STRINGLENGTH,f)) != NULL) { q = index (p,'\n'); if (q) *q = 0; if (index ("#;:",*p)) continue; q = nxtarg (&p," \t="); if (strcmp(q,collname) == 0) { basedir = skipover(p," \t="); basedir = salloc (basedir); break; } } (void) fclose (f); } if (basedir == NULL) { (void) sprintf (buf,FILEBASEDEFAULT,collname); basedir = salloc(buf); } } if (chdir (basedir) < 0) goaway ("Can't chdir to base directory %s",basedir); (void) sprintf (buf,FILEPREFIX,collname); f = fopen (buf,"r"); if (f) { while ((p = fgets (buf,STRINGLENGTH,f)) != NULL) { q = index (p,'\n'); if (q) *q = 0; if (index ("#;:",*p)) continue; prefix = salloc(p); if (chdir (prefix) < 0) goaway ("Can't chdir to %s from base directory %s", prefix,basedir); break; } (void) fclose (f); } x = stat (".",&sbuf); if (prefix) (void) chdir (basedir); if (x < 0) goaway ("Can't stat base/prefix directory"); if (nchildren >= maxchildren) { setupack = FSETUPBUSY; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Sup client told to try again later"); } if (sbuf.st_dev == basedev && sbuf.st_ino == baseino && samehost()) { setupack = FSETUPSAME; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Attempt to upgrade to same directory on same host"); } /* obtain release information */ if (!getrelease (release)) { setupack = FSETUPRELEASE; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Invalid release information"); } /* check host access file */ cryptkey = NULL; for (tl = listTL; tl != NULL; tl = tl->TLnext) { char *h; if ((h = tl->TLhost) == NULL) h = FILEHOSTDEF; (void) sprintf (buf,FILEHOST,collname,h); f = fopen (buf,"r"); if (f) { int hostok = FALSE; while ((p = fgets (buf,STRINGLENGTH,f)) != NULL) { int not; q = index (p,'\n'); if (q) *q = 0; if (index ("#;:",*p)) continue; q = nxtarg (&p," \t"); if ((not = (*q == '!')) && *++q == '\0') q = nxtarg (&p," \t"); hostok = (not == (matchhost(q) == 0)); if (hostok) { while ((*p == ' ') || (*p == '\t')) p++; if (*p) cryptkey = salloc (p); break; } } (void) fclose (f); if (!hostok) { setupack = FSETUPHOST; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Host not on access list for %s", collname); } } } /* try to lock collection */ (void) sprintf (buf,FILELOCK,collname); #ifdef LOCK_SH x = open (buf,O_RDONLY,0); if (x >= 0) { if (flock (x,(LOCK_SH|LOCK_NB)) < 0) { (void) close (x); if (errno != EWOULDBLOCK) goaway ("Can't lock collection %s",collname); setupack = FSETUPBUSY; (void) msgsetupack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Sup client told to wait for lock"); } lockfd = x; } #endif setupack = FSETUPOK; x = msgsetupack (); if (x != SCMOK) goaway ("Error sending setup reply to client"); } void /** Test data encryption **/ docrypt () { register int x; char *p,*q; char buf[STRINGLENGTH]; register FILE *f; struct stat sbuf; if (!xpatch) { (void) sprintf (buf,FILECRYPT,collname); /* Turn off link following */ if (link_nofollow(1) != -1) { /* get stat info before open */ if (stat(buf, &sbuf) == -1) (void) bzero((char *)&sbuf, sizeof(sbuf)); if ((f = fopen (buf,"r")) != NULL) { struct stat fsbuf; if (cryptkey == NULL && (p = fgets (buf,STRINGLENGTH,f))) { if ((q = index (p,'\n')) != NULL) *q = '\0'; if (*p) cryptkey = salloc (buf); } if (local_file(fileno(f), &fsbuf) > 0 && stat_info_ok(&sbuf, &fsbuf)) { runas_uid = sbuf.st_uid; runas_gid = sbuf.st_gid; } (void) fclose (f); } /* Restore link following */ if (link_nofollow(0) == -1) goaway ("Restore link following"); } } if ( netcrypt (cryptkey) != SCMOK ) goaway ("Runing non-crypting supfilesrv"); x = msgcrypt (); if (x != SCMOK) goaway ("Error reading encryption test request from client"); (void) netcrypt ((char *)NULL); if (strcmp(crypttest,CRYPTTEST) != 0) goaway ("Client not encrypting data properly"); free (crypttest); crypttest = NULL; x = msgcryptok (); if (x != SCMOK) goaway ("Error sending encryption test reply to client"); } /*************************************************************** *** C O N N E C T T O P R O P E R A C C O U N T *** ***************************************************************/ void srvlogin () { register int x,fileuid = -1,filegid = -1; (void) netcrypt (PSWDCRYPT); /* encrypt acct name and password */ x = msglogin (); (void) netcrypt ((char *)NULL); /* turn off encryption */ if (x != SCMOK) goaway ("Error reading login request from client"); if ( logcrypt ) { if (strcmp(logcrypt,CRYPTTEST) != 0) { logack = FLOGNG; logerror = "Improper login encryption"; (void) msglogack (); goaway ("Client not encrypting login information properly"); } free (logcrypt); logcrypt = NULL; } if (loguser == NULL) { if (cryptkey) { if (runas_uid >= 0 && runas_gid >= 0) { fileuid = runas_uid; filegid = runas_gid; loguser = NULL; } else loguser = salloc (DEFUSER); } else loguser = salloc (DEFUSER); } if ((logerror = changeuid (loguser,logpswd,fileuid,filegid)) != NULL) { logack = FLOGNG; (void) msglogack (); if (protver >= 6) longjmp (sjbuf,TRUE); goaway ("Client denied login access"); } if (loguser) free (loguser); if (logpswd) free (logpswd); logack = FLOGOK; x = msglogack (); if (x != SCMOK) goaway ("Error sending login reply to client"); if (!xpatch) /* restore desired encryption */ if (netcrypt (cryptkey) != SCMOK) goaway("Running non-crypting supfilesrv"); free (cryptkey); cryptkey = NULL; } /***************************************** *** M A K E N A M E L I S T *** *****************************************/ void listfiles () { register int x; refuseT = NULL; x = msgrefuse (); if (x != SCMOK) goaway ("Error reading refuse list from client"); getscanlists (); Tfree (&refuseT); x = msglist (); if (x != SCMOK) goaway ("Error sending file list to client"); Tfree (&listT); listT = NULL; needT = NULL; x = msgneed (); if (x != SCMOK) goaway ("Error reading needed files list from client"); denyT = NULL; (void) Tprocess (needT,denyone, NULL); Tfree (&needT); x = msgdeny (); if (x != SCMOK) goaway ("Error sending denied files list to client"); Tfree (&denyT); } int denyone (t, v) register TREE *t; void *v; { register TREELIST *tl; register char *name = t->Tname; register int update = (t->Tflags&FUPDATE) != 0; struct stat sbuf; register TREE *tlink; char slinkname[STRINGLENGTH]; register int x; for (tl = listTL; tl != NULL; tl = tl->TLnext) if ((t = Tsearch (tl->TLtree,name)) != NULL) break; if (t == NULL) { (void) Tinsert (&denyT,name,FALSE); return (SCMOK); } cdprefix (tl->TLprefix); if (S_ISLNK(t->Tmode)) x = lstat(name,&sbuf); else x = stat(name,&sbuf); if (x < 0 || (sbuf.st_mode&S_IFMT) != (t->Tmode&S_IFMT)) { (void) Tinsert (&denyT,name,FALSE); return (SCMOK); } switch (t->Tmode&S_IFMT) { case S_IFLNK: if ((x = readlink (name,slinkname,STRINGLENGTH)) <= 0) { (void) Tinsert (&denyT,name,FALSE); return (SCMOK); } slinkname[x] = '\0'; (void) Tinsert (&t->Tlink,slinkname,FALSE); break; case S_IFREG: if (sbuf.st_nlink > 1 && (tlink = linkcheck (t,(int)sbuf.st_dev,(int)sbuf.st_ino))) { (void) Tinsert (&tlink->Tlink,name,FALSE); return (SCMOK); } if (update) t->Tflags |= FUPDATE; case S_IFDIR: t->Tuid = sbuf.st_uid; t->Tgid = sbuf.st_gid; break; default: (void) Tinsert (&denyT,name,FALSE); return (SCMOK); } t->Tflags |= FNEEDED; return (SCMOK); } /********************************* *** S E N D F I L E S *** *********************************/ void sendfiles () { register TREELIST *tl; register int x; /* Does the protocol support compression */ if (cancompress) { /* Check for compression on sending files */ x = msgcompress(); if ( x != SCMOK) goaway ("Error sending compression check to server"); } /* send all files */ for (tl = listTL; tl != NULL; tl = tl->TLnext) { cdprefix (tl->TLprefix); #ifdef CVS if (candorcs) { cvs_root = getcwd(NULL, 256); if (access("CVSROOT", F_OK) < 0) dorcs = FALSE; else { loginfo("is a CVSROOT \"%s\"\n", cvs_root); dorcs = TRUE; } } #endif (void) Tprocess (tl->TLtree,sendone, NULL); } /* send directories in reverse order */ for (tl = listTL; tl != NULL; tl = tl->TLnext) { cdprefix (tl->TLprefix); (void) Trprocess (tl->TLtree,senddir, NULL); } x = msgsend (); if (x != SCMOK) goaway ("Error reading receive file request from client"); upgradeT = NULL; x = msgrecv (sendfile,0); if (x != SCMOK) goaway ("Error sending file to client"); } int sendone (t, v) TREE *t; void *v; { register int x,fd; char temp_file[STRINGLENGTH]; char *av[50]; /* More than enough */ if ((t->Tflags&FNEEDED) == 0) /* only send needed files */ return (SCMOK); if (S_ISDIR(t->Tmode)) /* send no directories this pass */ return (SCMOK); x = msgsend (); if (x != SCMOK) goaway ("Error reading receive file request from client"); upgradeT = t; /* upgrade file pointer */ fd = -1; /* no open file */ if (S_ISREG(t->Tmode)) { if (!listonly && (t->Tflags&FUPDATE) == 0) { #ifdef RCS if (dorcs) { char rcs_release[STRINGLENGTH]; tmpnam(rcs_file); if (strcmp(&t->Tname[strlen(t->Tname)-2], ",v") == 0) { t->Tname[strlen(t->Tname)-2] = '\0'; ac = 0; #ifdef CVS av[ac++] = "cvs"; av[ac++] = "-d"; av[ac++] = cvs_root; av[ac++] = "-r"; av[ac++] = "-l"; av[ac++] = "-Q"; av[ac++] = "co"; av[ac++] = "-p"; if (rcs_branch != NULL) { av[ac++] = "-r"; av[ac++] = rcs_branch; } #else av[ac++] = "co"; av[ac++] = "-q"; av[ac++] = "-p"; if (rcs_branch != NULL) { sprintf(rcs_release, "-r%s", rcs_branch); av[ac++] = rcs_release; } #endif av[ac++] = t->Tname; av[ac++] = NULL; status = runio(av, NULL, rcs_file, "/dev/null"); /*loginfo("using rcs mode \n");*/ if (status < 0 || WEXITSTATUS(status)) { /* Just in case */ unlink(rcs_file); if (status < 0) { goaway ("We died trying to run cvs or rcs on %s", rcs_file); t->Tmode = 0; } else { #if 0 logerr("rcs command failed = %d\n", WEXITSTATUS(status)); #endif t->Tflags |= FUPDATE; } } else if (docompress) { tmpnam(temp_file); av[0] = "gzip"; av[1] = "-cf"; av[2] = NULL; if (runio(av, rcs_file, temp_file, NULL) != 0) { /* Just in case */ unlink(temp_file); unlink(rcs_file); goaway ("We died trying to gzip %s", rcs_file); t->Tmode = 0; } fd = open (temp_file,O_RDONLY,0); } else fd = open (rcs_file,O_RDONLY,0); } } #endif if (fd == -1) { if (docompress) { tmpnam(temp_file); av[0] = "gzip"; av[1] = "-cf"; av[2] = NULL; if (runio(av, t->Tname, temp_file, NULL) != 0) { /* Just in case */ unlink(temp_file); goaway ("We died trying to gzip %s", t->Tname); t->Tmode = 0; } fd = open (temp_file,O_RDONLY,0); } else fd = open (t->Tname,O_RDONLY,0); } if (fd < 0 && (t->Tflags&FUPDATE) == 0) t->Tmode = 0; } if (t->Tmode) { t->Tuser = salloc (uconvert (t->Tuid)); t->Tgroup = salloc (gconvert (t->Tgid)); } } x = msgrecv (sendfile,fd); if (docompress) unlink(temp_file); #ifdef RCS if (dorcs) unlink(rcs_file); #endif if (x != SCMOK) goaway ("Error sending file %s to client", t->Tname); return (SCMOK); } int senddir (t, v) TREE *t; void *v; { register int x; if ((t->Tflags&FNEEDED) == 0) /* only send needed files */ return (SCMOK); if (!S_ISDIR(t->Tmode)) /* send only directories this pass */ return (SCMOK); x = msgsend (); if (x != SCMOK) goaway ("Error reading receive file request from client"); upgradeT = t; /* upgrade file pointer */ t->Tuser = salloc (uconvert (t->Tuid)); t->Tgroup = salloc (gconvert (t->Tgid)); x = msgrecv (sendfile,0); if (x != SCMOK) goaway ("Error sending file %s to client", t->Tname); return (SCMOK); } int sendfile(t, ap) TREE *t; va_list ap; { register int x, fd; fd = va_arg(ap,int); if (!S_ISREG(t->Tmode) || listonly || (t->Tflags&FUPDATE)) return (SCMOK); x = writefile (fd); if (x != SCMOK) goaway ("Error sending file %s to client", t->Tname); (void) close (fd); return (SCMOK); } /***************************************** *** E N D C O N N E C T I O N *** *****************************************/ void srvfinishup (starttime) time_t starttime; { register int x = SCMOK; char tmpbuf[BUFSIZ], *p, lognam[STRINGLENGTH]; int logfd; time_t finishtime; char *releasename; (void) netcrypt ((char *)NULL); if (protver < 6) { if (goawayreason != NULL) free (goawayreason); goawayreason = (char *)NULL; x = msggoaway(); doneack = FDONESUCCESS; donereason = salloc ("Unknown"); } else if (goawayreason == (char *)NULL) x = msgdone (); else { doneack = FDONEGOAWAY; donereason = goawayreason; } if (x == SCMEOF || x == SCMERR) { doneack = FDONEUSRERROR; donereason = salloc ("Premature EOF on network"); } else if (x != SCMOK) { doneack = FDONESRVERROR; donereason = salloc ("Unknown SCM code"); } if (doneack == FDONEDONTLOG) return; if (donereason == NULL) donereason = salloc ("No reason"); if (doneack == FDONESRVERROR || doneack == FDONEUSRERROR) logerr ("%s", donereason); else if (doneack == FDONEGOAWAY) logerr ("GOAWAY: %s",donereason); else if (doneack != FDONESUCCESS) logerr ("Reason %d: %s",doneack,donereason); goawayreason = donereason; cdprefix ((char *)NULL); if (collname == NULL) { logerr ("NULL collection in svrfinishup"); return; } (void) sprintf (lognam,FILELOGFILE,collname); if ((logfd = open(lognam,O_APPEND|O_WRONLY,0644)) < 0) return; /* can not open file up...error */ finishtime = time ((time_t *)NULL); p = tmpbuf; (void) sprintf (p,"%s ",fmttime (lasttime)); p += strlen(p); (void) sprintf (p,"%s ",fmttime (starttime)); p += strlen(p); (void) sprintf (p,"%s ",fmttime (finishtime)); p += strlen(p); if ((releasename = release) == NULL) releasename = "UNKNOWN"; (void) sprintf (p,"%s %s %d %s\n",remotehost(),releasename, FDONESUCCESS-doneack,donereason); p += strlen(p); #if MACH /* if we are busy dont get stuck updating the disk if full */ if(setupack == FSETUPBUSY) { long l = FIOCNOSPC_ERROR; ioctl(logfd, FIOCNOSPC, &l); } #endif /* MACH */ (void) write(logfd,tmpbuf,(p - tmpbuf)); (void) close(logfd); } /*************************************************** *** H A S H T A B L E R O U T I N E S *** ***************************************************/ void Hfree (table) HASH **table; { register HASH *h; register int i; for (i = 0; i < HASHSIZE; i++) while ((h = table[i]) != NULL) { table[i] = h->Hnext; if (h->Hname) free (h->Hname); free ((char *)h); } } HASH *Hlookup (table,num1,num2) HASH **table; int num1,num2; { register HASH *h; register int hno; hno = HASHFUNC(num1,num2); for (h = table[hno]; h && (h->Hnum1 != num1 || h->Hnum2 != num2); h = h->Hnext); return (h); } void Hinsert (table,num1,num2,name,tree) HASH **table; int num1,num2; char *name; TREE *tree; { register HASH *h; register int hno; hno = HASHFUNC(num1,num2); h = (HASH *) malloc (sizeof(HASH)); h->Hnum1 = num1; h->Hnum2 = num2; h->Hname = name; h->Htree = tree; h->Hnext = table[hno]; table[hno] = h; } /********************************************* *** U T I L I T Y R O U T I N E S *** *********************************************/ TREE *linkcheck (t,d,i) TREE *t; int d,i; /* inode # and device # */ { register HASH *h; h = Hlookup (inodeH,i,d); if (h) return (h->Htree); Hinsert (inodeH,i,d,(char *)NULL,t); return ((TREE *)NULL); } char *uconvert (uid) int uid; { register struct passwd *pw; register char *p; register HASH *u; u = Hlookup (uidH,uid,0); if (u) return (u->Hname); pw = getpwuid (uid); if (pw == NULL) return (""); p = salloc (pw->pw_name); Hinsert (uidH,uid,0,p,(TREE*)NULL); return (p); } char *gconvert (gid) int gid; { register struct group *gr; register char *p; register HASH *g; g = Hlookup (gidH,gid,0); if (g) return (g->Hname); gr = getgrgid (gid); if (gr == NULL) return (""); p = salloc (gr->gr_name); Hinsert (gidH,gid,0,p,(TREE *)NULL); return (p); } char *changeuid (namep,passwordp,fileuid,filegid) char *namep,*passwordp; int fileuid,filegid; { char *group,*account,*pswdp; struct passwd *pwd; struct group *grp; #if CMUCS struct account *acc; struct ttyloc tlc; #endif /* CMUCS */ register int status = ACCESS_CODE_OK; char nbuf[STRINGLENGTH]; static char errbuf[STRINGLENGTH]; #if CMUCS int *grps; #endif /* CMUCS */ char *p = NULL; if (namep == NULL) { pwd = getpwuid (fileuid); if (pwd == NULL) { (void) sprintf (errbuf,"Reason: Unknown user id %d", fileuid); return (errbuf); } grp = getgrgid (filegid); if (grp) group = strcpy (nbuf,grp->gr_name); else group = NULL; account = NULL; pswdp = NULL; } else { (void) strcpy (nbuf,namep); account = group = index (nbuf,','); if (group != NULL) { *group++ = '\0'; account = index (group,','); if (account != NULL) { *account++ = '\0'; if (*account == '\0') account = NULL; } if (*group == '\0') group = NULL; } pwd = getpwnam (nbuf); if (pwd == NULL) { (void) sprintf (errbuf,"Reason: Unknown user %s", nbuf); return (errbuf); } if (strcmp (nbuf,DEFUSER) == 0) pswdp = NULL; else pswdp = passwordp ? passwordp : ""; #ifdef AFS if (strcmp (nbuf,DEFUSER) != 0) { char *reason; setpag(); /* set a pag */ if (ka_UserAuthenticate(pwd->pw_name, "", 0, pswdp, 1, &reason)) { (void) sprintf (errbuf,"AFS authentication failed, %s", reason); logerr ("Attempt by %s; %s", nbuf, errbuf); return (errbuf); } } #endif } if (getuid () != 0) { if (getuid () == pwd->pw_uid) return (NULL); if (strcmp (pwd->pw_name,DEFUSER) == 0) return (NULL); logerr ("Fileserver not superuser"); return ("Reason: fileserver is not running privileged"); } #if CMUCS tlc.tlc_hostid = TLC_UNKHOST; tlc.tlc_ttyid = TLC_UNKTTY; if (okaccess(pwd->pw_name,ACCESS_TYPE_SU,0,-1,tlc) != 1) status = ACCESS_CODE_DENIED; else { grp = NULL; acc = NULL; status = oklogin(pwd->pw_name,group,&account,pswdp,&pwd,&grp,&acc,&grps); if (status == ACCESS_CODE_OK) { if ((p = okpassword(pswdp,pwd->pw_name,pwd->pw_gecos)) != NULL) status = ACCESS_CODE_INSECUREPWD; } } #else /* CMUCS */ status = ACCESS_CODE_OK; if (namep && strcmp(pwd->pw_name, DEFUSER) != 0) if (strcmp(pwd->pw_passwd,(char *)crypt(pswdp,pwd->pw_passwd))) status = ACCESS_CODE_BADPASSWORD; #endif /* CMUCS */ switch (status) { case ACCESS_CODE_OK: break; case ACCESS_CODE_BADPASSWORD: p = "Reason: Invalid password"; break; #if CMUCS case ACCESS_CODE_INSECUREPWD: (void) sprintf (errbuf,"Reason: %s",p); p = errbuf; break; case ACCESS_CODE_DENIED: p = "Reason: Access denied"; break; case ACCESS_CODE_NOUSER: p = errbuf; break; case ACCESS_CODE_ACCEXPIRED: p = "Reason: Account expired"; break; case ACCESS_CODE_GRPEXPIRED: p = "Reason: Group expired"; break; case ACCESS_CODE_ACCNOTVALID: p = "Reason: Invalid account"; break; case ACCESS_CODE_MANYDEFACC: p = "Reason: User has more than one default account"; break; case ACCESS_CODE_NOACCFORGRP: p = "Reason: No account for group"; break; case ACCESS_CODE_NOGRPFORACC: p = "Reason: No group for account"; break; case ACCESS_CODE_NOGRPDEFACC: p = "Reason: No group for default account"; break; case ACCESS_CODE_NOTGRPMEMB: p = "Reason: Not member of group"; break; case ACCESS_CODE_NOTDEFMEMB: p = "Reason: Not member of default group"; break; case ACCESS_CODE_OOPS: p = "Reason: Internal error"; break; #endif /* CMUCS */ default: (void) sprintf (p = errbuf,"Reason: Status %d",status); break; } if (pwd == NULL) return (p); if (status != ACCESS_CODE_OK) { logerr ("Login failure for %s",pwd->pw_name); logerr ("%s",p); #if CMUCS logaccess (pwd->pw_name,ACCESS_TYPE_SUP,status,0,-1,tlc); #endif /* CMUCS */ return (p); } #if CMUCS if (setgroups (grps[0], &grps[1]) < 0) logerr ("setgroups: %%m"); if (setgid ((gid_t)grp->gr_gid) < 0) logerr ("setgid: %%m"); if (setuid ((uid_t)pwd->pw_uid) < 0) logerr ("setuid: %%m"); #else /* CMUCS */ if (initgroups (pwd->pw_name,pwd->pw_gid) < 0) return("Error setting group list"); if (setgid (pwd->pw_gid) < 0) logerr ("setgid: %%m"); if (setuid (pwd->pw_uid) < 0) logerr ("setuid: %%m"); #endif /* CMUCS */ return (NULL); } void #ifdef __STDC__ goaway (char *fmt,...) #else /*VARARGS*//*ARGSUSED*/ goaway (va_alist) va_dcl #endif { char buf[STRINGLENGTH]; va_list ap; #ifdef __STDC__ va_start(ap,fmt); #else register char *fmt; va_start(ap); fmt = va_arg(ap,char *); #endif (void) netcrypt ((char *)NULL); vsnprintf(buf, sizeof(buf), fmt, ap); va_end(ap); goawayreason = salloc (buf); (void) msggoaway (); logerr ("%s",buf); longjmp (sjbuf,TRUE); } char *fmttime (time) time_t time; { static char buf[STRINGLENGTH]; int len; (void) strcpy (buf,ctime (&time)); len = strlen(buf+4)-6; (void) strncpy (buf,buf+4,len); buf[len] = '\0'; return (buf); } /* * Determine whether the file referenced by the file descriptor 'handle' can * be trusted, namely is it a file resident in the local file system. * * The main method of operation is to perform operations on the file * descriptor so that an attempt to spoof the checks should fail, for * example renamimg the file from underneath us and/or changing where the * file lives from underneath us. * * returns: -1 for error, indicating that we can not tell * 0 for file is definately not local, or it is an RFS link * 1 for file is local and can be trusted * * Side effect: copies the stat information into the supplied buffer, * regardless of the type of file system the file resides. * * Currently, the cases that we try to distinguish are RFS, AFS, NFS and * UFS, where the latter is considered a trusted file. We assume that the * caller has disabled link following and will detect an attempt to access * a file through an RFS link, except in the case the the last component is * an RFS link. With link following disabled, the last component itself is * interpreted as a regular file if it is really an RFS link, so we * disallow the RFS link identified by group "symlink" and mode "IEXEC by * owner only". An AFS file is * detected by trying the VIOCIGETCELL ioctl, which is one of the few AFS * ioctls which operate on a file descriptor. Note, this AFS ioctl is * implemented in the cache manager, so the decision does not involve a * query with the AFS file server. An NFS file is detected by looking at * the major device number and seeing if it matches the known values for * MACH NSF/Sun OS 3.x or Sun OS 4.x. * * Having the fstatfs() system call would make this routine easier and * more reliable. * * Note, in order to make the checks simpler, the file referenced by the * file descriptor can not be a BSD style symlink. Even with symlink * following of the last path component disabled, the attempt to open a * file which is a symlink will succeed, so we check for the BSD symlink * file type here. Also, the link following on/off and RFS file types * are only relevant in a MACH environment. */ #ifdef AFS #include #endif #define SYMLINK_GRP 64 int local_file(handle, sinfo) int handle; struct stat *sinfo; { struct stat sb; #ifdef VIOCIGETCELL /* * dummies for the AFS ioctl */ struct ViceIoctl vdata; char cellname[512]; #endif /* VIOCIGETCELL */ if (fstat(handle, &sb) < 0) return(-1); if (sinfo != NULL) *sinfo = sb; #if CMUCS /* * If the following test succeeds, then the file referenced by * 'handle' is actually an RFS link, so we will not trust it. * See . */ if (sb.st_gid == SYMLINK_GRP && (sb.st_mode & (S_IFMT|S_IEXEC|(S_IEXEC>>3)|(S_IEXEC>>6))) == (S_IFREG|S_IEXEC)) return(0); #endif /* CMUCS */ /* * Do not trust BSD style symlinks either. */ if (S_ISLNK(sb.st_mode)) return(0); #ifdef VIOCIGETCELL /* * This is the VIOCIGETCELL ioctl, which takes an fd, not * a path name. If it succeeds, then the file is in AFS. * * On failure, ENOTTY indicates that the file was not in * AFS; all other errors are pessimistically assumed to be * a temporary AFS error. */ vdata.in_size = 0; vdata.out_size = sizeof(cellname); vdata.out = cellname; if (ioctl(handle, VIOCIGETCELL, (char *)&vdata) != -1) return(0); if (errno != ENOTTY) return(-1); #endif /* VIOCIGETCELL */ /* * Verify the file is not in NFS. * * Our current implementation and Sun OS 3.x use major device * 255 for NFS files; Sun OS 4.x seems to use 130 (I have only * determined this empirically -- DLC). Without a fstatfs() * system call, this will have to do for now. */ #ifdef __SVR4 { struct statvfs sf; if (fstatvfs(handle, &sf) == -1) return(-1); return strncmp(sf.f_basetype, "nfs", 3) != 0; } #elif defined(__NetBSD__) { struct statfs sf; if (fstatfs(handle, &sf) == -1) return(-1); return strncmp(sf.f_fstypename, "nfs", 3) != 0; } #else if (major(sb.st_dev) == 255 || major(sb.st_dev) == 130) return(0); else return(1); #endif } /* * Companion routine for ensuring that a local file can be trusted. Compare * various pieces of the stat information to make sure that the file can be * trusted. Returns true for stat information which meets the criteria * for being trustworthy. The main paranoia is to prevent a hard link to * a root owned file. Since the link could be removed after the file is * opened, a simply fstat() can not be relied upon. The two stat buffers * for comparison should come from a stat() on the file name and a following * fstat() on the open file. Some of the following checks are also an * additional level of paranoia. Also, this test will fail (correctly) if * either or both of the stat structures have all fields zeroed; typically * due to a stat() failure. */ int stat_info_ok(sb1, sb2) struct stat *sb1, *sb2; { return (sb1->st_ino == sb2->st_ino && /* Still the same file */ sb1->st_dev == sb2->st_dev && /* On the same device */ sb1->st_mode == sb2->st_mode && /* Perms (and type) same */ S_ISREG(sb1->st_mode) && /* Only allow reg files */ (sb1->st_mode & 077) == 0 && /* Owner only perms */ sb1->st_nlink == sb2->st_nlink && /* # hard links same... */ sb1->st_nlink == 1 && /* and only 1 */ sb1->st_uid == sb2->st_uid && /* owner and ... */ sb1->st_gid == sb2->st_gid && /* group unchanged */ sb1->st_mtime == sb2->st_mtime && /* Unmodified between stats */ sb1->st_ctime == sb2->st_ctime); /* Inode unchanged. Hopefully a catch-all paranoid test */ } #if MACH /* * Twiddle symbolic/RFS link following on/off. This is a no-op in a non * CMUCS/MACH environment. Also, the setmodes/getmodes interface is used * mainly because it is simpler than using table(2) directly. */ #include int link_nofollow(on) int on; { static int modes = -1; if (modes == -1 && (modes = getmodes()) == -1) return(-1); if (on) return(setmodes(modes | UMODE_NOFOLLOW)); return(setmodes(modes)); } #else /* MACH */ /*ARGSUSED*/ int link_nofollow(on) int on; { return(0); } #endif /* MACH */