[net.bugs.uucp] new version of uucp f-proto

piet@mcvax.UUCP (Piet Beertema) (10/08/85)

<>

Here's a new version of the f-proto for uucp. Compared to the one I posted
quite some time ago the major differences are:
- it has been adapted to run cost-effective on X.25 hosts; the old version
  produced too many short packets on such hosts;
- a few minor bugs have been fixed (hence the posting to net.bugs.uucp too).
If you're already running 4.3bsd uucp, you can replace the fio.c by this one
without modifications.

-----------------------------------------------------------
#This is a shell archive; extract with /bin/sh
echo -n 'Extracting n.fio.c... '
sed 's/^M//' >n.fio.c <<\SHAREND
M/*	fio.c	3.3	85/10/08	(CWI)	*/
M
M/*
M * flow control protocol.
M *
M * This protocol relies on flow control of the data stream.
M * It is meant for working over links that can (almost) be
M * guaranteed to be errorfree, specifically X.25/PAD links.
M * A sumcheck is carried out over a whole file only. If a
M * transport fails the receiver can request retransmission(s).
M * This protocol uses a 7-bit datapath only, so it can be
M * used on links that are not 8-bit transparent.
M *
M * When using this protocol with an X.25 PAD:
M * Although this protocol uses no control chars except CR,
M * control chars NULL and ^P are used before this protocol
M * is started; since ^P is the default char for accessing
M * PAD X.28 command mode, be sure to disable that access
M * (PAD par 1). Also make sure both flow control pars
M * (5 and 12) are set. The CR used in this proto is meant
M * to trigger packet transmission, hence par 3 should be 
M * set to 2; a good value for the Idle Timer (par 4) is 10.
M * All other pars should be set to 0.
M *
M * Normally a calling site will take care of setting the
M * local PAD pars via an X.28 command and those of the remote
M * PAD via an X.29 command, unless the remote site has a
M * special channel assigned for this protocol with the proper
M * par settings.
M *
M * Additional comments for hosts with direct X.25 access:
M * - the global variable IsTcpIp, when set, excludes ioctl's,
M *   so the same binary can run on X.25 and non-X.25 hosts;
M * - reads are done in small chunks, which can be smaller than
M *   the packet size; your X.25 driver must support that.
M *
M *
M * Author:
M *	Piet Beertema, CWI, Amsterdam, Sep 1984
M * Modified for X.25 hosts:
M *	Robert Elz, Melbourne Univ, Mar 1985
M */
M
M#include "uucp.h"
M#include <signal.h>
M#ifdef USG
M#include <termio.h>
M#else !USG
M#include <sgtty.h>
M#endif !USG
M#include <setjmp.h>
M
M#define FIBUFSIZ	256	/* for X.25 interfaces: set equal to packet size,
M				 * but see comment above
M				 */
M
M#define FOBUFSIZ	1024	/* for X.25 interfaces: set equal to packet size;
M				 * otherwise make as large as feasible to reduce
M				 * number of write system calls 
M				 */
M
M#ifndef MAXMSGLEN
M#define MAXMSGLEN	BUFSIZ
M#endif MAXMSGLEN
M
Mstatic int fchksum;
Mstatic jmp_buf Ffailbuf;
M
Mstatic
Mfalarm()
M{
M	signal(SIGALRM, falarm);
M	longjmp(Ffailbuf, 1);
M}
M
Mstatic int (*fsig)();
M
M#ifndef USG
M#define TCGETA	TIOCGETP
M#define TCSETA	TIOCSETP
M#define termio	sgttyb
M#endif USG
M
Mfturnon()
M{
M	int ret;
M	struct termio ttbuf;
M
M	if (!IsTcpIp) {
M		ioctl(Ifn, TCGETA, &ttbuf);
M#ifdef USG
M		ttbuf.c_iflag = IXOFF|IXON|ISTRIP;
M		ttbuf.c_cc[VMIN] = FIBUFSIZ > 64 ? 64 : FIBUFSIZ;
M		ttbuf.c_cc[VTIME] = 5;
M#else !USG
M		ttbuf.sg_flags = ANYP|CBREAK|TANDEM;
M#endif USG
M		ret = ioctl(Ifn, TCSETA, &ttbuf);
M		ASSERT(ret >= 0, "STTY FAILED", "", ret);
M	}
M	fsig = signal(SIGALRM, falarm);
M	/* give the other side time to perform its ioctl;
M	 * otherwise it may flush out the first data this
M	 * side is about to send.
M	 */
M	sleep(2);
M	return SUCCESS;
M}
M
Mfturnoff()
M{
M	(void) signal(SIGALRM, fsig);
M	return SUCCESS;
M}
M
Mfwrmsg(type, str, fn)
Mregister char *str;
Mint fn;
Mchar type;
M{
M	register char *s;
M	char bufr[MAXMSGLEN];
M
M	s = bufr;
M	*s++ = type;
M	while (*str)
M		*s++ = *str++;
M	if (*(s-1) == '\n')
M		s--;
M	*s++ = '\r';
M	*s = 0;
M	(void) write(fn, bufr, s - bufr);
M	return SUCCESS;
M}
M
Mfrdmsg(str, fn)
Mregister char *str;
Mregister int fn;
M{
M	register char *smax;
M
M	if (setjmp(Ffailbuf))
M		return FAIL;
M	smax = str + MAXMSGLEN - 1;
M	(void) alarm(2*MAXMSGTIME);
M	for (;;) {
M		if (read(fn, str, 1) <= 0)
M			goto msgerr;
M		*str &= 0177;
M		if (*str == '\r')
M			break;
M		if (*str < ' ') {
M			continue;
M		}
M		if (str++ >= smax)
M			goto msgerr;
M	}
M	*str = '\0';
M	(void) alarm(0);
M	return SUCCESS;
Mmsgerr:
M	(void) alarm(0);
M	return FAIL;
M}
M
Mfwrdata(fp1, fn)
MFILE *fp1;
Mint fn;
M{
M	register int alen, ret;
M	register char *obp;
M	char ack, ibuf[MAXMSGLEN];
M	int flen, mil, retries = 0;
M	long abytes, fbytes;
M	struct timeb t1, t2;
M
M	ret = FAIL;
Mretry:
M	fchksum = 0xffff;
M	abytes = fbytes = 0L;
M	ack = '\0';
M#ifdef USG
M	time(&t1.time);
M	t1.millitm = 0;
M#else !USG
M	ftime(&t1);
M#endif !USG
M	do {
M		alen = fwrblk(fn, fp1, &flen);
M		fbytes += flen;
M		if (alen <= 0) {
M			abytes -= alen;
M			goto acct;
M		}
M		abytes += alen;
M	} while (!feof(fp1) && !ferror(fp1));
M	DEBUG(8, "\nchecksum: %04x\n", fchksum);
M	if (frdmsg(ibuf, fn) != FAIL) {
M		if ((ack = ibuf[0]) == 'G')
M			ret = SUCCESS;
M		DEBUG(4, "ack - '%c'\n", ack);
M	}
Macct:
M#ifdef USG
M	time(&t2.time);
M	t2.millitm = 0;
M#else !USG
M	ftime(&t2);
M#endif !USG
M	Now = t2;
M	t2.time -= t1.time;
M	mil = t2.millitm - t1.millitm;
M	if (mil < 0) {
M		--t2.time;
M		mil += 1000;
M	}
M	sprintf(ibuf, "sent data %ld bytes %ld.%02d secs",
M		fbytes, (long)t2.time, mil / 10);
M	sysacct(abytes, t2.time - t1.time);
M	if (retries > 0)
M		sprintf(&ibuf[strlen(ibuf)], ", retry #%d", retries);
M	DEBUG(1, "%s\n", ibuf);
M	syslog(ibuf);
M	if (ack == 'R') {
M		DEBUG(4, "RETRY:\n", 0);
M		fseek(fp1, 0L, 0);
M		retries++;
M		goto retry;
M	}
M#ifdef SYSACCT
M	if (ret == FAIL)
M		sysaccf(NULL);		/* force accounting */
M#endif SYSACCT
M	return ret;
M}
M
M/* max. attempts to retransmit a file: */
M#define MAXRETRIES	(fbytes < 10000L ? 2 : 1)
M
Mfrddata(fn, fp2)
Mregister int fn;
Mregister FILE *fp2;
M{
M	register int flen;
M	register char eof;
M	char ibuf[FIBUFSIZ];
M	int ret, mil, retries = 0;
M	long alen, abytes, fbytes;
M	struct timeb t1, t2;
M
M	ret = FAIL;
Mretry:
M	fchksum = 0xffff;
M	abytes = fbytes = 0L;
M#ifdef USG
M	time(&t1.time);
M	t1.millitm = 0;
M#else !USG
M	ftime(&t1);
M#endif !USG
M	do {
M		flen = frdblk(ibuf, fn, &alen);
M		abytes += alen;
M		if (flen < 0)
M			goto acct;
M		if (eof = flen > FIBUFSIZ)
M			flen -= FIBUFSIZ + 1;
M		fbytes += flen;
M		if (fwrite(ibuf, sizeof (char), flen, fp2) != flen)
M			goto acct;
M	} while (!eof);
M	ret = SUCCESS;
Macct:
M#ifdef USG
M	time(&t2.time);
M	t2.millitm = 0;
M#else !USG
M	ftime(&t2);
M#endif !USG
M	Now = t2;
M	t2.time -= t1.time;
M	mil = t2.millitm - t1.millitm;
M	if (mil < 0) {
M		--t2.time;
M		mil += 1000;
M	}
M	sprintf(ibuf, "received data %ld bytes %ld.%02d secs",
M		fbytes, (long)t2.time, mil/10);
M	sysacct(abytes, t2.time - t1.time);
M	DEBUG(1, "%s\n", ibuf);
M	syslog(ibuf);
M	if (ret == FAIL) {
M		if (retries++ < MAXRETRIES) {
M			DEBUG(8, "send ack: 'R'\n", 0);
M			fwrmsg('R', "", fn);
M			fseek(fp2, 0L, 0);
M			DEBUG(4, "RETRY:\n", 0);
M			goto retry;
M		}
M		DEBUG(8, "send ack: 'Q'\n", 0);
M		fwrmsg('Q', "", fn);
M#ifdef SYSACCT
M		sysaccf(NULL);		/* force accounting */
M#endif SYSACCT
M	}
M	else {
M		DEBUG(8, "send ack: 'G'\n", 0);
M		fwrmsg('G', "", fn);
M	}
M	return ret;
M}
M
Mstatic
Mfrdbuf(blk, len, fn)
Mregister char *blk;
Mregister int len;
Mregister int fn;
M{
M	static int ret = FIBUFSIZ / 2;
M
M	if (setjmp(Ffailbuf))
M		return FAIL;
M	(void) alarm(MAXMSGTIME);
M	ret = read(fn, blk, len);
M	alarm(0);
M	return ret <= 0 ? FAIL : ret;
M}
M
M/* call ultouch every TC calls to either frdblk or fwrblk  */
M#define TC	20
Mstatic int tc = TC;
M
M/* Byte conversion:
M *
M *   from	 pre	   to
M * 000-037	 172	 100-137
M * 040-171		 040-171
M * 172-177	 173	 072-077
M * 200-237	 174	 100-137
M * 240-371	 175	 040-171
M * 372-377	 176	 072-077
M */
M
Mstatic
Mfwrblk(fn, fp, lenp)
Mint fn;
Mregister FILE *fp;
Mint *lenp;
M{
M	register char *op;
M	register int c, sum, nl, len;
M	char obuf[FOBUFSIZ + 8];
M	int ret;
M
M	/* call ultouch occasionally */
M	if (--tc < 0) {
M		tc = TC;
M		ultouch();
M	}
M	op = obuf;
M	nl = 0;
M	len = 0;
M	sum = fchksum;
M	while ((c = getc(fp)) != EOF) {
M		len++;
M		if (sum & 0x8000) {
M			sum <<= 1;
M			sum++;
M		} else
M			sum <<= 1;
M		sum += c;
M		sum &= 0xffff;
M		if (c & 0200) {
M			c &= 0177;
M			if (c < 040) {
M				*op++ = '\174';
M				*op++ = c + 0100;
M			} else
M			if (c <= 0171) {
M				*op++ = '\175';
M				*op++ = c;
M			}
M			else {
M				*op++ = '\176';
M				*op++ = c - 0100;
M			}
M			nl += 2;
M		} else {
M			if (c < 040) {
M				*op++ = '\172';
M				*op++ = c + 0100;
M				nl += 2;
M			} else
M			if (c <= 0171) {
M				*op++ = c;
M				nl++;
M			} else {
M				*op++ = '\173';
M				*op++ = c - 0100;
M				nl += 2;
M			}
M		}
M		if (nl >= FOBUFSIZ - 1) {
M			/*
M			 * peek at next char, see if it will fit
M			 */
M			c = getc(fp);
M			if (c == EOF)
M				break;
M			(void) ungetc(c, fp);
M			if (nl >= FOBUFSIZ || c < 040 || c > 0171)
M				goto writeit;
M		}
M	}
M	/*
M	 * At EOF - append checksum, there is space for it...
M	 */
M	sprintf(op, "\176\176%04x\r", sum);
M	nl += strlen(op);
Mwriteit:
M	*lenp = len;
M	fchksum = sum;
M	DEBUG(8, "%d/", len);
M	DEBUG(8, "%d ", nl);
M	ret = write(fn, obuf, nl);
M	return ret == nl ? nl : ret < 0 ? 0 : -ret;
M}
M
Mstatic
Mfrdblk(ip, fn, rlen)
Mregister char *ip;
Mint fn;
Mlong *rlen;
M{
M	register char *op, c;
M	register int sum, len, nl;
M	char buf[5], *erbp = ip;
M	int i;
M	static char special = 0;
M
M	/* call ultouch occasionally */
M	if (--tc < 0) {
M		tc = TC;
M		ultouch();
M	}
M	if ((len = frdbuf(ip, FIBUFSIZ, fn)) == FAIL) {
M		*rlen = 0;
M		goto dcorr;
M	}
M	*rlen = len;
M	DEBUG(8, "%d/", len);
M	op = ip;
M	nl = 0;
M	sum = fchksum;
M	do {
M		if ((*ip &= 0177) >= '\172') {
M			if (special) {
M				DEBUG(8, "%d", nl);
M				special = 0;
M				op = buf;
M				if (*ip++ != '\176' || (i = --len) > 5)
M					goto dcorr;
M				while (i--)
M					*op++ = *ip++ & 0177;
M				while (len < 5) {
M					i = frdbuf(&buf[len], 5 - len, fn);
M					if (i == FAIL) {
M						len = FAIL;
M						goto dcorr;
M					}
M					DEBUG(8, ",%d", i);
M					len += i;
M					*rlen += i;
M					while (i--)
M						*op++ &= 0177;
M				}
M				if (buf[4] != '\r')
M					goto dcorr;
M				sscanf(buf, "%4x", &fchksum);
M				DEBUG(8, "\nchecksum: %04x\n", sum);
M				if (fchksum == sum)
M					return FIBUFSIZ + 1 + nl;
M				else {
M					DEBUG(8, "\n", 0);
M					DEBUG(4, "Bad checksum\n", 0);
M					return FAIL;
M				}
M			}
M			special = *ip++;
M		} else {
M			if (*ip < '\040') {
M				/* error: shouldn't get control chars */
M				goto dcorr;
M			}
M			switch (special) {
M			case 0:
M				c = *ip++;
M				break;
M			case '\172':
M				c = *ip++ - 0100;
M				break;
M			case '\173':
M				c = *ip++ + 0100;
M				break;
M			case '\174':
M				c = *ip++ + 0100;
M				break;
M			case '\175':
M				c = *ip++ + 0200;
M				break;
M			case '\176':
M				c = *ip++ + 0300;
M				break;
M			}
M			*op++ = c;
M			if (sum & 0x8000) {
M				sum <<= 1;
M				sum++;
M			} else
M				sum <<= 1;
M			sum += c & 0377;
M			sum &= 0xffff;
M			special = 0;
M			nl++;
M		}
M	} while (--len);
M	fchksum = sum;
M	DEBUG(8, "%d ", nl);
M	return nl;
Mdcorr:
M	DEBUG(8, "\n", 0);
M	DEBUG(4, "Data corrupted\n", 0);
M	while (len != FAIL) {
M		if ((len = frdbuf(erbp, FIBUFSIZ, fn)) != FAIL)
M			*rlen += len;
M	}
M	return FAIL;
M}
SHAREND
echo done
exit

-- 
	Piet Beertema, CWI, Amsterdam
	(piet@mcvax.UUCP)