/* -*-C-*- dviinit.h */ /*-->dviinit*/ /**********************************************************************/ /****************************** dviinit *******************************/ /**********************************************************************/ void dviinit(filestr) /* initialize DVI file processing */ char *filestr; /* command line filename string */ { int i; /* temporary */ INT16 k; /* index in curpath[] and curname[] */ char* tcp; /* temporary string pointers */ char* tcp1; /* temporary string pointers */ #if OS_UNIX /*********************************************************************** Unix allows binary I/O on stdin and stdout; most other operating systems do not. In order to support this, we use the convention that a null filestr (NOT a (char*)NULL) means the dvifile is on stdin, and the translated output is on stdout. Error logging will be to dvixxx.err or dvixxx.dvi-err. Because random access is required of the dvifile, stdin may NOT be a pipe; it must be a disk file. ***********************************************************************/ #define NO_FILENAME (filestr[0] == '\0') #endif #if CANON_A2 #ifdef CANON_TEST char modestring[10]; /* for dynamic fopen() mode string */ #endif /* CANON_TEST */ #endif #if (OS_TOPS20 | OS_VAXVMS) char* tcp2; /* temporary string pointers */ #endif for (k = 0; k < 10; ++k) tex_counter[k] = (INT32)0L; fontptr = (struct font_entry *)NULL; hfontptr = (struct font_entry *)NULL; pfontptr = (struct font_entry *)NULL; fontfp = (FILE *)NULL; cache_size = 0; nopen = 0; /*********************************************************** Set up file names and open dvi and plot files. Simple filename parsing assumes forms like: Unix: name name.ext /dir/subdir/.../subdir/name /dir/subdir/.../subdir/name.ext TOPS-20: any Unix style (supported by PCC-20) name name.ext name.ext.gen device:name device:name.ext device:name.ext.gen [directory]name [directory]name.ext [directory]name.ext.gen device:[directory]name device:[directory]name.ext device:[directory]name.ext.gen logname:name logname:name.ext logname:name.ext.gen The Unix style should work under IBM PC DOS (backslash can usually be replaced for forward slash), and the TOPS-20 style contains VAX VMS as a subset. Fancier TOPS-20 names are possible (control-V quoting of arbitrary characters, and attributes), but they are rare enough that we do not support them. For TOPS-20 and VAX VMS, generation numbers are recognized as a digit following the last dot, and they are preserved for the DVI file only. For the output files, the highest generation will always be assumed. This is for convenience with the TOPS-20 TeX implementation, which on completion types in a command "TeXspool: foo.dvi.13" and waits for the user to type a carriage return. We only need to extract the directory path, name, and extension, so the parsing is simple-minded. ***********************************************************/ tcp = strrchr(filestr,'/'); /* find end of Unix file path */ #if (OS_ATARI | OS_PCDOS) if (tcp == (char*)NULL) /* no Unix-style file path */ tcp = strrchr(filestr, '\\'); /* try \dos\path\ */ #endif #if OS_TOPS20 if (tcp == (char*)NULL) /* no Unix-style file path */ { tcp = strrchr(filestr, '>'); /* try */ if (tcp == (char*)NULL) /* no */ tcp = strrchr(filestr, ']'); /* try [dir] */ if (tcp == (char*)NULL) /* no [dir] */ tcp = strrchr(filestr, ':'); /* try logname: */ } #endif #if OS_VAXVMS if (tcp == (char*)NULL) /* no Unix-style file path */ { tcp = strrchr(filestr, ']'); /* try [dir] */ if (tcp == (char*)NULL) /* no [dir] */ tcp = strrchr(filestr, ':'); /* try logname: */ } #endif if (tcp == (char*)NULL) /* no file path */ { curpath[0] = '\0'; /* empty path */ tcp = filestr; /* point to start of name */ } else /* save path for later use */ { k = (INT16)(tcp-filestr+1); (void)strncpy(curpath, filestr, (int)k); curpath[k] = '\0'; tcp++; /* point to start of name */ } tcp1 = strrchr(tcp, '.'); /* find last dot in filename */ #if (OS_TOPS20 | OS_VAXVMS) if ((tcp1 != (char*)NULL) && isdigit(*(tcp1+1))) { /* then assume generation number */ tcp2 = tcp1; /* remember dot position */ *tcp1 = '\0'; /* discard generation number */ tcp1 = strrchr(tcp,'.');/* find last dot in filename */ *tcp2 = '.'; /* restore dot */ } #endif if (tcp1 == (char*)NULL) { /* no dot, so name has no extension */ (void)strcpy(curname, tcp); /* save name */ tcp1 = strchr(tcp, '\0'); /* set empty extension */ } else /* save name part */ { k = (INT16)(tcp1-tcp); (void)strncpy(curname, tcp, (int)k); curname[k] = '\0'; } (void)strcpy(curext, tcp1); /* save extension */ /* DVI file must always have extension DVIEXT; supply one if necessary (e.g /u/jones/foo.dvi) */ (void)strcpy(dviname, curpath); (void)strcat(dviname, curname); if (curext[0] == '\0') (void)strcat(dviname, DVIEXT); else (void)strcat(dviname, curext); /* Device output file is PATH NAME . DVIPREFIX OUTFILE_EXT (e.g. /u/jones/foo.dvi-alw) */ (void)strcpy(dvoname, curpath); (void)strcat(dvoname, curname); (void)strcat(dvoname, "."); (void)strcat(dvoname, DVIPREFIX); (void)strcat(dvoname, OUTFILE_EXT); /* Font substitution file is PATH NAME SUBEXT (e.g. /u/jones/foo.sub). We do not tell user (via stderr) about it until it is needed. */ if (subfile[0] == '\0') /* no -fsubfile; make default */ { (void)strcpy(subfile, curpath); (void)strcat(subfile, curname); (void)strcat(subfile, subext); } #if OS_UNIX if (NO_FILENAME) { dvifp = stdin; /* already open, so no DEBUG_OPEN call */ if (fseek(dvifp,0L,0)) { (void)fprintf(stderr, "%s: Unable to fseek() on stdin--was it a pipe?\n", g_progname); (void)EXIT(1); } } else { dvifp = FOPEN(dviname,RB_OPEN); DEBUG_OPEN(dvifp,dviname,RB_OPEN); } #else dvifp = FOPEN(dviname,RB_OPEN); DEBUG_OPEN(dvifp,dviname,RB_OPEN); #endif if (dvifp == (FILE*)NULL) { (void)sprintf(message,"dviinit(): %s: can't open [%s]", g_progname, dviname); (void)fatal(message); } /* We try both PATH NAME . DVIPREFIX OUTFILE_EXT and NAME . DVIPREFIX OUTFILE_EXT, in case the user does not have write access to the directory PATH */ for (i = 1; i <= 2; ++i) /* two tries here */ { #if BBNBITGRAPH strcpy(dvoname,"stdout"); /* BitGraph always goes to stdout */ plotfp = stdout; #else /* NOT BBNBITGRAPH */ #if OS_VAXVMS #if POSTSCRIPT plotfp = FOPEN(dvoname,WB_OPEN); #else /* Create binary files for other devices; they cause less trouble with VMS spooler programs. */ plotfp = FOPEN(dvoname,WB_OPEN,"rfm=fix","bls=512","mrs=512"); #endif /* POSTSCRIPT */ #else /* NOT OS_VAXVMS */ #if OS_UNIX plotfp = NO_FILENAME ? stdout : FOPEN(dvoname,WB_OPEN); #else plotfp = FOPEN(dvoname,WB_OPEN); #endif /* OS_UNIX */ #endif /* OS_VAXVMS */ #endif /* BBNBITGRAPH */ #if OS_UNIX if (!NO_FILENAME) #endif DEBUG_OPEN(plotfp,dvoname,WB_OPEN); if (plotfp != (FILE*)NULL) break; /* open okay */ if (i == 2) { (void)strcpy(dvoname, curname); (void)strcat(dvoname, "."); (void)strcat(dvoname, DVIPREFIX); (void)strcat(dvoname, OUTFILE_EXT); } } if (plotfp == (FILE*)NULL) { (void)sprintf(message,"dviinit(): %s: can't open [%s]", g_progname, dvoname); (void)fatal(message); } #if OS_TOPS20 /* Because of the PCC-20 arithmetic (instead of logical) shift bug wherein a constant expression (1<<35) evaluates to 0 instead of 400000,,000000, we must set CF_nud using a variable, instead of just using a constant */ /* "ac1 = makefield(CF_nud,1);" */ ac1 = 1; ac1 = makefield(CF_nud,ac1); setfield(ac1,CF_jfn,jfnof(fileno(plotfp))); /* jfn */ setfield(ac1,CF_dsp,FBbyv); /* generation number index in fdb */ ac2 = makefield(FB_ret,-1); ac3 = makefield(FB_ret,0); /* set generation count to 0 (infinite) */ (void)jsys(JSchfdb,acs); /* ignore any errors */ #endif /* OS_TOPS20 */ #if CANON_A2 #ifdef CANON_TEST #if OS_ATARI strcpy(tmoname,"\\ca2XXX\."); #else strcpy(tmoname,"ca2XXXXXX"); #endif mktemp(tmoname); savefp = plotfp; (void)sprintf(modestring,"%s+",WB_OPEN); plotfp = FOPEN(tmoname, modestring); stor_total = 0; #endif /* CANON_TEST */ #endif /* CANON_A2 */ if (!quiet #if OS_UNIX && !NO_FILENAME #endif ) { (void)fprintf(stderr,"[Input from DVI file %s]",dviname); NEWLINE(stderr); (void)fprintf(stderr,"[Output on file %s]",dvoname); NEWLINE(stderr); } if (strncmp(curpath,dvoname,strlen(curpath)) == 0) (void)strcpy(g_logname, curpath); /* used PATH on output file */ else g_logname[0] = '\0'; /* no PATH on output file */ #if OS_UNIX if (NO_FILENAME) /* use dvixxx as file prefix */ { (void)strcpy(g_logname, "dvi"); (void)strcat(g_logname, OUTFILE_EXT); } #endif (void)strcat(g_logname, curname); (void)strcat(g_logname, "."); (void)strcat(g_logname, DVIPREFIX); #if (OS_ATARI | OS_PCDOS | OS_UNIX) (void)strcat(g_logname, "err"); #endif #if (OS_TOPS20 | OS_VAXVMS) /* Force new generation; otherwise fopen(,"w+") reuses existing file */ (void)strcat(g_logname, "err.-1"); #endif lmargin = (COORDINATE)(leftmargin*((float)XDPI)); tmargin = (COORDINATE)(topmargin*((float)YDPI)); #if (CANON_A2 | IMPRESS | HPJETPLUS) /* Printer has (0,0) origin at (XORIGIN,YORIGIN) near, but not at, upper left corner */ lmargin -= XORIGIN; tmargin -= YORIGIN; #endif /* (CANON_A2 | IMPRESS | HPJETPLUS) */ if ((BYTE)nosignex(dvifp,(BYTE)1) != PRE) (void)fatal("dviinit(): PRE doesn't occur first--are you sure \ this is a DVI file?"); i = (int)signex(dvifp,(BYTE)1); if (i != DVIFORMAT) { (void)sprintf(message, "dviinit(): DVI format = %d, can only process DVI format %d files.", i, (int)DVIFORMAT); (void)fatal(message); } }