⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 xmodem_old.c

📁 Tornado的源代码资源包
💻 C
字号:
/*
	Copyright 2001, 2002 Georges Menie (www.menie.org)

    This program is free software; you can redistribute it and/or modify
    it under the terms of the GNU Lesser General Public License as published by
    the Free Software Foundation; either version 2 of the License, or
    (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU Lesser General Public License for more details.

    You should have received a copy of the GNU Lesser General Public License
    along with this program; if not, write to the Free Software
    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
*/

/* this code needs standard functions memcpy() and memset()
   and input/output functions port_inbyte() and port_outbyte().

   the prototypes of the input/output functions are:
     int port_inbyte(unsigned short timeout); // msec timeout
     void port_outbyte(int c);

 */

#include "crc16.h"

#define SOH  0x01
#define STX  0x02
#define EOT  0x04
#define ACK  0x06
#define NAK  0x15
#define CAN  0x18
#define CTRLZ 0x1A

#define DLY_1S 1000
#define MAXRETRANS 25
static int last_error = 0;
/****************Portting Start *******************/
#include "string.h"

#include "vxworks.h"
#include "tylib.h"


int  consoleFd;

void serial_init( )
{
	consoleFd=open("/tyCo/0",O_RDWR,0666);
	ioctl(consoleFd,FIOSETOPTIONS,OPT_RAW);
	ioctl(consoleFd,FIOFLUSH,0);
}

/* Test if there's chars at serial port */
int sio_ischar()
{
	int cnt;
	ioctl (consoleFd, FIONREAD, (int) &cnt);
	return cnt;
}

void port_outbyte(unsigned char trychar)
{
	unsigned char keyout;
	keyout = trychar;
	write(consoleFd, &keyout, 1);
}

unsigned char port_inbyte(unsigned int time_out)
{
	unsigned char ch;
	
	last_error = 0;

	while( !sio_ischar() );
	read(consoleFd,&ch,1);

	last_error = 1;
	return ch;
}
/****************Portting End*******************/
static int check(int crc, const unsigned char *buf, int sz)
{
	if (crc) 
	{
		unsigned short crc = crc16_ccitt(buf, sz);
		unsigned short tcrc = (buf[sz]<<8)+buf[sz+1];
		if (crc == tcrc)
			return 1;
	}
	else 
	{
		int i;
		unsigned char cks = 0;
		for (i = 0; i < sz; ++i) 
		{
			cks += buf[i];
		}
		if (cks == buf[sz])
		return 1;
	}

	return 0;
}

static void flushinput(void)
{
	while (port_inbyte(((DLY_1S)*3)>>1) >= 0)
		;
}

int xmodemReceive(unsigned char *dest, int destsz)
{
	unsigned char xbuff[1030]; /* 1024 for XModem 1k + 3 head chars + 2 crc + nul */
	unsigned char *p;
	int bufsz, crc = 0;
	unsigned char trychar = 'C';
	unsigned char packetno = 1;
	int i, c, len = 0;
	int retry, retrans = MAXRETRANS;

	for(;;) 
	{
		for( retry = 0; retry < 16; ++retry) 
		{
			if (trychar) 
				port_outbyte(trychar);
			c = port_inbyte((DLY_1S)<<1);
			if (last_error == 0) 
			{
				switch (c) 
				{
					case SOH:
						bufsz = 128;
						goto start_recv;
					case STX:
						bufsz = 1024;
						goto start_recv;
					case EOT:
						flushinput();
						port_outbyte(ACK);
						return len; /* normal end */
					case CAN:
						c = port_inbyte(DLY_1S);

						if (c == CAN) 
						{
							flushinput();
							port_outbyte(ACK);
							return -1; /* canceled by remote */
						}
						break;
					default:
						break;
				}
			}
		}
		if (trychar == 'C') 
		{ 
			trychar = NAK; 
			continue; 
		}
		flushinput();
		port_outbyte(CAN);
		port_outbyte(CAN);
		port_outbyte(CAN);
		return -2; /* sync error */

	start_recv:
		if (trychar == 'C') crc = 1;
		trychar = 0;
		p = xbuff;
		*p++ = c;
		for (i = 0;  i < (bufsz+(crc?1:0)+3); ++i) 
		{
			c = port_inbyte(DLY_1S);

			if (last_error != 0) 
				goto reject;
			*p++ = c;
		}

		if (xbuff[1] == (unsigned char)(~xbuff[2]) && 
			(xbuff[1] == packetno || xbuff[1] == (unsigned char)packetno-1) &&
			check(crc, &xbuff[3], bufsz)) 
		{
			if (xbuff[1] == packetno)	
			{
				int count = destsz - len;
				if (count > bufsz) 
					count = bufsz;
				if (count > 0) 
				{
					memcpy (&dest[len], &xbuff[3], count);
					len += count;
				}
				++packetno;
				retrans = MAXRETRANS+1;
			}
			if (--retrans <= 0) 
			{
				flushinput();
				port_outbyte(CAN);
				port_outbyte(CAN);
				port_outbyte(CAN);
				return -3; /* too many retry error */
			}
			port_outbyte(ACK);
			continue;
		}
	reject:
		flushinput();
		port_outbyte(NAK);
	}
}

int xmodemTransmit(unsigned char *src, int srcsz)
{
	unsigned char xbuff[1030]; /* 1024 for XModem 1k + 3 head chars + 2 crc + nul */
	int bufsz, crc = -1;
	unsigned char packetno = 1;
	int i, c, len = 0;
	int retry;

	for(;;) {
		for( retry = 0; retry < 16; ++retry) 
		{
			c = port_inbyte((DLY_1S)<<1);
			if (last_error == 0) 
			{
				switch (c) 
				{
					case 'C':
						crc = 1;
						goto start_trans;
					case NAK:
						crc = 0;
						goto start_trans;
					case CAN:
						c = port_inbyte(DLY_1S);
						if (c == CAN) 
						{
							port_outbyte(ACK);
							flushinput();
							return -1; /* canceled by remote */
						}
						break;
					default:
						break;
				}
			}
		}
		port_outbyte(CAN);
		port_outbyte(CAN);
		port_outbyte(CAN);
		flushinput();
		return -2; /* no sync */

		for(;;) 
		{
		start_trans:
			xbuff[0] = SOH; bufsz = 128;
			xbuff[1] = packetno;
			xbuff[2] = ~packetno;
			c = srcsz - len;
			if (c > bufsz) c = bufsz;
			if (c >= 0) 
			{
				memset (&xbuff[3], 0, bufsz);
				if (c == 0) 
				{
					xbuff[3] = CTRLZ;
				}
				else 
				{
					memcpy (&xbuff[3], &src[len], c);
					if (c < bufsz) xbuff[3+c] = CTRLZ;
				}
				if (crc) 
				{
					unsigned short ccrc = crc16_ccitt(&xbuff[3], bufsz);
					xbuff[bufsz+3] = (ccrc>>8) & 0xFF;
					xbuff[bufsz+4] = ccrc & 0xFF;
				}
				else 
				{
					unsigned char ccks = 0;
					for (i = 3; i < bufsz+3; ++i) 
					{
						ccks += xbuff[i];
					}
					xbuff[bufsz+3] = ccks;
				}
				for (retry = 0; retry < MAXRETRANS; ++retry) 
				{
					for (i = 0; i < bufsz+4+(crc?1:0); ++i) 
					{
						port_outbyte(xbuff[i]);
					}
					c = port_inbyte(DLY_1S);
					if (last_error == 0 ) 
					{
						switch (c) 
						{
							case ACK:
								++packetno;
								len += bufsz;
								goto start_trans;
							case CAN:
								c = port_inbyte(DLY_1S);
								if ( c == CAN) 
								{
									port_outbyte(ACK);
									flushinput();
									return -1; /* canceled by remote */
								}
								break;
							case NAK:
							default:
								break;
						}
					}
				}
				port_outbyte(CAN);
				port_outbyte(CAN);
				port_outbyte(CAN);
				flushinput();
				return -4; /* xmit error */
			}
			else 
			{
				for (retry = 0; retry < 10; ++retry) 
				{
					port_outbyte(EOT);
					c = port_inbyte((DLY_1S)<<1);
					if (c == ACK) break;
				}
				flushinput();
				return (c == ACK)?len:-5;
			}
		}
	}
}

void getfile()
{
    char text[3000];
	int i;
	serial_init();
	xmodemReceive(text,12);
	for (i = strlen(text)-2; i> 0; i--)
		if (text[i] == 26)
			text[i] = 0;
	printf("%s\n",text);
	
	return;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -