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

通讯/手机编程

开发平台:

Windows_Unix

  1. x = strlen(atmbuf);
  2. if (ckstrcmp(atmbuf,xdef,x,0)) /* Matches default? */
  3.   bleep(BP_WARN); /* No */
  4. else { /* Yes */
  5.     p = xdef + x;
  6.     printf("%s ", p);
  7. #ifdef GEMDOS
  8.     fflush(stdout);
  9. #endif /* GEMDOS */
  10.     addbuf(p);
  11.     inword = cmflgs = 0;
  12.     debug(F110,"cmtxt: addbuf",cmdbuf,0);
  13. }
  14.     }
  15.     break;
  16.   case 3: /* Question Mark */
  17.     if (*xhlp == NUL)
  18.       printf(" Text string");
  19.     else
  20.       printf(" %s",xhlp);
  21.     printf("n%s%s",cmprom,cmdbuf);
  22.     fflush(stdout);
  23.     break;
  24.   default:
  25.     printf("?Unexpected return code from gtword() - %dn",x);
  26.     return(-2);
  27.         }
  28.         x = gtword(0);
  29.     }
  30. }
  31. /*  C M K E Y  --  Parse a keyword  */
  32. /*
  33.  Call with:
  34.    table    --  keyword table, in 'struct keytab' format;
  35.    n        --  number of entries in table;
  36.    xhlp     --  pointer to help string;
  37.    xdef     --  pointer to default keyword;
  38.    f        --  processing function (e.g. to evaluate variables)
  39.    pmsg     --  0 = don't print error messages
  40.                 1 = print error messages
  41.                 2 = include CM_HLP keywords even if invisible
  42.                 3 = 1+2
  43.                 4 = parse a switch (keyword possibly ending in : or =)
  44.  Returns:
  45.    -3       --  no input supplied and no default available
  46.    -2       --  input doesn't uniquely match a keyword in the table
  47.    -1       --  user deleted too much, command reparse required
  48.     n >= 0  --  value associated with keyword
  49. */
  50. int
  51. cmkey(table,n,xhlp,xdef,f)
  52. /* cmkey */  struct keytab table[]; int n; char *xhlp, *xdef; xx_strp f; {
  53.     return(cmkey2(table,n,xhlp,xdef,"",f,1));
  54. }
  55. int
  56. cmkeyx(table,n,xhlp,xdef,f)
  57. /* cmkeyx */  struct keytab table[]; int n; char *xhlp, *xdef; xx_strp f; {
  58.     return(cmkey2(table,n,xhlp,xdef,"",f,0));
  59. }
  60. int
  61. cmswi(table,n,xhlp,xdef,f)
  62. /* cmswi */  struct keytab table[]; int n; char *xhlp, *xdef; xx_strp f; {
  63.     return(cmkey2(table,n,xhlp,xdef,"",f,4));
  64. }
  65. int
  66. cmkey2(table,n,xhlp,xdef,tok,f,pmsg)
  67.     struct keytab table[];
  68.     int n;
  69.     char *xhlp,
  70.     *xdef;
  71.     char *tok;
  72.     xx_strp f;
  73.     int pmsg;
  74. { /* cmkey2 */
  75.     int i, tl, y, z = 0, zz, xc, wordlen = 0, cmswitch;
  76.     char *xp, *zq;
  77.     if (!xhlp) xhlp = "";
  78.     if (!xdef) xdef = "";
  79.     cmfldflgs = 0;
  80.     if (!table) {
  81. printf("?Keyword table missingn");
  82. return(-9);
  83.     }
  84.     tl = (int)strlen(tok);
  85.     inword = xc = cc = 0; /* Clear character counters. */
  86.     cmswitch = pmsg & 4; /* Flag for parsing a switch */
  87.     debug(F101,"cmkey: pmsg","",pmsg);
  88.     debug(F101,"cmkey: cmflgs","",cmflgs);
  89.     /* debug(F101,"cmkey: cmdbuf","",cmdbuf);*/
  90.     ppvnambuf[0] = NUL;
  91.     if ((zz = cmflgs) == 1) { /* Command already entered? */
  92. if (setatm(xdef,0) < 0) { /* Yes, copy default into atom buf */
  93.     printf("?Default too longn");
  94.     return(-9);
  95. }
  96.     } else
  97.       zz = gtword((pmsg == 4) ? 1 : 0); /* Otherwise get a command word */
  98.     debug(F101,"cmkey table length","",n);
  99.     debug(F101,"cmkey cmflgs","",cmflgs);
  100.     debug(F101,"cmkey zz","",zz);
  101.     debug(F101,"cmkey cc","",cc);
  102.     rtimer(); /* Reset timer */
  103.     while (1) {
  104. xc += cc;
  105. debug(F111,"cmkey gtword xc",atmbuf,xc);
  106. switch (zz) {
  107.   case -10: /* Timeout */
  108.     if (gtimer() < timelimit) {
  109. zz = gtword((pmsg == 4) ? 1 : 0);
  110. continue;
  111.     } else {
  112. #ifdef IKSD
  113.                 extern int inserver;
  114.                 if (inserver) {
  115.                     printf("rnIKSD IDLE TIMEOUT: %d secrn", timelimit);
  116.                     doexit(GOOD_EXIT,0);
  117.                 }
  118. #endif /* IKSD */
  119. return(-10);
  120.             }
  121.   case -5:
  122.     return(cmflgs = 0);
  123.   case -9:
  124.     printf("Command or field too longn");
  125.   case -4: /* EOF */
  126.   case -3: /* Null Command/Quit/Timeout */
  127.   case -2: /* Buffer overflow */
  128.   case -1: /* Or user did some deleting. */
  129.     return(cmflgs = zz);
  130.   case 0: /* User terminated word with space */
  131.   case 1: /* or newline */
  132.   case 4: /* or switch ending in : or = */
  133.     wordlen = cc; /* Length if no conversion */
  134.     if (cc == 0) { /* Supply default if we got nothing */
  135. if ((wordlen = setatm(xdef,(zz == 4) ? 2 : 0)) < 0) {
  136.     printf("?Default too longn");
  137.     return(-9);
  138. }
  139.     }
  140.     if (zz == 1 && cc == 0) /* Required field missing */
  141.       return(-3);
  142.     if (f) { /* If a conversion function is given */
  143. char * pp;
  144. zq = atxbuf; /* apply it */
  145. pp = atxbuf;
  146. atxn = CMDBL;
  147. if ((*f)(atmbuf,&zq,&atxn) < 0) return(-2);
  148. debug(F110,"cmkey atxbuf after *f",atxbuf,0);
  149. if (!*pp) /* Supply default if we got nothing */
  150.   pp = xdef;
  151. ckstrncpy(ppvnambuf,atmbuf,PPVLEN);
  152. if ((wordlen = setatm(pp,(zz == 4) ? 2 : 0)) < 0) {
  153.     printf("Evaluated keyword too longn");
  154.     return(-9);
  155. }
  156.     }
  157.     if (cmswitch && *atmbuf != '/') {
  158. if (pmsg & 1) {
  159.     bleep(BP_FAIL);
  160.                     printf("?Not a switch - %sn",atmbuf);
  161. }
  162. cmflgs = -2;
  163. return(-6);
  164.     }
  165.     if (cmswitch) {
  166. int i;
  167. for (i = 0; i < wordlen; i++) {
  168.     if (atmbuf[i] == ':' || atmbuf[i] == '=') {
  169. atmbuf[i] = NUL;
  170. break;
  171.     }
  172. }
  173.     }
  174.     y = lookup(table,atmbuf,n,&z); /* Look up the word in the table */
  175.     switch (y) {
  176.       case -3: /* Nothing to look up */
  177. break;
  178.       case -2: /* Ambiguous */
  179. cmflgs = -2;
  180. if (pmsg & 1) {
  181.     bleep(BP_FAIL);
  182.                     printf("?Ambiguous - %sn",atmbuf);
  183.     return(-9);
  184. }
  185. return(-2);
  186.       case -1: /* Not found at all */
  187. if (tl) {
  188.     for (i = 0; i < tl; i++) /* Check for token */
  189.       if (tok[i] == *atmbuf) { /* Got one */
  190.   debug(F000,"cmkey token:",atmbuf,*atmbuf);
  191.   ungword();  /* Put back the following word */
  192.   return(-5); /* Special return code for token */
  193.       }
  194. }
  195. if (tl == 0) { /* No tokens were included */
  196. #ifdef OS2
  197.     /* In OS/2 and Windows, allow for a disk letter like DOS */
  198.     if (isalpha(*atmbuf) && *(atmbuf+1) == ':')
  199.       return(-7);
  200. #endif /* OS2 */
  201.     if ((pmsg & 1) && !quiet) {
  202. bleep(BP_FAIL);
  203. printf("?No keywords match - %sn",atmbuf); /* cmkey */
  204.     }
  205.     return(cmflgs = -9);
  206. } else {
  207.     if (cmflgs == 1 || cmswitch) /* cmkey2 or cmswi */
  208.       return(cmflgs = -6);
  209.     else
  210.       return(cmflgs = -2);
  211.     /* The -6 code is to let caller try another table */
  212. }
  213. break;
  214.       default:
  215. #ifdef CK_RECALL
  216. if (test(table[z].flgs,CM_NOR)) no_recall = 1;
  217. #endif /* CK_RECALL */
  218. if (zz == 4)
  219.   swarg = 1;
  220. cmkwflgs = table[z].flgs;
  221. break;
  222.     }
  223.     return(y);
  224.   case 2: /* User terminated word with ESC */
  225.     debug(F101,"cmkey Esc cc","",cc);
  226.             if (cc == 0) {
  227. if (*xdef != NUL) {     /* Nothing in atmbuf */
  228.     printf("%s ",xdef); /* Supply default if any */
  229. #ifdef GEMDOS
  230.     fflush(stdout);
  231. #endif /* GEMDOS */
  232.     addbuf(xdef);
  233.     if (setatm(xdef,0) < 0) {
  234. printf("?Default too longn");
  235. return(-9);
  236.     }
  237.     inword = cmflgs = 0;
  238.     debug(F111,"cmkey: default",atmbuf,cc);
  239. } else {
  240.     debug(F101,"cmkey Esc pmsg","",0);
  241. #ifdef COMMENT
  242. /*
  243.   Chained FDBs...  The idea is that this function might not have a default,
  244.   but the next one might.  But if it doesn't, there is no way to come back to
  245.   this one.  To be revisited later...
  246. */
  247.     if (xcmfdb) /* Chained fdb -- try next one */
  248.       return(-3);
  249. #endif /* COMMENT */
  250.     if (pmsg & (1|4)) { /* So for now just beep */
  251. bleep(BP_WARN);
  252.     }
  253.     break;
  254. }
  255.             }
  256.     if (f) { /* If a conversion function is given */
  257. char * pp;
  258. zq = atxbuf; /* apply it */
  259. pp = atxbuf;
  260. atxn = CMDBL;
  261. if ((*f)(atmbuf,&zq,&atxn) < 0)
  262.   return(-2);
  263. if (!*pp)
  264.   pp = xdef;
  265. if (setatm(pp,0) < 0) {
  266.     printf("Evaluated keyword too longn");
  267.     return(-9);
  268. }
  269.     }
  270.     y = lookup(table,atmbuf,n,&z); /* Something in atmbuf */
  271.     debug(F111,"cmkey lookup y",atmbuf,y);
  272.     debug(F111,"cmkey lookup z",atmbuf,z);
  273.     if (y == -2 && z >= 0 && z < n) { /* Ambiguous */
  274. #ifndef NOPARTIAL
  275. int j, k, len = 9999; /* Do partial completion */
  276. /* Skip past any abbreviations in the table */
  277. for ( ; z < n; z++) {
  278.     if (table[z].flgs & CM_ABR == 0)
  279.       break;
  280.     if (!(table[z].flgs & CM_HLP) || (pmsg & 2))
  281.       break;
  282. }
  283. debug(F111,"cmkey partial z",atmbuf,z);
  284. debug(F111,"cmkey partial n",atmbuf,n);
  285. for (j = z+1; j < n; j++) {
  286.     debug(F111,"cmkey partial j",table[j].kwd,j);
  287.     if (ckstrcmp(atmbuf,table[j].kwd,cc,0))
  288.       break;
  289.     if (table[j].flgs & CM_ABR)
  290.       continue;
  291.     if ((table[j].flgs & CM_HLP) && !(pmsg & 2))
  292.       continue;
  293.     k = ckstrpre(table[z].kwd,table[j].kwd);
  294.     debug(F111,"cmkey partial k",table[z].kwd,k);
  295.     if (k < len)
  296.       len = k; /* Length of longest common prefix */
  297. }
  298. debug(F111,"cmkey partial len",table[z].kwd,len);
  299. if (len != 9999 && len > cc) {
  300.     strcat(atmbuf,table[z].kwd+cc);
  301.     atmbuf[len] = NUL;
  302.     printf("%s",atmbuf+cc);
  303.     strcat(cmdbuf,atmbuf+cc);
  304.     xc += (len - cc);
  305.     cc = len;
  306. }
  307. #endif /* NOPARTIAL */
  308. bleep(BP_WARN);
  309. break;
  310.     } else if (y == -3) {
  311. bleep(BP_WARN);
  312. break;
  313.     } else if (y == -1) { /* Not found */
  314. if ((pmsg & 1) && !quiet) {
  315.     bleep(BP_FAIL);
  316.     printf("?No keywords match - "%s"n",atmbuf);
  317. }
  318. cmflgs = -2;
  319. return(-9);
  320.     }
  321. /*
  322.   If we found it, but it's a help-only keyword and the "help" bit is not
  323.   set in pmsg, then not found.
  324. */
  325.     debug(F101,"cmkey flgs","",table[z].flgs);
  326.     if (test(table[z].flgs,CM_HLP) && ((pmsg & 2) == 0)) {
  327. if ((pmsg & 1) && !quiet) {
  328.     bleep(BP_FAIL);
  329.     printf("?No keywords match - %sn",atmbuf);
  330. }
  331. cmflgs = -2;
  332. return(-9);
  333.     }
  334. /*
  335.   See if the keyword just found has the CM_ABR bit set in its flgs field, and
  336.   if so, search forwards in the table for a keyword that has the same kwval
  337.   but does not have CM_ABR (or CM_INV?) set, and then expand using the full
  338.   keyword.  WARNING: This assumes that (a) keywords are in alphabetical order,
  339.   and (b) the CM_ABR bit is set only if the the abbreviated keyword is a true
  340.   abbreviation (left substring) of the full keyword.
  341. */
  342.     if (test(table[z].flgs,CM_ABR)) {
  343. int zz;
  344. for (zz = z+1; zz < n; zz++)
  345.   if ((table[zz].kwval == table[z].kwval) &&
  346.       (!test(table[zz].flgs,CM_ABR)) &&
  347.       (!test(table[zz].flgs,CM_INV))) {
  348.       z = zz;
  349.       break;
  350.   }
  351.     }
  352.     xp = table[z].kwd + cc;
  353.     if (cmswitch && test(table[z].flgs,CM_ARG)) {
  354. #ifdef VMS
  355. printf("%s=",xp);
  356. brkchar = '=';
  357. #else
  358. printf("%s:",xp);
  359. brkchar = ':';
  360. #endif /* VMS */
  361.     } else {
  362. printf("%s ",xp);
  363. brkchar = SP;
  364.     }
  365. #ifdef CK_RECALL
  366.     if (test(table[z].flgs,CM_NOR)) no_recall = 1;
  367. #endif /* CK_RECALL */
  368.     cmkwflgs = table[z].flgs;
  369. #ifdef GEMDOS
  370.     fflush(stdout);
  371. #endif /* GEMDOS */
  372.     addbuf(xp);
  373.     if (cmswitch && test(table[z].flgs,CM_ARG)) {
  374. bp--; /* Replace trailing space with : */
  375. #ifdef VMS
  376. *bp++ = '=';
  377. #else
  378. *bp++ = ':';
  379. #endif /* VMS */
  380. *bp = NUL;
  381. np = bp;
  382. swarg = 1;
  383.     }
  384.     inword = 0;
  385.     cmflgs = 0;
  386.     debug(F110,"cmkey: addbuf",cmdbuf,0);
  387.     return(y);
  388.   case 3: /* User typed "?" */
  389.     if (f) { /* If a conversion function is given */
  390. char * pp;
  391. zq = atxbuf; /* do the conversion now. */
  392. pp = atxbuf;
  393. atxn = CMDBL;
  394. if ((*f)(atmbuf,&zq,&atxn) < 0) return(-2);
  395. if (setatm(pp,0) < 0) {
  396.     printf("?Evaluated keyword too longn");
  397.     return(-9);
  398. }
  399.     }
  400.     y = lookup(table,atmbuf,n,&z); /* Look up what we have so far. */
  401.     if (y == -1) {
  402. cmflgs = -2;
  403. if ((pmsg & 1) && !quiet) {
  404.     bleep(BP_FAIL);
  405.     printf(" No keywords matchn");
  406.     return(-9);
  407. }
  408. return(-2);
  409.     }
  410.     if (*xhlp == NUL)
  411.       printf(" One of the following:n");
  412.     else
  413.       printf(" %s, one of the following:n",xhlp);
  414. #ifdef OLDHELP
  415.     if ((y > -1) &&
  416. !test(table[z].flgs,CM_ABR) &&
  417. ((z >= n-1) || ckstrcmp(table[z].kwd,table[z+1].kwd,cc,0))
  418. ) {
  419. printf(" %sn",table[z].kwd);
  420.     } else {
  421. clrhlp();
  422. for (i = 0; i < n; i++) {
  423.     if (!ckstrcmp(table[i].kwd,atmbuf,cc,0)
  424. && !test(table[i].flgs,CM_INV)
  425. )
  426.       addhlp(table[i].kwd);
  427. }
  428. dmphlp();
  429.     }
  430. #else  /* New way ... */
  431.     {
  432. int x;
  433. x = pmsg & (2|4); /* See kwdhelp() comments */
  434. if (atmbuf[0]) /* If not at beginning of field */
  435.   x |= 1; /* also show invisibles */
  436. kwdhelp(table,n,atmbuf,"","",1,x);
  437.     }
  438. #endif /* OLDHELP */
  439. #ifndef NOSPL
  440.     {
  441. extern int topcmd;
  442. if (tl > 0 && topcmd != XXHLP) /* This is bad... */
  443.   printf("or the name of a macro ("do ?" for a list)n");
  444.     }
  445. #endif /* NOSPL */
  446.     if (*atmbuf == NUL) {
  447. if (tl == 1)
  448.   printf("or the token %cn",*tok);
  449. else if (tl > 1)
  450.   printf("or one of the tokens: %sn",ckspread(tok));
  451.     }
  452.     printf("%s%s", cmprom, cmdbuf);
  453.     fflush(stdout);
  454.     break;
  455.   default:
  456.     printf("n%d - Unexpected return code from gtwordn",zz);
  457.     return(cmflgs = -2);
  458. }
  459. zz = gtword(0);
  460. debug(F111,"cmkey gtword zz",atmbuf,zz);
  461.     }
  462. }
  463. int
  464. chktok(tlist) char *tlist; {
  465.     char *p;
  466.     p = tlist;
  467.     while (*p != NUL && *p != *atmbuf) p++;
  468.     return((*p) ? (int) *p : 0);
  469. }
  470. /* Routines for parsing and converting dates and times */
  471. #define isdatesep(c) (c==SP||c=='-'||c=='/'||c=='.'||c=='_')
  472. #define istimesep(c) (c==':' || c=='P'||c=='p'||c=='A'||c=='a')
  473. char cmdatebuf[18] = { NUL, NUL };
  474. #define TU_DAYS   0
  475. #define TU_WEEKS  1
  476. #define TU_MONTHS 2
  477. #define TU_YEARS  3
  478. static struct keytab timeunits[] = {
  479.   { "days",   TU_DAYS,   0 },
  480.   { "months", TU_MONTHS, 0 },
  481.   { "weeks",  TU_WEEKS,  0 },
  482.   { "wks",    TU_WEEKS,  0 },
  483.   { "years",  TU_YEARS,  0 },
  484.   { "yrs",    TU_YEARS,  0 }
  485. };
  486. static int nunits = (sizeof(timeunits) / sizeof(struct keytab));
  487. /*  C M C V T D A T E  --  Converts free-form date to standard form.  */
  488. /*
  489.    Call with
  490.      s = pointer to free-format date-time
  491.      t = 0: return time only if time was given in s
  492.      t = 1: always return time (00:00:00 if no time given in s)
  493.      t = 2: allow time to be > 24:00:00
  494.    Returns:
  495.      -1 on failure, 0 or greater on success with result in cmdate[].
  496. */
  497. int
  498. cmcvtdate(s,t) char * s; int t; {
  499.     int rc = 0, x, i, hh, mm, ss, pmflag = 0, nodate = 0, len;
  500.     int units;
  501.     char * fld[3], * p;
  502.     char * year, * month = NULL, * day;
  503.     char * hour = "00", * min = "00", * sec = "00";
  504.     char tmpbuf[8];
  505.     char xbuf[32];
  506.     char ybuf[32];
  507.     char dbuf[26];
  508.     char daybuf[3];
  509.     char monbuf[3];
  510.     char yearbuf[5];
  511.     debug(F110,"cmcvtdate",s,0);
  512.     if (!s) s = "";
  513.     tmpbuf[0] = NUL;
  514.     while (*s == SP) s++; /* Gobble any leading blanks */
  515.     len = strlen(s);
  516.     if (len == 0) {
  517. s = ckdate();
  518. len = 17;
  519. debug(F110,"cmcvtdate now",s,0);
  520.     }
  521.     if (len > 30) { /* Check length of arg */
  522. debug(F101,"cmcvtdate date too long","",-1);
  523. return(-1);
  524.     }
  525.     ckstrncpy(xbuf,s,32); /* Make a local copy we can poke */
  526.     s = xbuf; /* Point to it */
  527.     s[len] = NUL;
  528. /* First we handle "today", "yesterday", "tomorrow", etc */
  529.     if (ckstrchr("+-TtYy",s[0])) {
  530. int i, k, n, minus = 0;
  531. long jd;
  532. jd = mjd(ckdate());
  533. debug(F111,"cmcvtdate NEW",s,jd);
  534. if (s[0] == '+' || s[0] == '-') { /* {+,-} <number> <timeunits> */
  535.     char * kp, * np;
  536.     char tmpyear[5];
  537.     char * dp;
  538.     if (s[0] == '-') /* Sign */
  539.       minus = 1;
  540.     kp = s+1; /* Skip intervening spaces */
  541.     while (*kp) {
  542. if (*kp == SP) kp++;
  543. else break;
  544.     }
  545.     if (!*kp)
  546.       return(-1);
  547.     np = kp; /* Number */
  548.     while (*kp) {
  549. if (isdigit(*kp)) kp++;
  550. else break;
  551.     }
  552.     if (!*kp)
  553.       return(-1);
  554.     n = atoi(np);
  555.     if (minus) n = 0 - n;
  556.     debug(F101,"cmcvtdate offset n","",n);
  557.     while (*kp) { /* Find end of number */
  558. if (isdigit(*kp)) kp++;
  559. else break;
  560.     }
  561.     while (*kp) { /* Skip spaces again */
  562. if (*kp == SP) kp++;
  563. else break;
  564.     }
  565.     debug(F110,"cmcvtdate unit start",kp,0);
  566.     p = kp; /* Units */
  567.     while (*p) {
  568. if (isalpha(*p)) {
  569.     p++;
  570. } else if (isdatesep(*p)) { /* Have a date separator */
  571.     *p++ = NUL;
  572.     break;
  573. } else
  574.   return(-1);
  575.     }
  576.     while (*p) { /* Skip any spaces */
  577. if (*p == SP) p++;
  578. else break;
  579.     }
  580.     debug(F110,"cmcvtdate time pointer",p,0);
  581.     debug(F110,"cmcvtdate unit",kp,0);
  582.     x = lookup(timeunits,kp,nunits,&k); /* Look up units */
  583.     if (x < 0) {
  584. debug(F111,"cmcvtdate lookup fails",kp,x);
  585. return(-1);
  586.     }
  587.     units = x;
  588.     debug(F111,"cmcvtdate offset units",timeunits[k].kwd,units);
  589.     switch (units) { /* Handle each unit */
  590.       case TU_DAYS: /* Days */
  591. jd += n;
  592. strcpy(ybuf,mjd2date(jd));
  593. if (*p) {
  594.     ybuf[8] = ' ';
  595.     strcpy(ybuf+9,p);
  596. } else
  597.   strcpy(ybuf+8," 00:00:00");
  598. s = ybuf;
  599. len = strlen(ybuf);
  600. debug(F111,"cmcvtdate days",s,len);
  601. goto normal;
  602.       case TU_WEEKS: /* Weeks */
  603. jd += (7 * n);
  604. strcpy(ybuf,mjd2date(jd));
  605. if (*p) {
  606.     ybuf[8] = ' ';
  607.     strcpy(ybuf+9,p);
  608. } else
  609.   strcpy(ybuf+8," 00:00:00");
  610. s = ybuf;
  611. len = strlen(ybuf);
  612. debug(F111,"cmcvtdate weeks",s,len);
  613. goto normal;
  614.       case TU_MONTHS: { /* Months */
  615.   char tmpmonth[3];
  616.   int xyear, xmonth;
  617.   dp = ckdate();
  618.   tmpyear[0] = dp[0];
  619.   tmpyear[1] = dp[1];
  620.   tmpyear[2] = dp[2];
  621.   tmpyear[3] = dp[3];
  622.   tmpyear[4] = NUL;
  623.   tmpmonth[0] = dp[4];
  624.   tmpmonth[1] = dp[5];
  625.   tmpmonth[2] = NUL;
  626.   xyear = atoi(tmpyear);
  627.   xmonth = atoi(tmpmonth);
  628.   xmonth += n;
  629.   xyear += (xmonth / 12);
  630.   xmonth = (xmonth % 12);
  631.   if (xmonth <= 0) {
  632.       xmonth += 12;
  633.       xyear--;
  634.   }
  635.   sprintf(ybuf,"%04d%02d%s",xyear,xmonth,dp+6);
  636.   if (*p) {
  637.       ybuf[8] = ' ';
  638.       strcpy(ybuf+9,p);
  639.   } else
  640.     strcpy(ybuf+8," 00:00:00");
  641.   s = ybuf;
  642.   len = strlen(ybuf);
  643.   debug(F111,"cmcvtdate months",s,len);
  644.   goto normal;
  645.       }
  646.       case TU_YEARS: { /* Years */
  647.   dp = ckdate();
  648.   tmpyear[0] = dp[0];
  649.   tmpyear[1] = dp[1];
  650.   tmpyear[2] = dp[2];
  651.   tmpyear[3] = dp[3];
  652.   tmpyear[4] = NUL;
  653.   sprintf(ybuf,"%04d%s",(atoi(tmpyear)+n),dp+4);
  654.   if (*p) {
  655.       ybuf[8] = ' ';
  656.       strcpy(ybuf+9,p);
  657.   } else
  658.     strcpy(ybuf+8," 00:00:00");
  659.   s = ybuf;
  660.   len = strlen(ybuf);
  661.   debug(F111,"cmcvtdate years",s,len);
  662.   goto normal;
  663.       }
  664.     }
  665.     return(-1);
  666. }
  667. i = ckstrpre(s,"today"); /* TODAY */
  668. if (i > 2 && (s[i] == NUL || isdatesep(s[i]))) {
  669.     strncpy(ybuf,ckdate(),8);
  670.     strcpy(ybuf+8," 00:00:00");
  671.     if (s[i])
  672.       strcpy(ybuf+8,s+i);
  673.     s = ybuf;
  674.     len = strlen(s);
  675.     debug(F111,"cmcvtdate today",s,len);
  676.     goto normal;
  677. }
  678. i = ckstrpre(s,"tomorrow"); /* TOMORROW */
  679. if (i > 2 && (s[i] == NUL || isdatesep(s[i]))) {
  680.     jd++;
  681.     strncpy(ybuf,mjd2date(jd),8);
  682.     strcpy(ybuf+8," 00:00:00");
  683.     if (s[i]) strcpy(ybuf+8,s+i);
  684.     s = ybuf;
  685.     len = strlen(s);
  686.     debug(F111,"cmcvtdate tomorrow",s,len);
  687.     goto normal;
  688. }
  689. i = ckstrpre(s,"yesterday"); /* YESTERDAY */
  690. if (i > 0 && (s[i] == NUL || isdatesep(s[i]))) {
  691.     jd--;
  692.     strncpy(ybuf,mjd2date(jd),8);
  693.     strcpy(ybuf+8," 00:00:00");
  694.     if (s[i]) strcpy(ybuf+8,s+i);
  695.     s = ybuf;
  696.     len = strlen(s);
  697.     debug(F111,"cmcvtdate yesterday",s,len);
  698.     goto normal;
  699. }
  700.     }
  701.   normal:
  702.     if (len >= 8) { /* Already in right format? */
  703. if (isdigit(s[0])  && isdigit(s[1])  &&
  704.     isdigit(s[2])  && isdigit(s[3])  &&
  705.     isdigit(s[4])  && isdigit(s[5])  &&
  706.     isdigit(s[6])  && isdigit(s[7]))
  707.   if (!s[8]) {
  708.       strcat(s," 00:00:00");
  709.       ckstrncpy(cmdatebuf,s,18);
  710.       debug(F111,"cmcvtdate yyyymmdd",s,rc);
  711.       return(rc);
  712.   } else if (len == 17 &&
  713.      isdigit(s[9])  && isdigit(s[10]) &&
  714.      isdigit(s[12]) && isdigit(s[13]) &&
  715.      isdigit(s[15]) && isdigit(s[16]) &&
  716.      s[11] == ':'   && s[14] == ':'   &&
  717.      (s[8] == SP || s[8] == '-' || s[8] == '_')) {
  718.       ckstrncpy(cmdatebuf,s,18);
  719.       debug(F111,"cmcvtdate yyyymmdd hh:mm:ss",s,rc);
  720.       return(rc);
  721. } else { /* We have a numeric date */
  722.     debug(F111,"cmcvtdate yyyymmdd xxx",s,rc);
  723.     p = s+9; /* Start of time field */
  724.     yearbuf[0] = s[0]; yearbuf[1] = s[1];
  725.     yearbuf[2] = s[2]; yearbuf[3] = s[3];
  726.     yearbuf[4] = NUL;  year = yearbuf;
  727.     monbuf[0] = s[4];  monbuf[1] = s[5];
  728.     monbuf[2] = NUL;   month = monbuf;
  729.     daybuf[0] = s[6];  daybuf[1] = s[7];
  730.     daybuf[2] = NUL;   day = daybuf;
  731.     goto dotime;
  732. }
  733.     }
  734.     fld[i = 0] = (p = s); /* First field */
  735.     while (*p) { /* Get next two fields */
  736. if (isdatesep(*p)) { /* Have a date separator */
  737.     *p++ = NUL; /* Replace by NUL */
  738.     if (*p) { /* Now we're at the next field */
  739. while (*p == SP) p++; /* Skip leading spaces */
  740. if (!*p) break; /* Make sure we still have something */
  741. if (i == 2) /* Last one? */
  742.   break;
  743. fld[++i] = p; /* No, record pointer to this one */
  744.     } else
  745.       break;
  746. } else if (istimesep(*p)) { /* Have a time separator */
  747.     if (isalpha(*p) && !isdigit(*(p-1))) { /* Might be letter */
  748. p++; /* in month name... */
  749. continue;
  750.     }
  751.     if (i != 0) { /* After a date */
  752. debug(F111,"cmcvtdate date bad timesep",p,-1);
  753. return(-1);
  754.     }
  755.     nodate = 1; /* Or without a date */
  756.     break;
  757. }
  758. p++;
  759.     }
  760.     if (p > s && i == 0) /* Make sure we have a date */
  761.       nodate = 1;
  762.     if (nodate) { /* No date */
  763. char *tmp; /* Substitute today's date */
  764. ztime(&tmp);
  765. if (!tmp) {
  766.     debug(F101,"cmcvtdate null ztime","",-1);
  767.     return(-1);
  768. }
  769. if (!*tmp) {
  770.     debug(F101,"cmcvtdate emtpy ztime","",-1);
  771.     return(-1);
  772. }
  773. ckstrncpy(dbuf,tmp,26); /* Reformat */
  774. if (dbuf[8] == SP) dbuf[8] = '0';
  775. fld[0] = dbuf+8;
  776. dbuf[10] = NUL;
  777. fld[1] = dbuf+4;
  778. dbuf[7] = NUL;
  779. fld[2] = dbuf+20;
  780. dbuf[24] = NUL;
  781. p = s; /* Back up source pointer to reparse */
  782.     } else if (i != 2) {
  783. debug(F101,"cmcvtdate fail A","",-1);
  784. return(-1);
  785.     }
  786.     if (!rdigits(fld[0])) { /* Now parse the date */
  787. debug(F101,"cmcvtdate fail B","",-1);
  788. return(-1);
  789.     }
  790.     if (!rdigits(fld[1])) {
  791. if ((x = lookup(cmonths,fld[1],12,NULL)) < 0) {
  792.     debug(F101,"cmcvtdate fail C","",-1);
  793.     return(-1);
  794. }
  795. sprintf(tmpbuf,"%02d",x);
  796. month = tmpbuf;
  797.     }
  798.     if (((int)strlen(fld[0]) == 4)) { /* yyyy-xx-dd */
  799. year = fld[0];
  800. day = fld[2];
  801. if (!month)
  802.   month = fld[1]; /* yyyy-mm-dd */
  803.     } else if (((int)strlen(fld[2]) == 4)) { /* xx-xx-yyyy */
  804. year = fld[2];
  805. if (month) { /* dd-name-yyyy */
  806.     day = fld[0];
  807. } else { /* xx-xx-yyyy */
  808.     int f0, f1;
  809.     f0 = atoi(fld[0]);
  810.     f1 = atoi(fld[1]);
  811.     if ((f0 > 12) && (f1 <= 12)) {
  812. day = fld[0]; /* mm-dd-yyyy */
  813. month = fld[1];
  814.     } else if ((f0 <= 12) && (f1 > 12)) {
  815. if (!rdigits(fld[1]))
  816.   return(-1);
  817. else
  818.   day = fld[1]; /* dd-mm-yyyy */
  819. month = fld[0];
  820. #ifdef COMMENT
  821.     } else if ((f0 <= 12) && (f1 <= 12)) {
  822. if (!quiet)
  823.   printf("?Day and month are ambiguous - "%s"n", o);
  824. return(-9);
  825. #endif /* COMMENT */
  826.     } else {
  827. debug(F101,"cmcvtdate fail D","",-1);
  828. return(-1);
  829.     }
  830. }
  831.     } else {
  832. debug(F101,"cmcvtdate fail E","",-1);
  833. return(-1);
  834.     }
  835.     x = atoi(month);
  836.     sprintf(tmpbuf,"%02d",x); /* 2-digit numeric month */
  837.   dotime:
  838.     debug(F110,"cmcvtdate dotime s",s,0);
  839.     debug(F110,"cmcvtdate dotime p",p,0);
  840.     if ((x  = atoi(day)) > 31) {
  841. debug(F101,"cmcvtdate fail K","",-1);
  842. return(-1);
  843.     }
  844.     if (!*p && t == 0) {
  845. sprintf(cmdatebuf,"%s%s%02d",year,month,x);
  846. return(0);
  847.     }
  848.     fld[i = 0] = *p ? p : "00"; /* First time field */
  849.     fld[1] = "00";
  850.     fld[2] = "00";
  851.     while (*p) { /* Get the rest, if any */
  852. if (istimesep(*p)) {
  853.     debug(F000,"cmcvtdate timesep:",p,*p);
  854.     if (*p == 'P' || *p == 'p') {
  855. if (*(p+1) != 'M' && *(p+1) != 'm') {
  856.     debug(F101,"cmcvtdate fail F","",-1);
  857.     return(-1);
  858. } else if (!*(p+2)) {
  859.     pmflag = 1;
  860.     *p = NUL;
  861.     break;
  862. } else {
  863.     debug(F101,"cmcvtdate fail F","",-1);
  864.     return(-1);
  865. }
  866.     } else if (*p == 'A' || *p == 'a') {
  867. if (*(p+1) != 'M' && *(p+1) != 'm') {
  868.     debug(F101,"cmcvtdate fail F","",-1);
  869.     return(-1);
  870. }
  871. if (*(p+2)) {
  872.     debug(F101,"cmcvtdate fail F","",-1);
  873.     return(-1);
  874. }
  875. *p = NUL;
  876. break;
  877.     }
  878.     *p++ = NUL;
  879.     if (*p) {
  880. while (*p == SP) p++;
  881. if (!*p) break;
  882. if (i == 2)
  883.   break;
  884. fld[++i] = p;
  885.     } else
  886.       break;
  887. }
  888. p++;
  889.     }
  890.     debug(F101,"cmcvtdate time i","",i);
  891.     debug(F110,"cmcvtdate time fld[0]",fld[0],0);
  892.     debug(F110,"cmcvtdate time fld[1]",fld[1],0);
  893.     debug(F110,"cmcvtdate time fld[2]",fld[2],0);
  894.     if (!rdigits(fld[0]))
  895.       return(-1);
  896.     hour = fld[0];
  897.     if (i >= 1) {
  898. if (!rdigits(fld[1]))
  899.   return(-1);
  900. else
  901.   min = fld[1];
  902.     }
  903.     if (i == 2) {
  904. if (!rdigits(fld[2]))
  905.   return(-1);
  906. else
  907.   sec = fld[2];
  908.     }
  909.     hh = atoi(hour);
  910.     if (pmflag && hh <= 11)
  911.       hh += 12;
  912.     if ((t != 2 && hh > 24) || hh < 0) {
  913. debug(F101,"cmcvtdate fail G","",-1);
  914. return(-1);
  915.     }
  916.     if ((mm = atoi(min)) > 59) {
  917. debug(F101,"cmcvtdate fail H","",-1);
  918. return(-1);
  919.     }
  920.     if ((ss = atoi(sec)) > 59) {
  921. debug(F101,"cmcvtdate fail I","",-1);
  922. return(-1);
  923.     }
  924.     if (mm < 0 || ss < 0) return(-1);
  925.     if (t != 2 && (ss > 0 || mm > 0) && hh > 23) {
  926. debug(F101,"cmcvtdate fail J","",-1);
  927. return(-1);
  928.     }
  929.     debug(F110,"cmcvtdate year",year,0);
  930.     debug(F110,"cmcvtdate month",month,0);
  931.     debug(F101,"cmcvtdate x","",x);
  932.     debug(F101,"cmcvtdate hh","",hh);
  933.     debug(F101,"cmcvtdate mm","",mm);
  934.     debug(F101,"cmcvtdate ss","",ss);
  935.     sprintf(cmdatebuf,"%s%s%02d %02d:%02d:%02d",year,month,x,hh,mm,ss);
  936. #ifdef DEBUG
  937.     if (deblog) {
  938. debug(F101,"cmcvtdate hour","",hh);
  939. debug(F101,"cmcvtdate minute","",mm);
  940. debug(F101,"cmcvtdate second","",ss);
  941. debug(F111,"cmcvtdate result",cmdatebuf,rc);
  942.     }
  943. #endif /* DEBLOG */
  944.     return(0);
  945. }
  946. /*  C K C V T D A T E  --  Like cmcvtdate(), but returns string.  */
  947. /*  For use by date-related functions */
  948. /*  See calling conventions for cmcvtdate() above. */
  949. char *
  950. ckcvtdate(p,t) char * p; int t; {
  951.     if (cmcvtdate(p,t) < 0)
  952.       return("<BAD_DATE_OR_TIME>"); /* fblah() error message */
  953.     else
  954.       return((char *) cmdatebuf);
  955. }
  956. /*  C M D A T E  --  Parse a date and/or time  */
  957. /*
  958.   Accepts date in various formats.  If the date is recognized,
  959.   this routine returns 0 or greater with the result string pointer
  960.   pointing to a buffer containing the date as "yyyymmdd hh:mm:ss".
  961. */
  962. int
  963. cmdate(xhlp,xdef,xp,quiet,f) char *xhlp, *xdef, **xp; int quiet; xx_strp f; {
  964.     int /* i, */ x, rc;
  965.     char *o, *s, *zq;
  966.     cmfldflgs = 0;
  967.     if (!xhlp) xhlp = "";
  968.     if (!xdef) xdef = "";
  969.     if (!*xhlp) xhlp = "Date and/or time";
  970.     *xp = "";
  971.     rc = cmfld(xhlp,xdef,&s,(xx_strp)0);
  972.     debug(F101,"cmdate cmfld rc","",rc);
  973.     if (rc < 0)
  974.       return(rc);
  975.     debug(F110,"cmdate 1",s,0);
  976.     o = s; /* Remember what they typed. */
  977.     s = brstrip(s);
  978.     debug(F110,"cmdate 2",s,0);
  979.     x = 0;
  980.     if (f) { /* If a conversion function is given */
  981. char * pp;
  982. zq = atxbuf; /* do the conversion. */
  983. pp = atxbuf;
  984. atxn = CMDBL;
  985. if ((x = (*f)(s,&zq,&atxn)) < 0) return(-2);
  986. if (!*pp)
  987.   pp = xdef;
  988. if (setatm(pp,0) < 0) {
  989.     if (!quiet) printf("?Evaluated date too longn");
  990.     return(-9);
  991. }
  992. s = atxbuf;
  993.     }
  994.     rc = cmcvtdate(s,1);
  995.     if (rc < 0) {
  996. if (!quiet) printf("Invalid date or time - "%s"n", o);
  997. return(-9);
  998.     }
  999.     *xp = cmdatebuf;
  1000.     return(rc);
  1001. }
  1002. #ifdef CK_RECALL /* Command-recall functions */
  1003. /*  C M R I N I  --  Initialize or change size of command recall buffer */
  1004. int
  1005. cmrini(n) int n; {
  1006.     int i;
  1007.     if (recall && in_recall) { /* Free old storage, if any */
  1008. for (i = 0; i < cm_recall; i++) {
  1009.     if (recall[i]) {
  1010. free(recall[i]);
  1011. recall[i] = NULL;
  1012.     }
  1013. }
  1014. free(recall);
  1015. recall = NULL;
  1016.     }
  1017.     cm_recall = n; /* Set new size */
  1018.     rlast = current = -1; /* Initialize pointers */
  1019.     if (n > 0) {
  1020. recall = (char **)malloc((cm_recall + 1) * sizeof(char *));
  1021. if (!recall)
  1022.   return(1);
  1023. for (i = 0; i < cm_recall; i++) {
  1024.     recall[i] = NULL;
  1025. }
  1026. in_recall = 1; /* Recall buffers init'd */
  1027.     }
  1028.     return(0);
  1029. }
  1030. /*  C M A D D N E X T  --  Force addition of next command */
  1031. VOID
  1032. cmaddnext() {
  1033.     if (on_recall && in_recall) { /* Even if it doesn't come */
  1034. force_add = 1; /* from the keyboard */
  1035. no_recall = 0;
  1036.     }
  1037. }
  1038. /*  C M G E T C M D  --  Find most recent matching command  */
  1039. char *
  1040. cmgetcmd(s) char * s; {
  1041.     int i;
  1042.     for (i = current; i >= 0; i--) { /* Search backward thru history list */
  1043. if (!recall[i]) continue; /* This one's null, skip it */
  1044. if (ckmatch(s,recall[i],0,1)) /* Match? */
  1045.   return(recall[i]); /* Yes, return pointer */
  1046.     }
  1047.     return(NULL); /* No match, return NULL pointer */
  1048. }
  1049. /*  A D D C M D  --  Add a command to the recall buffer  */
  1050. VOID
  1051. addcmd(s) char * s; {
  1052.     int len;
  1053.     if (!s) s = cmdbuf;
  1054.     len = strlen(s);
  1055. #ifdef CKSYSLOG
  1056.     /* Log all interactive commands */
  1057.     /* Logging macros & TAKE files is way too much */
  1058.     if (ckxlogging) {
  1059. if (ckxsyslog >= SYSLG_CX || ckxsyslog >= SYSLG_CM && !cmdsrc())
  1060.   cksyslog(SYSLG_CX, 1, "command", s, NULL);
  1061.     }
  1062. #endif /* CKSYSLOG */
  1063.     if ((!cmdsrc() || force_add) && /* Reading commands from keyboard? */
  1064. (on_recall) && /* Recall is turned on? */
  1065. (cm_recall > 0) && /* Saving commands? */
  1066. !no_recall && /* Not not saving this command? */
  1067. len > 0) { /* Non-null command? */
  1068. force_add = 0;
  1069.         if (rlast >= cm_recall - 1) { /* Yes, buffer full? */
  1070.     int i; /* Yes. */
  1071.     if (recall[0]) { /* Discard oldest command */
  1072. free(recall[0]);
  1073. recall[0] = NULL;
  1074.     }
  1075.     for (i = 0; i < rlast; i++) { /* The rest */
  1076. recall[i] = recall[i+1]; /* move back */
  1077.     }
  1078.     rlast--; /* Now we have one less */
  1079. }
  1080.         rlast++; /* Index of last command in buffer */
  1081. current = rlast; /* Also now the current command */
  1082. if (current >= cm_recall) {
  1083.     printf("Oops, command recall errorn");
  1084. } else {
  1085.     recall[current] = malloc(len+1);
  1086.     if (recall[current])
  1087.       strcpy(recall[current],s);
  1088. }
  1089.     }
  1090. }
  1091. #endif /* CK_RECALL */
  1092. int
  1093. cmgetlc(s) char * s; { /* Get leading char */
  1094.     char c;
  1095.     while ((c = *s++) <= SP) ;
  1096.     return(c);
  1097. }
  1098. /*  C M C F M  --  Parse command confirmation (end of line)  */
  1099. /*
  1100.  Returns
  1101.    -2: User typed anything but whitespace or newline
  1102.    -1: Reparse needed
  1103.     0: Confirmation was received
  1104. */
  1105. int
  1106. cmcfm() {
  1107.     int x, xc;
  1108.     debug(F101,"cmcfm: cmflgs","",cmflgs);
  1109.     debug(F110,"cmcfm: atmbuf",atmbuf,0);
  1110.     inword = xc = cc = 0;
  1111.     setatm("",0); /* (Probably unnecessary) */
  1112.     while (cmflgs != 1) {
  1113.         x = gtword(0);
  1114.         xc += cc;
  1115.         switch (x) {
  1116.   case -9:
  1117.     printf("Command or field too longn");
  1118.   case -4: /* EOF */
  1119.   case -2:
  1120.   case -1:
  1121.     return(x);
  1122.   case 1: /* End of line */
  1123.     if (xc > 0) {
  1124. if (xcmfdb) {
  1125.     return(-6);
  1126. } else {
  1127.     printf("?Not confirmed - %sn",atmbuf);
  1128.     return(-9);
  1129. }
  1130.     } else break; /* Finish up below */
  1131.   case 2: /* ESC */
  1132.     if (xc == 0) {
  1133. bleep(BP_WARN);
  1134. continue; /* or fall thru. */
  1135.     }
  1136.   case 0: /* Space */
  1137.     if (xc == 0) /* If no chars typed, continue, */
  1138.       continue; /* else fall thru. */
  1139.     /* else fall thru... */
  1140.   case 3: /* Question mark */
  1141.     if (xc > 0) {
  1142. if (xcmfdb) {
  1143.     return(-6);
  1144. } else {
  1145.     printf("?Not confirmed - %sn",atmbuf);
  1146.     return(-9);
  1147. }
  1148.     }
  1149.     printf(
  1150.        "n Press the Return or Enter key to confirm the commandn");
  1151.     printf("%s%s",cmprom,cmdbuf);
  1152.     fflush(stdout);
  1153.     continue;
  1154. }
  1155.     }
  1156. #ifdef CK_RECALL
  1157.     addcmd(cmdbuf);
  1158. #endif /* CK_RECALL */
  1159.     return(0);
  1160. }
  1161. /* The following material supports chained parsing functions. */
  1162. /* See ckucmd.h for FDB and OFDB definitions. */
  1163. struct OFDB cmresult = { /* Universal cmfdb result holder */
  1164.     NULL,
  1165.     0,
  1166.     NULL,
  1167.     0
  1168. };
  1169. VOID
  1170. cmfdbi(p,fc,s1,s2,s3,n1,n2,f,k,nxt) /* Initialize an FDB */
  1171.     struct FDB * p;
  1172.     int fc;
  1173.     char * s1, * s2, * s3;
  1174.     int n1, n2;
  1175.     xx_strp f;
  1176.     struct keytab * k;
  1177.     struct FDB * nxt; {
  1178.     p->fcode = fc;
  1179.     p->hlpmsg = s1;
  1180.     p->dflt = s2;
  1181.     p->sdata = s3;
  1182.     p->ndata1 = n1;
  1183.     p->ndata2 = n2;
  1184.     p->spf = f;
  1185.     p->kwdtbl = k;
  1186.     p->nxtfdb = nxt;
  1187. }
  1188. /*  C M F D B  --  Parse a field with several possible functions  */
  1189. int
  1190. cmfdb(fdbin) struct FDB * fdbin; {
  1191. #ifndef NOSPL
  1192.     extern int x_ifnum;                 /* IF NUMERIC - disables warnings */
  1193. #endif /* NOSPL */
  1194.     struct FDB * in = fdbin;
  1195.     struct OFDB * out = &cmresult;
  1196.     int x = 0, n;
  1197.     char *s, *xp, *m = NULL;
  1198.     int errbits = 0;
  1199.     xp = bp;
  1200.     out->fcode = -1; /* Initialize output struct */
  1201.     out->fdbaddr = NULL;
  1202.     out->sresult = NULL;
  1203.     out->nresult = 0;
  1204. /*
  1205.   Currently we make one trip through the FDBs.  So if the user types Esc or
  1206.   Tab at the beginning of a field, only the first FDB is examined for a
  1207.   default.  If the user types ?, help is given only for one FDB.  We should
  1208.   search through the FDBs for all matching possibilities -- and in particular
  1209.   display the pertinent context-sensitive help for each function, rather than
  1210.   the only the first one that works, and then rewind the FDB pointer so we
  1211.   are not locked out of the earlier ones.
  1212. */
  1213.     cmfldflgs = 0;
  1214.     while (1) { /* Loop through the chain of FDBs */
  1215. nomsg = 1;
  1216. xcmfdb = 1;
  1217. s = NULL;
  1218. n = 0;
  1219. debug(F101,"cmfdb in->fcode","",in->fcode);
  1220. switch (in->fcode) { /* Current parsing function code */
  1221.   case _CMNUM:
  1222. #ifndef NOSPL
  1223.             x_ifnum = 1;                /* Disables warning messages */
  1224. #endif /* NOSPL */
  1225.     x = cmnum(in->hlpmsg,in->dflt,10,&n,in->spf);
  1226. #ifndef NOSPL
  1227.             x_ifnum = 0;
  1228. #endif /* NOSPL */
  1229.     debug(F101,"cmfdb cmnum","",x);
  1230.     if (x < 0) errbits |= 1;
  1231.     break;
  1232.   case _CMOFI:
  1233.     x = cmofi(in->hlpmsg,in->dflt,&s,in->spf);
  1234.     debug(F101,"cmfdb cmofi","",x);
  1235.     if (x < 0) errbits |= 2;
  1236.     break;
  1237.   case _CMIFI:
  1238.     x = cmifi2(in->hlpmsg,
  1239.        in->dflt,
  1240.        &s,
  1241.        &n,
  1242.        in->ndata1,
  1243.        in->sdata,
  1244.        in->spf,
  1245.        in->ndata2
  1246.        );
  1247.     debug(F101,"cmfdb cmifi2 x","",x);
  1248.     debug(F101,"cmfdb cmifi2 n","",n);
  1249.     if (x < 0) errbits |= 4;
  1250.     break;
  1251.   case _CMFLD:
  1252.     cmfldflgs = in->ndata1;
  1253.     x = cmfld(in->hlpmsg,in->dflt,&s,in->spf);
  1254.     debug(F101,"cmfdb cmfld","",x);
  1255.     if (x < 0) errbits |= 8;
  1256.     break;
  1257.   case _CMTXT:
  1258.     x = cmtxt(in->hlpmsg,in->dflt,&s,in->spf);
  1259.     debug(F101,"cmfdb cmtxt","",x);
  1260.     if (x < 0) errbits |= 16;
  1261.     break;
  1262.   case _CMKEY:
  1263.     x = cmkey2(in->kwdtbl,
  1264.        in->ndata1,
  1265.        in->hlpmsg,in->dflt,in->sdata,in->spf,in->ndata2);
  1266.     debug(F101,"cmfdb cmkey","",x);
  1267.     if (x < 0) errbits |= ((in->ndata2 & 4) ? 32 : 64);
  1268.     break;
  1269.   case _CMCFM:
  1270.     x = cmcfm();
  1271.     debug(F101,"cmfdb cmcfm","",x);
  1272.     if (x < 0) errbits |= 128;
  1273.     break;
  1274.   default:
  1275.     debug(F101,"cmfdb - unexpected function code","",in->fcode);
  1276.     printf("?cmfdb - unexpected function code: %dn",in->fcode);
  1277. }
  1278. debug(F101,"cmfdb x","",x);
  1279. debug(F101,"cmfdb cmflgs","",cmflgs);
  1280. debug(F101,"cmfdb crflag","",crflag);
  1281. debug(F101,"cmfdb qmflag","",qmflag);
  1282. debug(F101,"cmfdb esflag","",esflag);
  1283. if (x > -1) { /* Success */
  1284.     out->fcode = in->fcode; /* Fill in output struct */
  1285.     out->fdbaddr = in;
  1286.     out->sresult = s;
  1287.     out->nresult = (in->fcode == _CMKEY) ? x : n;
  1288.     out->kflags = (in->fcode == _CMKEY) ? cmkwflgs : 0;
  1289.     debug(F101,"cmfdb out->nresult","",out->nresult);
  1290.     nomsg = 0;
  1291.     xcmfdb = 0;
  1292.     debug(F111,"cmfdb cmdbuf & crflag",cmdbuf,crflag);
  1293.     if (crflag) {
  1294. cmflgs = 1;
  1295. #ifdef CK_RECALL
  1296. debug(F101,"cmfdb code","",in->fcode);
  1297. if (in->fcode != _CMCFM)
  1298.   addcmd(cmdbuf);
  1299. #endif /* CK_RECALL */
  1300.     }
  1301.     return(x); /* and return */
  1302. }
  1303. in = in->nxtfdb; /* Failed, get next parsing function */
  1304. nomsg = 0;
  1305. xcmfdb = 0;
  1306. if (!in) { /* No more */
  1307.     debug(F101,"cmfdb failure x","",x);
  1308.     debug(F101,"cmfdb failure errbits","",errbits);
  1309.     if (x == -6)
  1310.       x = -9;
  1311.     if (x == -9) {
  1312. /* Make informative messages for a few common cases */
  1313. switch (errbits) {
  1314.   case 4+32: m = "Does not match filename or switch"; break;
  1315.   case 4+64: m = "Does not match filename or keyword"; break;
  1316.   case 1+32: m = "Not a number or valid keyword"; break;
  1317.   case 1+64: m = "Not a number or valid switch"; break;
  1318.   default: m = "Not valid in this position";
  1319. }
  1320. printf("?%s: "%s"n",m, atmbuf);
  1321.     }
  1322.     return(x);
  1323. }
  1324. if (x != -2 && x != -6 && x != -9 && x != -3) /* Editing or somesuch */
  1325.   return(x); /* Go back and reparse */
  1326. pp = np = bp = xp; /* Back up pointers */
  1327. cmflgs = -1; /* Force a reparse */
  1328. #ifndef NOSPL
  1329. if (!askflag) { /* If not executing ASK-class cmd... */
  1330. #endif /* NOSPL */
  1331.     if (crflag) { /* If CR was typed, put it back */
  1332. pushc = LF; /* But as a linefeed */
  1333.     } else if (qmflag) { /* Ditto for Question mark */
  1334. pushc = '?';
  1335.     } else if (esflag) { /* and Escape or Tab */
  1336. pushc = ESC;
  1337.     }
  1338. #ifndef NOSPL
  1339. }
  1340. #endif /* NOSPL */
  1341.     }
  1342. }
  1343. #ifdef OLDHELP
  1344. /* Keyword help routines */
  1345. /*  C L R H L P -- Initialize/Clear the help line buffer  */
  1346. static VOID
  1347. clrhlp() {                              /* Clear the help buffer */
  1348.     hlpbuf[0] = NUL;
  1349.     hh = hx = 0;
  1350. }
  1351. /*  A D D H L P  --  Add a string to the help line buffer  */
  1352. static VOID
  1353. addhlp(s) char *s; {                    /* Add a word to the help buffer */
  1354.     int j;
  1355.     hh++;                               /* Count this column */
  1356.     for (j = 0; (j < hc) && (*s != NUL); j++) { /* Fill the column */
  1357.         hlpbuf[hx++] = *s++;
  1358.     }
  1359.     if (*s != NUL)                      /* Still some chars left in string? */
  1360.         hlpbuf[hx-1] = '+';             /* Mark as too long for column. */
  1361.     if (hh < (hw / hc)) {               /* Pad col with spaces if necessary */
  1362.         for (; j < hc; j++) {
  1363.             hlpbuf[hx++] = SP;
  1364.         }
  1365.     } else {                            /* If last column, */
  1366.         hlpbuf[hx++] = NUL;             /* no spaces. */
  1367.         dmphlp();                       /* Print it. */
  1368.         return;
  1369.     }
  1370. }
  1371. /*  D M P H L P  --  Dump the help line buffer  */
  1372. static VOID
  1373. dmphlp() {                              /* Print the help buffer */
  1374.     hlpbuf[hx++] = NUL;
  1375.     if ( hlpbuf[0] )
  1376.        printf(" %sn",hlpbuf);
  1377.     clrhlp();
  1378. }
  1379. #endif /* OLDHELP */
  1380. /*  G T W O R D  --  Gets a "word" from the command input stream  */
  1381. /*
  1382. Usage: retcode = gtword(brk);
  1383.   brk = 0 for normal word breaks (space, CR, Esc, ?)
  1384.   brk = 1 to add / : = (for parsing switches)
  1385. Returns:
  1386. -10 Timelimit set and timed out
  1387.  -9 if input was too long
  1388.  -4 if end of file (e.g. pipe broken)
  1389.  -3 if null field
  1390.  -2 if command buffer overflows
  1391.  -1 if user did some deleting
  1392.   0 if word terminates with SP or tab
  1393.   1 if ... CR
  1394.   2 if ... ESC
  1395.   3 if ... ? (question mark)
  1396.   4 if ... : or = and called with brk != 0
  1397. With:
  1398.   pp pointing to beginning of word in buffer
  1399.   bp pointing to after current position
  1400.   atmbuf containing a copy of the word
  1401.   cc containing the number of characters in the word copied to atmbuf
  1402. */
  1403. int
  1404. ungword() { /* Unget a word */
  1405.     debug(F101,"ungword cmflgs","",cmflgs);
  1406.     if (ungw) return(0);
  1407.     cmfsav = cmflgs;
  1408.     ungw = 1;
  1409.     cmflgs = 0;
  1410.     return(0);
  1411. }
  1412. /* Un-un-get word.  Undo ungword() if it has been done. */
  1413. VOID
  1414. unungw() {
  1415.     debug(F110,"unungw atmbuf",atmbuf,cmflgs);
  1416.     if (ungw) {
  1417. ungw = 0;
  1418. cmflgs = cmfsav;
  1419. atmbuf[0] = NUL;
  1420.     }
  1421. }
  1422. static int
  1423. gtword(brk) int brk; {
  1424.     int c;                              /* Current char */
  1425.     int quote = 0;                      /* Flag for quote character */
  1426.     int echof = 0;                      /* Flag for whether to echo */
  1427.     int comment = 0; /* Flag for in comment */
  1428.     char *cp = NULL; /* Comment pointer */
  1429.     int eintr = 0;
  1430.     int bracelvl = 0; /* nested brace counter [jrs] */
  1431.     int iscontd = 0;
  1432.     char lastchar = NUL;
  1433.     char prevchar = NUL;
  1434.     char lbrace, rbrace;
  1435. #ifdef RTU
  1436.     extern int rtu_bug;
  1437. #endif /* RTU */
  1438. #ifdef IKSD
  1439.     extern int inserver;
  1440. #endif /* IKSD */
  1441. #ifdef CK_LOGIN
  1442.     extern int x_logged;
  1443. #endif /* CK_LOGIN */
  1444.     extern int kstartactive;
  1445. #ifdef datageneral
  1446.     extern int termtype;                /* DG terminal type flag */
  1447.     extern int con_reads_mt;            /* Console read asynch is active */
  1448.     if (con_reads_mt) connoi_mt();      /* Task would interfere w/cons read */
  1449. #endif /* datageneral */
  1450.     if (cmfldflgs & 1) {
  1451. lbrace = '(';
  1452. rbrace = ')';
  1453.     } else {
  1454. lbrace = '{';
  1455. rbrace = '}';
  1456.     }
  1457.     crflag = 0;
  1458.     qmflag = 0;
  1459.     esflag = 0;
  1460.     if (swarg) { /* No leading space for switch args */
  1461. inword = 1;
  1462. swarg = 0;
  1463.     }
  1464.     debug(F000,"gtword brkchar","",brkchar);
  1465.     debug(F101,"gtword brk","",brk);
  1466.     if (ungw) { /* Have a word saved? */
  1467. int x;
  1468. debug(F110,"gtword ungetting from pp",pp,0);
  1469. while (*pp++ == SP) ;
  1470. if (setatm(pp,2) < 0) {
  1471.     printf("?Saved word too longn");
  1472.     return(-9);
  1473. }
  1474. strncpy(atmbuf,pp,ATMBL);
  1475. atmbuf[ATMBL] = NUL;
  1476. x = strlen(atmbuf);
  1477. while (x > 0 && atmbuf[x-1] == SP) /* Trim trailing spaces */
  1478.   atmbuf[--x] = NUL;
  1479. ungw = 0;
  1480. cmflgs = cmfsav;
  1481. debug(F111,"gtword returning atmbuf",atmbuf,cmflgs);
  1482. return(cmflgs);
  1483.     }
  1484.     pp = np;                            /* Start of current field */
  1485.     debug(F110,"gtword cmdbuf",cmdbuf,0);
  1486.     debug(F110,"gtword bp",bp,0);
  1487.     debug(F110,"gtword pp",pp,0);
  1488.     while (bp < cmdbuf+CMDBL) {         /* Big get-a-character loop */
  1489. echof = 0; /* Assume we don't echo because */
  1490. chsrc = 0; /* character came from reparse buf. */
  1491. #ifdef BS_DIRSEP
  1492. CMDIRPARSE:
  1493. #endif /* BS_DIRSEP */
  1494. c = *bp;
  1495.         if (!c) { /* If no char waiting in reparse buf */
  1496.     if (dpx && (!pushc
  1497. #ifndef NOSPL
  1498. || askflag
  1499. #endif /* NOSPL */
  1500. )) /* get from tty, set echo flag */
  1501.       echof = 1;
  1502.     debug(F101,"gtword timelimit","",timelimit);
  1503.     c = cmdgetc(timelimit); /* Read a command character. */
  1504.     debug(F101,"gtword c","",c);
  1505.     debug(F111,"gtword dpx,echof",ckitoa(dpx),echof);
  1506.     if (timelimit && c < -1) { /* Timed out */
  1507. return(-10);
  1508.     }
  1509. #ifndef NOXFER
  1510. /*
  1511.   The following allows packet recognition in the command parser.
  1512.   Presently it works only for Kermit packets, and if our current protocol
  1513.   happens to be anything besides Kermit, we simply force it to Kermit.
  1514.   We don't use the APC mechanism here for mechanical reasons, and also
  1515.   because this way, it works even with minimally configured interactive
  1516.   versions.  Add Zmodem later...
  1517. */
  1518. #ifdef CK_AUTODL
  1519.     if (!local && cmdadl /* Autodownload enabled? */
  1520. #ifdef IKS_OPTION
  1521. || TELOPT_SB(TELOPT_KERMIT).kermit.me_start
  1522. #endif /* IKS_OPTION */
  1523. ) {
  1524. int k;
  1525. k = kstart((CHAR)c); /* Kermit S or I packet? */
  1526. if (k) {
  1527.     int ksign = 0;
  1528.     if (k < 0) { /* Minus-Protocol? */
  1529. #ifdef NOSERVER
  1530. goto noserver; /* Need server mode for this */
  1531. #else
  1532. ksign = 1; /* Remember */
  1533. k = 0 - k; /* Convert to actual protocol */
  1534. justone = 1; /* Flag for protocol module */
  1535. #endif /* NOSERVER */
  1536.     } else
  1537.       justone = 0;
  1538.     k--; /* Adjust kstart's return value */
  1539.     if (k == PROTO_K) {
  1540. extern int protocol, g_proto;
  1541. extern CHAR sstate;
  1542. g_proto = protocol;
  1543. protocol = PROTO_K; /* Crude... */
  1544. sstate = ksign ? 'x' : 'v';
  1545. cmdbuf[0] = NUL;
  1546. return(-3);
  1547.     }
  1548. }
  1549.     }
  1550. #ifdef NOSERVER
  1551.   noserver:
  1552. #endif /* NOSERVER */
  1553. #endif /* CK_AUTODL */
  1554. #endif /* NOXFER */
  1555.     chsrc = 1; /* Remember character source is tty. */
  1556.     brkchar = c;
  1557. #ifdef IKSD
  1558.             if (inserver && c < 0) {    /* End of session? */
  1559.                 debug(F111,"gtword c < 0","exiting",c);
  1560.                 return(-4);             /* Cleanup and terminate */
  1561.             }
  1562. #endif /* IKSD */
  1563. #ifdef OS2
  1564.            if (c < 0) { /* Error */
  1565.        if (c == -3) { /* Empty word? */
  1566.    if (blocklvl > 0) /* In a block */
  1567.      continue; /* so keep looking for block end */
  1568.    else
  1569.      return(-3); /* Otherwise say we got nothing */
  1570.        } else { /* Not empty word */
  1571.    return(-4); /* So some kind of i/o error */
  1572.        }
  1573.            }
  1574. #else
  1575. #ifdef MAC
  1576.    if (c == -3) /* Empty word... */
  1577.      if (blocklvl > 0)
  1578.        continue;
  1579.      else
  1580.        return(-3);
  1581. #endif /* MAC */
  1582. #endif /* OS2 */
  1583.    if (c == EOF) { /* This can happen if stdin not tty. */
  1584. #ifdef EINTR
  1585. /*
  1586.   Some operating and/or C runtime systems return EINTR for no good reason,
  1587.   when the end of the standard input "file" is encountered.  In cases like
  1588.   this, we get into an infinite loop; hence the eintr counter, which is reset
  1589.   to 0 upon each call to this routine.
  1590. */
  1591. debug(F101,"gtword EOF","",errno);
  1592. if (errno == EINTR && ++eintr < 4) /* When bg'd process is */
  1593.   continue; /* fg'd again. */
  1594. #endif /* EINTR */
  1595. return(-4);
  1596.     }
  1597.     c &= cmdmsk; /* Strip any parity bit */
  1598. } /* if desired. */
  1599. /* Now we have the next character */
  1600. debug(F000,"gtword char","",c);
  1601. if (quote && (c == CR || c == LF)) { /* Enter key following quote */
  1602.     *bp++ = CMDQ; /* Double it */
  1603.     *bp = NUL;
  1604.     quote = 0;
  1605. }
  1606.         if (quote == 0) { /* If this is not a quoted character */
  1607.             if (c == CMDQ) { /* Got the quote character itself */
  1608. if (!comment && quoting)
  1609.   quote = 1; /* Flag it if not in a comment */
  1610.             }
  1611.     if (c == FF) { /* Formfeed. */
  1612.                 c = NL;                 /* Replace with newline */
  1613. cmdclrscn(); /* Clear the screen */
  1614.             }
  1615.     if (c == HT) { /* Tab */
  1616. if (comment) /* If in comment, */
  1617.   c = SP; /* substitute space */
  1618. else /* otherwise */
  1619.   c = ESC; /* substitute ESC (for completion) */
  1620.     }
  1621.     if (c == ';' || c == '#') { /* Trailing comment */
  1622. if (inword == 0 && quoting) { /* If not in a word */
  1623.     comment = 1; /* start a comment. */
  1624.     cp = bp; /* remember where it starts. */
  1625. }
  1626.     }
  1627.     if (!kstartactive && /* Not in possible Kermit packet */
  1628. !comment && c == SP) { /* Space not in comment */
  1629.                 *bp++ = (char) c; /* deposit in buffer if not already */
  1630. debug(F101,"gtword echof 2","",echof);
  1631. #ifdef BEBOX
  1632.                 if (echof) {
  1633.                     putchar(c); /* echo it. */
  1634.                     fflush(stdout);
  1635.                     fflush(stderr);
  1636.                 }
  1637. #else
  1638.                 if (echof) { /* echo it. */
  1639.     putchar((CHAR)c);
  1640.     if (timelimit)
  1641.       fflush(stdout);
  1642. }
  1643. #endif /* BEBOX */
  1644.                 if (inword == 0) {      /* If leading, gobble it. */
  1645.                     pp++;
  1646.                     continue;
  1647.                 } else {                /* If terminating, return. */
  1648.     if ((*pp != lbrace) || (bracelvl == 0)) {
  1649. np = bp;
  1650. cmbptr = np;
  1651. if (setatm(pp,0) < 0) {
  1652.     printf("?Field too long error 1n");
  1653.     debug(F111,"gtword too long #1",pp,strlen(pp));
  1654.     return(-9);
  1655. }
  1656. brkchar = c;
  1657. inword = cmflgs = 0;
  1658. return(0);
  1659.     }
  1660.                     continue;
  1661.                 }
  1662.             }
  1663.     if (!kstartactive && !comment && brk && (c == '=' || c == ':')) {
  1664.                 *bp++ = (char) c;
  1665. #ifdef BEBOX
  1666.                 if (echof) {
  1667.                     putchar(c); /* echo it. */
  1668.                     fflush(stdout);
  1669.                     fflush(stderr);
  1670.                 }
  1671. #else
  1672. if (echof) {
  1673.     putchar((CHAR)c);
  1674.     if (timelimit)
  1675.       fflush(stdout);
  1676. }
  1677. #endif /* BEBOX */
  1678. if ((*pp != lbrace) || (bracelvl == 0)) {
  1679.     np = bp;
  1680.     cmbptr = np;
  1681.     if (setatm(pp,0) < 0) {
  1682. printf("?Field too long error 1n");
  1683. debug(F111,"gtword too long #1",pp,strlen(pp));
  1684. return(-9);
  1685.     }
  1686.     inword = cmflgs = 0;
  1687.     brkchar = c;
  1688.     return(4);
  1689. }
  1690.             }
  1691.             if (c == lbrace)
  1692.               bracelvl++;
  1693.             if (c == rbrace) {
  1694.                 bracelvl--;
  1695.                 if (linebegin)
  1696.   blocklvl--;
  1697.             }
  1698.             if (c == LF || c == CR) { /* CR or LF. */
  1699. if (echof) {
  1700.                     cmdnewl((char)c); /* echo it. */
  1701. #ifdef BEBOX
  1702.                     fflush(stdout);
  1703.                     fflush(stderr);
  1704. #endif /* BEBOX */
  1705.                 }
  1706. {
  1707.     /* Trim trailing comment and whitespace */
  1708.     char *qq;
  1709.     if (comment) { /* Erase comment */
  1710. while (bp >= cp) /* Back to comment pointer */
  1711.   *bp-- = NUL;
  1712. bp++;
  1713. pp = bp; /* Adjust other pointers */
  1714. inword = 0; /* and flags */
  1715. comment = 0;
  1716. cp = NULL;
  1717.     }
  1718.     qq = inword ? pp : (char *)cmdbuf;
  1719.     /* Erase trailing whitespace */
  1720.     while (bp > qq && (*(bp-1) == SP || *(bp-1) == HT)) {
  1721. bp--;
  1722. debug(F000,"erasing","",*bp);
  1723. *bp = NUL;
  1724.     }
  1725.     lastchar = (bp > qq) ? *(bp-1) : NUL;
  1726.     prevchar = (bp > qq+1) ? *(bp-2) : NUL;
  1727. }
  1728. if (linebegin && blocklvl > 0) /* Blank line in {...} block */
  1729.   continue;
  1730. linebegin = 1; /* At beginning of next line */
  1731. iscontd = prevchar != CMDQ &&
  1732.   (lastchar == '-' || lastchar == lbrace);
  1733. debug(F101,"gtword iscontd","",iscontd);
  1734.                 if (iscontd) { /* If line is continued... */
  1735.                     if (chsrc) { /* If reading from tty, */
  1736.                         if (*(bp-1) == lbrace) { /* Check for "begin block" */
  1737.                             *bp++ = SP; /* Insert a space for neatness */
  1738.                             blocklvl++; /* Count block nesting level */
  1739.                         } else { /* Or hyphen */
  1740.     bp--; /* Overwrite the hyphen */
  1741.                         }
  1742.                         *bp = NUL; /* erase the dash, */
  1743.                         continue; /* and go back for next char now. */
  1744.                     }
  1745. } else if (blocklvl > 0) { /* No continuation character */
  1746.     if (chsrc) { /* But we're in a "block" */
  1747. *bp++ = ','; /* Add comma */
  1748. *bp = NUL;
  1749. continue;
  1750.     }
  1751. } else { /* No continuation, end of command. */
  1752.     *bp = NUL; /* Terminate the command string. */
  1753.     if (comment) { /* If we're in a comment, */
  1754. comment = 0; /* Say we're not any more, */
  1755. *cp = NUL; /* cut it off. */
  1756.     }
  1757.     np = bp; /* Where to start next field. */
  1758.     cmbptr = np;
  1759.     if (setatm(pp,0) < 0) { /* Copy field to atom buffer */
  1760. debug(F111,"gtword too long #2",pp,strlen(pp));
  1761. printf("?Field too long error 2n");
  1762. return(-9);
  1763.     }
  1764.     inword = 0; /* Not in a word any more. */
  1765.     crflag = 1;
  1766.                     debug(F110,"gtword","crflag is set",0);
  1767.     return(cmflgs = 1);
  1768. }
  1769.             }
  1770.     /* Question mark */
  1771.             if ((c == '?')
  1772. && quoting
  1773. && !kstartactive
  1774. && !comment
  1775. && cmdsrc() == 0 /* Commands coming from terminal */
  1776. && chsrc != 0 /* and NOT from reparse buffer! */
  1777. ) {
  1778.                 putchar((CHAR)c);
  1779.                 *bp = NUL;
  1780.                 if (setatm(pp,0) < 0) {
  1781.     debug(F111,"gtword too long #3",pp,strlen(pp));
  1782.     printf("?Too long #3n");
  1783.     return(-9);
  1784. }
  1785. qmflag = 1;
  1786.                 return(cmflgs = 3);
  1787.             }
  1788.             if (c == ESC) { /* ESC */
  1789. if (!comment) {
  1790.     *bp = NUL;
  1791.     if (setatm(pp,0) < 0) {
  1792. debug(F111,"gtword too long #4",pp,strlen(pp));
  1793. printf("?Too long #4n");
  1794. return(-9);
  1795.     }
  1796.     esflag = 1;
  1797.     return(cmflgs = 2);
  1798. } else {
  1799.     bleep(BP_WARN);
  1800.     continue;
  1801. }
  1802.             }
  1803.             if (c == BS || c == RUB) {  /* Character deletion */
  1804.                 if (bp > cmdbuf) {      /* If still in buffer... */
  1805.     cmdchardel(); /* erase it. */
  1806.                     bp--;               /* point behind it, */
  1807. #ifdef COMMENT
  1808.                     if (*bp == SP) inword = 0; /* Flag if current field gone */
  1809. #else
  1810. /* fixed by Ulli Schlueter */
  1811.                     if (*bp == lbrace) bracelvl--; /* Adjust brace count */
  1812.                     if (*bp == rbrace) bracelvl++;
  1813.                     if ((*bp == SP) &&       /* Flag if current field gone */
  1814. (*pp != lbrace || bracelvl == 0))
  1815.       inword = 0;
  1816. #endif /* COMMENT */
  1817.                     *bp = NUL;          /* Erase character from buffer. */
  1818.                 } else {                /* Otherwise, */
  1819.     bleep(BP_WARN);
  1820.                     cmres();            /* and start parsing a new command. */
  1821.     *bp = *atmbuf = NUL;
  1822.                 }
  1823.                 if (pp < bp) continue;
  1824.                 else return(cmflgs = -1);
  1825.             }
  1826.             if (c == LDEL) {            /* ^U, line deletion */
  1827.                 while ((bp--) > cmdbuf) {
  1828.                     cmdchardel();
  1829.                     *bp = NUL;
  1830.                 }
  1831.                 cmres();                /* Restart the command. */
  1832. *bp = *atmbuf = NUL;
  1833.                 inword = 0;
  1834.                 return(cmflgs = -1);
  1835.             }
  1836.             if (c == WDEL) {            /* ^W, word deletion */
  1837.                 if (bp <= cmdbuf) {     /* Beep if nothing to delete */
  1838.     bleep(BP_WARN);
  1839.                     cmres();
  1840.     *bp = *atmbuf = NUL;
  1841.                     return(cmflgs = -1);
  1842.                 }
  1843.                 bp--;
  1844. /* Back up over any trailing nonalphanums */
  1845. /* This is dependent on ASCII collating sequence */
  1846. /* but isalphanum() is not available everywhere. */
  1847.                 for ( ; (bp >= cmdbuf) &&
  1848.      (*bp < '0') ||
  1849.      (*bp > '9' && *bp < '@') ||
  1850.      (*bp > 'Z' && *bp < 'a') ||
  1851.      (*bp > 'z') ; bp--) {
  1852.                     cmdchardel();
  1853.                     *bp = NUL;
  1854.                 }
  1855. /* Now delete back to rightmost remaining nonalphanum */
  1856.                 for ( ; (bp >= cmdbuf) && (*bp) ; bp--) {
  1857.     if ((*bp < '0') ||
  1858. (*bp > '9' && *bp < '@') ||
  1859. (*bp > 'Z' && *bp < 'a') ||
  1860. (*bp > 'z'))
  1861.       break;
  1862.                     cmdchardel();
  1863.                     *bp = NUL;
  1864.                 }
  1865.                 bp++;
  1866.                 inword = 0;
  1867.                 return(cmflgs = -1);
  1868.             }
  1869.             if (c == RDIS) {            /* ^R, redisplay */
  1870. #ifdef COMMENT
  1871.                 *bp = NUL;
  1872.                 printf("n%s%s",cmprom,cmdbuf);
  1873. #else
  1874. char *cpx; char cx;
  1875.                 *bp = NUL;
  1876.                 printf("n%s",cmprom);
  1877. cpx = cmdbuf;
  1878. while (cx = *cpx++) {
  1879. #ifdef isprint
  1880.     putchar((CHAR) (isprint(cx) ? cx : '^'));
  1881. #else
  1882.     putchar((CHAR) ((cx >= SP && cx < DEL) ? cx : '^'));
  1883. #endif /* isprint */
  1884. }
  1885. #endif /* COMMENT */
  1886. fflush(stdout);
  1887.                 continue;
  1888.             }
  1889. #ifdef CK_RECALL
  1890.     if (chsrc && on_recall && /* Reading commands from keyboard? */
  1891. (cm_recall > 0) && /* Saving commands? */
  1892. (c == C_UP || c == C_UP2)) { /* Go up one */
  1893. if (current < 0) { /* Nowhere to go, */
  1894.     bleep(BP_WARN);
  1895.     continue;
  1896. }
  1897. if (recall[current]) {
  1898.     if (!strcmp(recall[current],cmdbuf)) {
  1899. if (current > 0) {
  1900.     current--;
  1901. } else {
  1902.     bleep(BP_WARN);
  1903.     continue;
  1904. }
  1905.     }
  1906. }
  1907. if (recall[current]) { /* We have a previous command */
  1908.     while ((bp--) > cmdbuf) { /* Erase current line */
  1909. cmdchardel();
  1910. *bp = NUL;
  1911.     }
  1912.     ckstrncpy(cmdbuf,recall[current],CMDBL);
  1913. #ifdef OSK
  1914.     fflush(stdout);
  1915.     write(fileno(stdout), "r", 1);
  1916.     printf("%s%s",cmprom,cmdbuf);
  1917. #else
  1918.     printf("r%s%s",cmprom,cmdbuf);
  1919. #endif /* OSK */
  1920.     current--;
  1921. }
  1922. return(cmflgs = -1); /* Force a reparse */
  1923.     }
  1924.     if (chsrc && on_recall && /* Reading commands from keyboard? */
  1925. (cm_recall > 0) && /* Saving commands? */
  1926. (c == C_DN)) { /* Down one */
  1927. if (current + 1 > rlast) { /* Already at bottom, just beep */
  1928.     bleep(BP_WARN);
  1929.     continue;
  1930. }
  1931. current++; /* OK to go down */
  1932. if (recall[current]) {
  1933.     if (!strcmp(recall[current],cmdbuf)) {
  1934. if (current + 1 > rlast) { /* At bottom, beep */
  1935.     bleep(BP_WARN);
  1936.     continue;
  1937. } else
  1938.   current++;
  1939.     }
  1940. }
  1941. if (recall[current]) {
  1942.     while ((bp--) > cmdbuf) { /* Erase current line */
  1943. cmdchardel();
  1944. *bp = NUL;
  1945.     }
  1946.     ckstrncpy(cmdbuf,recall[current],CMDBL);
  1947. #ifdef OSK
  1948.     fflush(stdout);
  1949.     write(fileno(stdout), "r", 1);
  1950.     printf("%s%s",cmprom,cmdbuf);
  1951. #else
  1952.     printf("r%s%s",cmprom,cmdbuf);
  1953. #endif /* OSK */
  1954.     return(cmflgs = -1); /* Force reparse */
  1955. }
  1956.     }
  1957. #endif /* CK_RECALL */
  1958.     if (c < SP && quote == 0) { /* Any other unquoted control char */
  1959. if (!chsrc) { /* If cmd file, point past it */
  1960.     bp++;
  1961. } else {
  1962.     bleep(BP_WARN);
  1963. }
  1964. continue; /* continue, don't put in buffer */
  1965.     }
  1966.     linebegin = 0; /* Not at beginning of line */
  1967. #ifdef BEBOX
  1968.     if (echof) {
  1969.                 cmdecho((char) c, 0); /* Echo what was typed. */
  1970.                 fflush (stdout);
  1971.                 fflush(stderr);
  1972.             }
  1973. #else
  1974.             if (echof) cmdecho((char) c, 0); /* Echo what was typed. */
  1975. #endif /* BEBOX */
  1976.         } else { /* This character was quoted. */
  1977.     int qf = 1;
  1978.     quote = 0; /* Unset the quote flag. */
  1979.     debug(F000,"gtword quote 0","",c);
  1980.     /* Quote character at this level is only for SP, ?, and controls */
  1981.             /* If anything else was quoted, leave quote in, and let */
  1982.     /* the command-specific parsing routines handle it, e.g. 07 */
  1983.     if (c > 32 && c != '?' && c != RUB && chsrc != 0) {
  1984. debug(F000,"gtword quote 1","",c);
  1985. *bp++ = CMDQ; /* Deposit  if it came from tty */
  1986. qf = 0; /* and don't erase it from screen */
  1987. linebegin = 0; /* Not at beginning of line */
  1988. #ifdef BS_DIRSEP
  1989. /*
  1990.   This is a hack to handle "cd " or "cd foo" on OS/2 and similar systems.
  1991.   If we were called from cmdir() and the previous character was the quote
  1992.   character, i.e. backslash, and this character is the command terminator,
  1993.   then we stuff an extra backslash into the buffer without echoing, then
  1994.   we stuff the carriage return back in again, and go back and process it,
  1995.   this time with the quote flag off.
  1996. */
  1997.     } else if (dirnamflg && (c == CR || c == LF || c == SP)) {
  1998. debug(F000,"gtword quote 2","",c);
  1999. *bp++ = CMDQ;
  2000. linebegin = 0; /* Not at beginning of line */
  2001. *bp = (c == SP ? SP : CR);
  2002.         goto CMDIRPARSE ;
  2003. #endif /* BS_DIRSEP */
  2004.     } else {
  2005. debug(F000,"gtword quote 3","",c);
  2006.     }
  2007. #ifdef BEBOX
  2008.     if (echof) {
  2009.                 cmdecho((char) c, qf); /* Echo what was typed. */
  2010.                 fflush (stdout);
  2011.                 fflush(stderr);
  2012.             }
  2013. #else
  2014.     if (echof) cmdecho((char) c, qf); /* Now echo quoted character */
  2015. #endif /* BEBOX */
  2016.     debug(F111,"gtword quote",cmdbuf,c);
  2017. }
  2018. #ifdef COMMENT
  2019.         if (echof) cmdecho((char) c,quote); /* Echo what was typed. */
  2020. #endif /* COMMENT */
  2021.         if (!comment) inword = 1; /* Flag we're in a word. */
  2022. if (quote) continue; /* Don't deposit quote character. */
  2023.         if (c != NL) { /* Deposit command character. */
  2024.     *bp++ = (char) c; /* and make sure there is a NUL */
  2025. #ifdef COMMENT
  2026.     *bp = NUL; /* after it */
  2027. #endif /* COMMENT */
  2028. }
  2029.     }                                   /* End of big while */
  2030.     bleep(BP_WARN);
  2031.     printf("?Command too long, maximum length: %d.n",CMDBL);
  2032.     cmflgs = -2;
  2033.     return(-9);
  2034. }
  2035. /* Utility functions */
  2036. /* A D D B U F  -- Add the string pointed to by cp to the command buffer  */
  2037. static int
  2038. addbuf(cp) char *cp; {
  2039.     int len = 0;
  2040.     while ((*cp != NUL) && (bp < cmdbuf+CMDBL)) {
  2041.         *bp++ = *cp++;                  /* Copy and */
  2042.         len++;                          /* count the characters. */
  2043.     }
  2044.     *bp++ = SP;                         /* Put a space at the end */
  2045.     *bp = NUL;                          /* Terminate with a null */
  2046.     np = bp;                            /* Update the next-field pointer */
  2047.     cmbptr = np;
  2048.     return(len);                        /* Return the length */
  2049. }
  2050. /*  S E T A T M  --  Deposit a token in the atom buffer.  */
  2051. /*
  2052.   Break on space, newline, carriage return, or NUL.
  2053.   Call with:
  2054.     cp = Pointer to string to copy to atom buffer.
  2055.     fcode = 0 means break on whitespace or EOL.
  2056.     fcode = 1 means don't break on space.
  2057.     fcode = 2 means break on space, ':', or '='.
  2058.   Null-terminate the result.
  2059.   If the source pointer is the atom buffer itself, do nothing.
  2060.   (no longer true.  now we copy the buffer, edit it, and copy it back.)
  2061.   Return length of token, and also set global "cc" to this length.
  2062.   Return -1 if token was too long.
  2063. */
  2064. static int
  2065. setatm(cp,fcode) char *cp; int fcode; {
  2066.     char *ap, *xp;
  2067.     int  bracelvl, n;
  2068.     char lbrace, rbrace;
  2069.     if (cmfldflgs & 1) {
  2070. lbrace = '(';
  2071. rbrace = ')';
  2072.     } else {
  2073. lbrace = '{';
  2074. rbrace = '}';
  2075.     }
  2076.     cc = 0; /* Character counter */
  2077.     ap = atmbuf; /* Address of atom buffer */
  2078.     if ((int) strlen(cp) > ATMBL)
  2079.       return(-1);
  2080.     if (cp == ap) { /* In case source is atom buffer */
  2081. xp = atybuf; /* make a copy */
  2082. strncpy(xp,ap,ATMBL); /* so we can copy it back, edited. */
  2083. cp = xp;
  2084.     }
  2085.     *ap = NUL; /* Zero the atom buffer */
  2086.     if (fcode == 1) { /* Trim trailing blanks */
  2087. n = strlen(cp);
  2088. while (--n >= 0)
  2089.   if (cp[n] != SP) break;
  2090. cp[n+1] = NUL;
  2091.     }
  2092.     while (*cp == SP) cp++; /* Trim leading spaces */
  2093.     bracelvl = 0;
  2094.     while (*cp) {
  2095.         if (*cp == lbrace) bracelvl++;
  2096.         if (*cp == rbrace) bracelvl--;
  2097. if (bracelvl < 0) bracelvl = 0;
  2098. if (bracelvl == 0) {
  2099.     if ((*cp == SP || *cp == HT) && (fcode != 1)) break;
  2100.     if ((fcode == 2) && (*cp == '=' || *cp == ':')) break;
  2101.     if (*cp == LF || *cp == CR) break;
  2102. }
  2103.         *ap++ = *cp++;
  2104.         cc++;
  2105.     }
  2106.     *ap = NUL; /* Terminate the string. */
  2107.     return(cc);                         /* Return length. */
  2108. }
  2109. /*
  2110.   These functions attempt to hide system dependencies from the mainline
  2111.   code in gtword().  Dummy arg for cmdgetc() needed for compatibility with
  2112.   coninc(), ttinc(), etc, since a pointer to this routine can be passed in
  2113.   place of those to tn_doop().
  2114.   No longer static.  Used by askmore().  Fri Aug 20 15:03:34 1999.
  2115. */
  2116. int
  2117. cmdgetc(timelimit) int timelimit; { /* Get a character from the tty. */
  2118.     int c;
  2119. #ifdef IKSD
  2120.     extern int inserver;
  2121. #endif /* IKSD */
  2122. #ifdef CK_LOGIN
  2123.     extern int x_logged;
  2124. #endif /* CK_LOGIN */
  2125. #ifdef TNCODE
  2126.     static int got_cr = 0;
  2127.     extern int ckxech;
  2128.     int tx = 0, is_tn = 0;
  2129. #endif /* TNCODE */
  2130.     if (pushc
  2131. #ifndef NOSPL
  2132. && !askflag
  2133. #endif /* NOSPL */
  2134. ) {
  2135.         debug(F111,"cmdgetc()","pushc",pushc);
  2136. c = pushc;
  2137. pushc = NUL;
  2138. if (xcmfdb && c == '?') /* Don't echo ? twice if chaining. */
  2139.   cmdchardel();
  2140. return(c);
  2141.     }
  2142. #ifdef datageneral
  2143.     {
  2144. char ch;
  2145. c = dgncinb(0,&ch,1); /* -1 is EOF, -2 TO,
  2146.                                          * -c is AOS/VS error */
  2147. if (c == -2) { /* timeout was enabled? */
  2148.     resto(channel(0)); /* reset timeouts */
  2149.     c = dgncinb(0,&ch,1); /* retry this now! */
  2150. }
  2151. if (c < 0) return(-4); /* EOF or some error */
  2152. else c = (int) ch & 0177; /* Get char without parity */
  2153. /* echof = 1; */
  2154.     }
  2155. #else /* Not datageneral */
  2156. #ifndef MINIX2
  2157.     if (
  2158. #ifdef IKSD
  2159. (!local && inserver) ||
  2160. #endif /* IKSD */
  2161. timelimit > 0) {
  2162. #ifdef TNCODE
  2163.           GETNEXTCH:
  2164.             is_tn = !pushc && !local && sstelnet;
  2165. #endif /* TNCODE */
  2166. #ifdef COMMENT
  2167.     c = coninc(timelimit > 0 ? 1 : 0);
  2168. #else /* COMMENT */
  2169.     /* This is likely to break the asktimeout... */
  2170.     c = coninc(timelimit);
  2171. #endif /* COMMENT */
  2172.     /* debug(F101,"cmdgetc coninc","",c); */
  2173. #ifdef TNCODE
  2174.             if (c >= 0 && is_tn) { /* Server-side Telnet */
  2175.                 switch (c) {
  2176.   case IAC:
  2177.                     debug(F111,"gtword IAC","c",c);
  2178.                     got_cr = 0;
  2179.                     if ((tx = tn_doop((CHAR)(c & 0xff),ckxech,coninc)) == 0) {
  2180.                         goto GETNEXTCH;
  2181.                     } else if (tx <= -1) { /* I/O error */
  2182.                         /* If there was a fatal I/O error then ttclos()    */
  2183.                         /* has been called and the next GETNEXTCH attempt  */
  2184.                         /* will be !is_tn since ttclos() sets sstelnet = 0 */
  2185.                         doexit(BAD_EXIT,-1); /* (or return(-4)? */
  2186.                     } else if (tx == 1) { /* ECHO change */
  2187.                         ckxech = dpx = 1; /* Get next char */
  2188.                         goto GETNEXTCH;
  2189.                     } else if (tx == 2) { /* ECHO change */
  2190.                         ckxech = dpx = 0; /* Get next char */
  2191.                         goto GETNEXTCH;
  2192.                     } else if (tx == 3) { /* Quoted IAC */
  2193.                         c = 255; /* proceeed with it. */
  2194.                     }
  2195. #ifdef IKS_OPTION
  2196.                     else if (tx == 4) { /* IKS State Change */
  2197.                         goto GETNEXTCH;
  2198.                     }
  2199. #endif /* IKS_OPTION */
  2200.                     else if (tx == 6) { /* Remote Logout */
  2201. doexit(GOOD_EXIT,0);
  2202.                     } else {
  2203. goto GETNEXTCH; /* Unknown, get next char */
  2204.     }
  2205.                     break;
  2206. #ifdef COMMENT
  2207.                   case CR:
  2208.                     if (!TELOPT_U(TELOPT_BINARY)) {
  2209. if ( got_cr ) {
  2210.     /* This means the sender is violating Telnet   */
  2211.     /* protocol because we received two CRs in a   */
  2212.     /* row without getting either LF or NUL.       */
  2213.     /* This will not solve the problem but it      */
  2214.     /* will at least allow two CRs to do something */
  2215.     /* whereas before the user would have to guess */
  2216.     /* to send LF or NUL after the CR.             */
  2217.     debug(F100,"gtword CR telnet error","",0);
  2218.     c = LF;
  2219. } else {
  2220.     debug(F100,"gtword skipping CR","",0);
  2221.     got_cr = 1; /* Remember a CR was received */
  2222.     goto GETNEXTCH;
  2223. }
  2224.                     } else {
  2225. debug(F100,"gtword CR to LF","",0);
  2226. c = LF;
  2227.                     }
  2228.                     break;
  2229.                   case LF:
  2230.                     if (!TELOPT_U(TELOPT_BINARY)) {
  2231. got_cr = 0;
  2232. debug(F100,"gtword LF","",0);
  2233.                     } else {
  2234. if (got_cr) {
  2235.     got_cr = 0;
  2236.     debug(F100,"gtword skipping LF","",0);
  2237.     goto GETNEXTCH;
  2238. }
  2239.                     }
  2240.                     break;
  2241.                   case NUL:
  2242.                     if (!TELOPT_U(TELOPT_BINARY) && got_cr) {
  2243. c = LF;
  2244. debug(F100,"gtword NUL to LF","",0);
  2245.                     } else {
  2246. debug(F100,"gtword NUL","",0);
  2247.                     }
  2248.                     got_cr = 0;
  2249.                     break;
  2250. #else /* COMMENT */
  2251.                   case CR:
  2252.                     if ( !TELOPT_U(TELOPT_BINARY) && got_cr ) {
  2253.                         /* This means the sender is violating Telnet   */
  2254.                         /* protocol because we received two CRs in a   */
  2255.                         /* row without getting either LF or NUL.       */
  2256.                         /* This will not solve the problem but it      */
  2257.                         /* will at least allow two CRs to do something */
  2258.                         /* whereas before the user would have to guess */
  2259.                         /* to send LF or NUL after the CR.             */
  2260.                         debug(F100,"gtword CR telnet error","",0);
  2261.                     } else {
  2262.                         got_cr = 1; /* Remember a CR was received */
  2263.                     }
  2264.                     debug(F100,"gtword CR to LF","",0);
  2265.                     c = LF;
  2266.     break;
  2267.                   case LF:
  2268.                     if (got_cr) {
  2269.                         got_cr = 0;
  2270.                         debug(F100,"gtword skipping LF","",0);
  2271.                         goto GETNEXTCH;
  2272.                     }
  2273.     break;
  2274.                   case NUL:
  2275.                     if (got_cr) {
  2276.                         got_cr = 0;
  2277.                         debug(F100,"gtword skipping NUL","",0);
  2278.                         goto GETNEXTCH;
  2279.                     } else {
  2280.                       debug(F100,"gtword NUL","",0);
  2281.                     }
  2282.                     break;
  2283. #endif /* COMMENT */
  2284. #ifdef IKSD
  2285.   case ETX: /* Ctrl-C... */
  2286.                   case EOT: /* EOT = EOF */
  2287.                       debug(F000,"gtword","",c);
  2288.                       if (inserver
  2289. #ifdef CK_LOGIN
  2290.   && !x_logged
  2291. #endif /* CK_LOGIN */
  2292.   )
  2293.                           return(-4);
  2294.     break;
  2295. #endif /* IKSD */
  2296.   default:
  2297.                       got_cr = 0;
  2298.                 }
  2299.             }
  2300. #endif /* TNCODE */
  2301.     } else {
  2302. /* debug(F100,"cmdgetc getchar 1","",0); */
  2303. #ifdef OS2
  2304. c = coninc(0);
  2305. #else /* OS2 */
  2306. c = getchar();
  2307. /* debug(F101,"cmdgetc getchar 2","",c); */
  2308. #endif /* OS2 */
  2309.     }
  2310. #else  /* MINIX2 */
  2311. #undef getc
  2312.     c = getc(stdin);
  2313. #endif /* MINIX2 */
  2314. #ifdef RTU
  2315.     if (rtu_bug) {
  2316. c = getchar(); /* RTU doesn't discard the ^Z */
  2317. rtu_bug = 0;
  2318.     }
  2319. #endif /* RTU */
  2320. #endif /* datageneral */
  2321.     return(c); /* Return what we got */
  2322. }
  2323. static VOID
  2324. cmdclrscn() { /* Clear the screen */
  2325.     ck_cls();
  2326. }
  2327. static VOID /* What to echo at end of command */
  2328. #ifdef CK_ANSIC
  2329. cmdnewl(char c)
  2330. #else
  2331. cmdnewl(c) char c;
  2332. #endif /* CK_ANSIC */
  2333. /* cmdnewl */ {
  2334. #ifdef OS2
  2335. #ifdef IKSD
  2336.     extern int inserver;
  2337.     if (inserver && c == LF)
  2338.       putchar(CR);
  2339. #endif /* IKSD */
  2340. #endif /* OS2 */
  2341.     putchar(c); /* c is the terminating character */
  2342. #ifdef WINTCP /* what is this doing here? */
  2343.     if (c == CR) putchar(NL);
  2344. #endif /* WINTCP */
  2345. /*
  2346.   A.A. Chernov, who sent in changes for FreeBSD, said we also needed this
  2347.   for SVORPOSIX because "setup terminal by termios and curses does
  2348.   not convert r to n, so additional n needed in newline function."  But
  2349.   it is also very likely to result in unwanted blank lines.
  2350. */
  2351. #ifdef BSD44
  2352.     if (c == CR) putchar(NL);
  2353. #endif /* BSD44 */
  2354. #ifdef COMMENT
  2355.     /* OS2 no longer needs this as all CR are converted to NL in coninc() */
  2356.     /* This eliminates the ugly extra blank lines discussed above.        */
  2357. #ifdef OS2
  2358.     if (c == CR) putchar(NL);
  2359. #endif /* OS2 */
  2360. #endif /* COMMENT */
  2361. #ifdef aegis
  2362.     if (c == CR) putchar(NL);
  2363. #endif /* aegis */
  2364. #ifdef AMIGA
  2365.     if (c == CR) putchar(NL);
  2366. #endif /* AMIGA */
  2367. #ifdef datageneral
  2368.     if (c == CR) putchar(NL);
  2369. #endif /* datageneral */
  2370. #ifdef GEMDOS
  2371.     if (c == CR) putchar(NL);
  2372. #endif /* GEMDOS */
  2373. #ifdef STRATUS
  2374.     if (c == CR) putchar(NL);
  2375. #endif /* STRATUS */
  2376. }
  2377. static VOID
  2378. cmdchardel() { /* Erase a character from the screen */
  2379.     if (!dpx) return;
  2380. #ifdef datageneral
  2381.     /* DG 'b' is EM (^y or 31) */
  2382.     if (termtype == 1)
  2383.       /* Erase a character from non-DG screen, */
  2384.       dgncoub(1,"10 10",3);
  2385.     else
  2386. #endif /* datageneral */
  2387.       printf("b b");
  2388. #ifdef GEMDOS
  2389.     fflush(stdout);
  2390. #else
  2391. #ifdef BEBOX
  2392.     fflush(stdout);
  2393. #endif /* BEBOX */
  2394. #endif /* GEMDOS */
  2395. }
  2396. static VOID
  2397. #ifdef CK_ANSIC
  2398. cmdecho(char c, int quote)
  2399. #else
  2400. cmdecho(c,quote) char c; int quote;
  2401. #endif /* CK_ANSIC */
  2402. { /* cmdecho */
  2403.     if (!dpx) return;
  2404.     /* Echo tty input character c */
  2405.     if (quote) {
  2406. putchar(BS);
  2407. putchar(SP);
  2408. putchar(BS);
  2409. #ifdef isprint
  2410. putchar((CHAR) (isprint(c) ? c : '^' ));
  2411. #else
  2412. putchar((CHAR) ((c >= SP && c < DEL) ? c : '^'));
  2413. #endif /* isprint */
  2414.     } else putchar(c);
  2415. #ifdef OS2
  2416.     if (quote==1 && c==CR) putchar((CHAR) NL);
  2417. #endif /* OS2 */
  2418.     if (timelimit)
  2419.       fflush(stdout);
  2420. }
  2421. /* Return pointer to current position in command buffer. */
  2422. char *
  2423. cmpeek() {
  2424.     return(np);
  2425. }
  2426. #endif /* NOICP */
  2427. #ifdef NOICP
  2428. #include "ckcdeb.h"
  2429. #include "ckucmd.h"
  2430. #include "ckcasc.h"
  2431. #endif /* NOICP */
  2432. /*  X X E S C  --  Interprets backslash codes  */
  2433. /*  Returns the int value of the backslash code if it is > -1 and < 256 */
  2434. /*  and updates the string pointer to first character after backslash code. */
  2435. /*  If the argument is invalid, leaves pointer unchanged and returns -1. */
  2436. int
  2437. xxesc(s) char **s; { /* Expand backslash escapes */
  2438.     int x, y, brace, radix; /* Returns the int value */
  2439.     char hd = '9'; /* Highest digit in radix */
  2440.     char *p;
  2441.     p = *s; /* pointer to beginning */
  2442.     if (!p) return(-1); /* watch out for null pointer */
  2443.     x = *p++; /* character at beginning */
  2444.     if (x != CMDQ) return(-1); /* make sure it's a backslash code */
  2445.     x = *p; /* it is, get the next character */
  2446.     if (x == '{') { /* bracketed quantity? */
  2447. p++; /* begin past bracket */
  2448. x = *p;
  2449. brace = 1;
  2450.     } else brace = 0;
  2451.     switch (x) { /* Start interpreting */
  2452.       case 'd': /* Decimal radix indicator */
  2453.       case 'D':
  2454. p++; /* Just point past it and fall thru */
  2455.       case '0': /* Starts with digit */
  2456.       case '1':
  2457.       case '2':  case '3':  case '4':  case '5':
  2458.       case '6':  case '7':  case '8':  case '9':
  2459. radix = 10; /* Decimal */
  2460. hd = '9'; /* highest valid digit */
  2461. break;
  2462.       case 'o': /* Starts with o or O */
  2463.       case 'O':
  2464. radix = 8; /* Octal */
  2465. hd = '7'; /* highest valid digit */
  2466. p++; /* point past radix indicator */
  2467. break;
  2468.       case 'x': /* Starts with x or X */
  2469.       case 'X':
  2470. radix = 16; /* Hexadecimal */
  2471. p++; /* point past radix indicator */
  2472. break;
  2473.       default: /* All others */
  2474. #ifdef COMMENT
  2475. *s = p+1; /* Treat as quote of next char */
  2476. return(*p);
  2477. #else
  2478. return(-1);
  2479. #endif /* COMMENT */
  2480.     }
  2481.     /* For OS/2, there are "wide" characters required for the keyboard
  2482.      * binding, i.e 644 and similar codes larger than 255 (byte).
  2483.      * For this purpose, give up checking for < 256. If someone means
  2484.      * 266 should result in 26 followed by a "6" character, he should
  2485.      * always write {26}6 anyway.  Now, return only the lower byte of
  2486.      * the result, i.e. 10, but eat up the whole 266 sequence and
  2487.      * put the wide result 266 into a global variable.  Yes, that's not
  2488.      * the most beautiful programming style but requires the least
  2489.      * amount of changes to other routines.
  2490.      */
  2491.     if (radix <= 10) { /* Number in radix 8 or 10 */
  2492. for ( x = y = 0;
  2493.         (*p) && (*p >= '0') && (*p <= hd)
  2494. #ifdef OS2
  2495.                    && (y < 5) && (x*radix < KMSIZE);
  2496.               /* the maximum needed value 8196 is 4 digits long */
  2497.               /* while as octal it requires 1377, i.e. 5 digits */
  2498. #else
  2499.                    && (y < 3) && (x*radix < 256);
  2500. #endif /* OS2 */
  2501.       p++,y++) {
  2502.     x = x * radix + (int) *p - 48;
  2503. }
  2504. #ifdef OS2
  2505.         wideresult = x; /* Remember wide result */
  2506.         x &= 255;
  2507. #endif /* OS2 */
  2508. if (y == 0 || x > 255) { /* No valid digits? */
  2509.     *s = p; /* point after it */
  2510.     return(-1); /* return failure. */
  2511. }
  2512.     } else if (radix == 16) { /* Special case for hex */
  2513. if ((x = unhex(*p++)) < 0) { *s = p - 1; return(-1); }
  2514. if ((y = unhex(*p++)) < 0) { *s = p - 2; return(-1); }
  2515. x = ((x << 4) & 0xF0) | (y & 0x0F);
  2516. #ifdef OS2
  2517.         wideresult = x;
  2518.         if ((y = unhex(*p)) >= 0) {
  2519.            p++;
  2520.    wideresult = ((x << 4) & 0xFF0) | (y & 0x0F);
  2521.            x = wideresult & 255;
  2522.         }
  2523. #endif /* OS2 */
  2524.     } else x = -1;
  2525.     if (brace && *p == '}' && x > -1) /* Point past closing brace, if any */
  2526.       p++;
  2527.     *s = p; /* Point to next char after sequence */
  2528.     return(x); /* Return value of sequence */
  2529. }
  2530. int /* Convert hex string to int */
  2531. #ifdef CK_ANSIC
  2532. unhex(char x)
  2533. #else
  2534. unhex(x) char x;
  2535. #endif /* CK_ANSIC */
  2536. /* unhex */ {
  2537.     if (x >= '0' && x <= '9') /* 0-9 is offset by hex 30 */
  2538.       return(x - 0x30);
  2539.     else if (x >= 'A' && x <= 'F') /* A-F offset by hex 37 */
  2540.       return(x - 0x37);
  2541.     else if (x >= 'a' && x <= 'f') /* a-f offset by hex 57 */
  2542.       return(x - 0x57); /* (obviously ASCII dependent) */
  2543.     else return(-1);
  2544. }
  2545. /*  L O O K U P  --  Lookup the string in the given array of strings  */
  2546. /*
  2547.  Call this way:  v = lookup(table,word,n,&x);
  2548.    table - a 'struct keytab' table.
  2549.    word  - the target string to look up in the table.
  2550.    n     - the number of elements in the table.
  2551.    x     - address of an integer for returning the table array index,
  2552.            or NULL if you don't need a table index.
  2553.  The keyword table must be arranged in ascending alphabetical order.
  2554.  Alphabetic case doesn't matter.
  2555.  Returns the keyword's associated value (zero or greater) if found,
  2556.  with the variable x set to the keyword-table index, or:
  2557.   -3 if nothing to look up (target was null),
  2558.   -2 if ambiguous,
  2559.   -1 if not found.
  2560.  A match is successful if the target matches a keyword exactly, or if
  2561.  the target is a prefix of exactly one keyword.  It is ambiguous if the
  2562.  target matches two or more keywords from the table.
  2563. */
  2564. int
  2565. lookup(table,cmd,n,x) char *cmd; struct keytab table[]; int n, *x; {
  2566.     int i, v, len, cmdlen;
  2567. /* Get length of search object, if it's null return code -3. */
  2568.     if (!cmd)
  2569.       return(-3);
  2570.     if (((cmdlen = (int) strlen(cmd)) == 0) || (n < 1))
  2571.       return(-3);
  2572. /* Not null, look it up */
  2573.     for (i = 0; i < n-1; i++) {
  2574. len = strlen(table[i].kwd);
  2575.         if ((len == cmdlen && !ckstrcmp(table[i].kwd,cmd,len,0)) ||
  2576.     ((v = !ckstrcmp(table[i].kwd,cmd,cmdlen,0)) &&
  2577.      ckstrcmp(table[i+1].kwd,cmd,cmdlen,0))) {
  2578.     if (x) *x = i;
  2579.     return(table[i].kwval);
  2580. }
  2581.         if (v) { /* Ambiguous */
  2582.     if (x) *x = i; /* Set index of first match */
  2583.     return(-2);
  2584. }
  2585.     }
  2586. /* Last (or only) element */
  2587.     if (!ckstrcmp(table[n-1].kwd,cmd,cmdlen,0)) {
  2588.         if (x) *x = n-1;
  2589.         return(table[n-1].kwval);
  2590.     } else return(-1);
  2591. }
  2592. /* Like lookup, but requires a full (but case-independent) match */
  2593. int
  2594. xlookup(table,cmd,n,x) struct keytab table[]; char *cmd; int n, *x; {
  2595.     int i, cmdlen;
  2596.     if (!cmd)
  2597.       return(-3);
  2598.     if (((cmdlen = (int) strlen(cmd)) == 0) || (n < 1))
  2599.       return(-3);
  2600.     for (i = 0; i < n; i++) {
  2601. if (((int)strlen(table[i].kwd) == cmdlen) &&
  2602.     (!ckstrcmp(table[i].kwd,cmd,cmdlen,0))) {
  2603.     if (x) *x = i;
  2604.     return(table[i].kwval);
  2605. }
  2606.     }
  2607.     return(-1);
  2608. }
  2609. #ifndef NOICP
  2610. int
  2611. cmdsquo(x) int x; {
  2612.     quoting = x;
  2613.     return(1);
  2614. }
  2615. int
  2616. cmdgquo() {
  2617.     return(quoting);
  2618. }
  2619. #endif /* NOICP */