DataMuseum.dk

Presents historical artifacts from the history of:

DKUUG/EUUG Conference tapes

This is an automatic "excavation" of a thematic subset of
artifacts from Datamuseum.dk's BitArchive.

See our Wiki for more about DKUUG/EUUG Conference tapes

Excavated with: AutoArchaeologist - Free & Open Source Software.


top - metrics - download
Index: T c

⟦a0f0982d1⟧ TextFile

    Length: 6224 (0x1850)
    Types: TextFile
    Names: »ckdecod.c«

Derivation

└─⟦9ae75bfbd⟧ Bits:30007242 EUUGD3: Starter Kit
    └─⟦3da672b63⟧ »EurOpenD3/misc/kermit.tar.Z« 
        └─⟦126dd3ca0⟧ 
            └─⟦this⟧ »ckdecod.c« 

TextFile

#ifndef lint
static char sccsid[] = "@(#)uudecode.c	5.3-1 (Berkeley) 4/10/85";
#endif
/* modified by ajr to use checksums */
/* modified by fnf to use Keith Pyle's suggestion for compatibility */

/* Adapted by Phil Julian, SAS Institute, Inc., for use on the Data General
 * minicomputers.  #ifdefs surround host-dependent code.
 * 27 May 1987
 */

/*
 * uudecode [input]
 *
 * create the specified file, decoding as you go.
 * used with uuencode.
 */
#include <stdio.h>

#ifdef unix
#include <pwd.h>
#include <sys/types.h>
#include <sys/stat.h>
#endif

#ifdef datageneral
#include <sys_calls.h>
#include <packets/filestatus.h>         /* Used for ?GNFN */
#include <packets:normal_io.h>
#include <paru.h>
int rec_format, file_type;
struct p_nio_ex w_io_parms;             /* ?write system call structure */
int chanout;                            /* Output channel */
#ifdef putc
#undef putc
#endif
#define putc(c,file) { char ch = (char) (c); dg_binw(fchannel(file),&ch,1); }
#endif

#define SUMSIZE 64

/* single character decode */
#define DEC(c)	(((c) - ' ') & 077)

main(argc, argv)
char **argv;
{
	FILE *in, *out;
	int mode;
#ifdef datageneral
	char dest[256];     /* Filenames can be larger */
	char buf[512];
#else
	char dest[128];
	char buf[80];
#endif

	/* optional input arg */
	if (argc > 1) {
		if ((in = fopen(argv[1], "r")) == NULL) {
			perror(argv[1]);
			exit(1);
		}
		argv++; argc--;
	} else
		in = stdin;

	if (argc != 1) {
		printf("Usage: uudecode [infile]\n");
		exit(2);
	}

	/* search for header line */
	for (;;) {
		if (fgets(buf, sizeof buf, in) == NULL) {
			fprintf(stderr, "No begin line\n");
			exit(3);
		}
		if (strncmp(buf, "begin ", 6) == 0)
			break;
	}
#ifdef datageneral
        /* DG format has extra information, and the record format field must
         * be a legal value.  Other systems only store two items: the file 
         * mode and the file name.  We store the record format, the file
         * name and the file type.
         * Legal record formats are current betwee $RTDY and $RTVB (which 
         * is IBM tape only).
         */
	if ((sscanf(buf, "begin %o %s %o", &rec_format, dest, &file_type) < 3) 
	|| ((rec_format < $RTDY) && (rec_format > $RTVB))) {
            /* Not a DG generated file -- use defaults of Dynamic data and
             * a user data file (UDF).
             */
	    rec_format = $RTDY;
	    file_type  = $FUDF;
	}
#else
	sscanf(buf, "begin %o %s", &mode, dest);
#endif

#ifdef unix
	/* handle ~user/file format */
	if (dest[0] == '~') {
		char *sl;
		struct passwd *getpwnam();
		char *index();
		struct passwd *user;
		char dnbuf[100];

		sl = index(dest, '/');
		if (sl == NULL) {
			fprintf(stderr, "Illegal ~user\n");
			exit(3);
		}
		*sl++ = 0;
		user = getpwnam(dest+1);
		if (user == NULL) {
			fprintf(stderr, "No such user as %s\n", dest);
			exit(4);
		}
		strcpy(dnbuf, user->pw_dir);
		strcat(dnbuf, "/");
		strcat(dnbuf, sl);
		strcpy(dest, dnbuf);
	}
#endif

	/* create output file */
#ifdef datageneral
        zero((char *) &w_io_parms, sizeof(w_io_parms));
        w_io_parms.isti = $IBIN|$RTDY|$ICRF|$OFOT;
        w_io_parms.isti &= ~$IPST;
        w_io_parms.imrs = 2048;
        w_io_parms.ibad = -1;
        w_io_parms.ircl = -1;

        if (rec_format == 0) rec_format = $RTUN; 
        out = dg_open(dest, $OFOT|$IBIN|$OFCR|$OFCE|rec_format,file_type);
        if (out == NULL) {
		perror(dest);
		exit(4);
	}
	chanout = fchannel(out);
#else
	out = fopen(dest, "w");
	if (out == NULL) {
		perror(dest);
		exit(4);
	}
#if (unix || !MANX)
	chmod(dest, mode);
#endif
#endif datageneral

	decode(in, out);

	if (fgets(buf, sizeof buf, in) == NULL || strcmp(buf, "end\n")) {
		fprintf(stderr, "No end line\n");
		exit(5);
	}
	exit(0);
}

/*
 * copy from in to out, decoding as you go along.
 */
decode(in, out)
FILE *in;
FILE *out;
{
	char buf[80];
	char *bp;
	int n, checksum, line;
	int warnings = 5;

	for (line = 1; ; line++) {
		/* for each input line */
		if (fgets(buf, sizeof buf, in) == NULL) {
			printf("Short file\n");
			exit(10);
		}

		checksum = 0;
		n = DEC(buf[0]);
		if (n <= 0)
			break;

		bp = &buf[1];
		while (n > 0) {
			checksum = (checksum+outdec(bp, out, n)) % SUMSIZE;
			bp += 4;
			n -= 3;
		}

		if (*bp != '\n' && checksum != DEC(*bp))
			if (warnings > 0) {
				printf("Checksum error, line %d.\n",line);
				warnings--;
			} else if (warnings == 0) {
				printf("more...\n");
				warnings--;
			}
	}
}

/*
 * output a group of 3 bytes (4 input characters).
 * the input chars are pointed to by p, they are to
 * be output to file f.  n is used to tell us not to
 * output all of them at the end of the file.
 * we return a checksum increment.
 */
int outdec(p, f, n)
char *p;
FILE *f;
{
	int c1, c2, c3;

	c1 = DEC(*p) << 2 | DEC(p[1]) >> 4;
	c2 = DEC(p[1]) << 4 | DEC(p[2]) >> 2;
	c3 = DEC(p[2]) << 6 | DEC(p[3]);

	if (n >= 1)
		putc(c1, f);
	if (n >= 2)
		putc(c2, f);
	if (n >= 3)
		putc(c3, f);

	return((c1+c2+c3) % SUMSIZE);
}


/* fr: like read but stdio */
int
fr(fd, buf, cnt)
FILE *fd;
char *buf;
int cnt;
{
	int c, i;

	for (i=0; i<cnt; i++) {
		c = getc(fd);
		if (c == EOF)
			return(i);
		buf[i] = c;
	}
	return (cnt);
}

/*
 * Return the ptr in sp at which the character c appears;
 * NULL if not found
 */

#ifndef NULL		/* MANX has bogus "#define NULL 0L" somewhere */
#define	NULL	0
#endif

char *
index(sp, c)
register char *sp, c;
{
	do {
		if (*sp == c)
			return(sp);
	} while (*sp++);
	return(NULL);
}

#if (!unix && MANX)

perror (sp)
char *sp;
{
	if (sp && *sp) {
		fprintf (stderr, "%s: ");
	}
	fprintf (stderr, "<unknown error>\n");
}

#endif	/* Unix */

#ifdef datageneral
/* D G _ B I N W -- Output len characters to the file number filenum 
 * 
 *  The syntax is like the Unix write command.
 *  This code was borrowed from my Kermit source -- ckdtio.c
 */

dg_binw(channel,chs,len) int channel, len; char *chs; 
{
     int ac2,err;

     if (len == 0) return(0);

     w_io_parms.ich = channel;
     w_io_parms.ibad = chs;
     w_io_parms.ircl = len;
     ac2 = &w_io_parms;
     
     if ((err = sys_write(ac2)) == 0) return(0);

     if ( err != ERLTL && err != EREOF ) {
    	  perror("dg_binw: sys_write ");
          exit(err);
     }
}
#endif