blob: 8ab840fea35da9d0e8e339837e1988c3cecfb1da [file] [log] [blame]
#include <u.h>
#include <libc.h>
#include <bio.h>
#include <stdio.h>
#include "../common/common.h"
#include "ps_include.h"
extern int curpostfontid;
extern int curfontsize;
typedef struct {long start, end;} Section;
static char *buf;
static void
copy(Biobuf *fin, Biobuf *fout, Section *s) {
if (s->end <= s->start)
return;
Bseek(fin, s->start, 0);
while (Bseek(fin, 0L, 1) < s->end && (buf=Brdline(fin, '\n')) != NULL){
/*
* We have to be careful here, because % can legitimately appear
* in Ascii85 encodings, and must not be elided.
* The goal here is to make any DSC comments impotent without
* actually changing the behavior of the Postscript.
* Since stripping ``comments'' breaks Ascii85, we can instead just
* indent comments a space, which turns DSC comments into non-DSC comments
* and has no effect on binary encodings, which are whitespace-blind.
*/
if(buf[0] == '%')
Bputc(fout, ' ');
Bwrite(fout, buf, Blinelen(fin));
}
}
/*
*
* Reads a PostScript file (*fin), and uses structuring comments to locate the
* prologue, trailer, global definitions, and the requested page. After the whole
* file is scanned, the special ps_include PostScript definitions are copied to
* *fout, followed by the prologue, global definitions, the requested page, and
* the trailer. Before returning the initial environment (saved in PS_head) is
* restored.
*
* By default we assume the picture is 8.5 by 11 inches, but the BoundingBox
* comment, if found, takes precedence.
*
*/
/* *fin, *fout; /* input and output files */
/* page_no; /* physical page number from *fin */
/* whiteout; /* erase picture area */
/* outline; /* draw a box around it and */
/* scaleboth; /* scale both dimensions - if not zero */
/* cx, cy; /* center of the picture and */
/* sx, sy; /* its size - in current coordinates */
/* ax, ay; /* left-right, up-down adjustment */
/* rot; /* rotation - in clockwise degrees */
void
ps_include(Biobuf *fin, Biobuf *fout, int page_no, int whiteout,
int outline, int scaleboth, double cx, double cy, double sx, double sy,
double ax, double ay, double rot) {
char **strp;
int foundpage = 0; /* found the page when non zero */
int foundpbox = 0; /* found the page bounding box */
int nglobal = 0; /* number of global defs so far */
int maxglobal = 0; /* and the number we've got room for */
Section prolog, page, trailer; /* prologue, page, and trailer offsets */
Section *global = 0; /* offsets for all global definitions */
double llx, lly; /* lower left and */
double urx, ury; /* upper right corners - default coords */
double w = whiteout != 0; /* mostly for the var() macro */
double o = outline != 0;
double s = scaleboth != 0;
int i; /* loop index */
#define has(word) (strncmp(buf, word, strlen(word)) == 0)
#define grab(n) ((Section *)(nglobal \
? realloc((char *)global, n*sizeof(Section)) \
: calloc(n, sizeof(Section))))
llx = lly = 0; /* default BoundingBox - 8.5x11 inches */
urx = 72 * 8.5;
ury = 72 * 11.0;
/* section boundaries and bounding box */
prolog.start = prolog.end = 0;
page.start = page.end = 0;
trailer.start = 0;
Bseek(fin, 0L, 0);
while ((buf=Brdline(fin, '\n')) != NULL) {
buf[Blinelen(fin)-1] = '\0';
if (!has("%%"))
continue;
else if (has("%%Page: ")) {
if (!foundpage)
page.start = Bseek(fin, 0L, 1);
sscanf(buf, "%*s %*s %d", &i);
if (i == page_no)
foundpage = 1;
else if (foundpage && page.end <= page.start)
page.end = Bseek(fin, 0L, 1);
} else if (has("%%EndPage: ")) {
sscanf(buf, "%*s %*s %d", &i);
if (i == page_no) {
foundpage = 1;
page.end = Bseek(fin, 0L, 1);
}
if (!foundpage)
page.start = Bseek(fin, 0L, 1);
} else if (has("%%PageBoundingBox: ")) {
if (i == page_no) {
foundpbox = 1;
sscanf(buf, "%*s %lf %lf %lf %lf",
&llx, &lly, &urx, &ury);
}
} else if (has("%%BoundingBox: ")) {
if (!foundpbox)
sscanf(buf,"%*s %lf %lf %lf %lf",
&llx, &lly, &urx, &ury);
} else if (has("%%EndProlog") || has("%%EndSetup") || has("%%EndDocumentSetup"))
prolog.end = page.start = Bseek(fin, 0L, 1);
else if (has("%%Trailer"))
trailer.start = Bseek(fin, 0L, 1);
else if (has("%%BeginGlobal")) {
if (page.end <= page.start) {
if (nglobal >= maxglobal) {
maxglobal += 20;
global = grab(maxglobal);
}
global[nglobal].start = Bseek(fin, 0L, 1);
}
} else if (has("%%EndGlobal"))
if (page.end <= page.start)
global[nglobal++].end = Bseek(fin, 0L, 1);
}
Bseek(fin, 0L, 2);
if (trailer.start == 0)
trailer.start = Bseek(fin, 0L, 1);
trailer.end = Bseek(fin, 0L, 1);
if (page.end <= page.start)
page.end = trailer.start;
/*
fprint(2, "prolog=(%d,%d)\n", prolog.start, prolog.end);
fprint(2, "page=(%d,%d)\n", page.start, page.end);
for(i = 0; i < nglobal; i++)
fprint(2, "global[%d]=(%d,%d)\n", i, global[i].start, global[i].end);
fprint(2, "trailer=(%d,%d)\n", trailer.start, trailer.end);
*/
/* all output here */
for (strp = PS_head; *strp != NULL; strp++)
Bwrite(fout, *strp, strlen(*strp));
Bprint(fout, "/llx %g def\n", llx);
Bprint(fout, "/lly %g def\n", lly);
Bprint(fout, "/urx %g def\n", urx);
Bprint(fout, "/ury %g def\n", ury);
Bprint(fout, "/w %g def\n", w);
Bprint(fout, "/o %g def\n", o);
Bprint(fout, "/s %g def\n", s);
Bprint(fout, "/cx %g def\n", cx);
Bprint(fout, "/cy %g def\n", cy);
Bprint(fout, "/sx %g def\n", sx);
Bprint(fout, "/sy %g def\n", sy);
Bprint(fout, "/ax %g def\n", ax);
Bprint(fout, "/ay %g def\n", ay);
Bprint(fout, "/rot %g def\n", rot);
for (strp = PS_setup; *strp != NULL; strp++)
Bwrite(fout, *strp, strlen(*strp));
copy(fin, fout, &prolog);
for(i = 0; i < nglobal; i++)
copy(fin, fout, &global[i]);
copy(fin, fout, &page);
copy(fin, fout, &trailer);
for (strp = PS_tail; *strp != NULL; strp++)
Bwrite(fout, *strp, strlen(*strp));
if(nglobal)
free(global);
/* force the program to reestablish its state */
curpostfontid = -1;
curfontsize = -1;
}