This repository has been archived on 2024-04-08. You can view files and clone it, but cannot push or open issues or pull requests.
deb-mbse/mbsebbs/zmmisc.c

910 lines
19 KiB
C
Raw Normal View History

2004-11-07 13:21:59 +00:00
/*
* $Id$
*
* Z M . C
* Copyright 1994 Omen Technology Inc All Rights Reserved
* ZMODEM protocol primitives
*
* Entry point Functions:
* zsbhdr(type, hdr) send binary header
* zshhdr(type, hdr) send hex header
* zgethdr(hdr) receive header - binary or hex
* zsdata(buf, len, frameend) send data
* zrdata(buf, len) receive data
* stohdr(pos) store position data in Txhdr
* long rclhdr(hdr) recover position offset from header
*
*
* This version implements numerous enhancements including ZMODEM
* Run Length Encoding and variable length headers. These
* features were not funded by the original Telenet development
* contract.
*
* This software may be freely used for educational (didactic
* only) purposes. This software may also be freely used to
* support file transfer operations to or from licensed Omen
* Technology products. Use with other commercial or shareware
* programs (Crosstalk, Procomm, etc.) REQUIRES REGISTRATION.
*
* Any programs which use part or all of this software must be
* provided in source form with this notice intact except by
* written permission from Omen Technology Incorporated.
*
* Use of this software for commercial or administrative purposes
* except when exclusively limited to interfacing Omen Technology
* products requires a per port license payment of $20.00 US per
* port (less in quantity). Use of this code by inclusion,
* decompilation, reverse engineering or any other means
* constitutes agreement to these conditions and acceptance of
* liability to license the materials and payment of reasonable
* legal costs necessary to enforce this license agreement.
*
*
* Omen Technology Inc
* Post Office Box 4681
* Portland OR 97208
*
* This code is made available in the hope it will be useful,
* BUT WITHOUT ANY WARRANTY OF ANY KIND OR LIABILITY FOR ANY
* DAMAGES OF ANY KIND.
*
*/
static void zputhex(int);
2004-11-20 13:30:13 +00:00
static void zsbh32(char*, int);
static void zsda32(char*, int, int);
2004-11-07 13:35:34 +00:00
static int zrdat32(char*,int);
static int noxrd7(void);
static int zrbhd32(char*);
static int zrbhdr(char*);
static int zrhhdr(char*);
static int zgethex(void);
static int zgeth1(void);
2004-11-07 13:21:59 +00:00
static void garbitch(void);
2004-11-07 13:35:34 +00:00
2004-11-07 13:21:59 +00:00
#include "../config.h"
#include "../lib/mbselib.h"
#include "ttyio.h"
#include "zmmisc.h"
2004-11-20 18:31:13 +00:00
/*
* Original zm.c timing was in tenths of seconds, but our current ttyio driver
* does timing in whole seconds.
*/
2004-11-07 13:21:59 +00:00
static int Rxtimeout = 10; /* Seconds to wait for something */
char *txbuf=NULL;
2004-11-12 21:25:59 +00:00
static int lastsent; /* Last char we sent */
static int Not8bit; /* Seven bits seen on header */
2004-11-07 13:21:59 +00:00
2004-11-20 13:30:13 +00:00
extern unsigned Baudrate;
2004-11-07 13:21:59 +00:00
char *frametypes[] = {
(char *)"EMPTY", /* -16 */
(char *)"Can't be (-15)",
(char *)"Can't be (-14)",
(char *)"Can't be (-13)",
(char *)"Can't be (-12)",
(char *)"Can't be (-11)",
(char *)"Can't be (-10)",
(char *)"Can't be (-9)",
(char *)"HANGUP", /* -8 */
(char *)"Can't be (-7)",
(char *)"Can't be (-6)",
(char *)"Can't be (-5)",
(char *)"EOFILE", /* -4 */
(char *)"Can't be (-3)",
(char *)"TIMEOUT", /* -2 */
(char *)"ERROR", /* -1 */
(char *)"ZRQINIT",
(char *)"ZRINIT",
(char *)"ZSINIT",
(char *)"ZACK",
(char *)"ZFILE",
(char *)"ZSKIP",
(char *)"ZNAK",
(char *)"ZABORT",
(char *)"ZFIN",
(char *)"ZRPOS",
(char *)"ZDATA",
(char *)"ZEOF",
(char *)"ZFERR",
(char *)"ZCRC",
(char *)"ZCHALLENGE",
(char *)"ZCOMPL",
(char *)"ZCAN",
(char *)"ZFREECNT",
(char *)"ZCOMMAND",
(char *)"ZSTDERR",
(char *)"xxxxx"
#define FRTYPES 22 /* Total number of frame types in this array */
/* not including psuedo negative entries */
};
/***** Hack by mj ***********************************************************/
/*
* Buffer for outgoing frames. Sending them with single character write()'s
* is a waste of processor time and causes severe performance degradation
* on TCP and ISDN connections.
*/
#define FRAME_BUFFER_SIZE 16384
static char *frame_buffer=NULL;
static int frame_length = 0;
#define BUFFER_CLEAR() do { frame_length=0; } while(0)
#define BUFFER_BYTE(c) do { frame_buffer[frame_length++]=(c); } while(0)
#define BUFFER_FLUSH() do { PUT(frame_buffer, frame_length); \
frame_length=0; } while(0);
/****************************************************************************/
void get_frame_buffer(void)
{
2004-11-07 13:35:34 +00:00
if (frame_buffer == NULL)
frame_buffer = malloc(FRAME_BUFFER_SIZE);
2004-11-07 13:21:59 +00:00
}
void free_frame_buffer(void)
{
2004-11-07 13:35:34 +00:00
if (frame_buffer)
free(frame_buffer);
frame_buffer = NULL;
2004-11-07 13:21:59 +00:00
}
/*
* Send ZMODEM binary header hdr of type type
*/
2004-11-20 13:30:13 +00:00
void zsbhdr(int type, char *shdr)
2004-11-07 13:21:59 +00:00
{
2004-11-07 13:35:34 +00:00
register int n;
register unsigned short crc;
2004-11-07 13:21:59 +00:00
2004-11-20 13:30:13 +00:00
Syslog('z', "zsbhdr: %s %lx", frametypes[type+FTOFFSET], rclhdr(shdr));
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
BUFFER_CLEAR();
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
if (type == ZDATA)
for (n = Znulls; --n >=0; )
BUFFER_BYTE(0);
2004-11-12 21:25:59 +00:00
BUFFER_BYTE(ZPAD);
BUFFER_BYTE(ZDLE);
2004-11-07 13:35:34 +00:00
switch (Crc32t=Txfcs32) {
2004-11-20 13:30:13 +00:00
case 2: zsbh32(shdr, type);
2004-11-07 13:35:34 +00:00
BUFFER_FLUSH();
break;
2004-11-20 13:30:13 +00:00
case 1: zsbh32(shdr, type);
2004-11-07 13:35:34 +00:00
break;
2004-11-20 13:30:13 +00:00
default: BUFFER_BYTE(ZBIN);
2004-11-07 13:35:34 +00:00
zsendline(type);
crc = updcrc16(type, 0);
2004-11-07 13:21:59 +00:00
2004-11-20 13:30:13 +00:00
for (n=4; --n >= 0; ++shdr) {
2004-11-07 13:21:59 +00:00
zsendline(*shdr);
crc = updcrc16((0377& *shdr), crc);
2004-11-07 13:35:34 +00:00
}
crc = updcrc16(0,updcrc16(0,crc));
zsendline(((int)(crc>>8)));
zsendline(crc);
}
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
BUFFER_FLUSH();
2004-11-07 13:21:59 +00:00
}
/*
* Send ZMODEM binary header hdr of type type
*/
2004-11-20 13:30:13 +00:00
void zsbh32(char *shdr, int type)
2004-11-07 13:21:59 +00:00
{
2004-11-07 13:35:34 +00:00
register int n;
register unsigned long crc;
2004-11-20 13:30:13 +00:00
BUFFER_BYTE(ZBIN32);
2004-11-07 13:35:34 +00:00
zsendline(type);
2004-11-12 21:25:59 +00:00
crc = 0xFFFFFFFFL;
crc = updcrc32(type, crc);
2004-11-07 13:35:34 +00:00
2004-11-20 13:30:13 +00:00
for (n=4; --n >= 0; ++shdr) {
2004-11-07 13:35:34 +00:00
crc = updcrc32((0377 & *shdr), crc);
zsendline(*shdr);
}
crc = ~crc;
for (n=4; --n >= 0;) {
zsendline((int)crc);
crc >>= 8;
}
2004-11-07 13:21:59 +00:00
}
/*
* Send ZMODEM HEX header hdr of type type
*/
2004-11-20 13:30:13 +00:00
void zshhdr(int type, register char *shdr)
2004-11-07 13:21:59 +00:00
{
2004-11-07 13:35:34 +00:00
register int n;
register unsigned short crc;
2004-11-07 13:21:59 +00:00
2004-11-20 13:30:13 +00:00
Syslog('z', "zshhdr: %s %lx", frametypes[type+FTOFFSET], rclhdr(shdr));
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
BUFFER_CLEAR();
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
BUFFER_BYTE(ZPAD);
BUFFER_BYTE(ZPAD);
BUFFER_BYTE(ZDLE);
2004-11-20 13:30:13 +00:00
BUFFER_BYTE(ZHEX);
zputhex(type & 0x7f);
2004-11-07 13:35:34 +00:00
Crc32t = 0;
2004-11-20 13:30:13 +00:00
crc = updcrc16((type & 0x7f), 0);
for (n=4; --n >= 0; ++shdr) {
2004-11-12 21:25:59 +00:00
zputhex(*shdr);
crc = updcrc16((0377 & *shdr), crc);
2004-11-07 13:35:34 +00:00
}
crc = updcrc16(0,updcrc16(0,crc));
2004-11-20 13:30:13 +00:00
zputhex(((int)(crc>>8)));
zputhex(crc);
2004-11-07 13:35:34 +00:00
/*
* Make it printable on remote machine
*/
BUFFER_BYTE(015);
BUFFER_BYTE(0212);
/*
* Uncork the remote in case a fake XOFF has stopped data flow
*/
if (type != ZFIN && type != ZACK)
BUFFER_BYTE(021);
BUFFER_FLUSH();
2004-11-07 13:21:59 +00:00
}
/*
* Send binary array buf of length length, with ending ZDLE sequence frameend
*/
char *Zendnames[] = {(char *)"ZCRCE",(char *)"ZCRCG",(char *)"ZCRCQ",(char *)"ZCRCW"};
void zsdata(register char *buf, int length, int frameend)
{
2004-11-07 13:35:34 +00:00
register unsigned short crc;
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
Syslog('z', "zsdata: %d %s", length, Zendnames[(frameend-ZCRCE)&3]);
2004-11-07 13:21:59 +00:00
2004-11-07 13:35:34 +00:00
BUFFER_CLEAR();
2004-11-20 13:30:13 +00:00
if (Crc32t)
zsda32(buf, length, frameend);
else {
crc = 0;
for (;--length >= 0; ++buf) {
zsendline(*buf);
crc = updcrc16((0377 & *buf), crc);
}
BUFFER_BYTE(ZDLE);
BUFFER_BYTE(frameend);
crc = updcrc16(frameend, crc);
2004-11-07 13:35:34 +00:00
2004-11-20 13:30:13 +00:00
crc = updcrc16(0,updcrc16(0,crc));
zsendline(((int)(crc>>8)));
zsendline(crc);
2004-11-07 13:35:34 +00:00
}
2004-11-20 13:30:13 +00:00
2004-11-07 13:35:34 +00:00
if (frameend == ZCRCW)
BUFFER_BYTE(XON);
BUFFER_FLUSH();
2004-11-07 13:21:59 +00:00
}
void zsda32(register char *buf, int length, int frameend)
{
2004-11-07 13:35:34 +00:00
register int c;
register unsigned long crc;
crc = 0xFFFFFFFFL;
for (;--length >= 0; ++buf) {
c = *buf & 0377;
if (c & 0140)
BUFFER_BYTE(lastsent = c);
else
zsendline(c);
crc = updcrc32(c, crc);
}
BUFFER_BYTE(ZDLE);
BUFFER_BYTE(frameend);
crc = updcrc32(frameend, crc);
crc = ~crc;
for (c=4; --c >= 0;) {
2004-11-12 21:25:59 +00:00
zsendline((int)crc);
crc >>= 8;
2004-11-07 13:35:34 +00:00
}
2004-11-07 13:21:59 +00:00
}
/*
* Receive array buf of max length with ending ZDLE sequence
* and CRC. Returns the ending character or error code.
* NB: On errors may store length+1 bytes!
*/
int zrdata(register char *buf, int length)
{
2004-11-08 14:36:34 +00:00
register int c;
register unsigned short crc;
register char *end;
register int d;
2004-11-07 13:21:59 +00:00
2004-11-20 18:31:13 +00:00
Syslog('Z', "zrdata: len=%d, Crc32r=%s", length, Crc32r ? "true":"false");
2004-11-12 21:25:59 +00:00
2004-11-20 13:30:13 +00:00
if (Crc32r)
return zrdat32(buf, length);
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
crc = Rxcount = 0;
end = buf + length;
2004-11-08 14:36:34 +00:00
while (buf <= end) {
if ((c = zdlread()) & ~0377) {
2004-11-07 13:21:59 +00:00
crcfoo:
2004-11-08 14:36:34 +00:00
switch (c) {
case GOTCRCE:
case GOTCRCG:
case GOTCRCQ:
case GOTCRCW: crc = updcrc16((((d=c))&0377), crc);
2004-11-07 13:21:59 +00:00
if ((c = zdlread()) & ~0377)
2004-11-08 14:36:34 +00:00
goto crcfoo;
2004-11-07 13:21:59 +00:00
crc = updcrc16(c, crc);
if ((c = zdlread()) & ~0377)
2004-11-08 14:36:34 +00:00
goto crcfoo;
2004-11-07 13:21:59 +00:00
crc = updcrc16(c, crc);
if (crc & 0xFFFF) {
2004-11-08 14:36:34 +00:00
Syslog('+', "Zmodem zrdata: Bad CRC");
return TERROR;
2004-11-07 13:21:59 +00:00
}
Rxcount = length - (end - buf);
2004-11-20 18:31:13 +00:00
Syslog('z', "zrdata: %d %s", Rxcount, Zendnames[(d-GOTCRCE)&3]);
2004-11-07 13:21:59 +00:00
return d;
2004-11-08 14:36:34 +00:00
case GOTCAN: Syslog('+', "Zmodem: Sender Canceled");
2004-11-07 13:21:59 +00:00
return ZCAN;
2004-11-08 14:36:34 +00:00
case TIMEOUT: Syslog('+', "Zmodem: TIMEOUT receiving data");
2004-11-07 13:21:59 +00:00
return c;
2004-11-08 14:36:34 +00:00
case HANGUP: Syslog('+', "Zmodem: Carrier lost while receiving");
2004-11-07 13:21:59 +00:00
return c;
2004-11-08 14:36:34 +00:00
default: garbitch();
2004-11-07 13:21:59 +00:00
return c;
2004-11-08 14:36:34 +00:00
}
2004-11-07 13:21:59 +00:00
}
2004-11-08 14:36:34 +00:00
*buf++ = c;
crc = updcrc16(c, crc);
}
Syslog('+', "Zmodem: Data subpacket too long");
return TERROR;
2004-11-07 13:21:59 +00:00
}
int zrdat32(register char *buf, int length)
{
2004-11-08 14:36:34 +00:00
register int c;
register unsigned long crc;
register char *end;
register int d;
2004-11-12 21:25:59 +00:00
crc = 0xFFFFFFFFL;
Rxcount = 0;
end = buf + length;
2004-11-08 14:36:34 +00:00
while (buf <= end) {
if ((c = zdlread()) & ~0377) {
2004-11-07 13:21:59 +00:00
crcfoo:
2004-11-08 14:36:34 +00:00
switch (c) {
case GOTCRCE:
case GOTCRCG:
case GOTCRCQ:
case GOTCRCW: d = c; c &= 0377;
2004-11-07 13:21:59 +00:00
crc = updcrc32(c, crc);
if ((c = zdlread()) & ~0377)
2004-11-08 14:36:34 +00:00
goto crcfoo;
2004-11-07 13:21:59 +00:00
crc = updcrc32(c, crc);
if ((c = zdlread()) & ~0377)
2004-11-08 14:36:34 +00:00
goto crcfoo;
2004-11-07 13:21:59 +00:00
crc = updcrc32(c, crc);
if ((c = zdlread()) & ~0377)
2004-11-08 14:36:34 +00:00
goto crcfoo;
2004-11-07 13:21:59 +00:00
crc = updcrc32(c, crc);
if ((c = zdlread()) & ~0377)
2004-11-08 14:36:34 +00:00
goto crcfoo;
2004-11-07 13:21:59 +00:00
crc = updcrc32(c, crc);
if (crc != 0xDEBB20E3) {
2004-11-08 14:36:34 +00:00
Syslog('+', "Zmodem zrdat32: Bad CRC");
return TERROR;
2004-11-07 13:21:59 +00:00
}
Rxcount = length - (end - buf);
Syslog('z', "zrdat32: %d %s", Rxcount, Zendnames[(d-GOTCRCE)&3]);
return d;
2004-11-08 14:36:34 +00:00
case GOTCAN: Syslog('+', "Zmodem: Sender Canceled");
2004-11-07 13:21:59 +00:00
return ZCAN;
2004-11-08 14:36:34 +00:00
case TIMEOUT: Syslog('+', "Zmodem: TIMEOUT");
2004-11-07 13:21:59 +00:00
return c;
2004-11-08 14:36:34 +00:00
case HANGUP: Syslog('+', "Zmodem: Carrier lost while receiving");
2004-11-07 13:21:59 +00:00
return c;
2004-11-08 14:36:34 +00:00
default: garbitch();
2004-11-07 13:21:59 +00:00
return c;
2004-11-08 14:36:34 +00:00
}
2004-11-07 13:21:59 +00:00
}
2004-11-08 14:36:34 +00:00
*buf++ = c;
crc = updcrc32(c, crc);
}
Syslog('+', "Zmodem: Data subpacket too long");
return TERROR;
2004-11-07 13:21:59 +00:00
}
void garbitch(void)
{
2004-11-07 13:35:34 +00:00
Syslog('+', "Zmodem: Garbled data subpacket");
2004-11-07 13:21:59 +00:00
}
/*
* Read a ZMODEM header to hdr, either binary or hex.
*
* Set Rxhlen to size of header (default 4) (valid if good hdr)
* On success, set Zmodem to 1, set Rxpos and return type of header.
* Otherwise return negative on error.
* Return ERROR instantly if ZCRCW sequence, for fast error recovery.
*/
int zgethdr(char *shdr)
{
2004-11-12 21:25:59 +00:00
register int c, n, cancount;
int Zrwindow = 1400;
n = Zrwindow + Baudrate;
Rxframeind = Rxtype = 0;
2004-11-20 13:30:13 +00:00
// Syslog('z', "zgethdr(%lx)", rclhdr(shdr));
2004-11-07 13:21:59 +00:00
startover:
2004-11-12 21:25:59 +00:00
cancount = 5;
2004-11-07 13:21:59 +00:00
again:
2004-11-12 21:25:59 +00:00
/*
* Return immediate ERROR if ZCRCW sequence seen
*/
if (((c = GETCHAR(Rxtimeout)) < 0) && (c != TIMEOUT))
goto fifi;
else {
switch(c) {
case 021:
case 0221: goto again;
2004-11-07 13:21:59 +00:00
case HANGUP:
2004-11-12 21:25:59 +00:00
case TIMEOUT: goto fifi;
case CAN:
2004-11-07 13:21:59 +00:00
gotcan:
2004-11-12 21:25:59 +00:00
Syslog('z', "zgethdr: got CAN");
2004-11-07 13:21:59 +00:00
if (--cancount <= 0) {
2004-11-12 21:25:59 +00:00
c = ZCAN;
goto fifi;
2004-11-07 13:21:59 +00:00
}
2004-11-12 21:25:59 +00:00
switch (c = GETCHAR(Rxtimeout)) {
case TIMEOUT: goto again;
case ZCRCW: switch (GETCHAR(Rxtimeout)) {
case TIMEOUT: c = TERROR;
goto fifi;
case HANGUP: goto fifi;
default: goto agn2;
}
case HANGUP: goto fifi;
default: break;
case CAN: if (--cancount <= 0) {
c = ZCAN;
goto fifi;
}
goto again;
}
/* **** FALL THRU TO **** */
2004-11-07 13:21:59 +00:00
default:
agn2:
#define GCOUNT (-4)
2004-11-12 21:25:59 +00:00
if ( --n == 0) {
c = GCOUNT;
2004-11-20 13:30:13 +00:00
Syslog('z', "zgethdr: garbage count exceeded");
2004-11-12 21:25:59 +00:00
goto fifi;
}
goto startover;
case ZPAD|0200: /* This is what we want. */
Not8bit = c;
case ZPAD: /* This is what we want. */
break;
2004-11-07 13:21:59 +00:00
}
2004-11-12 21:25:59 +00:00
}
cancount = 5;
2004-11-07 13:21:59 +00:00
splat:
2004-11-12 21:25:59 +00:00
switch (c = noxrd7()) {
case ZPAD: goto splat;
2004-11-07 13:21:59 +00:00
case HANGUP:
2004-11-12 21:25:59 +00:00
case TIMEOUT: goto fifi;
default: goto agn2;
case ZDLE: /* This is what we want. */
break;
}
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
Rxframeind = c = noxrd7();
switch (c) {
2004-11-20 13:30:13 +00:00
case ZBIN32: Crc32r = 1;
2004-11-12 21:25:59 +00:00
c = zrbhd32(shdr);
break;
2004-11-07 13:21:59 +00:00
case HANGUP:
2004-11-12 21:25:59 +00:00
case TIMEOUT: goto fifi;
2004-11-20 13:30:13 +00:00
case ZBIN: Crc32r = 0;
2004-11-12 21:25:59 +00:00
c = zrbhdr(shdr);
break;
2004-11-20 13:30:13 +00:00
case ZHEX: Crc32r = 0;
2004-11-12 21:25:59 +00:00
c = zrhhdr(shdr);
break;
case CAN: goto gotcan;
default: goto agn2;
}
2004-11-20 13:30:13 +00:00
for (n = 4; ++n < ZMAXHLEN; ) /* Clear unused hdr bytes */
2004-11-12 21:25:59 +00:00
shdr[n] = 0;
Rxpos = shdr[ZP3] & 0377;
Rxpos = (Rxpos<<8) + (shdr[ZP2] & 0377);
Rxpos = (Rxpos<<8) + (shdr[ZP1] & 0377);
Rxpos = (Rxpos<<8) + (shdr[ZP0] & 0377);
2004-11-07 13:21:59 +00:00
fifi:
2004-11-12 21:25:59 +00:00
switch (c) {
case GOTCAN: c = ZCAN;
2004-11-07 13:21:59 +00:00
/* **** FALL THRU TO **** */
case ZNAK:
case ZCAN:
case TERROR:
case TIMEOUT:
2004-11-12 21:25:59 +00:00
case HANGUP: Syslog('+', "Zmodem: Got %s", frametypes[c+FTOFFSET]);
/* **** FALL THRU TO **** */
default: if (c >= -FTOFFSET && c <= FRTYPES)
2004-11-20 13:30:13 +00:00
Syslog('z', "zgethdr: %c %s %lx", Rxframeind, frametypes[c+FTOFFSET], Rxpos);
2004-11-12 21:25:59 +00:00
else
2004-11-20 13:30:13 +00:00
Syslog('z', "zgethdr: %d %d %lx", Rxframeind, c, Rxpos);
2004-11-12 21:25:59 +00:00
}
/* Use variable length headers if we got one */
if (c >= 0 && c <= FRTYPES && Rxframeind & 040) {
Syslog('z', "zgethdr: Usevhdrs");
}
return c;
2004-11-07 13:21:59 +00:00
}
/*
* Receive a binary style header (type and position)
*/
int zrbhdr(register char *shdr)
{
2004-11-12 21:25:59 +00:00
register int c, n;
register unsigned short crc;
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
if ((c = zdlread()) & ~0377)
return c;
Rxtype = c;
crc = updcrc16(c, 0);
2004-11-20 13:30:13 +00:00
for (n=4; --n >= 0; ++shdr) {
2004-11-07 13:21:59 +00:00
if ((c = zdlread()) & ~0377)
2004-11-12 21:25:59 +00:00
return c;
2004-11-07 13:21:59 +00:00
crc = updcrc16(c, crc);
2004-11-12 21:25:59 +00:00
*shdr = c;
}
if ((c = zdlread()) & ~0377)
return c;
crc = updcrc16(c, crc);
if ((c = zdlread()) & ~0377)
return c;
crc = updcrc16(c, crc);
if (crc & 0xFFFF) {
Syslog('+', "Zmodem zrbhdr: Bad CRC");
return TERROR;
}
return Rxtype;
2004-11-07 13:21:59 +00:00
}
/*
* Receive a binary style header (type and position) with 32 bit FCS
*/
int zrbhd32(register char *shdr)
{
2004-11-12 21:25:59 +00:00
register int c, n;
register unsigned long crc;
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
if ((c = zdlread()) & ~0377)
return c;
Rxtype = c;
crc = 0xFFFFFFFFL; crc = updcrc32(c, crc);
2004-11-20 13:30:13 +00:00
for (n=4; --n >= 0; ++shdr) {
2004-11-07 13:21:59 +00:00
if ((c = zdlread()) & ~0377)
2004-11-12 21:25:59 +00:00
return c;
crc = updcrc32(c, crc);
*shdr = c;
}
for (n=4; --n >= 0;) {
if ((c = zdlread()) & ~0377)
return c;
crc = updcrc32(c, crc);
}
if (crc != 0xDEBB20E3) {
Syslog('+', "Zmodem zrbhd32: Bad CRC");
return TERROR;
}
return Rxtype;
2004-11-07 13:21:59 +00:00
}
/*
* Receive a hex style header (type and position)
*/
int zrhhdr(char *shdr)
{
2004-11-12 21:25:59 +00:00
register int c;
register unsigned short crc;
register int n;
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
if ((c = zgethex()) < 0)
return c;
Rxtype = c;
crc = updcrc16(c, 0);
2004-11-20 13:30:13 +00:00
for (n=4; --n >= 0; ++shdr) {
2004-11-07 13:21:59 +00:00
if ((c = zgethex()) < 0)
2004-11-12 21:25:59 +00:00
return c;
2004-11-07 13:21:59 +00:00
crc = updcrc16(c, crc);
2004-11-12 21:25:59 +00:00
*shdr = c;
}
if ((c = zgethex()) < 0)
return c;
crc = updcrc16(c, crc);
if ((c = zgethex()) < 0)
return c;
crc = updcrc16(c, crc);
if (crc & 0xFFFF) {
Syslog('+', "Zmodem zrhhdr: Bad CRC");
return TERROR;
}
switch (c = GETCHAR(Rxtimeout)) {
case 0215: Not8bit = c;
/* **** FALL THRU TO **** */
case 015: /* Throw away possible cr/lf */
switch (c = GETCHAR(Rxtimeout)) {
case 012: Not8bit |= c;
}
}
if (c < 0)
return c;
return Rxtype;
2004-11-07 13:21:59 +00:00
}
/*
* Send a byte as two hex digits
*/
void zputhex(register int c)
{
2004-11-12 21:25:59 +00:00
static char digits[] = "0123456789abcdef";
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
BUFFER_BYTE(digits[(c&0xF0)>>4]);
BUFFER_BYTE(digits[(c)&0xF]);
2004-11-07 13:21:59 +00:00
}
/*
* Send character c with ZMODEM escape sequence encoding.
* Escape XON, XOFF. Escape CR following @ (Telenet net escape)
*/
void zsendline(int c)
{
2004-11-12 21:25:59 +00:00
/* Quick check for non control characters */
if (c & 0140)
BUFFER_BYTE(lastsent = c);
else {
switch (c &= 0377) {
case ZDLE: BUFFER_BYTE(ZDLE);
2004-11-07 13:21:59 +00:00
BUFFER_BYTE (lastsent = (c ^= 0100));
break;
2004-11-12 21:25:59 +00:00
case 015:
case 0215: if (!Zctlesc && (lastsent & 0177) != '@')
goto sendit;
/* **** FALL THRU TO **** */
case 020:
case 021:
case 023:
case 0220:
case 0221:
case 0223: BUFFER_BYTE(ZDLE);
2004-11-07 13:21:59 +00:00
c ^= 0100;
2004-11-12 21:25:59 +00:00
sendit:
2004-11-07 13:21:59 +00:00
BUFFER_BYTE(lastsent = c);
break;
2004-11-12 21:25:59 +00:00
default: if (Zctlesc && ! (c & 0140)) {
BUFFER_BYTE(ZDLE);
c ^= 0100;
2004-11-07 13:21:59 +00:00
}
BUFFER_BYTE(lastsent = c);
}
2004-11-12 21:25:59 +00:00
}
2004-11-07 13:21:59 +00:00
}
/* Decode two lower case hex digits into an 8 bit byte value */
int zgethex(void)
{
2004-11-12 21:25:59 +00:00
register int c;
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
c = zgeth1();
return c;
2004-11-07 13:21:59 +00:00
}
int zgeth1(void)
{
2004-11-12 21:25:59 +00:00
register int c, n;
if ((c = noxrd7()) < 0)
return c;
n = c - '0';
if (n > 9)
n -= ('a' - ':');
if (n & ~0xF)
return TERROR;
if ((c = noxrd7()) < 0)
2004-11-07 13:21:59 +00:00
return c;
2004-11-12 21:25:59 +00:00
c -= '0';
if (c > 9)
c -= ('a' - ':');
if (c & ~0xF)
return TERROR;
c += (n<<4);
return c;
2004-11-07 13:21:59 +00:00
}
/*
* Read a byte, checking for ZMODEM escape encoding
* including CAN*5 which represents a quick abort
*/
int zdlread(void)
{
2004-11-12 21:25:59 +00:00
register int c;
2004-11-07 13:21:59 +00:00
again:
2004-11-12 21:25:59 +00:00
/* Quick check for non control characters */
if ((c = GETCHAR(Rxtimeout)) & 0140)
return c;
switch (c) {
case ZDLE: break;
2004-11-20 13:30:13 +00:00
case XON:
case XON|0200:
case XOFF:
case XOFF|0200: goto again;
2004-11-12 21:25:59 +00:00
default: if (Zctlesc && !(c & 0140)) {
2004-11-07 13:21:59 +00:00
goto again;
2004-11-12 21:25:59 +00:00
}
return c;
}
2004-11-07 13:21:59 +00:00
again2:
2004-11-12 21:25:59 +00:00
if ((c = GETCHAR(Rxtimeout)) < 0)
return c;
if (c == CAN && (c = GETCHAR(Rxtimeout)) < 0)
return c;
if (c == CAN && (c = GETCHAR(Rxtimeout)) < 0)
return c;
if (c == CAN && (c = GETCHAR(Rxtimeout)) < 0)
return c;
switch (c) {
case CAN: return GOTCAN;
2004-11-07 13:21:59 +00:00
case ZCRCE:
case ZCRCG:
case ZCRCQ:
2004-11-12 21:25:59 +00:00
case ZCRCW: return (c | GOTOR);
case ZRUB0: return 0177;
case ZRUB1: return 0377;
2004-11-20 13:30:13 +00:00
case XON:
case XON|0200:
case XOFF:
case XOFF|0200: goto again2;
2004-11-12 21:25:59 +00:00
default: if (Zctlesc && ! (c & 0140)) {
2004-11-07 13:21:59 +00:00
goto again2;
2004-11-12 21:25:59 +00:00
}
if ((c & 0140) == 0100)
2004-11-07 13:21:59 +00:00
return (c ^ 0100);
2004-11-12 21:25:59 +00:00
break;
}
Syslog('+', "Zmodem: Bad escape sequence 0x%x", c);
return TERROR;
2004-11-07 13:21:59 +00:00
}
/*
* Read a character from the modem line with timeout.
* Eat parity, XON and XOFF characters.
*/
int noxrd7(void)
{
2004-11-12 21:25:59 +00:00
register int c;
for (;;) {
if ((c = GETCHAR(Rxtimeout)) < 0)
return c;
switch (c &= 0177) {
case XON:
case XOFF: continue;
default: if (Zctlesc && !(c & 0140))
continue;
case '\r':
case '\n':
case ZDLE: return c;
2004-11-07 13:21:59 +00:00
}
2004-11-12 21:25:59 +00:00
}
2004-11-07 13:21:59 +00:00
}
/*
* Store long integer pos in Txhdr
*/
void stohdr(long pos)
{
2004-11-12 21:25:59 +00:00
Txhdr[ZP0] = pos;
Txhdr[ZP1] = pos>>8;
Txhdr[ZP2] = pos>>16;
Txhdr[ZP3] = pos>>24;
2004-11-07 13:21:59 +00:00
}
/*
* Recover a long integer from a header
*/
long rclhdr(register char *shdr)
{
2004-11-12 21:25:59 +00:00
register long l;
2004-11-07 13:21:59 +00:00
2004-11-12 21:25:59 +00:00
l = (shdr[ZP3] & 0377);
l = (l << 8) | (shdr[ZP2] & 0377);
l = (l << 8) | (shdr[ZP1] & 0377);
l = (l << 8) | (shdr[ZP0] & 0377);
return l;
2004-11-07 13:21:59 +00:00
}
/* End of zmmisc.c */