/* * This is the main routine for the first (prescanning) pass. */ #include "dvips.h" /* The copyright notice in that file is included too! */ /* * These are all the external routines it calls: */ #include "protos.h" /* * These are the globals it accesses. */ #ifdef DEBUG extern integer debug_flag; #endif /* DEBUG */ extern fontdesctype *fonthead ; extern real conv ; extern real vconv ; extern real alpha ; extern integer firstpage, lastpage ; extern integer firstseq, lastseq ; extern integer maxsecsize ; extern Boolean notfirst, notlast ; extern Boolean evenpages, oddpages, pagelist ; extern integer fontmem ; extern integer pagecount ; extern integer pagenum ; extern integer maxpages ; extern sectiontype *sections ; extern FILE *dvifile ; extern integer num, den ; extern double mag ; extern int overridemag ; extern integer swmem ; extern int quiet ; extern int actualdpi ; extern int vactualdpi ; extern Boolean reverse ; extern int totalpages ; extern integer fsizetol ; extern char *oname ; extern Boolean pprescan ; extern Boolean abspage ; extern char preamblecomment[] ; /* * This routine handles the processing of the preamble in the dvi file. */ void readpreamble P1H(void) { register int i ; char *p ; if (dvibyte()!=247) error("! Bad DVI file: first byte not preamble") ; if (dvibyte()!=2) error("! Bad DVI file: id byte not 2") ; num = signedquad() ; den = signedquad() ; if (overridemag > 0) (void)signedquad() ; else if (overridemag < 0) mag = (mag * signedquad() + 500.0) / 1000.0 ; else mag = signedquad() ; conv = (real) num * DPI * mag / ( den * 254000000.0 ) ; vconv = (real) num * VDPI * mag / ( den * 254000000.0 ) ; alpha = (((real)den / 7227.0) / 0x100000) * (25400000.0 / (real) num) ; fsizetol = 1 + (integer)(DPI/(72270.0 * conv)) ; if (!pprescan) { for (i=dvibyte(),p=preamblecomment;i>0;i--,p++) *p=dvibyte() ; *p='\0' ; if (!quiet) { (void)fprintf(stderr, "'") ; #ifdef VMCMS /* IBM: VM/CMS */ for(p=preamblecomment;*p;p++) (void)putc(ascii2ebcdic[*p], stderr) ; #else #ifdef MVSXA /* IBM: MVS/XA */ for(p=preamblecomment;*p;p++) (void)putc(ascii2ebcdic[*p], stderr) ; #else for(p=preamblecomment;*p;p++) (void)putc(*p, stderr) ; #endif /* IBM: VM/CMS */ #endif (void)fprintf(stderr, "' -> %s\n", oname) ; } } else skipover(dvibyte()) ; } /* * Finally, here's our main prescan routine. */ static integer firstmatch = -1, lastmatch = -1 ; void prescanpages P1H(void) { register int cmd ; short ret = 0 ; register integer thispageloc, thissecloc ; register fontdesctype *f ; register shalfword c ; register long thissectionmem = 0 ; integer mpagenum ; integer pageseq = 0 ; int ntfirst = notfirst ; readpreamble() ; /* * Now we look for the first page to process. If we get to the end of * the file before the page, we complain (fatally). * Incidentally, we don't use the DVI file's bop backpointer to skip * over pages at high speed, because we want to look to for special * header that might be in skipped pages. */ while (1) { cmd = skipnop() ; if (cmd==248) error("! End of document before first specified page") ; if (cmd!=139) error("! Bad DVI file: expected bop") ; thispageloc = ftell(dvifile) ; /* the location FOLLOWING the bop */ #ifdef DEBUG if (dd(D_PAGE)) #ifdef SHORTINT (void)fprintf(stderr,"bop at %ld\n", thispageloc) ; #else /* ~SHORTINT */ (void)fprintf(stderr,"bop at %d\n", (int)thispageloc) ; #endif /* ~SHORTINT */ #endif /* DEBUG */ pagenum = signedquad() ; pageseq++ ; mpagenum = abspage ? pageseq : pagenum ; if (mpagenum == firstpage && ntfirst) firstmatch++ ; if (mpagenum == lastpage && notlast) lastmatch++ ; if (ntfirst && mpagenum == firstpage && firstmatch == firstseq) ntfirst = 0 ; if (ntfirst || ((evenpages && (pagenum & 1)) || (oddpages && (pagenum & 1)==0) || (pagelist && !InPageList(pagenum)))) { skipover(40) ; skippage() ; } else { if (notlast && mpagenum == lastpage) lastmatch-- ; break ; } } /* * Here we scan for each of the sections. First we initialize some of * the variables we need. */ while (maxpages > 0 && cmd != 248) { for (f=fonthead; f; f=f->next) { f->psname = 0 ; if (f->loaded==1) for (c=255; c>=0; c--) f->chardesc[c].flags &= (STATUSFLAGS) ; } fontmem = swmem - OVERCOST ; if (fontmem <= 1000) error("! Too little VM in printer") ; /* The section begins at the bop command just before thispageloc (which may * be a page that was aborted because the previous section overflowed memory). */ pagecount = 0 ; (void)fseek(dvifile, (long)thispageloc, 0) ; pagenum = signedquad() ; skipover(40) ; thissecloc = thispageloc ; /* * Now we have the loop that actually scans the pages. The scanpage routine * returns 1 if the page scans okay; it returns 2 if the memory ran out * before any pages were completed (in which case we'll try to carry on * and hope for the best); it returns 0 if a page was aborted for lack * of memory. After each page, we mark the characters seen on that page * as seen for this section so that they will be downloaded. */ ret = 0 ; while (maxpages>0) { if (!(evenpages && (pagenum & 1)) && !(oddpages && (pagenum & 1)==0) && !(pagelist && !InPageList(pagenum))) { ret = scanpage() ; if (ret == 0) break ; pagecount++ ; maxpages-- ; } else skippage() ; thissectionmem = swmem - fontmem - OVERCOST ; mpagenum = abspage ? pageseq : pagenum ; pageseq++ ; if (mpagenum == lastpage && notlast) lastmatch++ ; if (notlast && mpagenum == lastpage && lastmatch == lastseq) maxpages = -1 ; /* we are done after this page. */ if (reverse) thissecloc = thispageloc ; for (f=fonthead; f; f=f->next) if (f->loaded==1) { if (f->psflag & THISPAGE) f->psflag = PREVPAGE ; for (c=255; c>=0; c--) if (f->chardesc[c].flags & THISPAGE) f->chardesc[c].flags = PREVPAGE | (f->chardesc[c].flags & (STATUSFLAGS)) ; } cmd=skipnop() ; if (cmd==248) break ; if (cmd!=139) error("! Bad DVI file: expected bop") ; thispageloc = ftell(dvifile) ; #ifdef DEBUG if (dd(D_PAGE)) #ifdef SHORTINT (void)fprintf(stderr,"bop at %ld\n", thispageloc) ; #else /* ~SHORTINT */ (void)fprintf(stderr,"bop at %d\n", (int)thispageloc) ; #endif /* ~SHORTINT */ #endif /* DEBUG */ pagenum = signedquad() ; skipover(40) ; if (ret==2 || (maxsecsize && pagecount >= maxsecsize)) break ; } /* * Now we have reached the end of a section for some reason. * If there are any pages, we save the pagecount, section location, * and continue. */ if (pagecount>0) { register int fc = 0 ; register sectiontype *sp ; register charusetype *cp ; totalpages += pagecount ; for (f=fonthead; f; f=f->next) if (f->loaded==1 && f->psname) fc++ ; sp = (sectiontype *)mymalloc((integer)(sizeof(sectiontype) + fc * sizeof(charusetype) + sizeof(fontdesctype *))) ; sp->bos = thissecloc ; if (reverse) { sp->next = sections ; sections = sp ; } else { register sectiontype *p ; sp->next = NULL ; if (sections == NULL) sections = sp ; else { for (p=sections; p->next != NULL; p = p->next) ; p->next = sp ; } } sp->numpages = pagecount ; #ifdef DEBUG if (dd(D_PAGE)) #ifdef SHORTINT (void)fprintf(stderr,"Have a section: %ld pages at %ld fontmem %ld\n", #else /* ~SHORTINT */ (void)fprintf(stderr,"Have a section: %d pages at %d fontmem %d\n", #endif /* ~SHORTINT */ (integer)pagecount, (integer)thissecloc, (integer)thissectionmem) ; #endif /* DEBUG */ cp = (charusetype *) (sp + 1) ; fc = 0 ; for (f=fonthead; f; f=f->next) if (f->loaded==1 && f->psname) { register halfword b, bit ; cp->psfused = (f->psflag & PREVPAGE) ; f->psflag = 0 ; cp->fd = f ; c = 0 ; for (b=0; b<16; b++) { cp->bitmap[b] = 0 ; for (bit=32768; bit!=0; bit>>=1) { if (f->chardesc[c].flags & PREVPAGE) cp->bitmap[b] |= bit ; c++ ; } } cp++ ; } cp->fd = NULL ; } } }