ckcfns.c
上传用户:dufan58
上传日期:2007-01-05
资源大小:3407k
文件大小:187k
源码类别:

通讯/手机编程

开发平台:

Windows_Unix

  1. char *fnsv = "C-Kermit functions, 7.0.187, 20 Dec 1999";
  2. char *nm[] =  { "Disabled", "Local only", "Remote only", "Enabled" };
  3. /*  C K C F N S  --  System-independent Kermit protocol support functions.  */
  4. /*  ...Part 1 (others moved to ckcfn2,3 to make this module smaller) */
  5. /*
  6.   Author: Frank da Cruz <fdc@columbia.edu>,
  7.   Columbia University Academic Information Systems, New York City.
  8.   Copyright (C) 1985, 2000,
  9.     Trustees of Columbia University in the City of New York.
  10.     All rights reserved.  See the C-Kermit COPYING.TXT file or the
  11.     copyright text in the ckcmai.c module for disclaimer and permissions.
  12. */
  13. /*
  14.  System-dependent primitives defined in:
  15.    ck?tio.c -- terminal (communications) i/o
  16.    cx?fio.c -- file i/o, directory structure
  17. */
  18. #include "ckcsym.h" /* Needed for Stratus VOS */
  19. #include "ckcasc.h" /* ASCII symbols */
  20. #include "ckcdeb.h" /* Debug formats, typedefs, etc. */
  21. #include "ckcker.h" /* Symbol definitions for Kermit */
  22. #include "ckcxla.h" /* Character set symbols */
  23. #include "ckcnet.h"                     /* VMS definition of TCPSOCKET */
  24. int docrc  = 0; /* Accumulate CRC for v(crc16) */
  25. long crc16 = 0L; /* File CRC = v(crc16) */
  26. extern CHAR feol;
  27. extern int byteorder, xflg, remfile, what, fmask, cxseen, czseen;
  28. extern long ffc;
  29. #ifndef NOXFER
  30. /* (move these prototypes to the appropriate .h files...) */
  31. #ifdef COMMENT
  32. /* Not used */
  33. #ifdef VMS
  34. _PROTOTYP( int getvnum, (char *) );
  35. #endif /* VMS */
  36. #endif /* COMMENT */
  37. _PROTOTYP( static int bgetpkt, (int) );
  38. #ifndef NOCSETS
  39. _PROTOTYP( int lookup, (struct keytab[], char *, int, int *) );
  40. #endif /* NOCSETS */
  41. #ifndef NOSPL
  42. _PROTOTYP( int zzstring, (char *, char **, int *) );
  43. #endif /* NOSPL */
  44. #ifdef OS2ORUNIX
  45. _PROTOTYP( long zfsize, (char *) );
  46. #endif /* OS2ORUNIX */
  47. #ifdef OS2
  48. #include <io.h>
  49. #endif /* OS2 */
  50. #ifdef VMS
  51. #include <errno.h>
  52. #endif /* VMS */
  53. /* Externals from ckcmai.c */
  54. extern int srvcdmsg, srvidl, idletmo, bigendian;
  55. extern char * cdmsgfile[];
  56. extern int spsiz, spmax, rpsiz, timint, srvtim, rtimo, npad, ebq, ebqflg,
  57.  rpt, rptq, rptflg, capas, keep, fncact, pkttim, autopar, spsizr, xitsta;
  58. extern int pktnum, bctr, bctu, bctl, clfils, sbufnum, protocol,
  59.  size, osize, spktl, nfils, ckwarn, timef, spsizf, sndtyp, rcvtyp, success;
  60. extern int parity, turn, network, whatru, fsecs, justone, slostart,
  61.  ckdelay, displa, mypadn, moving, recursive, nettype;
  62. extern long filcnt, flci, flco, tlci, tlco, tfc, fsize, sendstart, rs_len;
  63. extern long filrej, oldcps, cps, peakcps, ccu, ccp, calibrate, filestatus;
  64. extern int fblksiz, frecl, frecfm, forg, fcctrl, fdispla, skipbup;
  65. extern int spackets, rpackets, timeouts, retrans, crunched, wmax, wcur;
  66. extern int hcflg, binary, fncnv, b_save, f_save, server;
  67. extern int nakstate, discard, rejection, local, xfermode, interrupted;
  68. extern int rq, rqf, sq, wslots, wslotn, wslotr, winlo, urpsiz, rln;
  69. extern int fnspath, fnrpath, eofmethod, diractive, whatru2, wearealike;
  70. extern int atcapr, atcapb, atcapu;
  71. extern int lpcapr, lpcapb, lpcapu;
  72. extern int swcapr, swcapb, swcapu;
  73. extern int lscapr, lscapb, lscapu;
  74. extern int rscapr, rscapb, rscapu;
  75. extern int rptena, rptmin;
  76. extern int sseqtbl[];
  77. extern int numerrs, nzxopts;
  78. extern long rptn;
  79. extern int maxtry;
  80. extern int stdouf;
  81. extern int sendmode;
  82. extern int carrier, ttprty;
  83. #ifdef TCPSOCKET
  84. extern int ttnproto;
  85. #endif /* TCPSOCKET */
  86. #ifndef NOSPL
  87. extern int sndxin, sndxhi, sndxlo;
  88. #endif /* NOSPL */
  89. extern int g_binary, g_fncnv;
  90. #ifdef GFTIMER
  91. extern CKFLOAT fpfsecs;
  92. #endif /* GFTIMER */
  93. #ifdef OS2
  94. extern struct zattr iattr;
  95. #endif /* OS2 */
  96. #ifdef PIPESEND
  97. extern int pipesend, usepipes;
  98. #endif /* PIPESEND */
  99. /* Per-file text/binary-mode recognition */
  100. #ifdef PATTERNS
  101. extern int patterns;
  102. extern char *txtpatterns[];
  103. extern char *binpatterns[];
  104. #endif /* PATTERNS */
  105. #ifdef STREAMING
  106. extern int streamrq, streaming, streamed, streamok;
  107. #endif /* STREAMING */
  108. extern int reliable, clearrq, cleared, urclear;
  109. extern int
  110.   atenci, atenco, atdati, atdato, atleni, atleno, atblki, atblko,
  111.   attypi, attypo, atsidi, atsido, atsysi, atsyso, atdisi, atdiso;
  112. extern int bigsbsiz, bigrbsiz;
  113. extern char *versio;
  114. extern char *filefile;
  115. extern char whoareu[], * cksysid;
  116. #ifndef NOSERVER
  117. extern int ngetpath;
  118. extern char * getpath[];
  119. extern int fromgetpath;
  120. #endif /* NOSERVER */
  121. extern int inserver;
  122. #ifdef CK_LOGIN
  123. extern int isguest;
  124. #endif /* CK_LOGIN */
  125. extern CHAR *srvcmd, * epktmsg;
  126. extern CHAR padch, mypadc, eol, seol, ctlq, myctlq, sstate, myrptq;
  127. extern CHAR *data, padbuf[], stchr, mystch;
  128. extern CHAR *srvptr;
  129. extern CHAR *rdatap;
  130. extern char *cmarg, *cmarg2, **cmlist, filnam[];
  131. extern char fspec[], *sfspec, *rfspec;
  132. extern int fspeclen;
  133. #ifndef NOMSEND
  134. extern struct filelist * filehead, * filenext;
  135. extern int addlist;
  136. #endif /* NOMSEND */
  137. _PROTOTYP( int lslook, (unsigned int b) ); /* Locking Shift Lookahead */
  138. _PROTOTYP( int szeof, (CHAR *s) );
  139. _PROTOTYP( VOID fnlist, (void) );
  140. #endif /* NOXFER */
  141. /* Character set Translation */
  142. #ifndef NOCSETS
  143. extern int tcharset, fcharset;
  144. extern int ntcsets, xlatype, cseqtab[];
  145. extern struct csinfo tcsinfo[], fcsinfo[];
  146. _PROTOTYP( CHAR ident, (CHAR) ); /* Identity translation function */
  147. /* Arrays of and pointers to character translation functions */
  148. #ifdef CK_ANSIC
  149. extern CHAR (*rx)(CHAR); /* Pointer to input character translation function */
  150. extern CHAR (*sx)(CHAR); /* Pointer to output character translation function */
  151. extern CHAR (*xls[MAXTCSETS+1][MAXFCSETS+1])(CHAR); /* Byte-to-Byte Send */
  152. extern CHAR (*xlr[MAXTCSETS+1][MAXFCSETS+1])(CHAR); /* Byte-to-Byte Recv */
  153. #ifdef UNICODE
  154. extern int (*xut)(USHORT); /* Translation function UCS to TCS */
  155. extern int (*xuf)(USHORT); /* Translation function UCS to FCS */
  156. extern USHORT (*xtu)(CHAR); /* Translation function TCS to UCS */
  157. extern USHORT (*xfu)(CHAR); /* Translation function FCS to UCS */
  158. #endif /* UNICODE */
  159. #else /* The same declarations again for non-ANSI comilers... */
  160. extern CHAR (*rx)();
  161. extern CHAR (*sx)();
  162. extern CHAR (*xls[MAXTCSETS+1][MAXFCSETS+1])();
  163. extern CHAR (*xlr[MAXTCSETS+1][MAXFCSETS+1])();
  164. #ifdef UNICODE
  165. extern int (*xut)();
  166. extern int (*xuf)();
  167. extern USHORT (*xtu)();
  168. extern USHORT (*xfu)();
  169. #endif /* UNICODE */
  170. #endif /* CK_ANSIC */
  171. #endif /* NOCSETS */
  172. /* (PWP) external def. of things used in buffered file input and output */
  173. #ifdef DYNAMIC
  174. extern char *zinbuffer, *zoutbuffer;
  175. #else
  176. extern char zinbuffer[], zoutbuffer[];
  177. #endif /* DYNAMIC */
  178. extern char *zinptr, *zoutptr;
  179. extern int zincnt, zoutcnt, zobufsize;
  180. extern long crcta[], crctb[]; /* CRC-16 generation tables */
  181. extern int rseqtbl[]; /* Rec'd-packet sequence # table */
  182. #ifndef NOXFER
  183. /* Variables defined in this module but shared by other modules. */
  184. int xfrbel = 1;
  185. char * ofperms = ""; /* Output file permissions */
  186. int autopath = 0; /* SET RECEIVE PATHNAMES AUTO flag */
  187. #ifdef CALIBRATE
  188. #define CAL_O 3
  189. #define CAL_M 253
  190. int cal_j = 0;
  191. CHAR
  192. cal_a[] = {
  193.  16, 45, 98,  3, 52, 41, 14,  7, 76,165,122, 11,104, 77,166, 15,
  194. 160, 93, 18, 19,112, 85, 54, 23,232,213, 90, 27, 12, 81,126, 31,
  195.   4,205, 34, 35,144, 73,110, 39, 28,133,218, 43,156, 65,102, 47,
  196.  84, 61, 50, 51,208,117, 86, 55,  8,245, 74, 59, 44,125,222, 63,
  197.  80,  1,162, 67,116,105,206, 71,120,  9,250, 75, 88, 97,  6, 79,
  198. 100,221, 82, 83, 36, 89, 94, 87, 40, 21,106, 91,236,145,150, 95,
  199. 228, 33,130, 99,148,137,198,103,108,169, 42,107,184,129, 78,111,
  200.   0, 49,114,115, 32,121,254,119,172, 57,138,123,152,177, 22,127,
  201. 240,193,  2,131,176,  5, 38,135,204,229, 10,139,200,161,174,143,
  202. 128, 17,146,147, 68,153, 30,151, 72,217,170,155, 24,209, 62,159,
  203.  64,225,194,163,244,201, 70,167,216,197,234,171,188,109,230,175,
  204. 212,113,178,179,132,185,190,183,136,249,202,187, 92,241,118,191,
  205.  48,237, 66,195, 96,233,142,199,248, 37, 58,203, 60, 13,134,207,
  206.  20, 29,210,211,164,149,182,215,220, 25, 26,219,124,157,246,223,
  207. 180,141,226,227,192,101,238,231, 56, 69,154,235,252,173, 46,239,
  208. 224,253,242,243,196, 53,214,247,168,181,186,251,140,189,158,255
  209. };
  210. #endif /* CALIBRATE */
  211. char * rf_err = "Error receiving file"; /* rcvfil() error message */
  212. #ifdef CK_SPEED
  213. short ctlp[256] = { /* Control-Prefix table */
  214.   1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* C0  */
  215.   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* G0  */
  216.   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
  217.   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1, /* DEL */
  218.   1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* C1  */
  219.   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* G1  */
  220.   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
  221.   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1  /* 255 */
  222. };
  223. #endif /* CK_SPEED */
  224. int sndsrc; /* Flag for where to get names of files to send: */
  225. /* -1: znext() function */
  226. /*  0: stdin */
  227. /* >0: list in cmlist or other list */
  228. /* -9: calibrate */
  229. int  memstr; /* Flag for input from memory string */
  230. int  funcstr; /* Flag for input from function */
  231. int  bestlen = 0;
  232. int  maxsend = 0;
  233. int gnf_binary = 0; /* Prevailing xfer mode for gnfile */
  234. #ifdef pdp11
  235. #define MYINITLEN 32
  236. #else
  237. #define MYINITLEN 100
  238. #endif /* pdp11 */
  239. CHAR myinit[MYINITLEN]; /* Copy of my Send-Init data */
  240. /* Variables local to this module */
  241. #ifdef TLOG
  242. #ifndef XYZ_INTERNAL
  243. static
  244. #endif /* XYZ_INTERNAL */
  245. char *fncnam[] = {
  246.   "rename", "replace", "backup", "append", "discard", "ask", "update", ""
  247. };
  248. #endif /* TLOG */
  249. static char *memptr; /* Pointer for memory strings */
  250. #ifdef VMS
  251. extern int batch;
  252. #else
  253. extern int backgrd;
  254. #endif /* VMS */
  255. #ifdef CK_CTRLZ
  256. static int lastchar = 0;
  257. #endif /* CK_CTRLZ */
  258. #ifdef CK_ANSIC
  259. static int (*funcptr)(void); /* Pointer for function strings */
  260. #else
  261. static int (*funcptr)();
  262. #endif /* CK_ANSIC */
  263. #ifdef pdp11
  264. #define CMDSTRL 50
  265. static char cmdstr[50]; /* System command string. */
  266. #else
  267. #ifdef BIGBUFOK
  268. #define CMDSTRL 1024
  269. #else
  270. #define CMDSTRL 256
  271. #endif /* BIGBUFOK */
  272. static char cmdstr[CMDSTRL+1];
  273. #endif /* pdp11 */
  274. static int drain; /* For draining stacked-up ACKs. */
  275. static int first; /* Flag for first char from input */
  276. static CHAR t; /* Current character */
  277. #ifdef COMMENT
  278. static CHAR next; /* Next character */
  279. #endif /* COMMENT */
  280. static int ebqsent = 0; /* 8th-bit prefix bid that I sent */
  281. static int lsstate = 0; /* Locking shift state */
  282. static int lsquote = 0; /* Locking shift quote */
  283. #ifdef datageneral
  284. extern int quiet;
  285. #endif /* datageneral */
  286. /*  E N C S T R  --  Encode a string from memory. */
  287. /*
  288.   Call this instead of getpkt() if source is a string, rather than a file.
  289.   Note: Character set translation is never done in this case.
  290. */
  291. #ifdef COMMENT
  292. #define ENCBUFL 200
  293. #ifndef pdp11
  294. CHAR encbuf[ENCBUFL];
  295. #else
  296. /* This is gross, but the pdp11 root segment is out of space */
  297. /* Will allocate it in ckuusr.c. */
  298. extern CHAR encbuf[];
  299. #endif /* pdp11 */
  300. #endif /* COMMENT */
  301. /*
  302.   Encode packet data from a string in memory rather than from a file.
  303.   Returns the length of the encoded string on success, -1 if the string
  304.   could not be completely encoded into the currently negotiated data
  305.   field length.
  306. */
  307. int
  308. #ifdef CK_ANSIC
  309. encstr(CHAR *s)
  310. #else /* CK_ANSIC */
  311. encstr(s) CHAR* s;
  312. #endif /* CK_ANSIC */
  313. {
  314. /*
  315.   Recoded 30 Jul 94 to use the regular data buffer and the negotiated
  316.   maximum packet size.  Previously we were limited to the length of encbuf[].
  317.   Also, to return a failure code if the entire encoded string would not fit.
  318.   Modified 14 Jul 98 to return length of encoded string.
  319. */
  320.     int m, rc, slen; char *p;
  321.     if (!data) { /* Watch out for null pointers. */
  322. debug(F100,"SERIOUS ERROR: encstr data == NULL","",0);
  323. return(-1);
  324.     }
  325.     if (!s) s = (CHAR *)""; /* Ditto. */
  326.     slen = strlen((char *)s); /* Length of source string. */
  327.     debug(F111,"encstr",s,slen);
  328.     rc = 0; /* Return code. */
  329.     m = memstr; p = memptr; /* Save these. */
  330.     memptr = (char *)s; /* Point to the string. */
  331.     debug(F101,"encstr memptr 1","",memptr);
  332.     memstr = 1; /* Flag memory string as source. */
  333.     first = 1; /* Initialize character lookahead. */
  334.     *data = NUL; /* In case s is empty */
  335.     debug(F101,"encstr spsiz","",spsiz);
  336.     rc = getpkt(spsiz,0); /* Fill a packet from the string. */
  337.     debug(F101,"encstr getpkt rc","",rc);
  338.     if (rc > -1 && memptr < (char *)(s + slen)) { /* Means we didn't encode */
  339. rc = -1; /* the whole string. */
  340. debug(F101,"encstr string too big","",size);
  341.     }
  342.     debug(F101,"encstr getpkt rc","",rc);
  343.     memstr = m; /* Restore memory string flag */
  344.     memptr = p; /* and pointer */
  345.     first = 1; /* Put this back as we found it. */
  346.     return(rc);
  347. }
  348. /*  Output functions passed to 'decode':  */
  349. int        /*  Put character in server command buffer  */
  350. #ifdef CK_ANSIC
  351. putsrv(char c)
  352. #else
  353. putsrv(c) register char c;
  354. #endif /* CK_ANSIC */
  355. /* putsrv */ {
  356.     *srvptr++ = c;
  357.     *srvptr = ''; /* Make sure buffer is null-terminated */
  358.     return(0);
  359. }
  360. int /*  Output character to console.  */
  361. #ifdef CK_ANSIC
  362. puttrm(char c)
  363. #else
  364. puttrm(c) register char c;
  365. #endif /* CK_ANSIC */
  366. /* puttrm */ {
  367. #ifndef NOSPL
  368.     extern int cmdsrc();
  369.     extern char * qbufp; /* If REMOTE QUERY active, */
  370.     extern int query, qbufn; /* also store response in */
  371.     if (query && qbufn++ < 1024) { /* query buffer. */
  372. *qbufp++ = c;
  373. *qbufp = NUL;
  374.     }
  375.     if (!query || !cmdsrc())
  376. #endif /* NOSPL */
  377.       conoc(c);
  378.     return(0);
  379. }
  380. #endif /* NOXFER */
  381. int /*  Output char to file. */
  382. #ifdef CK_ANSIC
  383. putmfil(char c) /* Just like putfil but to ZMFILE */
  384. #else /* rather than ZOFILE... */
  385. putmfil(c) register char c;
  386. #endif /* CK_ANSIC */
  387. /* putmfil */ {
  388.     debug(F000,"putfil","",c);
  389.     if (zchout(ZMFILE, (char) (c & fmask)) < 0) {
  390. czseen = 1;
  391. debug(F101,"putfil zchout write error, setting czseen","",1);
  392. return(-1);
  393.     }
  394.     return(0);
  395. }
  396. int /*  Output char to file. */
  397. #ifdef CK_ANSIC
  398. putfil(char c)
  399. #else
  400. putfil(c) register char c;
  401. #endif /* CK_ANSIC */
  402. /* putfil */ {
  403.     debug(F000,"putfil","",c);
  404.     if (zchout(ZOFILE, (char) (c & fmask)) < 0) {
  405. czseen = 1;    /* If write error... */
  406. debug(F101,"putfil zchout write error, setting czseen","",1);
  407. return(-1);
  408.     }
  409.     return(0);
  410. }
  411. /*
  412.   The following function is a wrapper for putfil().  The only reason for its
  413.   existence is to be passed as a function pointer to decode(), which treats
  414.   putfil() itself specially -- bypassing it and using an internal macro
  415.   instead to speed things up.  Use zputfil() instead of putfil() in cases where
  416.   we do not want this to happen, e.g. when we need to send output to the file
  417.   with a mixture of zchout() and zsout()/zsoutl() calls (as is the case with
  418.   incoming short-form REMOTE command replies redirected to a file), which would
  419.   otherwise result in data written to the file out of order.
  420. */
  421. int
  422. #ifdef CK_ANSIC
  423. zputfil(char c)
  424. #else
  425. zputfil(c) register char c;
  426. #endif /* CK_ANSIC */
  427. /* zputfil */ {
  428.     return(putfil(c));
  429. }
  430. #ifndef NOXFER
  431. /* D E C O D E  --  Kermit packet decoding procedure */
  432. /*
  433.  Call with string to be decoded and an output function.
  434.  Returns 0 on success, -1 on failure (e.g. disk full).
  435.  This is the "inner loop" when receiving files, and must be coded as
  436.  efficiently as possible.  Note some potential problems:  if a packet
  437.  is badly formed, having a prefixed sequence ending prematurely, this
  438.  function, as coded, could read past the end of the packet.  This has
  439.  never happened, thus the additional (time-consuming) tests have not
  440.  been added.
  441. */
  442. static CHAR *xdbuf; /* Global version of decode()'s buffer pointer */
  443.                         /* for use by translation functions. */
  444. /* Function for pushing a character onto decode()'s input stream. */
  445. VOID
  446. #ifdef CK_ANSIC
  447. zdstuff(CHAR c)
  448. #else
  449. zdstuff(c) CHAR c;
  450. #endif /* CK_ANSIC */
  451. /* zdstuff */ {
  452.     xdbuf--; /* Back up the pointer. */
  453.     *xdbuf = c; /* Stuff the character. */
  454. }
  455. #ifdef CKTUNING
  456. /*
  457.   Trimmed-down packet decoder for binary-mode no-parity transfers.
  458.   decode() is the full version.
  459. */
  460. int
  461. #ifdef CK_ANSIC
  462. bdecode(CHAR *buf, int (*fn)(char))
  463. #else
  464. bdecode(buf,fn) register CHAR *buf; register int (*fn)();
  465. #endif /* CK_ANSIC */
  466. /* bdecode */ {
  467.     register unsigned int a, a7; /* Various copies of current char */
  468.     int ccpflg; /* For Ctrl-unprefixing stats */
  469.     int t; /* Int version of character */
  470.     int len;
  471.     long z; /* For CRC calculation */
  472.     CHAR c; /* Current character */
  473.     if (!binary || parity || fn != putfil) /* JUST IN CASE */
  474.       return(decode(buf,fn,1));
  475.     debug(F100,"BDECODE","",0);
  476.     xdbuf = buf; /* Global copy of source pointer. */
  477.     len = rln; /* Number of bytes in data field */
  478.     while (len > 0) {
  479.         a = *xdbuf++ & 0xff; /* Get next character */
  480. len--;
  481. rpt = 0; /* Initialize repeat count. */
  482. if (a == rptq && rptflg) { /* Got a repeat prefix? */
  483.     rpt = xunchar(*xdbuf++ & 0xFF); /* Yes, get the repeat count, */
  484.     rptn += rpt;
  485.     a = *xdbuf++ & 0xFF; /* and get the prefixed character. */
  486.     len -= 2;
  487. }
  488. ccpflg = 0; /* Control prefix flag. */
  489. if (a == ctlq) { /* If control prefix, */
  490.     a  = *xdbuf++ & 0xFF; /* get its operand */
  491.     len--;
  492.     a7 = a & 0x7F; /* and its low 7 bits. */
  493.     if ((a7 >= 0100 && a7 <= 0137) || a7 == '?') { /* Controllify */
  494. a = ctl(a); /* if in control range. */
  495. a7 = a & 0x7F;
  496. ccpflg = 1; /* Note that we did this */
  497. ccp++; /* Count for stats */
  498.     }
  499. } else a7 = a & 0x7f; /* Not control quote */
  500. if (a7 < 32 || a7 == 127) /* A bare control character? */
  501.   if (!ccpflg) ccu++; /* Count it */
  502. if (!rpt) rpt = 1;
  503. for (; rpt > 0; rpt--) { /* Output the char RPT times */
  504. #ifdef CALIBRATE
  505.     if (calibrate) {
  506. ffc++;
  507. continue;
  508.     }
  509. #endif /* CALIBRATE */
  510. #ifdef OS2
  511.             if (xflg && !remfile) { /* Write to virtual screen */
  512.                 char _a;
  513.                 _a = a & fmask;
  514.                 t = conoc(_a);
  515.                 if (t < 1)
  516.                     t = -1;
  517.             } else
  518. #endif /* OS2 */
  519.       t = zmchout(a & fmask); /* zmchout is a macro */
  520.     if (t < 0) {
  521. debug(F101,"bdecode write error - errno","",errno);
  522. return(-1);
  523.     }
  524.     ffc++; /* Count the character */
  525.     if (docrc && !remfile) { /* Update file CRC */
  526. c = a; /* Force conversion to unsigned char */
  527. z = crc16 ^ (long)c;
  528. crc16 = (crc16 >> 8) ^
  529.   (crcta[(z & 0xF0) >> 4] ^ crctb[z & 0x0F]);
  530.     }
  531. }
  532. #ifdef CK_CTRLZ
  533. lastchar = a;
  534. #endif /* CK_CTRLZ */
  535.     }
  536.     return(0);
  537. }
  538. #endif /* CKTUNING */
  539. #endif /* NOXFER */
  540. /*  P N B Y T E  --  Output next byte to file or other destination  */
  541. static long offc = 0L;
  542. static int
  543. #ifdef CK_ANSIC
  544. pnbyte(CHAR c, int (*fn)(char))
  545. #else
  546. pnbyte(c,fn) CHAR c; int (*fn)();
  547. #endif /* CK_ANSIC */
  548. /* pnbyte */ {
  549.     int rc;
  550.     long z;
  551. #ifdef OS2
  552.     if (xflg && !remfile) { /* Write to virtual screen */
  553. char _a;
  554. _a = c & fmask;
  555. rc = conoc(_a);
  556. if (rc < 1)
  557.   return(-1);
  558.     } else {
  559. #endif /* OS2 */
  560. if (fn == putfil) /* Execute output function */
  561.   rc = zmchout(c); /* to-file macro (fast) */
  562. else if (!fn)
  563.   rc = putchar(c); /* to-screen macro (fast) */
  564.         else
  565.   rc = (*fn)(c); /* function call (not as fast) */
  566. if (rc < 0)
  567.   return(rc);
  568. #ifdef OS2
  569.     }
  570. #endif /* OS2 */
  571. /*
  572.   Both xgnbyte() and xpnbyte() increment ffc (the file byte counter).
  573.   During file transfer, only one of these functions is called.  However,
  574.   the TRANSLATE command is likely to call them both.  offc, therefore,
  575.   contains the output byte count, necessary for handling the UCS-2 BOM.
  576. */
  577.     offc++; /* Count the byte */
  578.     ffc++; /* Count the byte */
  579. #ifndef NOXFER
  580.     if (docrc && !xflg && !remfile) { /* Update file CRC */
  581. z = crc16 ^ (long)c;
  582. crc16 = (crc16 >> 8) ^
  583.   (crcta[(z & 0xF0) >> 4] ^ crctb[z & 0x0F]);
  584.     }
  585. #endif /* NOXFER */
  586.     return(1);
  587. }
  588. /*
  589.   X P N B Y T E  --  Translate and put next byte to file.
  590.   Only for Unicode.  Call with next untranslated byte from incoming
  591.   byte stream, which can be in any Transfer Character Set: UCS-2, UTF-8,
  592.   Latin-1, Latin-Hebrew, etc.  Translates to the file character set and
  593.   writes bytes to the output file.  Call with character as int to translate
  594.   as an int, plus the transfer character set (to translate from) and the
  595.   file character set (to translate to), or -1,0,0 to reset the UCS-2 byte
  596.   number (which should be done at the beginning of a file).  Returns:
  597.    -1: On error
  598.     0: Nothing to write (mid-sequence)
  599.    >0: Number of bytes written.
  600. */
  601. #ifdef KANJI
  602. static int jstate = 0, jx = 0; /* For outputting JIS-7 */
  603. static char jbuf[16] = { NUL, NUL };
  604. #endif /* KANJI */
  605. int
  606. #ifdef CK_ANSIC
  607. xpnbyte(int a, int tcs, int fcs, int (*fn)(char))
  608. #else
  609. xpnbyte(a,tcs,fcs,fn) int a, tcs, fcs; int (*fn)();
  610. #endif /* CK_ANSIC */
  611. /* xpnbyte */ {
  612. #ifdef UNICODE
  613.     extern int ucsorder, ucsbom; /* Byte order */
  614. #endif /* UNICODE */
  615.     CHAR c; /* Unsigned char worker */
  616.     static union ck_short uc, eu, sj; /* UCS-2, EUC, and Shift-JIS workers */
  617.     USHORT ch; /* ditto... */
  618.     USHORT * us = NULL; /* ditto... */
  619.     int c7, rc, haveuc = 0; /* Return code and UCS-2 flag */
  620.     int utferror = 0; /* UTF-8 error */
  621.     static int bn = 0; /* UCS-2 byte number */
  622.     int swapping = 0; /* Swapping UCS bytes to output? */
  623. /* swapping must be 0 or 1 */
  624.     if (a == -1 && (tcs | fcs) == 0) { /* Reset in case previous run */
  625. bn = 0; /* left bn at 1... */
  626. offc = 0L;
  627. debug(F101,"xpnbyte RESET","",bn);
  628. return(0);
  629.     }
  630.     debug(F001,"xpnbyte a","",a);
  631. #ifdef UNICODE
  632.     if (ucsorder != 1 && ucsorder != 0) /* Also just in case... */
  633.       ucsorder = 0;
  634.     if ((byteorder && !ucsorder) || (!byteorder && ucsorder))
  635.       swapping = 1; /* Swapping bytes to output */
  636.     if (tcs == TC_UTF8) { /* 'a' is from a UTF-8 stream */
  637. ch = a;
  638. if (fcs == TC_UTF8) /* Output is UTF-8 too */
  639.   return(pnbyte(ch,fn)); /* so just copy. */
  640. rc = utf8_to_ucs2(ch,&us); /* Otherwise convert to UCS-2 */
  641. if (rc == 0) { /* Done with this sequence */
  642.     uc.x_short = *us; /* We have a Unicode */
  643.     haveuc = 1;
  644. } else if (rc < 0) { /* Error */
  645.     debug(F101,"xpnbyte UTF-8 conversion error","",rc);
  646.     haveuc = 1; /* Replace by U+FFFD */
  647.     uc.x_short = *us;
  648.     utferror = 1;
  649. } else /* Sequence incomplete */
  650.   return(0);
  651.     } else if (tcs == TC_UCS2) { /* 'a' is UCS-2 */
  652. /* Here we have incoming UCS-2 in guaranteed Big Endian order */
  653. /* so we must exchange bytes if local machine is Little Endian. */
  654. switch (bn) { /* Which byte? */
  655.   case 0: /* High... */
  656.     uc.x_char[byteorder] = (unsigned)a & 0xff;
  657.     bn++;
  658.     return(0); /* Wait for next */
  659.   case 1: /* Low... */
  660.     uc.x_char[1-byteorder] = (unsigned)a & 0xff;
  661.     bn = 0; /* Done with sequence */
  662.     haveuc = 1; /* Have a Unicode */
  663. }
  664.     } else
  665. #endif /* UNICODE */
  666. #ifdef KANJI /* Whether UNICODE is defined or not */
  667.       if (tcs == TC_JEUC) { /* Incoming Japanese EUC */
  668. int bad = 0;
  669. static int kanji = 0; /* Flags set in case 0 for case 1 */
  670. static int kana = 0;
  671. switch (bn) { /* Byte number */
  672.   case 0: /* Byte 0 */
  673.     eu.x_short = 0;
  674.     if ((a & 0x80) == 0) {
  675. sj.x_short = (unsigned)a & 0xff; /* Single byte */
  676. kanji = kana = 0;
  677.     } else { /* Double byte */
  678. c7 = a & 0x7f;
  679. if (c7 > 0x20 && c7 < 0x7f) { /* Kanji */
  680.     eu.x_char[byteorder] = (CHAR) a;  /* Store first byte */
  681.     bn++;       /* Set up for second byte */
  682.     kanji = 1;
  683.     kana = 0;
  684.     return(0);
  685. } else if (a == 0x8e) { /* SS2 -- Katakana prefix */
  686.     eu.x_char[byteorder] = (CHAR) a; /* Save it */
  687.     bn++;
  688.     kana = 1;
  689.     kanji = 0;
  690.     return(0);
  691. } else {
  692.     bad++;
  693. }
  694.     }
  695.     break;
  696.   case 1: /* Byte 1 */
  697.     bn = 0;
  698.     if (kanji) {
  699. eu.x_char[1-byteorder] = (CHAR) a;
  700. sj.x_short = eu_to_sj(eu.x_short);
  701. break;
  702.     } else if (kana) {
  703. sj.x_short = (CHAR) (a | 0x80);
  704. break;
  705.     } else { /* (shouldn't happen) */
  706. bad++;
  707.     }
  708. }
  709. /* Come here with one Shift-JIS character */
  710. #ifdef UNICODE
  711. if (bad) {
  712.     uc.x_short = 0xfffd;
  713. } else {
  714.     uc.x_short = sj_to_un(sj.x_short); /* Convert to Unicode */
  715. }
  716. haveuc = 1;
  717. #endif /* UNICODE */
  718.     } else
  719. #endif /* KANJI */
  720. #ifdef UNICODE
  721. uc.x_short = (unsigned)a & 0xff; /* Latin-1 or whatever... */
  722.     /* Come here with uc = the character to be translated. */
  723.     /* If (haveuc) it's UCS-2 in native order, otherwise it's a byte. */
  724.     debug(F101,"xpnbyte haveuc","",haveuc);
  725.     if (haveuc) { /* If we have a Unicode... */
  726. debug(F101,"xpnbyte uc.x_short A","",uc.x_short);
  727. debug(F101,"xpnbyte feol","",feol);
  728. if (feol && uc.x_short == CR) {
  729.     return(0);
  730. } else if (feol && uc.x_short == LF) {
  731.     uc.x_short = feol;
  732. }
  733. debug(F101,"xpnbyte uc.x_short B","",uc.x_short);
  734. if (fcs == FC_UCS2) { /* And FCS is UCS-2 */
  735.     /* Write out the bytes in the appropriate byte order */
  736.     int count = 0;
  737.     if (offc == 0L && ucsbom) { /* Beginning of file? */
  738. if ((rc = pnbyte((ucsorder ? 0xff : 0xfe),fn)) < 0) /* BOM */
  739.   return(rc);
  740. if ((rc = pnbyte((ucsorder ? 0xfe : 0xff),fn)) < 0)
  741.   return(rc);
  742. count += 2;
  743.     }
  744.     if (utferror) {
  745. if ((rc = pnbyte((ucsorder ? 0xfd : 0xff),fn)) < 0)
  746.   return(rc);
  747. if ((rc = pnbyte((ucsorder ? 0xff : 0xfd),fn)) < 0)
  748.   return(rc);
  749. count += 2;
  750.     }
  751.     if ((rc = pnbyte(uc.x_char[swapping],fn)) < 0)
  752.       return(rc);
  753.     if ((rc = pnbyte(uc.x_char[1-swapping],fn)) < 0)
  754.       return(rc);
  755.     count += 2;
  756.     return(count);
  757. } else if (fcs == FC_UTF8) { /* Convert to UTF-8 */
  758.     CHAR * buf = NULL;
  759.     int i, count;
  760.     if (utferror) {
  761. if ((rc = pnbyte((ucsorder ? 0xbd : 0xff),fn)) < 0)
  762.   return(rc);
  763. if ((rc = pnbyte((ucsorder ? 0xff : 0xbd),fn)) < 0)
  764.   return(rc);
  765.     }
  766.     if ((count = ucs2_to_utf8(uc.x_short,&buf)) < 1)
  767.       return(-1);
  768.     debug(F011,"xpnbyte buf",buf,count);
  769.     for (i = 0; i < count; i++)
  770.       if ((rc = pnbyte(buf[i],fn)) < 0)
  771. return(rc);
  772.     if (utferror)
  773.       count += 2;
  774.     return(count);
  775. } else { /* Translate UCS-2 to byte */
  776.     if (uc.x_short == 0x2028 || uc.x_short == 0x2029) {
  777. if (utferror)
  778.   pnbyte(UNK,fn);
  779. if (feol)
  780.   return(pnbyte((CHAR)feol,fn));
  781. if ((rc = pnbyte((CHAR)CR,fn)) < 0)
  782.   return(rc);
  783. if ((rc = pnbyte((CHAR)LF,fn)) < 0)
  784.   return(rc);
  785. else
  786.   return(utferror ? 3 : 2);
  787.     } else if (xuf) { /* UCS-to-FCS function */
  788. int x = 0;
  789. if (utferror)
  790.   pnbyte(UNK,fn);
  791. if ((rc = (*xuf)(uc.x_short)) < 0) /* These can fail... */
  792.   ch = UNK;
  793. else
  794.   ch = (unsigned)((unsigned)rc & 0xffff);
  795. x = pnbyte(ch,fn);
  796. if (x < 0)
  797.   return(x);
  798. else if (utferror)
  799.   x++;
  800. return(x);
  801. #ifdef KANJI
  802. /*  Also see the non-Unicode Kanji section further down in this function. */
  803.     } else if (fcsinfo[fcs].alphabet == AL_JAPAN) {
  804. /* Translate UCS-2 to Japanese set */
  805. debug(F001,"xpnbyte uc","",uc.x_short);
  806. sj.x_short = un_to_sj(uc.x_short); /* First to Shift-JIS */
  807. debug(F001,"xpnbyte sj","",sj.x_short);
  808. switch (fcs) { /* File character set */
  809.   case FC_SHJIS: /* Shift-JIS -- just output it */
  810.     if (sj.x_char[byteorder]) /* But not high byte if zero */
  811.       if ((rc = pnbyte((CHAR)sj.x_char[byteorder],fn)) < 0)
  812. return(rc);
  813.     if ((rc = pnbyte((CHAR)sj.x_char[1-byteorder],fn)) < 0)
  814.       return(rc);
  815.     return(2);
  816.   case FC_JEUC: /* EUC-JP */
  817.     eu.x_short = sj_to_eu(sj.x_short); /* Shift-JIS to EUC */
  818.     debug(F001,"xpnbyte eu","",eu.x_short);
  819.     if (eu.x_short == 0xffff) { /* Bad */
  820. if ((rc = pnbyte(UNK,fn)) < 0)
  821.   return(rc);
  822. return(1);
  823.     } else { /* Good */
  824. int count = 0; /* Write high byte if not zero */
  825. if (eu.x_char[byteorder]) {
  826.     if ((rc=pnbyte((CHAR)eu.x_char[byteorder],fn)) < 0)
  827.       return(rc);
  828.     count++;
  829. }
  830. /* Always write low byte */
  831. if ((rc = pnbyte((CHAR)eu.x_char[1-byteorder],fn)) < 0)
  832.   return(rc);
  833. count++;
  834. return(count);
  835.     }
  836.     break;
  837.   case FC_JIS7: /* JIS-7 */
  838.   case FC_JDEC: /* DEC Kanji */
  839.     eu.x_short = sj_to_eu(sj.x_short); /* Shift-JIS to EUC */
  840.     if (eu.x_short == 0xffff) { /* Bad */
  841. debug(F001,"xpnbyte bad eu","",eu.x_short);
  842. if ((rc = pnbyte(UNK,fn)) < 0)
  843.   return(rc);
  844. return(1);
  845.     } else { /* Good */
  846. int i;
  847. /* ^^^ Use another name - 'a' hides parameter */
  848. /* It's OK as is but causes compiler warnings */
  849. char a = eu.x_char[1-byteorder]; /* Low byte */
  850. debug(F001,"xpnbyte eu","",eu.x_short);
  851. if (eu.x_char[byteorder] == 0) { /* Roman */
  852.     switch (jstate) {
  853.       case 1: /* Current state is Katakana */
  854. jbuf[0] = 0x0f; /* SI */
  855. jbuf[1] = a;
  856. jx = 2;
  857. break;
  858.       case 2: /* Current state is Kanji */
  859. jbuf[0] = 0x1b; /* ESC */
  860. jbuf[1] = 0x28; /* ( */
  861. jbuf[2] = 0x4a; /* J */
  862. jbuf[3] = a;
  863. jx = 4;
  864. break;
  865.       default: /* Current state is Roman */
  866. jbuf[0] = a;
  867. jx = 1;
  868. break;
  869.     }
  870.     jstate = 0; /* New state is Roman */
  871. } else if (eu.x_char[byteorder] == 0x8e) { /* Kana */
  872.     jx = 0;
  873.     switch (jstate) {
  874.       case 2:    /* from Kanji */
  875. jbuf[jx++] = 0x1b; /* ESC */
  876. jbuf[jx++] = 0x28; /* ( */
  877. jbuf[jx++] = 0x4a; /* J */
  878.       case 0:    /* from Roman */
  879. jbuf[jx++] = 0x0e; /* SO */
  880.       default:    /* State is already Kana*/
  881. jbuf[jx++] = (a & 0x7f); /* and the char */
  882. break;
  883.     }
  884.     jstate = 1; /* New state is Katakana */
  885. } else { /* Kanji */
  886.     jx = 0;
  887.     switch (jstate) {
  888.       case 1: /* Current state is Katakana */
  889. jbuf[jx++] = 0x0f; /* SI  */
  890.       case 0: /* Current state is Roman */
  891. jbuf[jx++] = 0x1b; /* ESC */
  892. jbuf[jx++] = 0x24; /* $   */
  893. jbuf[jx++] = 0x42; /* B   */
  894.       default: /* Current state is already Kanji */
  895. jbuf[jx++] = eu.x_char[byteorder] & 0x7f;
  896. jbuf[jx++] = eu.x_char[1-byteorder] & 0x7f;
  897. break;
  898.     }
  899.     jstate = 2; /* Set state to Kanji */
  900. }
  901. for (i = 0; i < jx; i++) /* Output the result */
  902.   if ((rc = pnbyte(jbuf[i],fn)) < 0)
  903.     return(rc);
  904. return(jx); /* Return its length */
  905.     }
  906. }
  907. #endif /* KANJI */
  908.     } else { /* No translation function */
  909. int count = 0;
  910. if (utferror) {
  911.     if ((rc = pnbyte((ucsorder ? 0xfd : 0xff),fn)) < 0)
  912.       return(rc);
  913.     if ((rc = pnbyte((ucsorder ? 0xff : 0xfd),fn)) < 0)
  914.       return(rc);
  915.     count += 2;
  916. }
  917. if ((rc = pnbyte(uc.x_char[swapping],fn)) < 0)
  918.   return(rc);
  919. if ((rc = pnbyte(uc.x_char[1-swapping],fn)) < 0)
  920.   return(rc);
  921. count += 2;
  922. return(count);
  923.     }
  924. }
  925.     } else { /* Byte to Unicode */
  926. if (xtu) { /* TCS-to-UCS function */
  927.     if (((tcsinfo[tcs].size > 128) && (uc.x_short & 0x80)) ||
  928. tcsinfo[tcs].size <= 128)
  929.       uc.x_short = (*xtu)(uc.x_short);
  930. }
  931. if (fcs == FC_UCS2) { /* And FCS is UCS-2 */
  932.     /* Write out the bytes in the appropriate byte order */
  933.     if (offc == 0 && ucsbom) { /* Beginning of file? */
  934. if ((rc = pnbyte((ucsorder ? 0xff : 0xfe),fn)) < 0) /* BOM */
  935.   return(rc);
  936. if ((rc = pnbyte((ucsorder ? 0xfe : 0xff),fn)) < 0)
  937.   return(rc);
  938.     }
  939.     if ((rc = pnbyte(uc.x_char[swapping],fn)) < 0)
  940.       return(rc);
  941.     if ((rc = pnbyte(uc.x_char[1-swapping],fn)) < 0)
  942.       return(rc);
  943.     return(2);
  944. } else if (fcs == FC_UTF8) { /* Convert to UTF-8 */
  945.     CHAR * buf = NULL;
  946.     int i, count;
  947.     if ((count = ucs2_to_utf8(uc.x_short,&buf)) < 1)
  948.       return(-1);
  949.     for (i = 0; i < count; i++)
  950.       if ((rc = pnbyte(buf[i],fn)) < 0)
  951. return(rc);
  952.     return(count);
  953. } else {
  954.     debug(F100,"xpnbyte impossible combination","",0);
  955.     return(-1);
  956. }
  957.     }
  958. #else
  959. #ifdef KANJI
  960. /*
  961.   This almost, but not quite, duplicates the Kanji section above.
  962.   There is no doubt a way to combine the sections more elegantly,
  963.   but probably only at the expense of additional execution overhead.
  964.   As matters stand, be careful to reflect any changes in this section
  965.   to the other Kanji section above.
  966. */
  967.     if (tcs == TC_JEUC) { /* Incoming Japanese EUC */
  968. int count = 0;
  969. switch (fcs) { /* File character set */
  970.   case FC_SHJIS: /* Shift-JIS -- just output it */
  971.     if (sj.x_char[byteorder]) /* But not high byte if zero */
  972.       if ((rc = pnbyte((CHAR)sj.x_char[byteorder],fn)) < 0)
  973. return(rc);
  974.     count++;
  975.     if ((rc = pnbyte((CHAR)sj.x_char[1-byteorder],fn)) < 0)
  976.       return(rc);
  977.     count++;
  978.     return(count);
  979.   case FC_JEUC: /* EUC-JP */
  980.     eu.x_short = sj_to_eu(sj.x_short); /* Shift-JIS to EUC */
  981.     debug(F001,"xpnbyte FC_JEUC eu","",eu.x_short);
  982.     if (eu.x_short == 0xffff) { /* Bad */
  983. if ((rc = pnbyte(UNK,fn)) < 0)
  984.   return(rc);
  985. return(1);
  986.     } else { /* Good */
  987. int count = 0; /* Write high byte if not zero */
  988. if (eu.x_char[byteorder]) {
  989.     if ((rc = pnbyte((CHAR)eu.x_char[byteorder],fn)) < 0)
  990.       return(rc);
  991.     count++;
  992. }
  993. /* Always write low byte */
  994. if ((rc = pnbyte((CHAR)eu.x_char[1-byteorder],fn)) < 0)
  995.   return(rc);
  996. count++;
  997. return(count);
  998.     }
  999.     break;
  1000.   case FC_JIS7: /* JIS-7 */
  1001.   case FC_JDEC: /* DEC Kanji */
  1002.     eu.x_short = sj_to_eu(sj.x_short); /* Shift-JIS to EUC */
  1003.     if (eu.x_short == 0xffff) { /* Bad */
  1004. debug(F001,"xpnbyte FC_JIS7 bad eu","",eu.x_short);
  1005. if ((rc = pnbyte(UNK,fn)) < 0)
  1006.   return(rc);
  1007. return(1);
  1008.     } else { /* Good */
  1009. int i;
  1010. char a = eu.x_char[1-byteorder]; /* Low byte */
  1011. debug(F001,"xpnbyte FC_JIS7 eu","",eu.x_short);
  1012. if (eu.x_char[byteorder] == 0) { /* Roman */
  1013.     switch (jstate) {
  1014.       case 1: /* Current state is Katakana */
  1015. jbuf[0] = 0x0f; /* SI */
  1016. jbuf[1] = a;
  1017. jx = 2;
  1018. break;
  1019.       case 2: /* Current state is Kanji */
  1020. jbuf[0] = 0x1b; /* ESC */
  1021. jbuf[1] = 0x28; /* ( */
  1022. jbuf[2] = 0x4a; /* J */
  1023. jbuf[3] = a;
  1024. jx = 4;
  1025. break;
  1026.       default: /* Current state is Roman */
  1027. jbuf[0] = a;
  1028. jx = 1;
  1029. break;
  1030.     }
  1031.     jstate = 0; /* New state is Roman */
  1032. } else if (eu.x_char[byteorder] == 0x8e) { /* Kana */
  1033.     jx = 0;
  1034.     switch (jstate) {
  1035.       case 2:    /* from Kanji */
  1036. jbuf[jx++] = 0x1b; /* ESC */
  1037. jbuf[jx++] = 0x28; /* ( */
  1038. jbuf[jx++] = 0x4a; /* J */
  1039.       case 0:    /* from Roman */
  1040. jbuf[jx++] = 0x0e; /* SO */
  1041.       default:    /* State is already Kana*/
  1042. jbuf[jx++] = (a & 0x7f); /* and the char */
  1043. break;
  1044.     }
  1045.     jstate = 1; /* New state is Katakana */
  1046. } else { /* Kanji */
  1047.     jx = 0;
  1048.     switch (jstate) {
  1049.       case 1: /* Current state is Katakana */
  1050. jbuf[jx++] = 0x0f; /* SI  */
  1051.       case 0: /* Current state is Roman */
  1052. jbuf[jx++] = 0x1b; /* ESC */
  1053. jbuf[jx++] = 0x24; /* $   */
  1054. jbuf[jx++] = 0x42; /* B   */
  1055.       default: /* Current state is already Kanji */
  1056. jbuf[jx++] = eu.x_char[byteorder] & 0x7f;
  1057. jbuf[jx++] = eu.x_char[1-byteorder] & 0x7f;
  1058. break;
  1059.     }
  1060.     jstate = 2; /* Set state to Kanji */
  1061. }
  1062. for (i = 0; i < jx; i++) /* Output the result */
  1063.   if ((rc = pnbyte(jbuf[i],fn)) < 0)
  1064.     return(rc);
  1065. return(jx); /* Return its length */
  1066.     }
  1067.   default:
  1068.     if (sj.x_short < 0x80)
  1069.       return(sj.x_short);
  1070.     else
  1071.       return('?');
  1072. }
  1073.     }
  1074. #endif /* KANJI */
  1075. #endif /* UNICODE */
  1076.     debug(F100,"xpnbyte BAD FALLTHRU","",0);
  1077.     return(-1);
  1078. }
  1079. #ifndef NOXFER
  1080. /*  D E C O D E  --  Kermit Data-packet decoder  */
  1081. int
  1082. #ifdef CK_ANSIC
  1083. decode(CHAR *buf, int (*fn)(char), int xlate)
  1084. #else
  1085. decode(buf,fn,xlate) register CHAR *buf; register int (*fn)(); int xlate;
  1086. #endif /* CK_ANSIC */
  1087. /* decode */ {
  1088.     register unsigned int a, a7, a8, b8; /* Various copies of current char */
  1089.     int t; /* Int version of character */
  1090.     int ssflg; /* Character was single-shifted */
  1091.     int ccpflg; /* For Ctrl-unprefixing stats */
  1092.     int len;
  1093.     long z;
  1094.     CHAR c;
  1095. /*
  1096.   Catch the case in which we are asked to decode into a file that is not open,
  1097.   for example, if the user interrupted the transfer, but the other Kermit
  1098.   keeps sending.
  1099. */
  1100.     if ((cxseen || czseen || discard) && (fn == putfil))
  1101.       return(0);
  1102. #ifdef COMMENT
  1103. #ifdef CKTUNING
  1104.     if (binary && !parity)
  1105.       return(bdecode(buf,fn));
  1106. #endif /* CKTUNING */
  1107. #endif /* COMMENT */
  1108.     debug(F100,"DECODE","",0);
  1109.     xdbuf = buf; /* Make global copy of pointer. */
  1110.     rpt = 0; /* Initialize repeat count. */
  1111.     len = rln; /* Number of bytes in data field */
  1112.     while (len > 0) { /* Loop for each byte */
  1113.         a = *xdbuf++ & 0xff; /* Get next character */
  1114. len--;
  1115. if (a == rptq && rptflg) { /* Got a repeat prefix? */
  1116.     rpt = xunchar(*xdbuf++ & 0xFF); /* Yes, get the repeat count, */
  1117.     rptn += rpt;
  1118.     a = *xdbuf++ & 0xFF; /* and get the prefixed character. */
  1119.     len -= 2;
  1120. }
  1121. b8 = lsstate ? 0200 : 0; /* 8th-bit value from SHIFT-STATE */
  1122. if (ebqflg && a == ebq) { /* Have 8th-bit prefix? */
  1123.     b8 ^= 0200; /* Yes, invert the 8th bit's value, */
  1124.     ssflg = 1; /* remember we did this, */
  1125.     a = *xdbuf++ & 0xFF; /* and get the prefixed character. */
  1126.     len--;
  1127. } else ssflg = 0;
  1128. ccpflg = 0;
  1129. if (a == ctlq) { /* If control prefix, */
  1130.     a  = *xdbuf++ & 0xFF; /* get its operand */
  1131.     len--;
  1132.     a7 = a & 0x7F; /* and its low 7 bits. */
  1133.     if ((a7 >= 0100 && a7 <= 0137) || a7 == '?') { /* Controllify */
  1134. a = ctl(a); /* if in control range. */
  1135. a7 = a & 0x7F;
  1136. ccpflg = 1; /* Note that we did this */
  1137. ccp++; /* Count for stats */
  1138.     }
  1139. } else a7 = a & 0x7f; /* Not control quote */
  1140. if (a7 < 32 || a7 == 127) { /* Control character? */
  1141.     if (!ccpflg) ccu++; /* A bare one, count it */
  1142.     if (lscapu) { /* If doing locking shifts... */
  1143. if (lsstate) /* If SHIFTED */
  1144.   a8 = (a & ~b8) & 0xFF; /* Invert meaning of 8th bit */
  1145. else /* otherwise */
  1146.   a8 = a | b8; /* OR in 8th bit */
  1147. /* If we're not in a quoted sequence */
  1148. if (!lsquote && (!lsstate || !ssflg)) {
  1149.     if (a8 == DLE) { /* Check for DLE quote */
  1150. lsquote = 1; /* prefixed by single shift! */
  1151. continue;
  1152.     } else if (a8 == SO) { /* Check for Shift-Out */
  1153. lsstate = 1; /* SHIFT-STATE = SHIFTED */
  1154. continue;
  1155.     } else if (a8 == SI) { /* or Shift-In */
  1156. lsstate = 0; /* SHIFT-STATE = UNSHIFTED */
  1157. continue;
  1158.     }
  1159. } else lsquote = 0;
  1160.     }
  1161. }
  1162. a |= b8; /* OR in the 8th bit */
  1163. if (rpt == 0) rpt = 1; /* If no repeats, then one */
  1164. #ifndef NOCSETS
  1165. if (!binary) { /* If in text mode, */
  1166.     if (tcharset != TC_UCS2) {
  1167. if (feol && a == CR) /* Convert CRLF to newline char */
  1168.   continue;
  1169. if (feol && a == LF)
  1170.   a = feol;
  1171.     }
  1172.     if (xlatype == XLA_BYTE) /* Byte-for-byte - do it now */
  1173.       if (xlate && rx) a = (*rx)((CHAR) a);
  1174.      }
  1175. #endif /* NOCSETS */
  1176. /* (PWP) Decoding speedup via buffered output and a macro... */
  1177. if (fn == putfil) {
  1178.     for (; rpt > 0; rpt--) { /* Output the char RPT times */
  1179. #ifdef CALIBRATE
  1180. if (calibrate) {
  1181.     ffc++;
  1182.     continue;
  1183. }
  1184. #endif /* CALIBRATE */
  1185. /* Note: The Unicode and Kanji sections can probably be combined now; */
  1186. /* the Unicode method (xpnbyte()) covers Kanji too. */
  1187. #ifdef UNICODE
  1188. if (!binary && xlatype == XLA_UNICODE)
  1189.   t = xpnbyte((unsigned)((unsigned)a & 0xff),
  1190.       tcharset,
  1191.       fcharset,
  1192.       fn
  1193.       );
  1194. else
  1195. #endif /* UNICODE */
  1196. #ifdef KANJI
  1197. if (!binary && tcharset == TC_JEUC &&
  1198.     fcharset != FC_JEUC) { /* Translating from J-EUC */
  1199.     if (ffc == 0L) xkanjf();
  1200.     if (xkanji(a,fn) < 0)  /* to something else? */
  1201.       return(-1);
  1202.     else t = 1;
  1203. } else
  1204. #endif /* KANJI */
  1205. {
  1206. #ifdef OS2
  1207.       if (xflg && !remfile) { /* Write to virtual screen */
  1208.   char _a;
  1209.   _a = a & fmask;
  1210.   t = conoc(_a);
  1211.   if (t < 1)
  1212.     t = -1;
  1213.       } else
  1214. #endif /* OS2 */
  1215. t = zmchout(a & fmask); /* zmchout is a macro */
  1216. }
  1217. if (t < 0) {
  1218.     debug(F101,"decode write errno","",errno);
  1219.     return(-1);
  1220. }
  1221. #ifdef UNICODE
  1222. if (xlatype != XLA_UNICODE || binary) {
  1223.     ffc++; /* Count the character */
  1224.     if (docrc && !xflg && !remfile) { /* Update file CRC */
  1225. c = a; /* Force conversion to unsigned char */
  1226. z = crc16 ^ (long)c;
  1227. crc16 = (crc16 >> 8) ^
  1228.   (crcta[(z & 0xF0) >> 4] ^ crctb[z & 0x0F]);
  1229.     }
  1230. }
  1231. #endif /* UNICODE */
  1232.     }
  1233. } else { /* Output to something else. */
  1234.     a &= fmask; /* Apply file mask */
  1235.     for (; rpt > 0; rpt--) { /* Output the char RPT times */
  1236. #ifdef CALIBRATE
  1237. if (calibrate) {
  1238.     ffc++;
  1239.     continue;
  1240. }
  1241. #endif /* CALIBRATE */
  1242. if ((*fn)((char) a) < 0) return(-1); /* Send to output func. */
  1243.     }
  1244. }
  1245. #ifdef CK_CTRLZ
  1246. lastchar = a;
  1247. #endif /* CK_CTRLZ */
  1248.     }
  1249.     return(0);
  1250. }
  1251. /*  G E T P K T -- Fill a packet data field  */
  1252. /*
  1253.   Gets characters from the current source -- file or memory string.
  1254.   Encodes the data into the packet, filling the packet optimally.
  1255.   Set first = 1 when calling for the first time on a given input stream
  1256.   (string or file).
  1257.   Call with:
  1258.     bufmax -- current send-packet size
  1259.     xlate  -- flag: 0 to skip character-set translation, 1 to translate
  1260.   Uses global variables:
  1261.     t     -- current character.
  1262.     first -- flag: 1 to start up, 0 for input in progress, -1 for EOF.
  1263.     next  -- next character (not used any more).
  1264.     data  -- pointer to the packet data buffer.
  1265.     size  -- number of characters in the data buffer.
  1266.     memstr - flag that input is coming from a memory string instead of a file.
  1267.     memptr - pointer to string in memory.
  1268.     (*sx)()  character set translation function
  1269.   Returns:
  1270.     The size as value of the function, and also sets global "size",
  1271.     and fills (and null-terminates) the global data array.
  1272.     Returns:
  1273.       0 on EOF.
  1274.      -1 on fatal (internal) error.
  1275.      -3 on timeout (e.g. when reading data from a pipe).
  1276.   Rewritten by Paul W. Placeway (PWP) of Ohio State University, March 1989.
  1277.   Incorporates old getchx() and encode() inline to reduce function calls,
  1278.   uses buffered input for much-improved efficiency, and clears up some
  1279.   confusion with line termination (CRLF vs LF vs CR).
  1280.   Rewritten again by Frank da Cruz to incorporate locking shift mechanism,
  1281.   May 1991.  And again in 1998 for efficiency, etc, with a separate
  1282.   bgetpkt() split out for binary-mode no-parity transfers.
  1283. */
  1284. /*
  1285.   Note: Separate Kanji support dates from circa 1991 and now (1999) can most
  1286.   likely be combined with the the Unicode support: the xgnbyte()/xpnbyte()
  1287.   mechanism works for both Unicode and Kanji.
  1288. */
  1289. #ifdef KANJI
  1290. int
  1291. kgetf(
  1292. #ifdef CK_ANSIC
  1293.       VOID
  1294. #endif /* CK_ANSIC */
  1295.       ) {
  1296.     if (funcstr)
  1297.       return((*funcptr)());
  1298.     else
  1299.       return(zminchar());
  1300. }
  1301. int
  1302. kgetm(
  1303. #ifdef CK_ANSIC
  1304.       VOID
  1305. #endif /* CK_ANSIC */
  1306.       ) {
  1307.     int x;
  1308.     if (x = *memptr++) return(x);
  1309.     else return(-1);
  1310. }
  1311. #endif /* KANJI */
  1312. /*
  1313.   Lookahead function to decide whether locking shift is worth it.  Looks at
  1314.   the next four input characters to see if all of their 8th bits match the
  1315.   argument.  Call with 0 or 0200.  Returns 1 on match, 0 if they don't match.
  1316.   If we don't happen to have at least 4 more characters waiting in the input
  1317.   buffer, returns 1.  Note that zinptr points two characters ahead of the
  1318.   current character because of repeat-count lookahead.
  1319. */
  1320. int
  1321. lslook(b) unsigned int b; { /* Locking Shift Lookahead */
  1322.     int i;
  1323.     if (zincnt < 3) /* If not enough chars in buffer, */
  1324.       return(1); /* force shift-state switch. */
  1325.     b &= 0200; /* Force argument to proper form. */
  1326.     for (i = -1; i < 3; i++) /* Look at next 5 characters to */
  1327.       if (((*(zinptr+i)) & 0200) != b) /* see if all their 8th bits match.  */
  1328. return(0); /* They don't. */
  1329.     return(1); /* They do. */
  1330. }
  1331. /* Routine to compute maximum data length for packet to be filled */
  1332. int
  1333. maxdata() { /* Get maximum data length */
  1334.     int n, len;
  1335.     debug(F101,"maxdata spsiz 1","",spsiz);
  1336.     if (spsiz < 0) /* How could this happen? */
  1337.       spsiz = DSPSIZ;
  1338.     debug(F101,"maxdata spsiz 2","",spsiz);
  1339.     n = spsiz - 5; /* Space for Data and Checksum */
  1340.     if (n > 92 && n < 96) n = 92; /* "Short" Long packets don't pay */
  1341.     if (n > 92 && lpcapu == 0) /* If long packets needed, */
  1342.       n = 92; /* make sure they've been negotiated */
  1343.     len = n - bctl; /* Space for data */
  1344.     if (n > 92) len -= 3; /* Long packet needs header chksum */
  1345.     debug(F101,"maxdata len 1","",len);
  1346.     if (len < 0) len = 10;
  1347.     debug(F101,"maxdata len 2","",len);
  1348.     return(len);
  1349. }
  1350. static CHAR leftover[9] = { '','','','','','','','','' };
  1351. static int nleft = 0;
  1352. #ifdef CKTUNING
  1353. /*
  1354.   When CKTUNING is defined we use this special trimmed-down version of getpkt
  1355.   to speed up binary-mode no-parity transfers.  When CKTUNING is not defined,
  1356.   or for text-mode or parity transfers, we use the regular getpkt() function.
  1357.   Call just like getpkt() but test first for transfer mode and parity.  NOTE:
  1358.   This routine is only to be called when sending a real file -- not for
  1359.   filenames, server responses, etc, because it only reads from the input file.
  1360.   See getpkt() for more detailed commentary.
  1361. */
  1362. static int
  1363. bgetpkt(bufmax) int bufmax; {
  1364.     register CHAR rt = t, rnext;
  1365.     register CHAR *dp, *odp, *p1, *p2;
  1366.     register int x = 0, a7;
  1367.     CHAR xxrc, xxcq; /* Pieces of prefixed sequence */
  1368.     long z; /* A long worker (for CRC) */
  1369.     if (!binary || parity || memstr) /* JUST IN CASE caller didn't test */
  1370.       return(getpkt(bufmax,!binary));
  1371.     if (!data) {
  1372. debug(F100,"SERIOUS ERROR: bgetpkt data == NULL","",0);
  1373. return(-1);
  1374.     }
  1375.     dp = data; /* Point to packet data buffer */
  1376.     size = 0; /* And initialize its size */
  1377.     bufmax = maxdata(); /* Get maximum data length */
  1378. #ifdef DEBUG
  1379.     if (deblog)
  1380.       debug(F101,"bgetpkt bufmax","",bufmax);
  1381. #endif /* DEBUG */
  1382.     if (first == 1) { /* If first character of this file.. */
  1383. ffc = 0L; /* reset file character counter */
  1384. #ifdef COMMENT
  1385. /* Moved to below */
  1386. first = 0; /* Next character won't be first */
  1387. #endif /* COMMENT */
  1388. *leftover = ''; /* Discard any interrupted leftovers */
  1389. nleft = 0;
  1390. /* Get first character of file into rt, watching out for null file */
  1391. #ifdef CALIBRATE
  1392. if (calibrate) {
  1393. #ifdef NORANDOM
  1394.     rt = 17;
  1395. #else
  1396.     rt = cal_a[rand() & 0xff];
  1397. #endif /* NORANDOM */
  1398.     first = 0;
  1399. } else
  1400. #endif /* CALIBRATE */
  1401. if ((x = zminchar()) < 0) { /* EOF or error */
  1402.     if (x == -3) { /* Timeout. */
  1403. size = (dp - data);
  1404. debug(F101,"bgetpkt timeout size","",size);
  1405. return((size == 0) ? x : size);
  1406.     }
  1407.     first = -1;
  1408.     size = 0;
  1409.     if (x == -2) { /* Error */
  1410. debug(F100,"bgetpkt: input error","",0);
  1411. cxseen = 1; /* Interrupt the file transfer */
  1412.     } else {
  1413. debug(F100,"bgetpkt empty file","",0);
  1414.     }
  1415.     return(0);
  1416. }
  1417. first = 0; /* Next char will not be the first */
  1418. ffc++; /* Count a file character */
  1419. rt = (CHAR) x; /* Convert int to char */
  1420. if (docrc && what == W_SEND) { /* Accumulate file crc */
  1421.     z = crc16 ^ (long)rt;
  1422.     crc16 = (crc16 >> 8) ^
  1423.       (crcta[(z & 0xF0) >> 4] ^ crctb[z & 0x0F]);
  1424. }
  1425. rt &= fmask; /* Apply SET FILE BYTESIZE mask */
  1426.     } else if (first == -1 && nleft == 0) { /* EOF from last time */
  1427.         return(size = 0);
  1428.     }
  1429. /*
  1430.   Here we handle characters that were encoded for the last packet but
  1431.   did not fit, and so were saved in the "leftover" array.
  1432. */
  1433.     if (nleft) {
  1434. for (p1 = leftover; nleft > 0; nleft--) /* Copy leftovers */
  1435.   *dp++ = *p1++;
  1436. *leftover = ''; /* Delete leftovers */
  1437. nleft = 0;
  1438.     }
  1439.     if (first == -1) /* Handle EOF */
  1440.       return(size = (dp - data));
  1441. /* Now fill up the rest of the packet. */
  1442.     rpt = 0; /* Initialize character repeat count */
  1443.     while (first > -1) { /* Until EOF... */
  1444. #ifdef CALIBRATE
  1445. if (calibrate) { /* We generate our own "file" */
  1446.     if (ffc >= calibrate) { /* EOF */
  1447. first = -1;
  1448. ffc--;
  1449.     } else { /* Generate next character */
  1450. if (cal_j > CAL_M * ffc)
  1451.   cal_j = cal_a[ffc & 0xff];
  1452. x = (unsigned)cal_a[(cal_j & 0xff)];
  1453. if (x == rt) x ^= 2;
  1454.     }
  1455.     ffc++;
  1456.     cal_j += (unsigned int)(ffc + CAL_O);
  1457. } else
  1458. #endif /* CALIBRATE */
  1459. if ((x = zminchar()) < 0) { /* Check for EOF */
  1460.     if (x == -3) { /* Timeout. */
  1461. t = rt;
  1462. size = (dp-data);
  1463. debug(F101,"bgetpkt timeout size","",size);
  1464. return((size == 0) ? x : size);
  1465.     }
  1466.     first = -1; /* Flag eof for next time. */
  1467.     if (x == -2) cxseen = 1; /* If error, cancel this file. */
  1468. } else {
  1469.     ffc++; /* Count the character */
  1470.     if (docrc && what == W_SEND) { /* Accumulate file crc */
  1471. z = crc16 ^ (long)((CHAR)x & 0xff);
  1472. crc16 = (crc16 >> 8) ^
  1473.   (crcta[(z & 0xF0) >> 4] ^ crctb[z & 0x0F]);
  1474.     }
  1475. }
  1476. rnext = (CHAR) (x & fmask); /* Apply file mask */
  1477. /*
  1478.   At this point, the character we just read is in rnext,
  1479.   and the character we are about to encode into the packet is in rt.
  1480. */
  1481. odp = dp; /* Remember where we started. */
  1482. xxrc = xxcq = NUL; /* Clear these. */
  1483. /*
  1484.   Now encode the character according to the options that are in effect:
  1485.     ctlp[]: whether this control character needs prefixing.
  1486.     rptflg: repeat counts enabled.
  1487.     Other options don't apply in this routine.
  1488. */
  1489. if (rptflg && rt == rnext && first == 0) { /* Got a run... */
  1490.     if (++rpt < 94) { /* Below max, just count */
  1491. continue; /* go back and get another */
  1492.     } else if (rpt == 94) { /* Reached max, must dump */
  1493. xxrc = (CHAR) tochar(rpt); /* Put the repeat count here */
  1494. rptn += rpt; /* Accumulate it for statistics */
  1495. rpt = 0; /* And reset it */
  1496.     }
  1497. } else if (rpt > 0) { /* End of run */
  1498.     xxrc = (CHAR)tochar(++rpt); /* The count */
  1499.     rptn += rpt; /* For stats */
  1500.     rpt = 0; /* Reset repeat count */
  1501. }
  1502. a7 = rt & 0177; /* Get low 7 bits of character */
  1503. if (
  1504. #ifdef CK_SPEED
  1505.     ctlp[(unsigned)(rt & 0xff)] /* Lop off any "sign" extension */
  1506. #else
  1507.     (a7 < SP) || (a7 == DEL)
  1508. #endif /* CK_SPEED */
  1509.     ) { /* Do control prefixing if necessary */
  1510.     xxcq = myctlq; /* The prefix */
  1511.     ccp++; /* Count it */
  1512.     rt = (CHAR) ctl(rt); /* Uncontrollify the character */
  1513. }
  1514. #ifdef CK_SPEED
  1515. else if ((a7 < SP) || (a7 == DEL)) /* Count an unprefixed one */
  1516.   ccu++;
  1517. #endif /* CK_SPEED */
  1518. if (a7 == myctlq) /* Always prefix the control prefix */
  1519.   xxcq = myctlq;
  1520. if ((rptflg) && (a7 == rptq)) /* If it's the repeat prefix, */
  1521.   xxcq = myctlq; /* prefix it if doing repeat counts */
  1522. /* Now construct the prefixed sequence */
  1523. if (xxrc) { /* Repeat count */
  1524.     if (xxrc == (CHAR) '"' && !xxcq) { /* 2 in a row & not prefixed */
  1525. *dp++ = rt; /* So just do this */
  1526.     } else { /* More than two or prefixed */
  1527. *dp++ = (CHAR) rptq; *dp++ = xxrc; /* Emit repeat sequence */
  1528.     }
  1529. }
  1530. if (xxcq) { *dp++ = myctlq; } /* Control prefix */
  1531. *dp++ = rt; /* Finally, the character itself */
  1532. rt = rnext; /* Next character is now current. */
  1533. /* Done encoding the character.  Now take care of packet buffer overflow. */
  1534. size = dp - data; /* How many bytes we put in buffer. */
  1535. if (size >= bufmax) { /* If too big, save some for next. */
  1536.     *dp = ''; /* Mark the end. */
  1537.     if (size > bufmax) { /* if packet is overfull */
  1538. /* Copy the part that doesn't fit into the leftover buffer, */
  1539. /* taking care not to split a prefixed sequence. */
  1540. int i;
  1541. nleft = dp - odp;
  1542. p1 = leftover;
  1543. p2 = odp;
  1544. for (i = 0; i < nleft; i++)
  1545.   *p1++ = *p2++;
  1546. size = odp - data; /* Return truncated packet. */
  1547. *odp = ''; /* Mark the new end */
  1548.     }
  1549.     t = rt; /* Save for next time */
  1550.     return(size);
  1551. }
  1552.     } /* Otherwise, keep filling. */
  1553.     size = dp - data; /* End of file */
  1554.     *dp = ''; /* Mark the end of the data. */
  1555.     return(size);      /* Return partially filled last packet. */
  1556. }
  1557. #endif /* CKTUNING */
  1558. VOID
  1559. dofilcrc(c) int c; { /* Accumulate file crc */
  1560.     long z;
  1561.     z = crc16 ^ (long)c;
  1562.     crc16 = (crc16 >> 8) ^
  1563.       (crcta[(z & 0xF0) >> 4] ^ crctb[z & 0x0F]);
  1564. }
  1565. /* For SENDing from an array... */
  1566. int
  1567. agnbyte() { /* Get next byte from array */
  1568. #ifndef NOSPL
  1569.     char c;
  1570.     static int save = 0; /* For CRLF */
  1571.     static char ** ap = NULL; /* Array pointer */
  1572.     static char * p = NULL; /* Character pointer */
  1573.     static int i = 0, n = 0; /* Array index and limit */
  1574.     extern int a_dim[]; /* Array dimension */
  1575.     int x = 1;
  1576.     if (!ap) { /* First time thru */
  1577. ap = sndarray; /* Set up array pointers */
  1578. if (!ap || (i = sndxlo) > a_dim[sndxin]) {
  1579.     sndarray = NULL;
  1580.     ap = NULL;
  1581.     return(-1);
  1582. }
  1583. p = ap[i]; /* Point to first element in range */
  1584. n = sndxhi; /* Index of last element in range */
  1585. if (sndxhi > a_dim[sndxin]) /* Adjust if necessary */
  1586.   n = a_dim[sndxin];
  1587.     }
  1588.     if (save) { /* If anything saved */
  1589. c = save; /* unsave it */
  1590. save = 0; /* and return it */
  1591. return(c & 0xff);
  1592.     }
  1593.     if (i > n) { /* No more elements */
  1594. sndarray = NULL;
  1595. ap = NULL;
  1596. return(-1);
  1597.     }
  1598.     if (!p) /* Source pointer is NULL */
  1599.       c = NUL; /* this means an empty line */
  1600.     else /* Source pointer not NULL */
  1601.       c = *p++; /* Next char */
  1602.     if (!c) { /* Char is empty? */
  1603. if (!binary) { /* Text: end of line. */
  1604.     if (feol) { /* Supply NL */
  1605. c = feol;
  1606.     } else { /* or CRLF */
  1607. save = LF;
  1608. c = CR;
  1609.     }
  1610.     p = ap[++i];
  1611.     return(c & 0xff);
  1612. }
  1613. while (i++ < n) { /* Binary - get next element */
  1614.     p = ap[i];
  1615.     if (!p) /* Empty line? */
  1616.       continue; /* Ignore it and get another */
  1617.     c = *p++; /* Get next char */
  1618.     if (!c) /* Emtpy char? */
  1619.       continue; /* Ignore it and get another */
  1620.     return(c & 0xff); /* Not empty - return it */
  1621. }
  1622. sndarray = NULL;
  1623. ap = NULL;
  1624. return(-1); /* Done */
  1625.     }
  1626.     return(c & 0xff); /* Char is not empty */
  1627. #else
  1628.     sndarray = NULL;
  1629.     return(-1);
  1630. #endif /* NOSPL */
  1631. }
  1632. #endif /* NOXFER */
  1633. #ifndef NOCSETS
  1634. static CHAR xlabuf[32] = { 0, 0, 0, 0, 0, 0, 0, 0 };
  1635. static int xlacount = 0;
  1636. static int xlaptr = 0;
  1637. static USHORT lastucs2 = 0;
  1638. /*
  1639.   X G N B Y T E --  Get next translated byte from the input file.
  1640.   Returns the next byte that is to be put into the packet, already translated.
  1641.   This isolates getpkt() from having to know anything about translation,
  1642.   single- vs multibyte character sets, one-to-many vs many-to-one, etc, but it
  1643.   has rather high overhead, so don't call it unless you know translation is
  1644.   needed to or from Unicode, Japanese, or other multibyte character set.
  1645.   Call with:
  1646.     fcs:  File character set (source, file we are reading from)
  1647.     tcs:  Target character set (use an FC_xxx code, not a TC_xxx code)
  1648.   Returns:
  1649.     >= 0: A translated byte suitable for writing.
  1650.     <  0: Fatal error (such as EOF on input source).
  1651. */
  1652. int
  1653. xgnbyte(tcs,fcs) int tcs, fcs; {
  1654.     _PROTOTYP( int (*xx), (USHORT) ) = NULL;
  1655. #ifdef UNICODE
  1656.     extern int ucsorder; /* SET FILE UCS BYTE-ORDER */
  1657. #endif /* UNICODE */
  1658.     int haveuc = 0; /* Flag for have Unicode character */
  1659. #ifdef KANJI
  1660.     int havesj = 0; /* Have Shift-JIS character */
  1661.     int haveeu = 0; /* Have EUC-JP character */
  1662. #endif /* KANJI */
  1663.     int rc = -1, x = 0, flag = 0;
  1664.     int utferror = 0;
  1665.     int eolflag = 0;
  1666.     unsigned int xc, thischar;
  1667.     static int swapping = 0;
  1668.     CHAR rt;
  1669.     USHORT ch;
  1670. #ifdef UNICODE
  1671.     union ck_short uc;
  1672. #endif /* UNICODE */
  1673. #ifdef KANJI
  1674.     union ck_short sj, eu; /* Shift-JIS character */
  1675. #endif /* KANJI */
  1676. #ifdef KANJI
  1677.     sj.x_short = 0;
  1678. #endif /* KANJI */
  1679.     if (xlacount-- > 0) { /* We already have some */
  1680. x = xlabuf[xlaptr++];
  1681. debug(F001,"xgnbyte SEND from buf","",x);
  1682. return(x);
  1683.     }
  1684.     if (xlatype != XLA_NONE) { /* Not not translating... */
  1685. haveuc = 0;
  1686. #ifdef UNICODE
  1687. if (fcs == FC_UCS2) { /* UCS-2: Read two bytes */
  1688.     if (ffc == 0) { /* Beginning of file? */
  1689. swapping = 0; /* Reset byte-swapping flag */
  1690.     }
  1691.     uc.x_short = 0;
  1692.   bomskip:
  1693.     x = zminchar(); /* Get first byte */
  1694.     flag = 1; /* Remember we called zminchar() */
  1695.     if (x > -1) { /* Didn't fail */
  1696. ffc++; /* Count a file byte */
  1697. uc.x_char[swapping] = x & 0xff;
  1698. #ifndef NOXFER
  1699. if (docrc && what == W_SEND)
  1700.   dofilcrc(x);
  1701. #endif /* NOXFER */
  1702. if ((x = zminchar()) > -1) { /* If didn't fail */
  1703.     ffc++; /* count another file byte */
  1704.     uc.x_char[1-swapping] = x & 0xff;
  1705.     haveuc = 1; /* And remember we have Unicode */
  1706. #ifndef NOXFER
  1707.     if (docrc && what == W_SEND)
  1708.       dofilcrc(x);
  1709. #endif /* NOXFER */
  1710.     if (ffc == 2) { /* Second char of file */
  1711. debug(F001,"xgnbyte 1st UCS2","",uc.x_short);
  1712. if (uc.x_short == (USHORT)0xfeff) {
  1713.     swapping = 0;
  1714.     debug(F101,
  1715.   "xgnbyte UCS2 goodbom swap","",swapping);
  1716.     goto bomskip;
  1717. } else if (uc.x_short == (USHORT)0xfffe) {
  1718.     swapping = 1;
  1719.     debug(F101,
  1720.   "xgnbyte UCS2 badbom swap","",swapping);
  1721.     goto bomskip;
  1722. } else if ((byteorder && !ucsorder) ||
  1723.    (!byteorder && ucsorder)) {
  1724.     CHAR c;
  1725.     c = uc.x_char[0];
  1726.     uc.x_char[0] = uc.x_char[1];
  1727.     uc.x_char[1] = c;
  1728.     swapping = 1;
  1729.     debug(F101,
  1730.   "xgnbyte UCS2 no BOM X swap","",swapping);
  1731. } else {
  1732.     swapping = 0;
  1733.     debug(F101,
  1734.   "xgnbyte UCS2 no BOM Y swap","",swapping);
  1735. }
  1736.     }
  1737. } else
  1738.   return(x);
  1739.     } else
  1740.       return(x);
  1741.     debug(F001,"xgnbyte UCS2","",uc.x_short);
  1742. } else if (fcs == FC_UTF8) { /* File is UTF-8 */
  1743.     CHAR ch = 0; /* Data types needed for API... */
  1744.     USHORT * us = NULL;
  1745.     uc.x_short = 0;
  1746.     flag = 1; /* We (will) have called zminchar() */
  1747.     while ((x = zminchar()) > -1) { /* Read source bytes */
  1748. ffc++; /* Got a byte - count it */
  1749. #ifndef NOXFER
  1750. if (docrc && what == W_SEND)
  1751.   dofilcrc(x);
  1752. #endif /* NOXFER */
  1753. ch = x;
  1754. rc = utf8_to_ucs2(ch,&us); /* Convert to UCS-2 */
  1755. if (rc == 0) { /* Done */
  1756.     uc.x_short = *us;
  1757.     haveuc = 1;
  1758.     break;
  1759. } else if (rc < 0) { /* Error */
  1760.     utferror = 1;
  1761.     debug(F101,"xgnbyte UTF-8 input error","",rc);
  1762.     haveuc = 1;
  1763.     uc.x_short = *us;
  1764.     break;
  1765. }
  1766.     }
  1767.     if (x < 0)
  1768.       return(x);
  1769.     debug(F001,"xgnbyte UTF8->UTF2","",uc.x_short);
  1770. }
  1771. #endif /* UNICODE */
  1772. #ifdef KANJI
  1773. #ifdef UNICODE
  1774. else
  1775. #endif /* UNICODE */
  1776.   if (fcsinfo[fcs].alphabet == AL_JAPAN) { /* Japanese source file */
  1777.     int c7, x, y, done = 0;
  1778.     if (fcs == FC_JIS7) { /* If file charset is JIS-7 */
  1779. if (ffc == 0L) /* If first byte of file */
  1780.   j7init(); /* Initialize JIS-7 parser */
  1781. x = getj7(); /* Get a JIS-7 byte */
  1782.     } else /* Otherwise */
  1783.       x = zminchar(); /* Just get byte */
  1784.     if (x < 0) { /* Propogate EOF or error */
  1785. debug(F100,"XGNBYTE EOF","",0);
  1786. return(x);
  1787.     }
  1788.     debug(F001,"XGNBYTE x","",x);
  1789.     ffc++; /* Count */
  1790. #ifndef NOXFER
  1791.     if (docrc && what == W_SEND) dofilcrc(x); /* Do CRC */
  1792. #endif /* NOXFER */
  1793.     switch (fcs) { /* What next depends on charset */
  1794.       case FC_SHJIS: /* Shift-JIS */
  1795. if ((x <= 0x80) || /* Any 7-bit char... */
  1796.     (x >= 0xa0 && x <= 0xdf)) { /* or halfwidth Katakana */
  1797.     sj.x_short = (USHORT) x;    /* we read one byte. */
  1798. } else { /* Anything else */
  1799.     if ((y = zminchar()) < 0) /* get another */
  1800.       return(y);
  1801. #ifndef NOXFER
  1802.     if (docrc && what == W_SEND) dofilcrc(y);
  1803. #endif /* NOXFER */
  1804.     ffc++;
  1805.     sj.x_char[byteorder] = (CHAR) x;
  1806.     sj.x_char[1-byteorder] = (CHAR) y;
  1807. }
  1808. break;
  1809.       case FC_JIS7: /* JIS-7 */
  1810.       case FC_JDEC: /* DEC Kanji */
  1811.       case FC_JEUC: /* EUC-JP */
  1812. if ((x & 0x80) == 0) { /* Convert to Shift-JIS */
  1813.     sj.x_short = (USHORT) x; /* C0 or G0: one byte */
  1814.     eu.x_short = (USHORT) x;
  1815.     haveeu = 1;
  1816. } else {
  1817.     c7 = x & 0x7f;
  1818.     if (c7 > 0x20 && c7 < 0x7f) { /* Kanji: two bytes */
  1819. if ((y = (fcs == FC_JEUC) ? zminchar() : getj7()) < 0)
  1820.   return(y);
  1821. ffc++;
  1822. #ifndef NOXFER
  1823. if (docrc && what == W_SEND) dofilcrc(y);
  1824. #endif /* NOXFER */
  1825. eu.x_char[byteorder] = (CHAR) x;
  1826. eu.x_char[1-byteorder] = (CHAR) y;
  1827. sj.x_short = eu_to_sj(eu.x_short);
  1828. haveeu = 1;
  1829.     } else if (x == 0x8e) { /* SS2 Katakana prefix: 2 bytes */
  1830. if ((y = (fcs == FC_JIS7) ? getj7() : zminchar()) < 0)
  1831.   return(y);
  1832. ffc++;
  1833. #ifndef NOXFER
  1834. if (docrc && what == W_SEND) dofilcrc(y);
  1835. #endif /* NOXFER */
  1836. sj.x_short = y | 0x80;
  1837. debug(F001,"XGNBYTE KANA SJ","",sj.x_short);
  1838.     } else {
  1839. /* Something that translates to U+FFFD */
  1840. sj.x_short = UNKSJIS;
  1841.     }
  1842. }
  1843. break;
  1844.     }
  1845.     havesj = 1; /* Have Shift-JIS */
  1846. #ifdef UNICODE
  1847.     uc.x_short = sj_to_un(sj.x_short); /* Translate to UCS-2 */
  1848.     haveuc = 1; /* Have Unicode */
  1849. #endif /* UNICODE */
  1850.     flag = 1; /* Have a char */
  1851. }
  1852. #endif /* KANJI */
  1853.     }
  1854.     if (!flag) { /* If no character was read yet... */
  1855. if ((x = zminchar()) > -1) /* read one now */
  1856.   ffc++;
  1857. debug(F101,"xgnbyte zminchar 1","",x);
  1858. if (x < 0)
  1859.   return(x);
  1860. haveuc = 0;
  1861.     }
  1862. #ifdef UNICODE
  1863.     if (haveuc) {
  1864. thischar = uc.x_short;
  1865. lastucs2 = uc.x_short;
  1866.     } else
  1867. #endif /* UNICODE */
  1868.       thischar = x;
  1869.     debug(F001,"xgnbyte thischar",haveuc ? "[UNICODE]" : "[other]",thischar);
  1870. #ifdef CK_CTRLZ /* SET EOF CTRLZ */
  1871.     if (eofmethod == XYEOF_Z && !binary && thischar == 26) {
  1872. debug(F100,"xgnbyte EOF on Ctrl-Z 1","",0);
  1873. return(-1);
  1874.     }
  1875. #endif /* CK_CTRLZ */
  1876. #ifdef UNICODE
  1877.     if (!haveuc) /* If not Unicode... */
  1878. #endif /* UNICODE */
  1879.       x &= fmask; /* Apply SET FILE BYTESIZE mask */
  1880.     switch (xlatype) { /* Translation type... */
  1881. #ifdef UNICODE
  1882.       case XLA_UNICODE: { /* Unicode is involved */
  1883.   xc = 0;
  1884. /*
  1885.   Here we must choose the appropriate translation function.  If we are being
  1886.   called by getpkt() (i.e. when transferring a file), we are translating from
  1887.   Unicode to the Transfer Character Set and therefore must use the function
  1888.   pointed to by xut.  Otherwise, e.g. during TRANSLATE, CONNECT, TRANSMIT, etc,
  1889.   we are translating from Unicode to the File Character Set and so must call
  1890.   the function pointed to by xuf.  There might be a cleaner way to set this
  1891.   up but I don't think so.  For example, setxlatype() might have been called
  1892.   too soon and so might not have known whether it was a file transfer or a
  1893.   local operation.
  1894. */
  1895.   xx = (what == W_SEND) ? xut : xuf;
  1896.   eolflag = 0;
  1897.   if (haveuc) { /* File is Unicode */
  1898.       /* See Unicode TR13, "Converting to Other Character Sets" */
  1899.       if (uc.x_short == 0x2028 || /* Line Separator? */
  1900.   uc.x_short == 0x2029 || /* Paragraph Separator */
  1901.   (feol && (uc.x_short == (USHORT)feol))
  1902.   ) {
  1903.   debug(F001,"xgnbyte uc eol","",uc.x_short);
  1904.   rc = 0;
  1905.   eolflag = 1; /* Don't translate and handle later */
  1906.       }
  1907.       if (xx && !eolflag) { /* UCS-to-TCS function (UCS->byte) */
  1908.   rc = (*xx)(uc.x_short); /* These can fail... */
  1909.   debug(F101,"xgnbyte xx rc","",rc);
  1910.   if (rc < 0) /* If it can't be translated */
  1911.     uc.x_short = UNK; /* Put unknown-character symbol */
  1912.   else
  1913.     uc.x_short = (unsigned)((unsigned)rc & 0xffff);
  1914.   debug(F101,"xgnbyte xx uc","",uc.x_short);
  1915.       }
  1916. #ifdef KANJI
  1917.       if (tcs == FC_JEUC) { /* Translating to EUC-JP */
  1918.   USHORT sj;
  1919.   union ck_short eu;
  1920.   debug(F001,"xgnbyte UCS->EUC UCS","",uc.x_short);
  1921.   if (!havesj) /* If we don't already have it */
  1922.     sj = un_to_sj(uc.x_short); /* convert to Shift-JIS */
  1923.   eu.x_short = sj_to_eu(sj);
  1924.   debug(F001,"xgnbyte UCS->EUC EUC","",eu.x_short);
  1925.   xlaptr = 0;
  1926.   xlacount = 0;
  1927.   if (eolflag) {
  1928.       if (what == W_SEND) {
  1929.   xlabuf[xlacount++] = LF;
  1930.   return(CR);
  1931.       } else {
  1932.   return(feol);
  1933.       }
  1934.   }
  1935.   if (eu.x_char[byteorder]) { /* Two bytes */
  1936.       rc = eu.x_char[byteorder];
  1937.       xlabuf[xlacount++] = eu.x_char[1-byteorder];
  1938.       debug(F001,"xgnbyte UCS->EUC xlabuf[0]","",xlabuf[0]);
  1939.   } else { /* One byte */
  1940.       rc = eu.x_char[1-byteorder];
  1941.   }
  1942.   debug(F101,"xgnbyte UCS->EUC xlacount","",xlacount);
  1943.   debug(F001,"xgnbyte UCS->EUC rc","",rc);
  1944.   return(rc);
  1945.       } else
  1946. #endif /* KANJI */
  1947.       if (tcs != FC_UCS2 && tcs != FC_UTF8) {
  1948.   if (uc.x_short & 0xff00) { /* Decoding error */
  1949.       debug(F001,"xgnbyte decoding error","",uc.x_short);
  1950.       return(-2);
  1951.   } else
  1952.     return((unsigned int)(uc.x_short & 0xff));
  1953.       }
  1954.       xc = uc.x_short;
  1955.   } else { /* File is not Unicode */
  1956.       USHORT ch;
  1957.       /* Translate from single FCS byte to UCS-2 */
  1958. /*
  1959.   This is a bit nonobvious...  The blah_u() (Blah-to-Unicode) routines are
  1960.   called only with pieces of character sets, in the ISO 2022 sense.  So,
  1961.   for example, if ch is a Latin-1 character, we call the translation
  1962.   routine only if it is in the right half; if it's in the left half, it
  1963.   isn't translated, and in fact will give the wrong result if sent to the
  1964.   translation function.  That's because those functions were designed for
  1965.   use with the ISO 2022 G0..G3 sets, not for file transfer.  On the other
  1966.   hand, if it's a 7-bit character set, we *do* call the translation
  1967.   function.  (To put it another way, the left half of any 8-bit character
  1968.   set is ASCII and therefore doesn't need to be translated but 7-bit sets
  1969.   such as ISO 646 German do need translation).
  1970. */
  1971.       ch = (unsigned)(thischar & 0xff);
  1972.       if (((fcsinfo[fcs].size > 128) && (ch & 0x80)) ||
  1973.   fcsinfo[fcs].size <= 128) {
  1974.   if (xfu) {  /* FCS-to-UCS function */
  1975.       ch = (*xfu)(ch);
  1976.   }
  1977.       }
  1978.       xc = ch;
  1979.   }
  1980.   /* At this point we have a UCS-2 character in native format */
  1981.   /* (Big Endian or Little Endian) in xc, which is an unsigned int. */
  1982.   debug(F001,"xgnbyte xc","",xc);
  1983.   if (tcs == FC_UTF8) { /* Now convert to UTF-8 */
  1984.       USHORT c; /* NOTE: this is FC_UTF8 on purpose! */
  1985.       CHAR * buf = NULL;
  1986.       int i, k = 0, x;
  1987.       xlaptr = 0;
  1988.       if (utferror) {
  1989.   xlabuf[k++] = 0xff;
  1990.   xlabuf[k++] = 0xbd;
  1991.       }
  1992.       if (eolflag) { /* We detected EOL in source file */
  1993.   if (what == W_SEND) { /* Convert to CRLF */
  1994.       xlabuf[k++] = LF;
  1995.       xlacount = k;
  1996.       return((unsigned int)CR);
  1997.   } else { /* Or to local line-end */
  1998.       xlacount = k;
  1999.       return((unsigned int)feol);
  2000.   }
  2001.       }
  2002.       c = xc;
  2003.       if ((x = ucs2_to_utf8(c,&buf)) < 1) {
  2004.   debug(F101,"xgnbyte ucs2_to_utf8 error","",c);
  2005.   return(-2);
  2006.       }
  2007.       debug(F101,"xgnbyte UTF8 buf[0]","",buf[0]);
  2008.       for (i = 1; i < x; i++) {
  2009.   xlabuf[k+i-1] = buf[i];
  2010.   debug(F111,"xgnbyte UTF8 xlabuf",ckitoa(i-1),buf[i]);
  2011.       }
  2012.       xlaptr = 0;
  2013.       xlacount = x - 1;
  2014.       debug(F101,"xgnbyte UTF8 xlacount","",xlacount);
  2015.       return((unsigned int)buf[0]);
  2016.   } else { /* Or keep it as UCS-2 */
  2017.       int k = 0;
  2018.       CHAR c;
  2019.       xlaptr = 0;
  2020.       if (utferror) {
  2021.   xlabuf[k++] = 0xff;
  2022.   xlabuf[k++] = 0xfd;
  2023.   debug(F101,"xgnbyte error","",k);
  2024.       }
  2025.       if (eolflag) { /* We detected EOL in source file */
  2026.   if (what == W_SEND) { /* Convert to CRLF */
  2027.       xlabuf[k++] = CR;
  2028.       xlabuf[k++] = NUL;
  2029.       xlabuf[k++] = LF;
  2030.       xlacount = k;
  2031.       debug(F101,"xgnbyte send CRLF","",k);
  2032.       return(0); /* Return NUL */
  2033.   } else { /* Or to local line-end */
  2034.       xlabuf[k++] = (CHAR)feol;
  2035.       xlacount = k;
  2036.       debug(F101,"xgnbyte send feol","",k);
  2037.       return(0); /* Return NUL */
  2038.   }
  2039.       }
  2040.       if (what == W_SEND || !ucsorder) {
  2041.   xlabuf[k++] = (xc >> 8) & 0xff; /* Big Endian */
  2042.   xlabuf[k++] = xc & 0xff;
  2043.   debug(F110,"xgnbyte to UCS2 BE",ckitoa(xlabuf[0]),0);
  2044.       } else { /* Little Endian */
  2045.   xlabuf[k++] = xc & 0xff;
  2046.   xlabuf[k++] = (xc >> 8) & 0xff;
  2047.   debug(F110,"xgnbyte to UCS2 LE",ckitoa(xlabuf[0]),0);
  2048.       }
  2049.       c = xlabuf[0];
  2050.       xlaptr = 1;
  2051.       xlacount = k-1;
  2052.       debug(F101,"xgnbyte c","",c);
  2053.       debug(F101,"xgnbyte xlaptr","",xlaptr);
  2054.       debug(F101,"xgnbyte xlacount","",xlacount);
  2055.       return((unsigned int)c);
  2056.   }
  2057.       }
  2058. #endif /* UNICODE */
  2059.       case XLA_NONE:
  2060. return(zminchar());
  2061.       case XLA_BYTE: /* Byte-for-Byte translation */
  2062. rt = x;
  2063. if (sx)
  2064.   rt = (*sx)(rt);
  2065. #ifdef UNICODE
  2066. if (utferror) {
  2067.     xlaptr = 0;
  2068.     xlacount = 1;
  2069.     xlabuf[0] = rt;
  2070.     return(UNK);
  2071. } else
  2072. #endif /* UNICODE */
  2073.   return((unsigned int)rt);
  2074. #ifdef KANJI
  2075.       case XLA_JAPAN: /* Come here with Shift-JIS */
  2076. if (tcs == FC_JEUC) { /* It better be... */
  2077.     xlaptr = 0;
  2078.     xlacount = 0;
  2079.     if (!havesj) {
  2080. printf("BAD BADn");
  2081. return(-2);
  2082.     }
  2083.     if (!haveeu) /* We might already have EUC too */
  2084.       eu.x_short = sj_to_eu(sj.x_short);
  2085.     if (eu.x_char[byteorder]) {
  2086. xlabuf[xlacount++] = eu.x_char[1-byteorder];
  2087. return(eu.x_char[byteorder]);
  2088.     } else {
  2089. return(eu.x_char[1-byteorder]);
  2090.     }
  2091.     break;
  2092. }
  2093. #endif /* KANJI */
  2094.       default:
  2095. debug(F101,"xgnbyte bad xlatype","",xlatype);
  2096. return(-2);
  2097.     }
  2098.     /* NOTREACHED */
  2099.     /* Some compilers complain if this is not here, others if it is. */
  2100.     debug(F100,"xgnbyte switch failure","",0);
  2101.     return(-2);
  2102. }
  2103. #endif /* NOCSETS */
  2104. #ifndef NOXFER
  2105. /*  G E T P K T  --  Fill a packet data field from the indicated source.  */
  2106. /*
  2107.   Parameters:
  2108.     bufmax: Maximum length of entire packet.
  2109.     xlate:  Flag for whether to translate charsets when in text mode.
  2110.   Returns:  Number of characters written to packet data field, 0 or more,
  2111.             Or -1 on failure (internal error),
  2112.             or -3 on timeout (e.g. when reading from a pipe).
  2113.   This is the full version allowing for parity and text-mode conversions;
  2114.   i.e. it works in all cases.   Also see bgetpkt(), a special lean/mean/fast
  2115.   packet encoder that works only for binary-mode no-parity transfers.
  2116. */
  2117. static int uflag = 0;
  2118. int
  2119. getpkt(bufmax,xlate) int bufmax, xlate; { /* Fill one packet buffer */
  2120.     register CHAR rt = t, rnext = NUL;   /* Register shadows of the globals */
  2121.     register CHAR *dp, *odp, *odp2, *p1, *p2; /* pointers... */
  2122.     register int x; /* Loop index. */
  2123.     register int a7; /* Low 7 bits of character */
  2124.     int thischar = 0; /* Might be byte or wide */
  2125.     CHAR xxls, xxdl, xxrc, xxss, xxcq; /* Pieces of prefixed sequence */
  2126.     if (binary) xlate = 0; /* We don't translate if binary */
  2127.     if (!data) {
  2128. debug(F100,"SERIOUS ERROR: getpkt data == NULL","",0);
  2129. return(-1);
  2130.     }
  2131.     dp = data; /* Point to packet data buffer */
  2132.     size = 0; /* And initialize its size */
  2133. /*
  2134.   Assume bufmax is the receiver's total receive-packet buffer length.
  2135.   Our whole packet has to fit into it, so we adjust the data field length.
  2136.   We also decide optimally whether it is better to use a short-format or
  2137.   long-format packet when we're near the borderline.
  2138. */
  2139.     bufmax = maxdata(); /* Get maximum data length */
  2140.     if (first == 1) { /* If first character of this file.. */
  2141. #ifdef UNICODE
  2142. /* Special end-of-line handling for Unicode */
  2143. if (tcharset == TC_UCS2 || tcharset == TC_UTF8)
  2144.   uflag = 1;
  2145. #endif /* UNICODE */
  2146. debug(F101,"getpkt first uflag","",uflag);
  2147. debug(F101,"getpkt first rt","",rt);
  2148. if (!memstr && !funcstr) /* and real file... */
  2149.   ffc = 0L; /* reset file character counter */
  2150. #ifdef COMMENT
  2151. /* Moved to below... */
  2152. first = 0; /* Next character won't be first */
  2153. #endif /* COMMENT */
  2154. *leftover = ''; /* Discard any interrupted leftovers */
  2155. nleft = 0;
  2156. #ifndef NOCSETS
  2157. setxlatype(tcharset,fcharset); /* Set up charset translations */
  2158. #endif /* NOCSETS */
  2159. /* Get first character of file into rt, watching out for null file */
  2160. #ifdef CALIBRATE
  2161. if (calibrate && !memstr) {
  2162. #ifdef NORANDOM
  2163.     x = rt = 53;
  2164. #else
  2165.     x = rt = cal_a[rand() & 0xff];
  2166. #endif /* NORANDOM */
  2167.     first = 0;
  2168.     ffc++;
  2169. } else
  2170. #endif /* CALIBRATE */
  2171. #ifdef KANJI
  2172. if (xlate && tcharset == TC_JEUC) { /* Kanji text */
  2173.     x = zkanjf();
  2174.     if ((x = zkanji(memstr ? kgetm : kgetf)) < 0) {
  2175.         first = -1;
  2176.         size = 0;
  2177.         if (x == -2) {
  2178.             debug(F100,"getpkt zkanji: input error","",0);
  2179.             cxseen = 1;
  2180.         } else debug(F100,"getpkt zkanji: empty string/file","",0);
  2181.         return(0);
  2182.     }
  2183.     rt = x;
  2184.     first = 0;
  2185.     if (!memstr) {
  2186. ffc++;
  2187. if (docrc && what == W_SEND) /* Accumulate file crc */
  2188.   dofilcrc((int)rt);
  2189.     }
  2190. } else { /* Not Kanji text */
  2191. #endif /* KANJI */
  2192.     if (memstr) { /* Reading data from memory string */
  2193. /* This will not be Unicode */
  2194. if ((rt = *memptr++) == '') { /* end of string ==> EOF */
  2195.     first = -1;
  2196.     size = 0;
  2197.     debug(F100,"getpkt: empty string","",0);
  2198.     return(0);
  2199. }
  2200. first = 0;
  2201.     } else if (funcstr) { /* Reading data from a function */
  2202. /* This will not be Unicode */
  2203. if ((x = (*funcptr)()) < 0) { /* End of input  */
  2204.     first = -1;
  2205.     size = 0; /* Empty */
  2206.     return(0);
  2207. }
  2208. ffc++; /* Count a file character */
  2209. rt = (CHAR) x; /* Convert int to char */
  2210. first = 0;
  2211. debug(F000,"getpkt funcstr","",rt);
  2212.     } else { /* Reading data from a file */
  2213. #ifndef NOCSETS
  2214. if (xlate && !binary) { /* Could be Unicode */
  2215.     if (xlatype == XLA_UNICODE) {
  2216. /* Get next translated byte */
  2217. x = xgnbyte(cseqtab[tcharset],fcharset);
  2218. debug(F101,"getpkt xgnbyte","",x);
  2219.     } else { /* Not Unicode */
  2220. x = zminchar(); /* Get next byte, translate below */
  2221. debug(F101,"getpkt zminchar A","",x);
  2222.     }
  2223. } else { /* Just get next byte */
  2224. #endif /* NOCSETS */
  2225.     x = zminchar();
  2226.     debug(F101,"getpkt zminchar B","",x);
  2227. #ifndef NOCSETS
  2228. }
  2229. #endif /* NOCSETS */
  2230. if (x < 0) { /* End of file or input error */
  2231.     if (x == -3) { /* Timeout. */
  2232. size = (dp-data);
  2233. debug(F101,"getpkt timeout size","",size);
  2234. return((size == 0) ? x : size);
  2235.     }
  2236.     first = -1;
  2237.     size = 0;
  2238.     if (x == -2) { /* Error */
  2239. debug(F100,"getpkt: input error","",0);
  2240. cxseen = 1; /* Interrupt the file transfer */
  2241.     } else {
  2242. debug(F100,"getpkt empty file","",0);
  2243.     }
  2244.     return(0);
  2245. }
  2246. first = 0; /* Next character won't be first */
  2247. rt = (CHAR) x; /* Convert int to char */
  2248. #ifndef NOCSETS
  2249. if (xlatype != XLA_UNICODE || binary) {
  2250.     ffc++;
  2251.     if (sx)
  2252.       rt = (*sx)(rt);
  2253.     if (docrc && what == W_SEND)
  2254.       dofilcrc(x);
  2255. }
  2256. #endif /* NOCSETS */
  2257. #ifdef DEBUG
  2258. if (deblog)
  2259.   debug(F101,"getpkt 1st char","",rt);
  2260. #endif /* DEBUG */
  2261. if (/* !haveuc && */ docrc && what == W_SEND) /* File CRC */
  2262.   dofilcrc(x);
  2263.     }
  2264. #ifdef KANJI
  2265. }
  2266. #endif /* KANJI */
  2267. /* PWP: handling of feol is done later (in the while loop)... */
  2268.     } else if ((first == -1) && (nleft == 0)) { /* EOF from last time */
  2269. #ifdef DEBUG
  2270. if (deblog) {
  2271.     debug(F101,"getpkt eof crc16","",crc16);
  2272.     debug(F101,"getpkt eof ffc","",ffc);
  2273. }
  2274. #endif /* DEBUG */
  2275.         return(size = 0);
  2276.     }
  2277. /*
  2278.   Here we handle characters that were encoded for the last packet but
  2279.   did not fit, and so were saved in the "leftover" array.
  2280. */
  2281.     debug(F101,"getpkt nleft","",nleft);
  2282.     if (nleft) {
  2283. for (p1 = leftover; nleft > 0; nleft--) /* Copy leftovers */
  2284.   *dp++ = *p1++;
  2285. *leftover = ''; /* Delete leftovers */
  2286. nleft = 0;
  2287.     }
  2288.     if (first == -1) /* Handle EOF */
  2289.       return(size = (dp - data));
  2290. /* Now fill up the rest of the packet. */
  2291.     rpt = 0; /* Initialize character repeat count */
  2292.     while (first > -1) { /* Until EOF... */
  2293. #ifdef CALIBRATE
  2294. if (calibrate && !memstr) { /* We generate our own "file" */
  2295.     if (ffc >= calibrate) { /* EOF */
  2296. first = -1;
  2297. ffc--;
  2298.     } else { /* Generate next character */
  2299. if (cal_j > CAL_M * ffc)
  2300.   cal_j = cal_a[ffc & 0xff];
  2301. x = (unsigned)cal_a[(cal_j & 0xff)];
  2302. if (x == rt) x ^= 2;
  2303.     }
  2304.     cal_j += (unsigned int)(ffc + CAL_O);
  2305.     ffc++;
  2306. } else
  2307. #endif /* CALIBRATE */
  2308. #ifdef KANJI
  2309.   if (xlate && tcharset == TC_JEUC) {
  2310.       if ((x = zkanji(memstr ? kgetm : kgetf)) < 0) {
  2311.   first = -1;
  2312.   if (x == -2) cxseen = 1;
  2313.       } else if (!memstr) ffc++;
  2314.       rnext = (CHAR) (x & fmask);
  2315.   } else {
  2316. #endif /* KANJI */
  2317.     if (memstr) { /* Get next char from memory string */
  2318. if ((x = *memptr++) == '') /* End of string means EOF */
  2319.   first = -1; /* Flag EOF for next time. */
  2320. rnext = (CHAR) (x & fmask); /* Apply file mask */
  2321.     } else if (funcstr) { /* Get next char from function */
  2322. if ((x = (*funcptr)()) < 0) /* End of string means EOF */
  2323.   first = -1; /* Flag EOF for next time. */
  2324. rnext = (CHAR) (x & fmask); /* Apply file mask */
  2325.     } else { /* From file... */
  2326. #ifndef NOCSETS
  2327. if (xlate && !binary) { /* Could be Unicode */
  2328.     if (xlatype == XLA_UNICODE) {
  2329. /* Get next translated byte */
  2330. x = xgnbyte(cseqtab[tcharset],fcharset);
  2331.     } else { /* Not Unicode */
  2332. x = zminchar(); /* Get next byte, translate below */
  2333. debug(F101,"xgnbyte B zminchar","",x);
  2334.     }
  2335. } else { /* Just get next byte */
  2336. #endif /* NOCSETS */
  2337.     x = zminchar();
  2338.     debug(F101,"xgnbyte C zminchar","",x);
  2339. #ifndef NOCSETS
  2340. }
  2341. #endif /* NOCSETS */
  2342. if (x < 0) { /* Check for EOF */
  2343.     if (x == -3) { /* Timeout reading from pipe */
  2344. t = rt;
  2345. size = (dp-data);
  2346. debug(F101,"getpkt timeout size","",size);
  2347. return((size == 0) ? x : size);
  2348.     }
  2349.     first = -1; /* Flag eof for next time. */
  2350.     if (x == -2) cxseen = 1; /* If error, cancel this file. */
  2351. }
  2352. rnext = (CHAR) (x & fmask); /* Apply file mask */
  2353. #ifndef NOCSETS
  2354. if (xlatype != XLA_UNICODE) {
  2355. #endif /* NOCSETS */
  2356.     ffc++;
  2357. #ifndef NOCSETS
  2358.     if (sx)
  2359.       rt = (*sx)(rt);
  2360. #endif /* NOCSETS */
  2361.     if (docrc && what == W_SEND)
  2362.       dofilcrc(x);
  2363. #ifndef NOCSETS
  2364. }
  2365. #endif /* NOCSETS */
  2366.     }
  2367. #ifdef KANJI
  2368. }
  2369. #endif /* KANJI */
  2370. /*
  2371.   At this point, the character we just read is in rnext,
  2372.   and the character we are about to encode into the packet is in rt.
  2373. */
  2374. odp = dp; /* Remember where we started. */
  2375.   xxls = xxdl = xxrc = xxss = xxcq = NUL; /* Clear these. */
  2376. /*
  2377.   Now encode the character according to the options that are in effect:
  2378.     ctlp[]: whether this control character needs prefixing.
  2379.     binary: text or binary mode.
  2380.     rptflg: repeat counts enabled.
  2381.     ebqflg: 8th-bit prefixing enabled.
  2382.     lscapu: locking shifts enabled.
  2383. */
  2384. if (rptflg) { /* Repeat processing is on? */
  2385.     if (!uflag &&
  2386. /*
  2387.  * If the next char is really CRLF, then we cannot
  2388.  * be doing a repeat (unless CR,CR,LF which becomes
  2389.  * "~ <n-1> CR CR LF", which is OK but not most efficient).
  2390.  * I just plain don't worry about this case.  The actual
  2391.  * conversion from NL to CRLF is done after the rptflg if...
  2392.  */
  2393. (!feol || binary || (feol && (rnext != feol))) &&
  2394. (rt == rnext) && (first == 0)) { /* Got a run... */
  2395. if (++rpt < 94) { /* Below max, just count */
  2396.     continue; /* go back and get another */
  2397. } else if (rpt == 94) { /* Reached max, must dump */
  2398.     xxrc = (CHAR) tochar(rpt); /* Put the repeat count here */
  2399.     rptn += rpt; /* Accumulate it for statistics */
  2400.     rpt = 0; /* And reset it */
  2401. }
  2402.     } else if (rpt > 1) { /* More than two */
  2403. xxrc = (CHAR) tochar(++rpt); /* and count. */
  2404. rptn += rpt;
  2405. rpt = 0; /* Reset repeat counter. */
  2406.     }
  2407.     /*
  2408.       If (rpt == 1) we must encode exactly two characters.
  2409.       This is done later, after the first character is encoded.
  2410.     */
  2411. }
  2412. /* If it's the newline character... */
  2413. if (!uflag && !binary && feol && (rt == feol)) {
  2414.     if (lscapu && lsstate) { /* If SHIFT-STATE is SHIFTED */
  2415. if (ebqflg) { /* If single shifts enabled, */
  2416.     *dp++ = (CHAR) ebq; /* insert a single shift. */
  2417. } else { /* Otherwise must shift in. */
  2418.     *dp++ = myctlq; /* Insert shift-out code */
  2419.     *dp++ = 'O';
  2420.     lsstate = 0; /* Change shift state */
  2421. }
  2422.     }
  2423. #ifdef CK_SPEED
  2424.     if (ctlp[CR]) {
  2425. *dp++ = myctlq; /* Insert carriage return directly */
  2426. *dp++ = 'M';
  2427. ccp++;
  2428.     } else {
  2429. *dp++ = CR; /* Perhaps literally */
  2430. ccu++;
  2431.     }
  2432. #else /* !CK_SPEED */
  2433.     *dp++ = myctlq; /* Insert carriage return directly */
  2434.     *dp++ = 'M';
  2435.     ccp++;
  2436. #endif /* CK_SPEED */
  2437.     rt = LF; /* Now make next char be linefeed. */
  2438. }
  2439. /*
  2440.   Now handle the 8th bit of the file character.  If we have an 8-bit
  2441.   connection, we preserve the 8th bit.  If we have a 7-bit connection,
  2442.   we employ either single or locking shifts (if they are enabled).
  2443. */
  2444. a7 = rt & 0177; /* Get low 7 bits of character */
  2445. if (rt & 0200) { /* 8-bit character? */
  2446.     if (lscapu) { /* Locking shifts enabled? */
  2447. if (!lsstate) { /* Not currently shifted? */
  2448.     x = lslook(0200); /* Look ahead */
  2449.     if (x != 0 || ebqflg == 0) { /* Locking shift decision */
  2450. xxls = 'N';    /* Need locking shift-out */
  2451. lsstate = 1;    /* and change to shifted state */
  2452.     } else if (ebqflg) {   /* Not worth it */
  2453. xxss = (CHAR) ebq; /* Use single shift */
  2454.     }
  2455. }
  2456. rt = (CHAR) a7; /* Replace character by 7-bit value */
  2457.     } else if (ebqflg) { /* 8th bit prefixing is on? */
  2458. xxss = (CHAR) ebq; /* Insert single shift */
  2459. rt = (CHAR) a7; /* Replace character by 7-bit value */
  2460.     }
  2461. /*
  2462.   In case we have a 7-bit connection and this is an 8-bit character, AND
  2463.   neither locking shifts nor single shifts are enabled, then the character's
  2464.   8th bit will be destroyed in transmission, and a block check error will
  2465.   occur.
  2466. */
  2467. } else if (lscapu) { /* 7-bit character */
  2468.     if (lsstate) { /* Comes while shifted out? */
  2469. x = lslook(0); /* Yes, look ahead */
  2470. if (x || ebqflg == 0) { /* Time to shift in. */
  2471.     xxls = 'O'; /* Set shift-in code */
  2472.     lsstate = 0; /* Exit shifted state */
  2473. } else if (ebqflg) { /* Not worth it, stay shifted out */
  2474.     xxss = (CHAR) ebq; /* Insert single shift */
  2475. }
  2476.     }
  2477. }
  2478. /* If data character is significant to locking shift protocol... */
  2479. if (lscapu && (a7 == SO || a7 == SI || a7 == DLE))
  2480.   xxdl = 'P'; /* Insert datalink escape */
  2481. if (
  2482. #ifdef CK_SPEED
  2483.     /*
  2484.       Thwart YET ANOTHER unwanted, unneeded, and unloved sign
  2485.       extension.  This one was particularly nasty because it prevented
  2486.       255 (Telnet IAC) from being prefixed on some platforms -- e.g.
  2487.       VMS with VAX C -- but not others, thus causing file transfers to
  2488.       fail on Telnet connections by sending bare IACs.  Not to mention
  2489.       the stray memory reference.  Signed chars are a BAD idea.
  2490.     */
  2491.     ctlp[(unsigned)(rt & 0xff)] /* Lop off any "sign" extension */
  2492. #else
  2493.     (a7 < SP) || (a7 == DEL)
  2494. #endif /* CK_SPEED */
  2495.     ) { /* Do control prefixing if necessary */
  2496.     xxcq = myctlq; /* The prefix */
  2497.     ccp++; /* Count it */
  2498.     rt = (CHAR) ctl(rt); /* Uncontrollify the character */
  2499. }
  2500. #ifdef CK_SPEED
  2501. else if ((a7 < SP) || (a7 == DEL)) /* Count an unprefixed one */
  2502.   ccu++;
  2503. #endif /* CK_SPEED */
  2504. if (a7 == myctlq) /* Always prefix the control prefix */
  2505.   xxcq = myctlq;
  2506. if ((rptflg) && (a7 == rptq)) /* If it's the repeat prefix, */
  2507.   xxcq = myctlq; /* prefix it if doing repeat counts */
  2508. if ((ebqflg) && (a7 == ebq)) /* Prefix the 8th-bit prefix */
  2509.   xxcq = myctlq; /* if doing 8th-bit prefixes */
  2510. /* Now construct the entire sequence */
  2511. if (xxls) { *dp++ = myctlq; *dp++ = xxls; } /* Locking shift */
  2512. odp2 = dp;     /* (Save this place) */
  2513. if (xxdl) { *dp++ = myctlq; *dp++ = xxdl; } /* Datalink escape */
  2514. if (xxrc) { *dp++ = (CHAR) rptq; *dp++ = xxrc; } /* Repeat count */
  2515. if (xxss) { *dp++ = (CHAR) ebq; }           /* Single shift */
  2516. if (xxcq) { *dp++ = myctlq; }          /* Control prefix */
  2517. *dp++ = rt; /* Finally, the character itself */
  2518. if (rpt == 1) { /* Exactly two copies? */
  2519.     rpt = 0;
  2520.     p2 = dp; /* Save place temporarily */
  2521.     for (p1 = odp2; p1 < p2; p1++) /* Copy the old chars over again */
  2522.       *dp++ = *p1;
  2523.     if ((p2-data) <= bufmax) odp = p2; /* Check packet bounds */
  2524.     if ((p2-data) < bufmax) odp = p2; /* Check packet bounds */
  2525. }
  2526. rt = rnext; /* Next character is now current. */
  2527. /* Done encoding the character.  Now take care of packet buffer overflow. */
  2528. if ((dp-data) >= bufmax) { /* If too big, save some for next. */
  2529.     debug(F000,"getpkt EOP","",rt);
  2530.     size = (dp-data); /* Calculate the size. */
  2531.     *dp = ''; /* Mark the end. */
  2532.     if (memstr) { /* No leftovers for memory strings */
  2533. if (rt) /* Char we didn't encode yet */
  2534.   memptr--; /* (for encstr()) */
  2535. return(size);
  2536.     }
  2537.     if ((dp-data) > bufmax) { /* if packet is overfull */
  2538. /* copy the part that doesn't fit into the leftover buffer, */
  2539. /* taking care not to split a prefixed sequence. */
  2540. int i;
  2541. nleft = dp - odp;
  2542. for (i = 0, p1 = leftover, p2 = odp; i < nleft; i++) {
  2543.     *p1++ = *p2++;
  2544.     if (memstr) memptr--; /* (for encstr) */
  2545. }
  2546. debug(F111,"getpkt leftover",leftover,size);
  2547. debug(F101,"getpkt osize","",(odp-data));
  2548. size = (odp-data); /* Return truncated packet. */
  2549. *odp = ''; /* Mark the new end */
  2550.     }
  2551.     t = rt; /* Save for next time */
  2552.     return(size);
  2553. }
  2554.     } /* Otherwise, keep filling. */
  2555.     size = (dp-data); /* End of file */
  2556.     *dp = ''; /* Mark the end of the data. */
  2557.     debug(F111,"getpkt eof/eot",data,size); /* Fell thru before packet full, */
  2558.     return(size);      /* return partially filled last packet. */
  2559. }
  2560. /*  T I N I T  --  Initialize a transaction  */
  2561. int epktrcvd = 0, epktsent = 0;
  2562. /*
  2563.   Call with 1 to reset everything before S/I/Y negotiation, or 0 to
  2564.   reset only the things that are not set in the S/I/Y negotiation.
  2565.   Returns -1 on failure (e.g. to create packet buffers), 0 on success.
  2566. */
  2567. int
  2568. tinit(flag) int flag; {
  2569.     int x;
  2570. #ifdef CK_TIMERS
  2571.     extern int rttflg;
  2572. #else
  2573.     extern int rcvtimo;
  2574. #endif /* CK_TIMERS */
  2575.     extern int fatalio;
  2576.     debug(F101,"tinit flag","",flag);
  2577.     *epktmsg = NUL;
  2578.     epktrcvd = 0;
  2579.     epktsent = 0;
  2580.     ofperms = "";
  2581.     diractive = 0; /* DIR / REMOTE DIR not active */
  2582.     interrupted = 0; /* Not interrupted */
  2583.     fatalio = 0; /* No fatal i/o error */
  2584.     if (server) {
  2585. moving  = 0;
  2586. #ifdef PIPESEND
  2587. pipesend = 0; /* This takes care of multiple GETs sent to a server. */
  2588. #endif /* PIPESEND */
  2589.     }
  2590.     bestlen = 0; /* For packet length optimization */
  2591.     maxsend = 0; /* Biggest data field we can send */
  2592. #ifdef STREAMING
  2593.     streamok = 0; /* Streaming negotiated */
  2594.     streaming = 0; /* Streaming being done now */
  2595. #endif /* STREAMING */
  2596.     binary = b_save; /* ... */
  2597.     gnf_binary = binary; /* Per-file transfer mode */
  2598.     retrans = 0; /* Packet retransmission count */
  2599.     sndtyp = 0; /* No previous packet */
  2600.     xflg = 0; /* Reset x-packet flag */
  2601.     memstr = 0; /* Reset memory-string flag */
  2602.     memptr = NULL; /*  and buffer pointer */
  2603.     funcstr = 0; /* Reset "read from function" flag */
  2604.     funcptr = NULL; /*  and function pointer */
  2605.     autopar = 0; /* Automatic parity detection flag */
  2606.     /* This stuff is only for BEFORE S/I/Y negotiation, not after */
  2607.     if (flag) {
  2608. bctu = bctl = 1; /* Reset block check type to 1 */
  2609. myinit[0] = ''; /* Haven't sent init string yet */
  2610. rqf = -1; /* Reset 8th-bit-quote request flag */
  2611. ebq = MYEBQ; /* Reset 8th-bit quoting stuff */
  2612. ebqflg = 0; /* 8th bit quoting not enabled */
  2613. ebqsent = 0; /* No 8th-bit prefix bid sent yet */
  2614. sq = 'Y'; /* 8th-bit prefix bid I usually send */
  2615. spsiz = spsizr; /* Initial send-packet size */
  2616. debug(F101,"tinit spsiz","",spsiz);
  2617. wslots = 1; /* One window slot */
  2618. wslotn = 1; /* No window negotiated yet */
  2619. justone = 0; /* (should this be zero'd here?) */
  2620. what = W_INIT; /* Doing nothing so far... */
  2621.     }
  2622.     fncnv = f_save; /* Back to what user last said */
  2623.     pktnum = 0; /* Initial packet number to send */
  2624.     cxseen = czseen = discard = 0; /* Reset interrupt flags */
  2625.     *filnam = ''; /* Clear file name */
  2626.     spktl = 0; /* And its length */
  2627.     nakstate = 0; /* Assume we're not in a NAK state */
  2628.     numerrs = 0; /* Transmission error counter */
  2629.     idletmo = 0; /* No idle timeout yet */
  2630.     if (server) {  /* If acting as server, */
  2631. if (srvidl > 0) /* If an idle timeout is given */
  2632.   timint = srvidl;
  2633. else
  2634.   timint = srvtim; /* use server timeout interval. */
  2635.     } else { /* Otherwise */
  2636. timint = chktimo(rtimo,timef); /* and use local timeout value */
  2637.     }
  2638.     debug(F101,"tinit timint","",timint);
  2639. #ifdef CK_TIMERS
  2640.     if (rttflg && timint > 0) /* Using round-trip timers? */
  2641.       rttinit();
  2642. #else
  2643.     rcvtimo = timint;
  2644. #endif /* CK_TIMERS */
  2645.     winlo = 0; /* Packet 0 is at window-low */
  2646.     debug(F101,"tinit winlo","",winlo);
  2647.     x = mksbuf(1); /* Make a 1-slot send-packet buffer */
  2648.     if (x < 0) return(x);
  2649.     x = getsbuf(0); /* Allocate first send-buffer. */
  2650.     debug(F101,"tinit getsbuf","",x);
  2651.     if (x < 0) return(x);
  2652.     dumpsbuf();
  2653.     x = mkrbuf(wslots); /* & a 1-slot receive-packet buffer. */
  2654.     if (x < 0) return(x);
  2655.     lsstate = 0; /* Initialize locking shift state */
  2656.     if (autopath) { /* SET RECEIVE PATHNAMES AUTO fixup */
  2657. fnrpath = PATH_AUTO;
  2658. autopath = 0;
  2659.     }
  2660.     return(0);
  2661. }
  2662. VOID
  2663. pktinit() { /* Initialize packet sequence */
  2664.     pktnum = 0; /* number & window low. */
  2665.     winlo = 0;
  2666.     debug(F101,"pktinit winlo","",winlo);
  2667. }
  2668. /*  R I N I T  --  Respond to S or I packet  */
  2669. VOID
  2670. rinit(d) CHAR *d; {
  2671.     char *tp;
  2672.     ztime(&tp);
  2673.     tlog(F110,"Transaction begins",tp,0L); /* Make transaction log entry */
  2674.     tlog(F110,"Global file mode:", binary ? "binary" : "text", 0L);
  2675.     tlog(F110,"Collision action:", fncnam[fncact],0);
  2676.     tlog(F100,"","",0);
  2677.     debug(F101,"rinit fncact","",fncact);
  2678.     filcnt = filrej = 0; /* Init file counters */
  2679.     spar(d);
  2680.     ack1(rpar());
  2681. #ifdef datageneral
  2682.     if ((local) && (!quiet))            /* Only do this if local & not quiet */
  2683.         consta_mt();                    /* Start the asynch read task */
  2684. #endif /* datageneral */
  2685. }
  2686. /*  R E S E T C  --  Reset per-transaction character counters */
  2687. VOID
  2688. resetc() {
  2689.     rptn = 0; /* Repeat counts */
  2690.     fsecs = flci = flco = 0L; /* File chars in and out */
  2691. #ifdef GFTIMER
  2692.     fpfsecs = 0.0;
  2693. #endif /* GFTIMER */
  2694.     tfc = tlci = tlco = 0L; /* Total file, line chars in & out */
  2695.     ccu = ccp = 0L; /* Control-char statistics */
  2696. #ifdef COMMENT
  2697.     fsize = -1L; /* File size */
  2698. #else
  2699.     if (what != W_SEND)
  2700.       fsize = -1L;
  2701.     debug(F101,"resetc fsize","",fsize);
  2702. #endif /* COMMENT */
  2703.     timeouts = retrans = 0; /* Timeouts, retransmissions */
  2704.     spackets = rpackets = 0; /* Packet counts out & in */
  2705.     crunched = 0; /* Crunched packets */
  2706.     wcur = 0; /* Current window size */
  2707.     wmax = 0; /* Maximum window size used */
  2708.     peakcps = 0;                        /* Peak chars per second */
  2709. }
  2710. /*  S I N I T  --  Get & verify first file name, then send Send-Init packet */
  2711. /*
  2712.  Returns:
  2713.    1 if send operation begins successfully
  2714.    0 if send operation fails
  2715. */
  2716. #ifdef DYNAMIC
  2717. char *cmargbuf = NULL;
  2718. #else
  2719. char cmargbuf[CKMAXPATH+1];
  2720. #endif /* DYNAMIC */
  2721. char *cmargp[2];
  2722. VOID
  2723. fnlist() {
  2724.     if (!calibrate)
  2725.       sndsrc = (nfils < 0) ? -1 : nfils; /* Source for filenames */
  2726. #ifdef DYNAMIC
  2727.     if (!cmargbuf && !(cmargbuf = malloc(CKMAXPATH+1)))
  2728.       fatal("fnlist: no memory for cmargbuf");
  2729. #endif /* DYNAMIC */
  2730.     cmargbuf[0] = NUL; /* Initialize name buffer */
  2731.     debug(F101,"fnlist nfils","",nfils);
  2732.     debug(F110,"fnlist cmarg",cmarg,0);
  2733.     debug(F110,"fnlist cmarg2",cmarg2,0);
  2734.     if (!cmarg2) cmarg2 = "";
  2735.     if (nfils == 0) { /* Sending from stdin or memory. */
  2736. if ((cmarg2 != NULL) && (*cmarg2)) {
  2737.     cmarg = cmarg2; /* If F packet, "as-name" is used */
  2738.     cmarg2 = ""; /* if provided */
  2739. } else
  2740.   cmarg = "stdin"; /* otherwise just use "stdin" */
  2741. ckstrncpy(cmargbuf,cmarg,CKMAXPATH+1);
  2742. cmargp[0] = cmargbuf;
  2743. cmargp[1] = "";
  2744. cmlist = cmargp;
  2745. nfils = 1;
  2746.     }
  2747. }
  2748. int
  2749. sinit() {
  2750.     int x; /* Worker int */
  2751.     char *tp, *xp, *m; /* Worker string pointers */
  2752.     filcnt = filrej = 0; /* Initialize file counters */
  2753.     fnlist();
  2754.     xp = "";
  2755.     if (nfils < 0) {
  2756. #ifdef PIPESEND
  2757. if (usepipes && protocol == PROTO_K && *cmarg == '!') {
  2758.     pipesend = 1;
  2759.     cmarg++;
  2760. }
  2761. #endif /* PIPESEND */
  2762. xp = cmarg;
  2763.     } else {
  2764. #ifndef NOMSEND
  2765. if (addlist)
  2766.   xp = filehead->fl_name;
  2767. else
  2768. #endif /* NOMSEND */
  2769.   if (filefile)
  2770.     xp = filefile;
  2771.   else if (calibrate)
  2772.     xp = "Calibration";
  2773.   else
  2774.     xp = *cmlist;
  2775.     }
  2776.     debug(F110,"sinit xp",xp,0);
  2777.     x = gnfile(); /* Get first filename. */
  2778.     m = NULL; /* Error message pointer */
  2779.     debug(F101,"sinit gnfil","",x);
  2780.     switch (x) {
  2781.       case -5: m = "Too many files match wildcard"; break;
  2782.       case -4: m = "Cancelled"; break;
  2783.       case -3: m = "Read access denied"; break;
  2784.       case -2: m = "File is not readable"; break;
  2785. #ifdef COMMENT
  2786.       case -1: m = iswild(filnam) ? "No files match" : "File not found";
  2787. break;
  2788.       case  0: m = "No filespec given!"; break;
  2789. #else
  2790.       case  0:
  2791.       case -1: m = iswild(filnam) ? "No files match" : "File not found";
  2792. break;
  2793. #endif /* COMMENT */
  2794.       default:
  2795. break;
  2796.     }
  2797.     debug(F101,"sinit nfils","",nfils);
  2798.     debug(F110,"sinit filnam",filnam,0);
  2799.     if (x < 1) { /* Didn't get a file. */
  2800. if (server) /* Doing GET command */
  2801.   errpkt((CHAR *)m); /* so send Error packet. */
  2802. else /* Doing SEND command */
  2803.   xxscreen(SCR_EM,0,0l,m); /* so print message. */
  2804. tlog(F110,xp,m,0L); /* Make transaction log entry. */
  2805. freerbuf(rseqtbl[0]); /* Free the buffer the GET came in. */
  2806. return(0); /* Return failure code */
  2807.     }
  2808.     if (!local && !server && ckdelay > 0) /* OS-9 sleep(0) == infinite */
  2809.       sleep(ckdelay); /* Delay if requested */
  2810. #ifdef datageneral
  2811.     if ((local) && (!quiet))            /* Only do this if local & not quiet */
  2812.         consta_mt();                    /* Start the asynch read task */
  2813. #endif /* datageneral */
  2814.     freerbuf(rseqtbl[0]); /* Free the buffer the GET came in. */
  2815.     sipkt('S'); /* Send the Send-Init packet. */
  2816.     ztime(&tp); /* Get current date/time */
  2817.     tlog(F110,"Transaction begins",tp,0L); /* Make transaction log entry */
  2818.     tlog(F110,"Global file mode:", binary ? "binary" : "text", 0L);
  2819.     tlog(F100,"","",0);
  2820.     debug(F111,"sinit ok",filnam,0);
  2821.     return(1);
  2822. }
  2823. int
  2824. #ifdef CK_ANSIC
  2825. sipkt(char c) /* Send S or I packet. */
  2826. #else
  2827. sipkt(c) char c;
  2828. #endif
  2829. /* sipkt */ {
  2830.     CHAR *rp; int k, x;
  2831.     debug(F101,"sipkt pktnum","",pktnum);
  2832.     k = sseqtbl[pktnum]; /* Find slot for this packet */
  2833.     debug(F101,"sipkt k","",k);
  2834.     if (k < 0) { /* No slot? */
  2835. k = getsbuf(winlo = pktnum); /* Make one. */
  2836. debug(F101,"sipkt getsbuf","",k);
  2837.     }
  2838.     ttflui(); /* Flush pending input. */
  2839.     rp = rpar(); /* Get protocol parameters. */
  2840.     x = spack(c,pktnum,(int)strlen((char *)rp),rp); /* Send them. */
  2841.     return(x);
  2842. }
  2843. /*  X S I N I T  --  Retransmit S-packet  */
  2844. /*
  2845.   For use in the GET-SEND sequence, when we start to send, but receive another
  2846.   copy of the GET command because the receiver didn't get our S packet.
  2847.   This retransmits the S packet and frees the receive buffer for the ACK.
  2848.   This special case is necessary because packet number zero is being re-used.
  2849. */
  2850. VOID
  2851. xsinit() {
  2852.     int k;
  2853.     k = rseqtbl[0];
  2854.     debug(F101,"xsinit k","",k);
  2855.     if (k > -1)
  2856.     freerbuf(k);
  2857.     resend(0);
  2858. }
  2859. /*  R C V F I L -- Receive a file  */
  2860. /*
  2861.   Incoming filename is in data field of F packet.
  2862.   This function decodes it into the srvcmd buffer, substituting an
  2863.   alternate "as-name", if one was given.
  2864.   Then it does any requested transformations (like converting to
  2865.   lowercase), and finally if a file of the same name already exists,
  2866.   takes the desired collision action.
  2867.   Returns:
  2868.     1 on success.
  2869.     0 on failure.
  2870. */
  2871. char ofn1[CKMAXPATH+4]; /* Buffer for output file name */
  2872. char * ofn2; /* Pointer to backup file name */
  2873. int ofn1x; /* Flag output file already exists */
  2874. int opnerr; /* Flag for open error */
  2875. int /* Returns success ? 1 : 0 */
  2876. rcvfil(n) char *n; {
  2877.     extern int en_cwd;
  2878.     extern char * rcvexcept[];
  2879.     int i, skipthis;
  2880.     char * n2;
  2881. #ifdef OS2ONLY
  2882.     char *zs, *longname, *newlongname, *pn; /* OS/2 long name items */
  2883. #endif /* OS2ONLY */
  2884. #ifdef DTILDE
  2885.     char *dirp;
  2886. #endif /* DTILDE */
  2887.     int dirflg, x, y;
  2888. #ifdef PIPESEND
  2889.     extern char * rcvfilter;
  2890. #endif /* PIPESEND */
  2891.     extern char * rrfspec;
  2892. #ifdef CALIBRATE
  2893.     extern int dest;
  2894.     int csave;
  2895.     csave = calibrate; /* So we can decode filename */
  2896.     calibrate = 0;
  2897. #endif /* CALIBRATE */
  2898.     ofperms = ""; /* Reset old-file permissions */
  2899.     opnerr = 0; /* No open error (yet) */
  2900.     ofn2 = NULL; /* No new name (yet) */
  2901.     lsstate = 0; /* Cancel locking-shift state */
  2902.     srvptr = srvcmd; /* Decode file name from packet. */
  2903. #ifdef UNICODE
  2904.     xpnbyte(-1,0,0,NULL); /* Reset UCS-2 byte counter. */
  2905. #endif /* UNICODE */
  2906.     debug(F110,"rcvfil rdatap",rdatap,0);
  2907.     decode(rdatap,putsrv,0); /* Don't xlate charsets. */
  2908. #ifdef CALIBRATE
  2909.     calibrate = csave;
  2910.     if (dest == DEST_N) {
  2911. calibrate = 1;
  2912. cmarg2 = "CALIBRATE";
  2913.     }
  2914. #endif /* CALIBRATE */
  2915.     if (*srvcmd == '') /* Watch out for null F packet. */
  2916.       strcpy((char *)srvcmd,"NONAME");
  2917.     makestr(&rrfspec,(char *)srvcmd);
  2918. #ifdef DTILDE
  2919.     if (*srvcmd == '~') {
  2920. dirp = tilde_expand((char *)srvcmd); /* Expand tilde, if any. */
  2921. if (*dirp != '') strcpy((char *)srvcmd,dirp);
  2922.     }
  2923. #else
  2924. #ifdef OS2
  2925.     if (isalpha(*srvcmd) && srvcmd[1] == ':' && srvcmd[2] == '')
  2926.       strcat((char *)srvcmd,"NONAME");
  2927. #endif /* OS2 */
  2928. #endif /* DTILDE */
  2929.     if (!ENABLED(en_cwd)) { /* CD is disabled */
  2930. zstrip((char *)(srvcmd+2),&n2); /* and they included a pathname, */
  2931. if (strcmp((char *)(srvcmd+2),n2)) { /* so refuse. */
  2932.     rf_err = "Access denied";
  2933.     return(0);
  2934. }
  2935.     }
  2936. #ifdef COMMENT
  2937.     /* Wrong place for this -- handle cmarg2 first -- see below...  */
  2938.     if (zchko((char *)srvcmd) < 0) { /* Precheck for write access */
  2939. debug(F110,"rcvfil access denied",srvcmd,0);
  2940. rf_err = "Write access denied";
  2941. discard = opnerr = 1;
  2942. return(0);
  2943.     }
  2944.     xxscreen(SCR_FN,0,0l,(char *)srvcmd); /* Put it on screen if local */
  2945.     debug(F110,"rcvfil srvcmd 1",srvcmd,0);
  2946.     tlog(F110,"Receiving",(char *)srvcmd,0L); /* Transaction log entry */
  2947. #endif /* COMMENT */
  2948.     skipthis = 0; /* This file in our exception list? */
  2949.     for (i = 0; i < 8; i++) {
  2950. if (!rcvexcept[i]) {
  2951.     break;
  2952. }
  2953. if (ckmatch(rcvexcept[i],(char *)srvcmd,filecase,1)) {
  2954.     skipthis = 1;
  2955.     break;
  2956. }
  2957.     }
  2958. #ifdef DEBUG
  2959.     if (deblog && skipthis) {
  2960. debug(F111,"rcvfil rcvexcept",rcvexcept[i],i);
  2961. debug(F110,"rcvfil skipping",srvcmd,0);
  2962.     }
  2963. #endif /* DEBUG */
  2964.     if (skipthis) { /* Skipping this file */
  2965. discard = 1;
  2966. rejection = 1;
  2967. rf_err = "Exception list";
  2968. debug(F101,"rcvfil discard","",discard);
  2969. tlog(F100," refused: exception list","",0);
  2970. return(1);
  2971.     }
  2972.     /* File is not in exception list */
  2973.     if (!cmarg2) /* No core dumps please */
  2974.       cmarg2 = "";
  2975.     debug(F110,"rcvfil cmarg2",cmarg2,0);
  2976.     if (*cmarg2) { /* Check for alternate name */
  2977. #ifndef NOSPL
  2978. int y; char *s; /* Pass it thru the evaluator */
  2979. extern int cmd_quoting;
  2980. if (cmd_quoting) {
  2981.     y = MAXRP;
  2982.     ckstrncpy(ofn1,(char *)srvcmd,CKMAXPATH+1); /* for v(filename) */
  2983.     s = (char *)srvcmd;
  2984.     zzstring(cmarg2,&s,&y);
  2985. } else
  2986.   *srvcmd = NUL;
  2987. if (!*srvcmd) /* If we got something */
  2988. #endif /* NOSPL */
  2989.   strcpy((char *)srvcmd,cmarg2);
  2990.     }
  2991.     debug(F110,"rcvfil srvcmd 2",srvcmd,0);
  2992. #ifdef PIPESEND
  2993.     /* If it starts with "bang", it's a pipe, not a file. */
  2994.     if (usepipes && protocol == PROTO_K && *srvcmd == '!' && !rcvfilter) {
  2995. CHAR *s;
  2996. s = srvcmd+1; /* srvcmd[] is not a pointer. */
  2997. while (*s) { /* So we have to slide the contents */
  2998.     *(s-1) = *s; /* over 1 space to the left. */
  2999.     s++;
  3000. }
  3001. *(s-1) = NUL;
  3002. pipesend = 1;
  3003.     }
  3004. #endif /* PIPESEND */
  3005. #ifdef COMMENT
  3006. /*
  3007.   This is commented out because we need to know whether the name we are
  3008.   using was specified by the local user as an override, or came from the
  3009.   incoming packet.  In the former case, we don't do stuff to it (like
  3010.   strip the pathname) that we might do to it in the latter.
  3011. */
  3012.     cmarg2 = ""; /* Done with alternate name */
  3013. #endif /* COMMENT */
  3014.     if ((int)strlen((char *)srvcmd) > CKMAXPATH) /* Watch out for overflow */
  3015.       *(srvcmd + CKMAXPATH - 1) = NUL;
  3016.     /* At this point, srvcmd[] contains the incoming filename or as-name. */
  3017.     /* So NOW we check for write access. */
  3018.     if (zchko((char *)srvcmd) < 0) { /* Precheck for write access */
  3019. debug(F110,"rcvfil access denied",srvcmd,0);
  3020. rf_err = "Write access denied";
  3021. discard = opnerr = 1;
  3022. return(0);
  3023.     }
  3024.     xxscreen(SCR_FN,0,0l,(char *)srvcmd); /* Put it on screen if local */
  3025.     debug(F110,"rcvfil srvcmd 1",srvcmd,0);
  3026.     tlog(F110,"Receiving",(char *)srvcmd,0L); /* Transaction log entry */
  3027. #ifdef CK_LABELED
  3028. #ifdef VMS
  3029. /*
  3030.   If we have an as-name, this overrides the internal name if we are doing
  3031.   a labeled-mode transfer.
  3032. */
  3033.     if (*cmarg2) {
  3034. extern int lf_opts;
  3035. lf_opts &= ~LBL_NAM;