/* External procedures for tangle/weave */
/*   Written by: H. Trickey, 11/17/83   */
/* Modified for 8-bit input by don, 8/31/89 */

/* Note: these procedures aren't necessary; the default input_ln and
* flush_buffer routines in tangle/weave work fine on UNIX.
* However a very big time improvement is achieved by using these.
*/

#define BUF_SIZE 100            /* should agree with tangle.web */

#include "h00vars.h"            /* defines Pascal I/O structure */

extern short buffer[];          /* 0..BUF_SIZE.  Input goes here */
extern short outbuf[];          /* 0..OUT_BUF_SIZE. Output from here */
extern short xord[];
extern char xchr[];     /* character translation arrays */
extern char limit;              /* index into buffer, defined in the range
                                  0..BUF_SIZE (and note that this is a
                                  char rather than a short because BUF_SIZE
                                  is 100) */

/*
* lineread reads from the Pascal text file with iorec pointer filep
* into buffer[0], buffer[1],..., buffer[limit-1] (and
* setting "limit").
* Characters are read until a newline is found (which isn't put in the
* buffer) or until the next character would go into buffer[BUF_SIZE].
* And trailing blanks are to be ignored, so limit is really set to
* one past the last non-blank.
* The characters need to be translated, so really xord[c] is put into
* the buffer when c is read.
* If end-of-file is encountered, the funit field of *filep is set
* appropriately.
*/
lineread(filep)
       struct iorec    *filep;
{
       register int    c;
       register short  *cs;    /* pointer into buffer where next char goes */
       register short  *cnb;   /* last non-blank character input */
       register FILE   *iop;   /* stdio-style FILE pointer */
       register int    l;      /* how many chars before buffer overflow */

       iop = filep->fbuf;
       cnb = cs = &buffer[0];
       l = BUF_SIZE;

       /* overflow when next char would go into buffer[BUF_SIZE] */
       while ((--l >= 0) && ((c = getc(iop)) != EOF) && (c != '\n')) {
               if ((*cs++ = xord[c]) != ' ')
                       cnb = cs;
       }

       if (c == EOF)
               filep->funit |= EOFF;   /* we hit end-of-file */

       limit = (cnb - &buffer[0]);
}

/*
* linewrite writes to the Pascal text file with iorec pointer filep
* from outbuf[0], outbuf[1],..., outbuf[cnt-1].
* Translation is done, so that xchr[c] is output instead of c.
*/
linewrite(filep, cnt)
       struct iorec    *filep;
       int             cnt;
{
       register FILE   *iop;   /* stdio-style FILE pointer */
       register short  *cs;    /* pointer to next character to output */
       register int    l;      /* how many characters left to output */

       iop = filep->fbuf;
       cs = &outbuf[0];
       l = cnt;

       while (--l >= 0)
               putc(xchr[*cs++], iop);
}

/*
*      testeof(filep)
*
*  Test whether or not the Pascal text file with iorec pointer filep
*  has reached end-of-file (when the only I/O on it is done with
*  lineread, above).
*  We may have to read the next character and unget it to see if perhaps
*  the end-of-file is next.
*/
bool
testeof(filep)
       register struct iorec   *filep;
{
       register char c;
       register FILE *iop; /* stdio-style FILE pointer */
       if (filep->funit & EOFF)
               return(TRUE);
       else { /* check to see if next is EOF */
               iop = filep->fbuf;
               c = getc(iop);
               if (c == EOF)
                       return(TRUE);
               else {
                       ungetc(c,iop);
                       return(FALSE);
                       }
               }
}