/* external procedures called by TeX */
#include <sys/signal.h>
#include <sys/time.h>
#include "texpaths.h" /* defines MAXINPATHCHARS, defaultinputpath, etc. */
#include "h00vars.h"
/* The following are for closing files */
closea(f)
int *f; /* actually, file pointer, but it doesn't matter */
{
close1(f);
}
closeb(f)
int *f; /* actually, file pointer, but it doesn't matter */
{
close1(f);
}
closew(f)
int *f; /* actually, file pointer, but it doesn't matter */
{
close1(f);
}
extern int interrupt; /* To be made nonzero to tell TeX about interrupts */
/*
** catchint()
**
** Gets called when the user hits his interrupt key. Sets the global
** "interrupt" nonzero, then sets it up so that the next interrupt will
** also be caught here.
**
*/
/*
** dateandtime(time, day, month, year)
**
** Stores minutes since midnight, current day, month and year into
** *time, *day, *month and *year, respectively.
**
** Also, set things up so that catchint() will get control on interrupts.
**
*/
dateandtime(minutes, day, month, year)
int *minutes, *day, *month, *year;
{
long clock;
struct tm *tmptr, *localtime();
/*
* setpaths is called to set up the arrays inputpath, fontpath, formatpath
* and poolpath as follows: if the user's environment has a value for the
* appropriate value, then use it; otherwise, leave the current value of
* the array (which may be the default path, or it may be the result of
* a call to setpaths on a previous run that was made the subject of
* an undump: this will give the maker of a preloaded TeX the option of
* providing a new set of "default" paths.
*
* Note that we have to copy the string from the environment area, since
* that will change on the next run (which matters if this is for a
* preloaded TeX).
*/
setpaths()
{
register char *envpath;
if ((envpath = getenv("TEXINPUTS")) != NULL)
copypath(inputpath,envpath,MAXINPATHCHARS);
if ((envpath = getenv("TEXFONTS")) != NULL)
copypath(fontpath,envpath,MAXOTHPATHCHARS);
if ((envpath = getenv("TEXFORMATS")) != NULL)
copypath(formatpath,envpath,MAXOTHPATHCHARS);
if ((envpath = getenv("TEXPOOL")) != NULL)
copypath(poolpath,envpath,MAXOTHPATHCHARS);
}
/*
* copypath(s1,s2,n) copies at most n characters (including the null)
* from string s2 to string s1, giving an error message for paths
* that are too long.
*/
copypath(s1,s2,n)
register char *s1,*s2;
register int n;
{
while ((*s1++ = *s2++) != '\0')
if (--n == 0) {
fprintf(stderr, "! Environment search path is too big\n");
*--s1 = '\0';
return; /* let user continue with truncated path */
}
}
extern short buffer[],xord[]; /* arrays of 0..255 */
extern short last,first; /* pointers into buffer */
/*
* lineread reads from the Pascal text file with iorec pointer filep
* into buffer[first], buffer[first+1],..., buffer[last-1] (and
* setting "last").
* Since 8-bit code is in the buffer, the elements are "short", not char.
* Characters are read until a newline is found (which isn't put in the
* buffer) or until the next character would go into buffer[lastlim].
* 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, lastlim)
struct iorec *filep;
int lastlim;
{
register c;
register short *cs; /* pointer into buffer where next char goes */
register FILE *iop; /* stdio-style FILE pointer */
int l; /* how many more chars are allowed before buffer overflow */
short *bf; /* hold address of buffer[first] */
iop = filep->fbuf;
bf = cs = &(buffer[first]);
l = lastlim-first;
/* overflow when next char would go into buffer[lastlim] */
while (--l>=0 && (c = getc(iop)) != EOF && c!='\n')
*cs++ = xord[c];
if (c == EOF)
filep->funit |= EOFF; /* we hit end-of-file */
last = first+cs-bf;
}
#define filenamesize 1024 /* should agree with initex.ch */
extern char nameoffile[],realnameoffile[]; /* these have size filenamesize */
/*
* testaccess(amode,filepath)
*
* Test whether or not the file whose name is in the global nameoffile
* can be opened for reading (if mode=READACCESS)
* or writing (if mode=WRITEACCESS).
*
* The filepath argument is one of the ...FILEPATH constants defined below.
* If the filename given in nameoffile does not begin with '/', we try
* prepending all the ':'-separated areanames in the appropriate path to the
* filename until access can be made, if it ever can.
*
* The realnameoffile global array will contain the name that yielded an
* access success.
*/
bool
testaccess(amode,filepath)
int amode,filepath;
{
register bool ok;
register char *p;
char *curpathplace;
int f;
switch(filepath) {
case NOFILEPATH: curpathplace = NULL; break;
case INPUTFILEPATH: case READFILEPATH:
curpathplace = inputpath; break;
case FONTFILEPATH: curpathplace = fontpath; break;
case FORMATFILEPATH: curpathplace = formatpath; break;
case POOLFILEPATH: curpathplace = poolpath; break;
}
if (nameoffile[0]=='/') /* file name has absolute path */
curpathplace = NULL;
do {
packrealnameoffile(&curpathplace);
if (amode==READACCESS)
/* use system call "access" to see if we could read it */
if (access(realnameoffile,READACCESS)==0) ok = TRUE;
else ok = FALSE;
else {
/* WRITEACCESS: use creat to see if we could create it, but close
the file again if we're OK, to let pc open it for real */
f = creat(realnameoffile,0666);
if (f>=0) ok = TRUE;
else ok = FALSE;
if (ok)
close(f);
}
} while (!ok && curpathplace != NULL);
if (ok) { /* pad realnameoffile with blanks, as Pascal wants */
for (p = realnameoffile; *p != '\0'; p++)
/* nothing: find end of string */ ;
while (p < &(realnameoffile[filenamesize]))
*p++ = ' ';
}
return (ok);
}
/*
* packrealnameoffile(cpp) makes realnameoffile contain the directory at *cpp,
* followed by '/', followed by the characters in nameoffile up until the
* first blank there, and finally a '\0'. The cpp pointer is left pointing
* at the next directory in the path.
* But: if *cpp == NULL, then we are supposed to use nameoffile as is.
*/
packrealnameoffile(cpp)
char **cpp;
{
register char *p,*realname;
realname = realnameoffile;
if ((p = *cpp)!=NULL) {
while ((*p != ':') && (*p != '\0')) {
*realname++ = *p++;
if (realname == &(realnameoffile[filenamesize-1]))
break;
}
if (*p == '\0') *cpp = NULL; /* at end of path now */
else *cpp = p+1; /* else get past ':' */
*realname++ = '/'; /* separate the area from the name to follow */
}
/* now append nameoffile to realname... */
p = nameoffile;
while (*p != ' ') {
if (realname >= &(realnameoffile[filenamesize-1])) {
fprintf(stderr,"! Full file name is too long\n");
break;
}
*realname++ = *p++;
}
*realname = '\0';
}
/*
** 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);
}
}
}
/*
** writedvi(a,b)
**
** writedvi is called to write dvibuf[a..b] to dvifile
** Unfortunately, dvibuf is declared as eight_bits, which in our
** implementation means that two bytes are taken by each entry.
** We only want to output the low order (first) byte of each pair.
*/
extern struct iorec dvifile;
extern short dvibuf[];
writedvi (a,b)
int a,b;
{
register short *ptr,*final;
register FILE *iop;
iop=dvifile.fbuf;
ptr= &(dvibuf[a]);
final= &(dvibuf[b]);
while (ptr<=final) {
putc((char) (*ptr & 0377), iop);
ptr += 1;
}
/* Note: The above code used to be machine dependent. By changing
* the declarations from "char *" to "short *" we get the machine
* to do its own byte ordering of the de-reference of "ptr" above.
* Then we AND off the low bits (now that it's been pulled from
* memory correctly) and cast it into a "char" for putc().
* --Peter Kessler's idea; explanation, AND, cast a la clp.
*/
}
/*
** close1(filep)
** close1 does the proper things to pc's runtime system file data structure
** to close a file
*/
close1(filep)
if (filep->fbuf != 0) {
if ((filep->funit & FDEF) == 0) {
next = (struct iorec *) &_fchain;
while (next->fchain != FILNIL
&& next->fchain != filep)
next = next->fchain;
if (next->fchain != FILNIL)
next->fchain = next->fchain->fchain;
if (filep->fblk > PREDEF) {
fflush(filep->fbuf);
setbuf(filep->fbuf, NULL);
}
fclose(filep->fbuf);
if (ferror(filep->fbuf)) {
/* ERROR and PERROR no longer exist in Pascal library; TeX doesn't know
what to do with them anyway! So I'm commenting this out... don 7/18/92 */
/* ERROR("%s: Close failed\n",
filep->pfname); */
return;
}
filep->fbuf=0;
}
if ((filep->funit & TEMP) != 0 &&
unlink(filep->pfname)) {
/* PERROR("Could not remove ", filep->pfname); */
return;
}
}
}
/*
** The following procedure is due to sjc@s1-c.
** (with mods for 8-bit changes --- don, Sep 89)
**
** calledit(filename, fnlength, linenumber)
**
** TeX82 can call this to implement the 'e' feature in error-recovery
** mode, invoking a text editor on the erroneous source file.
**
** You should pass to "filename" the first character of the packed array
** containing the filename, and to "fnlength" the size of the filename.
**
** New problem (8-bit mod): need to add 128 to all characters of filename!
**
** Ordinarily, this invokes emacs. If you want a different
** editor, create a shell environment variable TEXEDIT containing
** the string that invokes that editor, with "%s" indicating where
** the filename goes and "%d" indicating where the decimal
** linenumber (if any) goes. For example, a TEXEDIT string for "vi" might be:
**
** setenv TEXEDIT "/usr/bin/vi +%d %s"
**
*/