V10/cmd/uucp/cpmv.c

Compare this file to the similar file:
Show the results in this format:


/*	/sccs/src/cmd/uucp/s.cpmv.c
	cpmv.c	1.4	8/30/84 17:37:15
*/
#include "uucp.h"
VERSION(@(#) c cpmv.c 1.4);

/*
 * copy f1 to f2 locally
 *	f1	-> source file name
 *	f2	-> destination file name
 * return:
 *	0	-> ok
 *	FAIL	-> failed
 */

xcp(f1, f2)
char *f1, *f2;
{
	register FILE *fp1, *fp2;
	register int n;
	char buf[BUFSIZ];
	char full[MAXFULLNAME];

	if ((fp1 = fopen(f1, "r")) == NULL)
		return(FAIL);
	(void) strcpy(full, f2);
	if (DIRECTORY(f2)) {
	    (void) strcat(full, "/");
	    (void) strcat(full, BASENAME(f1, '/'));
	    (void) strcpy(f2, full);
	}

	DEBUG(4, "full %s\n", full);
	if ((fp2 = fopen(full, "w")) == NULL) {
	    (void) fclose(fp1);
	    return(FAIL);
	}
	(void) chmod(full, 0666);

	/* copy -- check errors later */
	while ( (n = fread(buf, sizeof (char), sizeof buf, fp1)) != 0)
	    (void) fwrite(buf, sizeof (char), n, fp2);

	/* check for any errors */
	n = ferror(fp1) | ferror(fp2);
	n |= fclose(fp1) | fclose(fp2);

	if (n)
	    return(FAIL);
	return(0);
}


/*
 * move f1 to f2 locally
 * returns:
 *	0	-> ok
 *	FAIL	-> failed
 */

xmv(f1, f2)
register char *f1, *f2;
{
	register int ret;
	int oerrno = 0;

	(void) unlink(f2);   /* i'm convinced this is the right thing to do */
	if ( (ret = link(f1, f2)) < 0) {
		oerrno = errno;
	    /* copy file */
	    ret = xcp(f1, f2);
	}

	if (ret == 0)
	    (void) unlink(f1);
/*
 * ugh
 */
	if (ret < 0) {
		char buf[BUFSIZ];
		char f2base[BUFSIZ];
		struct stat s1, s2;
		char *p;
		char *strrchr();

		s1.st_mode = 0;
		s2.st_mode = 0;
		if (stat(f1, &s1) < 0)
			s1.st_mode = -1;
		strcpy(f2base, f2);
		if ((p = strrchr(f2base, '/')) != 0)
			*p = 0;
		if (stat(f2base, &s2) < 0)
			s2.st_mode = -1;
		sprintf(buf, "bad xmv: %s (%o) -> %s (%o); ln errno %d",
			f1, s1.st_mode, f2base, s2.st_mode, oerrno);
		logent(buf, "DEBUG");
	}
	return(ret);
}


/* toCorrupt - move file to CORRUPTDIR
 * return - none
 */

void
toCorrupt(file)
char *file;
{
	char corrupt[MAXFULLNAME];

	(void) sprintf(corrupt, "%s/%s", CORRUPTDIR, BASENAME(file, '/'));
	(void) link(file, corrupt);
	ASSERT(unlink(file) == 0, Ct_UNLINK, file, errno);
	return;
}

/*
 * append f1 to f2
 *	f1	-> source FILE pointer
 *	f2	-> destination FILE pointer
 * return:
 *	SUCCESS	-> ok
 *	FAIL	-> failed
 */
xfappend(fp1, fp2)
register FILE	*fp1, *fp2;
{
	register int nc;
	char	buf[BUFSIZ];

	while ((nc = fread(buf, sizeof (char), BUFSIZ, fp1)) > 0)
		(void) fwrite(buf, sizeof (char), nc, fp2);

	return(ferror(fp1) || ferror(fp2) ? FAIL : SUCCESS);
}


/*
 * copy f1 to f2 locally under uid of uid argument
 *	f1	-> source file name
 *	f2	-> destination file name
 *	Uid and Euid are global
 * return:
 *	0	-> ok
 *	FAIL	-> failed
 * NOTES:
 *  for V7 systems, flip-flop between real and effective uid is
 *  not allowed, so fork must be done.  This code will not
 *  work correctly when realuid is root on System 5 because of
 *  a bug in setuid.
 */

#ifndef uidxcp

uidxcp(f1, f2)
char *f1, *f2;
{
	int status;
	char full[MAXFULLNAME];
	register pid, rpid;

	(void) strcpy(full, f2);
	if (DIRECTORY(f2)) {
	    (void) strcat(full, "/");
	    (void) strcat(full, BASENAME(f1, '/'));
	}

	/* create full owned by uucp */
	(void) close(creat(full, 0666));
	(void) chmod(full, 0666);

	/* do file copy as read uid */
#ifndef V7
	(void) setuid(Uid);
	status = xcp(f1, full);
	(void) setuid(Euid);
	return(status);

#else

	if ((pid = vfork()) == 0) {
	    setuid(Uid);
	    _exit (xcp(f1, full));
	}
	status = 1;
	while ((rpid = wait(&status)) != pid && rpid != -1)
		;
	return (rpid == -1 ? 1 : status);
#endif
}

#endif