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.
magicka/Xmodem/zmodemt.c

1153 lines
26 KiB
C
Raw Normal View History

2016-04-02 23:26:17 +00:00
#ifndef lint
static const char rcsid[] = "$Id: zmodemt.c,v 1.1.1.1 2001/03/08 00:01:48 efalk Exp $" ;
#endif
/*
* Copyright (c) 1995 by Edward A. Falk
*/
/**********
*
*
* @@@@@ @ @ @@@ @@@@ @@@@@ @ @ @@@@@
* @ @@ @@ @ @ @ @ @ @@ @@ @
* @ @ @ @ @ @ @ @ @@@ @ @ @ @
* @ @ @ @ @ @ @ @ @ @ @ @ @
* @@@@@ @ @ @ @@@ @@@@ @@@@@ @ @ @ @
*
* ZMODEMT - transmit side of zmodem protocol
*
* transmit side of zmodem protocol
*
* Caller sets flags defined in zmodem.h as appropriate.
* (default is basic zmodem)
*
* functions:
*
* ZmodemTInit(ZModem *info)
* YmodemTInit(ZModem *info)
* XmodemTInit(ZModem *info)
* Initiate a connection
*
* ZmodemTFile(char *filename, u_char flags[4], ZModem *info)
* Initiate a file transfer. Flags are as specified
* under "ZCBIN" in zmodem.h
*
* ZmodemTFinish(ZModem *info)
* last file
*
* ZmodemTAbort(ZModem *info)
* abort transfer
*
* all functions return 0 on success, 1 on abort
*
*
* Edward A. Falk
*
* January, 1995
*
*
*
**********/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/stat.h>
#include "zmodem.h"
#include "crctab.h"
int SendMoreFileData( register ZModem *info ) ;
extern int ZXmitData(int, int, u_char, u_char *data, ZModem *) ;
extern void ZFlowControl(int onoff, ZModem *info) ;
static int YXmitData() ;
static int YSendFilename() ;
static int YSendData() ;
static int YSendFin() ;
static int sendFilename() ;
static u_char zeros[4] = {0,0,0,0} ;
/* called by user to establish protocol */
int
ZmodemTInit( register ZModem *info )
{
int err ;
int i ;
info->state = TStart ;
info->Protocol = ZMODEM ;
info->crc32 = 0 ;
info->packetCount = 0 ;
info->errCount = 0 ;
info->escCtrl = info->escHibit = info->atSign = info->escape = 0 ;
info->InputState = Idle ;
info->canCount = info->chrCount = 0 ;
info->windowCount = 0 ;
info->filename = NULL ;
info->bufsize = 0 ;
info->interrupt = 0 ;
info->waitflag = 0 ;
if( info->packetsize == 0 )
info->packetsize = 256 ;
/* we won't be receiving much data, pick a reasonable buffer
* size (largest packet will do)
*/
i = info->packetsize*2 ;
if( i < 1024 ) i = 1024 ;
info->buffer = (u_char *)malloc(i) ;
ZIFlush(info) ;
/* optional: send "rz\r" to remote end */
if( DoInitRZ ) {
if( (err = ZXmitStr((u_char *)"rz\r", 3, info)) )
return err ;
}
if( (err = ZXmitHdr(ZRQINIT, ZHEX, zeros, info)) ) /* nudge receiver */
return err ;
info->timeout = 60 ;
zmodemlog("ZmodemTInit[%s]: sent ZRQINIT\n", sname(info)) ;
return 0 ;
}
/* called by user to establish Ymodem protocol */
int
YmodemTInit( register ZModem *info )
{
info->state = YTStart ;
info->Protocol = YMODEM ;
info->errCount = 0 ;
info->InputState = Ysend ;
info->canCount = info->chrCount = 0 ;
info->windowCount = 0 ;
info->filename = NULL ;
if( info->packetsize != 1024 )
info->packetsize = 128 ;
info->buffer = (u_char *)malloc(1024) ;
ZIFlush(info) ;
ZFlowControl(0, info) ;
info->timeout = 60 ;
return 0 ;
}
/* called by user to establish Xmodem protocol */
int
XmodemTInit( register ZModem *info )
{
(void) YmodemTInit(info) ;
info->Protocol = XMODEM ;
return 0 ;
}
/* called by user to begin transmission of a file */
int
ZmodemTFile(
char *file,
char *rfile,
u_int f0,
u_int f1,
u_int f2,
u_int f3,
int filesRem,
int bytesRem,
ZModem *info)
{
if( file == NULL || (info->file = fopen(file, "r")) == NULL )
return ZmErrCantOpen ;
info->fileEof = 0 ;
info->filename = file ;
info->rfilename = (rfile != NULL) ? rfile : "noname" ;
info->filesRem = filesRem ;
info->bytesRem = bytesRem ;
info->fileFlags[3] = f0 ;
info->fileFlags[2] = f1 ;
info->fileFlags[1] = f2 ;
info->fileFlags[0] = f3 ;
info->offset = info->lastOffset = 0 ;
info->len = info->date = info->fileType = info->mode = 0 ;
if( info->filename != NULL )
{
struct stat buf ;
if( stat(info->filename, &buf) == 0 ) {
info->len = buf.st_size ;
info->date = buf.st_mtime ;
info->fileType = 0 ;
info->mode = (buf.st_mode&0777)|0100000 ;
}
}
if( info->Protocol == XMODEM )
return YSendData(info) ;
if( info->Protocol == YMODEM )
return YSendFilename(info) ;
info->state = FileWait ;
zmodemlog("ZmodemTFile[%s]: send ZFILE(%s)\n",
sname(info), info->rfilename) ;
return sendFilename(info) ;
}
/* send ZFILE header and filename & info. Wait for response
* from receiver.
*/
static int
sendFilename( ZModem *info )
{
int err ;
int i ;
u_char obuf[2048] ;
u_char *ptr = obuf ;
info->state = FileWait ;
if( (err = ZXmitHdr(ZFILE, ZBIN, info->fileFlags, info)) )
return err ;
i = strlen(info->rfilename) ;
memcpy(ptr, info->rfilename, i+1) ; ptr += i+1 ;
sprintf((char *)ptr, "%d %lo %o 0 %d %d 0", info->len, info->date,
info->mode, info->filesRem, info->bytesRem) ;
ptr += strlen((char *)ptr) ;
*ptr++ = '\0' ;
return ZXmitData(ZBIN, ptr-obuf, ZCRCW, obuf, info) ;
}
/* called by user when there are no more files to send */
int
ZmodemTFinish( ZModem *info )
{
if( info->Protocol == XMODEM )
return ZmDone ;
if( info->Protocol == YMODEM )
return YSendFin(info) ;
info->state = TFinish ;
if( info->buffer != NULL ) {
free(info->buffer) ;
info->buffer = NULL ;
}
zmodemlog("ZmodemTFinish[%s]: send ZFIN\n", sname(info)) ;
return ZXmitHdr(ZFIN, ZHEX, zeros, info) ;
}
extern int ZPF() ;
extern int AnswerChallenge() ;
extern int GotAbort() ;
extern int Ignore() ;
extern int RetDone() ;
extern int GotCommand() ;
extern int GotStderr() ;
static int GotRinit() ;
static int SendZSInit() ;
static int SendFileCrc() ;
static int GotSendAck() ;
static int GotSendDoneAck() ;
static int GotSendNak() ;
static int GotSendWaitAck() ;
static int SkipFile() ;
static int GotSendPos() ;
static int SendFileData() ;
static int ResendEof() ;
static int OverAndOut() ;
/* sent ZRQINIT, waiting for response */
StateTable TStartOps[] = {
{ZRINIT,GotRinit,1,1,TStart},
{ZCHALLENGE,AnswerChallenge,1,0,TStart},
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{ZNAK,Ignore,0,0,TStart},
{ZCOMMAND,GotCommand,0,0,CommandData},
{ZSTDERR,GotStderr,0,0,StderrData},
{99,ZPF,0,0,TStart},
} ;
/* sent ZSINIT, waiting for response */
StateTable TInitOps[] = {
{ZACK,RetDone,1,0,TInit},
{ZNAK,SendZSInit,1,0,TInit},
{ZRINIT,GotRinit,1,1,TInit}, /* redundant, but who cares */
{ZCHALLENGE,AnswerChallenge,1,0,TInit},
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{ZCOMMAND,GotCommand,0,0,CommandData},
{ZSTDERR,GotStderr,0,0,StderrData},
{99,ZPF,0,0,TInit},
} ;
/* sent ZFILE, waiting for response */
StateTable FileWaitOps[] = {
{ZRPOS,SendFileData,1,0,Sending},
{ZSKIP,SkipFile,1,0,FileWait},
{ZCRC,SendFileCrc,1,0,FileWait},
{ZNAK,sendFilename,1,0,FileWait},
{ZRINIT,sendFilename,1,1,FileWait}, /* rcvr confused, retry file */
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{ZCHALLENGE,AnswerChallenge,1,0,FileWait},
{ZCOMMAND,GotCommand,0,0,CommandData},
{ZSTDERR,GotStderr,0,0,StderrData},
{99,ZPF,0,0,FileWait},
} ;
/* sent file CRC, waiting for response */
StateTable CrcWaitOps[] = {
{ZRPOS,SendFileData,1,0,Sending},
{ZSKIP,SkipFile,1,0,FileWait},
{ZNAK,SendFileCrc,1,0,CrcWait},
{ZRINIT,sendFilename,1,1,FileWait}, /* rcvr confused, retry file */
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{ZCRC,SendFileCrc,0,0,CrcWait},
{ZCHALLENGE,AnswerChallenge,0,0,CrcWait},
{ZCOMMAND,GotCommand,0,0,CommandData},
{ZSTDERR,GotStderr,0,0,StderrData},
{99,ZPF,0,0,CrcWait},
} ;
/* sending data, interruptable */
StateTable SendingOps[] = {
{ZACK,GotSendAck,0,0,Sending},
{ZRPOS,GotSendPos,1,1,Sending},
{ZSKIP,SkipFile,1,1,FileWait},
{ZNAK,GotSendNak,1,1,Sending},
{ZRINIT,sendFilename,1,1,FileWait}, /* rcvr confused, retry file */
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{99,ZPF,0,0,SendWait},
} ;
/* sent data, need to send EOF */
StateTable SendDoneOps[] = {
{ZACK,GotSendDoneAck,0,0,SendWait},
{ZRPOS,GotSendPos,1,1,Sending},
{ZSKIP,SkipFile,1,1,FileWait},
{ZNAK,GotSendNak,1,1,Sending},
{ZRINIT,sendFilename,1,1,FileWait}, /* rcvr confused, retry file */
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{99,ZPF,0,0,SendWait},
} ;
/* sending data, waiting for ACK */
StateTable SendWaitOps[] = {
{ZACK,GotSendWaitAck,0,0,Sending},
{ZRPOS,GotSendPos,0,0,SendWait},
{ZSKIP,SkipFile,1,1,FileWait},
{ZNAK,GotSendNak,0,0,Sending},
{ZRINIT,sendFilename,1,1,FileWait}, /* rcvr confused, retry file */
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{99,ZPF,0,0,SendWait},
} ;
/* sent ZEOF, waiting for new RINIT */
StateTable SendEofOps[] = {
{ZRINIT,SkipFile,1,0,TStart}, /* successful completion */
{ZACK,Ignore,0,0,SendEof}, /* probably ACK from last packet */
{ZRPOS,GotSendPos,1,1,SendWait},
{ZSKIP,SkipFile,1,1,TStart},
{ZNAK,ResendEof,1,0,SendEof},
{ZRINIT,sendFilename,1,1,FileWait}, /* rcvr confused, retry file */
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{99,ZPF,0,0,SendEof},
} ;
StateTable TFinishOps[] = {
{ZFIN,OverAndOut,1,1,Done},
{ZNAK,ZmodemTFinish,1,1,TFinish},
{ZRINIT,ZmodemTFinish,1,1,TFinish},
{ZABORT,GotAbort,1,1,TFinish},
{ZFERR,GotAbort,1,1,TFinish},
{99,ZPF,0,0,TFinish},
} ;
extern char *hdrnames[] ;
/* Received Rinit. Usually received while in start state,
* this can also be an attempt to resync after a protocol
* failure
*/
static int
GotRinit( register ZModem *info )
{
info->bufsize = info->hdrData[1] + info->hdrData[2]*256 ;
info->zrinitflags = info->hdrData[4] + info->hdrData[3]*256 ;
info->crc32 = info->zrinitflags & CANFC32 ;
info->escCtrl = info->zrinitflags & ESCCTL ;
info->escHibit = info->zrinitflags & ESC8 ;
/* Full streaming: If receiver can overlap I/O, and if
* the sender can sample the reverse channel without hanging,
* and the receiver has not specified a buffer size, then we
* can simply blast away with ZCRCG packets. If the receiver
* detects an error, it sends an attn sequence and a new ZRPOS
* header to restart the file where the error occurred.
*
* [note that zmodem8.doc does not define how much noise is
* required to trigger a ZCRCW packet. We arbitrarily choose
* 64 bytes]
*
* If 'windowsize' is nonzero, and the receiver can do full
* duplex, ZCRCQ packets are sent instead of ZCRCG, to keep track
* of the number of characters in the queue. If the queue fills
* up, we pause and wait for a ZACK.
*
*
* Full Streaming with Reverse Interrupt: If sender cannot
* sample the input stream, then we define an Attn sequence
* that will be used to interrupt transmission.
*
*
* Full Streaming with Sliding Window: If sender cannot
* sample input stream or respond to Attn signal, we send
* several ZCRCQ packets until we're sure the receiver must
* have sent back at least one ZACK. Then we stop sending and
* read that ZACK. Then we send one more packet and so on.
*
*
* Segmented Streaming: If receiver cannot overlap I/O or can't do
* full duplex and has specified a maximum receive buffer size,
* whenever the buffer size is reached, we send a ZCRCW packet.
*
* TODO: what if receiver can't overlap, but hasn't set a buffer
* size?
*
* ZCRCE: CRC next, frame ends, header follows
* ZCRCG: CRC next, frame continues nonstop
* ZCRCQ: CRC next, send ZACK, frame continues nonstop
* ZCRCW: CRC next, send ZACK, frame ends, header follows
*/
ZFlowControl(1,info) ;
if( (info->zrinitflags & (CANFDX|CANOVIO)) == (CANFDX|CANOVIO) &&
(SendSample||SendAttn) &&
info->bufsize == 0 )
{
if( info->windowsize == 0 )
info->Streaming = Full ;
else
info->Streaming = StrWindow ;
}
else if( (info->zrinitflags & (CANFDX|CANOVIO)) == (CANFDX|CANOVIO) &&
info->bufsize == 0 )
{
info->Streaming = SlidingWindow ;
}
else
info->Streaming = Segmented ;
zmodemlog("GotRinit[%s]\n", sname(info)) ;
if( AlwaysSinit || info->zsinitflags != 0 || info->attn != NULL )
return SendZSInit(info) ;
else
return ZmDone ; /* caller now calls ZmodemTFile() */
}
static int
SendZSInit( ZModem *info )
{
int err ;
char *at = (info->attn != NULL) ? info->attn : "" ;
u_char fbuf[4] ;
/* TODO: zmodem8.doc states: "If the ZSINIT header specifies
* ESCCTL or ESC8, a HEX header is used, and the receiver
* activates the specified ESC modes before reading the following
* data subpacket." What does that mean?
*/
zmodemlog("SendZSInit[%s]\n", sname(info)) ;
info->state = TInit ;
fbuf[0] = fbuf[1] = fbuf[2] = 0 ;
fbuf[3] = info->zsinitflags ;
if( (err = ZXmitHdr(ZSINIT, ZBIN, fbuf, info)) ||
(err = ZXmitData(ZBIN, strlen(at)+1, ZCRCW, (u_char *)at, info)) )
return err ;
return 0 ;
}
static int
SendFileCrc( register ZModem *info )
{
u_long crc ;
crc = FileCrc(info->filename) ;
zmodemlog("SendFileCrc[%s]: %lx\n", sname(info), crc) ;
return ZXmitHdrHex(ZCRC, ZEnc4(crc), info) ;
}
/* Utility: start sending data. */
static int
startFileData( register ZModem *info )
{
int err ;
info->offset =
info->lastOffset =
info->zrposOffset = ZDec4(info->hdrData+1) ;
fseek(info->file, info->offset, 0) ;
/* TODO: what if fseek fails? Send EOF? */
zmodemlog("startFileData[%s]: %ld\n", sname(info), info->offset) ;
if( (err = ZXmitHdr(ZDATA, ZBIN, info->hdrData+1, info)) )
return err ;
return SendMoreFileData(info) ;
}
/* send a chunk of file data in response to a ZRPOS. Whether this
* is followed by a ZCRCE, ZCRCG, ZCRCQ or ZCRCW depends on all
* sorts of protocol flags
*/
static int
SendFileData( register ZModem *info )
{
info->waitflag = 0 ;
return startFileData(info) ;
}
/* here if an ACK arrived while transmitting data. Update
* last known receiver offset, and try to send some more
* data.
*/
static int
GotSendAck( register ZModem *info )
{
u_long offset ;
offset = ZDec4(info->hdrData+1) ;
if( offset > info->lastOffset )
info->lastOffset = offset ;
zmodemlog("GotSendAck[%s]: %ld\n", sname(info), info->offset) ;
return 0 ; /* DONT send more data, that will happen
* later anyway */
}
/* here if an ACK arrived after last file packet sent. Send
* the EOF.
*/
static int
GotSendDoneAck( register ZModem *info )
{
u_long offset ;
offset = ZDec4(info->hdrData+1) ;
if( offset > info->lastOffset )
info->lastOffset = offset ;
zmodemlog("GotSendDoneAck[%s]: %ld\n", sname(info), info->offset) ;
info->state = SendEof ;
info->timeout = 60 ;
return ZXmitHdrHex(ZEOF, ZEnc4(info->offset), info) ;
}
/* off to a bad start; ZDATA header was corrupt. Start
* from beginning
*/
static int
GotSendNak( register ZModem *info )
{
info->offset = info->zrposOffset ;
fseek(info->file, info->offset, 0) ;
/* TODO: what if fseek fails? Send EOF? */
zmodemlog("GotSendNak[%s]: %ld\n", sname(info), info->offset) ;
return SendMoreFileData(info) ;
}
/* got a ZSKIP, receiver doesn't want this file. */
static int
SkipFile( register ZModem *info )
{
zmodemlog("SkipFile[%s]\n", sname(info)) ;
fclose(info->file) ;
return ZmDone ;
}
/* got a ZRPOS packet in the middle of sending a file,
* set new offset and try again
*/
static int
GotSendPos( register ZModem *info )
{
ZStatus(DataErr, ++info->errCount, NULL) ;
info->waitflag = 1 ; /* next pkt should wait, to resync */
zmodemlog("GotSendPos[%s]\n", sname(info), info->offset) ;
return startFileData(info) ;
}
/* here if an ACK arrived while waiting while transmitting data.
* Update last known receiver offset, and try to send some more
* data.
*/
static int
GotSendWaitAck( register ZModem *info )
{
u_long offset ;
int err ;
offset = ZDec4(info->hdrData+1) ;
if( offset > info->lastOffset )
info->lastOffset = offset ;
zmodemlog("GotSendWaitAck[%s]\n", sname(info), offset) ;
if( (err = ZXmitHdr(ZDATA, ZBIN, ZEnc4(info->offset), info)) )
return err ;
return SendMoreFileData(info) ;
}
/* utility: send a chunk of file data. Whether this is followed
* by a ZCRCE, ZCRCG, ZCRCQ or ZCRCW depends on all
* sorts of protocol flags, plus 'waitflag'. Exact amount of file
* data transmitted is variable, as escape sequences may pad out
* the buffer.
*/
int
SendMoreFileData( register ZModem *info )
{
int type ;
int qfull = 0 ;
int err ;
int len ; /* max # chars to send this packet */
long pending ; /* # of characters sent but not acknowledged */
/* ZCRCE: CRC next, frame ends, header follows
* ZCRCG: CRC next, frame continues nonstop
* ZCRCQ: CRC next, send ZACK, frame continues nonstop
* ZCRCW: CRC next, send ZACK, frame ends, header follows
*/
if( info->interrupt ) {
/* Bugger, receiver sent an interrupt. Enter a wait state
* and see what they want. Next header *should* be ZRPOS.
*/
info->state = SendWait ;
info->timeout = 60 ;
return 0 ;
}
/* Find out how many bytes we can transfer in the next packet */
len = info->packetsize ;
pending = info->offset - info->lastOffset ;
if( info->windowsize != 0 && info->windowsize - pending <= len ){
len = info->windowsize - pending ;
qfull = 1 ;
}
if( info->bufsize != 0 && info->bufsize - pending <= len ) {
len = info->bufsize - pending ;
qfull = 1 ;
}
if( len == 0 ) { /* window still full, keep waiting */
info->state = SendWait ;
info->timeout = 60 ;
return 0 ;
}
/* OK, we can safely transmit 'len' bytes of data. Start reading
* file until buffer is full.
*/
len -= 10 ; /* Pre-deduct 10 bytes for trailing CRC */
/* find out what kind of packet to send */
if( info->waitflag ) {
type = ZCRCW ;
info->waitflag = 0 ;
}
#ifdef COMMENT
else if( info->fileEof )
type = ZCRCE ;
#endif /* COMMENT */
else if( qfull )
type = ZCRCW ;
else
switch( info->Streaming ) {
case Full:
case Segmented: type = ZCRCG ; break ;
case StrWindow:
if( (info->windowCount += len) < info->windowsize/4 )
type = ZCRCG ;
else {
type = ZCRCQ ;
info->windowCount = 0 ;
}
break ;
default:
case SlidingWindow: type = ZCRCQ ; break ;
}
{
int crc32 = info->crc32 ;
register int c=0, c2, atSign=0 ;
register u_long crc ;
register u_char *ptr = info->buffer ;
crc = crc32 ? 0xffffffff : 0 ;
/* read characters from file and put into buffer until buffer is
* full or file is exhausted
*/
while( len > 0 && (c = getc(info->file)) != EOF )
{
if( !crc32 )
crc = updcrc(c, crc) ;
else
crc = UPDC32(c, crc) ;
/* zmodem protocol requires that CAN(ZDLE), DLE, XON, XOFF and
* a CR following '@' be escaped. In addition, I escape '^]'
* to protect telnet, "<CR>~." to protect rlogin, and ESC for good
* measure.
*/
c2 = c & 0177 ;
if( c == ZDLE || c2 == 020 || c2 == 021 || c2 == 023 ||
c2 == 0177 || c2 == '\r' || c2 == '\n' || c2 == 033 ||
c2 == 035 || (c2 < 040 && info->escCtrl) )
{
*ptr++ = ZDLE ;
if( c == 0177 )
*ptr = ZRUB0 ;
else if( c == 0377 )
*ptr = ZRUB1 ;
else
*ptr = c^0100 ;
len -= 2 ;
}
else {
*ptr = c ;
--len ;
}
++ptr ;
atSign = c2 == '@' ;
++info->offset ;
}
/* if we've reached file end, a ZEOF header will follow. If
* there's room in the outgoing buffer for it, end the packet
* with ZCRCE and append the ZEOF header. If there isn't room,
* we'll have to do a ZCRCW
*/
if( (info->fileEof = (c == EOF)) ) {
if( qfull || (info->bufsize != 0 && len < 24) )
type = ZCRCW ;
else
type = ZCRCE ;
}
*ptr++ = ZDLE ;
if( !crc32 )
crc = updcrc(type, crc) ;
else
crc = UPDC32(type, crc) ;
*ptr++ = type ;
if( !crc32 ) {
crc = updcrc(0,crc) ; crc = updcrc(0,crc) ;
ptr = putZdle(ptr, (crc>>8)&0xff, info) ;
ptr = putZdle(ptr, crc&0xff, info) ;
}
else {
crc = ~crc ;
for(len=4; --len >= 0; crc >>= 8)
ptr = putZdle(ptr, crc&0xff, info) ;
}
len = ptr - info->buffer ;
}
ZStatus(SndByteCount, info->offset, NULL) ;
if( (err = ZXmitStr(info->buffer, len, info)) )
return err ;
#ifdef COMMENT
if( (err = ZXmitData(ZBIN, info->buffer, len, type, info)) )
return err ;
#endif /* COMMENT */
/* finally, do we want to wait after this packet? */
switch( type ) {
case ZCRCE:
info->state = SendEof ;
info->timeout = 60 ;
return ZXmitHdrHex(ZEOF, ZEnc4(info->offset), info) ;
case ZCRCW:
info->state = info->fileEof ? SendDone : SendWait ;
info->timeout = 60 ;
break ;
default:
info->state = Sending ;
info->timeout = 0 ;
break ;
}
#ifdef COMMENT
if( info->fileEof ) { /* Yes, file is done, send EOF and wait */
info->state = SendEof ;
info->timeout = 60 ;
return ZXmitHdrHex(ZEOF, ZEnc4(info->offset), info) ;
}
else if( type == ZCRCW ) {
info->state = SendWait ;
info->timeout = 60 ;
}
else {
info->state = Sending ;
info->timeout = 0 ;
}
#endif /* COMMENT */
return 0 ;
}
static int
ResendEof( register ZModem *info )
{
return ZXmitHdrHex(ZEOF, ZEnc4(info->offset), info) ;
}
static int
OverAndOut( ZModem *info )
{
ZXmitStr((u_char *)"OO", 2, info) ;
return ZmDone ;
}
/* YMODEM */
static u_char eotstr[1] = {EOT} ;
/* ymodem parser */
int
YsendChar( char c, register ZModem *info )
{
int err ;
if( info->canCount >= 2 ) {
ZStatus(RmtCancel, 0, NULL) ;
return ZmErrCancel ;
}
switch( info->state ) {
case YTStart: /* wait for 'G', 'C' or NAK */
switch( c ) {
case 'G': /* streaming YModem */
case 'C': /* CRC YModem */
case NAK: /* checksum YModem */
info->PacketType = c ;
return ZmDone ;
default:
return 0 ;
}
case YTFile: /* sent filename, waiting for ACK or NAK */
switch( c ) {
case NAK: /* resend */
case 'C':
case 'G':
ZStatus(DataErr, ++info->errCount, NULL) ;
return YSendFilename(info) ;
case ACK:
info->state = YTDataWait ;
default:
return 0 ;
}
case YTDataWait: /* sent filename, waiting for G,C or NAK */
switch( c ) {
case NAK:
case 'C':
case 'G':
info->chrCount = 0 ;
if( info->PacketType == 'G' ) /* send it all at once */
{
while( info->state == YTData )
if( (err = YSendData(info)) )
return err ;
return 0 ;
}
else
return YSendData(info) ;
default:
return 0 ;
}
case YTData: /* sent data, waiting for ACK or NAK */
switch( c ) {
case 'C':
case 'G': /* protocol failure, resend filename */
if( info->Protocol == YMODEM ) {
ZStatus(DataErr, ++info->errCount, NULL) ;
info->state = YTFile ;
rewind(info->file) ;
return YSendFilename(info) ;
}
/* else XModem, treat it like a NAK */
case NAK:
ZStatus(DataErr, ++info->errCount, NULL) ;
return YXmitData(info->buffer + info->bufp, info->ylen, info) ;
case ACK:
info->offset += info->ylen ;
info->bufp += info->ylen ;
info->chrCount -= info->ylen ;
ZStatus(SndByteCount, info->offset, NULL) ;
return YSendData(info) ;
default:
return 0 ;
}
case YTEOF: /* sent EOF, waiting for ACK or NAK */
switch( c ) {
case NAK:
return ZXmitStr(eotstr, 1, info) ;
case ACK:
info->state = info->Protocol == YMODEM ? YTStart : Done ;
return ZmDone ;
default:
return 0 ;
}
case YTFin: /* sent Fin, waiting for ACK or NAK */
switch( c ) {
case NAK:
return YSendFin(info) ;
case ACK:
return ZmDone ;
default:
return 0 ;
}
default:
return 0 ;
}
}
static int
YXmitData( u_char *buffer, int len, ZModem *info )
{
u_char hdr[3] ;
u_char trail[2] ;
u_long crc = 0 ;
int i, err ;
hdr[0] = len == 1024 ? STX : SOH ;
hdr[1] = info->packetCount ;
hdr[2] = ~hdr[1] ;
if( (err = ZXmitStr(hdr, 3, info)) ||
(err = ZXmitStr(buffer, len, info)) )
return err ;
if( info->PacketType == NAK ) { /* checksum */
for(i=0; i<len; ++i)
crc += buffer[i] ;
trail[0] = crc % 256 ;
return ZXmitStr(trail,1, info) ;
}
else {
for(i=0; i<len; ++i)
crc = updcrc(buffer[i], crc) ;
crc = updcrc(0,crc) ; crc = updcrc(0,crc) ;
trail[0] = crc / 256 ;
trail[1] = crc % 256 ;
return ZXmitStr(trail,2, info) ;
}
}
static int
YSendFilename( ZModem *info )
{
int i,len ;
u_char obuf[1024] ;
u_char *ptr = obuf ;
info->state = info->PacketType != 'G' ? YTFile : YTDataWait ;
info->packetCount = 0 ;
info->offset = 0 ;
i = strlen(info->rfilename) ;
memcpy(ptr, info->rfilename, i+1) ; ptr += i+1 ;
sprintf((char *)ptr, "%d %lo %o 0", info->len, info->date, info->mode);
ptr += strlen((char *)ptr) ;
*ptr++ = '\0' ;
/* pad out to 128 bytes or 1024 bytes */
i = ptr-obuf ;
len = i>128 ? 1024 : 128 ;
for(; i<len; ++i)
*ptr++ = '\0' ;
return YXmitData(obuf, len, info) ;
}
/* send next buffer of data */
static int
YSendData( ZModem *info )
{
int i ;
/* are there characters still in the read buffer? */
if( info->chrCount <= 0 ) {
info->bufp = 0 ;
info->chrCount = fread(info->buffer, 1, info->packetsize, info->file);
info->fileEof = feof(info->file) ;
}
if( info->chrCount <= 0 ) {
fclose(info->file) ;
info->state = YTEOF ;
return ZXmitStr(eotstr, 1, info) ;
}
/* pad out to 128 bytes if needed */
if( info->chrCount < 128 ) {
i = 128 - info->chrCount ;
memset(info->buffer + info->bufp + info->chrCount, 0x1a, i) ;
info->chrCount = 128 ;
}
info->ylen = info->chrCount >= 1024 ? 1024 : 128 ;
++info->packetCount ;
info->state = YTData ;
return YXmitData(info->buffer + info->bufp, info->ylen, info) ;
}
static int
YSendFin( ZModem *info )
{
u_char obuf[128] ;
info->state = YTFin ;
info->packetCount = 0 ;
memset(obuf,0,128) ;
return YXmitData(obuf, 128, info) ;
}