home *** CD-ROM | disk | FTP | other *** search
/ Amiga ISO Collection / AmigaUtilCD1.iso / Fax / AVMA&GPFax-V1,33Sources.LHA / efax.c < prev    next >
Encoding:
C/C++ Source or Header  |  1994-06-19  |  64.3 KB  |  2,184 lines

  1. #define Copyright         "Copyright 1994  Ed Casas"
  2.  
  3. #define Version          "efax v 0.6b"
  4.  
  5. #define EFAXVER "1.04"
  6. static char* versionString = "$VER: EFax (Amiga) " EFAXVER;
  7.  
  8. /*
  9.     Copyright (C) 1994  Ed Casas
  10.  
  11.     This program is free software; you can redistribute it and/or modify
  12.     it under the terms of the GNU General Public License as published by
  13.     the Free Software Foundation; either version 2 of the License, or
  14.     (at your option) any later version.
  15.  
  16.     This program is distributed in the hope that it will be useful,
  17.     but WITHOUT ANY WARRANTY; without even the implied warranty of
  18.     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  19.     GNU General Public License for more details.
  20.  
  21.     You should have received a copy of the GNU General Public License
  22.     along with this program; if not, write to the Free Software
  23.     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  24.  
  25.     You may contact the author by e-mail at: edc@ee.ubc.ca,
  26.     or by mail at: 2629 West 3rd Ave, Vancouver, BC, Canada,
  27.     V6K 1M4.
  28.  
  29. */
  30.  
  31. const char *Usage =
  32.   "Usage:\n"
  33.   "  %s [ option ]... [ -r pat | -t num file... ]\n"
  34. "Options:\n"
  35.   "  -p portname load config from avm:servers.cfg\n"
  36.   "  -c cap  set file format or receive capabilites to cap\n"
  37.   "  -d dev  use modem on device dev\n"
  38.   "  -u unit unit number for device\n"
  39.   "  -g cmd  exec \"/bin/sh -c cmd\" for data calls\n"
  40.   "  -i str  send modem command ATstr at start\n"
  41.   "  -l id   set local indetification to id\n"
  42.   "  -o opt  use protocol option opt:\n"
  43.   "      1     use class 1 modem commands\n"
  44.   "      2     use class 2 modem commands\n"
  45.   "      a     if first [data mode] answer attempt fails retry as fax\n"
  46.   "      e     ignore errors in modem initialization commands\n"
  47.   "      r     reverse bit order on receive\n"  
  48.   "      t     reverse bit order on transmit\n"  
  49.   "      x     use XON instead of DC2 to trigger reception\n"
  50.   "      z     add 100 ms to pause before each modem comand (cumulative)\n"
  51.   "  -q ne   ask for retransmission if more than ne errors per page\n"
  52.   "  -s      share (unlock) modem device while waiting for call\n"
  53.   "  -v lvl  print messages of type in string lvl (ewinahcm)\n"
  54.   "  -w      don't answer phone, wait for OK or CONNECT instead\n"
  55.   "  -x fil  use uucp-style lock file fil\n"
  56.   "  -z str  send modem command ATstr when done\n"
  57.   "Commands:\n"
  58.   "  -r      answer and receive fax into files pat.001, pat.002, ... \n"
  59.   "  -t      send fax image files file... to telephone num\n"
  60.   ;
  61.  
  62. #include <ctype.h>        /* ANSI C */
  63. #include <errno.h>
  64. #include <signal.h>    
  65. #include <stdarg.h> 
  66. #include <stdio.h>
  67. #include <stdlib.h>
  68. #include <string.h>
  69. #include <time.h>
  70.  
  71. #include <fcntl.h>        /* UNIX */
  72. #include <sys/types.h>
  73.  
  74. #include <sys/stat.h>        /* required for SYSV (?) */
  75.  
  76. #include "voice.h"
  77. #include "callback_proto.h"
  78. #include "config_proto.h"
  79. #include "faxfile.h"
  80. #include "simplerexx.h"
  81.  
  82. extern long __stack = 10000L; /* max. stack size */
  83.  
  84. extern struct CallBack globalCallBack;
  85. extern int globalAborted;
  86.  
  87. AREXXCONTEXT OutgoingRexx;
  88.  
  89. void setSendSerialParms(void);
  90. void setReceiveSerialParms(void);
  91.  
  92. int verbose = 0;
  93.  
  94. struct faxout
  95. {
  96.     FILE           *fp;
  97.     int        fax_byte;
  98.     int        fax_weight;
  99.     int        pages;
  100.     int        raw;    /* 1: no header, 2: no EndOfPage */
  101.     char           *stdiobuf;
  102. };
  103.  
  104. struct faxout* FaxHandle = 0;
  105.  
  106. typedef struct faxin {
  107.     int     raw;
  108.     FAXHDR     faxhdr;
  109.     FAXHDR     pagehdr;
  110.     int     row;
  111.     int     endoffile;
  112.     int     eols;
  113.     int     rawzeros;
  114.     int     shdata;
  115.     int     shbit;
  116.     int     kludge;
  117.     int     reversebits;
  118.     int     stretch;
  119.     unsigned char  *bitp;
  120.     unsigned char    bitval;
  121.  
  122. } FAXIN;
  123.  
  124. extern FAXIN faxin;
  125.  
  126. /* prototypes */
  127. void releaseResources(void);
  128. void allocateResources(void);
  129. enum ReturnStatus waitForTimeout(int seconds, int microseconds, struct CallBack* cb);
  130. enum ReturnStatus readChar(char* c, int timeout, struct CallBack* cb);
  131. int writeToSerial(char* s, int len);
  132. void unReadChar(void);
  133. void mainError(char*);
  134.  
  135. extern char serialName[];
  136. extern int unitNumber;
  137. int dialTone = 1;
  138. int faxHardwareHandshaking = 1;
  139.  
  140. char portNameToLoad[40];
  141.  
  142. #ifndef FILENAME_MAX
  143. #define FILENAME_MAX 255
  144. #endif
  145.  
  146. #ifndef u_char
  147. #define u_char unsigned char
  148. #endif
  149.  
  150. /* constants... */
  151.  
  152. #define FAXFILE "serial.device"  /* default fax modem device */
  153. #define LOGF     stderr     /* session log written to this stream */
  154.  
  155.                 /* delays/timeouts, in deciseconds */
  156. #define TO_RESET  26        /* timeout for modem reset commands (per Hayes) */
  157. #define T_CMD     1        /* pause before each modem command */
  158. #define TO_A      1200        /* dial/answer (Phase A) - modem may T.O. first */
  159.  
  160. #define TO_DATAF  150        /* software adaptive answer data connect */
  161.  
  162. #define T1 350            /* T.30 T1 - waiting for DIS/DCS before Phase B */
  163. #define T2 60            /* T.30 T2 - waiting for frame in Phase B */
  164. #define T3S 30            /* T.30 response timeout (not T3) */
  165. #define T4 30            /* T.30 T4 - between [re]transmissions of DIS */
  166.  
  167. #define TO_DRAIN 140        /* drain 4k modem buffer at 2400bps (tx) */
  168. #define TO_RTCMD 10        /* return to command mode after DLE-ETX (rx) */
  169. #define TO_FT    31        /* max delay after +F[TR][MH] command */
  170. #define TO_CHAR  51        /* per data character (max FILL length) */
  171. #define TO_ABRT  20        /* max delay after sending abort sequence */
  172.  
  173. #define TO_C2B    450        /* Class 2 DIS to CONNECT:(DCS+TCF+CFR)xretries */
  174. #define TO_C2X    20        /* Class 2 wait for XON: 2/5 of 5s timeout */
  175. #define TO_C2PP   200        /* Class 2 wait for ppr: (ppm+ppr)x3retries + 2 */
  176. #define TO_C2R    600        /* Class 2 receive: (TCF+FTT)x11 retrains + 5 */
  177. #define TO_C2EOR  60        /* Class 2 end of data rx (2 retrans x 3 s) */
  178.  
  179.  
  180. #define CMDBUFSIZE 128      /* longest possible modem command or response */
  181. #define DEFDISLEN 3        /* length of DIS initially transmitted */
  182. #define DEFCAP 1,3,0,2,0,0,0,0    /* default local capabilities */
  183. #define DLE_ETX "\020\003"  /* DLE-ETX (end of data) string */
  184. #define FNAMFT "%m%d%H%M%S" /* strftime() format for default file name */
  185. #define HDBLKFLAG '#'        /* prefix to force HDB (text) lock file style */
  186. #define IDLEN 20        /* length of T.30 identification strings */
  187. #define LOCKPOLLDELAY 15    /* seconds between checks of lock files */
  188. #define MAXDIS 8        /* maximum DIS frames sent without response (T1) */
  189. #define MAXFIFLEN 125        /* max FIF len = MAXFRLEN - (adx+ctl+FCF) - FCS */
  190. #define MAXFRLEN 130        /* max frame length = 3.45s x 300 bps / 8 */
  191. #define MAXGETTY 512        /* maximum length of ``getty'' (-g) command */
  192. #define MAXIOPT  100        /* maximum # of modem initialization commands */
  193. #define MAXLKFILE 16        /* maximum number of lock files */
  194. #define MAXMSGBUF 8192        /* maximum size of message buffer */
  195. #define MAXNULLS 2        /* maximum consecutive received nulls saved */
  196. #define MAXTSTAMP 80        /* maximum length of a time stamp */
  197. #define MAXTRAINERR 0        /* maximum errors allowed in training check data */
  198. #define MAXTRAIN 2        /* maximum training retries at lowest speed */
  199. #define MAXRETRY 3        /* maximum retries of unacknowledged commands */
  200. #define MINWRITE  128       /* minimum bytes per write() to modem */
  201. #define NCAP 8              /* number of fields in a capability string */
  202. #define NTXRETRY  3        /* maximum re-sends per page */
  203. #define RCVBUFSIZE 1024        /* read up to this many bytes at a time from fax */
  204. #define SNDBUFSIZE 1024        /* maximum bytes to write at a time to fax */
  205. #define T4RTCLEN  9        /* end of page mark for T.4 coded images */
  206. #define T4RTC    "\000\020\001\000\020\001\000\020\001"
  207.  
  208. enum  cchar {
  209.   NUL, SOH, STX, ETX, EOT, ENQ, ACK, BEL, BS,  HT,  LF,
  210.   VT,  FF,  CR,  SO,  SI,  DLE, XON, DC2, XOFF,DC4, NAK,
  211.   SYN, ETB, CAN, EM,  SUB, ESC, FS,  GS,  RS,  US } ;
  212.  
  213. const char *prompts[] = {        /* modem responses that are prompts */
  214.   "OOK", "-CONNECT FAX", "CCONNECT", "NNO CARRIER", "EERROR",
  215.   "NNO DIALTONE", "BBUSY", "NNO ANSWER", "E+FCERROR", 0 } ;
  216.  
  217. enum promptcodes {            /* codes for modem prompts */
  218.    BUSY = 'B', CONNECT = 'C', OK = 'O', RING = 'R', NO = 'N',
  219.    ERROR = 'E' } ;
  220.  
  221. enum ttymodes                /* serial port modes */
  222.    { COMMAND, DROPDTR, SEND, ORIGINAL } ;
  223.  
  224.                 /* signals to be caught so can hang up phone */
  225. const int catch [] = { /*SIGHUP,*/ SIGINT, /*SIGQUIT, SIGIOT, SIGSEGV, SIGALRM,
  226.                SIGTERM,*/ SIGFPE, 0 } ;
  227.  
  228. typedef int cap [ NCAP ] ;        /* remote/local capabilities */
  229.  
  230.                                         /* capability fields... */
  231. enum  captype {                   VR, BR, WD, LN, DF, EC, BF, ST } ;
  232. const int capmax [ NCAP ] = {   1,  7,  3,  2,  3,  2,  1,  7 } ;
  233.                     /* & maximum values */
  234.  
  235.                     /* characters per second for br */
  236. const int cps [ 8 ] = { 300, 600, 900, 1200, 1500, 1800, 900, 1200 } ;
  237.  
  238.                     /* next br = fallback [ br ] */
  239. const int fallback [ 8 ] = { 0, 0, 1, 2, 7, 4, 3, 6 } ;
  240.  
  241.                     /* minimum scan time in ms  */
  242. const int delay [ 8 ] = { 0 , 5, 10, 10, 20, 20, 40, 40 } ;
  243.  
  244. /* Table to convert between T.30 DIS/DCS/DTC FIF and Class 2-like
  245.    capability codes. Uses br=6, 7 for V.17 at 7200, 9600. */
  246.  
  247. typedef const struct t30tabstruct
  248.   char *name ; 
  249.   u_char byte, shift, mask ; 
  250.   u_char captodis[8], distocap[16], captodcs[8], dcstocap[16] ; 
  251. } t30tabst ;
  252.  
  253. #define X 0xff                /* invalid values */
  254.  
  255. t30tabst t30tab [ NCAP ] = {
  256.   { "vr", 1, 1, 0x01, { 0, 1 } , { 0, 1 } , { 0, 1 } , { 0, 1 } },     
  257.   { "br", 1, 2, 0x0f, 
  258.       { 0, 4, 12, 12, 13, 13 } ,
  259.       { 0, X, X, X, 1, X, X, X, 3, X, X, X, 3, 5, 3, X } ,
  260.       { 0, 4, 12, 8, 5, 1 } ,
  261.       { 0, 5, 5, X, 1, 4, 4, X, 3, 7, X, X, 2, 6, X, X } } ,
  262.   { "wd", 2, 6, 0x03, { 0, 2, 1 } , { 0, 2, 1, 2 } ,
  263.       { 0, 2, 1 } , { 0, 2, 1, 2 } },
  264.   { "ln", 2, 4, 0x03, { 0, 2, 1 } , { 0, 2, 1, X } ,
  265.       { 0, 2, 1 } , { 0, 2, 1, X } },
  266.   { "df", 1, 0, 0x01, { 0, 1 } , { 0, 1 } , { 0, 1 } , { 0, 1 } },
  267.   { "ec", 3, 4, 0x03, { 0, 2, 2 } , { 0, X, 2, X } , 
  268.       { 0, 3, 2 } , { 0, 0, 2, 1 } }, 
  269.   { "bf", 5, 5, 0x01, { 0, 1 } , { 0, 1 } , { 0, 1 } , { 0, 1 } },
  270.   { "st", 2, 1, 0x07, 
  271.       { 7, 4, 3, 2, 6, 0, 5, 1 } , { 5, 7, 3, 2, 1, 6, 4, 0 } ,
  272.       { 7, 4, X, 2, X, 0, X, 1 } , { 5, 7, 3, 1, X, X, X, 0 } } 
  273. } ;
  274.  
  275.                     /* values of capability fields */
  276. const char *capvaluestr [ NCAP ] [8] = {
  277.   { " 98lpi", "196lpi" } , 
  278.   { " 2400bps", " 4800bps", " 7200bps", " 9600bps", "  12kbps", "14.4kbps",
  279.     "7200bpsV.17", "9600bpsV.17" } ,
  280.   { "8.5\"/215mm", " 10\"/255mm", " 12\"/303mm" } ,
  281.   { "11\"/A4", "14\"/B4", " any  " } ,
  282.   { "1D" , "2D" }, { "   -   ", "ECM-256", "ECM-64 " }, { " - ", "BFT" },
  283.   { "0ms", "5ms", "10/5ms", "10ms", "20/10ms", "20ms", "40/20ms", "40ms" }
  284. } ;
  285.  
  286. /* T.30 control frames */
  287.  
  288. enum frametype {        
  289.  DIS=0x01, CSI,    NSF=0x04,
  290.  CFR=0x21, FTT,
  291.  MCF=0x31, RTN, RTP, PIN, PIP,
  292.  DCS=0x41, TSI,    NSS=0x44,
  293.  CRP=0x58, DCN=0x5f,
  294.  EOM=0x71, MPS, EOP=0x074, PRI_EOM=0x79, PRI_MPS, PRI_EOP=0x7c,
  295.  DTC=0x80, CIG, NSC=0x84
  296.  } ;
  297.  
  298. /* Class 1 commands to [receive=0/transmit=1] [data=0/training=1] for
  299.    [baud rate=BR]. */
  300.  
  301. const char *c1cmd [ 2 ]  [ 2 ] [ 8 ] = { 
  302. { { "+FRM=24", "+FRM=48", "+FRM=72", "+FRM=96", "+FRM=122", "+FRM=146" ,
  303.     "+FRM=74", "+FRM=98" } ,
  304.   { "+FRM=24", "+FRM=48", "+FRM=72", "+FRM=96", "+FRM=121", "+FRM=145" ,
  305.     "+FRM=73", "+FRM=97" } } ,
  306. { { "+FTM=24", "+FTM=48", "+FTM=72", "+FTM=96", "+FTM=122", "+FTM=146" ,
  307.     "+FTM=74", "+FTM=98", } ,
  308.   { "+FTM=24", "+FTM=48", "+FTM=72", "+FTM=96", "+FTM=121", "+FTM=145" ,
  309.     "+FTM=73", "+FTM=97" } }
  310. } ;
  311.  
  312.  
  313. /* World's fastest T.4 decoder.  Implements a tree search.  Each bit of
  314.    each byte is tested using the T4TST(byte,bitmask) macro. This macro
  315.    invokes the macro T4CODE(len) when it detects a valid T.4 code or on the
  316.    first EOL after an invalid code.  The macro's parameter len will be the
  317.    run length, -1 on a normal EOL or -2 for the EOL following an error.  1D
  318.    coding only.
  319.  
  320.    As per T.4, run lengths >63 (make-up codes) should be added to the
  321.    subsequent (terminating) code.  The first run of a line is white and
  322.    colors alternate.
  323.  
  324.    Each node of the tree contains pointers to the next node for a '0' or a
  325.    '1' bit.  NULL pointers indicate terminal nodes in which case, the run
  326.    length (or -1 or -2) is given by the zlen or olen member and the search
  327.    continues with the node given by the znext or onext pointer.
  328.  
  329.    The decoding tree, stored in packed format (LSB to MSB plus guard bit),
  330.    is built by calling mktree() which returns a pointer to the initial
  331.    value of 't4p' to be used by the T4TST macro.
  332.  
  333. */
  334.  
  335. typedef struct dtree { 
  336.   struct dtree *zero, *one, *onext, *znext ; 
  337.   short int zlen, olen ;
  338. } dtree ;
  339.  
  340. #define T4TST(c,m) t4p = ( c & m ) ? \
  341. ( t4p->one  ? t4p->one  : ( T4CODE ( t4p->olen ), t4p->onext ) ) : \
  342. ( t4p->zero ? t4p->zero : ( T4CODE ( t4p->zlen ), t4p->znext ) ) ;
  343.  
  344. #define T4CTST(c) T4TST(c,0x80) ;  T4TST(c,0x40) ; \
  345.                   T4TST(c,0x20) ;  T4TST(c,0x10) ; \
  346.                   T4TST(c,0x08) ;  T4TST(c,0x04) ; \
  347.                   T4TST(c,0x02) ;  T4TST(c,0x01) ;
  348.  
  349. #define T4NODES 228
  350.  
  351. const short int pkdt4tree [] = {
  352.  
  353. 7,1,29,53,243,263,9,1,11,1,13,1,15,1,17,1,19,1,21,1,23,1,25,1,  
  354. 27,1,27,2,-2,31,49,43,33,75,35,37,4,10,121,39,41,173,4,63,4,0,  
  355. 79,45,95,47,93,4,1,71,51,167,4,2,55,61,57,59,4,3,67,69,4,4,63,65,  
  356. 4,5,83,4,6,4,7,2,128,4,8,4,9,87,73,109,4,11,89,77,99,4,12,117,  
  357. 103,81,107,4,13,85,2,64,4,14,4,15,4,16,4,17,119,91,203,4,18,4,19,127,  
  358. 97,131,4,20,129,137,101,143,4,21,123,105,145,4,22,4,23,147,111,155,  
  359. 113,115,4,24,149,151,4,25,153,4,26,4,27,163,4,28,165,209,125,  
  360. 4,29,4,30,4,31,4,32,4,33,4,34,133,135,4,35,4,36,4,37,4,38,139,141,  
  361. 4,39,4,40,4,41,4,42,4,43,4,44,4,45,4,46,4,47,4,48,4,49,4,50,4,51,4,52,  
  362. 4,53,4,54,157,2,192,159,161,4,55,4,56,4,57,4,58,4,59,4,60,4,61,4,62,  
  363. 175,169,181,171,197,2,256,2,320,2,384,2,1664,177,179,185,2,448,2,512,  
  364. 183,191,2,576,189,187,2,640,2,704,2,768,2,832,2,896,193,195,2,960,2,1024,  
  365. 2,1088,2,1152,199,201,2,1216,2,1280,2,1344,2,1408,205,207,2,1472,2,1536,  
  366. 2,1600,2,1728,235,211,213,217,215,223,2,1792,221,219,229,2,1856,2,1920,  
  367. 2,1984,2,2048,225,227,2,2112,2,2176,2,2240,2,2304,231,233,2,2368,2,2432,  
  368. 2,2496,2,2560,237,1,239,1,241,1,241,2,-1,245,261,247,265,249,267,  
  369. 273,251,271,253,255,2,12,281,257,303,259,307,2,0,2,1,2,4,2,3,2,2,  
  370. 2,6,2,5,269,2,7,2,9,2,8,2,10,2,11,291,275,277,279,2,13,283,287,2,14,  
  371. 2,15,299,313,285,383,2,16,289,309,2,17,357,423,293,295,321,297,317,  
  372. 2,18,363,327,301,331,2,19,305,341,2,20,339,2,21,349,333,311,347,2,22,  
  373. 315,351,2,23,361,379,319,389,2,24,323,371,325,367,2,25,397,395,329,  
  374. 2,26,2,27,2,28,2,29,335,337,2,30,2,31,2,32,2,33,2,34,2,35,343,345,  
  375. 2,36,2,37,2,38,2,39,2,40,2,41,2,42,2,43,353,355,2,44,2,45,2,46,2,47,  
  376. 359,393,2,48,2,49,2,50,2,51,365,377,2,52,403,399,369,401,2,53,  
  377. 373,4,64,375,409,2,54,407,405,2,55,381,387,2,56,415,385,391,2,57,2,58,  
  378. 417,2,59,2,60,419,2,61,4,256,2,62,2,63,4,128,4,192,421,4,320,  
  379. 4,384,4,448,4,512,4,576,4,640,4,704,4,768,4,832,4,896,4,960,411,413,  
  380. 4,1024,4,1088,4,1152,4,1216,4,1280,4,1344,4,1408,4,1472,4,1536,4,1600,  
  381. 4,1664,4,1728,449,425,427,431,429,437,4,1792,435,433,443,4,1856,4,1920,  
  382. 4,1984,4,2048,439,441,4,2112,4,2176,4,2240,4,2304,445,447,4,2368,4,2432,  
  383. 4,2496,4,2560,451,1,453,1,455,1,455,2,-1,
  384. -1 } ;
  385.  
  386.  
  387. /* Program options. Not changed except in main(). */
  388.  
  389. cap local = { DEFCAP } ;      /* -c  local capabilities */
  390. char *argv0 = "?" ;          /*     program name (argv[0]) */
  391. char *fnamepat= FNAMFT ;      /* -r  pattern for received file names */
  392. char **fnames = 0 ;          /* -t  files to send */
  393. char *getty = "startbbs %d" ;          /* -g  command for data calls */
  394. char *iopt[ MAXIOPT ] ;          /* -i  modem init commands */
  395. char *zopt[ MAXIOPT ] ;          /* -z  modem reset commands */
  396. char *lkfiles [ MAXLKFILE+1 ] ;      /* -x  lock files  */
  397. char localid  [ IDLEN + 1 ] ;      /* -l  local ID (telephone #) */
  398. char *verb = "ewin" ;          /* -v  verbosity level */
  399. int c1 = 0 ;              /* -o1 use class 1 protocol */
  400. int softadans = 0 ;          /* -oa use software adaptive answer */
  401. int igniniterr=0 ;          /* -oe ignore initialization errors */
  402. int niopt=0 ;              /*     number of init commands */
  403. int nzopt=0 ;              /*     number of reset commands */
  404. int cmdpause = T_CMD ;          /* -oz delay before each init command */
  405. int maxpgerr = 10 ;          /* -q  maximum errors per page */
  406. int nfiles=0 ;              /*     number of files to send */
  407. int share = 0 ;              /* -s  share fax device */
  408. int waitforcall = 0 ;          /* -w  wait for incoming call */
  409. u_char startchar = DC2 ;      /* -ox character to start reception */
  410.  
  411.  
  412.  
  413. /* Bit ordering: serial devices transmit LS bit first.  T.4/T.30 says MS
  414.    bit is sent first. `Normal' order therefore reverses bit order. */
  415.  
  416. u_char                           /* bit reversal lookup tables*/
  417.    reversebits [ 256 ] , normalbits [ 256 ] , 
  418.    *rxbitorder = normalbits , *txbitorder = normalbits ;
  419.  
  420. dtree *t4tree ;                /* T.4 decoding tree */
  421.  
  422. /* fax stream variables (private). */
  423.  
  424. u_char faxobuf [ SNDBUFSIZE ] ,        /* fax output stream. */
  425.   *faxop=faxobuf , *faxoq=faxobuf+SNDBUFSIZE ;
  426.  
  427. int faxdev=-1 ;                /* fax device file descriptor. */
  428.  
  429. /* session state variables. */
  430.  
  431. FILE *file=0 ;                /* current image file  */
  432. cap remote ;                            /* remote capabilities  */ 
  433. cap session ;                        /* session capabilities */
  434. char remoteid [ IDLEN + 10 ] ;        /* ID (tel no) of remote station */
  435. int crate = 19200 ;                /* CONNECT rate */
  436. int datamode=0 ;            /* true if answered in data mode */
  437. int decimate = 0 ;            /* drop 1/2 lines to get 98 lpi */
  438. int good=1 ;                /* page received OK */
  439. int hsc=-1 ;                /* hangup status code (Class 2) */
  440. int minlen = 0 ;            /* pad lines to this many bytes */
  441.  
  442.  
  443. /* Functions... */
  444.  
  445. /* Print time stamp. */
  446.  
  447. void tstamp( void )
  448. {
  449.   static time_t last = 0 , now ;
  450.   char *fmt = 0, tbuf [ MAXTSTAMP ] ;
  451.  
  452.   now = time ( 0 ) ;
  453.  
  454.   if ( now - last >   0 ) fmt = " (%M:%S)" ;
  455.   if ( now - last > 600 ) fmt = " %c" ;
  456.  
  457.   if ( fmt ) {
  458.     strftime ( tbuf , MAXTSTAMP , fmt , localtime( &now ) ) ;
  459.     mainError(tbuf);
  460.     last = now ; 
  461.   }
  462. }
  463.  
  464.  
  465. /* For systems without strerror(3) */
  466.  
  467. char *strerror( int i )       
  468. {
  469.   extern int sys_nerr;
  470.   extern char *sys_errlist[];
  471.   return ( i >= 0 && i < sys_nerr ) ? sys_errlist[i] : "Unknown Error" ;
  472. }
  473.  
  474.  
  475. /* Return string corresponding to character c. */
  476.  
  477. char *cname ( u_char c ) 
  478. {
  479. #define CNAMEFMT "<0x%02x>"
  480. #define CNAMELEN 6+1
  481.   static char *cname [ 256 ] = {        /* character names */
  482.   "<NUL>","<SOH>","<STX>","<ETX>", "<EOT>","<ENQ>","<ACK>","<BEL>",
  483.   "<BS>", "<HT>", "<LF>", "<VT>",  "<FF>", "<CR>", "<SO>", "<SI>", 
  484.   "<DLE>","<XON>","<DC2>","<XOFF>","<DC4>","<NAK>","<SYN>","<ETB>",
  485.   "<CAN>","<EM>", "<SUB>","<ESC>", "<FS>", "<GS>", "<RS>", "<US>" } ;
  486.   static char names[ (127-32)*2 + 129*(CNAMELEN) ], *p=names ;
  487.   static int i=0 ;
  488.     
  489.   if ( ! i ) 
  490.     for ( i=32 ; i<256 ; i++ ) {
  491.       cname [ i ] = p ;
  492.       sprintf ( p, i<127 ? "%c" : CNAMEFMT , i ) ;
  493.       p += strlen ( p ) + 1 ;
  494.     }
  495.  
  496.   return cname [ c ] ;
  497.  
  498. /* Return name of frame of type 'fr'. */
  499.  
  500. const char *frname ( int fr )
  501. {
  502. static struct framenamestruct {  int code ;  const char *name ; } 
  503. framenames [] = {
  504.  {DIS,"DIS"},{CSI,"CSI"},{NSF,"NSF"},{CFR,"CFR"},{FTT,"FTT"},{MCF,"MCF"},
  505.  {RTN,"RTN"},{RTP,"RTP"},{PIN,"PIN"},{PIP,"PIP"},{DCS,"DCS"},{TSI,"TSI"},
  506.  {NSS,"NSS"},{CRP,"CRP"},{DCN,"DCN"},{EOM,"EOM"},{MPS,"MPS"},{EOP,"EOP"},
  507.  {PRI_EOM,"PRI-EOM"},{PRI_MPS,"PRI-MPS"},{PRI_EOP,"PRI-EOP"},
  508.  {DTC,"DTC"},{CIG,"CIG"},{NSC,"NSC"}, {0,0} }, *p ;
  509.  
  510.   for ( p=framenames ; p->code && p->code != fr ; p++ ) ;
  511.   return p->code ? p->name : "UNKNOWN" ;
  512. }
  513.  
  514.  
  515. /* Print a message with a variable number of printf()-type arguments if the
  516.    first character appears in the verb[ose] string.  Other leading
  517.    characters and digits do additional actions: + allows the message to be
  518.    continued on the same line, '-' buffers the message instead of printing
  519.    it, E, and W expand into strings, S prints the error message for the
  520.    most recent system error, 0-4 set the return value, a space is skipped
  521.    and ends prefix.  Returns 0 if no prefix digit. */
  522.  
  523. int msg ( const char *fmt, ... ) 
  524. {
  525.   char debug[128];
  526.   int err=0, dolf=1, flush=1 ;
  527.   char *ps="", *pe="", *pw="" ;
  528.   const char *p ;
  529.   static int atcol1=1 ;
  530.   va_list ap ;
  531.   va_start ( ap, fmt ) ;
  532.  
  533.   for ( p=fmt ; *p ; p++ ) {
  534.     switch ( *p ) {
  535.     case ' ': p++ ; goto print ;
  536.     case 'A': break ;                   /* program args */
  537.     case 'C': break ;                   /* commands/responses */
  538.     case 'M': break ;                   /* modem dialogue */
  539.     case 'N': break ;                   /* negotiation */
  540.     case 'E': pw = "Error: " ; break ;
  541.     case 'H': break ;                   /* HDLC frame data */
  542.     case 'I': break ;                   /* information */
  543.     case '+': dolf = 0 ; break ;
  544.     case '-': flush = 0 ; break ;
  545.     case 'R': break ;                   /* reception errors */
  546.     case 'S': ps = strerror ( errno ) ; break ;
  547.     case 'W': pw = "Warning: " ; break ;
  548.     case '0': case '1': case '2': case '3': case '4': case '5': 
  549.       err = *p - '0' ; break ;
  550.     default: goto print ;
  551.     }
  552.   }
  553.  
  554.   print:  
  555.   if ( strchr ( verb , tolower ( *fmt ) ) ) {
  556.     if ( atcol1 ) {
  557.       sprintf ( debug , "%s: %s%s" , argv0 , pe , pw ) ;
  558.       mainError(debug);
  559.     }
  560.     vsprintf( debug, p , ap ) ;
  561.     mainError(debug);
  562.     mainError(ps);
  563.     if ( ( atcol1 = dolf ) ) {
  564.       tstamp ( ) ; 
  565.     }
  566.   }
  567.  
  568.   va_end ( ap ) ;
  569.   return err ;
  570. }
  571.  
  572.  
  573. /* Unpack the decoding tree. If the LS bit is 1, the entry is a
  574.    pointer to the next node in a code word, else it is a terminal
  575.    node and the value is a pointer to where to continue scanning
  576.    for the next code word and the following value is the run
  577.    length (or -1 or -2). */
  578.  
  579. dtree *unpktree ( void )
  580. {
  581.   static dtree nodes [ T4NODES ], *p ;
  582.   const short int *l ;
  583.   for ( p=nodes, l=pkdt4tree ; *l > 0 ; p++ ) {
  584.     if ( p - nodes >= T4NODES ) {
  585.       /* fprintf ( stderr, "error in T.4 decoding tree\n" ) ; */ break ;
  586.     }
  587.     if ( *l & 1 ) { p->zero = nodes + *l++/2 ; }
  588.     else { p->znext = nodes + *l++/2 ; p->zlen = *l++ ; } ;
  589.     if ( *l & 1 ) { p->one  = nodes + *l++/2 ; } 
  590.     else { p->onext = nodes + *l++/2 ; p->olen = *l++ ; } ;
  591.   }
  592.   return nodes + 1 ;        /* search starts at node 1 (white runs) */
  593. }
  594.  
  595.  
  596. /* millisecond delays (for systems without usleep). */
  597.  
  598. void msleep ( int t )
  599. {
  600.   waitForTimeout(t / 1000, (t % 1000) * 1000, &globalCallBack);
  601. }
  602.  
  603. /* faxgetc returns the next character
  604.    from the fax device or EOF after idle time t. */
  605.  
  606. /* t is in tenths of a second */
  607. /* returns EOF on timeout */
  608. int faxgetc(int t) {
  609.   enum ReturnStatus status;
  610.   char c;
  611.  
  612.   if (t < 0) {
  613.     t = 6000; /* 60 seconds */
  614.     while ((status = readChar(&c, t, &globalCallBack)) == TimedOut);
  615.     return c;
  616.   } else {
  617.     status = readChar(&c, t, &globalCallBack);
  618.     if (status == TimedOut) return(EOF);
  619.     return c;
  620.   }
  621. }
  622.  
  623. /* A function like faxgetc that also removes DLE escapes, detects DLE-ETX
  624.    terminators and fixes bit order. Does not ignore invalid escape
  625.    sequences. Returns the character read, EOF on error/timeout, or -2 on
  626.    DLE-ETX.  */
  627.  
  628. int faxgetdatac ( int t )
  629. {
  630.   int c ;
  631.   return ( c = faxgetc(t) ) == DLE && ( c = faxgetc(t) ) == ETX ? -2 : 
  632.     normalbits [ c & 0xff ] ;
  633. }
  634.  
  635.  
  636. /* Get a modem response into buffer s, storing up to n bytes.  The response
  637.    begins with most recent non-control character and ends with CR/LF.
  638.    Returns s or null if times-out in t deciseconds or on i/o error. Trace
  639.    messages are buffered to reduce possible timing problems. */
  640.  
  641. char *faxgets( char *s , int n , int t )
  642. {
  643.   int c=0, sta=0 ;
  644.   char *p = s ;
  645.  
  646.   while ( sta < 4 ) {
  647.     if ( ( c = faxgetc( t ) ) == EOF ) break ;
  648.     c &= 0x7f ;
  649.     if ( ! sta ) msg ( "M-+ [" ) ; 
  650.     msg ( "M-+ %s" , cname ( c ) ) ;
  651.     switch ( sta ) {
  652.     case 0 :
  653.     case 1 : sta =               ( iscntrl ( c ) ? 1 : (p=s, 2) ) ; break ;
  654.     case 2 : sta = c == CR ? 3 : ( iscntrl ( c ) ? 1 :       2  ) ; break ;
  655.     case 3 : sta = c == LF ? 4 : ( iscntrl ( c ) ? 1 : (p=s, 2) ) ; break ;
  656.     default: msg ( "Ecan't happen (faxgets)" ) ;
  657.     }
  658.     if ( sta == 2 && p < s+n-1 ) *p++ = c ;
  659.   }
  660.  
  661.   *p = 0 ;
  662.   if ( p >= s+n-1 ) msg ( "W- modem response overflow" ) ;
  663.   if ( sta ) msg ( "M- %s]" , c == EOF ? "<timeout>" : "" ) ;
  664.  
  665.   return c == EOF ? 0 : s ;
  666. }
  667.  
  668.  
  669. /* Write the fax output buffer to fax output device. To allow zero-fill
  670.    underflow padding, should be called after writing the EOL zero
  671.    byte. Warns if called with full buffer (presumably called from
  672.    faxputc()). Returns 0 or EOF on error. */
  673.  
  674. int faxflush ( void )
  675.   int n=0 ;
  676.   u_char *p = faxobuf ;
  677.  
  678.   if ( faxop >= faxoq ) msg ("Wfax output buffer overflow") ;
  679.  
  680.   n = writeToSerial(p, faxop - p);
  681.  
  682.   return n >= 0 ? (faxop = faxobuf , 0) : (msg ("ESfax device write:") , EOF) ;
  683.  
  684. /* faxputc() is a macro like putc().  It returns the character written or
  685.    EOF on error. */
  686.  
  687. #define faxputc(c) ( faxop >= faxoq && faxflush() == EOF ? EOF : \
  688.             (u_char) ( *faxop++ = (c) ) )
  689.  
  690. /* faxobytes() returns number of bytes in fax output buffer */
  691.  
  692. #define faxobytes() ( faxop - faxobuf ) 
  693.  
  694.  
  695. /* A function like faxputc but also escapes DLEs and sets proper bit
  696.    order. */
  697.  
  698. int faxputdatac ( u_char c )
  699. {
  700.   int x ;
  701.   x  = normalbits [ c ] ;
  702.   if ( x == DLE ) x = faxputc ( x ) ;
  703.   return x < 0 ? x : faxputc ( x ) ;
  704. }
  705.  
  706.  
  707. /* Send character or string to modem immediately (for commands).  Returns
  708.    like putc() and puts(). */
  709.  
  710. int faxputcnow ( char c ) 
  711.   return faxputc ( c ) < 0 ? EOF : ( faxflush() ? EOF : c ) ; 
  712. }
  713.  
  714. int faxputs ( const char *s )
  715. {
  716.   int n=0 ;
  717.   while ( s && *s && ( n = faxputcnow ( *s++ ) ) != EOF ) ;
  718.   return n ;
  719. }
  720.  
  721. /* Convert capability string to cap[ability]. Returns 0 or 2 on errors. */
  722.  
  723. int str2cap ( char *s, cap c )
  724. {
  725.   int err=0, i=0, n ;
  726.   char *p ;
  727.  
  728.   for ( n=0, p=s ; ! err && *p && i<NCAP ; p++ )
  729.     if ( isdigit ( *p ) ) c [ n++ ] = *p - '0' ;
  730.     else if ( isspace ( *p ) || *p == ',' ) ;
  731.     else err = msg ("E2invalid character (%c) in (%s)", *p , s ) ;
  732.  
  733.   if ( n < NCAP ) msg ( "W0missing value(s) in capability \"%s\"", s ) ;
  734.  
  735.   for ( i=0 ; i<n ; i++ ) 
  736.     if ( c [ i ] > capmax [ i ] || c [ i ] < 0 ) 
  737.       err = msg ( "E2%s = %d is %s", s , t30tab[i].name , c [ i ], 
  738.         c[i]<0 ? "negative" : "too big" ) ;
  739.   
  740.   return err ;
  741. }
  742.  
  743. /* Print cap[ability] 'c' using text values and prefix 's'. */
  744.  
  745. void printcap ( char *s , cap c )
  746. {
  747.   int i ;
  748.   msg ( "N-+ %s" , s ) ;
  749.   for ( i=0 ; i<NCAP ; i++ ) 
  750.     msg ( "N-+  %s" , c[i] > capmax [ i ] || c[i] < 0 ?  "ERROR" :
  751.      capvaluestr [ i ] [ c[i] ] ) ;
  752.   msg ( "N-" , s ) ;
  753. }
  754.  
  755.  
  756. /* Convert a cap[ability] 'c' to a DIS/DCS/DTC FIF 'fif' of 'len' bytes.
  757.    Converts into DIS format if 'isdis' is true, else into DCS/DTC
  758.    format. Returns 0 or 3 on internal error (bad conversion tables). */
  759.  
  760. int mkdis ( cap c , u_char *fif , int len , int isdis ) 
  761. {
  762.   int err=0, i, j, k ;
  763.   t30tabst *p ;
  764.  
  765.   if ( len < 3 || len > 5 ) 
  766.     len = msg ( "W3bad DCS/DIS length (%d) set to 3" , len ) ;
  767.  
  768.   fif[0] = 0 ;
  769.   fif[1] = isdis ? 0xc0 : 0x40 ;
  770.   for ( i=2 ; i<len-1 ; i++ ) fif[i] = 0x01 ;       /* add extension bits */
  771.   fif[i] = 0 ;
  772.  
  773.   for ( i=0 ; p=t30tab+i, i<NCAP ; i++ ) {
  774.  
  775.     j = c [ i ] ;
  776.     if ( j > capmax [ i ] || j < 0 ) {
  777.       j = 0 ;
  778.       err = msg ( "E3mkdis: bad %s = %d set to 0", p->name, j ) ;
  779.     }
  780.     k = ( isdis ? p->captodis : p->captodcs ) [ j ] ;
  781.     if ( k == X ) {
  782.       k = 0 ;
  783.       err = msg ( "E3mkdis: can't happen (invalid %s)", p->name ) ;
  784.     }
  785.     if ( p->byte < len ) fif [ p->byte ] |=  k << p->shift ;
  786.   }
  787.   return err ;
  788. }
  789.  
  790. /* Return length of DIS/DTC FIF (counts extension bits). */
  791.  
  792. int dislen ( u_char *fif )
  793. {
  794.   int n ;
  795.   for ( n=3 ; fif [ n-1 ] & 0x01 ; n++ ) ;
  796.   return n ;
  797. }
  798.  
  799. /* Convert received DIS/DCS/DTC FIF to cap. Returns 0 or 3 on error (bad
  800.    DIS/DCS field). */
  801.  
  802. int mkcap ( u_char *fif, cap c , int dis ) 
  803. {
  804.   int err=0, i, j, k, len ;
  805.   t30tabst *p ;
  806.  
  807.   len = dislen ( fif ) ;
  808.  
  809.   for ( i=0 ; i<NCAP ; i++ ) {
  810.     p=t30tab+i ;
  811.     if ( p->byte >= len ) {
  812.       c [ i ] = 0 ;
  813.     } else {
  814.       j = fif [ p->byte ] >> p->shift & p->mask ;
  815.       k = ( dis ? p->distocap : p->dcstocap ) [ j ] ;
  816.       if ( k == X ) {
  817.     c [ i ] = 0 ;
  818.     err = msg("E3mkcap: bad %s field (%d) set to 0", p->name, j) ;
  819.       } else { 
  820.     c [ i ] = k ;
  821.       }
  822.     }
  823.   }
  824.   return err ;
  825. }
  826.  
  827.  
  828. /* Compute compatible local/remote capabilities. As side effect, sets
  829.    minimum line length and decimation. This routine is used by the sending
  830.    station only (or both in Class 2). Returns 0 if OK or 3 if no compatible
  831.    settings possible. */
  832.  
  833. int mincap ( cap local, cap remote, cap session )
  834. {
  835.   int err=0, i ;
  836.   int msttab[2][8] = { { 0,1,3,3,5,5,7,7 } , { 0,1,1,3,3,5,5,7 } } ;
  837.  
  838.   for ( i=0 ; i<NCAP && i!=ST ; i++ )
  839.     session[i] = remote[i] < local[i] ? remote[i] : local[i] ;
  840.  
  841.   session[ST] = msttab [ session[VR] ] [ remote[ST] ] ;
  842.  
  843.   minlen = (cps[session[BR]] * delay[session[ST]] + 500) / 1000 ;
  844.  
  845.   decimate = local[VR] == 1 && session[VR] == 0 ;
  846.  
  847.   printcap ( "local  ", local ) ;
  848.   printcap ( "remote ", remote ) ;
  849.   printcap ( "session", session ) ;
  850.   msg ( "N- padding to %d bytes/line.%s", minlen, 
  851.        decimate ? " reducing 198->96 lpi." : "" ) ;
  852.  
  853.   if ( local[WD] != session[WD] || local[LN] != session[LN] || 
  854.       local[DF] != session[DF] ) 
  855.     err = msg ("E3can't convert image to remote capabilities" ) ;
  856.  
  857.   return err ;
  858. }
  859.  
  860.  
  861. /* Search for a match to the string s in a null-terminated array of
  862.    possible prefix strings pointed to by p.  The first character of each
  863.    prefix string is skipped.  Returns pointer to the table entry or NULL if
  864.    not found.  */
  865.  
  866. char *strtabmatch ( char **p, char *s )
  867. {
  868.   while ( *p && strncmp ( *p+1 , s , strlen ( *p+1 ) ) ) p++ ;
  869.   return ( ! *p || **p == '-' ) ? NULL : *p ;
  870. }
  871.  
  872.  
  873. /* Send command to modem and check responses.  Collects pending
  874.    (unexpected) responses and then pauses for inter-command delay
  875.    (cmdpause) if t is negative.  Writes command s to modem if s is not
  876.    null.  Reads responses and terminates when a response is one of the
  877.    prompts in responses[] or if times out in t deciseconds.  Notes and
  878.    processes important class 2 responses (FPTS: Page Transfer Status, FHNG:
  879.    HaNGup status, and FDCS: Current Session capabilities).  Repeats command
  880.    if detects a RING response (probable collision). Returns the first
  881.    character of the matching prefix string (e.g. 'O' for OK, 'C' for
  882.    CONNECT, etc.)  or EOF if no such response was received within timeout
  883.    t. */
  884.  
  885. int cmd ( const char *s , int t )
  886. {
  887.   char buf [ CMDBUFSIZE ] , *p = "" ;
  888.   int ppr ;
  889.  
  890.  retry:
  891.  
  892.   while ( s && faxgets ( buf , CMDBUFSIZE , t<0 ? cmdpause : 0 ) )
  893.     msg ( "W- unexpected response \"%s\"", buf ) ;
  894.  
  895.   msg ( s ? "C- command  \"%s\"" : "C- waiting" , s ) ;
  896.  
  897.   if ( s ) { faxputs ( "AT" ) ; faxputs ( s ) ; faxputcnow ( CR ) ; }
  898.  
  899.   while ( t && ( p = faxgets( buf , CMDBUFSIZE , t<0 ? -t : t ) ) ) {
  900.  
  901.     msg ( "C- response \"%s\"" , p ) ;
  902.  
  903.     if ( ! strncmp ( buf, "DATA" , 4 ) ||
  904.     ! strncmp ( buf, "CONNECT DATA" , 12 ) )
  905.       datamode = 1 ;
  906.     if ( ! strncmp ( buf, "+FDCS:", 6 ) && ! str2cap ( buf+6 , session ) ) 
  907.       mincap ( local, session, session ) ; /* set decimation & minlen */
  908.  
  909.     if (strncmp(buf, "+FCSI:", 6) == 0 || strncmp(buf, "+FTSI:", 6) == 0) {
  910.       strcpy(remoteid, buf + 6);
  911.     }
  912.  
  913.     if ( sscanf ( buf , "+FPTS: %d", &ppr ) > 0 ) good = ppr & 1 ;
  914.     sscanf ( buf, "+FHNG: %d", &hsc ) ;
  915.     sscanf ( buf, "CONNECT %d", &crate ) ;
  916.  
  917.     if ( ( p = strtabmatch ( (char**) prompts , buf ) ) ) break ;
  918.  
  919.     if ( ! strncmp ( buf, "RING", 4 ) ) { msleep(100) ; goto retry ; }
  920.   }
  921.  
  922.   return p ? *p : EOF ;
  923. }
  924.  
  925. /* Send command to modem and wait for either of two possible replies after
  926.    testing (and possibly setting) current error status via err pointer. */
  927.  
  928. void ckcmd ( int *err, const char *s, int t, int r1, int r2 )
  929. {
  930.   int c ;
  931.   if ( ! *err )
  932.     if ( ( c = cmd ( s, t ) ) != r1 && c != r2 ) 
  933.       if ( c == EOF ) 
  934.     *err = msg ("E3modem command (%s) timed out", s ? s : "none" ) ;
  935.       else
  936.     *err = msg ("E3wrong response to command (%s)", s ? s : "none" ) ;
  937. }
  938.  
  939. /* Resynchronize modem from an unknown state.  If no immediate response,
  940.    try pulsing DTR low (needs &D{2,3,4}), and cancelling data or fax data
  941.    modes.  In each case, discard any responses for about 2 seconds and then
  942.    try command s.  Returns 0 if OK or 4 if no response.  */
  943.  
  944. int faxsync( char *s )
  945. {
  946.   int err=0, method=0 ;
  947.  
  948.   while ( ! err ) {
  949.     switch ( method++ ) {
  950.     case 0 : 
  951.       break ;
  952.     case 1 : 
  953.       msg ("Isync: dropping DTR -- Unsupported on Amiga") ;
  954.       break ;
  955.     case 2 : 
  956.       msg ("Isync: sending escape") ;
  957.       faxputs ( DLE_ETX ) ; 
  958.       msleep ( 1500 ) ;
  959.       faxputs ( "+++" ) ; 
  960.       break ;
  961.     case 3 :
  962.       err = msg ("E4sync: modem not responding") ;
  963.       continue ;
  964.     }
  965.     flushBuffers();
  966.     if ( cmd ( s , -TO_RESET ) == OK ) break ;
  967.   }
  968.   return err ;
  969.  
  970.  
  971. /* Terminate session.  Makes sure modem is responding, sends modem reset
  972.    commands, resets fax device to original state, removes lock files. */
  973.  
  974. int end_session ( void )
  975. {
  976.   int i, err ;
  977.   err = faxsync ( "Q0V1" ) ;
  978.  
  979.   for ( i=0 ; ! err && i<nzopt ; i++ )
  980.     if ( cmd( zopt[i] , -TO_RESET ) != OK  && ! igniniterr )
  981.       err = msg ("E3modem reset command (%s) failed", zopt[i]) ;
  982.  
  983.   return err ;
  984.     
  985.  
  986. /* signal handler: hang up and exit */
  987.  
  988. void onsig ( int sig ) 
  989.   msg ( "Eterminating on signal %d", sig ) ; 
  990.   end_session ( ) ;
  991.   msg ("Idone, returning 5") ;
  992.   exit(5) ; 
  993.  
  994. int getResolution(char* fname) {
  995.   FILE* fp = fopen(fname, "rb");
  996.   if (fp) {
  997.     faxin_open_fp(fp, 0, 0);
  998.     faxin_begin_page(fp);
  999.     fclose(fp);
  1000.     return faxin.faxhdr.info.hires;
  1001.   }
  1002.   return 0;
  1003. }
  1004.  
  1005.  
  1006. /* Initialize session.  Try locking and opening fax device until opened or
  1007.    get error. Then set tty modes, register signal handler, setup modem. */
  1008.  
  1009. int begin_session ( char *faxfile )
  1010. {
  1011.   char txcmd[50];
  1012.   int i , err=0;
  1013.  
  1014.   if ( ! err ) msg ( "Iopened %s on fd %d", faxfile, faxdev ) ;
  1015.  
  1016.   for ( i=0 ; ! err && catch [ i ] ; i++ ) 
  1017.     if ( signal ( catch [ i ] , onsig ) == SIG_ERR ) 
  1018.       err = msg ( "ES2can't set signal %d handler:", catch [ i ] ) ;
  1019.   
  1020.   if ( !err ) err = faxsync("Q0V1") ;
  1021.  
  1022.   for ( i=0 ; ! err && i<niopt ; i++ )
  1023.     if ( cmd( iopt[i] , -TO_RESET ) != OK && ! igniniterr )
  1024.       err = msg ("E3modem initialization command (%s) failed", iopt[i]) ;
  1025.  
  1026.   if (nfiles) {
  1027.     /* we are transmitting */
  1028.     local[VR] = getResolution(fnames[0]);
  1029.   }
  1030.  
  1031.   if (!c1) {
  1032.     sprintf(txcmd, "+FDCC=%d,%d,%d,%d,%d,%d,%d,%d", local[VR], local[BR], local[WD], local[LN], local[DF], local[EC], local[BF], local[ST]);
  1033.  
  1034.     if (cmd(txcmd, -TO_RESET) != OK && ! igniniterr)
  1035.       err = msg("E3fax capability setup (%s) failed", txcmd);
  1036.   }
  1037.  
  1038.   return err ;
  1039. }
  1040.  
  1041.  
  1042. /* Position `file' to start of page p (1...nfiles).  Assumes global
  1043.    variables 'fnames' and 'nfiles,' have been initialized.  This function
  1044.    looks ahead to see if there are more pages and if they are of a
  1045.    different format.  Eventually will handle multi-page files with embedded
  1046.    format information (e.g. TIFF-F).  Sets ppm to MPS if more pages to be
  1047.    sent with same format, EOM if more pages with different format, EOP if
  1048.    no more pages. Returns 0 if OK, 2 on errors.  Currently sets ppm to MPS
  1049.    or EOP only.  */
  1050.  
  1051. int rdpage ( int p , int *ppm )
  1052. {
  1053.   int err=0 ;
  1054.  
  1055.   if ( file && fclose ( file ) ) msg ("ES2file close:") ; 
  1056.  
  1057.   if ( p <= nfiles && p > 0 ) {
  1058.     if ( ( file = fopen ( fnames[p-1] , "rb" ) ) ) {
  1059.       msg ( "Iopened \"%s\" (p %d/%d)", fnames[p-1] , p, nfiles ) ;
  1060.       *ppm = p >= nfiles ? EOP : MPS ;
  1061.       faxin_open_fp(file, 0, 0);
  1062.       faxin_begin_page(file);
  1063.     } else {
  1064.       err = msg ("ES2 file open\n%s:", fnames[p-1]) ;
  1065.     }
  1066.   } else {
  1067.     err = msg ("E2 bad page number (%d)", p ) ;
  1068.   }
  1069.  
  1070.   return err ;
  1071. }
  1072.  
  1073. /* Same as rdpage() but creates new file name and opens `file' for
  1074.    writing. If page is -1 removes the most recently opened file. Assumes
  1075.    'fnampat' contains the strftime(3) pattern to be used for generating
  1076.    file names. Returns 0 if OK, 2 on errors. */
  1077.  
  1078. int wrpage ( int p )
  1079. {
  1080.   int err=0 ;
  1081.   static char fname [ FILENAME_MAX + 1 ] , base [ FILENAME_MAX + 1 ] ;
  1082.   time_t t ;
  1083.  
  1084.   t = time(0)  ;
  1085.   if ( ! *base ) strftime( base, FILENAME_MAX, fnamepat, localtime ( &t ) ) ;
  1086.  
  1087. //  if ( file && fclose ( file ) ) msg ("ES2file close:") ; 
  1088.   if ( file ) {
  1089.     faxout_end_page(FaxHandle);
  1090.     faxout_close(FaxHandle);
  1091.     msg("ES2file close:");
  1092.   }
  1093.  
  1094.   if ( p == -1 ) {
  1095.     if ( remove ( fname ) )
  1096.       err = msg ( "ES2deleting file %s", fname ) ; 
  1097.     else
  1098.       msg ( "Iremoved %s", fname ) ; 
  1099.   } else {
  1100.     if ( *fname ) msg ( "Ireceived -> %s", fname ) ;
  1101.     sprintf ( fname, "%.*s.%03d", FILENAME_MAX - 5, base, p ) ;
  1102.     FaxHandle = faxout_open(fname, 0);
  1103.     if (FaxHandle) {
  1104.       file = FaxHandle->fp;
  1105.       err = 0;
  1106.     } else err = msg ("ES2opening file\n%s:", fname) ;
  1107.  
  1108.     if ( ! err ) msg ( "Iopened file %s", fname ) ;
  1109.   }
  1110.   return err ;
  1111. }
  1112.  
  1113.  
  1114. /* Send data for one page. Enable serial port flow control.  Read
  1115.    characters from file and pad lines if necessary.  Bit-reverse characters
  1116.    and escape DLE before storing in buffer.  Flush buffer if enough
  1117.    characters stored.  Append RTC if needed.  Send DLE-ETX and return
  1118.    serial port to command mode when done. Returns 0 if OK, non-0 on
  1119.    errors. */
  1120.  
  1121. int send_data( void )
  1122. {
  1123.   int done=0, err=0;
  1124.   int lastc=0, c ;
  1125.   int eol=0, len=0, llen=0, eolcnt=0, rtc=0, skip=0 ;
  1126.   int bytes=0, pad=0, linec=0, lines=0 ;
  1127.   int i, noise=0 ;
  1128.   time_t start, dt ;
  1129.   dtree *t4p = t4tree ;
  1130.  
  1131.   done = err = 0; 
  1132.   start = time(0) ;
  1133.  
  1134.   while ( ! done ) {
  1135.  
  1136.     if ( ( c = getc ( file ) ) != EOF ) {
  1137.  
  1138. #define T4CODE(l) ( l < 0 ? (eol=l,llen=len,len=0) : (len+=l) ) 
  1139.       T4CTST(c) ;
  1140. #undef T4CODE
  1141.  
  1142.       if ( eol ) {
  1143.     if ( ! llen ) {
  1144.       if ( ++eolcnt >= 6 ) done = rtc = 1 ; 
  1145.     } else {
  1146.       eolcnt=0 ;
  1147.     }
  1148.     if ( eol == -2 ) 
  1149.       msg ("W- image error (line %d)", lines ) ;
  1150.     if ( ! skip ) {
  1151.       lines++ ; 
  1152.       while ( ( linec < minlen || lastc ) ) { 
  1153.         faxputc ( lastc = 0 ) ; 
  1154.         linec++ ; 
  1155.         pad++ ; 
  1156.       }
  1157.       linec = 0 ;
  1158.     }
  1159.     if ( decimate ) {
  1160.       if ( skip ) {
  1161.         skip = 0 ;
  1162.       } else {
  1163.         faxputc ( lastc = 0 ) ; /* force enough zeroes for next EOL */
  1164.         pad++ ; 
  1165.         skip = 1 ;
  1166.       }
  1167.     }
  1168.     if ( faxobytes() > MINWRITE ) { 
  1169.       if ( faxflush() ) done=err=2 ;  
  1170.     }
  1171.     eol=0 ;
  1172.       }
  1173.       if ( ! skip || ! llen ) {
  1174.     faxputc ( lastc = txbitorder [ c ] ) ; 
  1175.     linec ++ ;
  1176.     bytes++ ;
  1177.     if ( lastc == DLE ) { faxputc ( lastc = DLE ) ; linec++ ; pad++ ; }
  1178.       }    
  1179.     } else {
  1180.       done = 1 ;
  1181.       if ( ferror ( file ) ) err = msg ("ES2file read:") ;
  1182.     }
  1183.   }
  1184.  
  1185.   if ( ! err && ! rtc ) {
  1186.     for ( i=0 ; i<T4RTCLEN ; i++ ) faxputdatac ( (T4RTC)[i] ) ;
  1187.     msg ( "I- added RTC (%d bytes)" , T4RTCLEN ) ;
  1188.     pad += T4RTCLEN ;
  1189.   } 
  1190.  
  1191.   for ( ; ( c = faxgetc ( 0 ) ) != EOF ; noise++ ) 
  1192.     msg ( "W-+%s" , cname ( c ) ) ; 
  1193.   if ( noise ) msg ("W- : %d byte(s) received while sending", noise ) ;
  1194.     
  1195.   faxputs ( DLE_ETX ) ;        /* and faxflush() */
  1196.  
  1197.   ckcmd ( &err, 0, TO_DRAIN, OK, 0 ) ;
  1198.  
  1199.   dt = time(0) - start ;
  1200.   msg ("Isent %d lines  %d+%d bytes  %d s  %d bps" , 
  1201.        lines, bytes, pad, (int) dt, ((bytes+pad)*8)/dt ) ;
  1202.  
  1203.   return err ;
  1204. }
  1205.  
  1206.  
  1207. /* Receive data. Get characters from the modem and check for
  1208.    errors/EOF.  Remove long runs of zeroes (T.4 FILL). If in DLE
  1209.    escape check for DLE, end of data or protocol violation.
  1210.    Otherwise, test for DLE & write the character out.  Check that
  1211.    the output file is still OK.  If not, send one CANcel
  1212.    character and wait for protocol to complete.  */
  1213.  
  1214. int receive_data ( void )
  1215. {
  1216.   int done=0, err=0, nulls=0, c ;
  1217.   int bytes=0, lines=0, nerr=0 ;
  1218.   int eol=0, eolcnt=0, len=0, llen=0, lastlen=0, rtc=0 ;
  1219.   dtree *t4p = t4tree ;
  1220.  
  1221.   faxout_begin_page(FaxHandle, session[VR] /* VR */, 1 /* dirty */);
  1222.  
  1223.   while ( ! done ) {
  1224.  
  1225.     if ( ( c = faxgetc( TO_CHAR ) ) == DLE ) {
  1226.       if ( ( c = faxgetc( TO_CHAR ) ) == ETX ) done = 1 ;
  1227.       else if ( c != DLE ) {
  1228.     msg ( "W- \"%s\" received after DLE", cname ( c ) ) ;
  1229.     continue ;
  1230.       }
  1231.     }    
  1232.     if ( c == EOF ) {
  1233.       done = err = msg ("E3fax device read timeout") ;
  1234.       continue ;
  1235.     } 
  1236.     if ( c == 0 ) {
  1237.       if ( ++nulls > MAXNULLS ) {
  1238.     continue ;
  1239.       }
  1240.     } else {
  1241.       nulls = 0 ;
  1242.     }
  1243.     if ( ! done && ! rtc ) {
  1244.       c = rxbitorder[c];
  1245.       putc ( c, file ) ;
  1246.       bytes++ ;
  1247.     }
  1248. #define T4CODE(l) ( l<0 ? (eol=l, llen=len, len=0) : (len+=l) ) 
  1249.       T4CTST(c) ;
  1250. #undef T4CODE
  1251.     if ( eol ) {
  1252.       lines++ ;
  1253.       if ( ( ( llen != lastlen ) && llen && lastlen && lines > 3 ) ) {
  1254.     nerr++ ;
  1255.     msg ("R-+ (%d:%d)", lines, llen ) ;
  1256.       }
  1257.       lastlen = llen ;
  1258.       if ( ! llen ) {
  1259.     if ( ++eolcnt >= 6 ) rtc = 1 ; 
  1260.       } else {
  1261.     eolcnt=0 ;
  1262.       }
  1263.       eol=0 ; 
  1264.     }
  1265.  
  1266.     if ( ! err && ferror ( file ) ) {
  1267.       err = msg ("ES2fax file write:") ;
  1268.       faxputcnow ( CAN ) ;
  1269.       msg ("Wreceive CANcelled") ;
  1270.     } 
  1271.  
  1272.   } /* while */
  1273.   
  1274.   if ( nerr ) msg ("R-  : reception errors" ) ;
  1275.   nerr /= 2 ;
  1276.   if ( nerr ) msg ("W- %d reception errors", nerr ) ;
  1277.  
  1278.   ckcmd ( &err, 0, TO_C2EOR, ( c1 ? NO : OK ) , 0 ) ;
  1279.   
  1280.   if ( ! err && ! rtc ) {
  1281.     fwrite ( T4RTC , sizeof ( u_char ) , T4RTCLEN , file ) ;
  1282.     msg ( "I- added RTC (%d bytes)" , T4RTCLEN ) ;
  1283.   }
  1284.  
  1285.   msg ( "I- received %d lines, %d bytes, %d errors", lines, bytes, nerr ) ;
  1286.  
  1287.   good = nerr < maxpgerr ;
  1288.   return err ;
  1289. }
  1290.  
  1291.  
  1292. /* Send training check sequence of n zeroes.  Returns 0 or 2 on error. */
  1293.  
  1294. int puttrain ( const char *s , int n )
  1295. {
  1296.   int i, err=0 ;
  1297.  
  1298.   ckcmd ( &err, s , TO_FT , CONNECT, 0 ) ;
  1299.  
  1300.   if ( ! err ) {
  1301.  
  1302.     for ( i=0 ; ! err && i < n ; i++ ) {
  1303.       faxputc ( 0 ) ;
  1304.       if ( ! ( i & 63 ) ) faxflush ( ) ;
  1305.     }
  1306.     faxputc ( 1 ) ;  /* disable zero-fill padding */
  1307.     faxputs ( DLE_ETX ) ;
  1308.  
  1309.     msg ( "I- sent TCF (%d bytes)", n ) ;
  1310.  
  1311.     ckcmd ( &err, 0, TO_DRAIN, OK, 0 ) ;
  1312.   }
  1313.  
  1314.   return err ;
  1315. }
  1316.  
  1317. /* Checks n bytes of received training check sequence. Skips first n/16
  1318.    bytes.  Returns 0 if OK or ignoring training errors, 1 if too many
  1319.    errors/not enough data, or 3 on other errors. */
  1320.  
  1321. int gettrain ( const char *s , int n ) 
  1322.   int err=0, c, i, errcnt=0, skip=n/16 ;
  1323.  
  1324.   ckcmd ( &err, s , TO_FT , CONNECT, 0 ) ;
  1325.   
  1326.   for ( i=0 ; ! err && ( c = faxgetdatac ( TO_CHAR ) ) >= 0 ; i++ )
  1327.     if ( c != 0 && i >= skip && i < n+skip ) errcnt++ ;
  1328.   
  1329.   ckcmd ( &err, 0, TO_RTCMD, NO, 0 ) ;
  1330.   
  1331.   if ( !err )
  1332.     msg ( "I- received TCF (%s: %d error(s) in %d/%d bytes)", 
  1333.      ( err = i<n || errcnt>MAXTRAINERR ) ? "failed" : "passed" , 
  1334.      errcnt, n, i ) ;
  1335.  
  1336.   return err ;
  1337. }
  1338.  
  1339. /* Log sent/received HDLC frame.  Display of these messages is delayed to
  1340.    avoid possible timing problems. */
  1341.  
  1342. void logfr ( const char *s , const char *nm , u_char *p , int n )
  1343. {
  1344.   int i=0 ;
  1345.   msg ( "I- %s %s", s, nm ) ;
  1346.   msg ( n > 10 ? "H- %s %d bytes:" : "H-+ %s %d bytes:" , s, n ) ;
  1347.   for ( i=0 ; i<n ; i++ ) {
  1348.     msg ( "H-+  %02x" , p[i] & 0xff ) ;
  1349.     if ( ( i&0xf ) == 0xf && i != n-1 ) msg ( "H-" ) ;
  1350.   }
  1351.   msg ( "H-") ;
  1352. }
  1353.  
  1354. /* Send HDLC control frame of type type.  Extra bits are OR'ed to the frame
  1355.    type (FCF) if this frame follows a previous one (no +FTH required) or if
  1356.    more frames will follow.  Sets up flag, address, and fax control field
  1357.    (FCF) bytes in `buf'.  Sends these plus `len` additional bytes.
  1358.    Terminates with DLE-ETX and checks response.  Returns 0 if OK, 2 or 3 on
  1359.    error. */
  1360.  
  1361. #define MORE_FR  0x100 
  1362. #define SUB_FR 0x200 
  1363.  
  1364. int putframe ( int type, u_char *buf , int len )
  1365. {
  1366.   int err=0, i ;
  1367.  
  1368.   buf [ 0 ] = 0xff ;
  1369.   buf [ 1 ] = type & MORE_FR ? 0xc0 : 0xc8 ;
  1370.   buf [ 2 ] = type & 0xff ;
  1371.  
  1372.   if ( ! ( type & SUB_FR ) )
  1373.     ckcmd ( &err, "+FTH=3" , TO_FT, CONNECT, 0 ) ;
  1374.  
  1375.   if ( ! err ) {
  1376.     for ( i=0 ; ! err && i< len+3 ; i++ )  faxputdatac ( buf [ i ] ) ;
  1377.     faxputs ( DLE_ETX ) ;
  1378.  
  1379.     ckcmd ( &err, 0, TO_DRAIN, type & MORE_FR ? CONNECT : OK , 0 ) ;
  1380.     if ( ! err ) 
  1381.       logfr ( "sent", frname(type & 0x7f), buf, len+3 ) ;
  1382.   }
  1383.  
  1384.   return err ;
  1385. }
  1386.  
  1387.  
  1388. /* Read HDLC frame and store it in buffer buf of size n.  Skips
  1389.    issuing +FRH command on pass==0.  Returns length of frame if
  1390.    OK, EOF on timeout, -3 if any errors as per T.30 5.4.2 (too
  1391.    long, FCS error) */
  1392.  
  1393. int getframe ( int pass , u_char *buf , int n , int t )
  1394. {
  1395.   int err=0, c, i=0 ;
  1396.  
  1397.   if ( pass && ( c = cmd ( "+FRH=3" , t ) ) != CONNECT ) 
  1398.     err = ( c == EOF ) ? -EOF : msg ( "E3get frame command failed") ;
  1399.   
  1400.   if ( err == -EOF ) { faxputc ( CAN ) ; cmd ( "" , TO_ABRT ) ; }
  1401.  
  1402.   if ( ! err ) {
  1403.     for ( i=0 ; ( c = faxgetdatac ( pass ? TO_CHAR : t ) ) >= 0  ; i++ )
  1404.       if ( i < n ) buf[ i ] = c ;
  1405.     if ( c == EOF ) 
  1406.       err = i ? msg ( "E3timed out reading frame data") : EOF ;
  1407.     if ( i >= n ) err = msg ( "E3frame too long (%d bytes)", i ) ;
  1408.   } 
  1409.  
  1410.  
  1411.   ckcmd ( &err, 0, TO_RTCMD, OK, CONNECT ) ;
  1412.  
  1413.   return err ? -err : ( i < n ? i : n ) ;
  1414. }
  1415.  
  1416. /* Reverse bit and byte order of ID strings as per T.30 5.3.6.2.4-6 */
  1417.  
  1418. void revcpy ( u_char *from , u_char *to )
  1419. {
  1420.   int i, j ;
  1421.   for ( i=0, j=IDLEN-1 ; i<IDLEN ; i++, j-- ) 
  1422.     to [ i ] = normalbits [ from [ j ] & 0xff ] ;
  1423. }
  1424.  
  1425. /* Handle procedure interrupt requests (just print message for now).
  1426.    Returns 0. */
  1427.  
  1428. int pr_int( void )
  1429. {
  1430.   return msg ("W0procedure interrupt request ignored" ) ;
  1431. }
  1432.  
  1433.  
  1434. /* Class 1 send.  Each received frame elicits an appropriate reply.
  1435.    Optional or unrecognized frames are ignored.  Terminates by sending DCN
  1436.    after receiving MCF after EOP or on error.  Timeouts, bad frames or CRP
  1437.    repeat last command up to MAXRETRY times.  On training check failures
  1438.    the speed (remote capability br) is reduced. The lowest speed is retried
  1439.    MAXTRAIN times.  Page transmission failures are retried NTXRETRY
  1440.    times. */
  1441.  
  1442.  
  1443. enum replies
  1444. { NONE=0x100, DCS1, DCS2, TXDATA, PPM, DONE, BADFR, TIMEOUT, SENDDIS } ;
  1445.  
  1446. int c1send ( int started ) 
  1447.   int err=0, done=0, pass=started, page=1, frame=NONE, reply=DONE ;
  1448.   int frlen, rxdislen=0, disbit=0 ;
  1449.   int cmdtry=0, pagetry=0, traintry=0 ;
  1450.   int ppm=EOP ;
  1451.   u_char buf [ MAXFRLEN ] , *fif=buf+3 ;
  1452.  
  1453.   for ( pass=started ; ! done ; pass++ ) {
  1454.  
  1455.     if ( err ) {
  1456.       frame = NONE ;
  1457.     } else {
  1458.       frlen = getframe ( pass, buf, MAXFRLEN, started ? T3S : T1 ) ;
  1459.       if ( frlen < 3 ) {
  1460.     frame = CRP ;
  1461.       } else {
  1462.     frame = buf [ 2 ] & 0x7f ;
  1463.     logfr ( "received" , frname(frame), buf , frlen ) ;
  1464.     cmdtry = 0 ;
  1465.       }
  1466.     }
  1467.  
  1468.     switch ( frame ) {
  1469.       
  1470.     case CRP:
  1471.       if ( !started || cmdtry++ >= MAXRETRY ) 
  1472.     err = msg ( "E3 no response from remote" ) ;
  1473.       break ;
  1474.  
  1475.     case NSF:
  1476.       reply = NONE ;
  1477.       break ;
  1478.  
  1479.     case CSI:
  1480.       revcpy ( fif , (u_char*) remoteid ) ;
  1481.       msg ( "I- remote ID=\"%*.*s\"" , IDLEN, IDLEN, remoteid ) ;
  1482.       reply = NONE ;
  1483.       break ;
  1484.  
  1485.     case DIS:
  1486.       started = 1 ;
  1487.       disbit = 0x80 ;
  1488.       rxdislen = dislen ( fif ) ;
  1489.       mkcap ( fif , remote , 1 ) ;
  1490.       reply = DCS1 ;
  1491.       break ;
  1492.  
  1493.     case CFR:
  1494.       reply = TXDATA ;
  1495.       break ;
  1496.  
  1497.     case FTT:
  1498.       if ( (remote[BR] = fallback[session[BR]]) > 0 || traintry++ < MAXTRAIN ) 
  1499.     reply = DCS2 ; 
  1500.       else
  1501.     err = msg ( "E1failed to train") ; 
  1502.       break ;
  1503.  
  1504.     case PIP: 
  1505.       pr_int ( ) ; 
  1506.     case RTP:
  1507.     case MCF:
  1508.       page++ ;
  1509.       pagetry=0 ;
  1510.       if ( ppm == MPS && frame == MCF ) reply = TXDATA ;
  1511.       else if ( ppm == EOP ) done = 1 ;
  1512.       else reply = DCS2 ; 
  1513.       break ;
  1514.  
  1515.     case PIN: 
  1516.       pr_int ( ) ;
  1517.     case RTN:
  1518.       if ( pagetry++ < NTXRETRY ) reply = DCS2 ;
  1519.       else err = msg( "E1too many page send retries" ) ;
  1520.       break ;
  1521.  
  1522.     case DCN:
  1523.       err = msg ( "E3remote disconnected") ;
  1524.       break ;
  1525.  
  1526.     case NONE:
  1527.       break ;
  1528.  
  1529.     default:
  1530.       break ;
  1531.  
  1532.     } /* switch ( frame ) */
  1533.  
  1534.     switch ( err || done ? DONE : reply ) {
  1535.  
  1536.     case  DCS1:
  1537.       revcpy ( (u_char*) localid , fif ) ;
  1538.       err = putframe ( TSI + MORE_FR + disbit , buf, IDLEN ) ;      
  1539.                                /* fall through */
  1540.     case DCS2:
  1541.       mincap ( local, remote, session ) ;
  1542.       mkdis ( session , fif , rxdislen , 0 ) ;
  1543.       if ( !err ) err = 
  1544.     putframe ( DCS + (reply==DCS1 ? SUB_FR : 0) + disbit, buf, rxdislen ) ;
  1545.       ckcmd ( &err, "+FTS=8", TO_FT, OK, 0 ) ;
  1546.       if ( !err ) err = 
  1547.     puttrain ( c1cmd [1][1][session[BR]] , (int)(1.65 * cps [ session[BR] ]) ) ;
  1548.       reply = DCS2 ;
  1549.       break ;
  1550.  
  1551.     case TXDATA:
  1552.       if ( !err ) err = rdpage ( page, &ppm ) ;
  1553.       ckcmd ( &err,  c1cmd [1][0][session[BR]] , TO_FT, CONNECT, 0 ) ;
  1554.       if ( !err ) err = send_data ( ) ;
  1555.       ckcmd ( &err, "+FTS=8", TO_FT, OK, 0 ) ;      
  1556.                                /* fall through */
  1557.     case  PPM:
  1558.       if ( !err ) err =    putframe ( ppm + disbit, buf, 0 ) ;
  1559.       reply = PPM ;
  1560.       break ;
  1561.  
  1562.     case DONE:
  1563.       putframe ( DCN + disbit, buf, 0 ) ; 
  1564.       done = 1 ;
  1565.       break ;
  1566.  
  1567.     case NONE:
  1568.       break ;
  1569.  
  1570.     default:
  1571.       err = msg ( "E3can't happen(reply)" ) ;
  1572.       break ;
  1573.  
  1574.     } /* switch ( reply ) */
  1575.  
  1576.   } /* for ( ! done ) */
  1577.  
  1578.   return err ;
  1579. }
  1580.  
  1581.  
  1582. /* Class 1 receive.  Sends DIS until gets a DCS or times out.  Sends ppr
  1583.   after ppm and receives data after DCS or MPS. Note: TCF (training check
  1584.   data) is received 75 +/- 20 ms after DCS so there should be no lengthy
  1585.   processing between DCS and gettrain(). */
  1586.  
  1587. int c1receive ( int started ) 
  1588.   int err=0, done=0, page=1, pass, frame, frlen ;
  1589.   u_char buf[MAXFRLEN], *fif=buf+3 ;
  1590.  
  1591.   err = wrpage ( page ) ;
  1592.  
  1593.   for ( pass=started ; ! err && ! done ; pass++ ) {
  1594.  
  1595.     if ( pass > 0 ) {
  1596.       frlen = getframe(pass, buf, MAXFRLEN, started ? T2 : T4 ) ;
  1597.       if ( frlen == EOF ) {
  1598.     frame = started ? TIMEOUT : ( pass < MAXDIS ? SENDDIS : TIMEOUT ) ;
  1599.       } else if ( frlen < 3 ) {
  1600.     frame = started ? NONE : ( pass < MAXDIS ? SENDDIS : NONE ) ;
  1601.       } else {
  1602.     frame = buf [ 2 ] & 0x7f ;
  1603.     logfr ( "received" , frname(frame), buf , frlen ) ;
  1604.       }
  1605.     } else {
  1606.       frame=SENDDIS ;
  1607.     }
  1608.  
  1609.     switch ( frame ) {
  1610.       
  1611.     case SENDDIS:
  1612.       revcpy ( (u_char*) localid, fif ) ;
  1613.       if ( !err ) err = 
  1614.     putframe ( CSI + MORE_FR + ( pass ? 0 : SUB_FR ), buf, IDLEN ) ;
  1615.       mkdis ( local , fif , DEFDISLEN , 1 ) ;
  1616.       if ( !err ) err = 
  1617.     putframe ( DIS + SUB_FR , buf, DEFDISLEN ) ;
  1618.       break ;
  1619.       
  1620.     case TIMEOUT:
  1621.       done = err = msg ( "E3timed out waiting for sender" ) ;
  1622.       break; 
  1623.  
  1624.     case BADFR :
  1625.       msg ( "W bad frame" ) ;
  1626.       break ;
  1627.  
  1628.     case TSI: ;
  1629.       revcpy ( fif , (u_char*) remoteid ) ;
  1630.       msg ( "I- remote ID=\"%*.*s\"" , IDLEN, IDLEN, remoteid ) ;
  1631.       break ;
  1632.       
  1633.     case DCS: ;
  1634.       started = 1 ;
  1635.       mkcap ( fif , session  , 0 ) ;
  1636.       printcap ( "session" , session ) ;
  1637.       if ( gettrain ( c1cmd [0][1][session[BR]], cps[session[BR]] ) == 0 ) {
  1638.     if ( !err ) err = putframe ( CFR, buf, 0 ) ;
  1639.     if ( !err ) goto getdata ;
  1640.       } else {
  1641.     if ( !err ) err = putframe ( FTT, buf, 0 ) ;
  1642.       }
  1643.       break ;
  1644.  
  1645.     case PRI_EOM:
  1646.     case PRI_MPS:
  1647.     case PRI_EOP:
  1648.       pr_int() ;
  1649.  
  1650.     case EOM:
  1651.     case MPS:
  1652.     case EOP:
  1653.       err = wrpage ( ++page ) ;    
  1654.       if ( ! err )
  1655.     if ( good ) err = putframe ( MCF, buf, 0 ) ;
  1656.     else        err = putframe ( RTN, buf, 0 ) ;
  1657.       if ( ! err && good && ( frame == MPS || frame == PRI_MPS ) ) 
  1658.     goto getdata ; 
  1659.       if ( ! err && good && (frame == EOM || frame == PRI_EOM ) ) {
  1660.     pass=started=0 ;
  1661.     ckcmd ( &err, "+FTS=8", TO_FT, OK, 0 ) ;
  1662.       }
  1663.       break ;
  1664.  
  1665.     getdata:
  1666.       ckcmd ( &err, c1cmd [0][0][session[BR]] , TO_FT, CONNECT, 0 ) ;
  1667.       if ( !err ) err = receive_data (  ) ;
  1668.       break ;
  1669.       
  1670.     case DCN:
  1671.       done = 1 ;
  1672.       break;
  1673.  
  1674.     } /* switch */
  1675.     
  1676.   } /* do */
  1677.  
  1678.   wrpage ( -1 ) ;        /* remove last file */
  1679.   return err ;
  1680. }
  1681.  
  1682. /* Class 2 fax transmission.  Retries each page up to NTXRETRY times.
  1683.    Transmission begins after DC2 or XON is received.  Sends the data and
  1684.    sends appropriate post-page message.  Checks for 'good' status set in
  1685.    cmd() by +FPTS: responses. */
  1686.  
  1687. int c2send ( void )
  1688. {
  1689.   int c, page, err=0, done=0, try, ppm=0, noise=0 ;
  1690.   char txcmd[50];
  1691.  
  1692.   for ( page=1 ; !err && ! done ; page++ ) {
  1693.  
  1694.     for ( try=good=0 ; !err && !good && try<NTXRETRY ; try++ ) {
  1695.  
  1696.       err = rdpage ( page , &ppm ) ;
  1697.  
  1698. /*
  1699.       local[VR] = faxin.faxhdr.info.hires;
  1700.       sprintf(txcmd, "+FDT=%d,%d,%d,%d", session[DF], local[VR], session[WD], session[LN]);
  1701. */
  1702.       sprintf(txcmd, "+FDT");
  1703.       ckcmd ( &err, txcmd , TO_C2B , CONNECT, 0 ) ;
  1704.   
  1705.       if ( ! err ) 
  1706.     while ( !err && ( c = faxgetc ( TO_C2X ) ) != XON && c != DC2 )
  1707.       if ( c == EOF ) {
  1708.         msg ( "Wno XON/DC2 received after CONNECT") ;
  1709.         break;
  1710.       } else {
  1711.         msg ( "W+%s", cname ( c ) ) ;
  1712.         noise++ ;
  1713.       }
  1714.  
  1715.       if ( noise ) {
  1716.     msg ( "Wreceived (%d) characters while waiting to send", noise ) ;
  1717.     noise = 0 ;
  1718.       }
  1719.  
  1720.       if ( ! err ) err = send_data ( ) ;
  1721.  
  1722.       good = 1 ; /* assume good if no +FPTS: */
  1723.       ckcmd ( &err, ppm == EOP ? "+FET=2" : "+FET=0" , TO_C2PP, OK, 0 ) ;
  1724.  
  1725.       if ( hsc > 0 ) err = msg ( "E2abnormal termination (code %d)", hsc ) ;
  1726.     }
  1727.     if ( ppm == EOP ) done = 1 ;
  1728.     if ( try >= NTXRETRY ) {
  1729.       err = msg ( "E2too many page send retries" ) ;
  1730.       cmd ( "+FK", T3S ) ;
  1731.     }
  1732.   } 
  1733.   return err ; 
  1734. }
  1735.  
  1736. /* Class 2 fax reception.  Send fax data receive command.  If
  1737.    response is OK, then no more pages are coming.  If it's
  1738.    CONNECT receive the data for one page.  Returns 0 or >= 2 for
  1739.    errors.  */
  1740.  
  1741. int c2receive ( void )
  1742. {
  1743.   int err=0, page=1, done=0, c ;
  1744.  
  1745.   err = wrpage ( page ) ;
  1746.  
  1747.   while ( ! err && ! done && hsc < 0 ) {
  1748.     if ( ( c = cmd ( "+FDR" , TO_C2R ) ) == CONNECT ) {
  1749.       faxputcnow ( startchar ) ;
  1750.       err = receive_data ( ) ;
  1751.       ckcmd ( &err, good ? "+FPTS=1" : "+FPTS=2", T3S , OK, 0 ) ;
  1752.       if ( ! err ) err = wrpage ( ++page ) ;
  1753.     } else {
  1754.       if ( c == OK ) { wrpage ( -1 ) ; done = 1 ; } 
  1755.       else err = msg ( "E3receive (+FDR) command failed") ;
  1756.     }
  1757.   } 
  1758.   if ( hsc > 0 ) err = msg ( "E2abnormal call termination (code %d)", hsc ) ;
  1759.  
  1760.   return err ;
  1761. }
  1762.  
  1763. /* Dial a number and send a fax. */
  1764.  
  1765. int send ( char *tel , char *faxfile )
  1766. {
  1767.   char c, s [ 128 ] ;
  1768.   int err=0 ;
  1769.  
  1770.   msg ( "Idialing %.127s", tel ) ;
  1771.   if (dialTone)
  1772.     sprintf ( s , "DT%.127s" , tel ) ;
  1773.   else
  1774.     sprintf ( s , "D%.127s" , tel ) ;
  1775.  
  1776.   err = begin_session ( faxfile ) ;
  1777.  
  1778.   if ( ! err ) {
  1779.     if ( ( ( c = cmd ( s , TO_A ) ) == ( c1 ? CONNECT : OK ) ) && hsc < 0 ) 
  1780.       msg ( "Iconnected" ) ; 
  1781.     else if ( c ==  BUSY ) 
  1782.       err = msg ( "W1number is busy" ) ; 
  1783.     else if ( c == EOF ) {
  1784.       err = msg ( "E2aborted manually" ) ;
  1785.     } else 
  1786.       err = msg ( "E2can't establish session" ) ;
  1787.   }
  1788.  
  1789.   if ( ! err ) 
  1790.     err = c1 ? c1send ( 0 ) : c2send ( ) ;
  1791.  
  1792.   if ( err != 1 && err < 4 ) end_session( ) ;
  1793.  
  1794.   return err ;
  1795. }
  1796.  
  1797.  
  1798. /* Receive a fax.  Open modem device and initialize it.  Remove locks if
  1799.    sharing device with outgoing calls.  If waiting for call, wait for modem
  1800.    activity, else answer phone.  Figure out what mode we answered in and
  1801.    handle call appropriately.  Re-lock if necessary. Then exec *getty or
  1802.    run class 1 or class 2 reception routines. Modems prompt as follows
  1803.    after answering: Class 0: CONNECT nnn for data; Class 1: FAX + CONNECT
  1804.    for fax, DATA + CONNECT nnn for data, just CONNECT for fax if +FAE=0;
  1805.    Class 2: CONNECT (data) or OK (fax). ("+FCON" and "CONNECT FAX" are 
  1806.    status messages, not prompts). */
  1807.  
  1808. int receive ( char *faxfile )
  1809. {
  1810.   enum connectmode { NONE, DATAMODE, FAXMODE } ; 
  1811.   enum connectmode mode=NONE ;
  1812.   int c=0, err ;
  1813.  
  1814.   if ( ( err = begin_session ( faxfile ) ) == 0 ) {
  1815.  
  1816.     if ( ! err && waitforcall ) {
  1817.       char ch;
  1818.  
  1819.       msg ( "Iwaiting for activity on %s", faxfile ) ;
  1820.       while (readChar(&ch, 6000, &globalCallBack) == TimedOut);
  1821.       unReadChar();
  1822.       msg ( "Iactivity detected at ") ;
  1823.     }
  1824.   
  1825.     if ( ! err && share ) {
  1826.       msleep ( 200 ) ;        /* let other programs lock port  */
  1827.     }
  1828.  
  1829.     if ( ! err && softadans && *getty ) {
  1830.       if ( cmd ( ( waitforcall ? 0 : "A" ) , TO_DATAF ) == CONNECT ) 
  1831.     mode = DATAMODE ;
  1832.       else {
  1833.     int i ; /* abort data answer mode & set fax mode to try again */
  1834.     for ( i=0 ; i<3 ; i++ ) 
  1835.       if ( cmd ( c1 ? "+FCLASS=1" : "+FCLASS=2" , -TO_RESET ) == OK )
  1836.         break ; 
  1837.       }
  1838.     }
  1839.     if ( ! err && mode == NONE ) {
  1840.       c = cmd ( ! waitforcall || ( softadans && *getty ) ? "A" : 0 , TO_A ) ;
  1841.       if (c == EOF) {
  1842.     err = msg ( "E2aborted manually" ) ;
  1843.       } else {
  1844.     if ( c1 )
  1845.       mode = ( c == CONNECT ) ? ( datamode ? DATAMODE : FAXMODE ) : NONE ;
  1846.     else
  1847.       mode = ( c == CONNECT ) ? DATAMODE : ( c == OK ? FAXMODE : NONE ) ;
  1848.       }
  1849.     }
  1850.   
  1851.     if ( err || hsc >= 0 ) mode = NONE ;
  1852.  
  1853.     if ( ! err )
  1854.       switch ( mode ) {
  1855.       case DATAMODE : {
  1856.         void* lastSent = 0;
  1857.         char buf [ MAXGETTY ] ;
  1858.     msg ( "Idata call answered") ;
  1859.     sprintf ( buf , getty , crate );
  1860.     msg ( "ISending ARexx \"%s\"" , buf ) ;
  1861.     lastSent = SendARexxMsg(OutgoingRexx, buf, FALSE);
  1862.     while (1) {
  1863.       Wait(ARexxSignal(OutgoingRexx));
  1864.       if (lastSent == GetReplyARexxMsg(OutgoingRexx)) break;
  1865.     }
  1866.     
  1867.     return 0;
  1868.     break ; }
  1869.       case FAXMODE :
  1870.     msg ( "Ifax call answered") ;
  1871.     break ;
  1872.       case NONE:
  1873.     err = msg ( "E3unable to answer call") ;
  1874.     break ;
  1875.       }
  1876.  
  1877.     setReceiveSerialParms();
  1878.  
  1879.     if ( ! err ) err = c1 ? c1receive ( 0 ) : c2receive ( ) ;
  1880.  
  1881.     setSendSerialParms();
  1882.  
  1883.     if ( err != 1 && err < 4 ) end_session( ) ;   /* if not locked */
  1884.   }
  1885.   
  1886.   return err  ;
  1887. }
  1888.  
  1889.  
  1890. /* Simple (one option per argument) version of getopt(3). */
  1891.  
  1892. int optind = 1 ;
  1893. char *optarg ;
  1894.  
  1895. int nextopt( int argc, char **argv, char *args )
  1896. {
  1897.   char *a, *p ;
  1898.  
  1899.   if ( optind >= argc || *(a = argv[optind]) != '-' ) return -1 ;
  1900.   optind++ ;
  1901.  
  1902.   if ( ! *(a+1) || ! ( p = strchr ( args , *(a+1) ) ) )
  1903.     return msg ( "Eunknown option (%s)" , a ) , '?' ; 
  1904.  
  1905.   if ( *(p+1) != ':' ) optarg = 0 ;
  1906.   else
  1907.     if ( *(a+2) ) optarg = a+2 ;
  1908.     else
  1909.       if ( optind >= argc ) return msg ( "Eno argument for (%s)", a ) , '?' ;
  1910.       else optarg = argv [ optind++ ] ;
  1911.   return *(a+1) ;
  1912. }
  1913.  
  1914. char modemSetup[6][50];
  1915. char modemReset[5][50];
  1916. char idInit[50];
  1917. char debugVerbosity[15];
  1918. char capInit[CMDBUFSIZE];
  1919.  
  1920. void processConfig(char* var, char* value) {
  1921.   static int foundPortName = 0;
  1922.   static int inPortConfig = 0;
  1923.   int d;
  1924.   
  1925.   if (stricmp(var, "PORT") == 0) {
  1926.     if (stricmp(value, portNameToLoad) == 0 && foundPortName == 0) {
  1927.       foundPortName = 1;
  1928.       inPortConfig = 1;
  1929.     }
  1930.   }
  1931.  
  1932.   if (inPortConfig == 1) {
  1933.     if (stricmp(var, "SERIALNAME") == 0) {
  1934.       strcpy(serialName, value);
  1935.     } else if (stricmp(var, "UNITNUMBER") == 0) {
  1936.       sscanf(value, "%d", &unitNumber);
  1937.     } else if (stricmp(var, "FAXCLASS") == 0) {
  1938.       if (stricmp(value, "CLASS 1") == 0) {
  1939.     strcpy(modemSetup[0], "+FCLASS=1");
  1940.     c1 = 1;
  1941.       } else if (stricmp(value, "CLASS 2") == 0) {
  1942.     strcpy(modemSetup[0], "+FCLASS=2");
  1943.     c1 = 0;
  1944.       }
  1945.       iopt[niopt++] = modemSetup[0];
  1946.     } else if (stricmp(var, "FAXID") == 0) {
  1947.       sprintf ( localid, "%*.*s", IDLEN, IDLEN, value ) ;
  1948.       sprintf ( idInit , "+FLID=\"%.*s\"" , CMDBUFSIZE-9, localid ) ;
  1949.       if ( !c1 ) { iopt[niopt++] = idInit; }
  1950.     } else if (stricmp(var, "REVERSETXBITORDER") == 0) {
  1951.       sscanf(value, "%d", &d);
  1952.       if (d) txbitorder = reversebits;
  1953.     } else if (stricmp(var, "REVERSERXBITORDER") == 0) {
  1954.       sscanf(value, "%d", &d);
  1955.       if (d) {
  1956.     rxbitorder = reversebits;
  1957.     msg("Ireversed rx bit order");
  1958.       }
  1959.     } else if (stricmp(var, "STARTRXWITHXON") == 0) {
  1960.       sscanf(value, "%d", &d);
  1961.       if (d)
  1962.     startchar = XON;
  1963.       else
  1964.     startchar = DC2;
  1965.     } else if (stricmp(var, "FAXHARDWAREHANDSHAKING") == 0) {
  1966.       sscanf(value, "%d", &faxHardwareHandshaking);
  1967.     } else if (stricmp(var, "COMMANDPAUSE") == 0) {
  1968.       sscanf(value, "%d", &d);
  1969.       for ( ; d > 0; d--) cmdpause += T_CMD;
  1970.     } else if (stricmp(var, "MAXPAGEERRORS") == 0) {
  1971.       sscanf(value, "%d", &maxpgerr);
  1972.     } else if (stricmp(var, "DIALTONE") == 0) {
  1973.       sscanf(value, "%d", &dialTone);
  1974.     } else if (stricmp(var, "DEBUGVERBOSITY") == 0) {
  1975.       strcpy(debugVerbosity, value);
  1976.       verb = debugVerbosity;
  1977.     } else if (stricmp(var, "FAXCAPABILITY") == 0) {
  1978.       str2cap(value, local);
  1979. /*
  1980.       sprintf ( capInit , "+FDCC=%.*s" , CMDBUFSIZE-7, value ) ;
  1981.       if ( !c1 ) {
  1982.     iopt[niopt++] = capInit;
  1983.       }
  1984. */
  1985.     } else if (stricmp(var, "FAXSETUP1") == 0) {
  1986.       strcpy(modemSetup[1], value);
  1987.       if (strcmp(modemSetup[1], "") != 0)
  1988.     iopt[niopt++] = modemSetup[1];
  1989.     } else if (stricmp(var, "FAXSETUP2") == 0) {
  1990.       strcpy(modemSetup[2], value);
  1991.       if (strcmp(modemSetup[2], "") != 0)
  1992.     iopt[niopt++] = modemSetup[2];
  1993.     } else if (stricmp(var, "FAXSETUP3") == 0) {
  1994.       strcpy(modemSetup[3], value);
  1995.       if (strcmp(modemSetup[3], "") != 0)
  1996.     iopt[niopt++] = modemSetup[3];
  1997.     } else if (stricmp(var, "FAXSETUP4") == 0) {
  1998.       strcpy(modemSetup[4], value);
  1999.       if (strcmp(modemSetup[4], "") != 0)
  2000.     iopt[niopt++] = modemSetup[4];
  2001.     } else if (stricmp(var, "FAXSETUP5") == 0) {
  2002.       strcpy(modemSetup[5], value);
  2003.       if (strcmp(modemSetup[5], "") != 0)
  2004.     iopt[niopt++] = modemSetup[5];
  2005.     } else if (stricmp(var, "FAXRESET1") == 0) {
  2006.       strcpy(modemReset[0], value);
  2007.       if (strcmp(modemReset[0], "") != 0)
  2008.     zopt[nzopt++] = modemReset[0];
  2009.     } else if (stricmp(var, "FAXRESET2") == 0) {
  2010.       strcpy(modemReset[1], value);
  2011.       if (strcmp(modemReset[1], "") != 0)
  2012.     zopt[nzopt++] = modemReset[1];
  2013.     } else if (stricmp(var, "FAXRESET3") == 0) {
  2014.       strcpy(modemReset[2], value);
  2015.       if (strcmp(modemReset[2], "") != 0)
  2016.     zopt[nzopt++] = modemReset[2];
  2017.     } else if (stricmp(var, "FAXRESET4") == 0) {
  2018.       strcpy(modemReset[3], value);
  2019.       if (strcmp(modemReset[3], "") != 0)
  2020.     zopt[nzopt++] = modemReset[3];
  2021.     } else if (stricmp(var, "FAXRESET5") == 0) {
  2022.       if (strcmp(modemReset[4], "") != 0)
  2023.     strcpy(modemReset[4], value);
  2024.       zopt[nzopt++] = modemReset[4];
  2025.     } else if (stricmp(var, "ENDRECORD") == 0) {
  2026.       inPortConfig = 0;
  2027.     } else if (stricmp(var, "STARTRECORD") == 0) {
  2028.       inPortConfig = 0;
  2029.     } else {
  2030.     }
  2031.   }
  2032. }
  2033.  
  2034.  
  2035.  
  2036. /* Fax send/receive program for Class 1 or 2 fax modems. Initialize
  2037.    character name, bit reversal and T.4 decoding tables and process
  2038.    arguments.  Returns 0 on success, 1 if number busy or device locked, 2
  2039.    for errors, 3 for protocol errors, 4 if no modem response, 5 on fatal
  2040.    signal. */
  2041.  
  2042. int main( int argc, char **argv)
  2043. {
  2044.   int i, err=0 , c, nlocks=0 ;
  2045.   char *faxfile = FAXFILE ;
  2046.   char capinit [ CMDBUFSIZE ] , idinit [ CMDBUFSIZE ] ;
  2047. //  char msgbuf [ MAXMSGBUF ] ;
  2048.   char tmp[200];
  2049.   void* lastSent = 0;
  2050.  
  2051.   argv0 = argv[0] ;
  2052.  
  2053. //  setvbuf ( LOGF , msgbuf , _IOFBF , MAXMSGBUF ) ;
  2054.  
  2055.   msg ( "I " Version "  starts ") ;
  2056.   msg ( "I " Copyright "  (compiled "__DATE__ " " __TIME__ ")" ) ;
  2057.  
  2058.   argv0 = strrchr ( argv0 , '/' )  ;
  2059.   if ( ! argv0 ) argv0 = argv[0] ; else argv0++ ;
  2060.  
  2061.   cname ( 0 ) ;
  2062.   for ( i=0 ; i<256 ; i++ ) normalbits [ reversebits [ i ] = i ] = 
  2063.     ( i& 1 ? 128:0 ) | ( i& 2 ? 64:0 ) | ( i& 4 ? 32:0 ) | ( i&  8 ? 16:0 ) |
  2064.     ( i&16 ?   8:0 ) | ( i&32 ?  4:0 ) | ( i&64 ?  2:0 ) | ( i&128 ?  1:0 ) ;
  2065.   t4tree = unpktree( ) ;
  2066.  
  2067.   while (!err && (c=nextopt(argc,argv,"p:c:d:g:i:l:o:q:r:st:v:wx:z:T") ) != -1) {
  2068.     switch (c) {
  2069.     case 'c': 
  2070.       err = str2cap ( optarg , local ) ;
  2071.       sprintf ( capinit , "+FDCC=%.*s" , CMDBUFSIZE-7, optarg ) ;
  2072.       if ( !err && !c1 ) { optarg = capinit ; goto addopt ; }
  2073.       break ;
  2074.     case 'l': 
  2075.       if ( strlen ( optarg ) > IDLEN ) 
  2076.     msg("Wlocal ID truncated to 20 characters" ) ;
  2077.       sprintf ( localid, "%*.*s", IDLEN, IDLEN, optarg ) ;
  2078.       sprintf ( idinit , "+FLID=\"%.*s\"" , CMDBUFSIZE-9, localid ) ;
  2079.       if ( !c1 ) { optarg = idinit ; goto addopt ; }
  2080.       break ;
  2081.     case 'i': 
  2082.     addopt:
  2083.       if ( niopt < MAXIOPT ) iopt [ niopt++ ] = optarg ;
  2084.       else err = msg ( "E2too many modem init commands"); 
  2085.       break ;
  2086.     case 'z': 
  2087.       if ( nzopt < MAXIOPT ) zopt [ nzopt++ ] = optarg ;
  2088.       else err = msg ( "E2too many modem reset commands"); 
  2089.       break ;
  2090.     case 'd': faxfile = optarg ; break ;
  2091.     case 'g': getty = optarg ; break ;
  2092.     case 'p': strcpy(portNameToLoad, optarg);
  2093.       loadConfig("avm:servers.cfg", processConfig);
  2094.       break;
  2095.     case 'o': 
  2096.       for ( ; *optarg ; optarg++ ) 
  2097.     switch ( *optarg ) {
  2098.     case '1' : c1 = 1 ; break ;
  2099.     case '2' : c1 = 0 ; break ;
  2100.     case 'a' : softadans = 1 ;  break ;
  2101.     case 'e' : igniniterr = igniniterr ? 0 : 1 ;  break ;
  2102.     case 'r' : rxbitorder = reversebits ; break ;
  2103.     case 't' : txbitorder = reversebits ; break ;
  2104.     case 'x' : startchar = XON ; break ;
  2105.     case 'z' : cmdpause += T_CMD ; break ;
  2106.      default : msg ( "Wunrecognized protocol option (%c)", *optarg ) ; 
  2107.     }
  2108.       break ;
  2109.     case 'q':
  2110.       if ( sscanf ( optarg , "%d", &maxpgerr ) != 1 )
  2111.     err=msg ("E2can't read quality (-q) argument (%s)", optarg ) ;
  2112.       break;
  2113.     case 'r': 
  2114.       fnamepat = optarg ;
  2115.  
  2116.       atexit(releaseResources);
  2117.       allocateResources();
  2118.   
  2119.       err = receive ( faxfile ) ; 
  2120.       sprintf(tmp, "call setclip(upper('%s.faxid'), '%s')", portNameToLoad, remoteid);
  2121.       //fprintf(stderr, tmp);
  2122.  
  2123.       lastSent = SendARexxMsg(OutgoingRexx, tmp, TRUE);
  2124.       while (1) {
  2125.     Wait(ARexxSignal(OutgoingRexx));
  2126.     if (lastSent == GetReplyARexxMsg(OutgoingRexx)) break;
  2127.       }
  2128.  
  2129.       break;
  2130.     case 's': share = 1 ; break;
  2131.     case 't': 
  2132.       fnames = argv + optind ;
  2133.       nfiles = argc - optind ;
  2134.  
  2135.       atexit(releaseResources);
  2136.       allocateResources();
  2137.  
  2138.       if (stricmp(optarg, "MAN!") == 0)
  2139.     err = send ( "", faxfile);
  2140.       else
  2141.     err = send ( optarg , faxfile ) ; 
  2142.  
  2143.       sprintf(tmp, "call setclip(upper('%s.faxid'), '%s')", portNameToLoad, remoteid);
  2144.       //fprintf(stderr, tmp);
  2145.  
  2146.       lastSent = SendARexxMsg(OutgoingRexx, tmp, TRUE);
  2147.       while (1) {
  2148.     Wait(ARexxSignal(OutgoingRexx));
  2149.     if (lastSent == GetReplyARexxMsg(OutgoingRexx)) break;
  2150.       }
  2151.       break;
  2152.     case 'v': 
  2153.       verb = optarg ;
  2154.       if ( strchr ( verb , 'a' ) )
  2155.       for ( i=0 ; i<argc ; i++ ) msg ( "Iargv[%d]=%s", i, argv[i]) ; 
  2156.       break ;
  2157.     case 'w': waitforcall = 1 ; break ;
  2158.     case 'x': 
  2159.       if ( nlocks < MAXLKFILE ) lkfiles [ nlocks++ ] = optarg ; 
  2160.       else err = msg ( "E2too many lock files" ) ; 
  2161.       break ;
  2162.     case 'T':
  2163.       if ( ( err = begin_session ( faxfile ) ) == 0) end_session( ) ;
  2164.       break;
  2165.     default : if (stderr) fprintf ( stderr, Usage, argv0 ) ; err = 2 ; break ;
  2166.     }
  2167.   }
  2168.  
  2169.   msg ( "Idone, returning %d", err ) ;
  2170.   exit(err);
  2171. }
  2172.